Win32 API programming with C - HTTP Server API

In the realm of network programming on Windows, the Win32 HTTP Server API offers a robust, low-level foundation for handling HTTP requests directly within a C application. This API, part of the Windows HTTP Services (WinHTTP), provides developers with the tools needed to create HTTP servers that can manage requests, send responses, and handle web traffic without the need for third-party libraries. This article explores how to use the Win32 HTTP Server API with the C programming language to create a simple HTTP server.

Understanding the Win32 HTTP Server API

The Win32 HTTP Server API is designed to be a high-performance server-side Web protocol listener. It's suitable for applications that need to serve HTTP content directly, such as web services, or for scenarios where embedding a lightweight web server into an application is necessary.

The API provides a set of functions that allow applications to:

  • Create and manage HTTP server sessions.

  • Configure URL namespaces and listen for requests on specific URLs.

  • Receive requests and send responses, including headers and body data.

  • Handle SSL/TLS encryption for secure communication.

Setting Up Your Environment

Before diving into code, ensure your development environment is set up correctly. You will need:

  • A Windows development environment.

  • A C compiler, such as the one provided with Visual Studio.

  • The Windows SDK, which includes the headers and libraries for the Win32 HTTP Server API.

Creating a Simple HTTP Server

To illustrate how to use the Win32 HTTP Server API, let's create a simple HTTP server that listens for requests on a specific URL and responds with a basic message. The following steps outline the process:

1. Initialize the HTTP Service

First, initialize the HTTP service using the HttpInitialize function. This function prepares the application to use the HTTP Server API.

#include <windows.h>
#include <http.h>

#pragma comment(lib, "httpapi.lib")

ULONG result;
HTTPAPI_VERSION HttpApiVersion = HTTPAPI_VERSION_1;

result = HttpInitialize(HttpApiVersion, HTTP_INITIALIZE_SERVER, NULL);
if (result != NO_ERROR) {
    return 1; // Initialization failed
}

2. Create an HTTP Server Session

Next, create a server session and configure it to listen for HTTP requests. Use HttpCreateHttpHandle to create a request queue handle.

HANDLE hReqQueue = NULL;
result = HttpCreateHttpHandle(&hReqQueue, 0);
if (result != NO_ERROR) {
    HttpTerminate(HTTP_INITIALIZE_SERVER, NULL);
    return 1; // Failed to create request queue
}

3. Add a URL to the Request Queue

Before the server can receive requests, you must add at least one URL to the request queue. This is done using the HttpAddUrl function.

PCWSTR urlPrefix = L"http://localhost:8080/";
result = HttpAddUrl(hReqQueue, urlPrefix, NULL);
if (result != NO_ERROR) {
    HttpCloseRequestQueue(hReqQueue);
    HttpTerminate(HTTP_INITIALIZE_SERVER, NULL);
    return 1; // Failed to add URL
}

4. Receive Requests and Send Responses

The server is now ready to receive HTTP requests. Use a loop to continuously call HttpReceiveHttpRequest, process each request, and then use HttpSendHttpResponse to send a response.

HTTP_REQUEST_ID reqId = 0;
HTTP_REQUEST* pRequest;
ULONG reqBufSize = sizeof(HTTP_REQUEST) + 2048; // Allocate a 2 KB buffer
pRequest = (HTTP_REQUEST*)malloc(reqBufSize);
DWORD bytesRead;

while (TRUE) {
    ZeroMemory(pRequest, reqBufSize);
    result = HttpReceiveHttpRequest(
        hReqQueue,    // Req Queue
        reqId,        // Req ID
        0,            // Flags
        pRequest,     // HTTP request buffer
        reqBufSize,   // req buffer length
        &bytesRead,   // bytes received
        NULL          // LPOVERLAPPED
    );

    if (result == NO_ERROR) {
        switch (pRequest->Verb)
        {
            case HttpVerbGET:
                wprintf(L"Got a GET request for %ws \n", pRequest->CookedUrl.pFullUrl);
                result = SendHttpResponse(hReqQueue, pRequest);
                break;
            case HttpVerbPOST:
                wprintf(L"Got a POST request for %ws \n", pRequest->CookedUrl.pFullUrl);
                break;
            default:
                break;
        }
    }
}

5. Clean Up

Finally, remember to clean up resources by removing URLs from the request queue, closing the queue, and terminating the HTTP service.

HttpRemoveUrl(hReqQueue, urlPrefix);
HttpCloseRequestQueue(hReqQueue);
HttpTerminate(HTTP_INITIALIZE_SERVER, NULL);

Conclusion

The Win32 HTTP Server API provides a powerful set of functions for creating HTTP servers in C. While the example provided here is very basic, it illustrates the core steps needed to receive HTTP requests and respond to them. The API's flexibility allows developers to handle more complex scenarios, including serving files, handling POST requests, and integrating with existing web services. With its direct access to the HTTP protocol and integration with Windows security models, the Win32 HTTP Server API is a valuable tool for C developers looking to implement web server functionality in their applications.

