@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.
- package/LICENSE +191 -0
- package/dist/context.d.ts +86 -0
- package/dist/context.js +76 -0
- package/dist/errors.d.ts +80 -0
- package/dist/errors.js +134 -0
- package/dist/handler.d.ts +71 -0
- package/dist/handler.js +176 -0
- package/dist/index.d.ts +18 -0
- package/dist/index.js +53 -0
- package/dist/logger.d.ts +72 -0
- package/dist/logger.js +105 -0
- package/dist/middleware.d.ts +46 -0
- package/dist/middleware.js +147 -0
- package/dist/response.d.ts +75 -0
- package/dist/response.js +107 -0
- package/dist/ws.d.ts +138 -0
- package/dist/ws.js +277 -0
- package/package.json +61 -0
package/dist/handler.js
ADDED
|
@@ -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"]}
|
package/dist/index.d.ts
ADDED
|
@@ -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==
|
package/dist/logger.d.ts
ADDED
|
@@ -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;
|