@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.
- package/LICENSE +202 -0
- package/README.md +51 -0
- package/lib/constants/ai-providers.d.ts +41 -0
- package/lib/constants/ai-providers.js +88 -0
- package/lib/constants/http-status.d.ts +24 -0
- package/lib/constants/http-status.js +29 -0
- package/lib/constants/index.d.ts +3 -0
- package/lib/constants/index.js +22 -0
- package/lib/constants/time.d.ts +10 -0
- package/lib/constants/time.js +16 -0
- package/lib/errors/app-errors.d.ts +30 -0
- package/lib/errors/app-errors.js +62 -0
- package/lib/errors/index.d.ts +1 -0
- package/lib/errors/index.js +20 -0
- package/lib/helpers/access-helpers.d.ts +40 -0
- package/lib/helpers/access-helpers.js +56 -0
- package/lib/helpers/crud-helpers.d.ts +16 -0
- package/lib/helpers/crud-helpers.js +34 -0
- package/lib/helpers/index.d.ts +4 -0
- package/lib/helpers/index.js +23 -0
- package/lib/helpers/mask-helpers.d.ts +33 -0
- package/lib/helpers/mask-helpers.js +54 -0
- package/lib/helpers/sse-helpers.d.ts +13 -0
- package/lib/helpers/sse-helpers.js +40 -0
- package/lib/index.d.ts +57 -0
- package/lib/index.js +86 -0
- package/lib/middleware/auth.d.ts +50 -0
- package/lib/middleware/auth.js +171 -0
- package/lib/middleware/index.d.ts +1 -0
- package/lib/middleware/index.js +20 -0
- package/lib/openapi/extend-zod.d.ts +1 -0
- package/lib/openapi/extend-zod.js +8 -0
- package/lib/openapi/index.d.ts +2 -0
- package/lib/openapi/index.js +10 -0
- package/lib/openapi/registry.d.ts +17 -0
- package/lib/openapi/registry.js +42 -0
- package/lib/openapi/routes/billing-routes.d.ts +1 -0
- package/lib/openapi/routes/billing-routes.js +69 -0
- package/lib/openapi/routes/index.d.ts +5 -0
- package/lib/openapi/routes/index.js +22 -0
- package/lib/openapi/routes/message-routes.d.ts +1 -0
- package/lib/openapi/routes/message-routes.js +108 -0
- package/lib/openapi/routes/pipeline-routes.d.ts +1 -0
- package/lib/openapi/routes/pipeline-routes.js +90 -0
- package/lib/openapi/routes/plugin-routes.d.ts +1 -0
- package/lib/openapi/routes/plugin-routes.js +99 -0
- package/lib/openapi/routes/quota-routes.d.ts +1 -0
- package/lib/openapi/routes/quota-routes.js +65 -0
- package/lib/openapi/schema-registry.d.ts +25 -0
- package/lib/openapi/schema-registry.js +95 -0
- package/lib/routes/health.d.ts +47 -0
- package/lib/routes/health.js +81 -0
- package/lib/routes/index.d.ts +1 -0
- package/lib/routes/index.js +20 -0
- package/lib/services/admin-audit.d.ts +13 -0
- package/lib/services/admin-audit.js +31 -0
- package/lib/services/cache-service.d.ts +108 -0
- package/lib/services/cache-service.js +212 -0
- package/lib/services/compliance-client.d.ts +46 -0
- package/lib/services/compliance-client.js +102 -0
- package/lib/services/compliance-event-subscriber.d.ts +11 -0
- package/lib/services/compliance-event-subscriber.js +60 -0
- package/lib/services/compliance-queue.d.ts +11 -0
- package/lib/services/compliance-queue.js +38 -0
- package/lib/services/entity-events.d.ts +44 -0
- package/lib/services/entity-events.js +63 -0
- package/lib/services/http-client.d.ts +108 -0
- package/lib/services/http-client.js +285 -0
- package/lib/services/index.d.ts +10 -0
- package/lib/services/index.js +40 -0
- package/lib/services/quota.d.ts +59 -0
- package/lib/services/quota.js +137 -0
- package/lib/services/retry-strategy.d.ts +74 -0
- package/lib/services/retry-strategy.js +127 -0
- package/lib/types/billing.d.ts +47 -0
- package/lib/types/billing.js +5 -0
- package/lib/types/common.d.ts +161 -0
- package/lib/types/common.js +53 -0
- package/lib/types/error-codes.d.ts +38 -0
- package/lib/types/error-codes.js +77 -0
- package/lib/types/feature-flags.d.ts +38 -0
- package/lib/types/feature-flags.js +107 -0
- package/lib/types/http.d.ts +37 -0
- package/lib/types/http.js +5 -0
- package/lib/types/index.d.ts +7 -0
- package/lib/types/index.js +26 -0
- package/lib/types/pipeline.d.ts +70 -0
- package/lib/types/pipeline.js +44 -0
- package/lib/types/quota-tiers.d.ts +23 -0
- package/lib/types/quota-tiers.js +26 -0
- package/lib/utils/alias-resolver.d.ts +16 -0
- package/lib/utils/alias-resolver.js +49 -0
- package/lib/utils/headers.d.ts +18 -0
- package/lib/utils/headers.js +24 -0
- package/lib/utils/identity.d.ts +61 -0
- package/lib/utils/identity.js +75 -0
- package/lib/utils/index.d.ts +7 -0
- package/lib/utils/index.js +26 -0
- package/lib/utils/logger.d.ts +28 -0
- package/lib/utils/logger.js +77 -0
- package/lib/utils/object.d.ts +13 -0
- package/lib/utils/object.js +21 -0
- package/lib/utils/params.d.ts +89 -0
- package/lib/utils/params.js +148 -0
- package/lib/utils/response.d.ts +142 -0
- package/lib/utils/response.js +237 -0
- package/lib/validation/ai-schemas.d.ts +61 -0
- package/lib/validation/ai-schemas.js +81 -0
- package/lib/validation/common-schemas.d.ts +72 -0
- package/lib/validation/common-schemas.js +58 -0
- package/lib/validation/index.d.ts +6 -0
- package/lib/validation/index.js +25 -0
- package/lib/validation/message-schemas.d.ts +79 -0
- package/lib/validation/message-schemas.js +42 -0
- package/lib/validation/middleware.d.ts +60 -0
- package/lib/validation/middleware.js +77 -0
- package/lib/validation/pipeline-schemas.d.ts +135 -0
- package/lib/validation/pipeline-schemas.js +85 -0
- package/lib/validation/plugin-schemas.d.ts +127 -0
- package/lib/validation/plugin-schemas.js +84 -0
- package/openapi.yaml +292 -0
- 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>;
|