geni-bioinfo 0.1.2 → 0.1.4

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.
@@ -13,6 +13,7 @@ exports.activityLogCommand
13
13
  .option('--entity-type <type>', 'Filter by entity type')
14
14
  .option('--entity-id <id>', 'Filter by entity ID')
15
15
  .option('--user-id <id>', 'Filter by user ID')
16
+ .option('--limit <n>', 'Maximum number of results to return', '10')
16
17
  .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
17
18
  .action(async (opts) => {
18
19
  const params = new URLSearchParams();
@@ -24,8 +25,8 @@ exports.activityLogCommand
24
25
  params.set('entityId', opts.entityId);
25
26
  if (opts.userId)
26
27
  params.set('userId', opts.userId);
27
- const query = params.toString() ? `?${params.toString()}` : '';
28
- const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/activity-logs${query}`));
28
+ params.set('limit', opts.limit);
29
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/activity-logs?${params}`));
29
30
  (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.activityLogColumns);
30
31
  });
31
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWN0aXZpdHktbG9nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbW1hbmRzL2FjdGl2aXR5LWxvZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5Q0FBMkM7QUFDM0Msa0NBQTJDO0FBQzNDLDRDQUFrRjtBQUVyRSxRQUFBLGtCQUFrQixHQUFHLElBQUksbUJBQU8sQ0FBQyxjQUFjLENBQUM7S0FDMUQsV0FBVyxDQUFDLHNCQUFzQixDQUFDLENBQUE7QUFFdEMsMEJBQWtCO0tBQ2YsT0FBTyxDQUFDLE1BQU0sQ0FBQztLQUNmLFdBQVcsQ0FBQyxvQkFBb0IsQ0FBQztLQUNqQyxNQUFNLENBQUMsbUJBQW1CLEVBQUUsMkNBQTJDLENBQUM7S0FDeEUsTUFBTSxDQUFDLHNCQUFzQixFQUFFLHVCQUF1QixDQUFDO0tBQ3ZELE1BQU0sQ0FBQyxrQkFBa0IsRUFBRSxxQkFBcUIsQ0FBQztLQUNqRCxNQUFNLENBQUMsZ0JBQWdCLEVBQUUsbUJBQW1CLENBQUM7S0FDN0MsU0FBUyxDQUFDLElBQUksa0JBQU0sQ0FBQyxtQkFBbUIsRUFBRSxlQUFlLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0tBQzlHLE1BQU0sQ0FBQyxLQUFLLEVBQUUsSUFNZCxFQUFFLEVBQUU7SUFDSCxNQUFNLE1BQU0sR0FBRyxJQUFJLGVBQWUsRUFBRSxDQUFBO0lBQ3BDLElBQUksSUFBSSxDQUFDLE1BQU07UUFBRSxNQUFNLENBQUMsR0FBRyxDQUFDLFFBQVEsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDbEQsSUFBSSxJQUFJLENBQUMsVUFBVTtRQUFFLE1BQU0sQ0FBQyxHQUFHLENBQUMsWUFBWSxFQUFFLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQTtJQUM5RCxJQUFJLElBQUksQ0FBQyxRQUFRO1FBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxVQUFVLEVBQUUsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFBO0lBQ3hELElBQUksSUFBSSxDQUFDLE1BQU07UUFBRSxNQUFNLENBQUMsR0FBRyxDQUFDLFFBQVEsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDbEQsTUFBTSxLQUFLLEdBQUcsTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLE1BQU0sQ0FBQyxRQUFRLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUE7SUFDOUQsTUFBTSxJQUFJLEdBQUcsTUFBTSxJQUFBLGdCQUFTLEVBQUMsSUFBQSxhQUFNLEVBQUMsaUJBQWlCLEtBQUssRUFBRSxDQUFDLENBQXlCLENBQUE7SUFDdEYsSUFBQSx3QkFBWSxFQUFDLElBQUksQ0FBQyxNQUFNLEtBQUssTUFBTSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBRSw4QkFBa0IsQ0FBQyxDQUFBO0FBQzNGLENBQUMsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbWFuZCwgT3B0aW9uIH0gZnJvbSAnY29tbWFuZGVyJ1xuaW1wb3J0IHsgYXBpVXJsLCBmZXRjaEpzb24gfSBmcm9tICcuLi9hdXRoJ1xuaW1wb3J0IHsgZm9ybWF0T3V0cHV0LCBhY3Rpdml0eUxvZ0NvbHVtbnMsIHR5cGUgT3V0cHV0Rm9ybWF0IH0gZnJvbSAnLi4vZm9ybWF0LmpzJ1xuXG5leHBvcnQgY29uc3QgYWN0aXZpdHlMb2dDb21tYW5kID0gbmV3IENvbW1hbmQoJ2FjdGl2aXR5LWxvZycpXG4gIC5kZXNjcmlwdGlvbignTWFuYWdlIGFjdGl2aXR5IGxvZ3MnKVxuXG5hY3Rpdml0eUxvZ0NvbW1hbmRcbiAgLmNvbW1hbmQoJ2xpc3QnKVxuICAuZGVzY3JpcHRpb24oJ0xpc3QgYWN0aXZpdHkgbG9ncycpXG4gIC5vcHRpb24oJy0tYWN0aW9uIDxhY3Rpb24+JywgJ0ZpbHRlciBieSBhY3Rpb24gKENSRUFURSwgVVBEQVRFLCBERUxFVEUpJylcbiAgLm9wdGlvbignLS1lbnRpdHktdHlwZSA8dHlwZT4nLCAnRmlsdGVyIGJ5IGVudGl0eSB0eXBlJylcbiAgLm9wdGlvbignLS1lbnRpdHktaWQgPGlkPicsICdGaWx0ZXIgYnkgZW50aXR5IElEJylcbiAgLm9wdGlvbignLS11c2VyLWlkIDxpZD4nLCAnRmlsdGVyIGJ5IHVzZXIgSUQnKVxuICAuYWRkT3B0aW9uKG5ldyBPcHRpb24oJy0tZm9ybWF0IDxmb3JtYXQ+JywgJ091dHB1dCBmb3JtYXQnKS5jaG9pY2VzKFsndGFibGUnLCAnanNvbicsICdjc3YnXSkuZGVmYXVsdCgndGFibGUnKSlcbiAgLmFjdGlvbihhc3luYyAob3B0czoge1xuICAgIGFjdGlvbj86IHN0cmluZ1xuICAgIGVudGl0eVR5cGU/OiBzdHJpbmdcbiAgICBlbnRpdHlJZD86IHN0cmluZ1xuICAgIHVzZXJJZD86IHN0cmluZ1xuICAgIGZvcm1hdDogT3V0cHV0Rm9ybWF0XG4gIH0pID0+IHtcbiAgICBjb25zdCBwYXJhbXMgPSBuZXcgVVJMU2VhcmNoUGFyYW1zKClcbiAgICBpZiAob3B0cy5hY3Rpb24pIHBhcmFtcy5zZXQoJ2FjdGlvbicsIG9wdHMuYWN0aW9uKVxuICAgIGlmIChvcHRzLmVudGl0eVR5cGUpIHBhcmFtcy5zZXQoJ2VudGl0eVR5cGUnLCBvcHRzLmVudGl0eVR5cGUpXG4gICAgaWYgKG9wdHMuZW50aXR5SWQpIHBhcmFtcy5zZXQoJ2VudGl0eUlkJywgb3B0cy5lbnRpdHlJZClcbiAgICBpZiAob3B0cy51c2VySWQpIHBhcmFtcy5zZXQoJ3VzZXJJZCcsIG9wdHMudXNlcklkKVxuICAgIGNvbnN0IHF1ZXJ5ID0gcGFyYW1zLnRvU3RyaW5nKCkgPyBgPyR7cGFyYW1zLnRvU3RyaW5nKCl9YCA6ICcnXG4gICAgY29uc3QgZGF0YSA9IGF3YWl0IGZldGNoSnNvbihhcGlVcmwoYC9hY3Rpdml0eS1sb2dzJHtxdWVyeX1gKSkgYXMgeyBpdGVtczogdW5rbm93bltdIH1cbiAgICBmb3JtYXRPdXRwdXQob3B0cy5mb3JtYXQgPT09ICdqc29uJyA/IGRhdGEgOiBkYXRhLml0ZW1zLCBvcHRzLmZvcm1hdCwgYWN0aXZpdHlMb2dDb2x1bW5zKVxuICB9KVxuIl19
32
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWN0aXZpdHktbG9nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbW1hbmRzL2FjdGl2aXR5LWxvZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5Q0FBMkM7QUFDM0Msa0NBQTJDO0FBQzNDLDRDQUFrRjtBQUVyRSxRQUFBLGtCQUFrQixHQUFHLElBQUksbUJBQU8sQ0FBQyxjQUFjLENBQUM7S0FDMUQsV0FBVyxDQUFDLHNCQUFzQixDQUFDLENBQUE7QUFFdEMsMEJBQWtCO0tBQ2YsT0FBTyxDQUFDLE1BQU0sQ0FBQztLQUNmLFdBQVcsQ0FBQyxvQkFBb0IsQ0FBQztLQUNqQyxNQUFNLENBQUMsbUJBQW1CLEVBQUUsMkNBQTJDLENBQUM7S0FDeEUsTUFBTSxDQUFDLHNCQUFzQixFQUFFLHVCQUF1QixDQUFDO0tBQ3ZELE1BQU0sQ0FBQyxrQkFBa0IsRUFBRSxxQkFBcUIsQ0FBQztLQUNqRCxNQUFNLENBQUMsZ0JBQWdCLEVBQUUsbUJBQW1CLENBQUM7S0FDN0MsTUFBTSxDQUFDLGFBQWEsRUFBRSxxQ0FBcUMsRUFBRSxJQUFJLENBQUM7S0FDbEUsU0FBUyxDQUFDLElBQUksa0JBQU0sQ0FBQyxtQkFBbUIsRUFBRSxlQUFlLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0tBQzlHLE1BQU0sQ0FBQyxLQUFLLEVBQUUsSUFPZCxFQUFFLEVBQUU7SUFDSCxNQUFNLE1BQU0sR0FBRyxJQUFJLGVBQWUsRUFBRSxDQUFBO0lBQ3BDLElBQUksSUFBSSxDQUFDLE1BQU07UUFBRSxNQUFNLENBQUMsR0FBRyxDQUFDLFFBQVEsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDbEQsSUFBSSxJQUFJLENBQUMsVUFBVTtRQUFFLE1BQU0sQ0FBQyxHQUFHLENBQUMsWUFBWSxFQUFFLElBQUksQ0FBQyxVQUFVLENBQUMsQ0FBQTtJQUM5RCxJQUFJLElBQUksQ0FBQyxRQUFRO1FBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxVQUFVLEVBQUUsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFBO0lBQ3hELElBQUksSUFBSSxDQUFDLE1BQU07UUFBRSxNQUFNLENBQUMsR0FBRyxDQUFDLFFBQVEsRUFBRSxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDbEQsTUFBTSxDQUFDLEdBQUcsQ0FBQyxPQUFPLEVBQUUsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFBO0lBQy9CLE1BQU0sSUFBSSxHQUFHLE1BQU0sSUFBQSxnQkFBUyxFQUFDLElBQUEsYUFBTSxFQUFDLGtCQUFrQixNQUFNLEVBQUUsQ0FBQyxDQUF5QixDQUFBO0lBQ3hGLElBQUEsd0JBQVksRUFBQyxJQUFJLENBQUMsTUFBTSxLQUFLLE1BQU0sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLElBQUksQ0FBQyxNQUFNLEVBQUUsOEJBQWtCLENBQUMsQ0FBQTtBQUMzRixDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1hbmQsIE9wdGlvbiB9IGZyb20gJ2NvbW1hbmRlcidcbmltcG9ydCB7IGFwaVVybCwgZmV0Y2hKc29uIH0gZnJvbSAnLi4vYXV0aCdcbmltcG9ydCB7IGZvcm1hdE91dHB1dCwgYWN0aXZpdHlMb2dDb2x1bW5zLCB0eXBlIE91dHB1dEZvcm1hdCB9IGZyb20gJy4uL2Zvcm1hdC5qcydcblxuZXhwb3J0IGNvbnN0IGFjdGl2aXR5TG9nQ29tbWFuZCA9IG5ldyBDb21tYW5kKCdhY3Rpdml0eS1sb2cnKVxuICAuZGVzY3JpcHRpb24oJ01hbmFnZSBhY3Rpdml0eSBsb2dzJylcblxuYWN0aXZpdHlMb2dDb21tYW5kXG4gIC5jb21tYW5kKCdsaXN0JylcbiAgLmRlc2NyaXB0aW9uKCdMaXN0IGFjdGl2aXR5IGxvZ3MnKVxuICAub3B0aW9uKCctLWFjdGlvbiA8YWN0aW9uPicsICdGaWx0ZXIgYnkgYWN0aW9uIChDUkVBVEUsIFVQREFURSwgREVMRVRFKScpXG4gIC5vcHRpb24oJy0tZW50aXR5LXR5cGUgPHR5cGU+JywgJ0ZpbHRlciBieSBlbnRpdHkgdHlwZScpXG4gIC5vcHRpb24oJy0tZW50aXR5LWlkIDxpZD4nLCAnRmlsdGVyIGJ5IGVudGl0eSBJRCcpXG4gIC5vcHRpb24oJy0tdXNlci1pZCA8aWQ+JywgJ0ZpbHRlciBieSB1c2VyIElEJylcbiAgLm9wdGlvbignLS1saW1pdCA8bj4nLCAnTWF4aW11bSBudW1iZXIgb2YgcmVzdWx0cyB0byByZXR1cm4nLCAnMTAnKVxuICAuYWRkT3B0aW9uKG5ldyBPcHRpb24oJy0tZm9ybWF0IDxmb3JtYXQ+JywgJ091dHB1dCBmb3JtYXQnKS5jaG9pY2VzKFsndGFibGUnLCAnanNvbicsICdjc3YnXSkuZGVmYXVsdCgndGFibGUnKSlcbiAgLmFjdGlvbihhc3luYyAob3B0czoge1xuICAgIGFjdGlvbj86IHN0cmluZ1xuICAgIGVudGl0eVR5cGU/OiBzdHJpbmdcbiAgICBlbnRpdHlJZD86IHN0cmluZ1xuICAgIHVzZXJJZD86IHN0cmluZ1xuICAgIGxpbWl0OiBzdHJpbmdcbiAgICBmb3JtYXQ6IE91dHB1dEZvcm1hdFxuICB9KSA9PiB7XG4gICAgY29uc3QgcGFyYW1zID0gbmV3IFVSTFNlYXJjaFBhcmFtcygpXG4gICAgaWYgKG9wdHMuYWN0aW9uKSBwYXJhbXMuc2V0KCdhY3Rpb24nLCBvcHRzLmFjdGlvbilcbiAgICBpZiAob3B0cy5lbnRpdHlUeXBlKSBwYXJhbXMuc2V0KCdlbnRpdHlUeXBlJywgb3B0cy5lbnRpdHlUeXBlKVxuICAgIGlmIChvcHRzLmVudGl0eUlkKSBwYXJhbXMuc2V0KCdlbnRpdHlJZCcsIG9wdHMuZW50aXR5SWQpXG4gICAgaWYgKG9wdHMudXNlcklkKSBwYXJhbXMuc2V0KCd1c2VySWQnLCBvcHRzLnVzZXJJZClcbiAgICBwYXJhbXMuc2V0KCdsaW1pdCcsIG9wdHMubGltaXQpXG4gICAgY29uc3QgZGF0YSA9IGF3YWl0IGZldGNoSnNvbihhcGlVcmwoYC9hY3Rpdml0eS1sb2dzPyR7cGFyYW1zfWApKSBhcyB7IGl0ZW1zOiB1bmtub3duW10gfVxuICAgIGZvcm1hdE91dHB1dChvcHRzLmZvcm1hdCA9PT0gJ2pzb24nID8gZGF0YSA6IGRhdGEuaXRlbXMsIG9wdHMuZm9ybWF0LCBhY3Rpdml0eUxvZ0NvbHVtbnMpXG4gIH0pXG4iXX0=
@@ -0,0 +1,2 @@
1
+ import { Command } from 'commander';
2
+ export declare const analyticsCommand: Command;
@@ -0,0 +1,79 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.analyticsCommand = void 0;
4
+ const commander_1 = require("commander");
5
+ const auth_1 = require("../auth");
6
+ const format_js_1 = require("../format.js");
7
+ function addCommonOptions(cmd) {
8
+ return cmd
9
+ .option('--from <iso8601>', 'Start of date range (ISO 8601, e.g. 2025-01-01T00:00:00Z)')
10
+ .option('--to <iso8601>', 'End of date range (ISO 8601, exclusive)')
11
+ .option('--submission-status <status>', 'Filter by submission status (e.g. SUCCEEDED, FAILED, RUNNING)');
12
+ }
13
+ function buildParams(opts) {
14
+ const params = new URLSearchParams();
15
+ if (opts.from)
16
+ params.set('from', opts.from);
17
+ if (opts.to)
18
+ params.set('to', opts.to);
19
+ if (opts.submissionStatus)
20
+ params.set('submissionStatus', opts.submissionStatus);
21
+ return params;
22
+ }
23
+ function qs(params) {
24
+ const s = params.toString();
25
+ return s ? `?${s}` : '';
26
+ }
27
+ exports.analyticsCommand = new commander_1.Command('analytics')
28
+ .description('View cost analytics');
29
+ addCommonOptions(exports.analyticsCommand
30
+ .command('summary')
31
+ .description('Overall cost totals for the tenant')
32
+ .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))).action(async (opts) => {
33
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/analytics/costs/summary${qs(buildParams(opts))}`));
34
+ (0, format_js_1.formatOutput)(data, opts.format, format_js_1.analyticsSummaryColumns);
35
+ });
36
+ addCommonOptions(exports.analyticsCommand
37
+ .command('by-submission')
38
+ .description('Per-submission cost breakdown')
39
+ .option('--limit <n>', 'Max results (1–200)', '50')
40
+ .option('--offset <n>', 'Pagination offset', '0')
41
+ .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))).action(async (opts) => {
42
+ const params = buildParams(opts);
43
+ params.set('limit', opts.limit);
44
+ params.set('offset', opts.offset);
45
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/analytics/costs/by-submission${qs(params)}`));
46
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.analyticsBySubmissionColumns);
47
+ });
48
+ addCommonOptions(exports.analyticsCommand
49
+ .command('by-environment')
50
+ .description('Cost grouped by environment')
51
+ .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))).action(async (opts) => {
52
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/analytics/costs/by-environment${qs(buildParams(opts))}`));
53
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.analyticsByEnvironmentColumns);
54
+ });
55
+ addCommonOptions(exports.analyticsCommand
56
+ .command('by-queue')
57
+ .description('Cost grouped by queue')
58
+ .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))).action(async (opts) => {
59
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/analytics/costs/by-queue${qs(buildParams(opts))}`));
60
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.analyticsByQueueColumns);
61
+ });
62
+ addCommonOptions(exports.analyticsCommand
63
+ .command('by-instance-type')
64
+ .description('Cost grouped by EC2 instance type and market type')
65
+ .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))).action(async (opts) => {
66
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/analytics/costs/by-instance-type${qs(buildParams(opts))}`));
67
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.analyticsByInstanceTypeColumns);
68
+ });
69
+ addCommonOptions(exports.analyticsCommand
70
+ .command('over-time')
71
+ .description('Cost bucketed by time period')
72
+ .addOption(new commander_1.Option('--interval <interval>', 'Time bucket size').choices(['day', 'week', 'month']).default('day'))
73
+ .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))).action(async (opts) => {
74
+ const params = buildParams(opts);
75
+ params.set('interval', opts.interval);
76
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/analytics/costs/over-time${qs(params)}`));
77
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.analyticsOverTimeColumns);
78
+ });
79
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"analytics.js","sourceRoot":"","sources":["../../src/commands/analytics.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander'\nimport { apiUrl, fetchJson } from '../auth'\nimport {\n  formatOutput,\n  analyticsSummaryColumns,\n  analyticsBySubmissionColumns,\n  analyticsByEnvironmentColumns,\n  analyticsByQueueColumns,\n  analyticsByInstanceTypeColumns,\n  analyticsOverTimeColumns,\n  type OutputFormat,\n} from '../format.js'\n\ntype BaseOpts = { from?: string; to?: string; submissionStatus?: string }\n\nfunction addCommonOptions(cmd: Command): Command {\n  return cmd\n    .option('--from <iso8601>', 'Start of date range (ISO 8601, e.g. 2025-01-01T00:00:00Z)')\n    .option('--to <iso8601>', 'End of date range (ISO 8601, exclusive)')\n    .option('--submission-status <status>', 'Filter by submission status (e.g. SUCCEEDED, FAILED, RUNNING)')\n}\n\nfunction buildParams(opts: BaseOpts): URLSearchParams {\n  const params = new URLSearchParams()\n  if (opts.from) params.set('from', opts.from)\n  if (opts.to) params.set('to', opts.to)\n  if (opts.submissionStatus) params.set('submissionStatus', opts.submissionStatus)\n  return params\n}\n\nfunction qs(params: URLSearchParams): string {\n  const s = params.toString()\n  return s ? `?${s}` : ''\n}\n\nexport const analyticsCommand = new Command('analytics')\n  .description('View cost analytics')\n\naddCommonOptions(\n  analyticsCommand\n    .command('summary')\n    .description('Overall cost totals for the tenant')\n    .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n).action(async (opts: BaseOpts & { format: OutputFormat }) => {\n  const data = await fetchJson(apiUrl(`/analytics/costs/summary${qs(buildParams(opts))}`))\n  formatOutput(data, opts.format, analyticsSummaryColumns)\n})\n\naddCommonOptions(\n  analyticsCommand\n    .command('by-submission')\n    .description('Per-submission cost breakdown')\n    .option('--limit <n>', 'Max results (1–200)', '50')\n    .option('--offset <n>', 'Pagination offset', '0')\n    .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n).action(async (opts: BaseOpts & { limit: string; offset: string; format: OutputFormat }) => {\n  const params = buildParams(opts)\n  params.set('limit', opts.limit)\n  params.set('offset', opts.offset)\n  const data = await fetchJson(apiUrl(`/analytics/costs/by-submission${qs(params)}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsBySubmissionColumns)\n})\n\naddCommonOptions(\n  analyticsCommand\n    .command('by-environment')\n    .description('Cost grouped by environment')\n    .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n).action(async (opts: BaseOpts & { format: OutputFormat }) => {\n  const data = await fetchJson(apiUrl(`/analytics/costs/by-environment${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByEnvironmentColumns)\n})\n\naddCommonOptions(\n  analyticsCommand\n    .command('by-queue')\n    .description('Cost grouped by queue')\n    .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n).action(async (opts: BaseOpts & { format: OutputFormat }) => {\n  const data = await fetchJson(apiUrl(`/analytics/costs/by-queue${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByQueueColumns)\n})\n\naddCommonOptions(\n  analyticsCommand\n    .command('by-instance-type')\n    .description('Cost grouped by EC2 instance type and market type')\n    .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n).action(async (opts: BaseOpts & { format: OutputFormat }) => {\n  const data = await fetchJson(apiUrl(`/analytics/costs/by-instance-type${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByInstanceTypeColumns)\n})\n\naddCommonOptions(\n  analyticsCommand\n    .command('over-time')\n    .description('Cost bucketed by time period')\n    .addOption(new Option('--interval <interval>', 'Time bucket size').choices(['day', 'week', 'month']).default('day'))\n    .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n).action(async (opts: BaseOpts & { interval: string; format: OutputFormat }) => {\n  const params = buildParams(opts)\n  params.set('interval', opts.interval)\n  const data = await fetchJson(apiUrl(`/analytics/costs/over-time${qs(params)}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsOverTimeColumns)\n})\n"]}
@@ -11,8 +11,8 @@ const apiTokenColumns = [
11
11
  { header: 'Description', value: (r) => r.description },
12
12
  { header: 'Environment ID', value: (r) => r.environment_id ?? '' },
13
13
  { header: 'Created By', value: (r) => r.created_by },
14
- { header: 'Expires At', value: (r) => r.expires_at },
15
- { header: 'Last Used At', value: (r) => r.last_used_at },
14
+ { header: 'Expires At', value: (r) => (0, format_js_1.formatDateTime)(r.expires_at) },
15
+ { header: 'Last Used At', value: (r) => (0, format_js_1.formatDateTime)(r.last_used_at) },
16
16
  { header: 'Active', value: (r) => r.is_active },
17
17
  ];
