Reference

Monetization Programmatic Quotas

When adding monetization to your API, you would usually set the number of meters a request will consume in the settings of the Monetization Policy. For example, the policy below specifies that each request will consume 1 requests meter and 5 computeUnits meters.

{
  "export": "MonetizationInboundPolicy",
  "module": "$import(@zuplo/runtime)",
  "options": {
    "allowRequestsOverQuota": false,
    "allowedSubscriptionStatuses": ["active", "incomplete"],
    "meterOnStatusCodes": "200-399",
    "meters": {
      "requests": 1,
      "computeUnits": 5
    }
  }
}
json

However, in some cases, you may not know up front how many units of a particular meter will be consumed until after the response is sent. For example, maybe your backend is responsible for computing the computeUnits on a request and send the result in the response in the compute-units header.

In Zuplo, you can support these dynamic meters by writing a little code. To make the computeUnits meter dynamic, first update the policy by setting the computeUnits meter to 0 as shown below.

{
  "export": "MonetizationInboundPolicy",
  "module": "$import(@zuplo/runtime)",
  "options": {
    "allowRequestsOverQuota": false,
    "allowedSubscriptionStatuses": ["active", "incomplete"],
    "meterOnStatusCodes": "200-399",
    "meters": {
      "requests": 1,
      "computeUnits": 0
    }
  }
}
json

Next you can create a custom code outbound policy that reads data from the Response (in this case the compute-units header) and sets the meter programmatically.

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

export default async function (
  response: Response,
  request: ZuploRequest,
  context: ZuploContext,
  options: any,
  policyName: string,
) {
  const headerValue = response.headers.get("compute-units");
  let computeUnitsValue;
  if (headerValue && typeof headerValue === "string") {
    computeUnitsValue = parseInt(headerValue);
  }

  // Throw an error if the server doesn't send compute units
  // Alternatively, you could have a default value
  if (!computeUnitsValue) {
    throw new Error("Invalid response, no compute units sent.");
  }

  // Set the compute units for the request
  MonetizationInboundPolicy.setMeters(context, {
    computeUnits: computeUnitsValue,
  });

  return response;
}
ts