Skip to main content
Version: 20 R8 BETA

HTTP Request handler

By default, HTTP requests received by the 4D web server are handled through built-in processing features or the REST server.

In addition, 4D supports the implementation of custom HTTP Request handlers, allowing you to intercept specific incoming HTTP requests and process them using your own code.

When a custom HTTP request handler intercepts a request, it is processed directly and no other processing features (e.g. On Web authentication or On Web connection database methods) are called.

Custom HTTP request handlers meet various needs, including:

  • using a given URL as a resource provider or a file-uploading box (to download or upload various files),
  • redirecting on specific pages according to a context (user authenticated, privileges granted...),
  • handle an authentication via oAuth 2.0.

Requirements

Custom HTTP Request handlers are supported:

HTTPHandlers.json File

You define your custom HTTP Request handlers in a configuration file named HTTPHandlers.json stored in the Project/Sources folder.

This file contains all listened URL patterns, the handled verbs, and the code to be called. Handlers are provided as a collection in JSON format.

At runtime, the first pattern matching the URL is executed, the others are ignored.

Here is an example of a HTTPHandlers.json file contents:


[
{
"class": "GeneralHandling",
"method": "gettingStarted",
"pattern": "start",
"verbs": "get, post"
}
]

This handler declaration can be read as: when any request starting by /start/ with a GET or POST verb is received by the server, the gettingStarted function of the GeneralHandling singleton is executed.

note

You must restart the Web server so that modifications made in this file are taken into account.

Handler definition

A handler is defined by:

  • a listened URL pattern
  • a function and its class where the code is implemented to handle the listened URL pattern
  • the verbs with which the URL can be called to trigger the handler

The handler identifier is the couple [pattern + a verb among the verbs list].

URL patterns

URL patterns can be given as prefixes or using regular expressions.

  • To declare a prefix pattern, use the "pattern" property name in the HTTPHandlers.json file. Prefixes are considered as regular expressions already containing starting and ending /.
    Ex: "pattern" : "docs" or "pattern" : "docs/invoices"

  • To declare a regular expression pattern, use the "regexPattern" property name in the HTTPHandlers.json file. Regular expressions patterns are handled directly. Ex: "regexPattern" : "/docs/**/index.html"

"Pattern" and "regexPattern" properties cannot be used in the same handler definition (in this case, only the "regexPattern" property is taken into account).

Pattern matching

URL patterns are triggered in the given order:

  • the first matching pattern is executed
  • the following patterns are not executed even if they match the URL

As a consequence, you need to apply a accurate strategy when writing your handlers: the most detailed patterns must be written before the more general patterns.

[
{
"class": "InvoiceslHandling",
"method": "handleTheInvoice",
"regexPattern": "/docs/invoices/details/theInvoice",
"verbs": "GET"
},
{
"class": "InvoiceslHandling",
"method": "handleUnauthorizedVerbs",
"regexPattern": "/docs/invoices/details/theInvoice",
"comment": "This handler is triggered for all verbs except GET (handled above)"
},
{
"class": "DocsHandling",
"method": "handleDocs",
"regexPattern": "/docs",
"comment": "This handler is triggered for all the verbs"
}
]

Forbidden patterns

URL patterns matching 4D built-in HTTP processing features are not allowed in custom HTTP handlers. For example, the following patterns cannot be handled:

  • /4DACTION
  • /rest
  • /$lib/renderer
  • /$shared

Class and method

You declare the code to be executed when a defined URL pattern is intercepted using the "class" and "method" properties.

  • "class": class name without cs., e.g. "UsersHandling" for the cs.UsersHandling user class. It must be a shared and singleton class.
  • "method": class function belonging to the class.

See below for information about the request handler code.

Verbs

You can use the "verbs" property in the handler definition to declare HTTP verbs that are supported in incoming requests for this handler. A request that uses a verb that is not explicitely allowed is automatically rejected by the server.

You can declare several verbs, separated by a comma. Verb names are not case sensitive.

Ex: "verbs" : "PUT, POST"

note

No control is done on verb names. All names can be used.

By default, if the "verbs" property is not used for a handler, all HTTP verbs are supported in incoming requests for this handler (except those possibly used beforehand in a more detailed pattern, as shown in the example above).

note

The HTTP verb can also be evaluated using the .verb property within the request handler code to be accepted or rejected.

Example

Here is a detailed example of a HTTPHandlers.json file:


