agent-smith/dist-in/commons/decorators.js
2026-02-26 19:41:09 +01:00

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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"}