geni-bioinfo 0.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/dist/auth.d.ts +8 -0
- package/dist/auth.js +76 -0
- package/dist/aws/clients.d.ts +18 -0
- package/dist/aws/clients.js +34 -0
- package/dist/commands/activate.d.ts +2 -0
- package/dist/commands/activate.js +20 -0
- package/dist/commands/activity-log.d.ts +2 -0
- package/dist/commands/activity-log.js +31 -0
- package/dist/commands/api-token.d.ts +2 -0
- package/dist/commands/api-token.js +96 -0
- package/dist/commands/auth.d.ts +2 -0
- package/dist/commands/auth.js +55 -0
- package/dist/commands/engine.d.ts +2 -0
- package/dist/commands/engine.js +83 -0
- package/dist/commands/environment.d.ts +2 -0
- package/dist/commands/environment.js +69 -0
- package/dist/commands/image.d.ts +2 -0
- package/dist/commands/image.js +40 -0
- package/dist/commands/instance.d.ts +2 -0
- package/dist/commands/instance.js +39 -0
- package/dist/commands/log.d.ts +3 -0
- package/dist/commands/log.js +43 -0
- package/dist/commands/login.d.ts +2 -0
- package/dist/commands/login.js +24 -0
- package/dist/commands/plugin.d.ts +2 -0
- package/dist/commands/plugin.js +80 -0
- package/dist/commands/queue.d.ts +2 -0
- package/dist/commands/queue.js +74 -0
- package/dist/commands/registry.d.ts +2 -0
- package/dist/commands/registry.js +72 -0
- package/dist/commands/setup/create.d.ts +2 -0
- package/dist/commands/setup/create.js +254 -0
- package/dist/commands/setup/delete.d.ts +2 -0
- package/dist/commands/setup/delete.js +97 -0
- package/dist/commands/setup/status.d.ts +2 -0
- package/dist/commands/setup/status.js +46 -0
- package/dist/commands/setup.d.ts +2 -0
- package/dist/commands/setup.js +13 -0
- package/dist/commands/storage.d.ts +2 -0
- package/dist/commands/storage.js +67 -0
- package/dist/commands/submission.d.ts +2 -0
- package/dist/commands/submission.js +87 -0
- package/dist/commands/task.d.ts +2 -0
- package/dist/commands/task.js +21 -0
- package/dist/commands/tenant.d.ts +2 -0
- package/dist/commands/tenant.js +68 -0
- package/dist/commands/user.d.ts +2 -0
- package/dist/commands/user.js +82 -0
- package/dist/commands/workflow.d.ts +2 -0
- package/dist/commands/workflow.js +80 -0
- package/dist/errors.d.ts +38 -0
- package/dist/errors.js +194 -0
- package/dist/format.d.ts +22 -0
- package/dist/format.js +155 -0
- package/dist/index.d.ts +2 -0
- package/dist/index.js +91 -0
- package/dist/templates/setup.yaml +503 -0
- package/package.json +49 -0
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.submissionCommand = void 0;
|
|
7
|
+
const commander_1 = require("commander");
|
|
8
|
+
const fs_1 = __importDefault(require("fs"));
|
|
9
|
+
const path_1 = __importDefault(require("path"));
|
|
10
|
+
const auth_1 = require("../auth");
|
|
11
|
+
const errors_1 = require("../errors");
|
|
12
|
+
const format_js_1 = require("../format.js");
|
|
13
|
+
function parseTimeToSeconds(value) {
|
|
14
|
+
const parts = value.split(':').map(Number);
|
|
15
|
+
if (parts.length !== 3 || parts.some(isNaN)) {
|
|
16
|
+
throw (0, errors_1.invalidArgument)('Time must be in DD:HH:MM format (days:hours:minutes).', 'Example: 00:12:30');
|
|
17
|
+
}
|
|
18
|
+
const [days, hours, minutes] = parts;
|
|
19
|
+
return (days * 86400) + (hours * 3600) + (minutes * 60);
|
|
20
|
+
}
|
|
21
|
+
exports.submissionCommand = new commander_1.Command('submission')
|
|
22
|
+
.description('Manage workflow submissions');
|
|
23
|
+
exports.submissionCommand
|
|
24
|
+
.command('list')
|
|
25
|
+
.description('List submissions')
|
|
26
|
+
.option('--status <status>', 'Filter by status (SUBMITTED, SUCCEEDED, FAILED)')
|
|
27
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
28
|
+
.action(async (opts) => {
|
|
29
|
+
const query = opts.status ? `?status=${opts.status}` : '';
|
|
30
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/submissions${query}`));
|
|
31
|
+
(0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.submissionColumns);
|
|
32
|
+
});
|
|
33
|
+
exports.submissionCommand
|
|
34
|
+
.command('get <id>')
|
|
35
|
+
.description('Get a submission by ID')
|
|
36
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
37
|
+
.action(async (id, opts) => {
|
|
38
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/submissions/${id}`));
|
|
39
|
+
(0, format_js_1.formatOutput)(data, opts.format, format_js_1.submissionColumns);
|
|
40
|
+
});
|
|
41
|
+
exports.submissionCommand
|
|
42
|
+
.command('create')
|
|
43
|
+
.description('Submit a workflow run')
|
|
44
|
+
.requiredOption('--workflow-name <name>', 'Workflow name')
|
|
45
|
+
.option('--workflow-version <version>', 'Workflow version')
|
|
46
|
+
.requiredOption('--params-file <file>', 'Path to params file (.json/.yaml/.yml)')
|
|
47
|
+
.requiredOption('--engine-id <id>', 'Engine ID')
|
|
48
|
+
.requiredOption('--queue-id <id>', 'Queue ID')
|
|
49
|
+
.requiredOption('--output-folder <s3-uri>', 'S3 URI prefix for workflow output files (e.g. s3://my-bucket/outputs/)')
|
|
50
|
+
.option('--alert-time <DD:HH:MM>', 'Alert time override (DD:HH:MM)')
|
|
51
|
+
.option('--cancel-time <DD:HH:MM>', 'Cancel time override (DD:HH:MM)')
|
|
52
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
53
|
+
.action(async (opts) => {
|
|
54
|
+
const form = new FormData();
|
|
55
|
+
form.append('workflowName', opts.workflowName);
|
|
56
|
+
if (opts.workflowVersion)
|
|
57
|
+
form.append('version', opts.workflowVersion);
|
|
58
|
+
form.append('engineId', opts.engineId);
|
|
59
|
+
form.append('queueId', opts.queueId);
|
|
60
|
+
form.append('output', opts.outputFolder);
|
|
61
|
+
if (opts.alertTime)
|
|
62
|
+
form.append('alertTimeSeconds', String(parseTimeToSeconds(opts.alertTime)));
|
|
63
|
+
if (opts.cancelTime)
|
|
64
|
+
form.append('cancelTimeSeconds', String(parseTimeToSeconds(opts.cancelTime)));
|
|
65
|
+
const paramsBuffer = await fs_1.default.promises.readFile(opts.paramsFile);
|
|
66
|
+
form.append('params', new Blob([paramsBuffer]), path_1.default.basename(opts.paramsFile));
|
|
67
|
+
const res = await fetch((0, auth_1.apiUrl)('/submissions'), {
|
|
68
|
+
method: 'POST',
|
|
69
|
+
headers: (0, auth_1.getAuthHeaders)(),
|
|
70
|
+
body: form,
|
|
71
|
+
});
|
|
72
|
+
const data = await (0, auth_1.readJsonResponse)(res, { method: 'POST', url: (0, auth_1.apiUrl)('/submissions') });
|
|
73
|
+
(0, format_js_1.formatOutput)(data, opts.format, format_js_1.submissionColumns);
|
|
74
|
+
});
|
|
75
|
+
exports.submissionCommand
|
|
76
|
+
.command('cancel <id>')
|
|
77
|
+
.description('Cancel a running submission')
|
|
78
|
+
.action(async (id) => {
|
|
79
|
+
const url = (0, auth_1.apiUrl)(`/submissions/${id}/cancel`);
|
|
80
|
+
const res = await fetch(url, {
|
|
81
|
+
method: 'POST',
|
|
82
|
+
headers: (0, auth_1.getAuthHeaders)(),
|
|
83
|
+
});
|
|
84
|
+
const data = await (0, auth_1.readJsonResponse)(res, { method: 'POST', url });
|
|
85
|
+
console.log(`${data.message} (${data.submissionId})`);
|
|
86
|
+
});
|
|
87
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"submission.js","sourceRoot":"","sources":["../../src/commands/submission.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander'\nimport fs from 'fs'\nimport path from 'path'\nimport { apiUrl, fetchJson, getAuthHeaders, readJsonResponse } from '../auth'\nimport { invalidArgument } from '../errors'\nimport { formatOutput, submissionColumns, type OutputFormat } from '../format.js'\n\nfunction parseTimeToSeconds(value: string): number {\n  const parts = value.split(':').map(Number)\n  if (parts.length !== 3 || parts.some(isNaN)) {\n    throw invalidArgument(\n      'Time must be in DD:HH:MM format (days:hours:minutes).',\n      'Example: 00:12:30',\n    )\n  }\n  const [days, hours, minutes] = parts\n  return (days * 86400) + (hours * 3600) + (minutes * 60)\n}\n\n\nexport const submissionCommand = new Command('submission')\n  .description('Manage workflow submissions')\n\nsubmissionCommand\n  .command('list')\n  .description('List submissions')\n  .option('--status <status>', 'Filter by status (SUBMITTED, SUCCEEDED, FAILED)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { status?: string; format: OutputFormat }) => {\n    const query = opts.status ? `?status=${opts.status}` : ''\n    const data = await fetchJson(apiUrl(`/submissions${query}`)) as { items: unknown[] }\n    formatOutput(opts.format === 'json' ? data : data.items, opts.format, submissionColumns)\n  })\n\nsubmissionCommand\n  .command('get <id>')\n  .description('Get a submission by ID')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (id: string, opts: { format: OutputFormat }) => {\n    const data = await fetchJson(apiUrl(`/submissions/${id}`))\n    formatOutput(data, opts.format, submissionColumns)\n  })\n\nsubmissionCommand\n  .command('create')\n  .description('Submit a workflow run')\n  .requiredOption('--workflow-name <name>', 'Workflow name')\n  .option('--workflow-version <version>', 'Workflow version')\n  .requiredOption('--params-file <file>', 'Path to params file (.json/.yaml/.yml)')\n  .requiredOption('--engine-id <id>', 'Engine ID')\n  .requiredOption('--queue-id <id>', 'Queue ID')\n  .requiredOption('--output-folder <s3-uri>', 'S3 URI prefix for workflow output files (e.g. s3://my-bucket/outputs/)')\n  .option('--alert-time <DD:HH:MM>', 'Alert time override (DD:HH:MM)')\n  .option('--cancel-time <DD:HH:MM>', 'Cancel time override (DD:HH:MM)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: {\n    workflowName: string\n    workflowVersion?: string\n    paramsFile: string\n    engineId: string\n    queueId: string\n    outputFolder: string\n    alertTime?: string\n    cancelTime?: string\n    format: OutputFormat\n  }) => {\n    const form = new FormData()\n    form.append('workflowName', opts.workflowName)\n    if (opts.workflowVersion) form.append('version', opts.workflowVersion)\n    form.append('engineId', opts.engineId)\n    form.append('queueId', opts.queueId)\n    form.append('output', opts.outputFolder)\n    if (opts.alertTime) form.append('alertTimeSeconds', String(parseTimeToSeconds(opts.alertTime)))\n    if (opts.cancelTime) form.append('cancelTimeSeconds', String(parseTimeToSeconds(opts.cancelTime)))\n    const paramsBuffer = await fs.promises.readFile(opts.paramsFile)\n    form.append('params', new Blob([paramsBuffer]), path.basename(opts.paramsFile))\n\n    const res = await fetch(apiUrl('/submissions'), {\n      method: 'POST',\n      headers: getAuthHeaders(),\n      body: form,\n    })\n    const data = await readJsonResponse(res, { method: 'POST', url: apiUrl('/submissions') })\n    formatOutput(data, opts.format, submissionColumns)\n  })\n\nsubmissionCommand\n  .command('cancel <id>')\n  .description('Cancel a running submission')\n  .action(async (id: string) => {\n    const url = apiUrl(`/submissions/${id}/cancel`)\n    const res = await fetch(url, {\n      method: 'POST',\n      headers: getAuthHeaders(),\n    })\n    const data = await readJsonResponse(res, { method: 'POST', url }) as { submissionId: string; message: string }\n    console.log(`${data.message} (${data.submissionId})`)\n  })\n"]}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.taskCommand = void 0;
|
|
4
|
+
const commander_1 = require("commander");
|
|
5
|
+
const auth_1 = require("../auth");
|
|
6
|
+
const format_js_1 = require("../format.js");
|
|
7
|
+
exports.taskCommand = new commander_1.Command('task')
|
|
8
|
+
.description('Manage tasks');
|
|
9
|
+
exports.taskCommand
|
|
10
|
+
.command('list')
|
|
11
|
+
.description('List tasks')
|
|
12
|
+
.option('--submission-id <id>', 'Filter by submission ID')
|
|
13
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
14
|
+
.action(async (opts) => {
|
|
15
|
+
const url = opts.submissionId
|
|
16
|
+
? (0, auth_1.apiUrl)(`/tasks?submissionId=${opts.submissionId}`)
|
|
17
|
+
: (0, auth_1.apiUrl)('/tasks');
|
|
18
|
+
const data = await (0, auth_1.fetchJson)(url);
|
|
19
|
+
(0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.taskColumns);
|
|
20
|
+
});
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFzay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jb21tYW5kcy90YXNrLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLHlDQUEyQztBQUMzQyxrQ0FBMkM7QUFDM0MsNENBQTJFO0FBRTlELFFBQUEsV0FBVyxHQUFHLElBQUksbUJBQU8sQ0FBQyxNQUFNLENBQUM7S0FDM0MsV0FBVyxDQUFDLGNBQWMsQ0FBQyxDQUFBO0FBRTlCLG1CQUFXO0tBQ1IsT0FBTyxDQUFDLE1BQU0sQ0FBQztLQUNmLFdBQVcsQ0FBQyxZQUFZLENBQUM7S0FDekIsTUFBTSxDQUFDLHNCQUFzQixFQUFFLHlCQUF5QixDQUFDO0tBQ3pELFNBQVMsQ0FBQyxJQUFJLGtCQUFNLENBQUMsbUJBQW1CLEVBQUUsZUFBZSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztLQUM5RyxNQUFNLENBQUMsS0FBSyxFQUFFLElBQXFELEVBQUUsRUFBRTtJQUN0RSxNQUFNLEdBQUcsR0FBRyxJQUFJLENBQUMsWUFBWTtRQUMzQixDQUFDLENBQUMsSUFBQSxhQUFNLEVBQUMsdUJBQXVCLElBQUksQ0FBQyxZQUFZLEVBQUUsQ0FBQztRQUNwRCxDQUFDLENBQUMsSUFBQSxhQUFNLEVBQUMsUUFBUSxDQUFDLENBQUE7SUFDcEIsTUFBTSxJQUFJLEdBQUcsTUFBTSxJQUFBLGdCQUFTLEVBQUMsR0FBRyxDQUF5QixDQUFBO0lBQ3pELElBQUEsd0JBQVksRUFBQyxJQUFJLENBQUMsTUFBTSxLQUFLLE1BQU0sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLElBQUksQ0FBQyxNQUFNLEVBQUUsdUJBQVcsQ0FBQyxDQUFBO0FBQ3BGLENBQUMsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbWFuZCwgT3B0aW9uIH0gZnJvbSAnY29tbWFuZGVyJ1xuaW1wb3J0IHsgYXBpVXJsLCBmZXRjaEpzb24gfSBmcm9tICcuLi9hdXRoJ1xuaW1wb3J0IHsgZm9ybWF0T3V0cHV0LCB0YXNrQ29sdW1ucywgdHlwZSBPdXRwdXRGb3JtYXQgfSBmcm9tICcuLi9mb3JtYXQuanMnXG5cbmV4cG9ydCBjb25zdCB0YXNrQ29tbWFuZCA9IG5ldyBDb21tYW5kKCd0YXNrJylcbiAgLmRlc2NyaXB0aW9uKCdNYW5hZ2UgdGFza3MnKVxuXG50YXNrQ29tbWFuZFxuICAuY29tbWFuZCgnbGlzdCcpXG4gIC5kZXNjcmlwdGlvbignTGlzdCB0YXNrcycpXG4gIC5vcHRpb24oJy0tc3VibWlzc2lvbi1pZCA8aWQ+JywgJ0ZpbHRlciBieSBzdWJtaXNzaW9uIElEJylcbiAgLmFkZE9wdGlvbihuZXcgT3B0aW9uKCctLWZvcm1hdCA8Zm9ybWF0PicsICdPdXRwdXQgZm9ybWF0JykuY2hvaWNlcyhbJ3RhYmxlJywgJ2pzb24nLCAnY3N2J10pLmRlZmF1bHQoJ3RhYmxlJykpXG4gIC5hY3Rpb24oYXN5bmMgKG9wdHM6IHsgc3VibWlzc2lvbklkPzogc3RyaW5nOyBmb3JtYXQ6IE91dHB1dEZvcm1hdCB9KSA9PiB7XG4gICAgY29uc3QgdXJsID0gb3B0cy5zdWJtaXNzaW9uSWRcbiAgICAgID8gYXBpVXJsKGAvdGFza3M/c3VibWlzc2lvbklkPSR7b3B0cy5zdWJtaXNzaW9uSWR9YClcbiAgICAgIDogYXBpVXJsKCcvdGFza3MnKVxuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCBmZXRjaEpzb24odXJsKSBhcyB7IGl0ZW1zOiB1bmtub3duW10gfVxuICAgIGZvcm1hdE91dHB1dChvcHRzLmZvcm1hdCA9PT0gJ2pzb24nID8gZGF0YSA6IGRhdGEuaXRlbXMsIG9wdHMuZm9ybWF0LCB0YXNrQ29sdW1ucylcbiAgfSlcbiJdfQ==
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.tenantCommand = void 0;
|
|
4
|
+
const commander_1 = require("commander");
|
|
5
|
+
const auth_1 = require("../auth");
|
|
6
|
+
const format_1 = require("../format");
|
|
7
|
+
exports.tenantCommand = new commander_1.Command('tenant')
|
|
8
|
+
.description('Manage tenants');
|
|
9
|
+
exports.tenantCommand
|
|
10
|
+
.command('create')
|
|
11
|
+
.description('Create a new tenant')
|
|
12
|
+
.requiredOption('--name <name>', 'Tenant name (lowercase, hyphens allowed)')
|
|
13
|
+
.requiredOption('--contact-email <email>', 'Contact email address for the tenant')
|
|
14
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
15
|
+
.action(async (opts) => {
|
|
16
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)('/support/tenants'), {
|
|
17
|
+
method: 'POST',
|
|
18
|
+
headers: { 'Content-Type': 'application/json' },
|
|
19
|
+
body: JSON.stringify({ name: opts.name, contactEmail: opts.contactEmail }),
|
|
20
|
+
});
|
|
21
|
+
(0, format_1.formatOutput)(data, opts.format, format_1.tenantColumns);
|
|
22
|
+
});
|
|
23
|
+
exports.tenantCommand
|
|
24
|
+
.command('list')
|
|
25
|
+
.description('List all tenants')
|
|
26
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
27
|
+
.action(async (opts) => {
|
|
28
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)('/support/tenants'));
|
|
29
|
+
(0, format_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_1.tenantColumns);
|
|
30
|
+
});
|
|
31
|
+
exports.tenantCommand
|
|
32
|
+
.command('get <id>')
|
|
33
|
+
.description('Get a tenant by ID')
|
|
34
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
35
|
+
.action(async (id, opts) => {
|
|
36
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/support/tenants/${id}`));
|
|
37
|
+
(0, format_1.formatOutput)(data, opts.format, format_1.tenantColumns);
|
|
38
|
+
});
|
|
39
|
+
exports.tenantCommand
|
|
40
|
+
.command('enable <id>')
|
|
41
|
+
.description('Enable a tenant')
|
|
42
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
43
|
+
.action(async (id, opts) => {
|
|
44
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/support/tenants/${id}/enable`), {
|
|
45
|
+
method: 'PUT',
|
|
46
|
+
});
|
|
47
|
+
(0, format_1.formatOutput)(data, opts.format, format_1.tenantColumns);
|
|
48
|
+
});
|
|
49
|
+
exports.tenantCommand
|
|
50
|
+
.command('disable <id>')
|
|
51
|
+
.description('Disable a tenant')
|
|
52
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
53
|
+
.action(async (id, opts) => {
|
|
54
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/support/tenants/${id}/disable`), {
|
|
55
|
+
method: 'PUT',
|
|
56
|
+
});
|
|
57
|
+
(0, format_1.formatOutput)(data, opts.format, format_1.tenantColumns);
|
|
58
|
+
});
|
|
59
|
+
exports.tenantCommand
|
|
60
|
+
.command('delete <id>')
|
|
61
|
+
.description('Delete a tenant')
|
|
62
|
+
.action(async (id) => {
|
|
63
|
+
await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/support/tenants/${id}`), {
|
|
64
|
+
method: 'DELETE',
|
|
65
|
+
});
|
|
66
|
+
console.log('Tenant deleted successfully.');
|
|
67
|
+
});
|
|
68
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVuYW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbW1hbmRzL3RlbmFudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5Q0FBNEM7QUFDNUMsa0NBQTRDO0FBQzVDLHNDQUEyRTtBQUU5RCxRQUFBLGFBQWEsR0FBRyxJQUFJLG1CQUFPLENBQUMsUUFBUSxDQUFDO0tBQy9DLFdBQVcsQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO0FBRWpDLHFCQUFhO0tBQ1YsT0FBTyxDQUFDLFFBQVEsQ0FBQztLQUNqQixXQUFXLENBQUMscUJBQXFCLENBQUM7S0FDbEMsY0FBYyxDQUFDLGVBQWUsRUFBRSwwQ0FBMEMsQ0FBQztLQUMzRSxjQUFjLENBQUMseUJBQXlCLEVBQUUsc0NBQXNDLENBQUM7S0FDakYsU0FBUyxDQUFDLElBQUksa0JBQU0sQ0FBQyxtQkFBbUIsRUFBRSxlQUFlLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0tBQzlHLE1BQU0sQ0FBQyxLQUFLLEVBQUUsSUFBa0UsRUFBRSxFQUFFO0lBQ25GLE1BQU0sSUFBSSxHQUFHLE1BQU0sSUFBQSxnQkFBUyxFQUFDLElBQUEsYUFBTSxFQUFDLGtCQUFrQixDQUFDLEVBQUU7UUFDdkQsTUFBTSxFQUFFLE1BQU07UUFDZCxPQUFPLEVBQUUsRUFBRSxjQUFjLEVBQUUsa0JBQWtCLEVBQUU7UUFDL0MsSUFBSSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsRUFBRSxJQUFJLEVBQUUsSUFBSSxDQUFDLElBQUksRUFBRSxZQUFZLEVBQUUsSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDO0tBQzNFLENBQUMsQ0FBQztJQUNILElBQUEscUJBQVksRUFBQyxJQUFJLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBRSxzQkFBYSxDQUFDLENBQUM7QUFDakQsQ0FBQyxDQUFD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LE1BQU0sRUFBRSxzQkFBYSxDQUFDLENBQUM7QUFDakQsQ0FBQyxDQUFDLENBQUM7QUFFTCxxQkFBYTtLQUNWLE9BQU8sQ0FBQyxhQUFhLENBQUM7S0FDdEIsV0FBVyxDQUFDLGlCQUFpQixDQUFDO0tBQzlCLE1BQU0sQ0FBQyxLQUFLLEVBQUUsRUFBVSxFQUFFLEVBQUU7SUFDM0IsTUFBTSxJQUFBLGdCQUFTLEVBQUMsSUFBQSxhQUFNLEVBQUMsb0JBQW9CLEVBQUUsRUFBRSxDQUFDLEVBQUU7UUFDaEQsTUFBTSxFQUFFLFFBQVE7S0FDakIsQ0FBQyxDQUFDO0lBQ0gsT0FBTyxDQUFDLEdBQUcsQ0FBQyw4QkFBOEIsQ0FBQyxDQUFDO0FBQzlDLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbWFuZCwgT3B0aW9uIH0gZnJvbSAnY29tbWFuZGVyJztcbmltcG9ydCB7IGFwaVVybCwgZmV0Y2hKc29uIH0gZnJvbSAnLi4vYXV0aCc7XG5pbXBvcnQgeyBmb3JtYXRPdXRwdXQsIHRlbmFudENvbHVtbnMsIHR5cGUgT3V0cHV0Rm9ybWF0IH0gZnJvbSAnLi4vZm9ybWF0JztcblxuZXhwb3J0IGNvbnN0IHRlbmFudENvbW1hbmQgPSBuZXcgQ29tbWFuZCgndGVuYW50JylcbiAgLmRlc2NyaXB0aW9uKCdNYW5hZ2UgdGVuYW50cycpO1xuXG50ZW5hbnRDb21tYW5kXG4gIC5jb21tYW5kKCdjcmVhdGUnKVxuICAuZGVzY3JpcHRpb24oJ0NyZWF0ZSBhIG5ldyB0ZW5hbnQnKVxuICAucmVxdWlyZWRPcHRpb24oJy0tbmFtZSA8bmFtZT4nLCAnVGVuYW50IG5hbWUgKGxvd2VyY2FzZSwgaHlwaGVucyBhbGxvd2VkKScpXG4gIC5yZXF1aXJlZE9wdGlvbignLS1jb250YWN0LWVtYWlsIDxlbWFpbD4nLCAnQ29udGFjdCBlbWFpbCBhZGRyZXNzIGZvciB0aGUgdGVuYW50JylcbiAgLmFkZE9wdGlvbihuZXcgT3B0aW9uKCctLWZvcm1hdCA8Zm9ybWF0PicsICdPdXRwdXQgZm9ybWF0JykuY2hvaWNlcyhbJ3RhYmxlJywgJ2pzb24nLCAnY3N2J10pLmRlZmF1bHQoJ3RhYmxlJykpXG4gIC5hY3Rpb24oYXN5bmMgKG9wdHM6IHsgbmFtZTogc3RyaW5nOyBjb250YWN0RW1haWw6IHN0cmluZzsgZm9ybWF0OiBPdXRwdXRGb3JtYXQgfSkgPT4ge1xuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCBmZXRjaEpzb24oYXBpVXJsKCcvc3VwcG9ydC90ZW5hbnRzJyksIHtcbiAgICAgIG1ldGhvZDogJ1BPU1QnLFxuICAgICAgaGVhZGVyczogeyAnQ29udGVudC1UeXBlJzogJ2FwcGxpY2F0aW9uL2pzb24nIH0sXG4gICAgICBib2R5OiBKU09OLnN0cmluZ2lmeSh7IG5hbWU6IG9wdHMubmFtZSwgY29udGFjdEVtYWlsOiBvcHRzLmNvbnRhY3RFbWFpbCB9KSxcbiAgICB9KTtcbiAgICBmb3JtYXRPdXRwdXQoZGF0YSwgb3B0cy5mb3JtYXQsIHRlbmFudENvbHVtbnMpO1xuICB9KTtcblxudGVuYW50Q29tbWFuZFxuICAuY29tbWFuZCgnbGlzdCcpXG4gIC5kZXNjcmlwdGlvbignTGlzdCBhbGwgdGVuYW50cycpXG4gIC5hZGRPcHRpb24obmV3IE9wdGlvbignLS1mb3JtYXQgPGZvcm1hdD4nLCAnT3V0cHV0IGZvcm1hdCcpLmNob2ljZXMoWyd0YWJsZScsICdqc29uJywgJ2NzdiddKS5kZWZhdWx0KCd0YWJsZScpKVxuICAuYWN0aW9uKGFzeW5jIChvcHRzOiB7IGZvcm1hdDogT3V0cHV0Rm9ybWF0IH0pID0+IHtcbiAgICBjb25zdCBkYXRhID0gYXdhaXQgZmV0Y2hKc29uKGFwaVVybCgnL3N1cHBvcnQvdGVuYW50cycpKSBhcyB7IGl0ZW1zOiB1bmtub3duW10gfTtcbiAgICBmb3JtYXRPdXRwdXQob3B0cy5mb3JtYXQgPT09ICdqc29uJyA/IGRhdGEgOiBkYXRhLml0ZW1zLCBvcHRzLmZvcm1hdCwgdGVuYW50Q29sdW1ucyk7XG4gIH0pO1xuXG50ZW5hbnRDb21tYW5kXG4gIC5jb21tYW5kKCdnZXQgPGlkPicpXG4gIC5kZXNjcmlwdGlvbignR2V0IGEgdGVuYW50IGJ5IElEJylcbiAgLmFkZE9wdGlvbihuZXcgT3B0aW9uKCctLWZvcm1hdCA8Zm9ybWF0PicsICdPdXRwdXQgZm9ybWF0JykuY2hvaWNlcyhbJ3RhYmxlJywgJ2pzb24nLCAnY3N2J10pLmRlZmF1bHQoJ3RhYmxlJykpXG4gIC5hY3Rpb24oYXN5bmMgKGlkOiBzdHJpbmcsIG9wdHM6IHsgZm9ybWF0OiBPdXRwdXRGb3JtYXQgfSkgPT4ge1xuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCBmZXRjaEpzb24oYXBpVXJsKGAvc3VwcG9ydC90ZW5hbnRzLyR7aWR9YCkpO1xuICAgIGZvcm1hdE91dHB1dChkYXRhLCBvcHRzLmZvcm1hdCwgdGVuYW50Q29sdW1ucyk7XG4gIH0pO1xuXG50ZW5hbnRDb21tYW5kXG4gIC5jb21tYW5kKCdlbmFibGUgPGlkPicpXG4gIC5kZXNjcmlwdGlvbignRW5hYmxlIGEgdGVuYW50JylcbiAgLmFkZE9wdGlvbihuZXcgT3B0aW9uKCctLWZvcm1hdCA8Zm9ybWF0PicsICdPdXRwdXQgZm9ybWF0JykuY2hvaWNlcyhbJ3RhYmxlJywgJ2pzb24nLCAnY3N2J10pLmRlZmF1bHQoJ3RhYmxlJykpXG4gIC5hY3Rpb24oYXN5bmMgKGlkOiBzdHJpbmcsIG9wdHM6IHsgZm9ybWF0OiBPdXRwdXRGb3JtYXQgfSkgPT4ge1xuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCBmZXRjaEpzb24oYXBpVXJsKGAvc3VwcG9ydC90ZW5hbnRzLyR7aWR9L2VuYWJsZWApLCB7XG4gICAgICBtZXRob2Q6ICdQVVQnLFxuICAgIH0pO1xuICAgIGZvcm1hdE91dHB1dChkYXRhLCBvcHRzLmZvcm1hdCwgdGVuYW50Q29sdW1ucyk7XG4gIH0pO1xuXG50ZW5hbnRDb21tYW5kXG4gIC5jb21tYW5kKCdkaXNhYmxlIDxpZD4nKVxuICAuZGVzY3JpcHRpb24oJ0Rpc2FibGUgYSB0ZW5hbnQnKVxuICAuYWRkT3B0aW9uKG5ldyBPcHRpb24oJy0tZm9ybWF0IDxmb3JtYXQ+JywgJ091dHB1dCBmb3JtYXQnKS5jaG9pY2VzKFsndGFibGUnLCAnanNvbicsICdjc3YnXSkuZGVmYXVsdCgndGFibGUnKSlcbiAgLmFjdGlvbihhc3luYyAoaWQ6IHN0cmluZywgb3B0czogeyBmb3JtYXQ6IE91dHB1dEZvcm1hdCB9KSA9PiB7XG4gICAgY29uc3QgZGF0YSA9IGF3YWl0IGZldGNoSnNvbihhcGlVcmwoYC9zdXBwb3J0L3RlbmFudHMvJHtpZH0vZGlzYWJsZWApLCB7XG4gICAgICBtZXRob2Q6ICdQVVQnLFxuICAgIH0pO1xuICAgIGZvcm1hdE91dHB1dChkYXRhLCBvcHRzLmZvcm1hdCwgdGVuYW50Q29sdW1ucyk7XG4gIH0pO1xuXG50ZW5hbnRDb21tYW5kXG4gIC5jb21tYW5kKCdkZWxldGUgPGlkPicpXG4gIC5kZXNjcmlwdGlvbignRGVsZXRlIGEgdGVuYW50JylcbiAgLmFjdGlvbihhc3luYyAoaWQ6IHN0cmluZykgPT4ge1xuICAgIGF3YWl0IGZldGNoSnNvbihhcGlVcmwoYC9zdXBwb3J0L3RlbmFudHMvJHtpZH1gKSwge1xuICAgICAgbWV0aG9kOiAnREVMRVRFJyxcbiAgICB9KTtcbiAgICBjb25zb2xlLmxvZygnVGVuYW50IGRlbGV0ZWQgc3VjY2Vzc2Z1bGx5LicpO1xuICB9KTtcbiJdfQ==
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.userCommand = void 0;
|
|
4
|
+
const commander_1 = require("commander");
|
|
5
|
+
const auth_1 = require("../auth");
|
|
6
|
+
const format_js_1 = require("../format.js");
|
|
7
|
+
const ROLE_CHOICES = ['support', 'admin', 'analyst', 'viewer'];
|
|
8
|
+
const userColumns = [
|
|
9
|
+
{ header: 'ID', value: (r) => r.id },
|
|
10
|
+
{ header: 'Name', value: (r) => r.name },
|
|
11
|
+
{ header: 'Email', value: (r) => r.email },
|
|
12
|
+
{ header: 'Role', value: (r) => r.role ?? '' },
|
|
13
|
+
{ header: 'Status', value: (r) => r.status },
|
|
14
|
+
{ header: 'Tenant ID', value: (r) => r.tenantId },
|
|
15
|
+
{ header: 'Created At', value: (r) => r.createdAt },
|
|
16
|
+
];
|
|
17
|
+
exports.userCommand = new commander_1.Command('user')
|
|
18
|
+
.description('Manage users');
|
|
19
|
+
exports.userCommand
|
|
20
|
+
.command('create')
|
|
21
|
+
.description('Create a new user.')
|
|
22
|
+
.requiredOption('--name <name>', 'User name')
|
|
23
|
+
.requiredOption('--email <email>', 'User email')
|
|
24
|
+
.addOption(new commander_1.Option('--role <role>', 'RBAC role for the new user')
|
|
25
|
+
.choices([...ROLE_CHOICES])
|
|
26
|
+
.default('analyst'))
|
|
27
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
28
|
+
.action(async (opts) => {
|
|
29
|
+
const body = { name: opts.name, email: opts.email, role: opts.role };
|
|
30
|
+
if (process.env.GENI_TENANT_ID)
|
|
31
|
+
body.tenantId = process.env.GENI_TENANT_ID;
|
|
32
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)('/admin/users'), {
|
|
33
|
+
method: 'POST',
|
|
34
|
+
headers: { 'Content-Type': 'application/json' },
|
|
35
|
+
body: JSON.stringify(body),
|
|
36
|
+
});
|
|
37
|
+
(0, format_js_1.formatOutput)(data, opts.format, userColumns);
|
|
38
|
+
});
|
|
39
|
+
exports.userCommand
|
|
40
|
+
.command('list')
|
|
41
|
+
.description('List users')
|
|
42
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
43
|
+
.action(async (opts) => {
|
|
44
|
+
const qs = process.env.GENI_TENANT_ID ? `?tenantId=${encodeURIComponent(process.env.GENI_TENANT_ID)}` : '';
|
|
45
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/admin/users${qs}`));
|
|
46
|
+
(0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, userColumns);
|
|
47
|
+
});
|
|
48
|
+
exports.userCommand
|
|
49
|
+
.command('get <id>')
|
|
50
|
+
.description('Get a user by ID')
|
|
51
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
52
|
+
.action(async (id, opts) => {
|
|
53
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/admin/users/${id}`));
|
|
54
|
+
(0, format_js_1.formatOutput)(data, opts.format, userColumns);
|
|
55
|
+
});
|
|
56
|
+
exports.userCommand
|
|
57
|
+
.command('assign-role <userId>')
|
|
58
|
+
.description('Assign a new RBAC role to a user')
|
|
59
|
+
.requiredOption('--role <role>', `One of: ${ROLE_CHOICES.join(', ')}`, (v) => {
|
|
60
|
+
if (!ROLE_CHOICES.includes(v)) {
|
|
61
|
+
throw new Error(`role must be one of: ${ROLE_CHOICES.join(', ')}`);
|
|
62
|
+
}
|
|
63
|
+
return v;
|
|
64
|
+
})
|
|
65
|
+
.action(async (userId, opts) => {
|
|
66
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/admin/users/${encodeURIComponent(userId)}/role`), {
|
|
67
|
+
method: 'PUT',
|
|
68
|
+
headers: { 'Content-Type': 'application/json' },
|
|
69
|
+
body: JSON.stringify({ role: opts.role }),
|
|
70
|
+
});
|
|
71
|
+
console.log(JSON.stringify(data, null, 2));
|
|
72
|
+
});
|
|
73
|
+
exports.userCommand
|
|
74
|
+
.command('delete <id>')
|
|
75
|
+
.description('Delete a user by ID')
|
|
76
|
+
.action(async (id) => {
|
|
77
|
+
await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/admin/users/${encodeURIComponent(id)}`), {
|
|
78
|
+
method: 'DELETE',
|
|
79
|
+
});
|
|
80
|
+
console.log('User deleted successfully.');
|
|
81
|
+
});
|
|
82
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"user.js","sourceRoot":"","sources":["../../src/commands/user.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander';\nimport { apiUrl, fetchJson } from '../auth';\nimport { formatOutput, type OutputFormat } from '../format.js';\n\nconst ROLE_CHOICES = ['support', 'admin', 'analyst', 'viewer'] as const;\n\nconst userColumns = [\n  { header: 'ID', value: (r: Record<string, unknown>) => r.id },\n  { header: 'Name', value: (r: Record<string, unknown>) => r.name },\n  { header: 'Email', value: (r: Record<string, unknown>) => r.email },\n  { header: 'Role', value: (r: Record<string, unknown>) => r.role ?? '' },\n  { header: 'Status', value: (r: Record<string, unknown>) => r.status },\n  { header: 'Tenant ID', value: (r: Record<string, unknown>) => r.tenantId },\n  { header: 'Created At', value: (r: Record<string, unknown>) => r.createdAt },\n];\n\nexport const userCommand = new Command('user')\n  .description('Manage users');\n\nuserCommand\n  .command('create')\n  .description('Create a new user.')\n  .requiredOption('--name <name>', 'User name')\n  .requiredOption('--email <email>', 'User email')\n  .addOption(\n    new Option('--role <role>', 'RBAC role for the new user')\n      .choices([...ROLE_CHOICES])\n      .default('analyst'),\n  )\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: {\n    name: string;\n    email: string;\n    role?: (typeof ROLE_CHOICES)[number];\n    format: OutputFormat;\n  }) => {\n    const body: Record<string, unknown> = { name: opts.name, email: opts.email, role: opts.role };\n    if (process.env.GENI_TENANT_ID) body.tenantId = process.env.GENI_TENANT_ID;\n\n    const data = await fetchJson(apiUrl('/admin/users'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify(body),\n    });\n    formatOutput(data, opts.format, userColumns);\n  });\n\nuserCommand\n  .command('list')\n  .description('List users')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { format: OutputFormat }) => {\n    const qs = process.env.GENI_TENANT_ID ? `?tenantId=${encodeURIComponent(process.env.GENI_TENANT_ID)}` : '';\n    const data = await fetchJson(apiUrl(`/admin/users${qs}`)) as { items: unknown[] };\n    formatOutput(opts.format === 'json' ? data : data.items, opts.format, userColumns);\n  });\n\nuserCommand\n  .command('get <id>')\n  .description('Get a user by ID')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (id: string, opts: { format: OutputFormat }) => {\n    const data = await fetchJson(apiUrl(`/admin/users/${id}`));\n    formatOutput(data, opts.format, userColumns);\n  });\n\nuserCommand\n  .command('assign-role <userId>')\n  .description('Assign a new RBAC role to a user')\n  .requiredOption(\n    '--role <role>',\n    `One of: ${ROLE_CHOICES.join(', ')}`,\n    (v: string) => {\n      if (!ROLE_CHOICES.includes(v as (typeof ROLE_CHOICES)[number])) {\n        throw new Error(`role must be one of: ${ROLE_CHOICES.join(', ')}`);\n      }\n      return v;\n    },\n  )\n  .action(async (userId: string, opts: { role: string }) => {\n    const data = await fetchJson(apiUrl(`/admin/users/${encodeURIComponent(userId)}/role`), {\n      method: 'PUT',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify({ role: opts.role }),\n    });\n    console.log(JSON.stringify(data, null, 2));\n  });\n\nuserCommand\n  .command('delete <id>')\n  .description('Delete a user by ID')\n  .action(async (id: string) => {\n    await fetchJson(apiUrl(`/admin/users/${encodeURIComponent(id)}`), {\n      method: 'DELETE',\n    });\n    console.log('User deleted successfully.');\n  });\n"]}
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.workflowCommand = void 0;
|
|
7
|
+
const commander_1 = require("commander");
|
|
8
|
+
const fs_1 = __importDefault(require("fs"));
|
|
9
|
+
const path_1 = __importDefault(require("path"));
|
|
10
|
+
const auth_1 = require("../auth");
|
|
11
|
+
const errors_1 = require("../errors");
|
|
12
|
+
const format_js_1 = require("../format.js");
|
|
13
|
+
function parseTimeToSeconds(value) {
|
|
14
|
+
const parts = value.split(':').map(Number);
|
|
15
|
+
if (parts.length !== 3 || parts.some(isNaN)) {
|
|
16
|
+
throw (0, errors_1.invalidArgument)('Time must be in DD:HH:MM format (days:hours:minutes).', 'Example: 00:12:30');
|
|
17
|
+
}
|
|
18
|
+
const [days, hours, minutes] = parts;
|
|
19
|
+
return (days * 86400) + (hours * 3600) + (minutes * 60);
|
|
20
|
+
}
|
|
21
|
+
function guessFileType(filePath) {
|
|
22
|
+
const lower = filePath.toLowerCase();
|
|
23
|
+
if (lower.endsWith('.nf'))
|
|
24
|
+
return 'TEXT';
|
|
25
|
+
if (lower.endsWith('.zip'))
|
|
26
|
+
return 'ZIP';
|
|
27
|
+
throw (0, errors_1.invalidArgument)(`Cannot determine file type from extension. File must be .nf (TEXT) or .zip (ZIP).`, `Provided: ${filePath}`);
|
|
28
|
+
}
|
|
29
|
+
exports.workflowCommand = new commander_1.Command('workflow')
|
|
30
|
+
.description('Manage workflows');
|
|
31
|
+
exports.workflowCommand
|
|
32
|
+
.command('list')
|
|
33
|
+
.description('List all registered workflows')
|
|
34
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
35
|
+
.action(async (opts) => {
|
|
36
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)('/workflows'));
|
|
37
|
+
(0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.workflowColumns);
|
|
38
|
+
});
|
|
39
|
+
exports.workflowCommand
|
|
40
|
+
.command('get <name>')
|
|
41
|
+
.description('Get a workflow by name')
|
|
42
|
+
.option('--version <version>', 'Workflow version (defaults to latest)')
|
|
43
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
44
|
+
.action(async (name, opts) => {
|
|
45
|
+
const query = opts.version ? `?version=${opts.version}` : '';
|
|
46
|
+
const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/workflows/${name}${query}`));
|
|
47
|
+
(0, format_js_1.formatOutput)(data, opts.format, format_js_1.workflowColumns);
|
|
48
|
+
});
|
|
49
|
+
exports.workflowCommand
|
|
50
|
+
.command('create')
|
|
51
|
+
.description('Register a new workflow (.nf file or .zip archive)')
|
|
52
|
+
.requiredOption('--workflow-name <name>', 'Workflow name')
|
|
53
|
+
.requiredOption('--workflow-version <alias>', 'Workflow version (alias)')
|
|
54
|
+
.requiredOption('--file <file>', 'Workflow file path (.nf or .zip)')
|
|
55
|
+
.option('--file-type <type>', 'File type (TEXT for .nf, ZIP for .zip archives; auto-guessed from extension if not provided)')
|
|
56
|
+
.option('--alert-time <DD:HH:MM>', 'Default alert time for submissions (DD:HH:MM)')
|
|
57
|
+
.option('--cancel-time <DD:HH:MM>', 'Default cancel time for submissions (DD:HH:MM)')
|
|
58
|
+
.addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
|
|
59
|
+
.action(async (opts) => {
|
|
60
|
+
const form = new FormData();
|
|
61
|
+
form.append('name', opts.workflowName);
|
|
62
|
+
form.append('version', opts.workflowVersion);
|
|
63
|
+
form.append('engine', 'nextflow');
|
|
64
|
+
const fileType = opts.fileType?.toUpperCase() || guessFileType(opts.file);
|
|
65
|
+
form.append('fileType', fileType);
|
|
66
|
+
if (opts.alertTime)
|
|
67
|
+
form.append('alertTimeSeconds', String(parseTimeToSeconds(opts.alertTime)));
|
|
68
|
+
if (opts.cancelTime)
|
|
69
|
+
form.append('cancelTimeSeconds', String(parseTimeToSeconds(opts.cancelTime)));
|
|
70
|
+
const fileBuffer = await fs_1.default.promises.readFile(opts.file);
|
|
71
|
+
form.append('file', new Blob([fileBuffer]), path_1.default.basename(opts.file));
|
|
72
|
+
const res = await fetch((0, auth_1.apiUrl)('/workflows'), {
|
|
73
|
+
method: 'POST',
|
|
74
|
+
headers: (0, auth_1.getAuthHeaders)(),
|
|
75
|
+
body: form,
|
|
76
|
+
});
|
|
77
|
+
const data = await (0, auth_1.readJsonResponse)(res, { method: 'POST', url: (0, auth_1.apiUrl)('/workflows') });
|
|
78
|
+
(0, format_js_1.formatOutput)(data, opts.format, format_js_1.workflowColumns);
|
|
79
|
+
});
|
|
80
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workflow.js","sourceRoot":"","sources":["../../src/commands/workflow.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander'\nimport fs from 'fs'\nimport path from 'path'\nimport { apiUrl, fetchJson, getAuthHeaders, readJsonResponse } from '../auth'\nimport { invalidArgument } from '../errors'\nimport { formatOutput, workflowColumns, type OutputFormat } from '../format.js'\n\nfunction parseTimeToSeconds(value: string): number {\n  const parts = value.split(':').map(Number)\n  if (parts.length !== 3 || parts.some(isNaN)) {\n    throw invalidArgument(\n      'Time must be in DD:HH:MM format (days:hours:minutes).',\n      'Example: 00:12:30',\n    )\n  }\n  const [days, hours, minutes] = parts\n  return (days * 86400) + (hours * 3600) + (minutes * 60)\n}\n\nfunction guessFileType(filePath: string): 'TEXT' | 'ZIP' {\n  const lower = filePath.toLowerCase()\n  if (lower.endsWith('.nf')) return 'TEXT'\n  if (lower.endsWith('.zip')) return 'ZIP'\n  throw invalidArgument(\n    `Cannot determine file type from extension. File must be .nf (TEXT) or .zip (ZIP).`,\n    `Provided: ${filePath}`,\n  )\n}\n\nexport const workflowCommand = new Command('workflow')\n  .description('Manage workflows')\n\nworkflowCommand\n  .command('list')\n  .description('List all registered workflows')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { format: OutputFormat }) => {\n    const data = await fetchJson(apiUrl('/workflows')) as { items: unknown[] }\n    formatOutput(opts.format === 'json' ? data : data.items, opts.format, workflowColumns)\n  })\n\nworkflowCommand\n  .command('get <name>')\n  .description('Get a workflow by name')\n  .option('--version <version>', 'Workflow version (defaults to latest)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (name: string, opts: { version?: string; format: OutputFormat }) => {\n    const query = opts.version ? `?version=${opts.version}` : ''\n    const data = await fetchJson(apiUrl(`/workflows/${name}${query}`))\n    formatOutput(data, opts.format, workflowColumns)\n  })\n\nworkflowCommand\n  .command('create')\n  .description('Register a new workflow (.nf file or .zip archive)')\n  .requiredOption('--workflow-name <name>', 'Workflow name')\n  .requiredOption('--workflow-version <alias>', 'Workflow version (alias)')\n  .requiredOption('--file <file>', 'Workflow file path (.nf or .zip)')\n  .option('--file-type <type>', 'File type (TEXT for .nf, ZIP for .zip archives; auto-guessed from extension if not provided)')\n  .option('--alert-time <DD:HH:MM>', 'Default alert time for submissions (DD:HH:MM)')\n  .option('--cancel-time <DD:HH:MM>', 'Default cancel time for submissions (DD:HH:MM)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { workflowName: string; workflowVersion: string; file: string; fileType?: string; alertTime?: string; cancelTime?: string; format: OutputFormat }) => {\n    const form = new FormData()\n    form.append('name', opts.workflowName)\n    form.append('version', opts.workflowVersion)\n    form.append('engine', 'nextflow')\n    \n    const fileType = opts.fileType?.toUpperCase() || guessFileType(opts.file)\n    form.append('fileType', fileType)\n    \n    if (opts.alertTime) form.append('alertTimeSeconds', String(parseTimeToSeconds(opts.alertTime)))\n    if (opts.cancelTime) form.append('cancelTimeSeconds', String(parseTimeToSeconds(opts.cancelTime)))\n    const fileBuffer = await fs.promises.readFile(opts.file)\n    form.append('file', new Blob([fileBuffer]), path.basename(opts.file))\n\n    const res = await fetch(apiUrl('/workflows'), {\n      method: 'POST',\n      headers: getAuthHeaders(),\n      body: form,\n    })\n    const data = await readJsonResponse(res, { method: 'POST', url: apiUrl('/workflows') })\n    formatOutput(data, opts.format, workflowColumns)\n  })\n"]}
|
package/dist/errors.d.ts
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
export type CliErrorCode = 'INVALID_ARGUMENT' | 'AUTH' | 'NOT_FOUND' | 'UPSTREAM' | 'INTERNAL';
|
|
2
|
+
export declare class CliError extends Error {
|
|
3
|
+
code: CliErrorCode;
|
|
4
|
+
exitCode: number;
|
|
5
|
+
hint?: string;
|
|
6
|
+
details?: unknown;
|
|
7
|
+
constructor(params: {
|
|
8
|
+
code: CliErrorCode;
|
|
9
|
+
message: string;
|
|
10
|
+
exitCode: number;
|
|
11
|
+
hint?: string;
|
|
12
|
+
details?: unknown;
|
|
13
|
+
cause?: unknown;
|
|
14
|
+
});
|
|
15
|
+
}
|
|
16
|
+
export declare class HttpRequestError extends Error {
|
|
17
|
+
status: number;
|
|
18
|
+
method: string;
|
|
19
|
+
url: string;
|
|
20
|
+
body: string;
|
|
21
|
+
parsedBody?: unknown;
|
|
22
|
+
constructor(params: {
|
|
23
|
+
status: number;
|
|
24
|
+
method: string;
|
|
25
|
+
url: string;
|
|
26
|
+
body: string;
|
|
27
|
+
parsedBody?: unknown;
|
|
28
|
+
});
|
|
29
|
+
}
|
|
30
|
+
export declare function invalidArgument(message: string, hint?: string): CliError;
|
|
31
|
+
export declare function notFound(message: string): CliError;
|
|
32
|
+
export declare function isDebugEnabled(argv?: string[]): boolean;
|
|
33
|
+
export declare function wantsJsonError(argv?: string[]): boolean;
|
|
34
|
+
export declare function toCliError(err: unknown): CliError;
|
|
35
|
+
export declare function printCliError(err: CliError, opts: {
|
|
36
|
+
json: boolean;
|
|
37
|
+
debug: boolean;
|
|
38
|
+
}): void;
|