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.
Files changed (58) hide show
  1. package/dist/auth.d.ts +8 -0
  2. package/dist/auth.js +76 -0
  3. package/dist/aws/clients.d.ts +18 -0
  4. package/dist/aws/clients.js +34 -0
  5. package/dist/commands/activate.d.ts +2 -0
  6. package/dist/commands/activate.js +20 -0
  7. package/dist/commands/activity-log.d.ts +2 -0
  8. package/dist/commands/activity-log.js +31 -0
  9. package/dist/commands/api-token.d.ts +2 -0
  10. package/dist/commands/api-token.js +96 -0
  11. package/dist/commands/auth.d.ts +2 -0
  12. package/dist/commands/auth.js +55 -0
  13. package/dist/commands/engine.d.ts +2 -0
  14. package/dist/commands/engine.js +83 -0
  15. package/dist/commands/environment.d.ts +2 -0
  16. package/dist/commands/environment.js +69 -0
  17. package/dist/commands/image.d.ts +2 -0
  18. package/dist/commands/image.js +40 -0
  19. package/dist/commands/instance.d.ts +2 -0
  20. package/dist/commands/instance.js +39 -0
  21. package/dist/commands/log.d.ts +3 -0
  22. package/dist/commands/log.js +43 -0
  23. package/dist/commands/login.d.ts +2 -0
  24. package/dist/commands/login.js +24 -0
  25. package/dist/commands/plugin.d.ts +2 -0
  26. package/dist/commands/plugin.js +80 -0
  27. package/dist/commands/queue.d.ts +2 -0
  28. package/dist/commands/queue.js +74 -0
  29. package/dist/commands/registry.d.ts +2 -0
  30. package/dist/commands/registry.js +72 -0
  31. package/dist/commands/setup/create.d.ts +2 -0
  32. package/dist/commands/setup/create.js +254 -0
  33. package/dist/commands/setup/delete.d.ts +2 -0
  34. package/dist/commands/setup/delete.js +97 -0
  35. package/dist/commands/setup/status.d.ts +2 -0
  36. package/dist/commands/setup/status.js +46 -0
  37. package/dist/commands/setup.d.ts +2 -0
  38. package/dist/commands/setup.js +13 -0
  39. package/dist/commands/storage.d.ts +2 -0
  40. package/dist/commands/storage.js +67 -0
  41. package/dist/commands/submission.d.ts +2 -0
  42. package/dist/commands/submission.js +87 -0
  43. package/dist/commands/task.d.ts +2 -0
  44. package/dist/commands/task.js +21 -0
  45. package/dist/commands/tenant.d.ts +2 -0
  46. package/dist/commands/tenant.js +68 -0
  47. package/dist/commands/user.d.ts +2 -0
  48. package/dist/commands/user.js +82 -0
  49. package/dist/commands/workflow.d.ts +2 -0
  50. package/dist/commands/workflow.js +80 -0
  51. package/dist/errors.d.ts +38 -0
  52. package/dist/errors.js +194 -0
  53. package/dist/format.d.ts +22 -0
  54. package/dist/format.js +155 -0
  55. package/dist/index.d.ts +2 -0
  56. package/dist/index.js +91 -0
  57. package/dist/templates/setup.yaml +503 -0
  58. 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,2 @@
1
+ import { Command } from 'commander';
2
+ export declare const taskCommand: Command;
@@ -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,2 @@
1
+ import { Command } from 'commander';
2
+ export declare const tenantCommand: Command;
@@ -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,2 @@
1
+ import { Command } from 'commander';
2
+ export declare const userCommand: Command;
@@ -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,2 @@
1
+ import { Command } from 'commander';
2
+ export declare const workflowCommand: Command;
@@ -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"]}
@@ -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;