195 lines
16 KiB
JavaScript
195 lines
16 KiB
JavaScript
import { trackUsage, updateUsageRecord } from '../middleware/usageTracking.js';
|
|
import { FunctionRegistry, PublicEndpointRegistry, AdminEndpointRegistry } from './registry.js';
|
|
import { logger } from './logger.js';
|
|
/**
|
|
* Decorator/Wrapper to mark an endpoint as public
|
|
* Registers the route in PublicEndpointRegistry
|
|
*/
|
|
export function Public(route) {
|
|
PublicEndpointRegistry.register(route.path, route.method);
|
|
return route;
|
|
}
|
|
/**
|
|
* Decorator/Wrapper to mark an endpoint as admin-only
|
|
* Registers the route in AdminEndpointRegistry
|
|
*/
|
|
export function Admin(route) {
|
|
AdminEndpointRegistry.register(route.path, route.method);
|
|
return route;
|
|
}
|
|
/**
|
|
* Decorator to mark a method as billable
|
|
* Handles usage tracking, context injection, and cancellation
|
|
*/
|
|
export function Billable(options) {
|
|
return function (target, propertyKey, descriptor) {
|
|
const originalMethod = descriptor.value;
|
|
descriptor.value = async function (...args) {
|
|
// 1. Extract context
|
|
// Assumes the first argument is BillableContext, or it's part of the first argument object
|
|
let context;
|
|
if (args.length > 0 && typeof args[0] === 'object') {
|
|
// Check if first arg is context
|
|
if ('userId' in args[0] && 'jobId' in args[0]) {
|
|
context = args[0];
|
|
}
|
|
}
|
|
if (!context) {
|
|
// If no context provided, we can't track usage properly
|
|
// For now, we'll log a warning and proceed without tracking
|
|
// In strict mode, we might want to throw an error
|
|
logger.warn(`[Billable] No context provided for ${options.productId}:${options.actionId}`);
|
|
return originalMethod.apply(this, args);
|
|
}
|
|
// 2. Get config
|
|
const config = FunctionRegistry.get(options.productId, options.actionId);
|
|
if (!config) {
|
|
logger.warn(`[Billable] No config found for ${options.productId}:${options.actionId}`);
|
|
return originalMethod.apply(this, args);
|
|
}
|
|
// 3. Start tracking
|
|
const usageId = await trackUsage({
|
|
userId: context.userId,
|
|
endpoint: 'function', // Internal function call
|
|
method: 'CALL',
|
|
product: options.productId,
|
|
action: options.actionId,
|
|
costUnits: config.costUnits,
|
|
cancellable: options.cancellable || false,
|
|
jobId: context.jobId,
|
|
metadata: context.metadata
|
|
});
|
|
const startTime = Date.now();
|
|
let error = null;
|
|
let result;
|
|
try {
|
|
// 4. Execute method
|
|
// If cancellable, we should ideally wrap the execution or check signal
|
|
if (options.cancellable && context.signal) {
|
|
if (context.signal.aborted) {
|
|
throw new Error('Operation cancelled');
|
|
}
|
|
// Add abort listener
|
|
context.signal.addEventListener('abort', () => {
|
|
logger.info(`[Billable] Job ${context?.jobId} aborted via signal`);
|
|
});
|
|
}
|
|
result = await originalMethod.apply(this, args);
|
|
return result;
|
|
}
|
|
catch (err) {
|
|
error = err;
|
|
throw err;
|
|
}
|
|
finally {
|
|
// 5. End tracking
|
|
if (usageId) {
|
|
const endTime = Date.now();
|
|
await updateUsageRecord({
|
|
usageId,
|
|
responseStatus: error ? 500 : 200,
|
|
responseTimeMs: endTime - startTime,
|
|
error
|
|
});
|
|
}
|
|
}
|
|
};
|
|
return descriptor;
|
|
};
|
|
}
|
|
/**
|
|
* Class Decorator: Registers the worker queue name
|
|
*/
|
|
export function Worker(queueName) {
|
|
return function (constructor) {
|
|
// We can't easily access the instance method 'handler' here without instantiating
|
|
// So we assume the class has a 'handler' method or we register the class itself
|
|
// For simplicity, let's assume we'll instantiate it later or the registry handles it.
|
|
// But wait, pg-boss needs a function.
|
|
// Let's store the constructor in the registry, and the registry (or bootstrap) will instantiate and bind.
|
|
// Actually, let's just attach the queue name to the class for now,
|
|
// and let a separate scanner or manual registration use it.
|
|
// OR, we can register a factory.
|
|
// Better approach for now: Register the prototype's handler if it exists.
|
|
// But 'handler' is on the instance usually.
|
|
// Let's just modify the class to have a static 'queueName' property
|
|
// and register it.
|
|
constructor.queueName = queueName;
|
|
};
|
|
}
|
|
import { getCache } from './cache/index.js';
|
|
const defaultKeyInfo = (c) => {
|
|
const url = new URL(c.req.url);
|
|
url.searchParams.sort();
|
|
return `auto-cache:${c.req.method}:${url.pathname}${url.search}`;
|
|
};
|
|
export const CachedHandler = (handler, options) => async (c) => {
|
|
const opts = options || {};
|
|
const ttl = opts.ttl || 300;
|
|
const varyByAuth = opts.varyByAuth || false;
|
|
const skipAuth = opts.skipAuth !== undefined ? opts.skipAuth : !varyByAuth; // Default true unless varyByAuth is true
|
|
const maxSizeBytes = opts.maxSizeBytes || 1024 * 1024; // 1MB
|
|
const keyGen = opts.keyGenerator || defaultKeyInfo;
|
|
// 1. Auth Bypass
|
|
const authHeader = c.req.header('Authorization');
|
|
if (skipAuth && authHeader) {
|
|
// Explicitly mark as skipped due to auth
|
|
c.header('X-Cache', 'SKIP');
|
|
return handler(c);
|
|
}
|
|
const cache = getCache();
|
|
let key = keyGen(c);
|
|
// Append Auth to key if requested (User Isolation)
|
|
if (varyByAuth && authHeader) {
|
|
key += `|auth=${authHeader}`;
|
|
}
|
|
const bypass = c.req.query('cache') === 'false' || c.req.query('nocache') === 'true';
|
|
// 2. Hit
|
|
if (!bypass) {
|
|
const cached = await cache.get(key);
|
|
if (cached) {
|
|
c.header('X-Cache', 'HIT');
|
|
const cachedVal = cached;
|
|
if (cachedVal.contentType)
|
|
c.header('Content-Type', cachedVal.contentType);
|
|
if (varyByAuth)
|
|
c.header('Vary', 'Authorization');
|
|
return c.body(cachedVal.data);
|
|
}
|
|
}
|
|
// 3. Miss
|
|
const response = await handler(c);
|
|
// 4. Save
|
|
if (response instanceof Response && response.ok) {
|
|
const cloned = response.clone();
|
|
try {
|
|
const contentType = response.headers.get('Content-Type') || 'application/json';
|
|
let data;
|
|
// Check content length if available
|
|
const contentLength = cloned.headers.get('Content-Length');
|
|
if (contentLength && parseInt(contentLength) > maxSizeBytes) {
|
|
return response;
|
|
}
|
|
if (contentType.includes('application/json')) {
|
|
const jsonObj = await cloned.json();
|
|
data = JSON.stringify(jsonObj);
|
|
}
|
|
else {
|
|
data = await cloned.text();
|
|
}
|
|
// Double check actual size after reading
|
|
if (data.length > maxSizeBytes) {
|
|
return response;
|
|
}
|
|
await cache.set(key, { data, contentType }, ttl);
|
|
c.header('X-Cache', bypass ? 'BYPASS' : 'MISS');
|
|
if (varyByAuth)
|
|
c.header('Vary', 'Authorization');
|
|
}
|
|
catch (e) {
|
|
logger.error({ err: e }, 'Cache interception failed');
|
|
}
|
|
}
|
|
return response;
|
|
};
|
|
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../src/commons/decorators.ts"],"names":[],"mappings":"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"}
|