18
18
  exports.apiTokenCommand = new commander_1.Command('api-token')
@@ -93,4 +93,4 @@ exports.apiTokenCommand
93
93
  });
94
94
  console.log(`${ids.length} token(s) revoked.`);
95
95
  });
96
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-token.js","sourceRoot":"","sources":["../../src/commands/api-token.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 apiTokenColumns = [\n  { header: 'ID', value: (r: Record<string, unknown>) => r.id },\n  { header: 'Prefix', value: (r: Record<string, unknown>) => r.token_prefix },\n  { header: 'Role', value: (r: Record<string, unknown>) => r.role ?? '' },\n  { header: 'Description', value: (r: Record<string, unknown>) => r.description },\n  { header: 'Environment ID', value: (r: Record<string, unknown>) => r.environment_id ?? '' },\n  { header: 'Created By', value: (r: Record<string, unknown>) => r.created_by },\n  { header: 'Expires At', value: (r: Record<string, unknown>) => r.expires_at },\n  { header: 'Last Used At', value: (r: Record<string, unknown>) => r.last_used_at },\n  { header: 'Active', value: (r: Record<string, unknown>) => r.is_active },\n];\n\nexport const apiTokenCommand = new Command('api-token')\n  .description('Manage API tokens');\n\napiTokenCommand\n  .command('list')\n  .description('List API tokens for the current tenant')\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('/admin/api-tokens'));\n    formatOutput(data, opts.format, apiTokenColumns);\n  });\n\napiTokenCommand\n  .command('get <id>')\n  .description('Get details of an API token')\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/api-tokens/${id}`));\n    formatOutput(data, opts.format, apiTokenColumns);\n  });\n\napiTokenCommand\n  .command('create')\n  .description('Create a new API token (token is shown only once)')\n  .requiredOption('--name <name>', 'Token name')\n  .addOption(\n    new Option('--role <role>', 'Token role').choices(['automation']).default('automation'),\n  )\n  .option('--description <text>', 'Human-readable label for this token')\n  .option('--expires-at <date>', 'Expiration date in ISO 8601 format (max 1 year, default 1 year)')\n  .option('--environment-id <uuid>', 'Optional — restrict token submissions to this environment')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { name: string; role: string; description?: string; expiresAt?: string; environmentId?: string; format: OutputFormat }) => {\n    const body: Record<string, unknown> = { name: opts.name, role: opts.role };\n    if (opts.description) body.description = opts.description;\n    if (opts.expiresAt) body.expires_at = opts.expiresAt;\n    if (opts.environmentId) body.environment_id = opts.environmentId;\n\n    const data = await fetchJson(apiUrl('/admin/api-tokens'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify(body),\n    }) as Record<string, unknown>;\n\n    if (opts.format === 'json') {\n      console.log(JSON.stringify(data, null, 2));\n    } else {\n      console.log('');\n      console.log('API token created. Save this token — it will not be shown again.');\n      console.log('');\n      console.log(`Token: ${data.token}`);\n      console.log('');\n      const row = { ...data };\n      delete row.token;\n      formatOutput(row, opts.format, apiTokenColumns);\n    }\n  });\n\napiTokenCommand\n  .command('revoke <id>')\n  .description('Revoke an API token')\n  .action(async (id: string) => {\n    await fetchJson(apiUrl(`/admin/api-tokens/${id}`), { method: 'DELETE' });\n    console.log('Token revoked.');\n  });\n\napiTokenCommand\n  .command('revoke-batch')\n  .description('Revoke multiple API tokens at once (Admin only)')\n  .requiredOption('--ids <ids>', 'Comma-separated list of token IDs to revoke')\n  .action(async (opts: { ids: string }) => {\n    const ids = opts.ids.split(',').map((id) => id.trim()).filter(Boolean);\n    if (ids.length === 0) {\n      console.error('No IDs provided.');\n      process.exit(1);\n    }\n    await fetchJson(apiUrl('/admin/api-tokens/revoke-batch'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify({ ids }),\n    });\n    console.log(`${ids.length} token(s) revoked.`);\n  });\n"]}
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api-token.js","sourceRoot":"","sources":["../../src/commands/api-token.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander';\nimport { apiUrl, fetchJson } from '../auth';\nimport { formatOutput, formatDateTime, type OutputFormat } from '../format.js';\n\nconst apiTokenColumns = [\n  { header: 'ID', value: (r: Record<string, unknown>) => r.id },\n  { header: 'Prefix', value: (r: Record<string, unknown>) => r.token_prefix },\n  { header: 'Role', value: (r: Record<string, unknown>) => r.role ?? '' },\n  { header: 'Description', value: (r: Record<string, unknown>) => r.description },\n  { header: 'Environment ID', value: (r: Record<string, unknown>) => r.environment_id ?? '' },\n  { header: 'Created By', value: (r: Record<string, unknown>) => r.created_by },\n  { header: 'Expires At', value: (r: Record<string, unknown>) => formatDateTime(r.expires_at) },\n  { header: 'Last Used At', value: (r: Record<string, unknown>) => formatDateTime(r.last_used_at) },\n  { header: 'Active', value: (r: Record<string, unknown>) => r.is_active },\n];\n\nexport const apiTokenCommand = new Command('api-token')\n  .description('Manage API tokens');\n\napiTokenCommand\n  .command('list')\n  .description('List API tokens for the current tenant')\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('/admin/api-tokens'));\n    formatOutput(data, opts.format, apiTokenColumns);\n  });\n\napiTokenCommand\n  .command('get <id>')\n  .description('Get details of an API token')\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/api-tokens/${id}`));\n    formatOutput(data, opts.format, apiTokenColumns);\n  });\n\napiTokenCommand\n  .command('create')\n  .description('Create a new API token (token is shown only once)')\n  .requiredOption('--name <name>', 'Token name')\n  .addOption(\n    new Option('--role <role>', 'Token role').choices(['automation']).default('automation'),\n  )\n  .option('--description <text>', 'Human-readable label for this token')\n  .option('--expires-at <date>', 'Expiration date in ISO 8601 format (max 1 year, default 1 year)')\n  .option('--environment-id <uuid>', 'Optional — restrict token submissions to this environment')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { name: string; role: string; description?: string; expiresAt?: string; environmentId?: string; format: OutputFormat }) => {\n    const body: Record<string, unknown> = { name: opts.name, role: opts.role };\n    if (opts.description) body.description = opts.description;\n    if (opts.expiresAt) body.expires_at = opts.expiresAt;\n    if (opts.environmentId) body.environment_id = opts.environmentId;\n\n    const data = await fetchJson(apiUrl('/admin/api-tokens'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify(body),\n    }) as Record<string, unknown>;\n\n    if (opts.format === 'json') {\n      console.log(JSON.stringify(data, null, 2));\n    } else {\n      console.log('');\n      console.log('API token created. Save this token — it will not be shown again.');\n      console.log('');\n      console.log(`Token: ${data.token}`);\n      console.log('');\n      const row = { ...data };\n      delete row.token;\n      formatOutput(row, opts.format, apiTokenColumns);\n    }\n  });\n\napiTokenCommand\n  .command('revoke <id>')\n  .description('Revoke an API token')\n  .action(async (id: string) => {\n    await fetchJson(apiUrl(`/admin/api-tokens/${id}`), { method: 'DELETE' });\n    console.log('Token revoked.');\n  });\n\napiTokenCommand\n  .command('revoke-batch')\n  .description('Revoke multiple API tokens at once (Admin only)')\n  .requiredOption('--ids <ids>', 'Comma-separated list of token IDs to revoke')\n  .action(async (opts: { ids: string }) => {\n    const ids = opts.ids.split(',').map((id) => id.trim()).filter(Boolean);\n    if (ids.length === 0) {\n      console.error('No IDs provided.');\n      process.exit(1);\n    }\n    await fetchJson(apiUrl('/admin/api-tokens/revoke-batch'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify({ ids }),\n    });\n    console.log(`${ids.length} token(s) revoked.`);\n  });\n"]}
@@ -56,6 +56,7 @@ exports.engineCommand
56
56
  .requiredOption('--environment-id <id>', 'Environment ID')