Full code below:

#ifndef UNICODE
#define UNICODE
#endif

#ifndef _WIN32_WINNT
#define _WIN32_WINNT 0x0600
#endif

#ifndef WIN32_LEAN_AND_MEAN
#define WIN32_LEAN_AND_MEAN
#endif

#include <windows.h>
#include <http.h>
#include <stdio.h>

#pragma comment(lib, "httpapi.lib")

ULONG result;
HTTPAPI_VERSION HttpApiVersion = HTTPAPI_VERSION_1;

// function declarations
DWORD SendHttpResponse(HANDLE hReqQueue, HTTP_REQUEST* pRequest);

int main() {
    result = HttpInitialize(HttpApiVersion, HTTP_INITIALIZE_SERVER, NULL);
    if (result != NO_ERROR) {
        return 1; // Initialization failed
    }

    // Create an HTTP Server Session
    HANDLE hReqQueue = NULL;
    result = HttpCreateHttpHandle(&hReqQueue, 0);
    if (result != NO_ERROR) {
        HttpTerminate(HTTP_INITIALIZE_SERVER, NULL);
        return 1; // Failed to create request queue
    }

    PCWSTR urlPrefix = L"http://localhost:8080/";
    result = HttpAddUrl(hReqQueue, urlPrefix, NULL);
    if (result != NO_ERROR) {
        HttpCloseRequestQueue(hReqQueue);
        HttpTerminate(HTTP_INITIALIZE_SERVER, NULL);
        return 1; // Failed to add URL
    }

    wprintf(L"Listening for requests at: %s\n", urlPrefix);

    HTTP_REQUEST_ID reqId = 0;
    HTTP_REQUEST* pRequest;
    ULONG reqBufSize = sizeof(HTTP_REQUEST) + 2048; // Allocate a 2 KB buffer
    pRequest = (HTTP_REQUEST*)malloc(reqBufSize);
    DWORD bytesRead;

    while (TRUE) {
        ZeroMemory(pRequest, reqBufSize);
        result = HttpReceiveHttpRequest(
            hReqQueue,    // Req Queue
            reqId,        // Req ID
            0,            // Flags
            pRequest,     // HTTP request buffer
            reqBufSize,   // req buffer length
            &bytesRead,   // bytes received
            NULL          // LPOVERLAPPED
        );


        if (result == NO_ERROR) {
            switch (pRequest->Verb)
            {
                case HttpVerbGET:
                    wprintf(L"Got a GET request for %ws \n", pRequest->CookedUrl.pFullUrl);
                    result = SendHttpResponse(hReqQueue, pRequest);
                    break;
                case HttpVerbPOST:
                    wprintf(L"Got a POST request for %ws \n", pRequest->CookedUrl.pFullUrl);
                    break;
                default:
                    break;
            }
        }
    }

    // cleanup
    HttpRemoveUrl(hReqQueue, urlPrefix);
    HttpCloseRequestQueue(hReqQueue);
    HttpTerminate(HTTP_INITIALIZE_SERVER, NULL);

    return 0;
}

DWORD SendHttpResponse(HANDLE hReqQueue, HTTP_REQUEST* pRequest)
{
    HTTP_RESPONSE response;
    HTTP_DATA_CHUNK dataChunk;
    DWORD result;
    DWORD bytesSent;

    // Prepare the response
    ZeroMemory(&response, sizeof(response));
    response.Headers.KnownHeaders[HttpHeaderContentType].pRawValue = "text/html";
    response.Headers.KnownHeaders[HttpHeaderContentType].RawValueLength = (USHORT)strlen("text/html");
    response.Version = (HTTP_VERSION)HTTP_VERSION_1_1;
    response.StatusCode = (USHORT)200;
    response.ReasonLength = (USHORT)strlen("OK");
    response.pReason = (PCSTR)"OK";
    response.EntityChunkCount = 1;
    dataChunk.DataChunkType = HttpDataChunkFromMemory;
    dataChunk.FromMemory.pBuffer = "Hello, World!";
    dataChunk.FromMemory.BufferLength = (ULONG)strlen("Hello, World!");
    response.pEntityChunks = &dataChunk;

    // Send the response
    result = HttpSendHttpResponse(
        hReqQueue,
        pRequest->RequestId,
        0,
        &response,
        NULL,
        &bytesSent,
        NULL,
        0,
        NULL,
        NULL
    );

    if (result != NO_ERROR)
    {
        wprintf(L"HttpSendHttpResponse failed with %lu \n", result);
    }

    return result;
}

More info about the Http Server API: https://learn.microsoft.com/en-us/windows/win32/http/http-api-start-page