Click here to Skip to main content
65,938 articles
CodeProject is changing. Read more.
Articles
(untagged)

Discover WIN32. How to use a sync/async retrieve HTTP content in your ASP & VB projects.

0.00/5 (No votes)
19 May 2002 2  
This article shows how to create one ATL COM component using the WinInet functions, how to use it in ASP programs and how to test it from the Visual Basic client. It also shows how to use multithreading support in this component.

Sample Image - snap.gif

Introduction

This article shows how to create one ATL COM component using the WinInet functions, how to use it in ASP programs and how to test it from the Visual Basic client. It also shows how to use multithreading support in this component.

This component can be used in the same way in Visual Basic, Delphi, Access or Microsoft SQL.

Overview

This application uses ATL, MFC, ASP, Visual Basic and WinInet functions. The application have an ATL project who provide some HTTP functions. It will demonstrate how to use this component in your ASP and Visual Basic projects.

The article is based on the Uwe Keim's article �Calling scripts within scripts� � many thanks to him for his good idea. Of course you will ask what is new in my component. The reason that made me build this component was the variable time delay at a HTTP request.

My solution was an asynchronous retrieve, which give the user the possibilities to check more often whether the page is completed or not.

The article also contains a Visual Basic client who tests the functionality of the component.

I assume the reader knows or at least has a fair idea about COM programming using ATL.

Details

The component consists of two parts: 

One standard WinInet retrieve module.

  • Retrieve method.
    Retrieve(BSTR newVal, BSTR *parOut)

  • GetPage function.
    bool GetPage(CString sURL, CString& sBody)

The Retrieve method just calls the GetPage function with the sURL parameter. Inside the function, there are the calls to WinInet API:

//Open one internet connection. The �INTERNET_OPEN_TYPE_PRECONFIG� 

//parameter tell the function to read the default proxy settings from 

// the registry(the proxy settings from connections in IE):

HINTERNET internet=InternetOpen("Daemon", INTERNET_OPEN_TYPE_PRECONFIG, 
                                 NULL, NULL, NULL);

//Make one connection with the desired URL. On the third parameter we 

//use the�INTERNET_FLAG_RELOAD� 

//while constrains the function not to usea local cached copy. 


//Take a look at the documentation of this functions to see all the 

// combination of parameters:

HINTERNET file_handle=InternetOpenUrl(internet, sURL, NULL, 0, 
                                      INTERNET_FLAG_RELOAD, 0);

//Read the data from the address:

InternetReadFile(file_handle, pagina, 100000, &bytes_read);

//Close the connection

InternetCloseHandle(internet);
The asp source code is very simple (full source code are in attachment file):
url="http://www.codeproject.com"

Set myORetrieve = CreateObject("RetrievePage.RetrievePage.1")

ret = myORetrieve.Retrieve(url)        

Response.Write ret

One asynchronous retrieve module.

  • RetrieveAsync method.
    RetrieveAsync(BSTR newVal, BSTR *parOut)

  • RetrieveAsyncCompleted method.
    RetrieveAsyncCompleted(BSTR newVal, BSTR *parOut)

  • RetrieveAsyncPage method.
    RetrieveAsyncPage(BSTR newVal, BSTR *parOut)

  • MyThreadGetPage function.
    UINT MyThreadGetPage(LPVOID pParam)

The client must use first the RetrieveAsync method with the desired URL parameter. This method launch one child thread that will retrieve the page in a separate thread. In this way the program will go to the next instruction and will not wait for the method to finish the request.

The program must implement one timer (or at the user's actions) who verify at some intervals if the RetrieveAsyncCompleted method return true (the requested page was retrieved). Only then the RetrieveAsyncPage method will provide the requested page.

While the RetrieveAsyncCompleted method return false you could send to the console a message such as �Please wait. Data is loading��

//Make a new thread. The �MyThreadGetPage� is the name of the 

//function who will resolve the HTTP request. Inside the 

// MyThreadGetPage function, there is one call to the previous 

//discussed GetPage function.:

AfxBeginThread( MyThreadGetPage , this, THREAD_PRIORITY_NORMAL )


// The program use a coleection to keep all request to the component.

// Add a new URL request:

pObject->m_MapStringToHTTPpage[sKey]        = oHTTPpage;


// Look at collection to see if it is a new URL or not:

m_MapStringToHTTPpage.Lookup(sURL, oHTTPpage)
The Visual Basic source code is very simple (full source code are in attachment file, in RetrieveVBClient director):
'Launch the thread with the specified address

str = myORetrieve.RetrieveAsync(URL.Text)

'Look if the retrieved page is completed:

lRetrieveAsyncCompleted = myORetrieve.RetrieveAsyncCompleted(URL.Text) 

'If it is true, read the data from the specified URL address

str = myORetrieve.RetrieveAsyncPage(URL.Text)

If you want to see how to build an ATL component, take a look at the Build an ATL Crypt component article.

Installation

  • Copy the DLL into a directory with system execute privilege and register it with regsvr32 command or put it on the MTS.  To put it on the MTS is better because if we want to modify the component and register it again, this is possible without restarting the computer � in case when using the regsvr32 command who �blocks� your dll file.
  • Use directly the visual basic dialog console. Input some string on the URL edit box and click on the button!
  • Copy the director with asp pages on the Web server � and just try it !

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here