[
{
"class": "GeneralHandling",
"method": "handle",
"pattern": "info", //URL prefix
"verbs": "GET"
},
{
"class": "UsersHandling",
"method": "manageAccount",
"pattern": "userAccount/update", //URL prefix
"verbs": "PUT,POST"
},
{
"class": "FinancialHandling",
"method": "handleInvoices",
"regexPattern": "/docs/invoices/(past|today)", //URL prefix given as a regex
"verbs": "GET"
},
{
"class": "DocsHandling",
"method": "handleDocs",
"regexPattern": "/docs/myPage.html", //URL prefix given as a regex
"verbs": "GET"
},
{
"class": "InvoicesHandling",
"method": "handleTheInvoice",
"pattern": "docs/invoices/details/theInvoice", // The most specific URL first
"verbs": "GET,POST"
},
{
"class": "InvoicesHandling",
"method": "handleDetails",
"pattern": "docs/invoices/details", // The general URLs after
"verbs": "GET"
},
{
"class": "InvoicesHandling",
"method": "handleInvoices", // The general URLs after
"pattern": "docs/invoices",
"verbs": "GET"
}
]

In this example, you must implement the following functions:

  • handle function in the GeneralHandling class
  • manageAccount in the UsersHandling class
  • handleInvoices in the FinancialHandling class
  • handleDocs in the DocsHandling class
  • handleTheInvoice / handleDetails / handleInvoices in the InvoicesHandling class

Examples of URLs triggering the handlers:

IP:port/info/ with a GET verb IP:port/info/general with a GET verb

IP:port/userAccount/update/ with a POST verb IP:port/userAccount/update/profile with a POST verb

IP:port/docs/invoices/past with a GET verb IP:port/docs/invoices/today/latest with a GET verb

IP:port//docs/myPage.html with a GET verb

IP:port//docs/invoices/ with a GET verb, calls handleInvoices function (InvoicesHandling class) IP:port//docs/invoices/details/ with a GET verb, calls handleDetails function (InvoicesHandling class) IP:port//docs/invoices/details/theInvoice/xxxxxx with a GET verb, calls handleTheInvoice function (InvoiceslHandling class)

Request handler code

Function configuration

The HTTP Request handler code must be implemented in a function of a Shared singleton class.

If the singleton is missing or not shared, an error "Cannot find singleton" is returned by the server. If the class or the function defined as handler in the HTTPHandlers.json file is not found, an error "Cannot find singleton function" is returned by the server.

Request handler functions are not necessarily shared, unless some request handler properties are updated by the functions. In this case, you need to declare its functions with the shared keyword.

note

It is not recommended to expose request handler functions to external REST calls using exposed or onHttpGet keywords.

Input: an instance of the 4D.IncomingMessage class

When a request has been intercepted by the handler, it is received on the server as an instance of the 4D.IncomingMessage class.

All necessary information about the request are available in this object, including the request url, verb, headers, and, if any, parameters (put in the URL) and body.

Then, the request handler can use this information to trigger appropriate business logic.

Output: an instance of the 4D.OutgoingMessage class

The request handler can return an object instance of the 4D.OutGoingMessage class, i.e. some full web content ready for a browser to handle, such as a file content.

Example

The 4D.IncomingMessage class provides functions to get the headers and the body of the request.

Here is a simple example to upload a file on the server.

The HTTPHandlers.json file:

[
{
"class": "UploadFile",
"method": "uploadFile",
"regexPattern": "/putFile",
"verbs": "POST"
}
]

The called URL is: http://127.0.0.1:8044/putFile?fileName=testFile

The binary content of the file is put in the body of the request and a POST verb is used. The file name is given as parameter (fileName) in the URL. It is received in the urlQuery object in the request.

    //UploadFile class

shared singleton Class constructor()


Function uploadFile($request : 4D.IncomingMessage) : 4D.OutgoingMessage

var $response:=4D.OutgoingMessage.new()

var $body:="Not supported file"
var $fileName; $fileType : Text
var $file : 4D.File
var $picture : Picture
var $created : Boolean

$fileName:=$request.urlQuery.fileName
$fileType:=$request.getHeader("Content-Type")

Case of
: ($fileType="application/pdf")
$file:=File("/PACKAGE/Files/"+$fileName+".pdf")
$created:=$file.create()
$file.setContent($request.getBlob())
$body:="Upload OK - File size: "+String($file.size)

: ($fileType="image/jpeg")
$file:=File("/PACKAGE/Files/"+$fileName+".jpg")
$picture:=$request.getPicture()
WRITE PICTURE FILE($file.platformPath; $picture)
$body:="Upload OK - Image size: "+String($file.size)

End case

$response.setBody($body)
$response.setHeader("Content-Type"; "text/plain")

return $response

See also

Perfect mastery of your back end business logic thanks to HTTP requests handlers (blog post)