@pipeline-builder/pipeline-manager 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 +74 -0
- package/cdk.json +91 -0
- package/config.yml +94 -0
- package/dist/boilerplate.d.ts +3 -0
- package/dist/boilerplate.d.ts.map +1 -0
- package/dist/boilerplate.js +58 -0
- package/dist/cdk.json +91 -0
- package/dist/cli.d.ts +62 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +372 -0
- package/dist/commands/bootstrap.d.ts +11 -0
- package/dist/commands/bootstrap.d.ts.map +1 -0
- package/dist/commands/bootstrap.js +159 -0
- package/dist/commands/create-pipeline.d.ts +12 -0
- package/dist/commands/create-pipeline.d.ts.map +1 -0
- package/dist/commands/create-pipeline.js +291 -0
- package/dist/commands/deploy.d.ts +15 -0
- package/dist/commands/deploy.d.ts.map +1 -0
- package/dist/commands/deploy.js +167 -0
- package/dist/commands/get-pipeline.d.ts +13 -0
- package/dist/commands/get-pipeline.d.ts.map +1 -0
- package/dist/commands/get-pipeline.js +97 -0
- package/dist/commands/get-plugin.d.ts +13 -0
- package/dist/commands/get-plugin.d.ts.map +1 -0
- package/dist/commands/get-plugin.js +98 -0
- package/dist/commands/list-pipelines.d.ts +20 -0
- package/dist/commands/list-pipelines.d.ts.map +1 -0
- package/dist/commands/list-pipelines.js +172 -0
- package/dist/commands/list-plugins.d.ts +20 -0
- package/dist/commands/list-plugins.d.ts.map +1 -0
- package/dist/commands/list-plugins.js +167 -0
- package/dist/commands/login.d.ts +21 -0
- package/dist/commands/login.d.ts.map +1 -0
- package/dist/commands/login.js +179 -0
- package/dist/commands/setup-events.d.ts +15 -0
- package/dist/commands/setup-events.d.ts.map +1 -0
- package/dist/commands/setup-events.js +177 -0
- package/dist/commands/status.d.ts +11 -0
- package/dist/commands/status.d.ts.map +1 -0
- package/dist/commands/status.js +89 -0
- package/dist/commands/store-token.d.ts +20 -0
- package/dist/commands/store-token.d.ts.map +1 -0
- package/dist/commands/store-token.js +233 -0
- package/dist/commands/synth.d.ts +21 -0
- package/dist/commands/synth.d.ts.map +1 -0
- package/dist/commands/synth.js +143 -0
- package/dist/commands/upload-plugin.d.ts +21 -0
- package/dist/commands/upload-plugin.d.ts.map +1 -0
- package/dist/commands/upload-plugin.js +311 -0
- package/dist/commands/version.d.ts +12 -0
- package/dist/commands/version.d.ts.map +1 -0
- package/dist/commands/version.js +223 -0
- package/dist/config/cli.constants.d.ts +101 -0
- package/dist/config/cli.constants.d.ts.map +1 -0
- package/dist/config/cli.constants.js +165 -0
- package/dist/config.yml +94 -0
- package/dist/templates/events-stack.json +141 -0
- package/dist/types/config.d.ts +44 -0
- package/dist/types/config.d.ts.map +1 -0
- package/dist/types/config.js +5 -0
- package/dist/types/error.d.ts +61 -0
- package/dist/types/error.d.ts.map +1 -0
- package/dist/types/error.js +39 -0
- package/dist/types/index.d.ts +8 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +26 -0
- package/dist/types/pipeline.d.ts +144 -0
- package/dist/types/pipeline.d.ts.map +1 -0
- package/dist/types/pipeline.js +5 -0
- package/dist/types/plugin.d.ts +160 -0
- package/dist/types/plugin.d.ts.map +1 -0
- package/dist/types/plugin.js +5 -0
- package/dist/utils/api-client.d.ts +26 -0
- package/dist/utils/api-client.d.ts.map +1 -0
- package/dist/utils/api-client.js +160 -0
- package/dist/utils/audit-log.d.ts +8 -0
- package/dist/utils/audit-log.d.ts.map +1 -0
- package/dist/utils/audit-log.js +53 -0
- package/dist/utils/auth-guard.d.ts +16 -0
- package/dist/utils/auth-guard.d.ts.map +1 -0
- package/dist/utils/auth-guard.js +25 -0
- package/dist/utils/aws-secrets.d.ts +21 -0
- package/dist/utils/aws-secrets.d.ts.map +1 -0
- package/dist/utils/aws-secrets.js +74 -0
- package/dist/utils/banner.d.ts +19 -0
- package/dist/utils/banner.d.ts.map +1 -0
- package/dist/utils/banner.js +59 -0
- package/dist/utils/cdk-utils.d.ts +51 -0
- package/dist/utils/cdk-utils.d.ts.map +1 -0
- package/dist/utils/cdk-utils.js +101 -0
- package/dist/utils/command-utils.d.ts +56 -0
- package/dist/utils/command-utils.d.ts.map +1 -0
- package/dist/utils/command-utils.js +138 -0
- package/dist/utils/config-loader.d.ts +27 -0
- package/dist/utils/config-loader.d.ts.map +1 -0
- package/dist/utils/config-loader.js +166 -0
- package/dist/utils/error-handler.d.ts +29 -0
- package/dist/utils/error-handler.d.ts.map +1 -0
- package/dist/utils/error-handler.js +255 -0
- package/dist/utils/list-command-utils.d.ts +23 -0
- package/dist/utils/list-command-utils.d.ts.map +1 -0
- package/dist/utils/list-command-utils.js +60 -0
- package/dist/utils/output-utils.d.ts +60 -0
- package/dist/utils/output-utils.d.ts.map +1 -0
- package/dist/utils/output-utils.js +320 -0
- package/dist/utils/rate-limiter.d.ts +14 -0
- package/dist/utils/rate-limiter.d.ts.map +1 -0
- package/dist/utils/rate-limiter.js +73 -0
- package/package.json +144 -0
|
@@ -0,0 +1,179 @@
|
|
|
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.login = login;
|
|
9
|
+
const https_1 = __importDefault(require("https"));
|
|
10
|
+
const axios_1 = __importDefault(require("axios"));
|
|
11
|
+
const picocolors_1 = __importDefault(require("picocolors"));
|
|
12
|
+
const cli_constants_1 = require("../config/cli.constants");
|
|
13
|
+
const error_handler_1 = require("../utils/error-handler");
|
|
14
|
+
const output_utils_1 = require("../utils/output-utils");
|
|
15
|
+
const rate_limiter_1 = require("../utils/rate-limiter");
|
|
16
|
+
const { bold, cyan, green, magenta } = picocolors_1.default;
|
|
17
|
+
/**
|
|
18
|
+
* Registers the `login` command with the CLI program.
|
|
19
|
+
*
|
|
20
|
+
* Authenticates against the platform API and prints an export statement
|
|
21
|
+
* for `PLATFORM_TOKEN` on success. Does NOT require `PLATFORM_TOKEN`
|
|
22
|
+
* to be set beforehand (unlike other commands).
|
|
23
|
+
*
|
|
24
|
+
* @param program - The root Commander program instance to attach the command to.
|
|
25
|
+
*
|
|
26
|
+
* @example
|
|
27
|
+
* ```bash
|
|
28
|
+
* pipeline-manager login --identifier admin@example.com --password secret
|
|
29
|
+
* pipeline-manager login -u admin@example.com -p secret
|
|
30
|
+
* pipeline-manager login -u admin@example.com -p secret --url https://myhost:8443
|
|
31
|
+
* eval $(pipeline-manager login -u admin@example.com -p secret --quiet)
|
|
32
|
+
* pipeline-manager login --refresh <refresh-token>
|
|
33
|
+
* ```
|
|
34
|
+
*/
|
|
35
|
+
function login(program) {
|
|
36
|
+
program
|
|
37
|
+
.command('login')
|
|
38
|
+
.description('Authenticate with the platform and obtain a PLATFORM_TOKEN')
|
|
39
|
+
.option('-u, --identifier <identifier>', 'Username or email')
|
|
40
|
+
.option('-p, --password <password>', 'Password')
|
|
41
|
+
.option('--refresh <refreshToken>', 'Use a refresh token instead of login credentials')
|
|
42
|
+
.option('--org <orgId>', 'Switch to a specific organization after login')
|
|
43
|
+
.option('--url <url>', 'Platform base URL', process.env.PLATFORM_BASE_URL || 'https://localhost:8443')
|
|
44
|
+
.option('--verify-ssl', 'Enable SSL certificate verification')
|
|
45
|
+
.option('--no-verify-ssl', 'Disable SSL certificate verification')
|
|
46
|
+
.option('--quiet', 'Only print the export statement (useful for eval)')
|
|
47
|
+
.action(async (options) => {
|
|
48
|
+
const executionId = (0, cli_constants_1.generateExecutionId)();
|
|
49
|
+
const isRefresh = !!options.refresh;
|
|
50
|
+
try {
|
|
51
|
+
// Validate required options
|
|
52
|
+
if (!isRefresh && (!options.identifier || !options.password)) {
|
|
53
|
+
(0, output_utils_1.printError)('Login requires --identifier and --password, or --refresh <token>');
|
|
54
|
+
process.exit(error_handler_1.ERROR_CODES.AUTHENTICATION);
|
|
55
|
+
}
|
|
56
|
+
// Rate limiting — prevent brute force (login only)
|
|
57
|
+
if (!isRefresh) {
|
|
58
|
+
const rateLimitMsg = (0, rate_limiter_1.checkAuthRateLimit)();
|
|
59
|
+
if (rateLimitMsg) {
|
|
60
|
+
(0, output_utils_1.printError)(rateLimitMsg);
|
|
61
|
+
process.exit(error_handler_1.ERROR_CODES.AUTHENTICATION);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
const quiet = options.quiet ?? false;
|
|
65
|
+
if (!quiet) {
|
|
66
|
+
(0, output_utils_1.printSection)(isRefresh ? 'Token Refresh' : 'Login');
|
|
67
|
+
console.log(`${magenta(`[EXE-${executionId}]`)} ${cyan(bold(isRefresh ? 'Token Refresh' : 'Platform Authentication'))}`);
|
|
68
|
+
console.log('');
|
|
69
|
+
(0, output_utils_1.printInfo)(isRefresh ? 'Refreshing access token' : 'Authenticating', {
|
|
70
|
+
...(options.identifier ? { identifier: options.identifier } : {}),
|
|
71
|
+
url: options.url,
|
|
72
|
+
verifySsl: options.verifySsl,
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
const httpsAgent = new https_1.default.Agent({
|
|
76
|
+
rejectUnauthorized: options.verifySsl ?? true,
|
|
77
|
+
});
|
|
78
|
+
let token;
|
|
79
|
+
if (isRefresh) {
|
|
80
|
+
const refreshUrl = `${options.url}/api/auth/refresh`;
|
|
81
|
+
(0, output_utils_1.printDebug)('POST', { url: refreshUrl });
|
|
82
|
+
const response = await axios_1.default.post(refreshUrl, { refreshToken: options.refresh }, {
|
|
83
|
+
headers: { 'Content-Type': 'application/json' },
|
|
84
|
+
httpsAgent,
|
|
85
|
+
timeout: cli_constants_1.TIMEOUTS.HTTP_REQUEST,
|
|
86
|
+
});
|
|
87
|
+
token = response.data?.data?.accessToken;
|
|
88
|
+
}
|
|
89
|
+
else {
|
|
90
|
+
const loginUrl = `${options.url}/api/auth/login`;
|
|
91
|
+
(0, output_utils_1.printDebug)('POST', { url: loginUrl });
|
|
92
|
+
const response = await axios_1.default.post(loginUrl, {
|
|
93
|
+
identifier: options.identifier,
|
|
94
|
+
password: options.password,
|
|
95
|
+
}, {
|
|
96
|
+
headers: { 'Content-Type': 'application/json' },
|
|
97
|
+
httpsAgent,
|
|
98
|
+
timeout: cli_constants_1.TIMEOUTS.HTTP_REQUEST,
|
|
99
|
+
});
|
|
100
|
+
token = response.data?.data?.accessToken;
|
|
101
|
+
}
|
|
102
|
+
if (!token) {
|
|
103
|
+
if (!isRefresh)
|
|
104
|
+
(0, rate_limiter_1.recordAuthFailure)();
|
|
105
|
+
(0, output_utils_1.printError)(`${isRefresh ? 'Token refresh' : 'Login'} failed: no access token in response`);
|
|
106
|
+
process.exit(error_handler_1.ERROR_CODES.AUTHENTICATION);
|
|
107
|
+
}
|
|
108
|
+
if (!isRefresh)
|
|
109
|
+
(0, rate_limiter_1.recordAuthSuccess)();
|
|
110
|
+
// Switch to a specific organization if --org is provided
|
|
111
|
+
if (options.org) {
|
|
112
|
+
if (!quiet) {
|
|
113
|
+
(0, output_utils_1.printInfo)('Switching to organization', { orgId: options.org });
|
|
114
|
+
}
|
|
115
|
+
const switchUrl = `${options.url}/api/auth/switch-org`;
|
|
116
|
+
(0, output_utils_1.printDebug)('POST', { url: switchUrl });
|
|
117
|
+
const switchResponse = await axios_1.default.post(switchUrl, { organizationId: options.org }, {
|
|
118
|
+
headers: {
|
|
119
|
+
'Content-Type': 'application/json',
|
|
120
|
+
'Authorization': `Bearer ${token}`,
|
|
121
|
+
},
|
|
122
|
+
httpsAgent,
|
|
123
|
+
timeout: cli_constants_1.TIMEOUTS.HTTP_REQUEST,
|
|
124
|
+
});
|
|
125
|
+
const switchedToken = switchResponse.data?.data?.accessToken;
|
|
126
|
+
if (!switchedToken) {
|
|
127
|
+
(0, output_utils_1.printError)('Organization switch failed: no access token in response');
|
|
128
|
+
process.exit(error_handler_1.ERROR_CODES.AUTHENTICATION);
|
|
129
|
+
}
|
|
130
|
+
token = switchedToken;
|
|
131
|
+
if (!quiet) {
|
|
132
|
+
(0, output_utils_1.printSuccess)(`Switched to organization ${options.org}`);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
if (!quiet) {
|
|
136
|
+
console.log('');
|
|
137
|
+
(0, output_utils_1.printSuccess)(isRefresh ? 'Token refreshed successfully' : 'Login successful');
|
|
138
|
+
console.log('');
|
|
139
|
+
}
|
|
140
|
+
console.log(`export PLATFORM_TOKEN=${token}`);
|
|
141
|
+
if (!quiet) {
|
|
142
|
+
console.log('');
|
|
143
|
+
(0, output_utils_1.printInfo)('Tip: Run the following to set the token in your shell:');
|
|
144
|
+
const orgFlag = options.org ? ` --org ${options.org}` : '';
|
|
145
|
+
if (isRefresh) {
|
|
146
|
+
console.log(green(` eval $(pipeline-manager login --refresh '<refresh-token>'${orgFlag} --quiet)`));
|
|
147
|
+
}
|
|
148
|
+
else {
|
|
149
|
+
console.log(green(` eval $(pipeline-manager login -u ${options.identifier} -p '***'${orgFlag} --quiet)`));
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
catch (error) {
|
|
154
|
+
if (!isRefresh)
|
|
155
|
+
(0, rate_limiter_1.recordAuthFailure)();
|
|
156
|
+
// Provide a clear failure message for auth errors
|
|
157
|
+
if (axios_1.default.isAxiosError(error)) {
|
|
158
|
+
const status = error.response?.status;
|
|
159
|
+
const message = error.response?.data?.message;
|
|
160
|
+
(0, output_utils_1.printError)(`${isRefresh ? 'Token refresh' : 'Login'} failed`, {
|
|
161
|
+
status: status ?? 'no response',
|
|
162
|
+
...(message ? { message } : {}),
|
|
163
|
+
});
|
|
164
|
+
process.exit(error_handler_1.ERROR_CODES.AUTHENTICATION);
|
|
165
|
+
}
|
|
166
|
+
(0, error_handler_1.handleError)(error, error_handler_1.ERROR_CODES.AUTHENTICATION, {
|
|
167
|
+
debug: program.opts().debug,
|
|
168
|
+
exit: true,
|
|
169
|
+
context: {
|
|
170
|
+
command: 'login',
|
|
171
|
+
executionId,
|
|
172
|
+
identifier: options.identifier,
|
|
173
|
+
url: options.url,
|
|
174
|
+
},
|
|
175
|
+
});
|
|
176
|
+
}
|
|
177
|
+
});
|
|
178
|
+
}
|
|
179
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"login.js","sourceRoot":"","sources":["../../src/commands/login.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport https from 'https';\nimport axios from 'axios';\nimport { Command } from 'commander';\nimport pico from 'picocolors';\nimport { generateExecutionId, TIMEOUTS } from '../config/cli.constants';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printDebug, printError, printInfo, printSection, printSuccess } from '../utils/output-utils';\nimport { checkAuthRateLimit, recordAuthFailure, recordAuthSuccess } from '../utils/rate-limiter';\n\nconst { bold, cyan, green, magenta } = pico;\n\n/**\n * Expected shape of the login/refresh API response.\n */\ninterface LoginResponse {\n  success: boolean;\n  data: {\n    accessToken: string;\n    refreshToken: string;\n  };\n}\n\n/**\n * Registers the `login` command with the CLI program.\n *\n * Authenticates against the platform API and prints an export statement\n * for `PLATFORM_TOKEN` on success.  Does NOT require `PLATFORM_TOKEN`\n * to be set beforehand (unlike other commands).\n *\n * @param program - The root Commander program instance to attach the command to.\n *\n * @example\n * ```bash\n * pipeline-manager login --identifier admin@example.com --password secret\n * pipeline-manager login -u admin@example.com -p secret\n * pipeline-manager login -u admin@example.com -p secret --url https://myhost:8443\n * eval $(pipeline-manager login -u admin@example.com -p secret --quiet)\n * pipeline-manager login --refresh <refresh-token>\n * ```\n */\nexport function login(program: Command): void {\n  program\n    .command('login')\n    .description('Authenticate with the platform and obtain a PLATFORM_TOKEN')\n    .option('-u, --identifier <identifier>', 'Username or email')\n    .option('-p, --password <password>', 'Password')\n    .option('--refresh <refreshToken>', 'Use a refresh token instead of login credentials')\n    .option('--org <orgId>', 'Switch to a specific organization after login')\n    .option('--url <url>', 'Platform base URL', process.env.PLATFORM_BASE_URL || 'https://localhost:8443')\n    .option('--verify-ssl', 'Enable SSL certificate verification')\n    .option('--no-verify-ssl', 'Disable SSL certificate verification')\n    .option('--quiet', 'Only print the export statement (useful for eval)')\n    .action(async (options) => {\n      const executionId = generateExecutionId();\n      const isRefresh = !!options.refresh;\n\n      try {\n\n        // Validate required options\n        if (!isRefresh && (!options.identifier || !options.password)) {\n          printError('Login requires --identifier and --password, or --refresh <token>');\n          process.exit(ERROR_CODES.AUTHENTICATION);\n        }\n\n        // Rate limiting — prevent brute force (login only)\n        if (!isRefresh) {\n          const rateLimitMsg = checkAuthRateLimit();\n          if (rateLimitMsg) {\n            printError(rateLimitMsg);\n            process.exit(ERROR_CODES.AUTHENTICATION);\n          }\n        }\n\n        const quiet = options.quiet ?? false;\n\n        if (!quiet) {\n          printSection(isRefresh ? 'Token Refresh' : 'Login');\n          console.log(`${magenta(`[EXE-${executionId}]`)} ${cyan(bold(isRefresh ? 'Token Refresh' : 'Platform Authentication'))}`);\n          console.log('');\n          printInfo(isRefresh ? 'Refreshing access token' : 'Authenticating', {\n            ...(options.identifier ? { identifier: options.identifier } : {}),\n            url: options.url,\n            verifySsl: options.verifySsl,\n          });\n        }\n\n        const httpsAgent = new https.Agent({\n          rejectUnauthorized: options.verifySsl ?? true,\n        });\n\n        let token: string | undefined;\n\n        if (isRefresh) {\n          const refreshUrl = `${options.url}/api/auth/refresh`;\n          printDebug('POST', { url: refreshUrl });\n\n          const response = await axios.post<LoginResponse>(\n            refreshUrl,\n            { refreshToken: options.refresh },\n            {\n              headers: { 'Content-Type': 'application/json' },\n              httpsAgent,\n              timeout: TIMEOUTS.HTTP_REQUEST,\n            },\n          );\n\n          token = response.data?.data?.accessToken;\n        } else {\n          const loginUrl = `${options.url}/api/auth/login`;\n          printDebug('POST', { url: loginUrl });\n\n          const response = await axios.post<LoginResponse>(\n            loginUrl,\n            {\n              identifier: options.identifier,\n              password: options.password,\n            },\n            {\n              headers: { 'Content-Type': 'application/json' },\n              httpsAgent,\n              timeout: TIMEOUTS.HTTP_REQUEST,\n            },\n          );\n\n          token = response.data?.data?.accessToken;\n        }\n\n        if (!token) {\n          if (!isRefresh) recordAuthFailure();\n          printError(`${isRefresh ? 'Token refresh' : 'Login'} failed: no access token in response`);\n          process.exit(ERROR_CODES.AUTHENTICATION);\n        }\n\n        if (!isRefresh) recordAuthSuccess();\n\n        // Switch to a specific organization if --org is provided\n        if (options.org) {\n          if (!quiet) {\n            printInfo('Switching to organization', { orgId: options.org });\n          }\n\n          const switchUrl = `${options.url}/api/auth/switch-org`;\n          printDebug('POST', { url: switchUrl });\n\n          const switchResponse = await axios.post<LoginResponse>(\n            switchUrl,\n            { organizationId: options.org },\n            {\n              headers: {\n                'Content-Type': 'application/json',\n                'Authorization': `Bearer ${token}`,\n              },\n              httpsAgent,\n              timeout: TIMEOUTS.HTTP_REQUEST,\n            },\n          );\n\n          const switchedToken = switchResponse.data?.data?.accessToken;\n          if (!switchedToken) {\n            printError('Organization switch failed: no access token in response');\n            process.exit(ERROR_CODES.AUTHENTICATION);\n          }\n\n          token = switchedToken;\n\n          if (!quiet) {\n            printSuccess(`Switched to organization ${options.org}`);\n          }\n        }\n\n        if (!quiet) {\n          console.log('');\n          printSuccess(isRefresh ? 'Token refreshed successfully' : 'Login successful');\n          console.log('');\n        }\n\n        console.log(`export PLATFORM_TOKEN=${token}`);\n\n        if (!quiet) {\n          console.log('');\n          printInfo('Tip: Run the following to set the token in your shell:');\n          const orgFlag = options.org ? ` --org ${options.org}` : '';\n          if (isRefresh) {\n            console.log(green(`  eval $(pipeline-manager login --refresh '<refresh-token>'${orgFlag} --quiet)`));\n          } else {\n            console.log(green(`  eval $(pipeline-manager login -u ${options.identifier} -p '***'${orgFlag} --quiet)`));\n          }\n        }\n      } catch (error) {\n        if (!isRefresh) recordAuthFailure();\n        // Provide a clear failure message for auth errors\n        if (axios.isAxiosError(error)) {\n          const status = error.response?.status;\n          const message = (error.response?.data as { message?: string })?.message;\n\n          printError(`${isRefresh ? 'Token refresh' : 'Login'} failed`, {\n            status: status ?? 'no response',\n            ...(message ? { message } : {}),\n          });\n          process.exit(ERROR_CODES.AUTHENTICATION);\n        }\n\n        handleError(error, ERROR_CODES.AUTHENTICATION, {\n          debug: program.opts().debug,\n          exit: true,\n          context: {\n            command: 'login',\n            executionId,\n            identifier: options.identifier,\n            url: options.url,\n          },\n        });\n      }\n    });\n}\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { Command } from 'commander';
|
|
2
|
+
/**
|
|
3
|
+
* Registers the `setup-events` command with the CLI program.
|
|
4
|
+
*
|
|
5
|
+
* Deploys EventBridge → SQS → Lambda infrastructure for pipeline reporting.
|
|
6
|
+
*
|
|
7
|
+
* 1. Deploys CloudFormation stack (rule + queue + Lambda shell + IAM)
|
|
8
|
+
* 2. Downloads @pipeline-builder/event-bridge from npm registry
|
|
9
|
+
* 3. Uploads handler code directly to Lambda (no S3 needed)
|
|
10
|
+
*
|
|
11
|
+
* Uses PLATFORM_BASE_URL from environment (same as all other commands).
|
|
12
|
+
* Idempotent — running again updates the stack and code.
|
|
13
|
+
*/
|
|
14
|
+
export declare function setupEvents(program: Command): void;
|
|
15
|
+
//# sourceMappingURL=setup-events.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"setup-events.d.ts","sourceRoot":"","sources":["../../src/commands/setup-events.ts"],"names":[],"mappings":"AASA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAUpC;;;;;;;;;;;GAWG;AACH,wBAAgB,WAAW,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAmIlD"}
|
|
@@ -0,0 +1,177 @@
|
|
|
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 __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
16
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
17
|
+
}) : function(o, v) {
|
|
18
|
+
o["default"] = v;
|
|
19
|
+
});
|
|
20
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
21
|
+
var ownKeys = function(o) {
|
|
22
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
23
|
+
var ar = [];
|
|
24
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
25
|
+
return ar;
|
|
26
|
+
};
|
|
27
|
+
return ownKeys(o);
|
|
28
|
+
};
|
|
29
|
+
return function (mod) {
|
|
30
|
+
if (mod && mod.__esModule) return mod;
|
|
31
|
+
var result = {};
|
|
32
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
33
|
+
__setModuleDefault(result, mod);
|
|
34
|
+
return result;
|
|
35
|
+
};
|
|
36
|
+
})();
|
|
37
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
38
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
39
|
+
};
|
|
40
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
41
|
+
exports.setupEvents = setupEvents;
|
|
42
|
+
const child_process_1 = require("child_process");
|
|
43
|
+
const fs = __importStar(require("fs"));
|
|
44
|
+
const os = __importStar(require("os"));
|
|
45
|
+
const path_1 = __importDefault(require("path"));
|
|
46
|
+
const client_cloudformation_1 = require("@aws-sdk/client-cloudformation");
|
|
47
|
+
const client_lambda_1 = require("@aws-sdk/client-lambda");
|
|
48
|
+
const command_utils_1 = require("../utils/command-utils");
|
|
49
|
+
const error_handler_1 = require("../utils/error-handler");
|
|
50
|
+
const output_utils_1 = require("../utils/output-utils");
|
|
51
|
+
const STACK_NAME = 'pipeline-builder-events';
|
|
52
|
+
// Existing CFN-managed Lambda function name — do NOT rename without a migration plan
|
|
53
|
+
const LAMBDA_NAME = 'pipeline-builder-event-ingestion';
|
|
54
|
+
const PACKAGE_NAME = '@pipeline-builder/event-bridge';
|
|
55
|
+
/**
|
|
56
|
+
* Registers the `setup-events` command with the CLI program.
|
|
57
|
+
*
|
|
58
|
+
* Deploys EventBridge → SQS → Lambda infrastructure for pipeline reporting.
|
|
59
|
+
*
|
|
60
|
+
* 1. Deploys CloudFormation stack (rule + queue + Lambda shell + IAM)
|
|
61
|
+
* 2. Downloads @pipeline-builder/event-bridge from npm registry
|
|
62
|
+
* 3. Uploads handler code directly to Lambda (no S3 needed)
|
|
63
|
+
*
|
|
64
|
+
* Uses PLATFORM_BASE_URL from environment (same as all other commands).
|
|
65
|
+
* Idempotent — running again updates the stack and code.
|
|
66
|
+
*/
|
|
67
|
+
function setupEvents(program) {
|
|
68
|
+
program
|
|
69
|
+
.command('setup-events')
|
|
70
|
+
.description('Deploy EventBridge event ingestion infrastructure for pipeline reporting')
|
|
71
|
+
.option('--package-version <version>', 'event-bridge package version (default: latest)')
|
|
72
|
+
.option('--secret-name <name>', 'Platform secret name (e.g. pipeline-builder/{orgId}/platform)')
|
|
73
|
+
.option('--region <region>', 'AWS region')
|
|
74
|
+
.option('--profile <profile>', 'AWS CLI profile', 'default')
|
|
75
|
+
.action(async (options) => {
|
|
76
|
+
const executionId = (0, command_utils_1.printCommandHeader)('Setup Event Ingestion');
|
|
77
|
+
try {
|
|
78
|
+
const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION;
|
|
79
|
+
if (!region) {
|
|
80
|
+
(0, output_utils_1.printError)('AWS region is required');
|
|
81
|
+
throw new Error('AWS region not provided');
|
|
82
|
+
}
|
|
83
|
+
const platformUrl = process.env.PLATFORM_BASE_URL;
|
|
84
|
+
if (!platformUrl) {
|
|
85
|
+
(0, output_utils_1.printError)('PLATFORM_BASE_URL environment variable is required');
|
|
86
|
+
throw new Error('PLATFORM_BASE_URL not set');
|
|
87
|
+
}
|
|
88
|
+
const secretName = options.secretName || process.env.PLATFORM_SECRET_NAME;
|
|
89
|
+
if (!secretName) {
|
|
90
|
+
(0, output_utils_1.printError)('--secret-name or PLATFORM_SECRET_NAME env var is required');
|
|
91
|
+
throw new Error('Platform secret name not provided');
|
|
92
|
+
}
|
|
93
|
+
(0, output_utils_1.printInfo)('Parameters', {
|
|
94
|
+
stack: STACK_NAME,
|
|
95
|
+
region,
|
|
96
|
+
platformUrl,
|
|
97
|
+
secretName,
|
|
98
|
+
packageVersion: options.packageVersion || 'latest',
|
|
99
|
+
});
|
|
100
|
+
// Step 1: Deploy CloudFormation (infra only — Lambda gets placeholder code)
|
|
101
|
+
(0, output_utils_1.printSection)('Deploy Infrastructure');
|
|
102
|
+
const templatePath = path_1.default.join(__dirname, '../templates/events-stack.json');
|
|
103
|
+
const cfnArgs = [
|
|
104
|
+
'cloudformation', 'deploy',
|
|
105
|
+
'--stack-name', STACK_NAME,
|
|
106
|
+
'--template-file', templatePath,
|
|
107
|
+
'--parameter-overrides',
|
|
108
|
+
`PlatformBaseUrl=${platformUrl}`,
|
|
109
|
+
`PlatformSecretName=${secretName}`,
|
|
110
|
+
'--capabilities', 'CAPABILITY_NAMED_IAM',
|
|
111
|
+
'--no-fail-on-empty-changeset',
|
|
112
|
+
'--region', region,
|
|
113
|
+
];
|
|
114
|
+
if (options.profile)
|
|
115
|
+
cfnArgs.push('--profile', options.profile);
|
|
116
|
+
(0, child_process_1.execFileSync)('aws', cfnArgs, { stdio: 'inherit' });
|
|
117
|
+
(0, output_utils_1.printSuccess)('Infrastructure deployed');
|
|
118
|
+
// Step 2: Download handler from registry and upload to Lambda
|
|
119
|
+
(0, output_utils_1.printSection)('Deploy Lambda Code');
|
|
120
|
+
const tmpDir = fs.mkdtempSync(path_1.default.join(os.tmpdir(), 'event-ingestion-'));
|
|
121
|
+
try {
|
|
122
|
+
const versionSpec = options.packageVersion ? `${PACKAGE_NAME}@${options.packageVersion}` : PACKAGE_NAME;
|
|
123
|
+
(0, output_utils_1.printInfo)('Installing from registry', { package: versionSpec });
|
|
124
|
+
(0, child_process_1.execFileSync)('npm', ['install', '--prefix', tmpDir, versionSpec], { stdio: 'pipe' });
|
|
125
|
+
const handlerSrc = path_1.default.join(tmpDir, 'node_modules', PACKAGE_NAME, 'lib', 'index.js');
|
|
126
|
+
if (!fs.existsSync(handlerSrc)) {
|
|
127
|
+
throw new Error(`Handler not found at ${handlerSrc}. Ensure the package is published.`);
|
|
128
|
+
}
|
|
129
|
+
// Get version from installed package
|
|
130
|
+
const pkgJsonPath = path_1.default.join(tmpDir, 'node_modules', PACKAGE_NAME, 'package.json');
|
|
131
|
+
const pkgJson = JSON.parse(fs.readFileSync(pkgJsonPath, 'utf-8'));
|
|
132
|
+
const version = pkgJson.version || 'unknown';
|
|
133
|
+
// ZIP the handler
|
|
134
|
+
const zipPath = path_1.default.join(tmpDir, 'index.zip');
|
|
135
|
+
(0, child_process_1.execFileSync)('zip', ['-j', zipPath, 'index.js'], { cwd: path_1.default.dirname(handlerSrc), stdio: 'pipe' });
|
|
136
|
+
// Upload directly to Lambda via SDK (no S3 needed)
|
|
137
|
+
(0, output_utils_1.printInfo)('Uploading code to Lambda', { function: LAMBDA_NAME, version });
|
|
138
|
+
const lambdaClient = new client_lambda_1.LambdaClient({ region });
|
|
139
|
+
await lambdaClient.send(new client_lambda_1.UpdateFunctionCodeCommand({
|
|
140
|
+
FunctionName: LAMBDA_NAME,
|
|
141
|
+
ZipFile: fs.readFileSync(zipPath),
|
|
142
|
+
}));
|
|
143
|
+
(0, output_utils_1.printSuccess)(`Lambda code deployed (${PACKAGE_NAME}@${version})`);
|
|
144
|
+
}
|
|
145
|
+
finally {
|
|
146
|
+
fs.rmSync(tmpDir, { recursive: true, force: true });
|
|
147
|
+
}
|
|
148
|
+
// Step 3: Show outputs via SDK
|
|
149
|
+
const cfnClient = new client_cloudformation_1.CloudFormationClient({ region });
|
|
150
|
+
const describeResult = await cfnClient.send(new client_cloudformation_1.DescribeStacksCommand({ StackName: STACK_NAME }));
|
|
151
|
+
const outputs = describeResult.Stacks?.[0]?.Outputs ?? [];
|
|
152
|
+
const outputMap = Object.fromEntries(outputs.map(o => [o.OutputKey ?? '', o.OutputValue ?? '']));
|
|
153
|
+
console.log('');
|
|
154
|
+
(0, output_utils_1.printSection)('Event Ingestion Ready');
|
|
155
|
+
(0, output_utils_1.printKeyValue)({
|
|
156
|
+
'Stack': STACK_NAME,
|
|
157
|
+
'Region': region,
|
|
158
|
+
'Platform URL': platformUrl,
|
|
159
|
+
'Lambda': outputMap.LambdaFunctionArn || LAMBDA_NAME,
|
|
160
|
+
'Event Queue': outputMap.EventQueueUrl || '(see AWS console)',
|
|
161
|
+
'Dead Letter Queue': outputMap.DeadLetterQueueUrl || '(see AWS console)',
|
|
162
|
+
'EventBridge Rule': outputMap.EventRuleName || '(see AWS console)',
|
|
163
|
+
'Status': '✓ Deployed',
|
|
164
|
+
});
|
|
165
|
+
console.log('');
|
|
166
|
+
(0, output_utils_1.printSuccess)('EventBridge event ingestion is active.');
|
|
167
|
+
}
|
|
168
|
+
catch (error) {
|
|
169
|
+
(0, error_handler_1.handleError)(error, error_handler_1.ERROR_CODES.API_REQUEST, {
|
|
170
|
+
debug: program.opts().debug,
|
|
171
|
+
exit: true,
|
|
172
|
+
context: { command: 'setup-events', executionId, stack: STACK_NAME },
|
|
173
|
+
});
|
|
174
|
+
}
|
|
175
|
+
});
|
|
176
|
+
}
|
|
177
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"setup-events.js","sourceRoot":"","sources":["../../src/commands/setup-events.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport { execFileSync } from 'child_process';\nimport * as fs from 'fs';\nimport * as os from 'os';\nimport path from 'path';\nimport { CloudFormationClient, DescribeStacksCommand } from '@aws-sdk/client-cloudformation';\nimport { LambdaClient, UpdateFunctionCodeCommand } from '@aws-sdk/client-lambda';\nimport { Command } from 'commander';\nimport { printCommandHeader } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printError, printInfo, printKeyValue, printSection, printSuccess } from '../utils/output-utils';\n\nconst STACK_NAME = 'pipeline-builder-events';\n// Existing CFN-managed Lambda function name — do NOT rename without a migration plan\nconst LAMBDA_NAME = 'pipeline-builder-event-ingestion';\nconst PACKAGE_NAME = '@pipeline-builder/event-bridge';\n\n/**\n * Registers the `setup-events` command with the CLI program.\n *\n * Deploys EventBridge → SQS → Lambda infrastructure for pipeline reporting.\n *\n * 1. Deploys CloudFormation stack (rule + queue + Lambda shell + IAM)\n * 2. Downloads @pipeline-builder/event-bridge from npm registry\n * 3. Uploads handler code directly to Lambda (no S3 needed)\n *\n * Uses PLATFORM_BASE_URL from environment (same as all other commands).\n * Idempotent — running again updates the stack and code.\n */\nexport function setupEvents(program: Command): void {\n  program\n    .command('setup-events')\n    .description('Deploy EventBridge event ingestion infrastructure for pipeline reporting')\n    .option('--package-version <version>', 'event-bridge package version (default: latest)')\n    .option('--secret-name <name>', 'Platform secret name (e.g. pipeline-builder/{orgId}/platform)')\n    .option('--region <region>', 'AWS region')\n    .option('--profile <profile>', 'AWS CLI profile', 'default')\n    .action(async (options) => {\n      const executionId = printCommandHeader('Setup Event Ingestion');\n\n      try {\n        const region = options.region || process.env.AWS_REGION || process.env.CDK_DEFAULT_REGION;\n        if (!region) {\n          printError('AWS region is required');\n          throw new Error('AWS region not provided');\n        }\n\n        const platformUrl = process.env.PLATFORM_BASE_URL;\n        if (!platformUrl) {\n          printError('PLATFORM_BASE_URL environment variable is required');\n          throw new Error('PLATFORM_BASE_URL not set');\n        }\n\n        const secretName = options.secretName || process.env.PLATFORM_SECRET_NAME;\n        if (!secretName) {\n          printError('--secret-name or PLATFORM_SECRET_NAME env var is required');\n          throw new Error('Platform secret name not provided');\n        }\n\n        printInfo('Parameters', {\n          stack: STACK_NAME,\n          region,\n          platformUrl,\n          secretName,\n          packageVersion: options.packageVersion || 'latest',\n        });\n\n        // Step 1: Deploy CloudFormation (infra only — Lambda gets placeholder code)\n        printSection('Deploy Infrastructure');\n\n        const templatePath = path.join(__dirname, '../templates/events-stack.json');\n\n        const cfnArgs = [\n          'cloudformation', 'deploy',\n          '--stack-name', STACK_NAME,\n          '--template-file', templatePath,\n          '--parameter-overrides',\n          `PlatformBaseUrl=${platformUrl}`,\n          `PlatformSecretName=${secretName}`,\n          '--capabilities', 'CAPABILITY_NAMED_IAM',\n          '--no-fail-on-empty-changeset',\n          '--region', region,\n        ];\n        if (options.profile) cfnArgs.push('--profile', options.profile);\n\n        execFileSync('aws', cfnArgs, { stdio: 'inherit' });\n\n        printSuccess('Infrastructure deployed');\n\n        // Step 2: Download handler from registry and upload to Lambda\n        printSection('Deploy Lambda Code');\n\n        const tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), 'event-ingestion-'));\n\n        try {\n          const versionSpec = options.packageVersion ? `${PACKAGE_NAME}@${options.packageVersion}` : PACKAGE_NAME;\n          printInfo('Installing from registry', { package: versionSpec });\n\n          execFileSync('npm', ['install', '--prefix', tmpDir, versionSpec], { stdio: 'pipe' });\n\n          const handlerSrc = path.join(tmpDir, 'node_modules', PACKAGE_NAME, 'lib', 'index.js');\n          if (!fs.existsSync(handlerSrc)) {\n            throw new Error(`Handler not found at ${handlerSrc}. Ensure the package is published.`);\n          }\n\n          // Get version from installed package\n          const pkgJsonPath = path.join(tmpDir, 'node_modules', PACKAGE_NAME, 'package.json');\n          const pkgJson = JSON.parse(fs.readFileSync(pkgJsonPath, 'utf-8'));\n          const version = pkgJson.version || 'unknown';\n\n          // ZIP the handler\n          const zipPath = path.join(tmpDir, 'index.zip');\n          execFileSync('zip', ['-j', zipPath, 'index.js'], { cwd: path.dirname(handlerSrc), stdio: 'pipe' });\n\n          // Upload directly to Lambda via SDK (no S3 needed)\n          printInfo('Uploading code to Lambda', { function: LAMBDA_NAME, version });\n\n          const lambdaClient = new LambdaClient({ region });\n          await lambdaClient.send(new UpdateFunctionCodeCommand({\n            FunctionName: LAMBDA_NAME,\n            ZipFile: fs.readFileSync(zipPath),\n          }));\n\n          printSuccess(`Lambda code deployed (${PACKAGE_NAME}@${version})`);\n\n        } finally {\n          fs.rmSync(tmpDir, { recursive: true, force: true });\n        }\n\n        // Step 3: Show outputs via SDK\n        const cfnClient = new CloudFormationClient({ region });\n        const describeResult = await cfnClient.send(new DescribeStacksCommand({ StackName: STACK_NAME }));\n        const outputs = describeResult.Stacks?.[0]?.Outputs ?? [];\n        const outputMap = Object.fromEntries(outputs.map(o => [o.OutputKey ?? '', o.OutputValue ?? '']));\n\n        console.log('');\n        printSection('Event Ingestion Ready');\n\n        printKeyValue({\n          'Stack': STACK_NAME,\n          'Region': region,\n          'Platform URL': platformUrl,\n          'Lambda': outputMap.LambdaFunctionArn || LAMBDA_NAME,\n          'Event Queue': outputMap.EventQueueUrl || '(see AWS console)',\n          'Dead Letter Queue': outputMap.DeadLetterQueueUrl || '(see AWS console)',\n          'EventBridge Rule': outputMap.EventRuleName || '(see AWS console)',\n          'Status': '✓ Deployed',\n        });\n\n        console.log('');\n        printSuccess('EventBridge event ingestion is active.');\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.API_REQUEST, {\n          debug: program.opts().debug,\n          exit: true,\n          context: { command: 'setup-events', executionId, stack: STACK_NAME },\n        });\n      }\n    });\n}\n"]}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { Command } from 'commander';
|
|
2
|
+
/**
|
|
3
|
+
* Registers the `status` command with the CLI program.
|
|
4
|
+
*
|
|
5
|
+
* Checks environment variables, CDK availability, platform connectivity,
|
|
6
|
+
* and token expiry to give a quick overview of the current setup.
|
|
7
|
+
*
|
|
8
|
+
* @param program - The root Commander program instance to attach the command to.
|
|
9
|
+
*/
|
|
10
|
+
export declare function status(program: Command): void;
|
|
11
|
+
//# sourceMappingURL=status.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"status.d.ts","sourceRoot":"","sources":["../../src/commands/status.ts"],"names":[],"mappings":"AAKA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAOpC;;;;;;;GAOG;AACH,wBAAgB,MAAM,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAqE7C"}
|
|
@@ -0,0 +1,89 @@
|
|
|
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.status = status;
|
|
9
|
+
const https_1 = __importDefault(require("https"));
|
|
10
|
+
const axios_1 = __importDefault(require("axios"));
|
|
11
|
+
const cli_constants_1 = require("../config/cli.constants");
|
|
12
|
+
const cdk_utils_1 = require("../utils/cdk-utils");
|
|
13
|
+
const command_utils_1 = require("../utils/command-utils");
|
|
14
|
+
const error_handler_1 = require("../utils/error-handler");
|
|
15
|
+
const output_utils_1 = require("../utils/output-utils");
|
|
16
|
+
/**
|
|
17
|
+
* Registers the `status` command with the CLI program.
|
|
18
|
+
*
|
|
19
|
+
* Checks environment variables, CDK availability, platform connectivity,
|
|
20
|
+
* and token expiry to give a quick overview of the current setup.
|
|
21
|
+
*
|
|
22
|
+
* @param program - The root Commander program instance to attach the command to.
|
|
23
|
+
*/
|
|
24
|
+
function status(program) {
|
|
25
|
+
program
|
|
26
|
+
.command('status')
|
|
27
|
+
.description('Show environment and connectivity status')
|
|
28
|
+
.option('--json', 'Output result as JSON', false)
|
|
29
|
+
.action(async (options) => {
|
|
30
|
+
const executionId = (0, command_utils_1.printCommandHeader)('Status');
|
|
31
|
+
try {
|
|
32
|
+
const results = {};
|
|
33
|
+
// 1. Check PLATFORM_TOKEN
|
|
34
|
+
const token = process.env[cli_constants_1.ENV_VARS.PLATFORM_TOKEN];
|
|
35
|
+
results.PLATFORM_TOKEN = token ? 'set' : 'not set';
|
|
36
|
+
// 2. Check PLATFORM_SECRET_NAME
|
|
37
|
+
const secretName = process.env.PLATFORM_SECRET_NAME;
|
|
38
|
+
results.PLATFORM_SECRET_NAME = secretName || 'not set';
|
|
39
|
+
// 3. Check CDK availability
|
|
40
|
+
results['CDK Available'] = (0, cdk_utils_1.checkCdkAvailable)() ? 'yes' : 'no';
|
|
41
|
+
// 4. Check platform health
|
|
42
|
+
const baseUrl = process.env[cli_constants_1.ENV_VARS.PLATFORM_BASE_URL] || 'https://localhost:8443';
|
|
43
|
+
results.PLATFORM_BASE_URL = baseUrl;
|
|
44
|
+
try {
|
|
45
|
+
const httpsAgent = new https_1.default.Agent({ rejectUnauthorized: false });
|
|
46
|
+
await axios_1.default.get(`${baseUrl}/health`, {
|
|
47
|
+
timeout: cli_constants_1.TIMEOUTS.HEALTH_CHECK,
|
|
48
|
+
httpsAgent,
|
|
49
|
+
});
|
|
50
|
+
results['Platform Health'] = 'reachable';
|
|
51
|
+
}
|
|
52
|
+
catch {
|
|
53
|
+
results['Platform Health'] = 'unreachable';
|
|
54
|
+
}
|
|
55
|
+
// 5. Decode token expiry
|
|
56
|
+
if (token) {
|
|
57
|
+
try {
|
|
58
|
+
const parts = token.split('.');
|
|
59
|
+
if (parts.length === 3 && parts[1]) {
|
|
60
|
+
const payload = JSON.parse(Buffer.from(parts[1], 'base64url').toString('utf-8'));
|
|
61
|
+
if (typeof payload.exp === 'number') {
|
|
62
|
+
const expiresAt = new Date(payload.exp * 1000);
|
|
63
|
+
results['Token Expires'] = expiresAt.toISOString();
|
|
64
|
+
results['Token Expired'] = expiresAt.getTime() < Date.now() ? 'yes' : 'no';
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
catch {
|
|
69
|
+
results['Token Expires'] = 'unable to decode';
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
if (options.json) {
|
|
73
|
+
console.log(JSON.stringify({ success: true, executionId, ...results }, null, 2));
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
(0, output_utils_1.printSection)('Environment Status');
|
|
77
|
+
(0, output_utils_1.printKeyValue)(results);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
catch (error) {
|
|
81
|
+
(0, error_handler_1.handleError)(error, error_handler_1.ERROR_CODES.GENERAL, {
|
|
82
|
+
debug: program.opts().debug,
|
|
83
|
+
exit: true,
|
|
84
|
+
context: { command: 'status', executionId },
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"status.js","sourceRoot":"","sources":["../../src/commands/status.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright 2026 Pipeline Builder Contributors\n// SPDX-License-Identifier: Apache-2.0\n\nimport https from 'https';\nimport axios from 'axios';\nimport { Command } from 'commander';\nimport { ENV_VARS, TIMEOUTS } from '../config/cli.constants';\nimport { checkCdkAvailable } from '../utils/cdk-utils';\nimport { printCommandHeader } from '../utils/command-utils';\nimport { ERROR_CODES, handleError } from '../utils/error-handler';\nimport { printKeyValue, printSection } from '../utils/output-utils';\n\n/**\n * Registers the `status` command with the CLI program.\n *\n * Checks environment variables, CDK availability, platform connectivity,\n * and token expiry to give a quick overview of the current setup.\n *\n * @param program - The root Commander program instance to attach the command to.\n */\nexport function status(program: Command): void {\n  program\n    .command('status')\n    .description('Show environment and connectivity status')\n    .option('--json', 'Output result as JSON', false)\n    .action(async (options) => {\n      const executionId = printCommandHeader('Status');\n\n      try {\n        const results: Record<string, string> = {};\n\n        // 1. Check PLATFORM_TOKEN\n        const token = process.env[ENV_VARS.PLATFORM_TOKEN];\n        results.PLATFORM_TOKEN = token ? 'set' : 'not set';\n\n        // 2. Check PLATFORM_SECRET_NAME\n        const secretName = process.env.PLATFORM_SECRET_NAME;\n        results.PLATFORM_SECRET_NAME = secretName || 'not set';\n\n        // 3. Check CDK availability\n        results['CDK Available'] = checkCdkAvailable() ? 'yes' : 'no';\n\n        // 4. Check platform health\n        const baseUrl = process.env[ENV_VARS.PLATFORM_BASE_URL] || 'https://localhost:8443';\n        results.PLATFORM_BASE_URL = baseUrl;\n\n        try {\n          const httpsAgent = new https.Agent({ rejectUnauthorized: false });\n          await axios.get(`${baseUrl}/health`, {\n            timeout: TIMEOUTS.HEALTH_CHECK,\n            httpsAgent,\n          });\n          results['Platform Health'] = 'reachable';\n        } catch {\n          results['Platform Health'] = 'unreachable';\n        }\n\n        // 5. Decode token expiry\n        if (token) {\n          try {\n            const parts = token.split('.');\n            if (parts.length === 3 && parts[1]) {\n              const payload = JSON.parse(Buffer.from(parts[1], 'base64url').toString('utf-8')) as Record<string, unknown>;\n              if (typeof payload.exp === 'number') {\n                const expiresAt = new Date(payload.exp * 1000);\n                results['Token Expires'] = expiresAt.toISOString();\n                results['Token Expired'] = expiresAt.getTime() < Date.now() ? 'yes' : 'no';\n              }\n            }\n          } catch {\n            results['Token Expires'] = 'unable to decode';\n          }\n        }\n\n        if (options.json) {\n          console.log(JSON.stringify({ success: true, executionId, ...results }, null, 2));\n        } else {\n          printSection('Environment Status');\n          printKeyValue(results);\n        }\n\n      } catch (error) {\n        handleError(error, ERROR_CODES.GENERAL, {\n          debug: program.opts().debug,\n          exit: true,\n          context: { command: 'status', executionId },\n        });\n      }\n    });\n}\n"]}
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import { Command } from 'commander';
|
|
2
|
+
/**
|
|
3
|
+
* Registers the `store-token` command with the CLI program.
|
|
4
|
+
*
|
|
5
|
+
* Generates a long-lived JWT token via the platform API and stores it
|
|
6
|
+
* in AWS Secrets Manager for use by the synth/deploy --store-tokens flag.
|
|
7
|
+
*
|
|
8
|
+
* Requires PLATFORM_TOKEN to be set, or use --email/--password to login inline.
|
|
9
|
+
*
|
|
10
|
+
* @example
|
|
11
|
+
* ```bash
|
|
12
|
+
* pipeline-manager store-token --region us-east-1
|
|
13
|
+
* pipeline-manager store-token --days 90 --region us-east-1
|
|
14
|
+
* pipeline-manager store-token --days 7 --secret-name my-custom-secret --no-verify-ssl
|
|
15
|
+
* pipeline-manager store-token --dry-run
|
|
16
|
+
* pipeline-manager store-token -e admin -p '***' --region us-east-1
|
|
17
|
+
* ```
|
|
18
|
+
*/
|
|
19
|
+
export declare function storeToken(program: Command): void;
|
|
20
|
+
//# sourceMappingURL=store-token.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"store-token.d.ts","sourceRoot":"","sources":["../../src/commands/store-token.ts"],"names":[],"mappings":"AAKA,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AAwBpC;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,UAAU,CAAC,OAAO,EAAE,OAAO,GAAG,IAAI,CAmLjD"}
|