@pipeline-builder/api-core 3.1.0

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.
Files changed (122) hide show
  1. package/LICENSE +202 -0
  2. package/README.md +51 -0
  3. package/lib/constants/ai-providers.d.ts +41 -0
  4. package/lib/constants/ai-providers.js +88 -0
  5. package/lib/constants/http-status.d.ts +24 -0
  6. package/lib/constants/http-status.js +29 -0
  7. package/lib/constants/index.d.ts +3 -0
  8. package/lib/constants/index.js +22 -0
  9. package/lib/constants/time.d.ts +10 -0
  10. package/lib/constants/time.js +16 -0
  11. package/lib/errors/app-errors.d.ts +30 -0
  12. package/lib/errors/app-errors.js +62 -0
  13. package/lib/errors/index.d.ts +1 -0
  14. package/lib/errors/index.js +20 -0
  15. package/lib/helpers/access-helpers.d.ts +40 -0
  16. package/lib/helpers/access-helpers.js +56 -0
  17. package/lib/helpers/crud-helpers.d.ts +16 -0
  18. package/lib/helpers/crud-helpers.js +34 -0
  19. package/lib/helpers/index.d.ts +4 -0
  20. package/lib/helpers/index.js +23 -0
  21. package/lib/helpers/mask-helpers.d.ts +33 -0
  22. package/lib/helpers/mask-helpers.js +54 -0
  23. package/lib/helpers/sse-helpers.d.ts +13 -0
  24. package/lib/helpers/sse-helpers.js +40 -0
  25. package/lib/index.d.ts +57 -0
  26. package/lib/index.js +86 -0
  27. package/lib/middleware/auth.d.ts +50 -0
  28. package/lib/middleware/auth.js +171 -0
  29. package/lib/middleware/index.d.ts +1 -0
  30. package/lib/middleware/index.js +20 -0
  31. package/lib/openapi/extend-zod.d.ts +1 -0
  32. package/lib/openapi/extend-zod.js +8 -0
  33. package/lib/openapi/index.d.ts +2 -0
  34. package/lib/openapi/index.js +10 -0
  35. package/lib/openapi/registry.d.ts +17 -0
  36. package/lib/openapi/registry.js +42 -0
  37. package/lib/openapi/routes/billing-routes.d.ts +1 -0
  38. package/lib/openapi/routes/billing-routes.js +69 -0
  39. package/lib/openapi/routes/index.d.ts +5 -0
  40. package/lib/openapi/routes/index.js +22 -0
  41. package/lib/openapi/routes/message-routes.d.ts +1 -0
  42. package/lib/openapi/routes/message-routes.js +108 -0
  43. package/lib/openapi/routes/pipeline-routes.d.ts +1 -0
  44. package/lib/openapi/routes/pipeline-routes.js +90 -0
  45. package/lib/openapi/routes/plugin-routes.d.ts +1 -0
  46. package/lib/openapi/routes/plugin-routes.js +99 -0
  47. package/lib/openapi/routes/quota-routes.d.ts +1 -0
  48. package/lib/openapi/routes/quota-routes.js +65 -0
  49. package/lib/openapi/schema-registry.d.ts +25 -0
  50. package/lib/openapi/schema-registry.js +95 -0
  51. package/lib/routes/health.d.ts +47 -0
  52. package/lib/routes/health.js +81 -0
  53. package/lib/routes/index.d.ts +1 -0
  54. package/lib/routes/index.js +20 -0
  55. package/lib/services/admin-audit.d.ts +13 -0
  56. package/lib/services/admin-audit.js +31 -0
  57. package/lib/services/cache-service.d.ts +108 -0
  58. package/lib/services/cache-service.js +212 -0
  59. package/lib/services/compliance-client.d.ts +46 -0
  60. package/lib/services/compliance-client.js +102 -0
  61. package/lib/services/compliance-event-subscriber.d.ts +11 -0
  62. package/lib/services/compliance-event-subscriber.js +60 -0
  63. package/lib/services/compliance-queue.d.ts +11 -0
  64. package/lib/services/compliance-queue.js +38 -0
  65. package/lib/services/entity-events.d.ts +44 -0
  66. package/lib/services/entity-events.js +63 -0
  67. package/lib/services/http-client.d.ts +108 -0
  68. package/lib/services/http-client.js +285 -0
  69. package/lib/services/index.d.ts +10 -0
  70. package/lib/services/index.js +40 -0
  71. package/lib/services/quota.d.ts +59 -0
  72. package/lib/services/quota.js +137 -0
  73. package/lib/services/retry-strategy.d.ts +74 -0
  74. package/lib/services/retry-strategy.js +127 -0
  75. package/lib/types/billing.d.ts +47 -0
  76. package/lib/types/billing.js +5 -0
  77. package/lib/types/common.d.ts +161 -0
  78. package/lib/types/common.js +53 -0
  79. package/lib/types/error-codes.d.ts +38 -0
  80. package/lib/types/error-codes.js +77 -0
  81. package/lib/types/feature-flags.d.ts +38 -0
  82. package/lib/types/feature-flags.js +107 -0
  83. package/lib/types/http.d.ts +37 -0
  84. package/lib/types/http.js +5 -0
  85. package/lib/types/index.d.ts +7 -0
  86. package/lib/types/index.js +26 -0
  87. package/lib/types/pipeline.d.ts +70 -0
  88. package/lib/types/pipeline.js +44 -0
  89. package/lib/types/quota-tiers.d.ts +23 -0
  90. package/lib/types/quota-tiers.js +26 -0
  91. package/lib/utils/alias-resolver.d.ts +16 -0
  92. package/lib/utils/alias-resolver.js +49 -0
  93. package/lib/utils/headers.d.ts +18 -0
  94. package/lib/utils/headers.js +24 -0
  95. package/lib/utils/identity.d.ts +61 -0
  96. package/lib/utils/identity.js +75 -0
  97. package/lib/utils/index.d.ts +7 -0
  98. package/lib/utils/index.js +26 -0
  99. package/lib/utils/logger.d.ts +28 -0
  100. package/lib/utils/logger.js +77 -0
  101. package/lib/utils/object.d.ts +13 -0
  102. package/lib/utils/object.js +21 -0
  103. package/lib/utils/params.d.ts +89 -0
  104. package/lib/utils/params.js +148 -0
  105. package/lib/utils/response.d.ts +142 -0
  106. package/lib/utils/response.js +237 -0
  107. package/lib/validation/ai-schemas.d.ts +61 -0
  108. package/lib/validation/ai-schemas.js +81 -0
  109. package/lib/validation/common-schemas.d.ts +72 -0
  110. package/lib/validation/common-schemas.js +58 -0
  111. package/lib/validation/index.d.ts +6 -0
  112. package/lib/validation/index.js +25 -0
  113. package/lib/validation/message-schemas.d.ts +79 -0
  114. package/lib/validation/message-schemas.js +42 -0
  115. package/lib/validation/middleware.d.ts +60 -0
  116. package/lib/validation/middleware.js +77 -0
  117. package/lib/validation/pipeline-schemas.d.ts +135 -0
  118. package/lib/validation/pipeline-schemas.js +85 -0
  119. package/lib/validation/plugin-schemas.d.ts +127 -0
  120. package/lib/validation/plugin-schemas.js +84 -0
  121. package/openapi.yaml +292 -0
  122. package/package.json +127 -0
