Extensions & Hooks

Request/Response Hooks

Hooks allow running code as part of the Request/Response pipeline. Hooks are typically added using a plugin as shown in this document, but they can also be added globally using Runtime Extensions.

All hooks can be either synchronous or asynchronous. To make your hook asynchronous simply add the async keyword on the function.

Hook: OnResponseSending

The OnResponseSending hook on ZuploContext fires just before the response is sent to the client. The Response can be modified by returning a new Response from this hook. This hook is useful for creating an inbound policy that also needs to run some logic after the response is returned from the handler.

The example below shows a simple tracing policy that adds a trace header to the request and ensures the same header is returned with the response.

import { ZuploContext, ZuploRequest } from "@zuplo/runtime";

export async function tracingPlugin(
  request: ZuploRequest,
  context: ZuploContext,
  policyName: string,
) {
  // Get the trace header
  let traceparent = request.headers.get("traceparent");

  // If not set, add the header to the request
  if (!traceparent) {
    traceparent = crypto.randomUUID();
    const headers = new Headers(request.headers);
    headers.set("traceparent", traceparent);
    return new ZuploRequest(request, { headers });
  }

  context.addResponseSendingHook((response, latestRequest, context) => {
    // If the response doesn't have the trace header that matches, set it
    if (response.headers.get("traceparent") !== traceparent) {
      const headers = new Headers(response.headers);
      headers.set("traceparent", traceparent);
      return new Response(response.body, {
        headers,
      });
    }
    return response;
  });

  return request;
}
ts

Hook: OnResponseSendingFinal

The OnResponseSendingFinal hook on ZuploContext fires immediately before the response is sent to the client. The Response in this hook is immutable. This hook is useful for custom tasks like caching, logging or analytics.

This simple example shows logging the time from the start of this policy until just before the response is sent.

import { ZuploContext, ZuploRequest } from "@zuplo/runtime";

export async function pluginWithHook(
  request: ZuploRequest,
  context: ZuploContext,
  policyName: string,
) {
  const start = Date.now();
  async (response, latestRequest, context) => {
    const end = Date.now();
    const delta = end - start;
    context.log.debug(`request took ${delta}ms`);
  });

  return request;
}
ts

Note that the call can block the response, so if you wish to run an activity asynchronously and not block the response you shouldn't await inside the hook and use context.waitUntil to ensure your asynchronous activity isn't terminated prematurely - this is shown in the example below. This example shows details of the request and final response being sent to an imaginary logging service.

import { ZuploContext, ZuploRequest } from "@zuplo/runtime";

export async function pluginWithHook(
  request: ZuploRequest,
  context: ZuploContext,
  policyName: string,
) {

  // Note we must clone the request here, before it's used by the handler
  const requestClone = request.clone();

  context.addResponseSendingFinalHook(
    async (response, latestRequest, context) => {

      // Note, we must clone the response to read the body
      // as the original will be used the by response pipeline
      // itself
      const responseClone = response.clone();

      const asyncInnerFunction = async () => {
        // Note - we must clone the response
        const requestBody = await requestClone.text();
        const responseBody = await responseClone.text();
        await fetch("https://example.com", {
          method: "GET",
          body: JSON.stringify({ requestBody, responseBody });
        });
      }

      const promise = asyncInnerFunction();

      // Don't block the response while waiting for the fetch
      context.waitUntil(promise);
    });

  return request;
}
ts