How wwAsyncWebRequest works
Asynchronous events can be used for many operations that take a while to run. The basic idea behind them in a Web application is:
-
* Browser submits a request to the server
* Server picks up request and submits it to the event queue
* Server returns a 'I'm working on it' to the browser which is refreshed every x seconds
* External application picks up the event and starts processing it
* Optionally external app updates the status of processing as available
* When done external app updates the event as completed
* Web Server app meanwhile keeps checking for completion
* When complete the Web Server app picks up the completed result (XML, a data file somewhere, or maybe a PDF or HTML string etc)
* Web Server can optionally perform more processing on the returned data
* Web Server then returns the result to the client browser
The following code sequence demonstrates how events can be set by clients and processed by servers:
Client initiating the Event:
SET PROCEDURE TO wwAsyncWebRequest ADDITIVE SET PROCEDURE TO wwXMLState ADDITIVE * ** Event Initiator code (Web App most likely) o = CREATEOBJECT("wwAsyncWebRequest") * ** SubmitEvent saves the request lcId = o.SubmitEvent("Test Data","Sumbited at: " + TIME()) * ** If you want to add additional data you can use dynamic properties o.SetProperty("Name","Rick") o.SetProperty("Report","custlist") o.SetProperty("Time",DATETIME()) o.SetProperty("SQL","select * from TT_Cust") * ** Or your can set properties on the even object directly o.oEvent.UserId = "rstrahl" o.oEvent.Title = "Test Event" * ** If you make manual changes make sure to Save o.SaveEvent()
Server receiving the event:
* ** Pass in an event id somehow LPARAMETERS lcID Application.Visible = .F. IF EMPTY(lcID) RETURN ENDIF SET PROCEDURE TO wwUtils ADDITIVE SET PROCEDURE TO wwXMLState Additive SET PROCEDURE TO wwAsyncWebRequest Additive SET CLASSLIB TO wwXML ADDITIVE loAsync = CREATEOBJECT("wwAsyncWebRequest") * ** Load the Event IF !loAsync.LoadEvent(lcID) RETURN ENDIF WAIT WINDOW "Simulating long request taking 20 seconds..." TIMEOUT 20 lcSQL = loAsync.GetProperty("SQL") * ** Run the SQL Statement &lcSQL INTO Cursor TTCustList * ** Do something with the output - in this case turn into XML loXML = CREATEOBJECT("wwXML") lcXML = loXML.CursorToXML() * ** Close out the request and pass the return data * ** into the ResultData property loAsync.CompleteEvent(lcID,lcXML) * ** EXIT app RETURN
While waiting for the request to complete on the server the client code polls the server. The polling code can look like this:
IF loAsync.CheckForCompletion(lcID) = 1 * ** Display result - XML document return in this case Response.ContentTypeHeader("text/xml") * ** Retrieve the data from the server and display as XML Response.Write(loAsync.oEvent.ReturnData) * ** Alternately you could read any other fields of the oEvent object * ** or any property set by the server application with GetProperty() RETURN ENDIF * ** Create the waiting output page lcBody = "
Waiting for report to complete" + ; REPLICATE(". ",loAsync.oEvent.ChkCounter + 1) + "" + ; "This report is now running in another process while leaving the " +; "Web Connection instance running free to process other requests. " + ; "It polls back every few seconds to check for completion and when done " +; "picks up the result data to be displayed in this case a simulated " + ; "long running report..." * ** Create the 'Waiting...' page. Note the Refresh and Refresh URL parameters * ** The URL contains a counter so we can abort after x number of tries. THIS.StandardPage("Running Report",lcBody,,; lnPageRefresh,"AsyncWebRequest.wwd?Action=Check&RequestId=" + lcId)
© West Wind Technologies, 2025 • Updated: 2025-03-12
Comment or report problem with topic