@venturekit/runtime 0.0.0-dev.20260307234057

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,176 @@
1
+ "use strict";
2
+ /**
3
+ * VentureKit Handler
4
+ *
5
+ * Single unified Lambda handler that adapts to context:
6
+ * - No scopes = public endpoint
7
+ * - With scopes = authenticated + scope check
8
+ * - Status code auto-detected from HTTP method or explicitly set
9
+ */
10
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
11
+ if (k2 === undefined) k2 = k;
12
+ var desc = Object.getOwnPropertyDescriptor(m, k);
13
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
14
+ desc = { enumerable: true, get: function() { return m[k]; } };
15
+ }
16
+ Object.defineProperty(o, k2, desc);
17
+ }) : (function(o, m, k, k2) {
18
+ if (k2 === undefined) k2 = k;
19
+ o[k2] = m[k];
20
+ }));
21
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
22
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
23
+ }) : function(o, v) {
24
+ o["default"] = v;
25
+ });
26
+ var __importStar = (this && this.__importStar) || (function () {
27
+ var ownKeys = function(o) {
28
+ ownKeys = Object.getOwnPropertyNames || function (o) {
29
+ var ar = [];
30
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
31
+ return ar;
32
+ };
33
+ return ownKeys(o);
34
+ };
35
+ return function (mod) {
36
+ if (mod && mod.__esModule) return mod;
37
+ var result = {};
38
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
39
+ __setModuleDefault(result, mod);
40
+ return result;
41
+ };
42
+ })();
43
+ Object.defineProperty(exports, "__esModule", { value: true });
44
+ exports.handler = handler;
45
+ const context_1 = require("./context");
46
+ const logger_1 = require("./logger");
47
+ const response_1 = require("./response");
48
+ const middleware_1 = require("./middleware");
49
+ const errors_1 = require("./errors");
50
+ /**
51
+ * Parse request body
52
+ */
53
+ function parseBody(event) {
54
+ if (!event.body)
55
+ return null;
56
+ try {
57
+ const body = event.isBase64Encoded
58
+ ? Buffer.from(event.body, 'base64').toString('utf-8')
59
+ : event.body;
60
+ return JSON.parse(body);
61
+ }
62
+ catch {
63
+ return null;
64
+ }
65
+ }
66
+ /**
67
+ * Auto-detect status code from HTTP method
68
+ */
69
+ function getStatusCode(method, explicit) {
70
+ if (explicit !== undefined)
71
+ return explicit;
72
+ switch (method.toUpperCase()) {
73
+ case 'POST':
74
+ return 201;
75
+ case 'DELETE':
76
+ return 204;
77
+ default:
78
+ return 200;
79
+ }
80
+ }
81
+ /**
82
+ * VentureKit unified handler
83
+ *
84
+ * Adapts to context with smart defaults:
85
+ * - No scopes = public endpoint
86
+ * - With scopes = authenticated + scope check
87
+ * - Status: GET/PUT → 200, POST → 201, DELETE → 204 (or set explicitly)
88
+ *
89
+ * @example
90
+ * ```typescript
91
+ * // Public endpoint
92
+ * export const main = handler(async (body, ctx, log) => {
93
+ * return { status: 'ok' };
94
+ * });
95
+ *
96
+ * // Authenticated endpoint
97
+ * export const main = handler(async (body, ctx, log) => {
98
+ * return { id: '123' };
99
+ * }, { scopes: ['projects.write'] });
100
+ * ```
101
+ */
102
+ function handler(fn, config = {}) {
103
+ const { scopes = [], status, middleware = [], logLevel = 'info', } = config;
104
+ const requiresAuth = scopes.length > 0;
105
+ const logger = (0, logger_1.createLogger)({ minLevel: logLevel });
106
+ // Build middleware stack
107
+ const middlewareStack = [
108
+ // Error boundary (outermost)
109
+ (0, middleware_1.errorBoundaryMiddleware)((error, ctx) => (0, response_1.errorResponse)(error, { requestId: ctx.requestId })),
110
+ // Logging
111
+ (0, middleware_1.loggingMiddleware)(logger),
112
+ // Custom middleware
113
+ ...middleware,
114
+ ];
115
+ const composedMiddleware = (0, middleware_1.compose)(middlewareStack);
116
+ return async (event, _context) => {
117
+ // Build request context
118
+ const ctx = (0, context_1.buildContext)(event, { supportedLocales: ['en'], defaultLocale: 'en' });
119
+ logger.setContext(ctx);
120
+ // Run through middleware and handler
121
+ return composedMiddleware(ctx, async () => {
122
+ // Auth check (only if scopes provided)
123
+ if (requiresAuth) {
124
+ if (!ctx.user) {
125
+ throw new errors_1.UnauthorizedError();
126
+ }
127
+ // Scope check
128
+ const hasAllScopes = scopes.every(scope => ctx.user.scopes.includes(scope));
129
+ if (!hasAllScopes) {
130
+ throw new errors_1.ForbiddenError('Insufficient permissions');
131
+ }
132
+ }
133
+ // Parse body
134
+ const body = parseBody(event);
135
+ // Transaction support
136
+ if (config.transactional) {
137
+ const data = await Promise.resolve().then(() => __importStar(require('@venturekit/data')));
138
+ const tx = await data.beginTransaction();
139
+ ctx.tx = tx;
140
+ try {
141
+ const result = await fn(body, ctx, logger);
142
+ await tx.commit();
143
+ const statusCode = getStatusCode(event.requestContext.http.method, status);
144
+ if (statusCode === 204) {
145
+ return (0, response_1.noContent)({ requestId: ctx.requestId });
146
+ }
147
+ else if (statusCode === 201) {
148
+ return (0, response_1.created)(result, { requestId: ctx.requestId });
149
+ }
150
+ else {
151
+ return (0, response_1.success)(result, { requestId: ctx.requestId });
152
+ }
153
+ }
154
+ catch (txError) {
155
+ await tx.rollback();
156
+ throw txError;
157
+ }
158
+ }
159
+ // Execute handler (non-transactional)
160
+ const result = await fn(body, ctx, logger);
161
+ // Determine status code
162
+ const statusCode = getStatusCode(event.requestContext.http.method, status);
163
+ // Return response based on status
164
+ if (statusCode === 204) {
165
+ return (0, response_1.noContent)({ requestId: ctx.requestId });
166
+ }
167
+ else if (statusCode === 201) {
168
+ return (0, response_1.created)(result, { requestId: ctx.requestId });
169
+ }
170
+ else {
171
+ return (0, response_1.success)(result, { requestId: ctx.requestId });
172
+ }
173
+ });
174
+ };
175
+ }
176
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"handler.js","sourceRoot":"","sources":["../src/handler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * VentureKit Handler\n * \n * Single unified Lambda handler that adapts to context:\n * - No scopes = public endpoint\n * - With scopes = authenticated + scope check\n * - Status code auto-detected from HTTP method or explicitly set\n */\n\nimport type { APIGatewayProxyEventV2, APIGatewayProxyResultV2, Context } from 'aws-lambda';\nimport { RequestContext, buildContext } from './context';\nimport { Logger, createLogger } from './logger';\nimport { success, created, noContent, errorResponse } from './response';\nimport { compose, Middleware, loggingMiddleware, errorBoundaryMiddleware } from './middleware';\nimport { UnauthorizedError, ForbiddenError } from './errors';\n\n/**\n * Handler function signature\n */\nexport type HandlerFn<TBody = unknown, TResult = unknown> = (\n  body: TBody,\n  ctx: RequestContext,\n  logger: Logger\n) => Promise<TResult>;\n\n/**\n * HTTP Status codes\n */\nexport type StatusCode = 200 | 201 | 204;\n\n/**\n * Handler configuration\n */\nexport interface HandlerConfig {\n  /** \n   * Required OAuth scopes\n   * - Empty array or undefined = public endpoint (no auth)\n   * - Non-empty array = authenticated + scope check\n   */\n  scopes?: string[];\n  \n  /** \n   * Response status code\n   * - undefined = auto-detect from HTTP method (GET/PUT=200, POST=201, DELETE=204)\n   * - number = explicit status code\n   */\n  status?: StatusCode;\n  \n  /** Custom middleware to apply */\n  middleware?: Middleware[];\n  \n  /** Log level */\n  logLevel?: 'debug' | 'info' | 'warn' | 'error';\n\n  /**\n   * Wrap the entire handler execution in a database transaction.\n   * When true, a transaction is started before the handler runs\n   * and is available via ctx.tx. It auto-commits on success\n   * and auto-rolls-back on error.\n   * Requires @venturekit/data to be installed.\n   */\n  transactional?: boolean;\n}\n\n/**\n * Parse request body\n */\nfunction parseBody<T>(event: APIGatewayProxyEventV2): T | null {\n  if (!event.body) return null;\n  \n  try {\n    const body = event.isBase64Encoded\n      ? Buffer.from(event.body, 'base64').toString('utf-8')\n      : event.body;\n    return JSON.parse(body) as T;\n  } catch {\n    return null;\n  }\n}\n\n/**\n * Auto-detect status code from HTTP method\n */\nfunction getStatusCode(method: string, explicit?: StatusCode): StatusCode {\n  if (explicit !== undefined) return explicit;\n  \n  switch (method.toUpperCase()) {\n    case 'POST':\n      return 201;\n    case 'DELETE':\n      return 204;\n    default:\n      return 200;\n  }\n}\n\n/**\n * VentureKit unified handler\n * \n * Adapts to context with smart defaults:\n * - No scopes = public endpoint\n * - With scopes = authenticated + scope check\n * - Status: GET/PUT → 200, POST → 201, DELETE → 204 (or set explicitly)\n * \n * @example\n * ```typescript\n * // Public endpoint\n * export const main = handler(async (body, ctx, log) => {\n *   return { status: 'ok' };\n * });\n * \n * // Authenticated endpoint\n * export const main = handler(async (body, ctx, log) => {\n *   return { id: '123' };\n * }, { scopes: ['projects.write'] });\n * ```\n */\nexport function handler<TBody = unknown, TResult = unknown>(\n  fn: HandlerFn<TBody, TResult>,\n  config: HandlerConfig = {}\n): (event: APIGatewayProxyEventV2, context: Context) => Promise<APIGatewayProxyResultV2> {\n  const {\n    scopes = [],\n    status,\n    middleware = [],\n    logLevel = 'info',\n  } = config;\n\n  const requiresAuth = scopes.length > 0;\n  const logger = createLogger({ minLevel: logLevel });\n\n  // Build middleware stack\n  const middlewareStack: Middleware[] = [\n    // Error boundary (outermost)\n    errorBoundaryMiddleware((error, ctx) => errorResponse(error, { requestId: ctx.requestId })),\n    // Logging\n    loggingMiddleware(logger),\n    // Custom middleware\n    ...middleware,\n  ];\n\n  const composedMiddleware = compose(middlewareStack);\n\n  return async (event: APIGatewayProxyEventV2, _context: Context): Promise<APIGatewayProxyResultV2> => {\n    // Build request context\n    const ctx = buildContext(event, { supportedLocales: ['en'], defaultLocale: 'en' });\n    logger.setContext(ctx);\n\n    // Run through middleware and handler\n    return composedMiddleware(ctx, async () => {\n      // Auth check (only if scopes provided)\n      if (requiresAuth) {\n        if (!ctx.user) {\n          throw new UnauthorizedError();\n        }\n\n        // Scope check\n        const hasAllScopes = scopes.every(scope => ctx.user!.scopes.includes(scope));\n        if (!hasAllScopes) {\n          throw new ForbiddenError('Insufficient permissions');\n        }\n      }\n\n      // Parse body\n      const body = parseBody<TBody>(event);\n\n      // Transaction support\n      if (config.transactional) {\n        const data = await import('@venturekit/data');\n        const tx = await data.beginTransaction();\n        ctx.tx = tx;\n        try {\n          const result = await fn(body as TBody, ctx, logger);\n          await tx.commit();\n          const statusCode = getStatusCode(event.requestContext.http.method, status);\n          if (statusCode === 204) {\n            return noContent({ requestId: ctx.requestId });\n          } else if (statusCode === 201) {\n            return created(result, { requestId: ctx.requestId });\n          } else {\n            return success(result, { requestId: ctx.requestId });\n          }\n        } catch (txError) {\n          await tx.rollback();\n          throw txError;\n        }\n      }\n\n      // Execute handler (non-transactional)\n      const result = await fn(body as TBody, ctx, logger);\n\n      // Determine status code\n      const statusCode = getStatusCode(event.requestContext.http.method, status);\n\n      // Return response based on status\n      if (statusCode === 204) {\n        return noContent({ requestId: ctx.requestId });\n      } else if (statusCode === 201) {\n        return created(result, { requestId: ctx.requestId });\n      } else {\n        return success(result, { requestId: ctx.requestId });\n      }\n    });\n  };\n}\n\n"]}
@@ -0,0 +1,18 @@
1
+ /**
2
+ * @venturekit/runtime
3
+ *
4
+ * Runtime utilities for VentureKit Lambda functions.
5
+ */
6
+ export { handler } from './handler';
7
+ export type { HandlerFn, HandlerConfig, StatusCode } from './handler';
8
+ export { buildContext, extractUserContext, extractLocale, } from './context';
9
+ export type { RequestContext, UserContext, TenantContext } from './context';
10
+ export { success, created, noContent, error, errorResponse, redirect, } from './response';
11
+ export type { SuccessResponse, ErrorResponse } from './response';
12
+ export { VentureError, BadRequestError, UnauthorizedError, ForbiddenError, NotFoundError, ConflictError, ValidationError, RateLimitError, InternalError, ServiceUnavailableError, isVentureError, } from './errors';
13
+ export { Logger, createLogger, logger, } from './logger';
14
+ export type { LogLevel, LogEntry, LoggerConfig } from './logger';
15
+ export { compose, loggingMiddleware, corsMiddleware, timeoutMiddleware, errorBoundaryMiddleware, } from './middleware';
16
+ export type { MiddlewareFn, Middleware } from './middleware';
17
+ export { connectionStore } from './ws';
18
+ export type { ConnectionRecord, ConnectionMetadata } from './ws';
package/dist/index.js ADDED
@@ -0,0 +1,53 @@
1
+ "use strict";
2
+ /**
3
+ * @venturekit/runtime
4
+ *
5
+ * Runtime utilities for VentureKit Lambda functions.
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ exports.connectionStore = exports.errorBoundaryMiddleware = exports.timeoutMiddleware = exports.corsMiddleware = exports.loggingMiddleware = exports.compose = exports.logger = exports.createLogger = exports.Logger = exports.isVentureError = exports.ServiceUnavailableError = exports.InternalError = exports.RateLimitError = exports.ValidationError = exports.ConflictError = exports.NotFoundError = exports.ForbiddenError = exports.UnauthorizedError = exports.BadRequestError = exports.VentureError = exports.redirect = exports.errorResponse = exports.error = exports.noContent = exports.created = exports.success = exports.extractLocale = exports.extractUserContext = exports.buildContext = exports.handler = void 0;
9
+ // Handler
10
+ var handler_1 = require("./handler");
11
+ Object.defineProperty(exports, "handler", { enumerable: true, get: function () { return handler_1.handler; } });
12
+ // Context
13
+ var context_1 = require("./context");
14
+ Object.defineProperty(exports, "buildContext", { enumerable: true, get: function () { return context_1.buildContext; } });
15
+ Object.defineProperty(exports, "extractUserContext", { enumerable: true, get: function () { return context_1.extractUserContext; } });
16
+ Object.defineProperty(exports, "extractLocale", { enumerable: true, get: function () { return context_1.extractLocale; } });
17
+ // Response utilities
18
+ var response_1 = require("./response");
19
+ Object.defineProperty(exports, "success", { enumerable: true, get: function () { return response_1.success; } });
20
+ Object.defineProperty(exports, "created", { enumerable: true, get: function () { return response_1.created; } });
21
+ Object.defineProperty(exports, "noContent", { enumerable: true, get: function () { return response_1.noContent; } });
22
+ Object.defineProperty(exports, "error", { enumerable: true, get: function () { return response_1.error; } });
23
+ Object.defineProperty(exports, "errorResponse", { enumerable: true, get: function () { return response_1.errorResponse; } });
24
+ Object.defineProperty(exports, "redirect", { enumerable: true, get: function () { return response_1.redirect; } });
25
+ // Errors
26
+ var errors_1 = require("./errors");
27
+ Object.defineProperty(exports, "VentureError", { enumerable: true, get: function () { return errors_1.VentureError; } });
28
+ Object.defineProperty(exports, "BadRequestError", { enumerable: true, get: function () { return errors_1.BadRequestError; } });
29
+ Object.defineProperty(exports, "UnauthorizedError", { enumerable: true, get: function () { return errors_1.UnauthorizedError; } });
30
+ Object.defineProperty(exports, "ForbiddenError", { enumerable: true, get: function () { return errors_1.ForbiddenError; } });
31
+ Object.defineProperty(exports, "NotFoundError", { enumerable: true, get: function () { return errors_1.NotFoundError; } });
32
+ Object.defineProperty(exports, "ConflictError", { enumerable: true, get: function () { return errors_1.ConflictError; } });
33
+ Object.defineProperty(exports, "ValidationError", { enumerable: true, get: function () { return errors_1.ValidationError; } });
34
+ Object.defineProperty(exports, "RateLimitError", { enumerable: true, get: function () { return errors_1.RateLimitError; } });
35
+ Object.defineProperty(exports, "InternalError", { enumerable: true, get: function () { return errors_1.InternalError; } });
36
+ Object.defineProperty(exports, "ServiceUnavailableError", { enumerable: true, get: function () { return errors_1.ServiceUnavailableError; } });
37
+ Object.defineProperty(exports, "isVentureError", { enumerable: true, get: function () { return errors_1.isVentureError; } });
38
+ // Logger
39
+ var logger_1 = require("./logger");
40
+ Object.defineProperty(exports, "Logger", { enumerable: true, get: function () { return logger_1.Logger; } });
41
+ Object.defineProperty(exports, "createLogger", { enumerable: true, get: function () { return logger_1.createLogger; } });
42
+ Object.defineProperty(exports, "logger", { enumerable: true, get: function () { return logger_1.logger; } });
43
+ // Middleware
44
+ var middleware_1 = require("./middleware");
45
+ Object.defineProperty(exports, "compose", { enumerable: true, get: function () { return middleware_1.compose; } });
46
+ Object.defineProperty(exports, "loggingMiddleware", { enumerable: true, get: function () { return middleware_1.loggingMiddleware; } });
47
+ Object.defineProperty(exports, "corsMiddleware", { enumerable: true, get: function () { return middleware_1.corsMiddleware; } });
48
+ Object.defineProperty(exports, "timeoutMiddleware", { enumerable: true, get: function () { return middleware_1.timeoutMiddleware; } });
49
+ Object.defineProperty(exports, "errorBoundaryMiddleware", { enumerable: true, get: function () { return middleware_1.errorBoundaryMiddleware; } });
50
+ // WebSocket connection store
51
+ var ws_1 = require("./ws");
52
+ Object.defineProperty(exports, "connectionStore", { enumerable: true, get: function () { return ws_1.connectionStore; } });
53
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7O0dBSUc7OztBQUVILFVBQVU7QUFDVixxQ0FBb0M7QUFBM0Isa0dBQUEsT0FBTyxPQUFBO0FBR2hCLFVBQVU7QUFDVixxQ0FJbUI7QUFIakIsdUdBQUEsWUFBWSxPQUFBO0FBQ1osNkdBQUEsa0JBQWtCLE9BQUE7QUFDbEIsd0dBQUEsYUFBYSxPQUFBO0FBSWYscUJBQXFCO0FBQ3JCLHVDQU9vQjtBQU5sQixtR0FBQSxPQUFPLE9BQUE7QUFDUCxtR0FBQSxPQUFPLE9BQUE7QUFDUCxxR0FBQSxTQUFTLE9BQUE7QUFDVCxpR0FBQSxLQUFLLE9BQUE7QUFDTCx5R0FBQSxhQUFhLE9BQUE7QUFDYixvR0FBQSxRQUFRLE9BQUE7QUFJVixTQUFTO0FBQ1QsbUNBWWtCO0FBWGhCLHNHQUFBLFlBQVksT0FBQTtBQUNaLHlHQUFBLGVBQWUsT0FBQTtBQUNmLDJHQUFBLGlCQUFpQixPQUFBO0FBQ2pCLHdHQUFBLGNBQWMsT0FBQTtBQUNkLHVHQUFBLGFBQWEsT0FBQTtBQUNiLHVHQUFBLGFBQWEsT0FBQTtBQUNiLHlHQUFBLGVBQWUsT0FBQTtBQUNmLHdHQUFBLGNBQWMsT0FBQTtBQUNkLHVHQUFBLGFBQWEsT0FBQTtBQUNiLGlIQUFBLHVCQUF1QixPQUFBO0FBQ3ZCLHdHQUFBLGNBQWMsT0FBQTtBQUdoQixTQUFTO0FBQ1QsbUNBSWtCO0FBSGhCLGdHQUFBLE1BQU0sT0FBQTtBQUNOLHNHQUFBLFlBQVksT0FBQTtBQUNaLGdHQUFBLE1BQU0sT0FBQTtBQUlSLGFBQWE7QUFDYiwyQ0FNc0I7QUFMcEIscUdBQUEsT0FBTyxPQUFBO0FBQ1AsK0dBQUEsaUJBQWlCLE9BQUE7QUFDakIsNEdBQUEsY0FBYyxPQUFBO0FBQ2QsK0dBQUEsaUJBQWlCLE9BQUE7QUFDakIscUhBQUEsdUJBQXVCLE9BQUE7QUFJekIsNkJBQTZCO0FBQzdCLDJCQUF1QztBQUE5QixxR0FBQSxlQUFlLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEB2ZW50dXJla2l0L3J1bnRpbWVcbiAqIFxuICogUnVudGltZSB1dGlsaXRpZXMgZm9yIFZlbnR1cmVLaXQgTGFtYmRhIGZ1bmN0aW9ucy5cbiAqL1xuXG4vLyBIYW5kbGVyXG5leHBvcnQgeyBoYW5kbGVyIH0gZnJvbSAnLi9oYW5kbGVyJztcbmV4cG9ydCB0eXBlIHsgSGFuZGxlckZuLCBIYW5kbGVyQ29uZmlnLCBTdGF0dXNDb2RlIH0gZnJvbSAnLi9oYW5kbGVyJztcblxuLy8gQ29udGV4dFxuZXhwb3J0IHtcbiAgYnVpbGRDb250ZXh0LFxuICBleHRyYWN0VXNlckNvbnRleHQsXG4gIGV4dHJhY3RMb2NhbGUsXG59IGZyb20gJy4vY29udGV4dCc7XG5leHBvcnQgdHlwZSB7IFJlcXVlc3RDb250ZXh0LCBVc2VyQ29udGV4dCwgVGVuYW50Q29udGV4dCB9IGZyb20gJy4vY29udGV4dCc7XG5cbi8vIFJlc3BvbnNlIHV0aWxpdGllc1xuZXhwb3J0IHtcbiAgc3VjY2VzcyxcbiAgY3JlYXRlZCxcbiAgbm9Db250ZW50LFxuICBlcnJvcixcbiAgZXJyb3JSZXNwb25zZSxcbiAgcmVkaXJlY3QsXG59IGZyb20gJy4vcmVzcG9uc2UnO1xuZXhwb3J0IHR5cGUgeyBTdWNjZXNzUmVzcG9uc2UsIEVycm9yUmVzcG9uc2UgfSBmcm9tICcuL3Jlc3BvbnNlJztcblxuLy8gRXJyb3JzXG5leHBvcnQge1xuICBWZW50dXJlRXJyb3IsXG4gIEJhZFJlcXVlc3RFcnJvcixcbiAgVW5hdXRob3JpemVkRXJyb3IsXG4gIEZvcmJpZGRlbkVycm9yLFxuICBOb3RGb3VuZEVycm9yLFxuICBDb25mbGljdEVycm9yLFxuICBWYWxpZGF0aW9uRXJyb3IsXG4gIFJhdGVMaW1pdEVycm9yLFxuICBJbnRlcm5hbEVycm9yLFxuICBTZXJ2aWNlVW5hdmFpbGFibGVFcnJvcixcbiAgaXNWZW50dXJlRXJyb3IsXG59IGZyb20gJy4vZXJyb3JzJztcblxuLy8gTG9nZ2VyXG5leHBvcnQge1xuICBMb2dnZXIsXG4gIGNyZWF0ZUxvZ2dlcixcbiAgbG9nZ2VyLFxufSBmcm9tICcuL2xvZ2dlcic7XG5leHBvcnQgdHlwZSB7IExvZ0xldmVsLCBMb2dFbnRyeSwgTG9nZ2VyQ29uZmlnIH0gZnJvbSAnLi9sb2dnZXInO1xuXG4vLyBNaWRkbGV3YXJlXG5leHBvcnQge1xuICBjb21wb3NlLFxuICBsb2dnaW5nTWlkZGxld2FyZSxcbiAgY29yc01pZGRsZXdhcmUsXG4gIHRpbWVvdXRNaWRkbGV3YXJlLFxuICBlcnJvckJvdW5kYXJ5TWlkZGxld2FyZSxcbn0gZnJvbSAnLi9taWRkbGV3YXJlJztcbmV4cG9ydCB0eXBlIHsgTWlkZGxld2FyZUZuLCBNaWRkbGV3YXJlIH0gZnJvbSAnLi9taWRkbGV3YXJlJztcblxuLy8gV2ViU29ja2V0IGNvbm5lY3Rpb24gc3RvcmVcbmV4cG9ydCB7IGNvbm5lY3Rpb25TdG9yZSB9IGZyb20gJy4vd3MnO1xuZXhwb3J0IHR5cGUgeyBDb25uZWN0aW9uUmVjb3JkLCBDb25uZWN0aW9uTWV0YWRhdGEgfSBmcm9tICcuL3dzJztcbiJdfQ==
@@ -0,0 +1,72 @@
1
+ /**
2
+ * VentureKit Logger
3
+ *
4
+ * Structured logging for Lambda functions, powered by pino.
5
+ * Outputs JSON for CloudWatch Logs Insights queries.
6
+ *
7
+ * Uses pino under the hood for:
8
+ * - Fast structured JSON serialization
9
+ * - Automatic level filtering
10
+ * - Child logger support with bound fields
11
+ * - Lambda-optimized (no worker threads, synchronous output)
12
+ */
13
+ import { RequestContext } from './context';
14
+ /**
15
+ * Log levels
16
+ */
17
+ export type LogLevel = 'debug' | 'info' | 'warn' | 'error';
18
+ /**
19
+ * Structured log entry (for type compatibility)
20
+ */
21
+ export interface LogEntry {
22
+ level: LogLevel;
23
+ message: string;
24
+ timestamp: string;
25
+ requestId?: string;
26
+ userId?: string;
27
+ tenantId?: string;
28
+ duration?: number;
29
+ [key: string]: unknown;
30
+ }
31
+ /**
32
+ * Logger configuration
33
+ */
34
+ export interface LoggerConfig {
35
+ /** Minimum log level to output */
36
+ minLevel: LogLevel;
37
+ /** Include request context in all logs */
38
+ includeContext: boolean;
39
+ }
40
+ /**
41
+ * Logger instance — thin wrapper over pino
42
+ */
43
+ export declare class Logger {
44
+ private pinoLogger;
45
+ private context;
46
+ private config;
47
+ constructor(config?: Partial<LoggerConfig>);
48
+ /**
49
+ * Set request context for all subsequent logs
50
+ */
51
+ setContext(context: RequestContext): void;
52
+ /**
53
+ * Clear request context
54
+ */
55
+ clearContext(): void;
56
+ /**
57
+ * Create a child logger with additional default fields
58
+ */
59
+ child(fields: Record<string, unknown>): Logger;
60
+ debug(message: string, data?: Record<string, unknown>): void;
61
+ info(message: string, data?: Record<string, unknown>): void;
62
+ warn(message: string, data?: Record<string, unknown>): void;
63
+ error(message: string, data?: Record<string, unknown>): void;
64
+ }
65
+ /**
66
+ * Default logger instance
67
+ */
68
+ export declare const logger: Logger;
69
+ /**
70
+ * Create a new logger with custom config
71
+ */
72
+ export declare function createLogger(config?: Partial<LoggerConfig>): Logger;
package/dist/logger.js ADDED
@@ -0,0 +1,105 @@
1
+ "use strict";
2
+ /**
3
+ * VentureKit Logger
4
+ *
5
+ * Structured logging for Lambda functions, powered by pino.
6
+ * Outputs JSON for CloudWatch Logs Insights queries.
7
+ *
8
+ * Uses pino under the hood for:
9
+ * - Fast structured JSON serialization
10
+ * - Automatic level filtering
11
+ * - Child logger support with bound fields
12
+ * - Lambda-optimized (no worker threads, synchronous output)
13
+ */
14
+ var __importDefault = (this && this.__importDefault) || function (mod) {
15
+ return (mod && mod.__esModule) ? mod : { "default": mod };
16
+ };
17
+ Object.defineProperty(exports, "__esModule", { value: true });
18
+ exports.logger = exports.Logger = void 0;
19
+ exports.createLogger = createLogger;
20
+ const pino_1 = __importDefault(require("pino"));
21
+ /**
22
+ * Logger instance — thin wrapper over pino
23
+ */
24
+ class Logger {
25
+ pinoLogger;
26
+ context = null;
27
+ config;
28
+ constructor(config = {}) {
29
+ this.config = {
30
+ minLevel: config.minLevel ?? 'info',
31
+ includeContext: config.includeContext ?? true,
32
+ };
33
+ this.pinoLogger = (0, pino_1.default)({
34
+ level: this.config.minLevel,
35
+ // Lambda-friendly: no worker threads, flat JSON, fast
36
+ transport: undefined,
37
+ timestamp: pino_1.default.stdTimeFunctions.isoTime,
38
+ formatters: {
39
+ level(label) {
40
+ return { level: label };
41
+ },
42
+ },
43
+ });
44
+ }
45
+ /**
46
+ * Set request context for all subsequent logs
47
+ */
48
+ setContext(context) {
49
+ this.context = context;
50
+ // Rebind pino child with request-scoped fields
51
+ this.pinoLogger = this.pinoLogger.child({
52
+ requestId: context.requestId,
53
+ ...(context.user?.id && { userId: context.user.id }),
54
+ ...(context.tenant?.id && { tenantId: context.tenant.id }),
55
+ });
56
+ }
57
+ /**
58
+ * Clear request context
59
+ */
60
+ clearContext() {
61
+ this.context = null;
62
+ this.pinoLogger = (0, pino_1.default)({
63
+ level: this.config.minLevel,
64
+ timestamp: pino_1.default.stdTimeFunctions.isoTime,
65
+ formatters: {
66
+ level(label) {
67
+ return { level: label };
68
+ },
69
+ },
70
+ });
71
+ }
72
+ /**
73
+ * Create a child logger with additional default fields
74
+ */
75
+ child(fields) {
76
+ const childLogger = new Logger(this.config);
77
+ childLogger.pinoLogger = this.pinoLogger.child(fields);
78
+ childLogger.context = this.context;
79
+ return childLogger;
80
+ }
81
+ debug(message, data) {
82
+ data ? this.pinoLogger.debug(data, message) : this.pinoLogger.debug(message);
83
+ }
84
+ info(message, data) {
85
+ data ? this.pinoLogger.info(data, message) : this.pinoLogger.info(message);
86
+ }
87
+ warn(message, data) {
88
+ data ? this.pinoLogger.warn(data, message) : this.pinoLogger.warn(message);
89
+ }
90
+ error(message, data) {
91
+ data ? this.pinoLogger.error(data, message) : this.pinoLogger.error(message);
92
+ }
93
+ }
94
+ exports.Logger = Logger;
95
+ /**
96
+ * Default logger instance
97
+ */
98
+ exports.logger = new Logger();
99
+ /**
100
+ * Create a new logger with custom config
101
+ */
102
+ function createLogger(config) {
103
+ return new Logger(config);
104
+ }
105
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logger.js","sourceRoot":"","sources":["../src/logger.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * VentureKit Logger\n * \n * Structured logging for Lambda functions, powered by pino.\n * Outputs JSON for CloudWatch Logs Insights queries.\n * \n * Uses pino under the hood for:\n * - Fast structured JSON serialization\n * - Automatic level filtering\n * - Child logger support with bound fields\n * - Lambda-optimized (no worker threads, synchronous output)\n */\n\nimport pino from 'pino';\nimport { RequestContext } from './context';\n\n/**\n * Log levels\n */\nexport type LogLevel = 'debug' | 'info' | 'warn' | 'error';\n\n/**\n * Structured log entry (for type compatibility)\n */\nexport interface LogEntry {\n  level: LogLevel;\n  message: string;\n  timestamp: string;\n  requestId?: string;\n  userId?: string;\n  tenantId?: string;\n  duration?: number;\n  [key: string]: unknown;\n}\n\n/**\n * Logger configuration\n */\nexport interface LoggerConfig {\n  /** Minimum log level to output */\n  minLevel: LogLevel;\n  /** Include request context in all logs */\n  includeContext: boolean;\n}\n\n/**\n * Logger instance — thin wrapper over pino\n */\nexport class Logger {\n  private pinoLogger: pino.Logger;\n  private context: RequestContext | null = null;\n  private config: LoggerConfig;\n\n  constructor(config: Partial<LoggerConfig> = {}) {\n    this.config = {\n      minLevel: config.minLevel ?? 'info',\n      includeContext: config.includeContext ?? true,\n    };\n\n    this.pinoLogger = pino({\n      level: this.config.minLevel,\n      // Lambda-friendly: no worker threads, flat JSON, fast\n      transport: undefined,\n      timestamp: pino.stdTimeFunctions.isoTime,\n      formatters: {\n        level(label: string) {\n          return { level: label };\n        },\n      },\n    });\n  }\n\n  /**\n   * Set request context for all subsequent logs\n   */\n  setContext(context: RequestContext): void {\n    this.context = context;\n    // Rebind pino child with request-scoped fields\n    this.pinoLogger = this.pinoLogger.child({\n      requestId: context.requestId,\n      ...(context.user?.id && { userId: context.user.id }),\n      ...(context.tenant?.id && { tenantId: context.tenant.id }),\n    });\n  }\n\n  /**\n   * Clear request context\n   */\n  clearContext(): void {\n    this.context = null;\n    this.pinoLogger = pino({\n      level: this.config.minLevel,\n      timestamp: pino.stdTimeFunctions.isoTime,\n      formatters: {\n        level(label: string) {\n          return { level: label };\n        },\n      },\n    });\n  }\n\n  /**\n   * Create a child logger with additional default fields\n   */\n  child(fields: Record<string, unknown>): Logger {\n    const childLogger = new Logger(this.config);\n    childLogger.pinoLogger = this.pinoLogger.child(fields);\n    childLogger.context = this.context;\n    return childLogger;\n  }\n\n  debug(message: string, data?: Record<string, unknown>): void {\n    data ? this.pinoLogger.debug(data, message) : this.pinoLogger.debug(message);\n  }\n\n  info(message: string, data?: Record<string, unknown>): void {\n    data ? this.pinoLogger.info(data, message) : this.pinoLogger.info(message);\n  }\n\n  warn(message: string, data?: Record<string, unknown>): void {\n    data ? this.pinoLogger.warn(data, message) : this.pinoLogger.warn(message);\n  }\n\n  error(message: string, data?: Record<string, unknown>): void {\n    data ? this.pinoLogger.error(data, message) : this.pinoLogger.error(message);\n  }\n}\n\n/**\n * Default logger instance\n */\nexport const logger = new Logger();\n\n/**\n * Create a new logger with custom config\n */\nexport function createLogger(config?: Partial<LoggerConfig>): Logger {\n  return new Logger(config);\n}\n"]}
@@ -0,0 +1,46 @@
1
+ /**
2
+ * VentureKit Middleware
3
+ *
4
+ * Composable middleware for request processing.
5
+ * Extensible for auth, tenancy, rate limiting, etc.
6
+ */
7
+ import type { APIGatewayProxyResultV2 } from 'aws-lambda';
8
+ import { RequestContext } from './context';
9
+ import { Logger } from './logger';
10
+ /**
11
+ * Middleware function signature
12
+ */
13
+ export type MiddlewareFn = (ctx: RequestContext, next: () => Promise<APIGatewayProxyResultV2>) => Promise<APIGatewayProxyResultV2>;
14
+ /**
15
+ * Middleware with configuration
16
+ */
17
+ export interface Middleware {
18
+ name: string;
19
+ fn: MiddlewareFn;
20
+ }
21
+ /**
22
+ * Compose multiple middleware into a single function
23
+ */
24
+ export declare function compose(middlewares: Middleware[]): MiddlewareFn;
25
+ /**
26
+ * Logging middleware - logs request/response
27
+ */
28
+ export declare function loggingMiddleware(logger: Logger): Middleware;
29
+ /**
30
+ * CORS middleware - adds CORS headers
31
+ */
32
+ export declare function corsMiddleware(options: {
33
+ allowOrigins: string[];
34
+ allowMethods: string[];
35
+ allowHeaders: string[];
36
+ allowCredentials: boolean;
37
+ maxAge: number;
38
+ }): Middleware;
39
+ /**
40
+ * Timeout middleware - enforces request timeout
41
+ */
42
+ export declare function timeoutMiddleware(timeoutMs: number): Middleware;
43
+ /**
44
+ * Error boundary middleware - catches and formats errors
45
+ */
46
+ export declare function errorBoundaryMiddleware(errorHandler: (error: unknown, ctx: RequestContext) => APIGatewayProxyResultV2): Middleware;