geni-bioinfo 0.1.5 → 0.1.7

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.
@@ -46,10 +46,13 @@ addCommonOptions(exports.costsCommand
46
46
  .command('submission')
47
47
  .description('Per-submission cost breakdown')
48
48
  .argument('[ids...]', 'Filter by submission IDs')
49
+ .option('--status <status>', 'Filter by submission status (case insensitive, e.g. succeeded, FAILED)')
49
50
  .option('--limit <n>', 'Max results (1–200)', '50')
50
51
  .option('--offset <n>', 'Pagination offset', '0')
51
52
  .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))).action(async (ids, opts) => {
52
53
  const params = buildParams(opts);
54
+ if (opts.status)
55
+ params.set('submissionStatus', opts.status.toUpperCase());
53
56
  ids.forEach(id => params.append('id', id));
54
57
  params.set('limit', opts.limit);
55
58
  params.set('offset', opts.offset);
@@ -77,6 +80,51 @@ addCommonOptions(exports.costsCommand
77
80
  const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/costs/instance-type${qs(buildParams(opts))}`));
78
81
  (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.analyticsByInstanceTypeColumns);
79
82
  });
83
+ exports.costsCommand
84
+ .command('task')
85
+ .description('Per-task cost breakdown')
86
+ .argument('[ids...]', 'Filter by task IDs')
87
+ .option('--submission-id <id>', 'Filter by submission ID')
88
+ .option('--limit <n>', 'Max results (1–200)', '50')
89
+ .option('--offset <n>', 'Pagination offset', '0')
90
+ .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
91
+ .action(async (ids, opts) => {
92
+ const params = new URLSearchParams();
93
+ ids.forEach(id => params.append('id', id));
94
+ if (opts.submissionId)
95
+ params.set('submissionId', opts.submissionId);
96
+ params.set('limit', opts.limit);
97
+ params.set('offset', opts.offset);
98
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/costs/task${qs(params)}`));
99
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.costsByTaskColumns);
100
+ });
101
+ addCommonOptions(exports.costsCommand
102
+ .command('instance')
103
+ .description('Per-instance cost breakdown')
104
+ .option('--environment-id <id>', 'Filter by environment ID')
105
+ .option('--engine-id <id>', 'Filter by engine ID')
106
+ .option('--queue-id <id>', 'Filter by queue ID')
107
+ .option('--instance-type <type>', 'Filter by EC2 instance type (e.g. m5.xlarge)')
108
+ .option('--execution-mode <mode>', 'Filter by execution mode (e.g. SPOT, ON_DEMAND)')
109
+ .option('--limit <n>', 'Max results (1–200)', '50')
110
+ .option('--offset <n>', 'Pagination offset', '0')
111
+ .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))).action(async (opts) => {
112
+ const params = buildParams(opts);
113
+ if (opts.environmentId)
114
+ params.set('environmentId', opts.environmentId);
115
+ if (opts.engineId)
116
+ params.set('engineId', opts.engineId);
117
+ if (opts.queueId)
118
+ params.set('queueId', opts.queueId);
119
+ if (opts.instanceType)
120
+ params.set('instanceType', opts.instanceType);
121
+ if (opts.executionMode)
122
+ params.set('executionMode', opts.executionMode);
123
+ params.set('limit', opts.limit);
124
+ params.set('offset', opts.offset);
125
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/costs/instance${qs(params)}`));
126
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.costsByInstanceColumns);
127
+ });
80
128
  addCommonOptions(exports.costsCommand
81
129
  .command('engine')
82
130
  .description('Cost grouped by engine')
@@ -104,8 +152,13 @@ addCommonOptions(exports.costsCommand
104
152
  exports.costsCommand
105
153
  .command('sync')
106
154
  .description('Trigger a CUR actual-cost sync')
107
- .action(async () => {
108
- const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)('/costs/sync'), { method: 'POST' });
155
+ .option('--reset', 'Clear all actual cost data before syncing, forcing a full recalculation from CUR')
156
+ .action(async (opts) => {
157
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)('/costs/sync'), {
158
+ method: 'POST',
159
+ headers: { 'Content-Type': 'application/json' },
160
+ body: JSON.stringify({ reset: opts.reset ?? false }),
161
+ });
109
162
  console.log(`Sync job enqueued: ${data.jobId}`);
110
163
  });
111
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"costs.js","sourceRoot":"","sources":["../../src/commands/costs.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  costsByEngineColumns,\n  costsByExecutionModeColumns,\n  type OutputFormat,\n} from '../format.js'\n\ntype BaseOpts = {\n  startDate?: string\n  endDate?: string\n  submissionStatus?: string\n}\n\nconst DATE_RE = /^\\d{4}-\\d{2}-\\d{2}$/\n\nfunction parseDate(value: string): string {\n  if (!DATE_RE.test(value)) throw new Error(`Expected YYYY-MM-DD, got \"${value}\"`)\n  return value\n}\n\nfunction addCommonOptions(cmd: Command): Command {\n  return cmd\n    .option('--start-date <YYYY-MM-DD>', 'Start date (UTC, inclusive)', parseDate)\n    .option('--end-date <YYYY-MM-DD>', 'End date (UTC, inclusive)', parseDate)\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\n  if (opts.startDate) params.set('from', `${opts.startDate}T00:00:00.000Z`)\n\n  if (opts.endDate) {\n    const d = new Date(`${opts.endDate}T00:00:00.000Z`)\n    d.setUTCDate(d.getUTCDate() + 1)\n    params.set('to', d.toISOString())\n  }\n\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 costsCommand = new Command('costs')\n  .description('View cost analytics')\n\naddCommonOptions(\n  costsCommand\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(`/costs/summary${qs(buildParams(opts))}`))\n  formatOutput(data, opts.format, analyticsSummaryColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('submission')\n    .description('Per-submission cost breakdown')\n    .argument('[ids...]', 'Filter by submission IDs')\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 (ids: string[], opts: BaseOpts & { limit: string; offset: string; format: OutputFormat }) => {\n  const params = buildParams(opts)\n  ids.forEach(id => params.append('id', id))\n  params.set('limit', opts.limit)\n  params.set('offset', opts.offset)\n  const data = await fetchJson(apiUrl(`/costs/submission${qs(params)}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsBySubmissionColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('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(`/costs/environment${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByEnvironmentColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('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(`/costs/queue${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByQueueColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('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(`/costs/instance-type${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByInstanceTypeColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('engine')\n    .description('Cost grouped by engine')\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(`/costs/engine${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, costsByEngineColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('execution-mode')\n    .description('Cost grouped by execution mode (spot / on-demand)')\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(`/costs/execution-mode${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, costsByExecutionModeColumns)\n})\n\naddCommonOptions(\n  costsCommand\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(`/costs/over-time${qs(params)}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsOverTimeColumns)\n})\n\ncostsCommand\n  .command('sync')\n  .description('Trigger a CUR actual-cost sync')\n  .action(async () => {\n    const data = await fetchJson(apiUrl('/costs/sync'), { method: 'POST' }) as { jobId: string }\n    console.log(`Sync job enqueued: ${data.jobId}`)\n  })\n"]}
164
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"costs.js","sourceRoot":"","sources":["../../src/commands/costs.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  costsByEngineColumns,\n  costsByExecutionModeColumns,\n  costsByTaskColumns,\n  costsByInstanceColumns,\n  type OutputFormat,\n} from '../format.js'\n\ntype BaseOpts = {\n  startDate?: string\n  endDate?: string\n  submissionStatus?: string\n}\n\nconst DATE_RE = /^\\d{4}-\\d{2}-\\d{2}$/\n\nfunction parseDate(value: string): string {\n  if (!DATE_RE.test(value)) throw new Error(`Expected YYYY-MM-DD, got \"${value}\"`)\n  return value\n}\n\nfunction addCommonOptions(cmd: Command): Command {\n  return cmd\n    .option('--start-date <YYYY-MM-DD>', 'Start date (UTC, inclusive)', parseDate)\n    .option('--end-date <YYYY-MM-DD>', 'End date (UTC, inclusive)', parseDate)\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\n  if (opts.startDate) params.set('from', `${opts.startDate}T00:00:00.000Z`)\n\n  if (opts.endDate) {\n    const d = new Date(`${opts.endDate}T00:00:00.000Z`)\n    d.setUTCDate(d.getUTCDate() + 1)\n    params.set('to', d.toISOString())\n  }\n\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 costsCommand = new Command('costs')\n  .description('View cost analytics')\n\naddCommonOptions(\n  costsCommand\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(`/costs/summary${qs(buildParams(opts))}`))\n  formatOutput(data, opts.format, analyticsSummaryColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('submission')\n    .description('Per-submission cost breakdown')\n    .argument('[ids...]', 'Filter by submission IDs')\n    .option('--status <status>', 'Filter by submission status (case insensitive, e.g. succeeded, FAILED)')\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 (ids: string[], opts: BaseOpts & { status?: string; limit: string; offset: string; format: OutputFormat }) => {\n  const params = buildParams(opts)\n  if (opts.status) params.set('submissionStatus', opts.status.toUpperCase())\n  ids.forEach(id => params.append('id', id))\n  params.set('limit', opts.limit)\n  params.set('offset', opts.offset)\n  const data = await fetchJson(apiUrl(`/costs/submission${qs(params)}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsBySubmissionColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('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(`/costs/environment${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByEnvironmentColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('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(`/costs/queue${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByQueueColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('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(`/costs/instance-type${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsByInstanceTypeColumns)\n})\n\ncostsCommand\n  .command('task')\n  .description('Per-task cost breakdown')\n  .argument('[ids...]', 'Filter by task IDs')\n  .option('--submission-id <id>', 'Filter by submission ID')\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 (ids: string[], opts: { submissionId?: string; limit: string; offset: string; format: OutputFormat }) => {\n    const params = new URLSearchParams()\n    ids.forEach(id => params.append('id', id))\n    if (opts.submissionId) params.set('submissionId', opts.submissionId)\n    params.set('limit', opts.limit)\n    params.set('offset', opts.offset)\n    const data = await fetchJson(apiUrl(`/costs/task${qs(params)}`)) as { items: unknown[] }\n    formatOutput(opts.format === 'json' ? data : data.items, opts.format, costsByTaskColumns)\n  })\n\naddCommonOptions(\n  costsCommand\n    .command('instance')\n    .description('Per-instance cost breakdown')\n    .option('--environment-id <id>',   'Filter by environment ID')\n    .option('--engine-id <id>',        'Filter by engine ID')\n    .option('--queue-id <id>',         'Filter by queue ID')\n    .option('--instance-type <type>',  'Filter by EC2 instance type (e.g. m5.xlarge)')\n    .option('--execution-mode <mode>', 'Filter by execution mode (e.g. SPOT, ON_DEMAND)')\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 & {\n  environmentId?: string; engineId?: string; queueId?: string;\n  instanceType?: string; executionMode?: string;\n  limit: string; offset: string; format: OutputFormat\n}) => {\n  const params = buildParams(opts)\n  if (opts.environmentId) params.set('environmentId', opts.environmentId)\n  if (opts.engineId)      params.set('engineId', opts.engineId)\n  if (opts.queueId)       params.set('queueId', opts.queueId)\n  if (opts.instanceType)  params.set('instanceType', opts.instanceType)\n  if (opts.executionMode) params.set('executionMode', opts.executionMode)\n  params.set('limit', opts.limit)\n  params.set('offset', opts.offset)\n  const data = await fetchJson(apiUrl(`/costs/instance${qs(params)}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, costsByInstanceColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('engine')\n    .description('Cost grouped by engine')\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(`/costs/engine${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, costsByEngineColumns)\n})\n\naddCommonOptions(\n  costsCommand\n    .command('execution-mode')\n    .description('Cost grouped by execution mode (spot / on-demand)')\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(`/costs/execution-mode${qs(buildParams(opts))}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, costsByExecutionModeColumns)\n})\n\naddCommonOptions(\n  costsCommand\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(`/costs/over-time${qs(params)}`)) as { items: unknown[] }\n  formatOutput(opts.format === 'json' ? data : data.items, opts.format, analyticsOverTimeColumns)\n})\n\ncostsCommand\n  .command('sync')\n  .description('Trigger a CUR actual-cost sync')\n  .option('--reset', 'Clear all actual cost data before syncing, forcing a full recalculation from CUR')\n  .action(async (opts: { reset?: boolean }) => {\n    const data = await fetchJson(apiUrl('/costs/sync'), {\n      method: 'POST',\n      headers: { 'Content-Type': 'application/json' },\n      body: JSON.stringify({ reset: opts.reset ?? false }),\n    }) as { jobId: string }\n    console.log(`Sync job enqueued: ${data.jobId}`)\n  })\n"]}
@@ -1,3 +1,3 @@
1
1
  import { Command } from 'commander';