57
57
  .option('--engine <engine>', 'Engine type', 'nextflow')
58
58
  .option('--engine-version <version>', 'Engine version', '25.10.4')
59
+ .option('--retry-attempts <n>', 'Number of spot-retry attempts', (v) => parseInt(v, 10), 5)
59
60
  .option('--read-only <uri>', 'Read-only access to S3 bucket/prefix, e.g. s3://bucket/prefix (repeatable)', collect, [])
60
61
  .option('--read-write <uri>', 'Read-write access to S3 bucket/prefix (repeatable)', collect, [])
61
62
  .option('--write-only <uri>', 'Write-only access to S3 bucket/prefix (repeatable)', collect, [])
@@ -74,10 +75,11 @@ exports.engineCommand
74
75
  environmentId: opts.environmentId,
75
76
  engine: opts.engine,
76
77
  engineVersion: opts.engineVersion,
78
+ retryAttempts: opts.retryAttempts,
77
79
  ...(s3Buckets.length > 0 ? { s3Buckets } : {}),
78
80
  ...(process.env.GENI_TENANT_ID ? { tenantId: process.env.GENI_TENANT_ID } : {}),
79
81
  }),
80
82
  });
81
83
  (0, format_js_1.formatOutput)(data, opts.format, format_js_1.engineColumns);
82
84
  });