@@ -0,0 +1,148 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.getParam = getParam;
6
+ exports.getOrgId = getOrgId;
7
+ exports.getAuthHeader = getAuthHeader;
8
+ exports.parseQueryBoolean = parseQueryBoolean;
9
+ exports.parseQueryInt = parseQueryInt;
10
+ exports.parseQueryString = parseQueryString;
11
+ exports.parsePositiveInt = parsePositiveInt;
12
+ const headers_1 = require("./headers");
13
+ /**
14
+ * Extract a single string parameter from Express 5 route params.
15
+ *
16
+ * @param params - Request params object
17
+ * @param key - Parameter key
18
+ * @returns The parameter value as a string, or undefined if not present
19
+ *
20
+ * @example
21
+ * ```typescript
22
+ * // Route: GET /plugins/:id
23
+ * const id = getParam(req.params, 'id');
24
+ * if (!id) {
25
+ * return sendError(res, 400, 'Missing id parameter');
26
+ * }
27
+ * ```
28
+ */
29
+ function getParam(params, key) {
30
+ const value = params[key];
31
+ if (value === undefined) {
32
+ return undefined;
33
+ }
34
+ return Array.isArray(value) ? value[0] : value;
35
+ }
36
+ /**
37
+ * Extract the organization ID from request.
38
+ * Checks params, headers, and user object.
39
+ *
40
+ * @param req - Express request object
41
+ * @returns Organization ID or undefined
42
+ *
43
+ * @example
44
+ * ```typescript
45
+ * const orgId = getOrgId(req);
46
+ * if (!orgId) {
47
+ * return sendError(res, 400, 'Organization ID required');
48
+ * }
49
+ * ```
50
+ */
51
+ function getOrgId(req) {
52
+ // Check route params first
53
+ const paramOrgId = getParam(req.params, 'orgId');
54
+ if (paramOrgId) {
55
+ return paramOrgId;
56
+ }
57
+ // Check x-org-id header
58
+ const headerOrgId = (0, headers_1.getHeaderString)(req.headers['x-org-id'])?.trim();
59
+ if (headerOrgId) {
60
+ return headerOrgId;
61
+ }
62
+ // Check authenticated user
63
+ const userOrgId = req.user?.organizationId?.trim();
64
+ if (userOrgId) {
65
+ return userOrgId;
66
+ }
67
+ return undefined;
68
+ }
69
+ /**
70
+ * Get the authorization header from request.
71
+ *
72
+ * @param req - Express request object
73
+ * @returns Authorization header value or empty string
74
+ */
75
+ function getAuthHeader(req) {
76
+ return (0, headers_1.getHeaderString)(req.headers.authorization) ?? '';
77
+ }
78
+ /**
79
+ * Parse a query parameter as a boolean.
80
+ *
81
+ * @param value - Query parameter value
82
+ * @returns Parsed boolean or undefined
83
+ *
84
+ * @example
85
+ * ```typescript
86
+ * const isActive = parseQueryBoolean(req.query.isActive);
87
+ * ```
88
+ */
89
+ function parseQueryBoolean(value) {
90
+ if (value === undefined || value === null || value === '')
91
+ return undefined;
92
+ if (typeof value === 'boolean')
93
+ return value;
94
+ if (typeof value === 'string') {
95
+ const lower = value.toLowerCase();
96
+ if (lower === 'true' || lower === '1')
97
+ return true;
98
+ if (lower === 'false' || lower === '0')
99
+ return false;
100
+ }
101
+ return undefined;
102
+ }
103
+ /**
104
+ * Parse a query parameter as an integer.
105
+ *
106
+ * @param value - Query parameter value
107
+ * @param defaultValue - Default value if parsing fails
108
+ * @returns Parsed integer
109
+ *
110
+ * @example
111
+ * ```typescript
112
+ * const limit = parseQueryInt(req.query.limit, 10);
113
+ * const offset = parseQueryInt(req.query.offset, 0);
114
+ * ```
115
+ */
116
+ function parseQueryInt(value, defaultValue) {
117
+ if (value === undefined || value === null || value === '')
118
+ return defaultValue;
119
+ const parsed = parseInt(String(value), 10);
120
+ return isNaN(parsed) ? defaultValue : parsed;
121
+ }
122
+ /**
123
+ * Parse a query parameter as a string.
124
+ *
125
+ * @param value - Query parameter value
126
+ * @returns String value or undefined
127
+ *
128
+ * @example
129
+ * ```typescript
130
+ * const search = parseQueryString(req.query.search);
131
+ * ```
132
+ */
133
+ function parseQueryString(value) {
134
+ if (value === undefined || value === null || value === '')
135
+ return undefined;
136
+ return String(value);
137
+ }
138
+ /**
139
+ * Parse a string to a positive integer, returning a fallback if invalid.
140
+ * Useful for parsing environment variables with numeric defaults.
141
+ */
142
+ function parsePositiveInt(value, fallback) {
143
+ if (!value)
144
+ return fallback;
145
+ const parsed = parseInt(value, 10);
146
+ return Number.isFinite(parsed) && parsed > 0 ? parsed : fallback;
147
+ }
148
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"params.js","sourceRoot":"","sources":["../../src/utils/params.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { Request } from 'express';\nimport { getHeaderString } from './headers';\n\n/**\n * Express 5 parameter type.\n */\ntype ParamValue = string | string[] | undefined;\n\n/**\n * Extract a single string parameter from Express 5 route params.\n *\n * @param params - Request params object\n * @param key - Parameter key\n * @returns The parameter value as a string, or undefined if not present\n *\n * @example\n * ```typescript\n * // Route: GET /plugins/:id\n * const id = getParam(req.params, 'id');\n * if (!id) {\n *   return sendError(res, 400, 'Missing id parameter');\n * }\n * ```\n */\nexport function getParam(\n  params: Record<string, ParamValue>,\n  key: string,\n): string | undefined {\n  const value = params[key];\n  if (value === undefined) {\n    return undefined;\n  }\n  return Array.isArray(value) ? value[0] : value;\n}\n\n/**\n * Extract the organization ID from request.\n * Checks params, headers, and user object.\n *\n * @param req - Express request object\n * @returns Organization ID or undefined\n *\n * @example\n * ```typescript\n * const orgId = getOrgId(req);\n * if (!orgId) {\n *   return sendError(res, 400, 'Organization ID required');\n * }\n * ```\n */\nexport function getOrgId(req: Request): string | undefined {\n  // Check route params first\n  const paramOrgId = getParam(req.params, 'orgId');\n  if (paramOrgId) {\n    return paramOrgId;\n  }\n\n  // Check x-org-id header\n  const headerOrgId = getHeaderString(req.headers['x-org-id'])?.trim();\n  if (headerOrgId) {\n    return headerOrgId;\n  }\n\n  // Check authenticated user\n  const userOrgId = req.user?.organizationId?.trim();\n  if (userOrgId) {\n    return userOrgId;\n  }\n\n  return undefined;\n}\n\n/**\n * Get the authorization header from request.\n *\n * @param req - Express request object\n * @returns Authorization header value or empty string\n */\nexport function getAuthHeader(req: Request): string {\n  return getHeaderString(req.headers.authorization) ?? '';\n}\n\n/**\n * Parse a query parameter as a boolean.\n *\n * @param value - Query parameter value\n * @returns Parsed boolean or undefined\n *\n * @example\n * ```typescript\n * const isActive = parseQueryBoolean(req.query.isActive);\n * ```\n */\nexport function parseQueryBoolean(value: unknown): boolean | undefined {\n  if (value === undefined || value === null || value === '') return undefined;\n  if (typeof value === 'boolean') return value;\n  if (typeof value === 'string') {\n    const lower = value.toLowerCase();\n    if (lower === 'true' || lower === '1') return true;\n    if (lower === 'false' || lower === '0') return false;\n  }\n  return undefined;\n}\n\n/**\n * Parse a query parameter as an integer.\n *\n * @param value - Query parameter value\n * @param defaultValue - Default value if parsing fails\n * @returns Parsed integer\n *\n * @example\n * ```typescript\n * const limit = parseQueryInt(req.query.limit, 10);\n * const offset = parseQueryInt(req.query.offset, 0);\n * ```\n */\nexport function parseQueryInt(value: unknown, defaultValue: number): number {\n  if (value === undefined || value === null || value === '') return defaultValue;\n  const parsed = parseInt(String(value), 10);\n  return isNaN(parsed) ? defaultValue : parsed;\n}\n\n/**\n * Parse a query parameter as a string.\n *\n * @param value - Query parameter value\n * @returns String value or undefined\n *\n * @example\n * ```typescript\n * const search = parseQueryString(req.query.search);\n * ```\n */\nexport function parseQueryString(value: unknown): string | undefined {\n  if (value === undefined || value === null || value === '') return undefined;\n  return String(value);\n}\n\n/**\n * Parse a string to a positive integer, returning a fallback if invalid.\n * Useful for parsing environment variables with numeric defaults.\n */\nexport function parsePositiveInt(value: string | undefined, fallback: number): number {\n  if (!value) return fallback;\n  const parsed = parseInt(value, 10);\n  return Number.isFinite(parsed) && parsed > 0 ? parsed : fallback;\n}\n"]}
@@ -0,0 +1,142 @@
1
+ import { Response } from 'express';
2
+ import { QuotaInfo } from '../types/common';
3
+ import { ErrorCode } from '../types/error-codes';
4
+ /**
5
+ * Send a standardized success response.
6
+ *
7
+ * @param res - Express response object
8
+ * @param statusCode - HTTP status code (default: 200)
9
+ * @param data - Response data
10
+ * @param message - Optional success message
11
+ *
12
+ * @example
13
+ * ```typescript
14
+ * sendSuccess(res, 200, { plugin: pluginData });
15
+ * sendSuccess(res, 201, { id: newId }, 'Resource created');
16
+ * ```
17
+ */
18
+ export declare function sendSuccess<T>(res: Response, statusCode?: number, data?: T, message?: string): void;
19
+ /**
20
+ * Send a standardized error response.
21
+ *
22
+ * @param res - Express response object
23
+ * @param statusCode - HTTP status code
24
+ * @param message - Error message
25
+ * @param code - Error code from ErrorCode enum
26
+ * @param details - Optional additional details
27
+ *
28
+ * @example
29
+ * ```typescript
30
+ * sendError(res, 404, 'Plugin not found', ErrorCode.NOT_FOUND);
31
+ * sendError(res, 400, 'Invalid input', ErrorCode.VALIDATION_ERROR, { field: 'name' });
32
+ * ```
33
+ */
34
+ export declare function sendError(res: Response, statusCode: number, message: string, code?: ErrorCode | string, details?: unknown): void;
35
+ /**
36
+ * Send a quota exceeded error response.
37
+ *
38
+ * @param res - Express response object
39
+ * @param quotaType - Type of quota exceeded
40
+ * @param quota - Current quota information
41
+ * @param resetAt - ISO timestamp when quota resets
42
+ *
43
+ * @example
44
+ * ```typescript
45
+ * sendQuotaExceeded(res, 'apiCalls', { type: 'apiCalls', limit: 10000, used: 10000, remaining: 0 }, resetAt);
46
+ * ```
47
+ */
48
+ export declare function sendQuotaExceeded(res: Response, quotaType: string, quota: QuotaInfo, resetAt?: string): void;
49
+ /**
50
+ * Paginated response interface.
51
+ */
52
+ export interface PaginatedResponse<T> {
53
+ success: true;
54
+ statusCode: number;
55
+ message?: string;
56
+ data: T[];
57
+ count: number;
58
+ limit: number;
59
+ offset: number;
60
+ total?: number;
61
+ }
62
+ /**
63
+ * Send a paginated list response.
64
+ *
65
+ * @param res - Express response object
66
+ * @param data - Array of items to return
67
+ * @param options - Pagination options
68
+ *
69
+ * @example
70
+ * ```typescript
71
+ * const plugins = await db.query.plugins.findMany({ limit, offset });
72
+ * sendPaginated(res, plugins, {
73
+ * limit: 10,
74
+ * offset: 0,
75
+ * total: 50,
76
+ * message: 'Plugins retrieved successfully',
77
+ * });
78
+ * ```
79
+ */
80
+ export declare function sendPaginated<T>(res: Response, data: T[], options: {
81
+ limit: number;
82
+ offset: number;
83
+ total?: number;
84
+ message?: string;
85
+ statusCode?: number;
86
+ }): void;
87
+ /**
88
+ * Send a paginated list response with the items nested under a named key.
89
+ * Use when the API contract is `{ <key>: [...], pagination: {...} }`
90
+ * (the dominant shape across api/* services).
91
+ *
92
+ * @example
93
+ * ```typescript
94
+ * sendPaginatedNested(res, 'pipelines', items, {
95
+ * total: 50, limit: 25, offset: 0, hasMore: true,
96
+ * });
97
+ * // → { pipelines: [...], pagination: { total: 50, limit: 25, offset: 0, hasMore: true } }
98
+ * ```
99
+ */
100
+ export declare function sendPaginatedNested<T>(res: Response, dataKey: string, data: T[], options: {
101
+ total?: number;
102
+ limit: number;
103
+ offset: number;
104
+ hasMore: boolean;
105
+ nextCursor?: string;
106
+ statusCode?: number;
107
+ }): void;
108
+ /**
109
+ * Extract database error details for logging/response.
110
+ *
111
+ * Extracts PostgreSQL error codes, details, hints, and constraint names
112
+ * from database errors for better error messages.
113
+ *
114
+ * @param error - Error object from database operation
115
+ * @returns Object with extracted error details
116
+ *
117
+ * @example
118
+ * ```typescript
119
+ * try {
120
+ * await db.insert(plugins).values(newPlugin);
121
+ * } catch (error) {
122
+ * const dbDetails = extractDbError(error);
123
+ * return sendError(res, 500, 'Database error', ErrorCode.DATABASE_ERROR, dbDetails);
124
+ * }
125
+ * ```
126
+ */
127
+ export declare function extractDbError(error: unknown): Record<string, unknown>;
128
+ /** Extract a message string from an unknown catch value. */
129
+ export declare function errorMessage(error: unknown): string;
130
+ /** Send a 400 bad-request response. */
131
+ export declare function sendBadRequest(res: Response, message: string, code?: ErrorCode | string): void;
132
+ /** Send a 500 internal error response. */
133
+ export declare function sendInternalError(res: Response, message: string, details?: Record<string, unknown>): void;
134
+ /** Common pagination + sorting parameters. */
135
+ export interface PaginationParams {
136
+ limit: number;
137
+ offset: number;
138
+ sortBy: string;
139
+ sortOrder: 'asc' | 'desc';
140
+ }
141
+ /** Parse pagination/sort params from query string, clamping to safe defaults. */
142
+ export declare function parsePaginationParams(query: Record<string, unknown>): PaginationParams;
@@ -0,0 +1,237 @@
1
+ "use strict";
2
+ // Copyright 2026 Pipeline Builder Contributors
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.sendSuccess = sendSuccess;
6
+ exports.sendError = sendError;
7
+ exports.sendQuotaExceeded = sendQuotaExceeded;
8
+ exports.sendPaginated = sendPaginated;
9
+ exports.sendPaginatedNested = sendPaginatedNested;
10
+ exports.extractDbError = extractDbError;
11
+ exports.errorMessage = errorMessage;
12
+ exports.sendBadRequest = sendBadRequest;
13
+ exports.sendInternalError = sendInternalError;
14
+ exports.parsePaginationParams = parsePaginationParams;
15
+ const logger_1 = require("./logger");
16
+ const error_codes_1 = require("../types/error-codes");
17
+ const common_schemas_1 = require("../validation/common-schemas");
18
+ const logger = (0, logger_1.createLogger)('response');
19
+ /**
20
+ * Send a standardized success response.
21
+ *
22
+ * @param res - Express response object
23
+ * @param statusCode - HTTP status code (default: 200)
24
+ * @param data - Response data
25
+ * @param message - Optional success message
26
+ *
27
+ * @example
28
+ * ```typescript
29
+ * sendSuccess(res, 200, { plugin: pluginData });
30
+ * sendSuccess(res, 201, { id: newId }, 'Resource created');
31
+ * ```
32
+ */
33
+ function sendSuccess(res, statusCode = 200, data, message) {
34
+ if (res.headersSent) {
35
+ logger.warn('Attempted to send success after headers already sent', { statusCode });
36
+ return;
37
+ }
38
+ const response = {
39
+ success: true,
40
+ statusCode,
41
+ };
42
+ if (data !== undefined) {
43
+ response.data = data;
44
+ }
45
+ if (message) {
46
+ response.message = message;
47
+ }
48
+ res.status(statusCode).json(response);
49
+ }
50
+ /**
51
+ * Send a standardized error response.
52
+ *
53
+ * @param res - Express response object
54
+ * @param statusCode - HTTP status code
55
+ * @param message - Error message
56
+ * @param code - Error code from ErrorCode enum
57
+ * @param details - Optional additional details
58
+ *
59
+ * @example
60
+ * ```typescript
61
+ * sendError(res, 404, 'Plugin not found', ErrorCode.NOT_FOUND);
62
+ * sendError(res, 400, 'Invalid input', ErrorCode.VALIDATION_ERROR, { field: 'name' });
63
+ * ```
64
+ */
65
+ function sendError(res, statusCode, message, code, details) {
66
+ if (res.headersSent) {
67
+ logger.warn('Attempted to send error after headers already sent', { statusCode, message });
68
+ return;
69
+ }
70
+ const response = {
71
+ success: false,
72
+ statusCode,
73
+ message,
74
+ };
75
+ if (code) {
76
+ response.code = code;
77
+ }
78
+ if (details !== undefined) {
79
+ response.details = details;
80
+ }
81
+ res.status(statusCode).json(response);
82
+ }
83
+ /**
84
+ * Send a quota exceeded error response.
85
+ *
86
+ * @param res - Express response object
87
+ * @param quotaType - Type of quota exceeded
88
+ * @param quota - Current quota information
89
+ * @param resetAt - ISO timestamp when quota resets
90
+ *
91
+ * @example
92
+ * ```typescript
93
+ * sendQuotaExceeded(res, 'apiCalls', { type: 'apiCalls', limit: 10000, used: 10000, remaining: 0 }, resetAt);
94
+ * ```
95
+ */
96
+ function sendQuotaExceeded(res, quotaType, quota, resetAt) {
97
+ const resetDate = resetAt ? new Date(resetAt) : new Date();
98
+ const resetIn = Math.max(0, Math.ceil((resetDate.getTime() - Date.now()) / 1000));
99
+ res.setHeader('Retry-After', resetIn);
100
+ res.setHeader('X-Quota-Limit', quota.limit);
101
+ res.setHeader('X-Quota-Used', quota.used);
102
+ res.setHeader('X-Quota-Remaining', Math.max(0, quota.remaining));
103
+ if (resetAt) {
104
+ res.setHeader('X-Quota-Reset', resetAt);
105
+ }
106
+ sendError(res, 429, `${quotaType} quota exceeded (${quota.used}/${quota.limit}). Please try again later.`, error_codes_1.ErrorCode.QUOTA_EXCEEDED, { quota });
107
+ }
108
+ /**
109
+ * Send a paginated list response.
110
+ *
111
+ * @param res - Express response object
112
+ * @param data - Array of items to return
113
+ * @param options - Pagination options
114
+ *
115
+ * @example
116
+ * ```typescript
117
+ * const plugins = await db.query.plugins.findMany({ limit, offset });
118
+ * sendPaginated(res, plugins, {
119
+ * limit: 10,
120
+ * offset: 0,
121
+ * total: 50,
122
+ * message: 'Plugins retrieved successfully',
123
+ * });
124
+ * ```
125
+ */
126
+ function sendPaginated(res, data, options) {
127
+ if (res.headersSent) {
128
+ logger.warn('Attempted to send paginated response after headers already sent');
129
+ return;
130
+ }
131
+ const { limit, offset, total, message, statusCode = 200 } = options;
132
+ const response = {
133
+ success: true,
134
+ statusCode,
135
+ data,
136
+ count: data.length,
137
+ limit,
138
+ offset,
139
+ };
140
+ if (message) {
141
+ response.message = message;
142
+ }
143
+ if (total !== undefined) {
144
+ response.total = total;
145
+ }
146
+ res.status(statusCode).json(response);
147
+ }
148
+ /**
149
+ * Send a paginated list response with the items nested under a named key.
150
+ * Use when the API contract is `{ <key>: [...], pagination: {...} }`
151
+ * (the dominant shape across api/* services).
152
+ *
153
+ * @example
154
+ * ```typescript
155
+ * sendPaginatedNested(res, 'pipelines', items, {
156
+ * total: 50, limit: 25, offset: 0, hasMore: true,
157
+ * });
158
+ * // → { pipelines: [...], pagination: { total: 50, limit: 25, offset: 0, hasMore: true } }
159
+ * ```
160
+ */
161
+ function sendPaginatedNested(res, dataKey, data, options) {
162
+ if (res.headersSent) {
163
+ logger.warn('Attempted to send paginated response after headers already sent');
164
+ return;
165
+ }
166
+ const { statusCode = 200, total, limit, offset, hasMore, nextCursor } = options;
167
+ const pagination = { limit, offset, hasMore };
168
+ if (total !== undefined)
169
+ pagination.total = total;
170
+ if (nextCursor)
171
+ pagination.nextCursor = nextCursor;
172
+ res.status(statusCode).json({ [dataKey]: data, pagination });
173
+ }
174
+ /**
175
+ * Extract database error details for logging/response.
176
+ *
177
+ * Extracts PostgreSQL error codes, details, hints, and constraint names
178
+ * from database errors for better error messages.
179
+ *
180
+ * @param error - Error object from database operation
181
+ * @returns Object with extracted error details
182
+ *
183
+ * @example
184
+ * ```typescript
185
+ * try {
186
+ * await db.insert(plugins).values(newPlugin);
187
+ * } catch (error) {
188
+ * const dbDetails = extractDbError(error);
189
+ * return sendError(res, 500, 'Database error', ErrorCode.DATABASE_ERROR, dbDetails);
190
+ * }
191
+ * ```
192
+ */
193
+ function extractDbError(error) {
194
+ if (!error || typeof error !== 'object') {
195
+ return {};
196
+ }
197
+ // Drizzle ORM wraps the original pg error in error.cause —
198
+ // check the cause first, then fall back to the error itself.
199
+ const err = error;
200
+ const source = (err.cause && typeof err.cause === 'object' ? err.cause : err);
201
+ const details = {};
202
+ if (source.code)
203
+ details.dbCode = source.code;
204
+ if (source.detail)
205
+ details.dbDetail = source.detail;
206
+ if (source.hint)
207
+ details.dbHint = source.hint;
208
+ if (source.constraint)
209
+ details.constraint = source.constraint;
210
+ if (source.table)
211
+ details.table = source.table;
212
+ if (source.column)
213
+ details.column = source.column;
214
+ return details;
215
+ }
216
+ // Convenience helpers (shared across plugin, pipeline, quota services)
217
+ /** Extract a message string from an unknown catch value. */
218
+ function errorMessage(error) {
219
+ return error instanceof Error ? error.message : String(error);
220
+ }
221
+ /** Send a 400 bad-request response. */
222
+ function sendBadRequest(res, message, code = error_codes_1.ErrorCode.VALIDATION_ERROR) {
223
+ sendError(res, 400, message, code);
224
+ }
225
+ /** Send a 500 internal error response. */
226
+ function sendInternalError(res, message, details) {
227
+ sendError(res, 500, message, error_codes_1.ErrorCode.INTERNAL_ERROR, details);
228
+ }
229
+ /** Parse pagination/sort params from query string, clamping to safe defaults. */
230
+ function parsePaginationParams(query) {
231
+ const limit = Math.min(Math.max(parseInt(String(query.limit), 10) || 10, 1), common_schemas_1.MAX_PAGE_LIMIT);
232
+ const offset = Math.max(parseInt(String(query.offset), 10) || 0, 0);
233
+ const sortBy = String(query.sortBy || 'createdAt');
234
+ const sortOrder = String(query.sortOrder || 'desc').toLowerCase() === 'asc' ? 'asc' : 'desc';
235
+ return { limit, offset, sortBy, sortOrder };
236
+ }
237
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"response.js","sourceRoot":"","sources":["../../src/utils/response.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { Response } from 'express';\nimport { createLogger } from './logger';\nimport { QuotaInfo } from '../types/common';\nimport { ErrorCode } from '../types/error-codes';\nimport { MAX_PAGE_LIMIT } from '../validation/common-schemas';\n\nconst logger = createLogger('response');\n\n/**\n * Send a standardized success response.\n *\n * @param res - Express response object\n * @param statusCode - HTTP status code (default: 200)\n * @param data - Response data\n * @param message - Optional success message\n *\n * @example\n * ```typescript\n * sendSuccess(res, 200, { plugin: pluginData });\n * sendSuccess(res, 201, { id: newId }, 'Resource created');\n * ```\n */\nexport function sendSuccess<T>(\n  res: Response,\n  statusCode: number = 200,\n  data?: T,\n  message?: string,\n): void {\n  if (res.headersSent) {\n    logger.warn('Attempted to send success after headers already sent', { statusCode });\n    return;\n  }\n\n  const response: {\n    success: true;\n    statusCode: number;\n    data?: T;\n    message?: string;\n  } = {\n    success: true,\n    statusCode,\n  };\n\n  if (data !== undefined) {\n    response.data = data;\n  }\n\n  if (message) {\n    response.message = message;\n  }\n\n  res.status(statusCode).json(response);\n}\n\n/**\n * Send a standardized error response.\n *\n * @param res - Express response object\n * @param statusCode - HTTP status code\n * @param message - Error message\n * @param code - Error code from ErrorCode enum\n * @param details - Optional additional details\n *\n * @example\n * ```typescript\n * sendError(res, 404, 'Plugin not found', ErrorCode.NOT_FOUND);\n * sendError(res, 400, 'Invalid input', ErrorCode.VALIDATION_ERROR, { field: 'name' });\n * ```\n */\nexport function sendError(\n  res: Response,\n  statusCode: number,\n  message: string,\n  code?: ErrorCode | string,\n  details?: unknown,\n): void {\n  if (res.headersSent) {\n    logger.warn('Attempted to send error after headers already sent', { statusCode, message });\n    return;\n  }\n\n  const response: {\n    success: false;\n    statusCode: number;\n    message: string;\n    code?: string;\n    details?: unknown;\n  } = {\n    success: false,\n    statusCode,\n    message,\n  };\n\n  if (code) {\n    response.code = code;\n  }\n\n  if (details !== undefined) {\n    response.details = details;\n  }\n\n  res.status(statusCode).json(response);\n}\n\n/**\n * Send a quota exceeded error response.\n *\n * @param res - Express response object\n * @param quotaType - Type of quota exceeded\n * @param quota - Current quota information\n * @param resetAt - ISO timestamp when quota resets\n *\n * @example\n * ```typescript\n * sendQuotaExceeded(res, 'apiCalls', { type: 'apiCalls', limit: 10000, used: 10000, remaining: 0 }, resetAt);\n * ```\n */\nexport function sendQuotaExceeded(\n  res: Response,\n  quotaType: string,\n  quota: QuotaInfo,\n  resetAt?: string,\n): void {\n  const resetDate = resetAt ? new Date(resetAt) : new Date();\n  const resetIn = Math.max(0, Math.ceil((resetDate.getTime() - Date.now()) / 1000));\n\n  res.setHeader('Retry-After', resetIn);\n  res.setHeader('X-Quota-Limit', quota.limit);\n  res.setHeader('X-Quota-Used', quota.used);\n  res.setHeader('X-Quota-Remaining', Math.max(0, quota.remaining));\n\n  if (resetAt) {\n    res.setHeader('X-Quota-Reset', resetAt);\n  }\n\n  sendError(\n    res,\n    429,\n    `${quotaType} quota exceeded (${quota.used}/${quota.limit}). Please try again later.`,\n    ErrorCode.QUOTA_EXCEEDED,\n    { quota },\n  );\n}\n\n/**\n * Paginated response interface.\n */\nexport interface PaginatedResponse<T> {\n  success: true;\n  statusCode: number;\n  message?: string;\n  data: T[];\n  count: number;\n  limit: number;\n  offset: number;\n  total?: number;\n}\n\n/**\n * Send a paginated list response.\n *\n * @param res - Express response object\n * @param data - Array of items to return\n * @param options - Pagination options\n *\n * @example\n * ```typescript\n * const plugins = await db.query.plugins.findMany({ limit, offset });\n * sendPaginated(res, plugins, {\n *   limit: 10,\n *   offset: 0,\n *   total: 50,\n *   message: 'Plugins retrieved successfully',\n * });\n * ```\n */\nexport function sendPaginated<T>(\n  res: Response,\n  data: T[],\n  options: {\n    limit: number;\n    offset: number;\n    total?: number;\n    message?: string;\n    statusCode?: number;\n  },\n): void {\n  if (res.headersSent) {\n    logger.warn('Attempted to send paginated response after headers already sent');\n    return;\n  }\n\n  const { limit, offset, total, message, statusCode = 200 } = options;\n\n  const response: PaginatedResponse<T> = {\n    success: true,\n    statusCode,\n    data,\n    count: data.length,\n    limit,\n    offset,\n  };\n\n  if (message) {\n    response.message = message;\n  }\n\n  if (total !== undefined) {\n    response.total = total;\n  }\n\n  res.status(statusCode).json(response);\n}\n\n/**\n * Send a paginated list response with the items nested under a named key.\n * Use when the API contract is `{ <key>: [...], pagination: {...} }`\n * (the dominant shape across api/* services).\n *\n * @example\n * ```typescript\n * sendPaginatedNested(res, 'pipelines', items, {\n *   total: 50, limit: 25, offset: 0, hasMore: true,\n * });\n * // → { pipelines: [...], pagination: { total: 50, limit: 25, offset: 0, hasMore: true } }\n * ```\n */\nexport function sendPaginatedNested<T>(\n  res: Response,\n  dataKey: string,\n  data: T[],\n  options: {\n    total?: number;\n    limit: number;\n    offset: number;\n    hasMore: boolean;\n    nextCursor?: string;\n    statusCode?: number;\n  },\n): void {\n  if (res.headersSent) {\n    logger.warn('Attempted to send paginated response after headers already sent');\n    return;\n  }\n  const { statusCode = 200, total, limit, offset, hasMore, nextCursor } = options;\n  const pagination: Record<string, unknown> = { limit, offset, hasMore };\n  if (total !== undefined) pagination.total = total;\n  if (nextCursor) pagination.nextCursor = nextCursor;\n  res.status(statusCode).json({ [dataKey]: data, pagination });\n}\n\n/**\n * Extract database error details for logging/response.\n *\n * Extracts PostgreSQL error codes, details, hints, and constraint names\n * from database errors for better error messages.\n *\n * @param error - Error object from database operation\n * @returns Object with extracted error details\n *\n * @example\n * ```typescript\n * try {\n *   await db.insert(plugins).values(newPlugin);\n * } catch (error) {\n *   const dbDetails = extractDbError(error);\n *   return sendError(res, 500, 'Database error', ErrorCode.DATABASE_ERROR, dbDetails);\n * }\n * ```\n */\nexport function extractDbError(error: unknown): Record<string, unknown> {\n  if (!error || typeof error !== 'object') {\n    return {};\n  }\n\n  // Drizzle ORM wraps the original pg error in error.cause —\n  // check the cause first, then fall back to the error itself.\n  const err = error as Record<string, unknown>;\n  const source = (err.cause && typeof err.cause === 'object' ? err.cause : err) as Record<string, unknown>;\n  const details: Record<string, unknown> = {};\n\n  if (source.code) details.dbCode = source.code;\n  if (source.detail) details.dbDetail = source.detail;\n  if (source.hint) details.dbHint = source.hint;\n  if (source.constraint) details.constraint = source.constraint;\n  if (source.table) details.table = source.table;\n  if (source.column) details.column = source.column;\n\n  return details;\n}\n\n// Convenience helpers (shared across plugin, pipeline, quota services)\n\n/** Extract a message string from an unknown catch value. */\nexport function errorMessage(error: unknown): string {\n  return error instanceof Error ? error.message : String(error);\n}\n\n/** Send a 400 bad-request response. */\nexport function sendBadRequest(\n  res: Response,\n  message: string,\n  code: ErrorCode | string = ErrorCode.VALIDATION_ERROR,\n): void {\n  sendError(res, 400, message, code);\n}\n\n/** Send a 500 internal error response. */\nexport function sendInternalError(\n  res: Response,\n  message: string,\n  details?: Record<string, unknown>,\n): void {\n  sendError(res, 500, message, ErrorCode.INTERNAL_ERROR, details);\n}\n\n// Pagination\n\n/** Common pagination + sorting parameters. */\nexport interface PaginationParams {\n  limit: number;\n  offset: number;\n  sortBy: string;\n  sortOrder: 'asc' | 'desc';\n}\n\n/** Parse pagination/sort params from query string, clamping to safe defaults. */\nexport function parsePaginationParams(query: Record<string, unknown>): PaginationParams {\n  const limit = Math.min(Math.max(parseInt(String(query.limit), 10) || 10, 1), MAX_PAGE_LIMIT);\n  const offset = Math.max(parseInt(String(query.offset), 10) || 0, 0);\n  const sortBy = String(query.sortBy || 'createdAt');\n  const sortOrder: 'asc' | 'desc' =\n    String(query.sortOrder || 'desc').toLowerCase() === 'asc' ? 'asc' : 'desc';\n\n  return { limit, offset, sortBy, sortOrder };\n}\n"]}
@@ -0,0 +1,61 @@
1
+ import { z } from 'zod';
2
+ /**
3
+ * Schema for AI generation request body.
4
+ * Used by both POST /pipelines/generate and POST /plugins/generate.
5
+ */
6
+ export declare const AIGenerateBodySchema: z.ZodObject<{
7
+ prompt: z.ZodString;
8
+ provider: z.ZodString;
9
+ model: z.ZodString;
10
+ apiKey: z.ZodOptional<z.ZodString>;
11
+ previousConfig: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodUnknown>>;
12
+ fallbackProviders: z.ZodOptional<z.ZodArray<z.ZodString>>;
13
+ }, z.core.$strip>;
14
+ /** Validated type for AI generation request body. */
15
+ export type ValidatedAIGenerateBody = z.infer<typeof AIGenerateBodySchema>;
16
+ /**
17
+ * Schema for POST /plugins/deploy-generated request body.
18
+ * Validates the AI-generated plugin configuration before Docker build.
19
+ */
20
+ export declare const PluginDeployGeneratedSchema: z.ZodObject<{
21
+ name: z.ZodString;
22
+ description: z.ZodOptional<z.ZodString>;
23
+ version: z.ZodString;
24
+ pluginType: z.ZodDefault<z.ZodEnum<{
25
+ CodeBuildStep: "CodeBuildStep";
26
+ ShellStep: "ShellStep";
27
+ ManualApprovalStep: "ManualApprovalStep";
28
+ }>>;
29
+ computeType: z.ZodDefault<z.ZodEnum<{
30
+ SMALL: "SMALL";
31
+ MEDIUM: "MEDIUM";
32
+ LARGE: "LARGE";
33
+ X2_LARGE: "X2_LARGE";
34
+ }>>;
35
+ keywords: z.ZodDefault<z.ZodOptional<z.ZodArray<z.ZodString>>>;
36
+ primaryOutputDirectory: z.ZodOptional<z.ZodNullable<z.ZodString>>;
37
+ installCommands: z.ZodDefault<z.ZodOptional<z.ZodArray<z.ZodString>>>;
38
+ commands: z.ZodArray<z.ZodString>;
39
+ env: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodString>>;
40
+ buildArgs: z.ZodOptional<z.ZodRecord<z.ZodString, z.ZodString>>;
41
+ dockerfile: z.ZodString;
42
+ accessModifier: z.ZodDefault<z.ZodEnum<{
43
+ public: "public";
44
+ private: "private";
45
+ }>>;
46
+ }, z.core.$strip>;
47
+ /** Validated type for plugin deploy-generated request body. */
48
+ export type ValidatedPluginDeployGenerated = z.infer<typeof PluginDeployGeneratedSchema>;
49
+ /**
50
+ * Schema for POST /pipelines/generate/from-url/stream request body.
51
+ * Validates Git URL + AI provider configuration for repo-based pipeline generation.
52
+ */
53
+ export declare const AIGenerateFromUrlBodySchema: z.ZodObject<{
54
+ gitUrl: z.ZodString;
55
+ provider: z.ZodString;
56
+ model: z.ZodString;
57
+ apiKey: z.ZodOptional<z.ZodString>;
58
+ repoToken: z.ZodOptional<z.ZodString>;
59
+ }, z.core.$strip>;
60
+ /** Validated type for AI generate-from-URL request body. */
61
+ export type ValidatedAIGenerateFromUrlBody = z.infer<typeof AIGenerateFromUrlBodySchema>;