2
- export declare const submissionLogCommand: Command;
3
- export declare const taskLogCommand: Command;
2
+ export declare const logsSubcommand: Command;
3
+ export declare const taskLogsSubcommand: Command;
@@ -1,99 +1,80 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.taskLogCommand = exports.submissionLogCommand = void 0;
3
+ exports.taskLogsSubcommand = exports.logsSubcommand = void 0;
4
4
  const commander_1 = require("commander");
5
5
  const auth_1 = require("../auth");
6
6
  const format_js_1 = require("../format.js");
7
- exports.submissionLogCommand = new commander_1.Command('submission-log')
8
- .description('Manage submission logs');
9
- exports.submissionLogCommand
10
- .command('list')
11
- .description('List logs for a submission')
12
- .requiredOption('--submission-id <id>', 'Submission ID')
7
+ exports.logsSubcommand = new commander_1.Command('logs')
8
+ .description('View logs for a submission')
9
+ .argument('<submission-id>', 'Submission ID')
10
+ .option('--tail', 'Show last N lines instead of all logs')
11
+ .option('--limit <n>', 'Number of lines to show with --tail', '50')
12
+ .option('--follow', 'Continuously poll for new log lines')
13
13
  .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
14
- .action(async (opts) => {
15
- const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/submission-logs?submissionId=${opts.submissionId}`));
16
- if (opts.format === 'table') {
17
- data.items.forEach(item => console.log(item.message));
14
+ .action(async (submissionId, opts) => {
15
+ if (opts.tail || opts.follow) {
16
+ const limit = parseInt(opts.limit, 10);
17
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/submission-logs?submissionId=${submissionId}&limit=${limit}`));
18
+ let lastTimestamp = printLogLines(data.items);
19
+ if (opts.follow) {
20
+ await pollLogs(lastTimestamp, async (since) => {
21
+ const res = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/submission-logs?submissionId=${submissionId}&since=${encodeURIComponent(since)}`));
22
+ return res.items;
23
+ }, (items) => {
24
+ lastTimestamp = printLogLines(items) ?? lastTimestamp;
25
+ });
26
+ }
18
27
  }
19
28
  else {
20
- (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.submissionLogColumns);
29
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/submission-logs?submissionId=${submissionId}`));
30
+ if (opts.format === 'table') {
31
+ data.items.forEach(item => console.log(item.message));
32
+ }
33
+ else {
34
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.submissionLogColumns);
35
+ }
21
36
  }