83
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.js","sourceRoot":"","sources":["../../src/commands/engine.ts"],"names":[],"mappings":";;;AAAA,yCAA2C;AAC3C,kCAA2C;AAC3C,4CAA6E;AAC7E,4CAA8C;AAU9C,SAAS,UAAU,CAAC,GAAW;IAC7B,MAAM,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC,2BAA2B,CAAC,CAAA;IACpD,IAAI,CAAC,KAAK;QAAE,MAAM,IAAA,2BAAe,EAAC,oBAAoB,GAAG,uDAAuD,CAAC,CAAA;IACjH,OAAO,EAAE,UAAU,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC,IAAI,SAAS,EAAE,CAAA;AAChE,CAAC;AAED,MAAM,OAAO,GAAG,CAAC,GAAW,EAAE,IAAc,EAAE,EAAE,CAAC,CAAC,GAAG,IAAI,EAAE,GAAG,CAAC,CAAA;AAElD,QAAA,aAAa,GAAG,IAAI,mBAAO,CAAC,QAAQ,CAAC;KAC/C,WAAW,CAAC,gBAAgB,CAAC,CAAA;AAEhC,qBAAa;KACV,OAAO,CAAC,MAAM,CAAC;KACf,WAAW,CAAC,cAAc,CAAC;KAC3B,MAAM,CAAC,mBAAmB,EAAE,0DAA0D,CAAC;KACvF,SAAS,CAAC,IAAI,kBAAM,CAAC,mBAAmB,EAAE,eAAe,CAAC,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;KAC9G,MAAM,CAAC,KAAK,EAAE,IAA+C,EAAE,EAAE;IAChE,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAA;IACpC,IAAI,IAAI,CAAC,MAAM;QAAE,MAAM,CAAC,GAAG,CAAC,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC,CAAA;IAChE,IAAI,OAAO,CAAC,GAAG,CAAC,cAAc;QAAE,MAAM,CAAC,GAAG,CAAC,UAAU,EAAE,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,CAAA;IAClF,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,MAAM,EAAE,CAAC,CAAC,CAAC,EAAE,CAAA;IAC7C,MAAM,IAAI,GAAG,MAAM,IAAA,gBAAS,EAAC,IAAA,aAAM,EAAC,iBAAiB,KAAK,EAAE,CAAC,CAAyB,CAAA;IACtF,IAAA,wBAAY,EAAC,IAAI,CAAC,MAAM,KAAK,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,EAAE,yBAAa,CAAC,CAAA;AACtF,CAAC,CAAC,CAAA;AAEJ,qBAAa;KACV,OAAO,CAAC,UAAU,CAAC;KACnB,WAAW,CAAC,qBAAqB,CAAC;KAClC,SAAS,CAAC,IAAI,kBAAM,CAAC,mBAAmB,EAAE,eAAe,CAAC,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;KAC9G,MAAM,CAAC,KAAK,EAAE,EAAU,EAAE,IAA8B,EAAE,EAAE;IAC3D,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC,CAAC,aAAa,OAAO,CAAC,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,EAAE,CAAA;IACzF,MAAM,IAAI,GAAG,MAAM,IAAA,gBAAS,EAAC,IAAA,aAAM,EAAC,kBAAkB,EAAE,GAAG,KAAK,EAAE,CAAC,CAAC,CAAA;IACpE,IAAA,wBAAY,EAAC,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE,yBAAa,CAAC,CAAA;AAChD,CAAC,CAAC,CAAA;AAEJ,qBAAa;KACV,OAAO,CAAC,aAAa,CAAC;KACtB,WAAW,CAAC,kBAAkB,CAAC;KAC/B,SAAS,CAAC,IAAI,kBAAM,CAAC,mBAAmB,EAAE,eAAe,CAAC,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;KAC9G,MAAM,CAAC,KAAK,EAAE,EAAU,EAAE,IAA8B,EAAE,EAAE;IAC3D,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC,CAAC,aAAa,OAAO,CAAC,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,EAAE,CAAA;IACzF,MAAM,IAAI,GAAG,MAAM,IAAA,gBAAS,EAAC,IAAA,aAAM,EAAC,kBAAkB,EAAE,GAAG,KAAK,EAAE,CAAC,EAAE;QACnE,MAAM,EAAE,QAAQ;KACjB,CAAC,CAAA;IACF,IAAA,wBAAY,EAAC,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE,yBAAa,CAAC,CAAA;AAChD,CAAC,CAAC,CAAA;AAEJ,qBAAa;KACV,OAAO,CAAC,QAAQ,CAAC;KACjB,WAAW,CAAC,kBAAkB,CAAC;KAC/B,cAAc,CAAC,eAAe,EAAE,aAAa,CAAC;KAC9C,cAAc,CAAC,uBAAuB,EAAE,gBAAgB,CAAC;KACzD,MAAM,CAAC,mBAAmB,EAAE,aAAa,EAAE,UAAU,CAAC;KACtD,MAAM,CAAC,4BAA4B,EAAE,gBAAgB,EAAE,SAAS,CAAC;KACjE,MAAM,CAAC,mBAAmB,EAAE,4EAA4E,EAAE,OAAO,EAAE,EAAc,CAAC;KAClI,MAAM,CAAC,oBAAoB,EAAE,oDAAoD,EAAE,OAAO,EAAE,EAAc,CAAC;KAC3G,MAAM,CAAC,oBAAoB,EAAE,oDAAoD,EAAE,OAAO,EAAE,EAAc,CAAC;KAC3G,SAAS,CAAC,IAAI,kBAAM,CAAC,mBAAmB,EAAE,eAAe,CAAC,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;KAC9G,MAAM,CAAC,KAAK,EAAE,IASd,EAAE,EAAE;IACH,MAAM,SAAS,GAAyB;QACtC,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,UAAU,CAAC,CAAC,CAAC,EAAE,cAAc,EAAE,WAA6B,EAAE,CAAC,CAAC;QAChG,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,UAAU,CAAC,CAAC,CAAC,EAAE,cAAc,EAAE,YAA8B,EAAE,CAAC,CAAC;QAClG,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,UAAU,CAAC,CAAC,CAAC,EAAE,cAAc,EAAE,YAA8B,EAAE,CAAC,CAAC;KACnG,CAAA;IACD,MAAM,IAAI,GAAG,MAAM,IAAA,gBAAS,EAAC,IAAA,aAAM,EAAC,gBAAgB,CAAC,EAAE;QACrD,MAAM,EAAE,MAAM;QACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;QAC/C,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;YACnB,IAAI,EAAE,IAAI,CAAC,IAAI;YACf,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,GAAG,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YAC9C,GAAG,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,OAAO,CAAC,GAAG,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;SAChF,CAAC;KACH,CAAC,CAAA;IACF,IAAA,wBAAY,EAAC,IAAI,EAAE,IAAI,CAAC,MAAM,EAAE,yBAAa,CAAC,CAAA;AAChD,CAAC,CAAC,CAAA","sourcesContent":["import { Command, Option } from 'commander'\nimport { apiUrl, fetchJson } from '../auth'\nimport { formatOutput, engineColumns, type OutputFormat } from '../format.js'\nimport { invalidArgument } from '../errors.js'\n\ntype PermissionMode = 'read-only' | 'read-write' | 'write-only'\n\ninterface S3BucketPermission {\n  bucketName: string\n  prefix?: string\n  permissionMode: PermissionMode\n}\n\nfunction parseS3Uri(uri: string): { bucketName: string; prefix?: string } {\n  const match = uri.match(/^s3:\\/\\/([^/]+)(\\/(.+))?$/)\n  if (!match) throw invalidArgument(`Invalid S3 URI: \"${uri}\". Expected format: s3://bucket or s3://bucket/prefix`)\n  return { bucketName: match[1], prefix: match[3] || undefined }\n}\n\nconst collect = (val: string, prev: string[]) => [...prev, val]\n\nexport const engineCommand = new Command('engine')\n  .description('Manage engines')\n\nengineCommand\n  .command('list')\n  .description('List engines')\n  .option('--status <status>', 'Filter by status (e.g. ACTIVE, PENDING, FAILED, DELETED)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { status?: string; format: OutputFormat }) => {\n    const params = new URLSearchParams()\n    if (opts.status) params.set('status', opts.status.toUpperCase())\n    if (process.env.GENI_TENANT_ID) params.set('tenantId', process.env.GENI_TENANT_ID)\n    const query = params.size ? `?${params}` : ''\n    const data = await fetchJson(apiUrl(`/admin/engines${query}`)) as { items: unknown[] }\n    formatOutput(opts.format === 'json' ? data : data.items, opts.format, engineColumns)\n  })\n\nengineCommand\n  .command('get <id>')\n  .description('Get an engine 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 query = process.env.GENI_TENANT_ID ? `?tenantId=${process.env.GENI_TENANT_ID}` : ''\n    const data = await fetchJson(apiUrl(`/admin/engines/${id}${query}`))\n    formatOutput(data, opts.format, engineColumns)\n  })\n\nengineCommand\n  .command('delete <id>')\n  .description('Delete an engine')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (id: string, opts: { format: OutputFormat }) => {\n    const query = process.env.GENI_TENANT_ID ? `?tenantId=${process.env.GENI_TENANT_ID}` : ''\n    const data = await fetchJson(apiUrl(`/admin/engines/${id}${query}`), {\n      method: 'DELETE',\n    })\n    formatOutput(data, opts.format, engineColumns)\n  })\n\nengineCommand\n  .command('create')\n  .description('Create an engine')\n  .requiredOption('--name <name>', 'Engine name')\n  .requiredOption('--environment-id <id>', 'Environment ID')\n  .option('--engine <engine>', 'Engine type', 'nextflow')\n  .option('--engine-version <version>', 'Engine version', '25.10.4')\n  .option('--read-only <uri>', 'Read-only access to S3 bucket/prefix, e.g. s3://bucket/prefix (repeatable)', collect, [] as string[])\n  .option('--read-write <uri>', 'Read-write access to S3 bucket/prefix (repeatable)', collect, [] as string[])\n  .option('--write-only <uri>', 'Write-only access to S3 bucket/prefix (repeatable)', collect, [] as string[])\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: {\n    name: string\n    environmentId: string\n    engine: string\n    engineVersion: string\n    readOnly: string[]\n    readWrite: string[]\n    writeOnly: string[]\n    format: OutputFormat\n  }) => {\n    const s3Buckets: S3BucketPermission[] = [\n      ...opts.readOnly.map(u => ({ ...parseS3Uri(u), permissionMode: 'read-only' as PermissionMode })),\n      ...opts.readWrite.map(u => ({ ...parseS3Uri(u), permissionMode: 'read-write' as PermissionMode })),\n      ...opts.writeOnly.map(u => ({ ...parseS3Uri(u), permissionMode: 'write-only' as PermissionMode })),\n    ]\n    const data = await fetchJson(apiUrl('/admin/engines'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify({\n        name: opts.name,\n        environmentId: opts.environmentId,\n        engine: opts.engine,\n        engineVersion: opts.engineVersion,\n        ...(s3Buckets.length > 0 ? { s3Buckets } : {}),\n        ...(process.env.GENI_TENANT_ID ? { tenantId: process.env.GENI_TENANT_ID } : {}),\n      }),\n    })\n    formatOutput(data, opts.format, engineColumns)\n  })\n"]}
85
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"engine.js","sourceRoot":"","sources":["../../src/commands/engine.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander'\nimport { apiUrl, fetchJson } from '../auth'\nimport { formatOutput, engineColumns, type OutputFormat } from '../format.js'\nimport { invalidArgument } from '../errors.js'\n\ntype PermissionMode = 'read-only' | 'read-write' | 'write-only'\n\ninterface S3BucketPermission {\n  bucketName: string\n  prefix?: string\n  permissionMode: PermissionMode\n}\n\nfunction parseS3Uri(uri: string): { bucketName: string; prefix?: string } {\n  const match = uri.match(/^s3:\\/\\/([^/]+)(\\/(.+))?$/)\n  if (!match) throw invalidArgument(`Invalid S3 URI: \"${uri}\". Expected format: s3://bucket or s3://bucket/prefix`)\n  return { bucketName: match[1], prefix: match[3] || undefined }\n}\n\nconst collect = (val: string, prev: string[]) => [...prev, val]\n\nexport const engineCommand = new Command('engine')\n  .description('Manage engines')\n\nengineCommand\n  .command('list')\n  .description('List engines')\n  .option('--status <status>', 'Filter by status (e.g. ACTIVE, PENDING, FAILED, DELETED)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { status?: string; format: OutputFormat }) => {\n    const params = new URLSearchParams()\n    if (opts.status) params.set('status', opts.status.toUpperCase())\n    if (process.env.GENI_TENANT_ID) params.set('tenantId', process.env.GENI_TENANT_ID)\n    const query = params.size ? `?${params}` : ''\n    const data = await fetchJson(apiUrl(`/admin/engines${query}`)) as { items: unknown[] }\n    formatOutput(opts.format === 'json' ? data : data.items, opts.format, engineColumns)\n  })\n\nengineCommand\n  .command('get <id>')\n  .description('Get an engine 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 query = process.env.GENI_TENANT_ID ? `?tenantId=${process.env.GENI_TENANT_ID}` : ''\n    const data = await fetchJson(apiUrl(`/admin/engines/${id}${query}`))\n    formatOutput(data, opts.format, engineColumns)\n  })\n\nengineCommand\n  .command('delete <id>')\n  .description('Delete an engine')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (id: string, opts: { format: OutputFormat }) => {\n    const query = process.env.GENI_TENANT_ID ? `?tenantId=${process.env.GENI_TENANT_ID}` : ''\n    const data = await fetchJson(apiUrl(`/admin/engines/${id}${query}`), {\n      method: 'DELETE',\n    })\n    formatOutput(data, opts.format, engineColumns)\n  })\n\nengineCommand\n  .command('create')\n  .description('Create an engine')\n  .requiredOption('--name <name>', 'Engine name')\n  .requiredOption('--environment-id <id>', 'Environment ID')\n  .option('--engine <engine>', 'Engine type', 'nextflow')\n  .option('--engine-version <version>', 'Engine version', '25.10.4')\n  .option('--retry-attempts <n>', 'Number of spot-retry attempts', (v) => parseInt(v, 10), 5)\n  .option('--read-only <uri>', 'Read-only access to S3 bucket/prefix, e.g. s3://bucket/prefix (repeatable)', collect, [] as string[])\n  .option('--read-write <uri>', 'Read-write access to S3 bucket/prefix (repeatable)', collect, [] as string[])\n  .option('--write-only <uri>', 'Write-only access to S3 bucket/prefix (repeatable)', collect, [] as string[])\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: {\n    name: string\n    environmentId: string\n    engine: string\n    engineVersion: string\n    retryAttempts: number\n    readOnly: string[]\n    readWrite: string[]\n    writeOnly: string[]\n    format: OutputFormat\n  }) => {\n    const s3Buckets: S3BucketPermission[] = [\n      ...opts.readOnly.map(u => ({ ...parseS3Uri(u), permissionMode: 'read-only' as PermissionMode })),\n      ...opts.readWrite.map(u => ({ ...parseS3Uri(u), permissionMode: 'read-write' as PermissionMode })),\n      ...opts.writeOnly.map(u => ({ ...parseS3Uri(u), permissionMode: 'write-only' as PermissionMode })),\n    ]\n    const data = await fetchJson(apiUrl('/admin/engines'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify({\n        name: opts.name,\n        environmentId: opts.environmentId,\n        engine: opts.engine,\n        engineVersion: opts.engineVersion,\n        retryAttempts: opts.retryAttempts,\n        ...(s3Buckets.length > 0 ? { s3Buckets } : {}),\n        ...(process.env.GENI_TENANT_ID ? { tenantId: process.env.GENI_TENANT_ID } : {}),\n      }),\n    })\n    formatOutput(data, opts.format, engineColumns)\n  })\n"]}
@@ -48,7 +48,7 @@ exports.environmentCommand
48
48
  .requiredOption('--cloud-id <cloudId>', 'Cloud account ID')
