ContextData - Sharing Request Data
It's often useful to store data throughout the life of a single request. This
data could be used on multiple policies, handlers, or for logging. However,
because the Zuplo runtime is asynchronous, you can't simply create global
variables and reference them in other modules if the value is unique across
requests. Additionally, using a traditional data structure like a Map
is also
not recommended as it can build up memory over time.
ContextData
The ContextData
utility store data that's tied to a specific request.
Additionally, this utility ensures that data stored is garbage collected
(removed from memory) when the request is complete.
ContextData.set
ContextData.set(context, "my-data", { prop1: "hello world" });ts
ContextData.get
const data = ContextData.get(context, "my-data");ts
set
(instance function)
const myData = new ContextData("my-data"); myData.set(context, { prop1: "hello world" });ts
get
(instance function)
const myData = new ContextData("my-data"); const data = myData.get(context);ts
Typing
The methods of ContextData
support generics in order to support typing.
const myData = new ContextData<{ key: string }>("my-data"); ContextData.get<{ key: string }>(context, "my-data"); ContextData.set<{ key: string }>(context, "my-data", { key: "hello" });ts
How NOT to Share Data
Below are a few examples of how NOT to share data in your API.
Don't write code like this in your API. It won't work reliably. These are examples of what NOT to do. See the next section for best practices.
The first example uses a simple shared global variable called currentRequestId
to store the current requestId
. On the surface this looks straightforward.
However, because the gateway is being shared among many requests who are all
running at the same time, the value of currentRequestId
is completely
unpredictable when your API is under load.
let currentRequestId: string | undefined; export function myFirstPolicy(request: ZuploRequest, context: ZuploContext) { currentRequestId = context.requestId; context.log.info(`The current requestId is: ${currentRequestId}`); } export function mySecondPolicy(request: ZuploRequest, context: ZuploContext) { currentRequestId = context.requestId; context.log.info(`The current requestId is: ${currentRequestId}`); }ts
Using ContextData
Below you will find examples of how to use ContextData
to safely share data
throughout the lifecycle of a request.
Using static methods
import { ContextData, ZuploContext, ZuploRequest } from "@zuplo/runtime"; export function myFirstPolicy(request: ZuploRequest, context: ZuploContext) { ContextData.set(context, "currentRequestId", context.requestId); const currentRequestId = ContextData.get(context, "currentRequestId"); context.log.info(`The current requestId is: ${currentRequestId}`); }ts
Using Shared Variable
Reusable class shared across modules The shared module allows other modules to access the same storage without passing the string name or type around.
export const myData = new ContextData<{ prop1: string }>("my-data");ts
Then use the class in another module.
import { myData } from "./my-module"; const data = myData.get(context); myData.set(context, data);ts