22
37
  });
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')
38
+ exports.taskLogsSubcommand = new commander_1.Command('logs')
39
+ .description('View logs for a task')
40
+ .argument('<task-id>', 'Task ID')
41
+ .option('--tail', 'Show last N lines instead of all logs')
42
+ .option('--limit <n>', 'Number of lines to show with --tail', '50')
27
43
  .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
- });
47
- exports.taskLogCommand = new commander_1.Command('task-log')
48
- .description('Manage task logs');
49
- exports.taskLogCommand
50
- .command('list')
51
- .description('List task logs for a submission')
52
- .requiredOption('--submission-id <id>', 'Submission ID')
53
- .option('--task-id <id>', 'Task ID (optional, filter by specific task)')
44
+ .option('--short', 'Omit task ID prefix')
54
45
  .addOption(new commander_1.Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))
55
- .action(async (opts) => {
56
- let url = `/task-logs?submissionId=${opts.submissionId}`;
57
- if (opts.taskId)
58
- url += `&taskId=${opts.taskId}`;
59
- const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(url));
60
- if (opts.format === 'table') {
61
- data.items.forEach(item => console.log(item.message));
46
+ .action(async (taskId, opts) => {
47
+ if (opts.tail || opts.follow) {
48
+ const limit = parseInt(opts.limit, 10);
49
+ const buildUrl = (since) => {
50
+ const params = new URLSearchParams();
51
+ params.set('taskId', taskId);
52
+ if (since)
53
+ params.set('since', since);
54
+ else
55
+ params.set('limit', String(limit));
56
+ return `/task-logs?${params}`;
57
+ };
58
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(buildUrl()));
59
+ const id = (item) => opts.short ? undefined : (item.jobName ?? item.taskId ?? taskId);
60
+ let lastTimestamp = printTaskLogLines(data.items, id);
61
+ if (opts.follow) {
62
+ await pollLogs(lastTimestamp, async (since) => {
63
+ const res = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(buildUrl(since)));
64
+ return res.items;
65
+ }, (items) => {
66
+ lastTimestamp = printTaskLogLines(items, id) ?? lastTimestamp;
67
+ });
68
+ }
62
69
  }
63
70
  else {
64
- (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.taskLogColumns);
65
- }
66
- });
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
- });
71
+ const data = await (0, auth_1.fetchJson)((0, auth_1.apiUrl)(`/task-logs?taskId=${taskId}`));
72
+ if (opts.format === 'table') {
73
+ data.items.forEach(item => console.log(item.message));
74
+ }
75
+ else {
76
+ (0, format_js_1.formatOutput)(opts.format === 'json' ? data : data.items, opts.format, format_js_1.taskLogColumns);
77
+ }
97
78
  }