49
49
  .requiredOption('--cloud-region <cloudRegion>', 'Cloud region')
50
50
  .option('--identity-to-assume <identityToAssume>', 'Cross-account identity ARN (default: created by geni-setup)')
51
- .option('--availability-zones <n>', 'Number of private subnets', '2')
51
+ .option('--availability-zones <n>', 'Number of availability zones (2–4, default: 2)', '2')
52
52
  .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
53
53
  .action(async (opts) => {
54
54
  const identityToAssume = opts.identityToAssume ?? `arn:aws:iam::${opts.cloudId}:role/GeniCrossAccountRole`;
@@ -66,4 +66,4 @@ exports.environmentCommand
66
66
  });
67
67
  (0, format_js_1.formatOutput)(data, opts.format, format_js_1.environmentColumns);
68
68
  });
69
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"environment.js","sourceRoot":"","sources":["../../src/commands/environment.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander'\nimport { apiUrl, fetchJson } from '../auth'\nimport { formatOutput, environmentColumns, type OutputFormat } from '../format.js'\n\nexport const environmentCommand = new Command('environment')\n  .description('Manage environments')\n\nenvironmentCommand\n  .command('list')\n  .description('List environments')\n  .option('--status <status>', 'Filter by status (e.g. ACTIVE, PENDING, FAILED, DELETED)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { status?: string; format: OutputFormat }) => {\n    const params = new URLSearchParams()\n    if (opts.status) params.set('status', opts.status.toUpperCase())\n    if (process.env.GENI_TENANT_ID) params.set('tenantId', process.env.GENI_TENANT_ID)\n    const query = params.size ? `?${params}` : ''\n    const data = await fetchJson(apiUrl(`/admin/environments${query}`)) as { items: unknown[] }\n    formatOutput(opts.format === 'json' ? data : data.items, opts.format, environmentColumns)\n  })\n\nenvironmentCommand\n  .command('get <id>')\n  .description('Get an environment 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 query = process.env.GENI_TENANT_ID ? `?tenantId=${process.env.GENI_TENANT_ID}` : ''\n    const data = await fetchJson(apiUrl(`/admin/environments/${id}${query}`))\n    formatOutput(data, opts.format, environmentColumns)\n  })\n\nenvironmentCommand\n  .command('delete <id>')\n  .description('Delete an environment')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (id: string, opts: { format: OutputFormat }) => {\n    const query = process.env.GENI_TENANT_ID ? `?tenantId=${process.env.GENI_TENANT_ID}` : ''\n    const data = await fetchJson(apiUrl(`/admin/environments/${id}${query}`), {\n      method: 'DELETE',\n    })\n    formatOutput(data, opts.format, environmentColumns)\n  })\n\nenvironmentCommand\n  .command('create')\n  .description('Create an environment')\n  .requiredOption('--name <name>', 'Environment name')\n  .requiredOption('--cloud-id <cloudId>', 'Cloud account ID')\n  .requiredOption('--cloud-region <cloudRegion>', 'Cloud region')\n  .option('--identity-to-assume <identityToAssume>', 'Cross-account identity ARN (default: created by geni-setup)')\n  .option('--availability-zones <n>', 'Number of private subnets', '2')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: {\n    name: string\n    cloudId: string\n    cloudRegion: string\n    identityToAssume?: string\n    availabilityZones: string\n    format: OutputFormat\n  }) => {\n    const identityToAssume = opts.identityToAssume ?? `arn:aws:iam::${opts.cloudId}:role/GeniCrossAccountRole`\n    const data = await fetchJson(apiUrl('/admin/environments'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify({\n        name: opts.name,\n        cloudId: opts.cloudId,\n        cloudRegion: opts.cloudRegion,\n        identityToAssume,\n        availabilityZones: parseInt(opts.availabilityZones, 10),\n        ...(process.env.GENI_TENANT_ID ? { tenantId: process.env.GENI_TENANT_ID } : {}),\n      }),\n    })\n    formatOutput(data, opts.format, environmentColumns)\n  })\n"]}
69
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"environment.js","sourceRoot":"","sources":["../../src/commands/environment.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander'\nimport { apiUrl, fetchJson } from '../auth'\nimport { formatOutput, environmentColumns, type OutputFormat } from '../format.js'\n\nexport const environmentCommand = new Command('environment')\n  .description('Manage environments')\n\nenvironmentCommand\n  .command('list')\n  .description('List environments')\n  .option('--status <status>', 'Filter by status (e.g. ACTIVE, PENDING, FAILED, DELETED)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { status?: string; format: OutputFormat }) => {\n    const params = new URLSearchParams()\n    if (opts.status) params.set('status', opts.status.toUpperCase())\n    if (process.env.GENI_TENANT_ID) params.set('tenantId', process.env.GENI_TENANT_ID)\n    const query = params.size ? `?${params}` : ''\n    const data = await fetchJson(apiUrl(`/admin/environments${query}`)) as { items: unknown[] }\n    formatOutput(opts.format === 'json' ? data : data.items, opts.format, environmentColumns)\n  })\n\nenvironmentCommand\n  .command('get <id>')\n  .description('Get an environment 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 query = process.env.GENI_TENANT_ID ? `?tenantId=${process.env.GENI_TENANT_ID}` : ''\n    const data = await fetchJson(apiUrl(`/admin/environments/${id}${query}`))\n    formatOutput(data, opts.format, environmentColumns)\n  })\n\nenvironmentCommand\n  .command('delete <id>')\n  .description('Delete an environment')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (id: string, opts: { format: OutputFormat }) => {\n    const query = process.env.GENI_TENANT_ID ? `?tenantId=${process.env.GENI_TENANT_ID}` : ''\n    const data = await fetchJson(apiUrl(`/admin/environments/${id}${query}`), {\n      method: 'DELETE',\n    })\n    formatOutput(data, opts.format, environmentColumns)\n  })\n\nenvironmentCommand\n  .command('create')\n  .description('Create an environment')\n  .requiredOption('--name <name>', 'Environment name')\n  .requiredOption('--cloud-id <cloudId>', 'Cloud account ID')\n  .requiredOption('--cloud-region <cloudRegion>', 'Cloud region')\n  .option('--identity-to-assume <identityToAssume>', 'Cross-account identity ARN (default: created by geni-setup)')\n  .option('--availability-zones <n>', 'Number of availability zones (2–4, default: 2)', '2')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: {\n    name: string\n    cloudId: string\n    cloudRegion: string\n    identityToAssume?: string\n    availabilityZones: string\n    format: OutputFormat\n  }) => {\n    const identityToAssume = opts.identityToAssume ?? `arn:aws:iam::${opts.cloudId}:role/GeniCrossAccountRole`\n    const data = await fetchJson(apiUrl('/admin/environments'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify({\n        name: opts.name,\n        cloudId: opts.cloudId,\n        cloudRegion: opts.cloudRegion,\n        identityToAssume,\n        availabilityZones: parseInt(opts.availabilityZones, 10),\n        ...(process.env.GENI_TENANT_ID ? { tenantId: process.env.GENI_TENANT_ID } : {}),\n      }),\n    })\n    formatOutput(data, opts.format, environmentColumns)\n  })\n"]}
@@ -13,6 +13,7 @@ exports.instanceCommand
13
13
  .option('--queue-id <id>', 'Filter by queue ID')
14
14
  .option('--submission-id <id>', 'Filter by submission ID')
15
15
  .option('--status <status>', 'Filter by status (e.g. RUNNING, TERMINATED, COST_CALCULATED, ERROR)')
16
+ .option('--limit <n>', 'Maximum number of results to return', '10')
16
17
  .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
17
18
  .action(async (opts) => {
18
19
  const params = new URLSearchParams();
@@ -24,8 +25,8 @@ exports.instanceCommand
24
25
  params.set('submissionId', opts.submissionId);
25
26
  if (opts.status)
26
27
  params.set('status', opts.status);
27
- const query = params.toString() ? `?${params.toString()}` : '';
28
- const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/instances${query}`));
28
+ params.set('limit', opts.limit);
29
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/instances?${params}`));
29
30
  (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.instanceColumns);
30
31
  });
31
32
  exports.instanceCommand
