@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,171 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright 2026 Pipeline Builder Contributors
|
|
3
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
5
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
6
|
+
};
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.SYSTEM_ORG_ID = void 0;
|
|
9
|
+
exports.requireAuth = requireAuth;
|
|
10
|
+
exports.requireOrganization = requireOrganization;
|
|
11
|
+
exports.requireAdmin = requireAdmin;
|
|
12
|
+
exports.isSystemOrgId = isSystemOrgId;
|
|
13
|
+
exports.isSystemOrg = isSystemOrg;
|
|
14
|
+
exports.isSystemAdmin = isSystemAdmin;
|
|
15
|
+
exports.requireSystemAdmin = requireSystemAdmin;
|
|
16
|
+
exports.requireFeature = requireFeature;
|
|
17
|
+
exports.resolveAccessModifier = resolveAccessModifier;
|
|
18
|
+
const jsonwebtoken_1 = __importDefault(require("jsonwebtoken"));
|
|
19
|
+
const http_status_1 = require("../constants/http-status");
|
|
20
|
+
const error_codes_1 = require("../types/error-codes");
|
|
21
|
+
const headers_1 = require("../utils/headers");
|
|
22
|
+
const logger_1 = require("../utils/logger");
|
|
23
|
+
const response_1 = require("../utils/response");
|
|
24
|
+
const logger = (0, logger_1.createLogger)('auth-middleware');
|
|
25
|
+
/** Cached JWT secret with periodic refresh from env var. */
|
|
26
|
+
let _jwtSecret;
|
|
27
|
+
let _jwtSecretRefreshedAt = 0;
|
|
28
|
+
const JWT_SECRET_REFRESH_INTERVAL_MS = 300_000; // 5 minutes
|
|
29
|
+
function getJwtSecret() {
|
|
30
|
+
const now = Date.now();
|
|
31
|
+
if (!_jwtSecret || now - _jwtSecretRefreshedAt > JWT_SECRET_REFRESH_INTERVAL_MS) {
|
|
32
|
+
const secret = process.env.JWT_SECRET;
|
|
33
|
+
if (!secret) {
|
|
34
|
+
logger.error('JWT_SECRET environment variable is not set');
|
|
35
|
+
throw new Error('JWT_SECRET environment variable is required');
|
|
36
|
+
}
|
|
37
|
+
if (_jwtSecret && _jwtSecret !== secret) {
|
|
38
|
+
logger.info('JWT secret rotated');
|
|
39
|
+
}
|
|
40
|
+
_jwtSecret = secret;
|
|
41
|
+
_jwtSecretRefreshedAt = now;
|
|
42
|
+
}
|
|
43
|
+
return _jwtSecret;
|
|
44
|
+
}
|
|
45
|
+
function requireAuth(reqOrOptions, res, next) {
|
|
46
|
+
if (reqOrOptions && res && next && 'headers' in reqOrOptions) {
|
|
47
|
+
return _requireAuth({}, reqOrOptions, res, next);
|
|
48
|
+
}
|
|
49
|
+
const options = reqOrOptions || {};
|
|
50
|
+
return (req, resInner, nextInner) => {
|
|
51
|
+
_requireAuth(options, req, resInner, nextInner);
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
function _requireAuth(options, req, res, next) {
|
|
55
|
+
const authHeader = req.headers.authorization;
|
|
56
|
+
if (!authHeader) {
|
|
57
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Authorization header required', error_codes_1.ErrorCode.TOKEN_MISSING);
|
|
58
|
+
}
|
|
59
|
+
const parts = authHeader.split(' ');
|
|
60
|
+
if (parts.length !== 2 || parts[0] !== 'Bearer') {
|
|
61
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Invalid authorization format. Use: Bearer <token>', error_codes_1.ErrorCode.TOKEN_INVALID);
|
|
62
|
+
}
|
|
63
|
+
try {
|
|
64
|
+
const decoded = jsonwebtoken_1.default.verify(parts[1], getJwtSecret());
|
|
65
|
+
if (decoded.type !== 'access') {
|
|
66
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Only access tokens can be used for API requests', error_codes_1.ErrorCode.TOKEN_INVALID);
|
|
67
|
+
}
|
|
68
|
+
if (!decoded.sub || !decoded.role) {
|
|
69
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Token missing required fields', error_codes_1.ErrorCode.TOKEN_INVALID);
|
|
70
|
+
}
|
|
71
|
+
req.user = { ...decoded };
|
|
72
|
+
if (options.allowOrgHeaderOverride) {
|
|
73
|
+
const headerOrgId = (0, headers_1.getHeaderString)(req.headers['x-org-id']);
|
|
74
|
+
const headerOrgName = (0, headers_1.getHeaderString)(req.headers['x-org-name']);
|
|
75
|
+
if (headerOrgId)
|
|
76
|
+
req.user.organizationId = headerOrgId;
|
|
77
|
+
if (headerOrgName)
|
|
78
|
+
req.user.organizationName = headerOrgName;
|
|
79
|
+
}
|
|
80
|
+
next();
|
|
81
|
+
}
|
|
82
|
+
catch (error) {
|
|
83
|
+
if (error instanceof jsonwebtoken_1.default.TokenExpiredError) {
|
|
84
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Token has expired', error_codes_1.ErrorCode.TOKEN_EXPIRED);
|
|
85
|
+
}
|
|
86
|
+
if (error instanceof jsonwebtoken_1.default.JsonWebTokenError) {
|
|
87
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Invalid token', error_codes_1.ErrorCode.TOKEN_INVALID);
|
|
88
|
+
}
|
|
89
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Authentication failed', error_codes_1.ErrorCode.UNAUTHORIZED);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
/** Requires organization membership. Use after requireAuth. */
|
|
93
|
+
function requireOrganization(req, res, next) {
|
|
94
|
+
if (!req.user) {
|
|
95
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Authentication required', error_codes_1.ErrorCode.UNAUTHORIZED);
|
|
96
|
+
}
|
|
97
|
+
if (!req.user.organizationId) {
|
|
98
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.BAD_REQUEST, 'Organization membership required', error_codes_1.ErrorCode.ORG_MISMATCH);
|
|
99
|
+
}
|
|
100
|
+
next();
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Requires admin role. Use after requireAuth.
|
|
104
|
+
* Permits users whose per-org role is 'admin' or 'owner'.
|
|
105
|
+
*/
|
|
106
|
+
function requireAdmin(req, res, next) {
|
|
107
|
+
if (!req.user) {
|
|
108
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Authentication required', error_codes_1.ErrorCode.UNAUTHORIZED);
|
|
109
|
+
}
|
|
110
|
+
if (req.user.role !== 'admin' && req.user.role !== 'owner') {
|
|
111
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.FORBIDDEN, 'Admin access required', error_codes_1.ErrorCode.INSUFFICIENT_PERMISSIONS);
|
|
112
|
+
}
|
|
113
|
+
next();
|
|
114
|
+
}
|
|
115
|
+
/** Organization ID/name that identifies the system (super-admin) tenant. */
|
|
116
|
+
exports.SYSTEM_ORG_ID = (process.env.SYSTEM_ORG_ID || 'system').toLowerCase();
|
|
117
|
+
/**
|
|
118
|
+
* Check if an orgId or orgName matches the system org.
|
|
119
|
+
* Use this instead of comparing directly against SYSTEM_ORG_ID,
|
|
120
|
+
* because the JWT orgId is a database ID (e.g. MongoDB ObjectId)
|
|
121
|
+
* while SYSTEM_ORG_ID is the well-known name "system".
|
|
122
|
+
*/
|
|
123
|
+
function isSystemOrgId(orgId, orgName) {
|
|
124
|
+
return orgId?.toLowerCase() === exports.SYSTEM_ORG_ID || orgName?.toLowerCase() === exports.SYSTEM_ORG_ID;
|
|
125
|
+
}
|
|
126
|
+
function isSystemOrg(req) {
|
|
127
|
+
if (!req.user)
|
|
128
|
+
return false;
|
|
129
|
+
return isSystemOrgId(req.user.organizationId, req.user.organizationName);
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Check if the request is from a system admin.
|
|
133
|
+
* A system admin has per-org role 'admin' or 'owner' in the system organization.
|
|
134
|
+
*/
|
|
135
|
+
function isSystemAdmin(req) {
|
|
136
|
+
return (req.user?.role === 'admin' || req.user?.role === 'owner') && isSystemOrg(req);
|
|
137
|
+
}
|
|
138
|
+
/** Requires system admin (admin role + system organization). */
|
|
139
|
+
function requireSystemAdmin(req, res, next) {
|
|
140
|
+
if (!isSystemAdmin(req)) {
|
|
141
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.FORBIDDEN, 'Access denied. Only system administrators can perform this action.', error_codes_1.ErrorCode.INSUFFICIENT_PERMISSIONS);
|
|
142
|
+
}
|
|
143
|
+
next();
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Require a specific feature flag. Use after requireAuth.
|
|
147
|
+
* Checks the `features` array in the JWT payload (set at token issuance).
|
|
148
|
+
* System org users always pass (all features enabled).
|
|
149
|
+
*/
|
|
150
|
+
function requireFeature(feature) {
|
|
151
|
+
return (req, res, next) => {
|
|
152
|
+
if (!req.user) {
|
|
153
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.UNAUTHORIZED, 'Authentication required', error_codes_1.ErrorCode.UNAUTHORIZED);
|
|
154
|
+
}
|
|
155
|
+
// System org always has all features
|
|
156
|
+
if (isSystemOrg(req))
|
|
157
|
+
return next();
|
|
158
|
+
if (!req.user.features?.includes(feature)) {
|
|
159
|
+
return (0, response_1.sendError)(res, http_status_1.HttpStatus.FORBIDDEN, `This feature requires a higher plan (${feature})`, error_codes_1.ErrorCode.INSUFFICIENT_PERMISSIONS);
|
|
160
|
+
}
|
|
161
|
+
next();
|
|
162
|
+
};
|
|
163
|
+
}
|
|
164
|
+
/** Only system admins can set access to 'public'; everyone else gets 'private'. */
|
|
165
|
+
function resolveAccessModifier(req, requested) {
|
|
166
|
+
if (requested === 'public' && isSystemAdmin(req)) {
|
|
167
|
+
return 'public';
|
|
168
|
+
}
|
|
169
|
+
return 'private';
|
|
170
|
+
}
|
|
171
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auth.js","sourceRoot":"","sources":["../../src/middleware/auth.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { Request, Response, NextFunction } from 'express';\nimport jwt from 'jsonwebtoken';\nimport { HttpStatus } from '../constants/http-status';\nimport { JwtPayload } from '../types/common';\nimport { ErrorCode } from '../types/error-codes';\nimport { getHeaderString } from '../utils/headers';\nimport { createLogger } from '../utils/logger';\nimport { sendError } from '../utils/response';\n\nconst logger = createLogger('auth-middleware');\n\n/** Cached JWT secret with periodic refresh from env var. */\nlet _jwtSecret: string | undefined;\nlet _jwtSecretRefreshedAt = 0;\nconst JWT_SECRET_REFRESH_INTERVAL_MS = 300_000; // 5 minutes\n\nfunction getJwtSecret(): string {\n  const now = Date.now();\n  if (!_jwtSecret || now - _jwtSecretRefreshedAt > JWT_SECRET_REFRESH_INTERVAL_MS) {\n    const secret = process.env.JWT_SECRET;\n    if (!secret) {\n      logger.error('JWT_SECRET environment variable is not set');\n      throw new Error('JWT_SECRET environment variable is required');\n    }\n    if (_jwtSecret && _jwtSecret !== secret) {\n      logger.info('JWT secret rotated');\n    }\n    _jwtSecret = secret;\n    _jwtSecretRefreshedAt = now;\n  }\n  return _jwtSecret;\n}\n\nexport interface RequireAuthOptions {\n  /**\n   * Allow x-org-id/x-org-name headers to override the JWT's organization fields.\n   *\n   * **SECURITY WARNING:** When enabled, a caller can set `x-org-id` to ANY\n   * organization ID, effectively impersonating that org. This MUST only be\n   * used on routes that are:\n   *   1. Internal service-to-service routes (not exposed to end users)\n   *   2. Behind network isolation (container network, VPC, etc.)\n   *\n   * NEVER enable this on user-facing API routes. If unsure, leave it disabled.\n   */\n  allowOrgHeaderOverride?: boolean;\n}\n\n/** JWT auth middleware. Use directly or call with options. */\nexport function requireAuth(\n  req: Request,\n  res: Response,\n  next: NextFunction,\n): void;\nexport function requireAuth(\n  options?: RequireAuthOptions,\n): (req: Request, res: Response, next: NextFunction) => void;\nexport function requireAuth(\n  reqOrOptions?: Request | RequireAuthOptions,\n  res?: Response,\n  next?: NextFunction,\n): void | ((req: Request, res: Response, next: NextFunction) => void) {\n  if (reqOrOptions && res && next && 'headers' in reqOrOptions) {\n    return _requireAuth({}, reqOrOptions as Request, res, next);\n  }\n\n  const options = (reqOrOptions as RequireAuthOptions) || {};\n  return (req: Request, resInner: Response, nextInner: NextFunction) => {\n    _requireAuth(options, req, resInner, nextInner);\n  };\n}\n\nfunction _requireAuth(\n  options: RequireAuthOptions,\n  req: Request,\n  res: Response,\n  next: NextFunction,\n): void {\n  const authHeader = req.headers.authorization;\n\n  if (!authHeader) {\n    return sendError(res, HttpStatus.UNAUTHORIZED, 'Authorization header required', ErrorCode.TOKEN_MISSING);\n  }\n\n  const parts = authHeader.split(' ');\n  if (parts.length !== 2 || parts[0] !== 'Bearer') {\n    return sendError(res, HttpStatus.UNAUTHORIZED, 'Invalid authorization format. Use: Bearer <token>', ErrorCode.TOKEN_INVALID);\n  }\n\n  try {\n    const decoded = jwt.verify(parts[1], getJwtSecret()) as JwtPayload;\n\n    if (decoded.type !== 'access') {\n      return sendError(res, HttpStatus.UNAUTHORIZED, 'Only access tokens can be used for API requests', ErrorCode.TOKEN_INVALID);\n    }\n\n    if (!decoded.sub || !decoded.role) {\n      return sendError(res, HttpStatus.UNAUTHORIZED, 'Token missing required fields', ErrorCode.TOKEN_INVALID);\n    }\n\n    req.user = { ...decoded };\n\n    if (options.allowOrgHeaderOverride) {\n      const headerOrgId = getHeaderString(req.headers['x-org-id']);\n      const headerOrgName = getHeaderString(req.headers['x-org-name']);\n      if (headerOrgId) req.user.organizationId = headerOrgId;\n      if (headerOrgName) req.user.organizationName = headerOrgName;\n    }\n\n    next();\n  } catch (error) {\n    if (error instanceof jwt.TokenExpiredError) {\n      return sendError(res, HttpStatus.UNAUTHORIZED, 'Token has expired', ErrorCode.TOKEN_EXPIRED);\n    }\n\n    if (error instanceof jwt.JsonWebTokenError) {\n      return sendError(res, HttpStatus.UNAUTHORIZED, 'Invalid token', ErrorCode.TOKEN_INVALID);\n    }\n\n    return sendError(res, HttpStatus.UNAUTHORIZED, 'Authentication failed', ErrorCode.UNAUTHORIZED);\n  }\n}\n\n\n/** Requires organization membership. Use after requireAuth. */\nexport function requireOrganization(\n  req: Request,\n  res: Response,\n  next: NextFunction,\n): void {\n  if (!req.user) {\n    return sendError(res, HttpStatus.UNAUTHORIZED, 'Authentication required', ErrorCode.UNAUTHORIZED);\n  }\n\n  if (!req.user.organizationId) {\n    return sendError(res, HttpStatus.BAD_REQUEST, 'Organization membership required', ErrorCode.ORG_MISMATCH);\n  }\n\n  next();\n}\n\n/**\n * Requires admin role. Use after requireAuth.\n * Permits users whose per-org role is 'admin' or 'owner'.\n */\nexport function requireAdmin(\n  req: Request,\n  res: Response,\n  next: NextFunction,\n): void {\n  if (!req.user) {\n    return sendError(res, HttpStatus.UNAUTHORIZED, 'Authentication required', ErrorCode.UNAUTHORIZED);\n  }\n\n  if (req.user.role !== 'admin' && req.user.role !== 'owner') {\n    return sendError(res, HttpStatus.FORBIDDEN, 'Admin access required', ErrorCode.INSUFFICIENT_PERMISSIONS);\n  }\n\n  next();\n}\n\n/** Organization ID/name that identifies the system (super-admin) tenant. */\nexport const SYSTEM_ORG_ID = (process.env.SYSTEM_ORG_ID || 'system').toLowerCase();\n\n/**\n * Check if an orgId or orgName matches the system org.\n * Use this instead of comparing directly against SYSTEM_ORG_ID,\n * because the JWT orgId is a database ID (e.g. MongoDB ObjectId)\n * while SYSTEM_ORG_ID is the well-known name \"system\".\n */\nexport function isSystemOrgId(orgId?: string, orgName?: string): boolean {\n  return orgId?.toLowerCase() === SYSTEM_ORG_ID || orgName?.toLowerCase() === SYSTEM_ORG_ID;\n}\n\nexport function isSystemOrg(req: Request): boolean {\n  if (!req.user) return false;\n  return isSystemOrgId(req.user.organizationId, req.user.organizationName);\n}\n\n/**\n * Check if the request is from a system admin.\n * A system admin has per-org role 'admin' or 'owner' in the system organization.\n */\nexport function isSystemAdmin(req: Request): boolean {\n  return (req.user?.role === 'admin' || req.user?.role === 'owner') && isSystemOrg(req);\n}\n\n/** Requires system admin (admin role + system organization). */\nexport function requireSystemAdmin(\n  req: Request,\n  res: Response,\n  next: NextFunction,\n): void {\n  if (!isSystemAdmin(req)) {\n    return sendError(\n      res, HttpStatus.FORBIDDEN,\n      'Access denied. Only system administrators can perform this action.',\n      ErrorCode.INSUFFICIENT_PERMISSIONS,\n    );\n  }\n  next();\n}\n\n/**\n * Require a specific feature flag. Use after requireAuth.\n * Checks the `features` array in the JWT payload (set at token issuance).\n * System org users always pass (all features enabled).\n */\nexport function requireFeature(feature: string) {\n  return (req: Request, res: Response, next: NextFunction): void => {\n    if (!req.user) {\n      return sendError(res, HttpStatus.UNAUTHORIZED, 'Authentication required', ErrorCode.UNAUTHORIZED);\n    }\n\n    // System org always has all features\n    if (isSystemOrg(req)) return next();\n\n    if (!req.user.features?.includes(feature)) {\n      return sendError(\n        res, HttpStatus.FORBIDDEN,\n        `This feature requires a higher plan (${feature})`,\n        ErrorCode.INSUFFICIENT_PERMISSIONS,\n      );\n    }\n\n    next();\n  };\n}\n\n/** Only system admins can set access to 'public'; everyone else gets 'private'. */\nexport function resolveAccessModifier(req: Request, requested: string | undefined): 'public' | 'private' {\n  if (requested === 'public' && isSystemAdmin(req)) {\n    return 'public';\n  }\n  return 'private';\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './auth';
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright 2026 Pipeline Builder Contributors
|
|
3
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
5
|
+
if (k2 === undefined) k2 = k;
|
|
6
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
7
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
8
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
9
|
+
}
|
|
10
|
+
Object.defineProperty(o, k2, desc);
|
|
11
|
+
}) : (function(o, m, k, k2) {
|
|
12
|
+
if (k2 === undefined) k2 = k;
|
|
13
|
+
o[k2] = m[k];
|
|
14
|
+
}));
|
|
15
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
16
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
17
|
+
};
|
|
18
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
19
|
+
__exportStar(require("./auth"), exports);
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvbWlkZGxld2FyZS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsK0NBQStDO0FBQy9DLHNDQUFzQzs7Ozs7Ozs7Ozs7Ozs7OztBQUV0Qyx5Q0FBdUIiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgMjAyNiBQaXBlbGluZSBCdWlsZGVyIENvbnRyaWJ1dG9yc1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcblxuZXhwb3J0ICogZnJvbSAnLi9hdXRoJztcbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright 2026 Pipeline Builder Contributors
|
|
3
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
5
|
+
const zod_to_openapi_1 = require("@asteasolutions/zod-to-openapi");
|
|
6
|
+
const zod_1 = require("zod");
|
|
7
|
+
(0, zod_to_openapi_1.extendZodWithOpenApi)(zod_1.z);
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXh0ZW5kLXpvZC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9vcGVuYXBpL2V4dGVuZC16b2QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLCtDQUErQztBQUMvQyxzQ0FBc0M7O0FBRXRDLG1FQUFzRTtBQUN0RSw2QkFBd0I7QUFFeEIsSUFBQSxxQ0FBb0IsRUFBQyxPQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG5pbXBvcnQgeyBleHRlbmRab2RXaXRoT3BlbkFwaSB9IGZyb20gJ0Bhc3RlYXNvbHV0aW9ucy96b2QtdG8tb3BlbmFwaSc7XG5pbXBvcnQgeyB6IH0gZnJvbSAnem9kJztcblxuZXh0ZW5kWm9kV2l0aE9wZW5BcGkoeik7XG4iXX0=
|
|
@@ -0,0 +1,10 @@
|
|
|
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.generateOpenApiSpec = exports.registry = void 0;
|
|
6
|
+
// Public API
|
|
7
|
+
var registry_1 = require("./registry");
|
|
8
|
+
Object.defineProperty(exports, "registry", { enumerable: true, get: function () { return registry_1.registry; } });
|
|
9
|
+
Object.defineProperty(exports, "generateOpenApiSpec", { enumerable: true, get: function () { return registry_1.generateOpenApiSpec; } });
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvb3BlbmFwaS9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsK0NBQStDO0FBQy9DLHNDQUFzQzs7O0FBRXRDLGFBQWE7QUFDYix1Q0FBMkQ7QUFBbEQsb0dBQUEsUUFBUSxPQUFBO0FBQUUsK0dBQUEsbUJBQW1CLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgMjAyNiBQaXBlbGluZSBCdWlsZGVyIENvbnRyaWJ1dG9yc1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcblxuLy8gUHVibGljIEFQSVxuZXhwb3J0IHsgcmVnaXN0cnksIGdlbmVyYXRlT3BlbkFwaVNwZWMgfSBmcm9tICcuL3JlZ2lzdHJ5JztcbmV4cG9ydCB0eXBlIHsgT3BlbkFwaVNwZWNPcHRpb25zIH0gZnJvbSAnLi9yZWdpc3RyeSc7XG4iXX0=
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { OpenAPIRegistry, OpenApiGeneratorV31 } from '@asteasolutions/zod-to-openapi';
|
|
2
|
+
/** Shared OpenAPI registry for all schema and path registrations. */
|
|
3
|
+
export declare const registry: OpenAPIRegistry;
|
|
4
|
+
/** Options for customizing the generated OpenAPI spec. */
|
|
5
|
+
export interface OpenApiSpecOptions {
|
|
6
|
+
title?: string;
|
|
7
|
+
version?: string;
|
|
8
|
+
description?: string;
|
|
9
|
+
serverUrl?: string;
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* Generate the complete OpenAPI 3.1.0 specification document.
|
|
13
|
+
*
|
|
14
|
+
* @param options - Optional overrides for spec metadata
|
|
15
|
+
* @returns OpenAPI 3.1 specification object
|
|
16
|
+
*/
|
|
17
|
+
export declare function generateOpenApiSpec(options?: OpenApiSpecOptions): ReturnType<OpenApiGeneratorV31['generateDocument']>;
|
|
@@ -0,0 +1,42 @@
|
|
|
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.registry = void 0;
|
|
6
|
+
exports.generateOpenApiSpec = generateOpenApiSpec;
|
|
7
|
+
const zod_to_openapi_1 = require("@asteasolutions/zod-to-openapi");
|
|
8
|
+
const routes_1 = require("./routes");
|
|
9
|
+
const schema_registry_1 = require("./schema-registry");
|
|
10
|
+
/** Shared OpenAPI registry for all schema and path registrations. */
|
|
11
|
+
exports.registry = new zod_to_openapi_1.OpenAPIRegistry();
|
|
12
|
+
let _initialized = false;
|
|
13
|
+
/** Execute all registrations once. */
|
|
14
|
+
function ensureInitialized() {
|
|
15
|
+
if (_initialized)
|
|
16
|
+
return;
|
|
17
|
+
_initialized = true;
|
|
18
|
+
(0, schema_registry_1.registerSchemas)();
|
|
19
|
+
(0, routes_1.registerAllRoutes)();
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Generate the complete OpenAPI 3.1.0 specification document.
|
|
23
|
+
*
|
|
24
|
+
* @param options - Optional overrides for spec metadata
|
|
25
|
+
* @returns OpenAPI 3.1 specification object
|
|
26
|
+
*/
|
|
27
|
+
function generateOpenApiSpec(options) {
|
|
28
|
+
ensureInitialized();
|
|
29
|
+
const generator = new zod_to_openapi_1.OpenApiGeneratorV31(exports.registry.definitions);
|
|
30
|
+
return generator.generateDocument({
|
|
31
|
+
openapi: '3.1.0',
|
|
32
|
+
info: {
|
|
33
|
+
title: options?.title ?? 'Pipeline Builder API',
|
|
34
|
+
version: options?.version ?? '1.0.0',
|
|
35
|
+
description: options?.description ?? 'CI/CD Pipeline Builder Platform API — manage pipelines, plugins, messages, quotas, and billing.',
|
|
36
|
+
},
|
|
37
|
+
servers: [
|
|
38
|
+
{ url: options?.serverUrl ?? '/api', description: 'API base path' },
|
|
39
|
+
],
|
|
40
|
+
});
|
|
41
|
+
}
|
|
42
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVnaXN0cnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvb3BlbmFwaS9yZWdpc3RyeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsK0NBQStDO0FBQy9DLHNDQUFzQzs7O0FBa0N0QyxrREFpQkM7QUFqREQsbUVBQXNGO0FBRXRGLHFDQUE2QztBQUM3Qyx1REFBb0Q7QUFFcEQscUVBQXFFO0FBQ3hELFFBQUEsUUFBUSxHQUFHLElBQUksZ0NBQWUsRUFBRSxDQUFDO0FBVTlDLElBQUksWUFBWSxHQUFHLEtBQUssQ0FBQztBQUV6QixzQ0FBc0M7QUFDdEMsU0FBUyxpQkFBaUI7SUFDeEIsSUFBSSxZQUFZO1FBQUUsT0FBTztJQUN6QixZQUFZLEdBQUcsSUFBSSxDQUFDO0lBQ3BCLElBQUEsaUNBQWUsR0FBRSxDQUFDO0lBQ2xCLElBQUEsMEJBQWlCLEdBQUUsQ0FBQztBQUN0QixDQUFDO0FBRUQ7Ozs7O0dBS0c7QUFDSCxTQUFnQixtQkFBbUIsQ0FDakMsT0FBNEI7SUFFNUIsaUJBQWlCLEVBQUUsQ0FBQztJQUVwQixNQUFNLFNBQVMsR0FBRyxJQUFJLG9DQUFtQixDQUFDLGdCQUFRLENBQUMsV0FBVyxDQUFDLENBQUM7SUFDaEUsT0FBTyxTQUFTLENBQUMsZ0JBQWdCLENBQUM7UUFDaEMsT0FBTyxFQUFFLE9BQU87UUFDaEIsSUFBSSxFQUFFO1lBQ0osS0FBSyxFQUFFLE9BQU8sRUFBRSxLQUFLLElBQUksc0JBQXNCO1lBQy9DLE9BQU8sRUFBRSxPQUFPLEVBQUUsT0FBTyxJQUFJLE9BQU87WUFDcEMsV0FBVyxFQUFFLE9BQU8sRUFBRSxXQUFXLElBQUksaUdBQWlHO1NBQ3ZJO1FBQ0QsT0FBTyxFQUFFO1lBQ1AsRUFBRSxHQUFHLEVBQUUsT0FBTyxFQUFFLFNBQVMsSUFBSSxNQUFNLEVBQUUsV0FBVyxFQUFFLGVBQWUsRUFBRTtTQUNwRTtLQUNGLENBQUMsQ0FBQztBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgMjAyNiBQaXBlbGluZSBCdWlsZGVyIENvbnRyaWJ1dG9yc1xuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcblxuaW1wb3J0IHsgT3BlbkFQSVJlZ2lzdHJ5LCBPcGVuQXBpR2VuZXJhdG9yVjMxIH0gZnJvbSAnQGFzdGVhc29sdXRpb25zL3pvZC10by1vcGVuYXBpJztcblxuaW1wb3J0IHsgcmVnaXN0ZXJBbGxSb3V0ZXMgfSBmcm9tICcuL3JvdXRlcyc7XG5pbXBvcnQgeyByZWdpc3RlclNjaGVtYXMgfSBmcm9tICcuL3NjaGVtYS1yZWdpc3RyeSc7XG5cbi8qKiBTaGFyZWQgT3BlbkFQSSByZWdpc3RyeSBmb3IgYWxsIHNjaGVtYSBhbmQgcGF0aCByZWdpc3RyYXRpb25zLiAqL1xuZXhwb3J0IGNvbnN0IHJlZ2lzdHJ5ID0gbmV3IE9wZW5BUElSZWdpc3RyeSgpO1xuXG4vKiogT3B0aW9ucyBmb3IgY3VzdG9taXppbmcgdGhlIGdlbmVyYXRlZCBPcGVuQVBJIHNwZWMuICovXG5leHBvcnQgaW50ZXJmYWNlIE9wZW5BcGlTcGVjT3B0aW9ucyB7XG4gIHRpdGxlPzogc3RyaW5nO1xuICB2ZXJzaW9uPzogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgc2VydmVyVXJsPzogc3RyaW5nO1xufVxuXG5sZXQgX2luaXRpYWxpemVkID0gZmFsc2U7XG5cbi8qKiBFeGVjdXRlIGFsbCByZWdpc3RyYXRpb25zIG9uY2UuICovXG5mdW5jdGlvbiBlbnN1cmVJbml0aWFsaXplZCgpOiB2b2lkIHtcbiAgaWYgKF9pbml0aWFsaXplZCkgcmV0dXJuO1xuICBfaW5pdGlhbGl6ZWQgPSB0cnVlO1xuICByZWdpc3RlclNjaGVtYXMoKTtcbiAgcmVnaXN0ZXJBbGxSb3V0ZXMoKTtcbn1cblxuLyoqXG4gKiBHZW5lcmF0ZSB0aGUgY29tcGxldGUgT3BlbkFQSSAzLjEuMCBzcGVjaWZpY2F0aW9uIGRvY3VtZW50LlxuICpcbiAqIEBwYXJhbSBvcHRpb25zIC0gT3B0aW9uYWwgb3ZlcnJpZGVzIGZvciBzcGVjIG1ldGFkYXRhXG4gKiBAcmV0dXJucyBPcGVuQVBJIDMuMSBzcGVjaWZpY2F0aW9uIG9iamVjdFxuICovXG5leHBvcnQgZnVuY3Rpb24gZ2VuZXJhdGVPcGVuQXBpU3BlYyhcbiAgb3B0aW9ucz86IE9wZW5BcGlTcGVjT3B0aW9ucyxcbik6IFJldHVyblR5cGU8T3BlbkFwaUdlbmVyYXRvclYzMVsnZ2VuZXJhdGVEb2N1bWVudCddPiB7XG4gIGVuc3VyZUluaXRpYWxpemVkKCk7XG5cbiAgY29uc3QgZ2VuZXJhdG9yID0gbmV3IE9wZW5BcGlHZW5lcmF0b3JWMzEocmVnaXN0cnkuZGVmaW5pdGlvbnMpO1xuICByZXR1cm4gZ2VuZXJhdG9yLmdlbmVyYXRlRG9jdW1lbnQoe1xuICAgIG9wZW5hcGk6ICczLjEuMCcsXG4gICAgaW5mbzoge1xuICAgICAgdGl0bGU6IG9wdGlvbnM/LnRpdGxlID8/ICdQaXBlbGluZSBCdWlsZGVyIEFQSScsXG4gICAgICB2ZXJzaW9uOiBvcHRpb25zPy52ZXJzaW9uID8/ICcxLjAuMCcsXG4gICAgICBkZXNjcmlwdGlvbjogb3B0aW9ucz8uZGVzY3JpcHRpb24gPz8gJ0NJL0NEIFBpcGVsaW5lIEJ1aWxkZXIgUGxhdGZvcm0gQVBJIOKAlCBtYW5hZ2UgcGlwZWxpbmVzLCBwbHVnaW5zLCBtZXNzYWdlcywgcXVvdGFzLCBhbmQgYmlsbGluZy4nLFxuICAgIH0sXG4gICAgc2VydmVyczogW1xuICAgICAgeyB1cmw6IG9wdGlvbnM/LnNlcnZlclVybCA/PyAnL2FwaScsIGRlc2NyaXB0aW9uOiAnQVBJIGJhc2UgcGF0aCcgfSxcbiAgICBdLFxuICB9KTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function registerBillingRoutes(): void;
|
|
@@ -0,0 +1,69 @@
|
|
|
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.registerBillingRoutes = registerBillingRoutes;
|
|
6
|
+
const registry_1 = require("../registry");
|
|
7
|
+
const tags = ['Billing'];
|
|
8
|
+
const auth = [{ bearerAuth: [] }];
|
|
9
|
+
function registerBillingRoutes() {
|
|
10
|
+
registry_1.registry.registerPath({
|
|
11
|
+
method: 'get',
|
|
12
|
+
path: '/billing/plans',
|
|
13
|
+
summary: 'List active plans',
|
|
14
|
+
description: 'List all active billing plans. No authentication required.',
|
|
15
|
+
tags,
|
|
16
|
+
responses: { 200: { description: 'List of active billing plans' } },
|
|
17
|
+
});
|
|
18
|
+
registry_1.registry.registerPath({
|
|
19
|
+
method: 'get',
|
|
20
|
+
path: '/billing/plans/{planId}',
|
|
21
|
+
summary: 'Get plan by ID',
|
|
22
|
+
tags,
|
|
23
|
+
responses: { 200: { description: 'Plan details' }, 404: { description: 'Plan not found' } },
|
|
24
|
+
});
|
|
25
|
+
registry_1.registry.registerPath({
|
|
26
|
+
method: 'post',
|
|
27
|
+
path: '/billing/subscriptions',
|
|
28
|
+
summary: 'Create subscription',
|
|
29
|
+
description: 'Create a new billing subscription for the current organization.',
|
|
30
|
+
tags,
|
|
31
|
+
security: auth,
|
|
32
|
+
responses: { 201: { description: 'Subscription created' }, 400: { description: 'Validation error' } },
|
|
33
|
+
});
|
|
34
|
+
registry_1.registry.registerPath({
|
|
35
|
+
method: 'get',
|
|
36
|
+
path: '/billing/subscriptions',
|
|
37
|
+
summary: 'List subscriptions',
|
|
38
|
+
description: 'List billing subscriptions for the current organization.',
|
|
39
|
+
tags,
|
|
40
|
+
security: auth,
|
|
41
|
+
responses: { 200: { description: 'List of subscriptions' } },
|
|
42
|
+
});
|
|
43
|
+
registry_1.registry.registerPath({
|
|
44
|
+
method: 'put',
|
|
45
|
+
path: '/billing/subscriptions/{id}',
|
|
46
|
+
summary: 'Update subscription',
|
|
47
|
+
tags,
|
|
48
|
+
security: auth,
|
|
49
|
+
responses: { 200: { description: 'Subscription updated' }, 404: { description: 'Not found' } },
|
|
50
|
+
});
|
|
51
|
+
registry_1.registry.registerPath({
|
|
52
|
+
method: 'delete',
|
|
53
|
+
path: '/billing/subscriptions/{id}',
|
|
54
|
+
summary: 'Cancel subscription',
|
|
55
|
+
tags,
|
|
56
|
+
security: auth,
|
|
57
|
+
responses: { 200: { description: 'Subscription cancelled' }, 404: { description: 'Not found' } },
|
|
58
|
+
});
|
|
59
|
+
registry_1.registry.registerPath({
|
|
60
|
+
method: 'post',
|
|
61
|
+
path: '/billing/marketplace',
|
|
62
|
+
summary: 'AWS Marketplace integration',
|
|
63
|
+
description: 'Handle AWS Marketplace subscription events.',
|
|
64
|
+
tags,
|
|
65
|
+
security: auth,
|
|
66
|
+
responses: { 200: { description: 'Marketplace event processed' } },
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmlsbGluZy1yb3V0ZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvb3BlbmFwaS9yb3V0ZXMvYmlsbGluZy1yb3V0ZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLCtDQUErQztBQUMvQyxzQ0FBc0M7O0FBT3RDLHNEQWlFQztBQXRFRCwwQ0FBdUM7QUFFdkMsTUFBTSxJQUFJLEdBQUcsQ0FBQyxTQUFTLENBQUMsQ0FBQztBQUN6QixNQUFNLElBQUksR0FBRyxDQUFDLEVBQUUsVUFBVSxFQUFFLEVBQUUsRUFBRSxDQUFDLENBQUM7QUFFbEMsU0FBZ0IscUJBQXFCO0lBQ25DLG1CQUFRLENBQUMsWUFBWSxDQUFDO1FBQ3BCLE1BQU0sRUFBRSxLQUFLO1FBQ2IsSUFBSSxFQUFFLGdCQUFnQjtRQUN0QixPQUFPLEVBQUUsbUJBQW1CO1FBQzVCLFdBQVcsRUFBRSw0REFBNEQ7UUFDekUsSUFBSTtRQUNKLFNBQVMsRUFBRSxFQUFFLEdBQUcsRUFBRSxFQUFFLFdBQVcsRUFBRSw4QkFBOEIsRUFBRSxFQUFFO0tBQ3BFLENBQUMsQ0FBQztJQUVILG1CQUFRLENBQUMsWUFBWSxDQUFDO1FBQ3BCLE1BQU0sRUFBRSxLQUFLO1FBQ2IsSUFBSSxFQUFFLHlCQUF5QjtRQUMvQixPQUFPLEVBQUUsZ0JBQWdCO1FBQ3pCLElBQUk7UUFDSixTQUFTLEVBQUUsRUFBRSxHQUFHLEVBQUUsRUFBRSxXQUFXLEVBQUUsY0FBYyxFQUFFLEVBQUUsR0FBRyxFQUFFLEVBQUUsV0FBVyxFQUFFLGdCQUFnQixFQUFFLEVBQUU7S0FDNUYsQ0FBQyxDQUFDO0lBRUgsbUJBQVEsQ0FBQyxZQUFZLENBQUM7UUFDcEIsTUFBTSxFQUFFLE1BQU07UUFDZCxJQUFJLEVBQUUsd0JBQXdCO1FBQzlCLE9BQU8sRUFBRSxxQkFBcUI7UUFDOUIsV0FBVyxFQUFFLGlFQUFpRTtRQUM5RSxJQUFJO1FBQ0osUUFBUSxFQUFFLElBQUk7UUFDZCxTQUFTLEVBQUUsRUFBRSxHQUFHLEVBQUUsRUFBRSxXQUFXLEVBQUUsc0JBQXNCLEVBQUUsRUFBRSxHQUFHLEVBQUUsRUFBRSxXQUFXLEVBQUUsa0JBQWtCLEVBQUUsRUFBRTtLQUN0RyxDQUFDLENBQUM7SUFFSCxtQkFBUSxDQUFDLFlBQVksQ0FBQztRQUNwQixNQUFNLEVBQUUsS0FBSztRQUNiLElBQUksRUFBRSx3QkFBd0I7UUFDOUIsT0FBTyxFQUFFLG9CQUFvQjtRQUM3QixXQUFXLEVBQUUsMERBQTBEO1FBQ3ZFLElBQUk7UUFDSixRQUFRLEVBQUUsSUFBSTtRQUNkLFNBQVMsRUFBRSxFQUFFLEdBQUcsRUFBRSxFQUFFLFdBQVcsRUFBRSx1QkFBdUIsRUFBRSxFQUFFO0tBQzdELENBQUMsQ0FBQztJQUVILG1CQUFRLENBQUMsWUFBWSxDQUFDO1FBQ3BCLE1BQU0sRUFBRSxLQUFLO1FBQ2IsSUFBSSxFQUFFLDZCQUE2QjtRQUNuQyxPQUFPLEVBQUUscUJBQXFCO1FBQzlCLElBQUk7UUFDSixRQUFRLEVBQUUsSUFBSTtRQUNkLFNBQVMsRUFBRSxFQUFFLEdBQUcsRUFBRSxFQUFFLFdBQVcsRUFBRSxzQkFBc0IsRUFBRSxFQUFFLEdBQUcsRUFBRSxFQUFFLFdBQVcsRUFBRSxXQUFXLEVBQUUsRUFBRTtLQUMvRixDQUFDLENBQUM7SUFFSCxtQkFBUSxDQUFDLFlBQVksQ0FBQztRQUNwQixNQUFNLEVBQUUsUUFBUTtRQUNoQixJQUFJLEVBQUUsNkJBQTZCO1FBQ25DLE9BQU8sRUFBRSxxQkFBcUI7UUFDOUIsSUFBSTtRQUNKLFFBQVEsRUFBRSxJQUFJO1FBQ2QsU0FBUyxFQUFFLEVBQUUsR0FBRyxFQUFFLEVBQUUsV0FBVyxFQUFFLHdCQUF3QixFQUFFLEVBQUUsR0FBRyxFQUFFLEVBQUUsV0FBVyxFQUFFLFdBQVcsRUFBRSxFQUFFO0tBQ2pHLENBQUMsQ0FBQztJQUVILG1CQUFRLENBQUMsWUFBWSxDQUFDO1FBQ3BCLE1BQU0sRUFBRSxNQUFNO1FBQ2QsSUFBSSxFQUFFLHNCQUFzQjtRQUM1QixPQUFPLEVBQUUsNkJBQTZCO1FBQ3RDLFdBQVcsRUFBRSw2Q0FBNkM7UUFDMUQsSUFBSTtRQUNKLFFBQVEsRUFBRSxJQUFJO1FBQ2QsU0FBUyxFQUFFLEVBQUUsR0FBRyxFQUFFLEVBQUUsV0FBVyxFQUFFLDZCQUE2QixFQUFFLEVBQUU7S0FDbkUsQ0FBQyxDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG5pbXBvcnQgeyByZWdpc3RyeSB9IGZyb20gJy4uL3JlZ2lzdHJ5JztcblxuY29uc3QgdGFncyA9IFsnQmlsbGluZyddO1xuY29uc3QgYXV0aCA9IFt7IGJlYXJlckF1dGg6IFtdIH1dO1xuXG5leHBvcnQgZnVuY3Rpb24gcmVnaXN0ZXJCaWxsaW5nUm91dGVzKCk6IHZvaWQge1xuICByZWdpc3RyeS5yZWdpc3RlclBhdGgoe1xuICAgIG1ldGhvZDogJ2dldCcsXG4gICAgcGF0aDogJy9iaWxsaW5nL3BsYW5zJyxcbiAgICBzdW1tYXJ5OiAnTGlzdCBhY3RpdmUgcGxhbnMnLFxuICAgIGRlc2NyaXB0aW9uOiAnTGlzdCBhbGwgYWN0aXZlIGJpbGxpbmcgcGxhbnMuIE5vIGF1dGhlbnRpY2F0aW9uIHJlcXVpcmVkLicsXG4gICAgdGFncyxcbiAgICByZXNwb25zZXM6IHsgMjAwOiB7IGRlc2NyaXB0aW9uOiAnTGlzdCBvZiBhY3RpdmUgYmlsbGluZyBwbGFucycgfSB9LFxuICB9KTtcblxuICByZWdpc3RyeS5yZWdpc3RlclBhdGgoe1xuICAgIG1ldGhvZDogJ2dldCcsXG4gICAgcGF0aDogJy9iaWxsaW5nL3BsYW5zL3twbGFuSWR9JyxcbiAgICBzdW1tYXJ5OiAnR2V0IHBsYW4gYnkgSUQnLFxuICAgIHRhZ3MsXG4gICAgcmVzcG9uc2VzOiB7IDIwMDogeyBkZXNjcmlwdGlvbjogJ1BsYW4gZGV0YWlscycgfSwgNDA0OiB7IGRlc2NyaXB0aW9uOiAnUGxhbiBub3QgZm91bmQnIH0gfSxcbiAgfSk7XG5cbiAgcmVnaXN0cnkucmVnaXN0ZXJQYXRoKHtcbiAgICBtZXRob2Q6ICdwb3N0JyxcbiAgICBwYXRoOiAnL2JpbGxpbmcvc3Vic2NyaXB0aW9ucycsXG4gICAgc3VtbWFyeTogJ0NyZWF0ZSBzdWJzY3JpcHRpb24nLFxuICAgIGRlc2NyaXB0aW9uOiAnQ3JlYXRlIGEgbmV3IGJpbGxpbmcgc3Vic2NyaXB0aW9uIGZvciB0aGUgY3VycmVudCBvcmdhbml6YXRpb24uJyxcbiAgICB0YWdzLFxuICAgIHNlY3VyaXR5OiBhdXRoLFxuICAgIHJlc3BvbnNlczogeyAyMDE6IHsgZGVzY3JpcHRpb246ICdTdWJzY3JpcHRpb24gY3JlYXRlZCcgfSwgNDAwOiB7IGRlc2NyaXB0aW9uOiAnVmFsaWRhdGlvbiBlcnJvcicgfSB9LFxuICB9KTtcblxuICByZWdpc3RyeS5yZWdpc3RlclBhdGgoe1xuICAgIG1ldGhvZDogJ2dldCcsXG4gICAgcGF0aDogJy9iaWxsaW5nL3N1YnNjcmlwdGlvbnMnLFxuICAgIHN1bW1hcnk6ICdMaXN0IHN1YnNjcmlwdGlvbnMnLFxuICAgIGRlc2NyaXB0aW9uOiAnTGlzdCBiaWxsaW5nIHN1YnNjcmlwdGlvbnMgZm9yIHRoZSBjdXJyZW50IG9yZ2FuaXphdGlvbi4nLFxuICAgIHRhZ3MsXG4gICAgc2VjdXJpdHk6IGF1dGgsXG4gICAgcmVzcG9uc2VzOiB7IDIwMDogeyBkZXNjcmlwdGlvbjogJ0xpc3Qgb2Ygc3Vic2NyaXB0aW9ucycgfSB9LFxuICB9KTtcblxuICByZWdpc3RyeS5yZWdpc3RlclBhdGgoe1xuICAgIG1ldGhvZDogJ3B1dCcsXG4gICAgcGF0aDogJy9iaWxsaW5nL3N1YnNjcmlwdGlvbnMve2lkfScsXG4gICAgc3VtbWFyeTogJ1VwZGF0ZSBzdWJzY3JpcHRpb24nLFxuICAgIHRhZ3MsXG4gICAgc2VjdXJpdHk6IGF1dGgsXG4gICAgcmVzcG9uc2VzOiB7IDIwMDogeyBkZXNjcmlwdGlvbjogJ1N1YnNjcmlwdGlvbiB1cGRhdGVkJyB9LCA0MDQ6IHsgZGVzY3JpcHRpb246ICdOb3QgZm91bmQnIH0gfSxcbiAgfSk7XG5cbiAgcmVnaXN0cnkucmVnaXN0ZXJQYXRoKHtcbiAgICBtZXRob2Q6ICdkZWxldGUnLFxuICAgIHBhdGg6ICcvYmlsbGluZy9zdWJzY3JpcHRpb25zL3tpZH0nLFxuICAgIHN1bW1hcnk6ICdDYW5jZWwgc3Vic2NyaXB0aW9uJyxcbiAgICB0YWdzLFxuICAgIHNlY3VyaXR5OiBhdXRoLFxuICAgIHJlc3BvbnNlczogeyAyMDA6IHsgZGVzY3JpcHRpb246ICdTdWJzY3JpcHRpb24gY2FuY2VsbGVkJyB9LCA0MDQ6IHsgZGVzY3JpcHRpb246ICdOb3QgZm91bmQnIH0gfSxcbiAgfSk7XG5cbiAgcmVnaXN0cnkucmVnaXN0ZXJQYXRoKHtcbiAgICBtZXRob2Q6ICdwb3N0JyxcbiAgICBwYXRoOiAnL2JpbGxpbmcvbWFya2V0cGxhY2UnLFxuICAgIHN1bW1hcnk6ICdBV1MgTWFya2V0cGxhY2UgaW50ZWdyYXRpb24nLFxuICAgIGRlc2NyaXB0aW9uOiAnSGFuZGxlIEFXUyBNYXJrZXRwbGFjZSBzdWJzY3JpcHRpb24gZXZlbnRzLicsXG4gICAgdGFncyxcbiAgICBzZWN1cml0eTogYXV0aCxcbiAgICByZXNwb25zZXM6IHsgMjAwOiB7IGRlc2NyaXB0aW9uOiAnTWFya2V0cGxhY2UgZXZlbnQgcHJvY2Vzc2VkJyB9IH0sXG4gIH0pO1xufVxuIl19
|
|
@@ -0,0 +1,22 @@
|
|
|
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.registerAllRoutes = registerAllRoutes;
|
|
6
|
+
const billing_routes_1 = require("./billing-routes");
|
|
7
|
+
const message_routes_1 = require("./message-routes");
|
|
8
|
+
const pipeline_routes_1 = require("./pipeline-routes");
|
|
9
|
+
const plugin_routes_1 = require("./plugin-routes");
|
|
10
|
+
const quota_routes_1 = require("./quota-routes");
|
|
11
|
+
/**
|
|
12
|
+
* Register all OpenAPI route definitions with the shared registry.
|
|
13
|
+
* Call this once during spec generation initialization.
|
|
14
|
+
*/
|
|
15
|
+
function registerAllRoutes() {
|
|
16
|
+
(0, billing_routes_1.registerBillingRoutes)();
|
|
17
|
+
(0, message_routes_1.registerMessageRoutes)();
|
|
18
|
+
(0, pipeline_routes_1.registerPipelineRoutes)();
|
|
19
|
+
(0, plugin_routes_1.registerPluginRoutes)();
|
|
20
|
+
(0, quota_routes_1.registerQuotaRoutes)();
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvb3BlbmFwaS9yb3V0ZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLCtDQUErQztBQUMvQyxzQ0FBc0M7O0FBWXRDLDhDQU1DO0FBaEJELHFEQUF5RDtBQUN6RCxxREFBeUQ7QUFDekQsdURBQTJEO0FBQzNELG1EQUF1RDtBQUN2RCxpREFBcUQ7QUFFckQ7OztHQUdHO0FBQ0gsU0FBZ0IsaUJBQWlCO0lBQy9CLElBQUEsc0NBQXFCLEdBQUUsQ0FBQztJQUN4QixJQUFBLHNDQUFxQixHQUFFLENBQUM7SUFDeEIsSUFBQSx3Q0FBc0IsR0FBRSxDQUFDO0lBQ3pCLElBQUEsb0NBQW9CLEdBQUUsQ0FBQztJQUN2QixJQUFBLGtDQUFtQixHQUFFLENBQUM7QUFDeEIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCAyMDI2IFBpcGVsaW5lIEJ1aWxkZXIgQ29udHJpYnV0b3JzXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG5pbXBvcnQgeyByZWdpc3RlckJpbGxpbmdSb3V0ZXMgfSBmcm9tICcuL2JpbGxpbmctcm91dGVzJztcbmltcG9ydCB7IHJlZ2lzdGVyTWVzc2FnZVJvdXRlcyB9IGZyb20gJy4vbWVzc2FnZS1yb3V0ZXMnO1xuaW1wb3J0IHsgcmVnaXN0ZXJQaXBlbGluZVJvdXRlcyB9IGZyb20gJy4vcGlwZWxpbmUtcm91dGVzJztcbmltcG9ydCB7IHJlZ2lzdGVyUGx1Z2luUm91dGVzIH0gZnJvbSAnLi9wbHVnaW4tcm91dGVzJztcbmltcG9ydCB7IHJlZ2lzdGVyUXVvdGFSb3V0ZXMgfSBmcm9tICcuL3F1b3RhLXJvdXRlcyc7XG5cbi8qKlxuICogUmVnaXN0ZXIgYWxsIE9wZW5BUEkgcm91dGUgZGVmaW5pdGlvbnMgd2l0aCB0aGUgc2hhcmVkIHJlZ2lzdHJ5LlxuICogQ2FsbCB0aGlzIG9uY2UgZHVyaW5nIHNwZWMgZ2VuZXJhdGlvbiBpbml0aWFsaXphdGlvbi5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHJlZ2lzdGVyQWxsUm91dGVzKCk6IHZvaWQge1xuICByZWdpc3RlckJpbGxpbmdSb3V0ZXMoKTtcbiAgcmVnaXN0ZXJNZXNzYWdlUm91dGVzKCk7XG4gIHJlZ2lzdGVyUGlwZWxpbmVSb3V0ZXMoKTtcbiAgcmVnaXN0ZXJQbHVnaW5Sb3V0ZXMoKTtcbiAgcmVnaXN0ZXJRdW90YVJvdXRlcygpO1xufVxuIl19
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function registerMessageRoutes(): void;
|
|
@@ -0,0 +1,108 @@
|
|
|
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.registerMessageRoutes = registerMessageRoutes;
|
|
6
|
+
const registry_1 = require("../registry");
|
|
7
|
+
const tags = ['Messages'];
|
|
8
|
+
const auth = [{ bearerAuth: [] }];
|
|
9
|
+
function registerMessageRoutes() {
|
|
10
|
+
registry_1.registry.registerPath({
|
|
11
|
+
method: 'get',
|
|
12
|
+
path: '/messages',
|
|
13
|
+
summary: 'List inbox messages',
|
|
14
|
+
description: 'List root messages (inbox) with optional filtering by type and pagination.',
|
|
15
|
+
tags,
|
|
16
|
+
security: auth,
|
|
17
|
+
responses: { 200: { description: 'Paginated list of messages' } },
|
|
18
|
+
});
|
|
19
|
+
registry_1.registry.registerPath({
|
|
20
|
+
method: 'get',
|
|
21
|
+
path: '/messages/announcements',
|
|
22
|
+
summary: 'List announcements',
|
|
23
|
+
description: 'List announcement messages only.',
|
|
24
|
+
tags,
|
|
25
|
+
security: auth,
|
|
26
|
+
responses: { 200: { description: 'List of announcements' } },
|
|
27
|
+
});
|
|
28
|
+
registry_1.registry.registerPath({
|
|
29
|
+
method: 'get',
|
|
30
|
+
path: '/messages/conversations',
|
|
31
|
+
summary: 'List conversations',
|
|
32
|
+
description: 'List conversation messages only.',
|
|
33
|
+
tags,
|
|
34
|
+
security: auth,
|
|
35
|
+
responses: { 200: { description: 'List of conversations' } },
|
|
36
|
+
});
|
|
37
|
+
registry_1.registry.registerPath({
|
|
38
|
+
method: 'get',
|
|
39
|
+
path: '/messages/unread/count',
|
|
40
|
+
summary: 'Get unread count',
|
|
41
|
+
description: 'Get the number of unread messages for the current organization.',
|
|
42
|
+
tags,
|
|
43
|
+
security: auth,
|
|
44
|
+
responses: { 200: { description: 'Unread message count' } },
|
|
45
|
+
});
|
|
46
|
+
registry_1.registry.registerPath({
|
|
47
|
+
method: 'get',
|
|
48
|
+
path: '/messages/{id}',
|
|
49
|
+
summary: 'Get message by ID',
|
|
50
|
+
tags,
|
|
51
|
+
security: auth,
|
|
52
|
+
responses: { 200: { description: 'Message details' }, 404: { description: 'Not found' } },
|
|
53
|
+
});
|
|
54
|
+
registry_1.registry.registerPath({
|
|
55
|
+
method: 'get',
|
|
56
|
+
path: '/messages/{id}/thread',
|
|
57
|
+
summary: 'Get message thread',
|
|
58
|
+
description: 'Get all messages in a thread including the root message and replies.',
|
|
59
|
+
tags,
|
|
60
|
+
security: auth,
|
|
61
|
+
responses: { 200: { description: 'Thread messages sorted by date' } },
|
|
62
|
+
});
|
|
63
|
+
registry_1.registry.registerPath({
|
|
64
|
+
method: 'post',
|
|
65
|
+
path: '/messages',
|
|
66
|
+
summary: 'Create a message',
|
|
67
|
+
description: 'Create a new announcement or conversation message.',
|
|
68
|
+
tags,
|
|
69
|
+
security: auth,
|
|
70
|
+
responses: { 201: { description: 'Message created' }, 400: { description: 'Validation error' } },
|
|
71
|
+
});
|
|
72
|
+
registry_1.registry.registerPath({
|
|
73
|
+
method: 'post',
|
|
74
|
+
path: '/messages/{id}/reply',
|
|
75
|
+
summary: 'Reply to a message',
|
|
76
|
+
description: 'Create a reply in an existing message thread.',
|
|
77
|
+
tags,
|
|
78
|
+
security: auth,
|
|
79
|
+
responses: { 201: { description: 'Reply created' }, 404: { description: 'Parent message not found' } },
|
|
80
|
+
});
|
|
81
|
+
registry_1.registry.registerPath({
|
|
82
|
+
method: 'put',
|
|
83
|
+
path: '/messages/{id}/read',
|
|
84
|
+
summary: 'Mark message as read',
|
|
85
|
+
tags,
|
|
86
|
+
security: auth,
|
|
87
|
+
responses: { 200: { description: 'Message marked as read' } },
|
|
88
|
+
});
|
|
89
|
+
registry_1.registry.registerPath({
|
|
90
|
+
method: 'put',
|
|
91
|
+
path: '/messages/{id}/thread/read',
|
|
92
|
+
summary: 'Mark thread as read',
|
|
93
|
+
description: 'Mark all messages in a thread as read.',
|
|
94
|
+
tags,
|
|
95
|
+
security: auth,
|
|
96
|
+
responses: { 200: { description: 'Thread marked as read' } },
|
|
97
|
+
});
|
|
98
|
+
registry_1.registry.registerPath({
|
|
99
|
+
method: 'delete',
|
|
100
|
+
path: '/messages/{id}',
|
|
101
|
+
summary: 'Delete a message',
|
|
102
|
+
description: 'Soft-delete a message.',
|
|
103
|
+
tags,
|
|
104
|
+
security: auth,
|
|
105
|
+
responses: { 200: { description: 'Message deleted' }, 404: { description: 'Not found' } },
|
|
106
|
+
});
|
|
107
|
+
}
|
|
108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-routes.js","sourceRoot":"","sources":["../../../src/openapi/routes/message-routes.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { registry } from '../registry';\n\nconst tags = ['Messages'];\nconst auth = [{ bearerAuth: [] }];\n\nexport function registerMessageRoutes(): void {\n  registry.registerPath({\n    method: 'get',\n    path: '/messages',\n    summary: 'List inbox messages',\n    description: 'List root messages (inbox) with optional filtering by type and pagination.',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'Paginated list of messages' } },\n  });\n\n  registry.registerPath({\n    method: 'get',\n    path: '/messages/announcements',\n    summary: 'List announcements',\n    description: 'List announcement messages only.',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'List of announcements' } },\n  });\n\n  registry.registerPath({\n    method: 'get',\n    path: '/messages/conversations',\n    summary: 'List conversations',\n    description: 'List conversation messages only.',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'List of conversations' } },\n  });\n\n  registry.registerPath({\n    method: 'get',\n    path: '/messages/unread/count',\n    summary: 'Get unread count',\n    description: 'Get the number of unread messages for the current organization.',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'Unread message count' } },\n  });\n\n  registry.registerPath({\n    method: 'get',\n    path: '/messages/{id}',\n    summary: 'Get message by ID',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'Message details' }, 404: { description: 'Not found' } },\n  });\n\n  registry.registerPath({\n    method: 'get',\n    path: '/messages/{id}/thread',\n    summary: 'Get message thread',\n    description: 'Get all messages in a thread including the root message and replies.',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'Thread messages sorted by date' } },\n  });\n\n  registry.registerPath({\n    method: 'post',\n    path: '/messages',\n    summary: 'Create a message',\n    description: 'Create a new announcement or conversation message.',\n    tags,\n    security: auth,\n    responses: { 201: { description: 'Message created' }, 400: { description: 'Validation error' } },\n  });\n\n  registry.registerPath({\n    method: 'post',\n    path: '/messages/{id}/reply',\n    summary: 'Reply to a message',\n    description: 'Create a reply in an existing message thread.',\n    tags,\n    security: auth,\n    responses: { 201: { description: 'Reply created' }, 404: { description: 'Parent message not found' } },\n  });\n\n  registry.registerPath({\n    method: 'put',\n    path: '/messages/{id}/read',\n    summary: 'Mark message as read',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'Message marked as read' } },\n  });\n\n  registry.registerPath({\n    method: 'put',\n    path: '/messages/{id}/thread/read',\n    summary: 'Mark thread as read',\n    description: 'Mark all messages in a thread as read.',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'Thread marked as read' } },\n  });\n\n  registry.registerPath({\n    method: 'delete',\n    path: '/messages/{id}',\n    summary: 'Delete a message',\n    description: 'Soft-delete a message.',\n    tags,\n    security: auth,\n    responses: { 200: { description: 'Message deleted' }, 404: { description: 'Not found' } },\n  });\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export declare function registerPipelineRoutes(): void;
|