98
79
  });
99
80
  function printLogLines(items, id) {
@@ -142,4 +123,4 @@ async function pollLogs(initialTimestamp, fetch, onItems) {
142
123
  });
143
124
  });
144
125
  }
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"]}
126
+ //# 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 logsSubcommand = new Command('logs')\n  .description('View logs for a submission')\n  .argument('<submission-id>', 'Submission ID')\n  .option('--tail', 'Show last N lines instead of all logs')\n  .option('--limit <n>', 'Number of lines to show with --tail', '50')\n  .option('--follow', 'Continuously poll for new log lines')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (submissionId: string, opts: { tail: boolean; limit: string; follow: boolean; format: OutputFormat }) => {\n    if (opts.tail || opts.follow) {\n      const limit = parseInt(opts.limit, 10)\n      const data = await fetchJson(apiUrl(`/submission-logs?submissionId=${submissionId}&limit=${limit}`)) as { items: LogItem[] }\n      let lastTimestamp = printLogLines(data.items)\n      if (opts.follow) {\n        await pollLogs(lastTimestamp, async (since) => {\n          const res = await fetchJson(apiUrl(`/submission-logs?submissionId=${submissionId}&since=${encodeURIComponent(since)}`)) as { items: LogItem[] }\n          return res.items\n        }, (items) => {\n          lastTimestamp = printLogLines(items) ?? lastTimestamp\n        })\n      }\n    } else {\n      const data = await fetchJson(apiUrl(`/submission-logs?submissionId=${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  })\n\nexport const taskLogsSubcommand = new Command('logs')\n  .description('View logs for a task')\n  .argument('<task-id>', 'Task ID')\n  .option('--tail', 'Show last N lines instead of all logs')\n  .option('--limit <n>', 'Number of lines to show with --tail', '50')\n  .option('--follow', 'Continuously poll for new log lines')\n  .option('--short', 'Omit task ID prefix')\n  .addOption(new Option('--format <format>', 'Output format').choices(['table', 'json', 'csv']).default('table'))\n  .action(async (taskId: string, opts: { tail: boolean; limit: string; follow: boolean; short: boolean; format: OutputFormat }) => {\n    if (opts.tail || opts.follow) {\n      const limit = parseInt(opts.limit, 10)\n      const buildUrl = (since?: string) => {\n        const params = new URLSearchParams()\n        params.set('taskId', 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 ?? taskId)\n      let lastTimestamp = printTaskLogLines(data.items, id)\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    } else {\n      const data = await fetchJson(apiUrl(`/task-logs?taskId=${taskId}`)) 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  })\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"]}
@@ -2,23 +2,57 @@
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.loginCommand = void 0;
4
4
  const commander_1 = require("commander");