@@ -36,4 +37,4 @@ exports.instanceCommand
36
37
  const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/instances/${id}`));
37
38
  (0, format_js_1.formatOutput)(data, opts.format, format_js_1.instanceColumns);
38
39
  });
39
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5zdGFuY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29tbWFuZHMvaW5zdGFuY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUNBQTJDO0FBQzNDLGtDQUEyQztBQUMzQyw0Q0FBK0U7QUFFbEUsUUFBQSxlQUFlLEdBQUcsSUFBSSxtQkFBTyxDQUFDLFVBQVUsQ0FBQztLQUNuRCxXQUFXLENBQUMsa0JBQWtCLENBQUMsQ0FBQTtBQUVsQyx1QkFBZTtLQUNaLE9BQU8sQ0FBQyxNQUFNLENBQUM7S0FDZixXQUFXLENBQUMsZ0JBQWdCLENBQUM7S0FDN0IsTUFBTSxDQUFDLHVCQUF1QixFQUFFLDBCQUEwQixDQUFDO0tBQzNELE1BQU0sQ0FBQyxpQkFBaUIsRUFBRSxvQkFBb0IsQ0FBQztLQUMvQyxNQUFNLENBQUMsc0JBQXNCLEVBQUUseUJBQXlCLENBQUM7S0FDekQsTUFBTSxDQUFDLG1CQUFtQixFQUFFLHFFQUFxRSxDQUFDO0tBQ2xHLFNBQVMsQ0FBQyxJQUFJLGtCQUFNLENBQUMsbUJBQW1CLEVBQUUsZUFBZSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztLQUM5RyxNQUFNLENBQUMsS0FBSyxFQUFFLElBTWQsRUFBRSxFQUFFO0lBQ0gsTUFBTSxNQUFNLEdBQUcsSUFBSSxlQUFlLEVBQUUsQ0FBQTtJQUNwQyxJQUFJLElBQUksQ0FBQyxhQUFhO1FBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxlQUFlLEVBQUUsSUFBSSxDQUFDLGFBQWEsQ0FBQyxDQUFBO0lBQ3ZFLElBQUksSUFBSSxDQUFDLE9BQU87UUFBRSxNQUFNLENBQUMsR0FBRyxDQUFDLFNBQVMsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUE7SUFDckQsSUFBSSxJQUFJLENBQUMsWUFBWTtRQUFFLE1BQU0sQ0FBQyxHQUFHLENBQUMsY0FBYyxFQUFFLElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBQTtJQUNwRSxJQUFJLElBQUksQ0FBQyxNQUFNO1FBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxRQUFRLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFBO0lBQ2xELE1BQU0sS0FBSyxHQUFHLE1BQU0sQ0FBQyxRQUFRLEVBQUUsQ0FBQyxDQUFDLENBQUMsSUFBSSxNQUFNLENBQUMsUUFBUSxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFBO0lBQzlELE1BQU0sSUFBSSxHQUFHLE1BQU0sSUFBQSxnQkFBUyxFQUFDLElBQUEsYUFBTSxFQUFDLGFBQWEsS0FBSyxFQUFFLENBQUMsQ0FBeUIsQ0FBQTtJQUNsRixJQUFBLHdCQUFZLEVBQUMsSUFBSSxDQUFDLE1BQU0sS0FBSyxNQUFNLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsTUFBTSxFQUFFLDJCQUFlLENBQUMsQ0FBQTtBQUN4RixDQUFDLENBQUMsQ0FBQTtBQUVKLHVCQUFlO0tBQ1osT0FBTyxDQUFDLFVBQVUsQ0FBQztLQUNuQixXQUFXLENBQUMsdUJBQXVCLENBQUM7S0FDcEMsU0FBUyxDQUFDLElBQUksa0JBQU0sQ0FBQyxtQkFBbUIsRUFBRSxlQUFlLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0tBQzlHLE1BQU0sQ0FBQyxLQUFLLEVBQUUsRUFBVSxFQUFFLElBQThCLEVBQUUsRUFBRTtJQUMzRCxNQUFNLElBQUksR0FBRyxNQUFNLElBQUEsZ0JBQVMsRUFBQyxJQUFBLGFBQU0sRUFBQyxjQUFjLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQTtJQUN4RCxJQUFBLHdCQUFZLEVBQUMsSUFBSSxFQUFFLElBQUksQ0FBQyxNQUFNLEVBQUUsMkJBQWUsQ0FBQyxDQUFBO0FBQ2xELENBQUMsQ0FBQyxDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbWFuZCwgT3B0aW9uIH0gZnJvbSAnY29tbWFuZGVyJ1xuaW1wb3J0IHsgYXBpVXJsLCBmZXRjaEpzb24gfSBmcm9tICcuLi9hdXRoJ1xuaW1wb3J0IHsgZm9ybWF0T3V0cHV0LCBpbnN0YW5jZUNvbHVtbnMsIHR5cGUgT3V0cHV0Rm9ybWF0IH0gZnJvbSAnLi4vZm9ybWF0LmpzJ1xuXG5leHBvcnQgY29uc3QgaW5zdGFuY2VDb21tYW5kID0gbmV3IENvbW1hbmQoJ2luc3RhbmNlJylcbiAgLmRlc2NyaXB0aW9uKCdNYW5hZ2UgaW5zdGFuY2VzJylcblxuaW5zdGFuY2VDb21tYW5kXG4gIC5jb21tYW5kKCdsaXN0JylcbiAgLmRlc2NyaXB0aW9uKCdMaXN0IGluc3RhbmNlcycpXG4gIC5vcHRpb24oJy0tZW52aXJvbm1lbnQtaWQgPGlkPicsICdGaWx0ZXIgYnkgZW52aXJvbm1lbnQgSUQnKVxuICAub3B0aW9uKCctLXF1ZXVlLWlkIDxpZD4nLCAnRmlsdGVyIGJ5IHF1ZXVlIElEJylcbiAgLm9wdGlvbignLS1zdWJtaXNzaW9uLWlkIDxpZD4nLCAnRmlsdGVyIGJ5IHN1Ym1pc3Npb24gSUQnKVxuICAub3B0aW9uKCctLXN0YXR1cyA8c3RhdHVzPicsICdGaWx0ZXIgYnkgc3RhdHVzIChlLmcuIFJVTk5JTkcsIFRFUk1JTkFURUQsIENPU1RfQ0FMQ1VMQVRFRCwgRVJST1IpJylcbiAgLmFkZE9wdGlvbihuZXcgT3B0aW9uKCctLWZvcm1hdCA8Zm9ybWF0PicsICdPdXRwdXQgZm9ybWF0JykuY2hvaWNlcyhbJ3RhYmxlJywgJ2pzb24nLCAnY3N2J10pLmRlZmF1bHQoJ3RhYmxlJykpXG4gIC5hY3Rpb24oYXN5bmMgKG9wdHM6IHtcbiAgICBlbnZpcm9ubWVudElkPzogc3RyaW5nXG4gICAgcXVldWVJZD86IHN0cmluZ1xuICAgIHN1Ym1pc3Npb25JZD86IHN0cmluZ1xuICAgIHN0YXR1cz86IHN0cmluZ1xuICAgIGZvcm1hdDogT3V0cHV0Rm9ybWF0XG4gIH0pID0+IHtcbiAgICBjb25zdCBwYXJhbXMgPSBuZXcgVVJMU2VhcmNoUGFyYW1zKClcbiAgICBpZiAob3B0cy5lbnZpcm9ubWVudElkKSBwYXJhbXMuc2V0KCdlbnZpcm9ubWVudElkJywgb3B0cy5lbnZpcm9ubWVudElkKVxuICAgIGlmIChvcHRzLnF1ZXVlSWQpIHBhcmFtcy5zZXQoJ3F1ZXVlSWQnLCBvcHRzLnF1ZXVlSWQpXG4gICAgaWYgKG9wdHMuc3VibWlzc2lvbklkKSBwYXJhbXMuc2V0KCdzdWJtaXNzaW9uSWQnLCBvcHRzLnN1Ym1pc3Npb25JZClcbiAgICBpZiAob3B0cy5zdGF0dXMpIHBhcmFtcy5zZXQoJ3N0YXR1cycsIG9wdHMuc3RhdHVzKVxuICAgIGNvbnN0IHF1ZXJ5ID0gcGFyYW1zLnRvU3RyaW5nKCkgPyBgPyR7cGFyYW1zLnRvU3RyaW5nKCl9YCA6ICcnXG4gICAgY29uc3QgZGF0YSA9IGF3YWl0IGZldGNoSnNvbihhcGlVcmwoYC9pbnN0YW5jZXMke3F1ZXJ5fWApKSBhcyB7IGl0ZW1zOiB1bmtub3duW10gfVxuICAgIGZvcm1hdE91dHB1dChvcHRzLmZvcm1hdCA9PT0gJ2pzb24nID8gZGF0YSA6IGRhdGEuaXRlbXMsIG9wdHMuZm9ybWF0LCBpbnN0YW5jZUNvbHVtbnMpXG4gIH0pXG5cbmluc3RhbmNlQ29tbWFuZFxuICAuY29tbWFuZCgnZ2V0IDxpZD4nKVxuICAuZGVzY3JpcHRpb24oJ0dldCBhbiBpbnN0YW5jZSBieSBJRCcpXG4gIC5hZGRPcHRpb24obmV3IE9wdGlvbignLS1mb3JtYXQgPGZvcm1hdD4nLCAnT3V0cHV0IGZvcm1hdCcpLmNob2ljZXMoWyd0YWJsZScsICdqc29uJywgJ2NzdiddKS5kZWZhdWx0KCd0YWJsZScpKVxuICAuYWN0aW9uKGFzeW5jIChpZDogc3RyaW5nLCBvcHRzOiB7IGZvcm1hdDogT3V0cHV0Rm9ybWF0IH0pID0+IHtcbiAgICBjb25zdCBkYXRhID0gYXdhaXQgZmV0Y2hKc29uKGFwaVVybChgL2luc3RhbmNlcy8ke2lkfWApKVxuICAgIGZvcm1hdE91dHB1dChkYXRhLCBvcHRzLmZvcm1hdCwgaW5zdGFuY2VDb2x1bW5zKVxuICB9KVxuIl19
40
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5zdGFuY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29tbWFuZHMvaW5zdGFuY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUNBQTJDO0FBQzNDLGtDQUEyQztBQUMzQyw0Q0FBK0U7QUFFbEUsUUFBQSxlQUFlLEdBQUcsSUFBSSxtQkFBTyxDQUFDLFVBQVUsQ0FBQztLQUNuRCxXQUFXLENBQUMsa0JBQWtCLENBQUMsQ0FBQTtBQUVsQyx1QkFBZTtLQUNaLE9BQU8sQ0FBQyxNQUFNLENBQUM7S0FDZixXQUFXLENBQUMsZ0JBQWdCLENBQUM7S0FDN0IsTUFBTSxDQUFDLHVCQUF1QixFQUFFLDBCQUEwQixDQUFDO0tBQzNELE1BQU0sQ0FBQyxpQkFBaUIsRUFBRSxvQkFBb0IsQ0FBQztLQUMvQyxNQUFNLENBQUMsc0JBQXNCLEVBQUUseUJBQXlCLENBQUM7S0FDekQsTUFBTSxDQUFDLG1CQUFtQixFQUFFLHFFQUFxRSxDQUFDO0tBQ2xHLE1BQU0sQ0FBQyxhQUFhLEVBQUUscUNBQXFDLEVBQUUsSUFBSSxDQUFDO0tBQ2xFLFNBQVMsQ0FBQyxJQUFJLGtCQUFNLENBQUMsbUJBQW1CLEVBQUUsZUFBZSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztLQUM5RyxNQUFNLENBQUMsS0FBSyxFQUFFLElBT2QsRUFBRSxFQUFFO0lBQ0gsTUFBTSxNQUFNLEdBQUcsSUFBSSxlQUFlLEVBQUUsQ0FBQTtJQUNwQyxJQUFJLElBQUksQ0FBQyxhQUFhO1FBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxlQUFlLEVBQUUsSUFBSSxDQUFDLGFBQWEsQ0FBQyxDQUFBO0lBQ3ZFLElBQUksSUFBSSxDQUFDLE9BQU87UUFBRSxNQUFNLENBQUMsR0FBRyxDQUFDLFNBQVMsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUE7SUFDckQsSUFBSSxJQUFJLENBQUMsWUFBWTtRQUFFLE1BQU0sQ0FBQyxHQUFHLENBQUMsY0FBYyxFQUFFLElBQUksQ0FBQyxZQUFZLENBQUMsQ0FBQTtJQUNwRSxJQUFJLElBQUksQ0FBQyxNQUFNO1FBQUUsTUFBTSxDQUFDLEdBQUcsQ0FBQyxRQUFRLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFBO0lBQ2xELE1BQU0sQ0FBQyxHQUFHLENBQUMsT0FBTyxFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQTtJQUMvQixNQUFNLElBQUksR0FBRyxNQUFNLElBQUEsZ0JBQVMsRUFBQyxJQUFBLGFBQU0sRUFBQyxjQUFjLE1BQU0sRUFBRSxDQUFDLENBQXlCLENBQUE7SUFDcEYsSUFBQSx3QkFBWSxFQUFDLElBQUksQ0FBQyxNQUFNLEtBQUssTUFBTSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsSUFBSSxDQUFDLE1BQU0sRUFBRSwyQkFBZSxDQUFDLENBQUE7QUFDeEYsQ0FBQyxDQUFDLENBQUE7QUFFSix1QkFBZTtLQUNaLE9BQU8sQ0FBQyxVQUFVLENBQUM7S0FDbkIsV0FBVyxDQUFDLHVCQUF1QixDQUFDO0tBQ3BDLFNBQVMsQ0FBQyxJQUFJLGtCQUFNLENBQUMsbUJBQW1CLEVBQUUsZUFBZSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztLQUM5RyxNQUFNLENBQUMsS0FBSyxFQUFFLEVBQVUsRUFBRSxJQUE4QixFQUFFLEVBQUU7SUFDM0QsTUFBTSxJQUFJLEdBQUcsTUFBTSxJQUFBLGdCQUFTLEVBQUMsSUFBQSxhQUFNLEVBQUMsY0FBYyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUE7SUFDeEQsSUFBQSx3QkFBWSxFQUFDLElBQUksRUFBRSxJQUFJLENBQUMsTUFBTSxFQUFFLDJCQUFlLENBQUMsQ0FBQTtBQUNsRCxDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1hbmQsIE9wdGlvbiB9IGZyb20gJ2NvbW1hbmRlcidcbmltcG9ydCB7IGFwaVVybCwgZmV0Y2hKc29uIH0gZnJvbSAnLi4vYXV0aCdcbmltcG9ydCB7IGZvcm1hdE91dHB1dCwgaW5zdGFuY2VDb2x1bW5zLCB0eXBlIE91dHB1dEZvcm1hdCB9IGZyb20gJy4uL2Zvcm1hdC5qcydcblxuZXhwb3J0IGNvbnN0IGluc3RhbmNlQ29tbWFuZCA9IG5ldyBDb21tYW5kKCdpbnN0YW5jZScpXG4gIC5kZXNjcmlwdGlvbignTWFuYWdlIGluc3RhbmNlcycpXG5cbmluc3RhbmNlQ29tbWFuZFxuICAuY29tbWFuZCgnbGlzdCcpXG4gIC5kZXNjcmlwdGlvbignTGlzdCBpbnN0YW5jZXMnKVxuICAub3B0aW9uKCctLWVudmlyb25tZW50LWlkIDxpZD4nLCAnRmlsdGVyIGJ5IGVudmlyb25tZW50IElEJylcbiAgLm9wdGlvbignLS1xdWV1ZS1pZCA8aWQ+JywgJ0ZpbHRlciBieSBxdWV1ZSBJRCcpXG4gIC5vcHRpb24oJy0tc3VibWlzc2lvbi1pZCA8aWQ+JywgJ0ZpbHRlciBieSBzdWJtaXNzaW9uIElEJylcbiAgLm9wdGlvbignLS1zdGF0dXMgPHN0YXR1cz4nLCAnRmlsdGVyIGJ5IHN0YXR1cyAoZS5nLiBSVU5OSU5HLCBURVJNSU5BVEVELCBDT1NUX0NBTENVTEFURUQsIEVSUk9SKScpXG4gIC5vcHRpb24oJy0tbGltaXQgPG4+JywgJ01heGltdW0gbnVtYmVyIG9mIHJlc3VsdHMgdG8gcmV0dXJuJywgJzEwJylcbiAgLmFkZE9wdGlvbihuZXcgT3B0aW9uKCctLWZvcm1hdCA8Zm9ybWF0PicsICdPdXRwdXQgZm9ybWF0JykuY2hvaWNlcyhbJ3RhYmxlJywgJ2pzb24nLCAnY3N2J10pLmRlZmF1bHQoJ3RhYmxlJykpXG4gIC5hY3Rpb24oYXN5bmMgKG9wdHM6IHtcbiAgICBlbnZpcm9ubWVudElkPzogc3RyaW5nXG4gICAgcXVldWVJZD86IHN0cmluZ1xuICAgIHN1Ym1pc3Npb25JZD86IHN0cmluZ1xuICAgIHN0YXR1cz86IHN0cmluZ1xuICAgIGxpbWl0OiBzdHJpbmdcbiAgICBmb3JtYXQ6IE91dHB1dEZvcm1hdFxuICB9KSA9PiB7XG4gICAgY29uc3QgcGFyYW1zID0gbmV3IFVSTFNlYXJjaFBhcmFtcygpXG4gICAgaWYgKG9wdHMuZW52aXJvbm1lbnRJZCkgcGFyYW1zLnNldCgnZW52aXJvbm1lbnRJZCcsIG9wdHMuZW52aXJvbm1lbnRJZClcbiAgICBpZiAob3B0cy5xdWV1ZUlkKSBwYXJhbXMuc2V0KCdxdWV1ZUlkJywgb3B0cy5xdWV1ZUlkKVxuICAgIGlmIChvcHRzLnN1Ym1pc3Npb25JZCkgcGFyYW1zLnNldCgnc3VibWlzc2lvbklkJywgb3B0cy5zdWJtaXNzaW9uSWQpXG4gICAgaWYgKG9wdHMuc3RhdHVzKSBwYXJhbXMuc2V0KCdzdGF0dXMnLCBvcHRzLnN0YXR1cylcbiAgICBwYXJhbXMuc2V0KCdsaW1pdCcsIG9wdHMubGltaXQpXG4gICAgY29uc3QgZGF0YSA9IGF3YWl0IGZldGNoSnNvbihhcGlVcmwoYC9pbnN0YW5jZXM/JHtwYXJhbXN9YCkpIGFzIHsgaXRlbXM6IHVua25vd25bXSB9XG4gICAgZm9ybWF0T3V0cHV0KG9wdHMuZm9ybWF0ID09PSAnanNvbicgPyBkYXRhIDogZGF0YS5pdGVtcywgb3B0cy5mb3JtYXQsIGluc3RhbmNlQ29sdW1ucylcbiAgfSlcblxuaW5zdGFuY2VDb21tYW5kXG4gIC5jb21tYW5kKCdnZXQgPGlkPicpXG4gIC5kZXNjcmlwdGlvbignR2V0IGFuIGluc3RhbmNlIGJ5IElEJylcbiAgLmFkZE9wdGlvbihuZXcgT3B0aW9uKCctLWZvcm1hdCA8Zm9ybWF0PicsICdPdXRwdXQgZm9ybWF0JykuY2hvaWNlcyhbJ3RhYmxlJywgJ2pzb24nLCAnY3N2J10pLmRlZmF1bHQoJ3RhYmxlJykpXG4gIC5hY3Rpb24oYXN5bmMgKGlkOiBzdHJpbmcsIG9wdHM6IHsgZm9ybWF0OiBPdXRwdXRGb3JtYXQgfSkgPT4ge1xuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCBmZXRjaEpzb24oYXBpVXJsKGAvaW5zdGFuY2VzLyR7aWR9YCkpXG4gICAgZm9ybWF0T3V0cHV0KGRhdGEsIG9wdHMuZm9ybWF0LCBpbnN0YW5jZUNvbHVtbnMpXG4gIH0pXG4iXX0=
@@ -20,6 +20,30 @@ exports.submissionLogCommand
20
20
  (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.submissionLogColumns);
21
21
  }
22
22
  });
23
+ exports.submissionLogCommand
24
+ .command('tail [submission-id]')
25
+ .description('Tail logs for a submission')
26
+ .option('--limit <n>', 'Number of recent log lines to show', '50')
27
+ .option('--follow', 'Continuously poll for new log lines')
28
+ .action(async (submissionId, opts) => {
29
+ const limit = parseInt(opts.limit, 10);
30
+ const url = submissionId
31
+ ? `/submission-logs?submissionId=${submissionId}&limit=${limit}`
32
+ : `/submission-logs?limit=${limit}`;
33
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(url));
34
+ let lastTimestamp = printLogLines(data.items);
35
+ if (opts.follow) {
36
+ await pollLogs(lastTimestamp, async (since) => {
37
+ const base = submissionId
38
+ ? `/submission-logs?submissionId=${submissionId}`
39
+ : `/submission-logs`;
40
+ const res = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`${base}&since=${encodeURIComponent(since)}`));
41
+ return res.items;
42
+ }, (items) => {
43
+ lastTimestamp = printLogLines(items) ?? lastTimestamp;
44
+ });
45
+ }
46
+ });
23
47
  exports.taskLogCommand = new commander_1.Command('task-log')
24
48
  .description('Manage task logs');
25
49
  exports.taskLogCommand
@@ -40,4 +64,82 @@ exports.taskLogCommand
40
64
  (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.taskLogColumns);
41
65
  }
42
66
  });
43
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbW1hbmRzL2xvZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5Q0FBMkM7QUFDM0Msa0NBQTJDO0FBQzNDLDRDQUFvRztBQUV2RixRQUFBLG9CQUFvQixHQUFHLElBQUksbUJBQU8sQ0FBQyxnQkFBZ0IsQ0FBQztLQUM5RCxXQUFXLENBQUMsd0JBQXdCLENBQUMsQ0FBQTtBQUV4Qyw0QkFBb0I7S0FDakIsT0FBTyxDQUFDLE1BQU0sQ0FBQztLQUNmLFdBQVcsQ0FBQyw0QkFBNEIsQ0FBQztLQUN6QyxjQUFjLENBQUMsc0JBQXNCLEVBQUUsZUFBZSxDQUFDO0tBQ3ZELFNBQVMsQ0FBQyxJQUFJLGtCQUFNLENBQUMsbUJBQW1CLEVBQUUsZUFBZSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsT0FBTyxFQUFFLE1BQU0sRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FBQztLQUM5RyxNQUFNLENBQUMsS0FBSyxFQUFFLElBQW9ELEVBQUUsRUFBRTtJQUNyRSxNQUFNLElBQUksR0FBRyxNQUFNLElBQUEsZ0JBQVMsRUFBQyxJQUFBLGFBQU0sRUFBQyxpQ0FBaUMsSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDLENBQXlCLENBQUE7SUFDbEgsSUFBSSxJQUFJLENBQUMsTUFBTSxLQUFLLE9BQU8sRUFBRSxDQUFDO1FBQzNCLElBQUksQ0FBQyxLQUErQixDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUE7SUFDbEYsQ0FBQztTQUFNLENBQUM7UUFDTixJQUFBLHdCQUFZLEVBQUMsSUFBSSxDQUFDLE1BQU0sS0FBSyxNQUFNLENBQUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsTUFBTSxFQUFFLGdDQUFvQixDQUFDLENBQUE7SUFDN0YsQ0FBQztBQUNILENBQUMsQ0FBQyxDQUFBO0FBRVMsUUFBQSxjQUFjLEdBQUcsSUFBSSxtQkFBTyxDQUFDLFVBQVUsQ0FBQztLQUNsRCxXQUFXLENBQUMsa0JBQWtCLENBQUMsQ0FBQTtBQUVsQyxzQkFBYztLQUNYLE9BQU8sQ0FBQyxNQUFNLENBQUM7S0FDZixXQUFXLENBQUMsaUNBQWlDLENBQUM7S0FDOUMsY0FBYyxDQUFDLHNCQUFzQixFQUFFLGVBQWUsQ0FBQztLQUN2RCxNQUFNLENBQUMsZ0JBQWdCLEVBQUUsNkNBQTZDLENBQUM7S0FDdkUsU0FBUyxDQUFDLElBQUksa0JBQU0sQ0FBQyxtQkFBbUIsRUFBRSxlQUFlLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEtBQUssQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDO0tBQzlHLE1BQU0sQ0FBQyxLQUFLLEVBQUUsSUFBcUUsRUFBRSxFQUFFO0lBQ3RGLElBQUksR0FBRyxHQUFHLDJCQUEyQixJQUFJLENBQUMsWUFBWSxFQUFFLENBQUE7SUFDeEQsSUFBSSxJQUFJLENBQUMsTUFBTTtRQUFFLEdBQUcsSUFBSSxXQUFXLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQTtJQUNoRCxNQUFNLElBQUksR0FBRyxNQUFNLElBQUEsZ0JBQVMsRUFBQyxJQUFBLGFBQU0sRUFBQyxHQUFHLENBQUMsQ0FBeUIsQ0FBQTtJQUNqRSxJQUFJLElBQUksQ0FBQyxNQUFNLEtBQUssT0FBTyxFQUFFLENBQUM7UUFDM0IsSUFBSSxDQUFDLEtBQStCLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQTtJQUNsRixDQUFDO1NBQU0sQ0FBQztRQUNOLElBQUEsd0JBQVksRUFBQyxJQUFJLENBQUMsTUFBTSxLQUFLLE1BQU0sQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLElBQUksQ0FBQyxNQUFNLEVBQUUsMEJBQWMsQ0FBQyxDQUFBO0lBQ3ZGLENBQUM7QUFDSCxDQUFDLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1hbmQsIE9wdGlvbiB9IGZyb20gJ2NvbW1hbmRlcidcbmltcG9ydCB7IGFwaVVybCwgZmV0Y2hKc29uIH0gZnJvbSAnLi4vYXV0aCdcbmltcG9ydCB7IGZvcm1hdE91dHB1dCwgc3VibWlzc2lvbkxvZ0NvbHVtbnMsIHRhc2tMb2dDb2x1bW5zLCB0eXBlIE91dHB1dEZvcm1hdCB9IGZyb20gJy4uL2Zvcm1hdC5qcydcblxuZXhwb3J0IGNvbnN0IHN1Ym1pc3Npb25Mb2dDb21tYW5kID0gbmV3IENvbW1hbmQoJ3N1Ym1pc3Npb24tbG9nJylcbiAgLmRlc2NyaXB0aW9uKCdNYW5hZ2Ugc3VibWlzc2lvbiBsb2dzJylcblxuc3VibWlzc2lvbkxvZ0NvbW1hbmRcbiAgLmNvbW1hbmQoJ2xpc3QnKVxuICAuZGVzY3JpcHRpb24oJ0xpc3QgbG9ncyBmb3IgYSBzdWJtaXNzaW9uJylcbiAgLnJlcXVpcmVkT3B0aW9uKCctLXN1Ym1pc3Npb24taWQgPGlkPicsICdTdWJtaXNzaW9uIElEJylcbiAgLmFkZE9wdGlvbihuZXcgT3B0aW9uKCctLWZvcm1hdCA8Zm9ybWF0PicsICdPdXRwdXQgZm9ybWF0JykuY2hvaWNlcyhbJ3RhYmxlJywgJ2pzb24nLCAnY3N2J10pLmRlZmF1bHQoJ3RhYmxlJykpXG4gIC5hY3Rpb24oYXN5bmMgKG9wdHM6IHsgc3VibWlzc2lvbklkOiBzdHJpbmc7IGZvcm1hdDogT3V0cHV0Rm9ybWF0IH0pID0+IHtcbiAgICBjb25zdCBkYXRhID0gYXdhaXQgZmV0Y2hKc29uKGFwaVVybChgL3N1Ym1pc3Npb24tbG9ncz9zdWJtaXNzaW9uSWQ9JHtvcHRzLnN1Ym1pc3Npb25JZH1gKSkgYXMgeyBpdGVtczogdW5rbm93bltdIH1cbiAgICBpZiAob3B0cy5mb3JtYXQgPT09ICd0YWJsZScpIHtcbiAgICAgIChkYXRhLml0ZW1zIGFzIHsgbWVzc2FnZTogc3RyaW5nIH1bXSkuZm9yRWFjaChpdGVtID0+IGNvbnNvbGUubG9nKGl0ZW0ubWVzc2FnZSkpXG4gICAgfSBlbHNlIHtcbiAgICAgIGZvcm1hdE91dHB1dChvcHRzLmZvcm1hdCA9PT0gJ2pzb24nID8gZGF0YSA6IGRhdGEuaXRlbXMsIG9wdHMuZm9ybWF0LCBzdWJtaXNzaW9uTG9nQ29sdW1ucylcbiAgICB9XG4gIH0pXG5cbmV4cG9ydCBjb25zdCB0YXNrTG9nQ29tbWFuZCA9IG5ldyBDb21tYW5kKCd0YXNrLWxvZycpXG4gIC5kZXNjcmlwdGlvbignTWFuYWdlIHRhc2sgbG9ncycpXG5cbnRhc2tMb2dDb21tYW5kXG4gIC5jb21tYW5kKCdsaXN0JylcbiAgLmRlc2NyaXB0aW9uKCdMaXN0IHRhc2sgbG9ncyBmb3IgYSBzdWJtaXNzaW9uJylcbiAgLnJlcXVpcmVkT3B0aW9uKCctLXN1Ym1pc3Npb24taWQgPGlkPicsICdTdWJtaXNzaW9uIElEJylcbiAgLm9wdGlvbignLS10YXNrLWlkIDxpZD4nLCAnVGFzayBJRCAob3B0aW9uYWwsIGZpbHRlciBieSBzcGVjaWZpYyB0YXNrKScpXG4gIC5hZGRPcHRpb24obmV3IE9wdGlvbignLS1mb3JtYXQgPGZvcm1hdD4nLCAnT3V0cHV0IGZvcm1hdCcpLmNob2ljZXMoWyd0YWJsZScsICdqc29uJywgJ2NzdiddKS5kZWZhdWx0KCd0YWJsZScpKVxuICAuYWN0aW9uKGFzeW5jIChvcHRzOiB7IHN1Ym1pc3Npb25JZDogc3RyaW5nOyB0YXNrSWQ/OiBzdHJpbmc7IGZvcm1hdDogT3V0cHV0Rm9ybWF0IH0pID0+IHtcbiAgICBsZXQgdXJsID0gYC90YXNrLWxvZ3M/c3VibWlzc2lvbklkPSR7b3B0cy5zdWJtaXNzaW9uSWR9YFxuICAgIGlmIChvcHRzLnRhc2tJZCkgdXJsICs9IGAmdGFza0lkPSR7b3B0cy50YXNrSWR9YFxuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCBmZXRjaEpzb24oYXBpVXJsKHVybCkpIGFzIHsgaXRlbXM6IHVua25vd25bXSB9XG4gICAgaWYgKG9wdHMuZm9ybWF0ID09PSAndGFibGUnKSB7XG4gICAgICAoZGF0YS5pdGVtcyBhcyB7IG1lc3NhZ2U6IHN0cmluZyB9W10pLmZvckVhY2goaXRlbSA9PiBjb25zb2xlLmxvZyhpdGVtLm1lc3NhZ2UpKVxuICAgIH0gZWxzZSB7XG4gICAgICBmb3JtYXRPdXRwdXQob3B0cy5mb3JtYXQgPT09ICdqc29uJyA/IGRhdGEgOiBkYXRhLml0ZW1zLCBvcHRzLmZvcm1hdCwgdGFza0xvZ0NvbHVtbnMpXG4gICAgfVxuICB9KVxuIl19
67
+ exports.taskLogCommand
68
+ .command('tail <submission-id>')
69
+ .description('Tail logs for a submission or task')
70
+ .option('--task-id <id>', 'Task ID (optional, filter by specific task)')
71
+ .option('--limit <n>', 'Number of recent log lines to show', '50')
72
+ .option('--follow', 'Continuously poll for new log lines')
73
+ .option('--short', 'Omit task ID prefix')
74
+ .action(async (submissionId, opts) => {
75
+ const limit = parseInt(opts.limit, 10);
76
+ const buildUrl = (since) => {
77
+ const params = new URLSearchParams();
78
+ params.set('submissionId', submissionId);
79
+ if (opts.taskId)
80
+ params.set('taskId', opts.taskId);
81
+ if (since)
82
+ params.set('since', since);
83
+ else
84
+ params.set('limit', String(limit));
85
+ return `/task-logs?${params}`;
86
+ };
87
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(buildUrl()));
88
+ const id = (item) => opts.short ? undefined : (item.jobName ?? item.taskId ?? opts.taskId);
89
+ let lastTimestamp = printTaskLogLines(data.items, id);
90
+ if (opts.follow) {
91
+ await pollLogs(lastTimestamp, async (since) => {
92
+ const res = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(buildUrl(since)));
93
+ return res.items;
94
+ }, (items) => {
95
+ lastTimestamp = printTaskLogLines(items, id) ?? lastTimestamp;
96
+ });
97
+ }
98
+ });
99
+ function printLogLines(items, id) {
100
+ for (const item of items) {
101
+ const ts = new Date(item.timestamp).toISOString();
102
+ if (id) {
103
+ console.log(`${ts} ${id} ${item.message}`);
104
+ }
105
+ else {
106
+ console.log(`${ts} ${item.message}`);
107
+ }
108
+ }
109
+ return items.length > 0 ? items[items.length - 1].timestamp : undefined;
110
+ }
111
+ function printTaskLogLines(items, id) {
112
+ for (const item of items) {
113
+ const ts = new Date(item.timestamp).toISOString();
114
+ const prefix = id(item);
115
+ if (prefix) {
116
+ console.log(`${ts} ${prefix} ${item.message}`);
117
+ }
118
+ else {
119
+ console.log(`${ts} ${item.message}`);
120
+ }
121
+ }
122
+ return items.length > 0 ? items[items.length - 1].timestamp : undefined;
123
+ }
124
+ async function pollLogs(initialTimestamp, fetch, onItems) {
125
+ let since = initialTimestamp ?? new Date().toISOString();
126
+ return new Promise((resolve) => {
127
+ const interval = setInterval(async () => {
128
+ try {
129
+ const items = await fetch(since);
130
+ if (items.length > 0) {
131
+ onItems(items);
132
+ since = items[items.length - 1].timestamp;
133
+ }
134
+ }
135
+ catch {
136
+ // ignore transient errors during polling
137
+ }
138
+ }, 3000);
139
+ process.on('SIGINT', () => {
140
+ clearInterval(interval);
141
+ resolve();
142
+ });
143
+ });
144
+ }
145
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"log.js","sourceRoot":"","sources":["../../src/commands/log.ts"],"names":[],"mappings":"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","sourcesContent":["import { Command, Option } from 'commander'\nimport { apiUrl, fetchJson } from '../auth'\nimport { formatOutput, submissionLogColumns, taskLogColumns, type OutputFormat } from '../format.js'\n\nexport const submissionLogCommand = new Command('submission-log')\n  .description('Manage submission logs')\n\nsubmissionLogCommand\n  .command('list')\n  .description('List logs for a submission')\n  .requiredOption('--submission-id <id>', 'Submission ID')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { submissionId: string; format: OutputFormat }) => {\n    const data = await fetchJson(apiUrl(`/submission-logs?submissionId=${opts.submissionId}`)) as { items: unknown[] }\n    if (opts.format === 'table') {\n      (data.items as { message: string }[]).forEach(item => console.log(item.message))\n    } else {\n      formatOutput(opts.format === 'json' ? data : data.items, opts.format, submissionLogColumns)\n    }\n  })\n\nsubmissionLogCommand\n  .command('tail [submission-id]')\n  .description('Tail logs for a submission')\n  .option('--limit <n>', 'Number of recent log lines to show', '50')\n  .option('--follow', 'Continuously poll for new log lines')\n  .action(async (submissionId: string | undefined, opts: { limit: string; follow: boolean }) => {\n    const limit = parseInt(opts.limit, 10)\n    const url = submissionId\n      ? `/submission-logs?submissionId=${submissionId}&limit=${limit}`\n      : `/submission-logs?limit=${limit}`\n    const data = await fetchJson(apiUrl(url)) as { items: LogItem[] }\n    let lastTimestamp = printLogLines(data.items)\n\n    if (opts.follow) {\n      await pollLogs(lastTimestamp, async (since) => {\n        const base = submissionId\n          ? `/submission-logs?submissionId=${submissionId}`\n          : `/submission-logs`\n        const res = await fetchJson(apiUrl(`${base}&since=${encodeURIComponent(since)}`)) as { items: LogItem[] }\n        return res.items\n      }, (items) => {\n        lastTimestamp = printLogLines(items) ?? lastTimestamp\n      })\n    }\n  })\n\nexport const taskLogCommand = new Command('task-log')\n  .description('Manage task logs')\n\ntaskLogCommand\n  .command('list')\n  .description('List task logs for a submission')\n  .requiredOption('--submission-id <id>', 'Submission ID')\n  .option('--task-id <id>', 'Task ID (optional, filter by specific task)')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (opts: { submissionId: string; taskId?: string; format: OutputFormat }) => {\n    let url = `/task-logs?submissionId=${opts.submissionId}`\n    if (opts.taskId) url += `&taskId=${opts.taskId}`\n    const data = await fetchJson(apiUrl(url)) as { items: unknown[] }\n    if (opts.format === 'table') {\n      (data.items as { message: string }[]).forEach(item => console.log(item.message))\n    } else {\n      formatOutput(opts.format === 'json' ? data : data.items, opts.format, taskLogColumns)\n    }\n  })\n\ntaskLogCommand\n  .command('tail <submission-id>')\n  .description('Tail logs for a submission or task')\n  .option('--task-id <id>', 'Task ID (optional, filter by specific task)')\n  .option('--limit <n>', 'Number of recent log lines to show', '50')\n  .option('--follow', 'Continuously poll for new log lines')\n  .option('--short', 'Omit task ID prefix')\n  .action(async (submissionId: string, opts: { taskId?: string; limit: string; follow: boolean; short: boolean }) => {\n    const limit = parseInt(opts.limit, 10)\n    const buildUrl = (since?: string) => {\n      const params = new URLSearchParams()\n      params.set('submissionId', submissionId)\n      if (opts.taskId) params.set('taskId', opts.taskId)\n      if (since) params.set('since', since)\n      else params.set('limit', String(limit))\n      return `/task-logs?${params}`\n    }\n    const data = await fetchJson(apiUrl(buildUrl())) as { items: TaskLogItem[] }\n    const id = (item: TaskLogItem) => opts.short ? undefined : (item.jobName ?? item.taskId ?? opts.taskId)\n    let lastTimestamp = printTaskLogLines(data.items, id)\n\n    if (opts.follow) {\n      await pollLogs(lastTimestamp, async (since) => {\n        const res = await fetchJson(apiUrl(buildUrl(since))) as { items: TaskLogItem[] }\n        return res.items\n      }, (items) => {\n        lastTimestamp = printTaskLogLines(items, id) ?? lastTimestamp\n      })\n    }\n  })\n\ninterface LogItem {\n  timestamp: string\n  message: string\n}\n\ninterface TaskLogItem extends LogItem {\n  taskId?: string\n  jobName?: string\n}\n\nfunction printLogLines(items: LogItem[], id?: string): string | undefined {\n  for (const item of items) {\n    const ts = new Date(item.timestamp).toISOString()\n    if (id) {\n      console.log(`${ts}  ${id}  ${item.message}`)\n    } else {\n      console.log(`${ts}  ${item.message}`)\n    }\n  }\n  return items.length > 0 ? items[items.length - 1].timestamp : undefined\n}\n\nfunction printTaskLogLines(items: TaskLogItem[], id: (item: TaskLogItem) => string | undefined): string | undefined {\n  for (const item of items) {\n    const ts = new Date(item.timestamp).toISOString()\n    const prefix = id(item)\n    if (prefix) {\n      console.log(`${ts}  ${prefix}  ${item.message}`)\n    } else {\n      console.log(`${ts}  ${item.message}`)\n    }\n  }\n  return items.length > 0 ? items[items.length - 1].timestamp : undefined\n}\n\nasync function pollLogs(\n  initialTimestamp: string | undefined,\n  fetch: (since: string) => Promise<LogItem[]>,\n  onItems: (items: LogItem[]) => void,\n): Promise<void> {\n  let since = initialTimestamp ?? new Date().toISOString()\n  return new Promise((resolve) => {\n    const interval = setInterval(async () => {\n      try {\n        const items = await fetch(since)\n        if (items.length > 0) {\n          onItems(items)\n          since = items[items.length - 1].timestamp\n        }\n      } catch {\n        // ignore transient errors during polling\n      }\n    }, 3000)\n\n    process.on('SIGINT', () => {\n      clearInterval(interval)\n      resolve()\n    })\n  })\n}\n"]}