5
+ const readline_1 = require("readline");
5
6
  const auth_1 = require("../auth");
7
+ async function prompt(question) {
8
+ const rl = (0, readline_1.createInterface)({ input: process.stdin, output: process.stdout });
9
+ return new Promise(resolve => rl.question(question, answer => { rl.close(); resolve(answer); }));
10
+ }
11
+ async function promptPassword(question) {
12
+ process.stdout.write(question);
13
+ process.stdin.setRawMode(true);
14
+ process.stdin.resume();
15
+ process.stdin.setEncoding('utf8');
16
+ return new Promise(resolve => {
17
+ let password = '';
18
+ process.stdin.on('data', function handler(ch) {
19
+ if (ch === '\n' || ch === '\r' || ch === '') {
20
+ process.stdin.setRawMode(false);
21
+ process.stdin.pause();
22
+ process.stdin.removeListener('data', handler);
23
+ process.stdout.write('\n');
24
+ resolve(password);
25
+ }
26
+ else if (ch === '' || ch === '\b') {
27
+ if (password.length > 0) {
28
+ password = password.slice(0, -1);
29
+ process.stdout.write('\b \b');
30
+ }
31
+ }
32
+ else {
33
+ password += ch;
34
+ process.stdout.write('*');
35
+ }
36
+ });
37
+ });
38
+ }
6
39
  exports.loginCommand = new commander_1.Command('login')
7
40
  .description('Login into GENI Workflows and store credentials')
8
- .option('--email <email>', 'User email')
9
41
  .option('--password <password>', 'User password')
10
42
  .action(async (opts) => {
11
- if (!opts.email || !opts.password) {
12
- exports.loginCommand.help();
13
- return;
43
+ const email = await prompt('Email: ');
44
+ const password = opts.password ?? await promptPassword('Password: ');
45
+ if (!email || !password) {
46
+ console.error('Email and password are required.');
47
+ process.exit(1);
14
48
  }
15
49
  const res = await fetch((0, auth_1.apiUrl)('/auth/login'), {
16
50
  method: 'POST',
17
51
  headers: { 'Content-Type': 'application/json' },
18
- body: JSON.stringify({ email: opts.email, password: opts.password }),
52
+ body: JSON.stringify({ email, password }),
19
53
  });
20
54
  const data = await (0, auth_1.readJsonResponse)(res, { method: 'POST', url: (0, auth_1.apiUrl)('/auth/login') });
21
55
  (0, auth_1.saveCredentials)(data.accessToken);
22
56
  console.log('Logged in. Credentials saved to ~/.geni/credentials.json');
23
57
  });
24
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9naW4uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29tbWFuZHMvbG9naW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUNBQW9DO0FBQ3BDLGtDQUFvRTtBQUV2RCxRQUFBLFlBQVksR0FBRyxJQUFJLG1CQUFPLENBQUMsT0FBTyxDQUFDO0tBQzdDLFdBQVcsQ0FBQyxpREFBaUQsQ0FBQztLQUM5RCxNQUFNLENBQUMsaUJBQWlCLEVBQUUsWUFBWSxDQUFDO0tBQ3ZDLE1BQU0sQ0FBQyx1QkFBdUIsRUFBRSxlQUFlLENBQUM7S0FDaEQsTUFBTSxDQUFDLEtBQUssRUFBRSxJQUEyQyxFQUFFLEVBQUU7SUFDNUQsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLElBQUksQ0FBQyxJQUFJLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDbEMsb0JBQVksQ0FBQyxJQUFJLEVBQUUsQ0FBQTtRQUNuQixPQUFNO0lBQ1IsQ0FBQztJQUNELE1BQU0sR0FBRyxHQUFHLE1BQU0sS0FBSyxDQUFDLElBQUEsYUFBTSxFQUFDLGFBQWEsQ0FBQyxFQUFFO1FBQzdDLE1BQU0sRUFBRSxNQUFNO1FBQ2QsT0FBTyxFQUFFLEVBQUUsY0FBYyxFQUFFLGtCQUFrQixFQUFFO1FBQy9DLElBQUksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLEVBQUUsS0FBSyxFQUFFLElBQUksQ0FBQyxLQUFLLEVBQUUsUUFBUSxFQUFFLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztLQUNyRSxDQUFDLENBQUM7SUFDSCxNQUFNLElBQUksR0FBRyxNQUFNLElBQUEsdUJBQWdCLEVBQUMsR0FBRyxFQUFFLEVBQUUsTUFBTSxFQUFFLE1BQU0sRUFBRSxHQUFHLEVBQUUsSUFBQSxhQUFNLEVBQUMsYUFBYSxDQUFDLEVBQUUsQ0FFdEYsQ0FBQztJQUNGLElBQUEsc0JBQWUsRUFBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQUM7SUFDbEMsT0FBTyxDQUFDLEdBQUcsQ0FBQywwREFBMEQsQ0FBQyxDQUFDO0FBQzFFLENBQUMsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tbWFuZCB9IGZyb20gJ2NvbW1hbmRlcic7XG5pbXBvcnQgeyBhcGlVcmwsIHJlYWRKc29uUmVzcG9uc2UsIHNhdmVDcmVkZW50aWFscyB9IGZyb20gJy4uL2F1dGgnO1xuXG5leHBvcnQgY29uc3QgbG9naW5Db21tYW5kID0gbmV3IENvbW1hbmQoJ2xvZ2luJylcbiAgLmRlc2NyaXB0aW9uKCdMb2dpbiBpbnRvIEdFTkkgV29ya2Zsb3dzIGFuZCBzdG9yZSBjcmVkZW50aWFscycpXG4gIC5vcHRpb24oJy0tZW1haWwgPGVtYWlsPicsICdVc2VyIGVtYWlsJylcbiAgLm9wdGlvbignLS1wYXNzd29yZCA8cGFzc3dvcmQ+JywgJ1VzZXIgcGFzc3dvcmQnKVxuICAuYWN0aW9uKGFzeW5jIChvcHRzOiB7IGVtYWlsPzogc3RyaW5nOyBwYXNzd29yZD86IHN0cmluZyB9KSA9PiB7XG4gICAgaWYgKCFvcHRzLmVtYWlsIHx8ICFvcHRzLnBhc3N3b3JkKSB7XG4gICAgICBsb2dpbkNvbW1hbmQuaGVscCgpXG4gICAgICByZXR1cm5cbiAgICB9XG4gICAgY29uc3QgcmVzID0gYXdhaXQgZmV0Y2goYXBpVXJsKCcvYXV0aC9sb2dpbicpLCB7XG4gICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgIGhlYWRlcnM6IHsgJ0NvbnRlbnQtVHlwZSc6ICdhcHBsaWNhdGlvbi9qc29uJyB9LFxuICAgICAgYm9keTogSlNPTi5zdHJpbmdpZnkoeyBlbWFpbDogb3B0cy5lbWFpbCwgcGFzc3dvcmQ6IG9wdHMucGFzc3dvcmQgfSksXG4gICAgfSk7XG4gICAgY29uc3QgZGF0YSA9IGF3YWl0IHJlYWRKc29uUmVzcG9uc2UocmVzLCB7IG1ldGhvZDogJ1BPU1QnLCB1cmw6IGFwaVVybCgnL2F1dGgvbG9naW4nKSB9KSBhcyB7XG4gICAgICBhY2Nlc3NUb2tlbjogc3RyaW5nO1xuICAgIH07XG4gICAgc2F2ZUNyZWRlbnRpYWxzKGRhdGEuYWNjZXNzVG9rZW4pO1xuICAgIGNvbnNvbGUubG9nKCdMb2dnZWQgaW4uIENyZWRlbnRpYWxzIHNhdmVkIHRvIH4vLmdlbmkvY3JlZGVudGlhbHMuanNvbicpO1xuICB9KTtcbiJdfQ==
58
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9naW4uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29tbWFuZHMvbG9naW4udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEseUNBQW9DO0FBQ3BDLHVDQUEyQztBQUMzQyxrQ0FBb0U7QUFFcEUsS0FBSyxVQUFVLE1BQU0sQ0FBQyxRQUFnQjtJQUNwQyxNQUFNLEVBQUUsR0FBRyxJQUFBLDBCQUFlLEVBQUMsRUFBRSxLQUFLLEVBQUUsT0FBTyxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsT0FBTyxDQUFDLE1BQU0sRUFBRSxDQUFDLENBQUM7SUFDN0UsT0FBTyxJQUFJLE9BQU8sQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxRQUFRLENBQUMsUUFBUSxFQUFFLE1BQU0sQ0FBQyxFQUFFLEdBQUcsRUFBRSxDQUFDLEtBQUssRUFBRSxDQUFDLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUNuRyxDQUFDO0FBRUQsS0FBSyxVQUFVLGNBQWMsQ0FBQyxRQUFnQjtJQUM1QyxPQUFPLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxRQUFRLENBQUMsQ0FBQztJQUMvQixPQUFPLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUMvQixPQUFPLENBQUMsS0FBSyxDQUFDLE1BQU0sRUFBRSxDQUFDO0lBQ3ZCLE9BQU8sQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQ2xDLE9BQU8sSUFBSSxPQUFPLENBQUMsT0FBTyxDQUFDLEVBQUU7UUFDM0IsSUFBSSxRQUFRLEdBQUcsRUFBRSxDQUFDO1FBQ2xCLE9BQU8sQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLE1BQU0sRUFBRSxTQUFTLE9BQU8sQ0FBQyxFQUFVO1lBQ2xELElBQUksRUFBRSxLQUFLLElBQUksSUFBSSxFQUFFLEtBQUssSUFBSSxJQUFJLEVBQUUsS0FBSyxHQUFHLEVBQUUsQ0FBQztnQkFDN0MsT0FBTyxDQUFDLEtBQUssQ0FBQyxVQUFVLENBQUMsS0FBSyxDQUFDLENBQUM7Z0JBQ2hDLE9BQU8sQ0FBQyxLQUFLLENBQUMsS0FBSyxFQUFFLENBQUM7Z0JBQ3RCLE9BQU8sQ0FBQyxLQUFLLENBQUMsY0FBYyxDQUFDLE1BQU0sRUFBRSxPQUFPLENBQUMsQ0FBQztnQkFDOUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUM7Z0JBQzNCLE9BQU8sQ0FBQyxRQUFRLENBQUMsQ0FBQztZQUNwQixDQUFDO2lCQUFNLElBQUksRUFBRSxLQUFLLEdBQUcsSUFBSSxFQUFFLEtBQUssSUFBSSxFQUFFLENBQUM7Z0JBQ3JDLElBQUksUUFBUSxDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQztvQkFDeEIsUUFBUSxHQUFHLFFBQVEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7b0JBQ2pDLE9BQU8sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDO2dCQUNoQyxDQUFDO1lBQ0gsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLFFBQVEsSUFBSSxFQUFFLENBQUM7Z0JBQ2YsT0FBTyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDNUIsQ0FBQztRQUNILENBQUMsQ0FBQyxDQUFDO0lBQ0wsQ0FBQyxDQUFDLENBQUM7QUFDTCxDQUFDO0FBRVksUUFBQSxZQUFZLEdBQUcsSUFBSSxtQkFBTyxDQUFDLE9BQU8sQ0FBQztLQUM3QyxXQUFXLENBQUMsaURBQWlELENBQUM7S0FDOUQsTUFBTSxDQUFDLHVCQUF1QixFQUFFLGVBQWUsQ0FBQztLQUNoRCxNQUFNLENBQUMsS0FBSyxFQUFFLElBQTJCLEVBQUUsRUFBRTtJQUM1QyxNQUFNLEtBQUssR0FBRyxNQUFNLE1BQU0sQ0FBQyxTQUFTLENBQUMsQ0FBQztJQUN0QyxNQUFNLFFBQVEsR0FBRyxJQUFJLENBQUMsUUFBUSxJQUFJLE1BQU0sY0FBYyxDQUFDLFlBQVksQ0FBQyxDQUFDO0lBQ3JFLElBQUksQ0FBQyxLQUFLLElBQUksQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUN4QixPQUFPLENBQUMsS0FBSyxDQUFDLGtDQUFrQyxDQUFDLENBQUM7UUFDbEQsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNsQixDQUFDO0lBQ0QsTUFBTSxHQUFHLEdBQUcsTUFBTSxLQUFLLENBQUMsSUFBQSxhQUFNLEVBQUMsYUFBYSxDQUFDLEVBQUU7UUFDN0MsTUFBTSxFQUFFLE1BQU07UUFDZCxPQUFPLEVBQUUsRUFBRSxjQUFjLEVBQUUsa0JBQWtCLEVBQUU7UUFDL0MsSUFBSSxFQUFFLElBQUksQ0FBQyxTQUFTLENBQUMsRUFBRSxLQUFLLEVBQUUsUUFBUSxFQUFFLENBQUM7S0FDMUMsQ0FBQyxDQUFDO0lBQ0gsTUFBTSxJQUFJLEdBQUcsTUFBTSxJQUFBLHVCQUFnQixFQUFDLEdBQUcsRUFBRSxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsR0FBRyxFQUFFLElBQUEsYUFBTSxFQUFDLGFBQWEsQ0FBQyxFQUFFLENBRXRGLENBQUM7SUFDRixJQUFBLHNCQUFlLEVBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxDQUFDO0lBQ2xDLE9BQU8sQ0FBQyxHQUFHLENBQUMsMERBQTBELENBQUMsQ0FBQztBQUMxRSxDQUFDLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1hbmQgfSBmcm9tICdjb21tYW5kZXInO1xuaW1wb3J0IHsgY3JlYXRlSW50ZXJmYWNlIH0gZnJvbSAncmVhZGxpbmUnO1xuaW1wb3J0IHsgYXBpVXJsLCByZWFkSnNvblJlc3BvbnNlLCBzYXZlQ3JlZGVudGlhbHMgfSBmcm9tICcuLi9hdXRoJztcblxuYXN5bmMgZnVuY3Rpb24gcHJvbXB0KHF1ZXN0aW9uOiBzdHJpbmcpOiBQcm9taXNlPHN0cmluZz4ge1xuICBjb25zdCBybCA9IGNyZWF0ZUludGVyZmFjZSh7IGlucHV0OiBwcm9jZXNzLnN0ZGluLCBvdXRwdXQ6IHByb2Nlc3Muc3Rkb3V0IH0pO1xuICByZXR1cm4gbmV3IFByb21pc2UocmVzb2x2ZSA9PiBybC5xdWVzdGlvbihxdWVzdGlvbiwgYW5zd2VyID0+IHsgcmwuY2xvc2UoKTsgcmVzb2x2ZShhbnN3ZXIpOyB9KSk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHByb21wdFBhc3N3b3JkKHF1ZXN0aW9uOiBzdHJpbmcpOiBQcm9taXNlPHN0cmluZz4ge1xuICBwcm9jZXNzLnN0ZG91dC53cml0ZShxdWVzdGlvbik7XG4gIHByb2Nlc3Muc3RkaW4uc2V0UmF3TW9kZSh0cnVlKTtcbiAgcHJvY2Vzcy5zdGRpbi5yZXN1bWUoKTtcbiAgcHJvY2Vzcy5zdGRpbi5zZXRFbmNvZGluZygndXRmOCcpO1xuICByZXR1cm4gbmV3IFByb21pc2UocmVzb2x2ZSA9PiB7XG4gICAgbGV0IHBhc3N3b3JkID0gJyc7XG4gICAgcHJvY2Vzcy5zdGRpbi5vbignZGF0YScsIGZ1bmN0aW9uIGhhbmRsZXIoY2g6IHN0cmluZykge1xuICAgICAgaWYgKGNoID09PSAnXFxuJyB8fCBjaCA9PT0gJ1xccicgfHwgY2ggPT09ICdcdTAwMDMnKSB7XG4gICAgICAgIHByb2Nlc3Muc3RkaW4uc2V0UmF3TW9kZShmYWxzZSk7XG4gICAgICAgIHByb2Nlc3Muc3RkaW4ucGF1c2UoKTtcbiAgICAgICAgcHJvY2Vzcy5zdGRpbi5yZW1vdmVMaXN0ZW5lcignZGF0YScsIGhhbmRsZXIpO1xuICAgICAgICBwcm9jZXNzLnN0ZG91dC53cml0ZSgnXFxuJyk7XG4gICAgICAgIHJlc29sdmUocGFzc3dvcmQpO1xuICAgICAgfSBlbHNlIGlmIChjaCA9PT0gJ38nIHx8IGNoID09PSAnXFxiJykge1xuICAgICAgICBpZiAocGFzc3dvcmQubGVuZ3RoID4gMCkge1xuICAgICAgICAgIHBhc3N3b3JkID0gcGFzc3dvcmQuc2xpY2UoMCwgLTEpO1xuICAgICAgICAgIHByb2Nlc3Muc3Rkb3V0LndyaXRlKCdcXGIgXFxiJyk7XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHBhc3N3b3JkICs9IGNoO1xuICAgICAgICBwcm9jZXNzLnN0ZG91dC53cml0ZSgnKicpO1xuICAgICAgfVxuICAgIH0pO1xuICB9KTtcbn1cblxuZXhwb3J0IGNvbnN0IGxvZ2luQ29tbWFuZCA9IG5ldyBDb21tYW5kKCdsb2dpbicpXG4gIC5kZXNjcmlwdGlvbignTG9naW4gaW50byBHRU5JIFdvcmtmbG93cyBhbmQgc3RvcmUgY3JlZGVudGlhbHMnKVxuICAub3B0aW9uKCctLXBhc3N3b3JkIDxwYXNzd29yZD4nLCAnVXNlciBwYXNzd29yZCcpXG4gIC5hY3Rpb24oYXN5bmMgKG9wdHM6IHsgcGFzc3dvcmQ/OiBzdHJpbmcgfSkgPT4ge1xuICAgIGNvbnN0IGVtYWlsID0gYXdhaXQgcHJvbXB0KCdFbWFpbDogJyk7XG4gICAgY29uc3QgcGFzc3dvcmQgPSBvcHRzLnBhc3N3b3JkID8/IGF3YWl0IHByb21wdFBhc3N3b3JkKCdQYXNzd29yZDogJyk7XG4gICAgaWYgKCFlbWFpbCB8fCAhcGFzc3dvcmQpIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoJ0VtYWlsIGFuZCBwYXNzd29yZCBhcmUgcmVxdWlyZWQuJyk7XG4gICAgICBwcm9jZXNzLmV4aXQoMSk7XG4gICAgfVxuICAgIGNvbnN0IHJlcyA9IGF3YWl0IGZldGNoKGFwaVVybCgnL2F1dGgvbG9naW4nKSwge1xuICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICBoZWFkZXJzOiB7ICdDb250ZW50LVR5cGUnOiAnYXBwbGljYXRpb24vanNvbicgfSxcbiAgICAgIGJvZHk6IEpTT04uc3RyaW5naWZ5KHsgZW1haWwsIHBhc3N3b3JkIH0pLFxuICAgIH0pO1xuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCByZWFkSnNvblJlc3BvbnNlKHJlcywgeyBtZXRob2Q6ICdQT1NUJywgdXJsOiBhcGlVcmwoJy9hdXRoL2xvZ2luJykgfSkgYXMge1xuICAgICAgYWNjZXNzVG9rZW46IHN0cmluZztcbiAgICB9O1xuICAgIHNhdmVDcmVkZW50aWFscyhkYXRhLmFjY2Vzc1Rva2VuKTtcbiAgICBjb25zb2xlLmxvZygnTG9nZ2VkIGluLiBDcmVkZW50aWFscyBzYXZlZCB0byB+Ly5nZW5pL2NyZWRlbnRpYWxzLmpzb24nKTtcbiAgfSk7XG4iXX0=