@agentuity/cli 0.1.15 → 0.1.17
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +18 -2
- package/dist/cli.js.map +1 -1
- package/dist/cmd/ai/opencode/install.js +1 -1
- package/dist/cmd/ai/opencode/install.js.map +1 -1
- package/dist/cmd/build/ast.d.ts.map +1 -1
- package/dist/cmd/build/ast.js +68 -2
- package/dist/cmd/build/ast.js.map +1 -1
- package/dist/cmd/build/vite/registry-generator.d.ts.map +1 -1
- package/dist/cmd/build/vite/registry-generator.js +112 -23
- package/dist/cmd/build/vite/registry-generator.js.map +1 -1
- package/dist/cmd/build/vite/route-discovery.d.ts +4 -0
- package/dist/cmd/build/vite/route-discovery.d.ts.map +1 -1
- package/dist/cmd/build/vite/route-discovery.js +4 -0
- package/dist/cmd/build/vite/route-discovery.js.map +1 -1
- package/dist/cmd/cloud/env/delete.d.ts.map +1 -1
- package/dist/cmd/cloud/env/delete.js +93 -34
- package/dist/cmd/cloud/env/delete.js.map +1 -1
- package/dist/cmd/cloud/env/get.d.ts.map +1 -1
- package/dist/cmd/cloud/env/get.js +53 -16
- package/dist/cmd/cloud/env/get.js.map +1 -1
- package/dist/cmd/cloud/env/import.d.ts.map +1 -1
- package/dist/cmd/cloud/env/import.js +80 -39
- package/dist/cmd/cloud/env/import.js.map +1 -1
- package/dist/cmd/cloud/env/index.d.ts.map +1 -1
- package/dist/cmd/cloud/env/index.js +6 -2
- package/dist/cmd/cloud/env/index.js.map +1 -1
- package/dist/cmd/cloud/env/list.d.ts.map +1 -1
- package/dist/cmd/cloud/env/list.js +99 -23
- package/dist/cmd/cloud/env/list.js.map +1 -1
- package/dist/cmd/cloud/env/org-util.d.ts +16 -0
- package/dist/cmd/cloud/env/org-util.d.ts.map +1 -0
- package/dist/cmd/cloud/env/org-util.js +28 -0
- package/dist/cmd/cloud/env/org-util.js.map +1 -0
- package/dist/cmd/cloud/env/pull.d.ts.map +1 -1
- package/dist/cmd/cloud/env/pull.js +61 -29
- package/dist/cmd/cloud/env/pull.js.map +1 -1
- package/dist/cmd/cloud/env/push.d.ts.map +1 -1
- package/dist/cmd/cloud/env/push.js +70 -30
- package/dist/cmd/cloud/env/push.js.map +1 -1
- package/dist/cmd/cloud/env/set.d.ts.map +1 -1
- package/dist/cmd/cloud/env/set.js +72 -26
- package/dist/cmd/cloud/env/set.js.map +1 -1
- package/dist/cmd/cloud/index.d.ts.map +1 -1
- package/dist/cmd/cloud/index.js +2 -0
- package/dist/cmd/cloud/index.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/create-namespace.js +1 -1
- package/dist/cmd/cloud/keyvalue/create-namespace.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/delete-namespace.js +2 -2
- package/dist/cmd/cloud/keyvalue/delete-namespace.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/delete.js +1 -1
- package/dist/cmd/cloud/keyvalue/delete.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/get.js +1 -1
- package/dist/cmd/cloud/keyvalue/get.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/index.js +1 -1
- package/dist/cmd/cloud/keyvalue/index.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/keys.js +1 -1
- package/dist/cmd/cloud/keyvalue/keys.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/list-namespaces.js +1 -1
- package/dist/cmd/cloud/keyvalue/list-namespaces.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/repl.d.ts.map +1 -1
- package/dist/cmd/cloud/keyvalue/repl.js +8 -5
- package/dist/cmd/cloud/keyvalue/repl.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/search.js +1 -1
- package/dist/cmd/cloud/keyvalue/search.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/set.js +1 -1
- package/dist/cmd/cloud/keyvalue/set.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/stats.js +1 -1
- package/dist/cmd/cloud/keyvalue/stats.js.map +1 -1
- package/dist/cmd/cloud/keyvalue/util.d.ts +4 -4
- package/dist/cmd/cloud/keyvalue/util.d.ts.map +1 -1
- package/dist/cmd/cloud/keyvalue/util.js +4 -9
- package/dist/cmd/cloud/keyvalue/util.js.map +1 -1
- package/dist/cmd/cloud/queue/ack.d.ts +3 -0
- package/dist/cmd/cloud/queue/ack.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/ack.js +45 -0
- package/dist/cmd/cloud/queue/ack.js.map +1 -0
- package/dist/cmd/cloud/queue/create.d.ts +3 -0
- package/dist/cmd/cloud/queue/create.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/create.js +80 -0
- package/dist/cmd/cloud/queue/create.js.map +1 -0
- package/dist/cmd/cloud/queue/delete.d.ts +3 -0
- package/dist/cmd/cloud/queue/delete.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/delete.js +50 -0
- package/dist/cmd/cloud/queue/delete.js.map +1 -0
- package/dist/cmd/cloud/queue/destinations.d.ts +3 -0
- package/dist/cmd/cloud/queue/destinations.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/destinations.js +232 -0
- package/dist/cmd/cloud/queue/destinations.js.map +1 -0
- package/dist/cmd/cloud/queue/dlq.d.ts +3 -0
- package/dist/cmd/cloud/queue/dlq.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/dlq.js +168 -0
- package/dist/cmd/cloud/queue/dlq.js.map +1 -0
- package/dist/cmd/cloud/queue/get.d.ts +3 -0
- package/dist/cmd/cloud/queue/get.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/get.js +130 -0
- package/dist/cmd/cloud/queue/get.js.map +1 -0
- package/dist/cmd/cloud/queue/index.d.ts +3 -0
- package/dist/cmd/cloud/queue/index.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/index.js +65 -0
- package/dist/cmd/cloud/queue/index.js.map +1 -0
- package/dist/cmd/cloud/queue/list.d.ts +3 -0
- package/dist/cmd/cloud/queue/list.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/list.js +71 -0
- package/dist/cmd/cloud/queue/list.js.map +1 -0
- package/dist/cmd/cloud/queue/messages.d.ts +3 -0
- package/dist/cmd/cloud/queue/messages.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/messages.js +137 -0
- package/dist/cmd/cloud/queue/messages.js.map +1 -0
- package/dist/cmd/cloud/queue/nack.d.ts +3 -0
- package/dist/cmd/cloud/queue/nack.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/nack.js +45 -0
- package/dist/cmd/cloud/queue/nack.js.map +1 -0
- package/dist/cmd/cloud/queue/pause.d.ts +3 -0
- package/dist/cmd/cloud/queue/pause.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/pause.js +36 -0
- package/dist/cmd/cloud/queue/pause.js.map +1 -0
- package/dist/cmd/cloud/queue/publish.d.ts +3 -0
- package/dist/cmd/cloud/queue/publish.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/publish.js +76 -0
- package/dist/cmd/cloud/queue/publish.js.map +1 -0
- package/dist/cmd/cloud/queue/receive.d.ts +3 -0
- package/dist/cmd/cloud/queue/receive.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/receive.js +67 -0
- package/dist/cmd/cloud/queue/receive.js.map +1 -0
- package/dist/cmd/cloud/queue/resume.d.ts +3 -0
- package/dist/cmd/cloud/queue/resume.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/resume.js +35 -0
- package/dist/cmd/cloud/queue/resume.js.map +1 -0
- package/dist/cmd/cloud/queue/sources.d.ts +3 -0
- package/dist/cmd/cloud/queue/sources.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/sources.js +290 -0
- package/dist/cmd/cloud/queue/sources.js.map +1 -0
- package/dist/cmd/cloud/queue/stats.d.ts +3 -0
- package/dist/cmd/cloud/queue/stats.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/stats.js +239 -0
- package/dist/cmd/cloud/queue/stats.js.map +1 -0
- package/dist/cmd/cloud/queue/util.d.ts +26 -0
- package/dist/cmd/cloud/queue/util.d.ts.map +1 -0
- package/dist/cmd/cloud/queue/util.js +19 -0
- package/dist/cmd/cloud/queue/util.js.map +1 -0
- package/dist/cmd/cloud/sandbox/snapshot/build.d.ts.map +1 -1
- package/dist/cmd/cloud/sandbox/snapshot/build.js +122 -28
- package/dist/cmd/cloud/sandbox/snapshot/build.js.map +1 -1
- package/dist/cmd/cloud/sandbox/snapshot/create.d.ts.map +1 -1
- package/dist/cmd/cloud/sandbox/snapshot/create.js +19 -7
- package/dist/cmd/cloud/sandbox/snapshot/create.js.map +1 -1
- package/dist/cmd/cloud/sandbox/snapshot/get.d.ts.map +1 -1
- package/dist/cmd/cloud/sandbox/snapshot/get.js +16 -0
- package/dist/cmd/cloud/sandbox/snapshot/get.js.map +1 -1
- package/dist/cmd/cloud/sandbox/snapshot/list.d.ts.map +1 -1
- package/dist/cmd/cloud/sandbox/snapshot/list.js +4 -0
- package/dist/cmd/cloud/sandbox/snapshot/list.js.map +1 -1
- package/dist/cmd/cloud/vector/stats.d.ts.map +1 -1
- package/dist/cmd/cloud/vector/stats.js +8 -0
- package/dist/cmd/cloud/vector/stats.js.map +1 -1
- package/dist/cmd/project/create.d.ts.map +1 -1
- package/dist/cmd/project/create.js +12 -0
- package/dist/cmd/project/create.js.map +1 -1
- package/dist/cmd/project/template-flow.d.ts +3 -0
- package/dist/cmd/project/template-flow.d.ts.map +1 -1
- package/dist/cmd/project/template-flow.js +157 -68
- package/dist/cmd/project/template-flow.js.map +1 -1
- package/dist/cmd/setup/index.d.ts.map +1 -1
- package/dist/cmd/setup/index.js +2 -1
- package/dist/cmd/setup/index.js.map +1 -1
- package/dist/env-util.d.ts +6 -1
- package/dist/env-util.d.ts.map +1 -1
- package/dist/env-util.js +16 -2
- package/dist/env-util.js.map +1 -1
- package/dist/errors.d.ts +4 -2
- package/dist/errors.d.ts.map +1 -1
- package/dist/errors.js +6 -0
- package/dist/errors.js.map +1 -1
- package/dist/onboarding/agentPrompt.d.ts +8 -0
- package/dist/onboarding/agentPrompt.d.ts.map +1 -0
- package/dist/onboarding/agentPrompt.js +263 -0
- package/dist/onboarding/agentPrompt.js.map +1 -0
- package/dist/schema-generator.d.ts +1 -1
- package/dist/schema-generator.d.ts.map +1 -1
- package/dist/schema-parser.d.ts +1 -1
- package/dist/schema-parser.d.ts.map +1 -1
- package/dist/schema-parser.js +36 -1
- package/dist/schema-parser.js.map +1 -1
- package/dist/tui/box.d.ts +4 -0
- package/dist/tui/box.d.ts.map +1 -1
- package/dist/tui/box.js +39 -0
- package/dist/tui/box.js.map +1 -1
- package/dist/tui.d.ts +11 -1
- package/dist/tui.d.ts.map +1 -1
- package/dist/tui.js +33 -15
- package/dist/tui.js.map +1 -1
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js.map +1 -1
- package/package.json +6 -6
- package/src/cli.ts +19 -2
- package/src/cmd/ai/opencode/install.ts +1 -1
- package/src/cmd/build/ast.ts +88 -2
- package/src/cmd/build/vite/registry-generator.ts +120 -24
- package/src/cmd/build/vite/route-discovery.ts +16 -0
- package/src/cmd/cloud/env/delete.ts +113 -41
- package/src/cmd/cloud/env/get.ts +60 -16
- package/src/cmd/cloud/env/import.ts +92 -44
- package/src/cmd/cloud/env/index.ts +6 -2
- package/src/cmd/cloud/env/list.ts +112 -27
- package/src/cmd/cloud/env/org-util.ts +37 -0
- package/src/cmd/cloud/env/pull.ts +72 -31
- package/src/cmd/cloud/env/push.ts +84 -35
- package/src/cmd/cloud/env/set.ts +89 -33
- package/src/cmd/cloud/index.ts +2 -0
- package/src/cmd/cloud/keyvalue/create-namespace.ts +1 -1
- package/src/cmd/cloud/keyvalue/delete-namespace.ts +2 -2
- package/src/cmd/cloud/keyvalue/delete.ts +1 -1
- package/src/cmd/cloud/keyvalue/get.ts +1 -1
- package/src/cmd/cloud/keyvalue/index.ts +1 -1
- package/src/cmd/cloud/keyvalue/keys.ts +1 -1
- package/src/cmd/cloud/keyvalue/list-namespaces.ts +1 -1
- package/src/cmd/cloud/keyvalue/repl.ts +8 -5
- package/src/cmd/cloud/keyvalue/search.ts +1 -1
- package/src/cmd/cloud/keyvalue/set.ts +1 -1
- package/src/cmd/cloud/keyvalue/stats.ts +1 -1
- package/src/cmd/cloud/keyvalue/util.ts +8 -17
- package/src/cmd/cloud/queue/ack.ts +50 -0
- package/src/cmd/cloud/queue/create.ts +91 -0
- package/src/cmd/cloud/queue/delete.ts +57 -0
- package/src/cmd/cloud/queue/destinations.ts +287 -0
- package/src/cmd/cloud/queue/dlq.ts +203 -0
- package/src/cmd/cloud/queue/get.ts +158 -0
- package/src/cmd/cloud/queue/index.ts +66 -0
- package/src/cmd/cloud/queue/list.ts +81 -0
- package/src/cmd/cloud/queue/messages.ts +160 -0
- package/src/cmd/cloud/queue/nack.ts +50 -0
- package/src/cmd/cloud/queue/pause.ts +41 -0
- package/src/cmd/cloud/queue/publish.ts +88 -0
- package/src/cmd/cloud/queue/receive.ts +76 -0
- package/src/cmd/cloud/queue/resume.ts +40 -0
- package/src/cmd/cloud/queue/sources.ts +352 -0
- package/src/cmd/cloud/queue/stats.ts +297 -0
- package/src/cmd/cloud/queue/util.ts +34 -0
- package/src/cmd/cloud/sandbox/snapshot/build.ts +146 -29
- package/src/cmd/cloud/sandbox/snapshot/create.ts +24 -7
- package/src/cmd/cloud/sandbox/snapshot/get.ts +16 -0
- package/src/cmd/cloud/sandbox/snapshot/list.ts +4 -0
- package/src/cmd/cloud/vector/stats.ts +9 -0
- package/src/cmd/project/create.ts +12 -0
- package/src/cmd/project/template-flow.ts +181 -69
- package/src/cmd/setup/index.ts +2 -1
- package/src/env-util.ts +17 -2
- package/src/errors.ts +8 -0
- package/src/onboarding/agentPrompt.ts +263 -0
- package/src/schema-generator.ts +1 -1
- package/src/schema-parser.ts +45 -3
- package/src/tui/box.ts +52 -0
- package/src/tui.ts +47 -17
- package/src/types.ts +0 -1
package/src/cmd/cloud/env/get.ts
CHANGED
|
@@ -1,55 +1,97 @@
|
|
|
1
1
|
import { z } from 'zod';
|
|
2
2
|
import { createSubcommand } from '../../../types';
|
|
3
3
|
import * as tui from '../../../tui';
|
|
4
|
-
import { projectGet } from '@agentuity/server';
|
|
4
|
+
import { projectGet, orgEnvGet } from '@agentuity/server';
|
|
5
5
|
import { getCommand } from '../../../command-prefix';
|
|
6
6
|
import { ErrorCode } from '../../../errors';
|
|
7
|
+
import { resolveOrgId, isOrgScope } from './org-util';
|
|
7
8
|
|
|
8
9
|
const EnvGetResponseSchema = z.object({
|
|
9
10
|
key: z.string().describe('Environment variable key name'),
|
|
10
11
|
value: z.string().describe('Environment variable value'),
|
|
11
12
|
secret: z.boolean().describe('Whether the value is stored as a secret'),
|
|
13
|
+
scope: z.enum(['project', 'org']).describe('The scope where the variable was found'),
|
|
12
14
|
});
|
|
13
15
|
|
|
14
16
|
export const getSubcommand = createSubcommand({
|
|
15
17
|
name: 'get',
|
|
16
18
|
description: 'Get an environment variable or secret value',
|
|
17
|
-
tags: ['read-only', 'fast', 'requires-auth'
|
|
19
|
+
tags: ['read-only', 'fast', 'requires-auth'],
|
|
18
20
|
examples: [
|
|
19
21
|
{ command: getCommand('env get NODE_ENV'), description: 'Get environment variable' },
|
|
20
22
|
{ command: getCommand('env get API_KEY'), description: 'Get a secret value' },
|
|
21
23
|
{ command: getCommand('env get API_KEY --no-mask'), description: 'Show unmasked value' },
|
|
24
|
+
{
|
|
25
|
+
command: getCommand('env get OPENAI_API_KEY --org'),
|
|
26
|
+
description: 'Get org-level variable',
|
|
27
|
+
},
|
|
22
28
|
],
|
|
23
|
-
requires: { auth: true,
|
|
29
|
+
requires: { auth: true, apiClient: true },
|
|
30
|
+
optional: { project: true },
|
|
24
31
|
schema: {
|
|
25
32
|
args: z.object({
|
|
26
33
|
key: z.string().describe('the environment variable or secret key'),
|
|
27
34
|
}),
|
|
28
35
|
options: z.object({
|
|
29
36
|
maskSecret: z.boolean().optional().describe('mask the secret value in output'),
|
|
37
|
+
org: z
|
|
38
|
+
.union([z.boolean(), z.string()])
|
|
39
|
+
.optional()
|
|
40
|
+
.describe(
|
|
41
|
+
'get from organization level (use --org for default org, or --org <orgId> for specific org)'
|
|
42
|
+
),
|
|
30
43
|
}),
|
|
31
44
|
response: EnvGetResponseSchema,
|
|
32
45
|
},
|
|
33
46
|
idempotent: true,
|
|
34
47
|
|
|
35
48
|
async handler(ctx) {
|
|
36
|
-
const { args, opts, apiClient, project, options } = ctx;
|
|
49
|
+
const { args, opts, apiClient, project, options, config } = ctx;
|
|
50
|
+
const useOrgScope = isOrgScope(opts?.org);
|
|
37
51
|
|
|
38
|
-
//
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
52
|
+
// Require project context if not using org scope
|
|
53
|
+
if (!useOrgScope && !project) {
|
|
54
|
+
tui.fatal(
|
|
55
|
+
'Project context required. Run from a project directory or use --org for organization scope.'
|
|
56
|
+
);
|
|
57
|
+
}
|
|
42
58
|
|
|
43
|
-
// Look for the key in both env and secrets
|
|
44
59
|
let value: string | undefined;
|
|
45
60
|
let isSecret = false;
|
|
61
|
+
let scope: 'project' | 'org';
|
|
62
|
+
|
|
63
|
+
if (useOrgScope) {
|
|
64
|
+
// Organization scope
|
|
65
|
+
const orgId = await resolveOrgId(apiClient, config, opts!.org!);
|
|
66
|
+
|
|
67
|
+
const orgData = await tui.spinner('Fetching organization variable', () => {
|
|
68
|
+
return orgEnvGet(apiClient, { id: orgId, mask: false });
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
if (orgData.secrets?.[args.key] !== undefined) {
|
|
72
|
+
value = orgData.secrets[args.key];
|
|
73
|
+
isSecret = true;
|
|
74
|
+
} else if (orgData.env?.[args.key] !== undefined) {
|
|
75
|
+
value = orgData.env[args.key];
|
|
76
|
+
isSecret = false;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
scope = 'org';
|
|
80
|
+
} else {
|
|
81
|
+
// Project scope
|
|
82
|
+
const projectData = await tui.spinner('Fetching variable', () => {
|
|
83
|
+
return projectGet(apiClient, { id: project!.projectId, mask: false });
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
if (projectData.secrets?.[args.key] !== undefined) {
|
|
87
|
+
value = projectData.secrets[args.key];
|
|
88
|
+
isSecret = true;
|
|
89
|
+
} else if (projectData.env?.[args.key] !== undefined) {
|
|
90
|
+
value = projectData.env[args.key];
|
|
91
|
+
isSecret = false;
|
|
92
|
+
}
|
|
46
93
|
|
|
47
|
-
|
|
48
|
-
value = projectData.secrets[args.key];
|
|
49
|
-
isSecret = true;
|
|
50
|
-
} else if (projectData.env?.[args.key] !== undefined) {
|
|
51
|
-
value = projectData.env[args.key];
|
|
52
|
-
isSecret = false;
|
|
94
|
+
scope = 'project';
|
|
53
95
|
}
|
|
54
96
|
|
|
55
97
|
if (value === undefined) {
|
|
@@ -62,13 +104,15 @@ export const getSubcommand = createSubcommand({
|
|
|
62
104
|
if (!options.json) {
|
|
63
105
|
const displayValue = shouldMask ? tui.maskSecret(value) : value;
|
|
64
106
|
const typeLabel = isSecret ? ' (secret)' : '';
|
|
65
|
-
|
|
107
|
+
const scopeLabel = useOrgScope ? ' [org]' : '';
|
|
108
|
+
tui.success(`${args.key}=${displayValue}${typeLabel}${scopeLabel}`);
|
|
66
109
|
}
|
|
67
110
|
|
|
68
111
|
return {
|
|
69
112
|
key: args.key,
|
|
70
113
|
value,
|
|
71
114
|
secret: isSecret,
|
|
115
|
+
scope,
|
|
72
116
|
};
|
|
73
117
|
},
|
|
74
118
|
});
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { z } from 'zod';
|
|
2
2
|
import { createSubcommand } from '../../../types';
|
|
3
3
|
import * as tui from '../../../tui';
|
|
4
|
-
import { projectEnvUpdate } from '@agentuity/server';
|
|
4
|
+
import { projectEnvUpdate, orgEnvUpdate } from '@agentuity/server';
|
|
5
5
|
import {
|
|
6
6
|
findExistingEnvFile,
|
|
7
7
|
readEnvFile,
|
|
@@ -13,6 +13,7 @@ import {
|
|
|
13
13
|
isReservedAgentuityKey,
|
|
14
14
|
} from '../../../env-util';
|
|
15
15
|
import { getCommand } from '../../../command-prefix';
|
|
16
|
+
import { resolveOrgId, isOrgScope } from './org-util';
|
|
16
17
|
|
|
17
18
|
const EnvImportResponseSchema = z.object({
|
|
18
19
|
success: z.boolean().describe('Whether import succeeded'),
|
|
@@ -20,21 +21,18 @@ const EnvImportResponseSchema = z.object({
|
|
|
20
21
|
envCount: z.number().describe('Number of env vars imported'),
|
|
21
22
|
secretCount: z.number().describe('Number of secrets imported'),
|
|
22
23
|
skipped: z.number().describe('Number of items skipped'),
|
|
23
|
-
path: z
|
|
24
|
+
path: z
|
|
25
|
+
.string()
|
|
26
|
+
.optional()
|
|
27
|
+
.describe('Local file path where variables were saved (project scope only)'),
|
|
24
28
|
file: z.string().describe('Source file path'),
|
|
29
|
+
scope: z.enum(['project', 'org']).describe('The scope where variables were imported'),
|
|
25
30
|
});
|
|
26
31
|
|
|
27
32
|
export const importSubcommand = createSubcommand({
|
|
28
33
|
name: 'import',
|
|
29
34
|
description: 'Import environment variables and secrets from a file to cloud and local .env',
|
|
30
|
-
tags: [
|
|
31
|
-
'mutating',
|
|
32
|
-
'creates-resource',
|
|
33
|
-
'slow',
|
|
34
|
-
'api-intensive',
|
|
35
|
-
'requires-auth',
|
|
36
|
-
'requires-project',
|
|
37
|
-
],
|
|
35
|
+
tags: ['mutating', 'creates-resource', 'slow', 'api-intensive', 'requires-auth'],
|
|
38
36
|
examples: [
|
|
39
37
|
{
|
|
40
38
|
command: getCommand('cloud env import .env.backup'),
|
|
@@ -44,18 +42,37 @@ export const importSubcommand = createSubcommand({
|
|
|
44
42
|
command: getCommand('cloud env import .env.local'),
|
|
45
43
|
description: 'Import from .env.local file',
|
|
46
44
|
},
|
|
45
|
+
{
|
|
46
|
+
command: getCommand('cloud env import .env.shared --org'),
|
|
47
|
+
description: 'Import to organization level',
|
|
48
|
+
},
|
|
47
49
|
],
|
|
48
50
|
idempotent: false,
|
|
49
|
-
requires: { auth: true,
|
|
51
|
+
requires: { auth: true, apiClient: true },
|
|
52
|
+
optional: { project: true },
|
|
50
53
|
schema: {
|
|
51
54
|
args: z.object({
|
|
52
55
|
file: z.string().describe('path to the .env file to import'),
|
|
53
56
|
}),
|
|
57
|
+
options: z.object({
|
|
58
|
+
org: z
|
|
59
|
+
.union([z.boolean(), z.string()])
|
|
60
|
+
.optional()
|
|
61
|
+
.describe('import to organization level (use --org for default org)'),
|
|
62
|
+
}),
|
|
54
63
|
response: EnvImportResponseSchema,
|
|
55
64
|
},
|
|
56
65
|
|
|
57
66
|
async handler(ctx) {
|
|
58
|
-
const { args, apiClient, project, projectDir } = ctx;
|
|
67
|
+
const { args, apiClient, project, projectDir, config, opts } = ctx;
|
|
68
|
+
const useOrgScope = isOrgScope(opts?.org);
|
|
69
|
+
|
|
70
|
+
// Require project context if not using org scope
|
|
71
|
+
if (!useOrgScope && !project) {
|
|
72
|
+
tui.fatal(
|
|
73
|
+
'Project context required. Run from a project directory or use --org for organization scope.'
|
|
74
|
+
);
|
|
75
|
+
}
|
|
59
76
|
|
|
60
77
|
// Read the import file
|
|
61
78
|
const importedVars = await readEnvFile(args.file);
|
|
@@ -68,8 +85,8 @@ export const importSubcommand = createSubcommand({
|
|
|
68
85
|
envCount: 0,
|
|
69
86
|
secretCount: 0,
|
|
70
87
|
skipped: 0,
|
|
71
|
-
path: '',
|
|
72
88
|
file: args.file,
|
|
89
|
+
scope: useOrgScope ? ('org' as const) : ('project' as const),
|
|
73
90
|
};
|
|
74
91
|
}
|
|
75
92
|
|
|
@@ -84,8 +101,8 @@ export const importSubcommand = createSubcommand({
|
|
|
84
101
|
envCount: 0,
|
|
85
102
|
secretCount: 0,
|
|
86
103
|
skipped: Object.keys(importedVars).length,
|
|
87
|
-
path: '',
|
|
88
104
|
file: args.file,
|
|
105
|
+
scope: useOrgScope ? ('org' as const) : ('project' as const),
|
|
89
106
|
};
|
|
90
107
|
}
|
|
91
108
|
|
|
@@ -104,40 +121,71 @@ export const importSubcommand = createSubcommand({
|
|
|
104
121
|
}
|
|
105
122
|
}
|
|
106
123
|
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
124
|
+
const envCount = Object.keys(env).length;
|
|
125
|
+
const secretCount = Object.keys(secrets).length;
|
|
126
|
+
const totalCount = envCount + secretCount;
|
|
127
|
+
|
|
128
|
+
if (useOrgScope) {
|
|
129
|
+
// Organization scope
|
|
130
|
+
const orgId = await resolveOrgId(apiClient, config, opts!.org!);
|
|
131
|
+
|
|
132
|
+
await tui.spinner('Importing variables to organization', () => {
|
|
133
|
+
return orgEnvUpdate(apiClient, {
|
|
134
|
+
id: orgId,
|
|
135
|
+
env,
|
|
136
|
+
secrets,
|
|
137
|
+
});
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
tui.success(
|
|
141
|
+
`Imported ${totalCount} variable${totalCount !== 1 ? 's' : ''} to organization from ${args.file} (${envCount} env, ${secretCount} secret${secretCount !== 1 ? 's' : ''})`
|
|
142
|
+
);
|
|
143
|
+
|
|
144
|
+
return {
|
|
145
|
+
success: true,
|
|
146
|
+
imported: totalCount,
|
|
147
|
+
envCount,
|
|
148
|
+
secretCount,
|
|
149
|
+
skipped: Object.keys(importedVars).length - totalCount,
|
|
150
|
+
file: args.file,
|
|
151
|
+
scope: 'org' as const,
|
|
152
|
+
};
|
|
153
|
+
} else {
|
|
154
|
+
// Project scope (existing behavior)
|
|
155
|
+
await tui.spinner('Importing variables to cloud', () => {
|
|
156
|
+
return projectEnvUpdate(apiClient, {
|
|
157
|
+
id: project!.projectId,
|
|
158
|
+
env,
|
|
159
|
+
secrets,
|
|
160
|
+
});
|
|
113
161
|
});
|
|
114
|
-
});
|
|
115
162
|
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
163
|
+
// Merge with local .env file only if we have a project directory
|
|
164
|
+
let localEnvPath: string | undefined;
|
|
165
|
+
if (projectDir) {
|
|
166
|
+
localEnvPath = await findExistingEnvFile(projectDir);
|
|
167
|
+
const localEnv = await readEnvFile(localEnvPath);
|
|
168
|
+
const mergedEnv = mergeEnvVars(localEnv, filteredVars);
|
|
120
169
|
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
170
|
+
await writeEnvFile(localEnvPath, mergedEnv, {
|
|
171
|
+
skipKeys: Object.keys(mergedEnv).filter(isReservedAgentuityKey),
|
|
172
|
+
});
|
|
173
|
+
}
|
|
124
174
|
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
175
|
+
tui.success(
|
|
176
|
+
`Imported ${totalCount} variable${totalCount !== 1 ? 's' : ''} from ${args.file} (${envCount} env, ${secretCount} secret${secretCount !== 1 ? 's' : ''})`
|
|
177
|
+
);
|
|
128
178
|
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
file: args.file,
|
|
141
|
-
};
|
|
179
|
+
return {
|
|
180
|
+
success: true,
|
|
181
|
+
imported: totalCount,
|
|
182
|
+
envCount,
|
|
183
|
+
secretCount,
|
|
184
|
+
skipped: Object.keys(importedVars).length - totalCount,
|
|
185
|
+
path: localEnvPath,
|
|
186
|
+
file: args.file,
|
|
187
|
+
scope: 'project' as const,
|
|
188
|
+
};
|
|
189
|
+
}
|
|
142
190
|
},
|
|
143
191
|
});
|
|
@@ -10,8 +10,8 @@ import { getCommand } from '../../../command-prefix';
|
|
|
10
10
|
|
|
11
11
|
export const command = createCommand({
|
|
12
12
|
name: 'env',
|
|
13
|
-
description: 'Manage environment variables and secrets for your project',
|
|
14
|
-
tags: ['fast', 'requires-auth'
|
|
13
|
+
description: 'Manage environment variables and secrets for your project or organization',
|
|
14
|
+
tags: ['fast', 'requires-auth'],
|
|
15
15
|
examples: [
|
|
16
16
|
{ command: getCommand('cloud env list'), description: 'List all variables and secrets' },
|
|
17
17
|
{
|
|
@@ -22,6 +22,10 @@ export const command = createCommand({
|
|
|
22
22
|
command: getCommand('cloud env set API_KEY "sk_..." --secret'),
|
|
23
23
|
description: 'Set a secret',
|
|
24
24
|
},
|
|
25
|
+
{
|
|
26
|
+
command: getCommand('cloud env set OPENAI_API_KEY "sk_..." --secret --org'),
|
|
27
|
+
description: 'Set an organization-wide secret',
|
|
28
|
+
},
|
|
25
29
|
],
|
|
26
30
|
subcommands: [
|
|
27
31
|
listSubcommand,
|
|
@@ -1,12 +1,14 @@
|
|
|
1
1
|
import { z } from 'zod';
|
|
2
2
|
import { createSubcommand } from '../../../types';
|
|
3
3
|
import * as tui from '../../../tui';
|
|
4
|
-
import { projectGet } from '@agentuity/server';
|
|
4
|
+
import { projectGet, orgEnvGet } from '@agentuity/server';
|
|
5
5
|
import { getCommand } from '../../../command-prefix';
|
|
6
|
+
import { resolveOrgId, isOrgScope } from './org-util';
|
|
6
7
|
|
|
7
8
|
const EnvItemSchema = z.object({
|
|
8
9
|
value: z.string().describe('Variable value'),
|
|
9
10
|
secret: z.boolean().describe('Whether this is a secret'),
|
|
11
|
+
scope: z.enum(['project', 'org']).describe('The scope of the variable'),
|
|
10
12
|
});
|
|
11
13
|
|
|
12
14
|
const EnvListResponseSchema = z.record(z.string(), EnvItemSchema);
|
|
@@ -15,14 +17,19 @@ export const listSubcommand = createSubcommand({
|
|
|
15
17
|
name: 'list',
|
|
16
18
|
aliases: ['ls'],
|
|
17
19
|
description: 'List all environment variables and secrets',
|
|
18
|
-
tags: ['read-only', 'fast', 'requires-auth'
|
|
20
|
+
tags: ['read-only', 'fast', 'requires-auth'],
|
|
19
21
|
examples: [
|
|
20
|
-
{ command: getCommand('env list'), description: 'List all variables' },
|
|
22
|
+
{ command: getCommand('env list'), description: 'List all variables (project + org merged)' },
|
|
21
23
|
{ command: getCommand('env list --no-mask'), description: 'Show unmasked values' },
|
|
22
24
|
{ command: getCommand('env list --secrets'), description: 'List only secrets' },
|
|
23
25
|
{ command: getCommand('env list --env-only'), description: 'List only env vars' },
|
|
26
|
+
{
|
|
27
|
+
command: getCommand('env list --org'),
|
|
28
|
+
description: 'List only organization-level variables',
|
|
29
|
+
},
|
|
24
30
|
],
|
|
25
|
-
requires: { auth: true,
|
|
31
|
+
requires: { auth: true, apiClient: true },
|
|
32
|
+
optional: { project: true },
|
|
26
33
|
idempotent: true,
|
|
27
34
|
schema: {
|
|
28
35
|
options: z.object({
|
|
@@ -32,39 +39,108 @@ export const listSubcommand = createSubcommand({
|
|
|
32
39
|
.describe('mask secret values in output (use --no-mask to show values)'),
|
|
33
40
|
secrets: z.boolean().default(false).describe('list only secrets'),
|
|
34
41
|
'env-only': z.boolean().default(false).describe('list only environment variables'),
|
|
42
|
+
org: z
|
|
43
|
+
.union([z.boolean(), z.string()])
|
|
44
|
+
.optional()
|
|
45
|
+
.describe('list organization-level variables only (use --org for default org)'),
|
|
35
46
|
}),
|
|
36
47
|
response: EnvListResponseSchema,
|
|
37
48
|
},
|
|
38
|
-
webUrl: (ctx) =>
|
|
49
|
+
webUrl: (ctx) => {
|
|
50
|
+
// If --org flag is used, link to org settings; otherwise link to project settings
|
|
51
|
+
if (ctx.opts?.org) {
|
|
52
|
+
return `/settings/organization/env`;
|
|
53
|
+
}
|
|
54
|
+
return ctx.project
|
|
55
|
+
? `/projects/${encodeURIComponent(ctx.project.projectId)}/settings`
|
|
56
|
+
: undefined;
|
|
57
|
+
},
|
|
39
58
|
|
|
40
59
|
async handler(ctx) {
|
|
41
|
-
const { opts, apiClient, project, options } = ctx;
|
|
42
|
-
|
|
43
|
-
// Fetch project with unmasked values
|
|
44
|
-
const projectData = await tui.spinner('Fetching variables', () => {
|
|
45
|
-
return projectGet(apiClient, { id: project.projectId, mask: false });
|
|
46
|
-
});
|
|
60
|
+
const { opts, apiClient, project, options, config } = ctx;
|
|
61
|
+
const useOrgScope = isOrgScope(opts?.org);
|
|
47
62
|
|
|
48
|
-
|
|
49
|
-
const
|
|
50
|
-
|
|
51
|
-
// Build combined result with type info
|
|
52
|
-
const result: Record<string, { value: string; secret: boolean }> = {};
|
|
63
|
+
// Build combined result with type info and scope
|
|
64
|
+
const result: Record<string, { value: string; secret: boolean; scope: 'project' | 'org' }> =
|
|
65
|
+
{};
|
|
53
66
|
|
|
54
67
|
// Filter based on options
|
|
55
68
|
const showEnv = !opts?.secrets;
|
|
56
69
|
const showSecrets = !opts?.['env-only'];
|
|
57
70
|
|
|
58
|
-
if (
|
|
59
|
-
|
|
60
|
-
|
|
71
|
+
if (useOrgScope) {
|
|
72
|
+
// Organization scope only
|
|
73
|
+
const orgId = await resolveOrgId(apiClient, config, opts!.org!);
|
|
74
|
+
|
|
75
|
+
const orgData = await tui.spinner('Fetching organization variables', () => {
|
|
76
|
+
return orgEnvGet(apiClient, { id: orgId, mask: false });
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
const orgEnv = orgData.env || {};
|
|
80
|
+
const orgSecrets = orgData.secrets || {};
|
|
81
|
+
|
|
82
|
+
if (showEnv) {
|
|
83
|
+
for (const [key, value] of Object.entries(orgEnv)) {
|
|
84
|
+
result[key] = { value, secret: false, scope: 'org' };
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
if (showSecrets) {
|
|
89
|
+
for (const [key, value] of Object.entries(orgSecrets)) {
|
|
90
|
+
result[key] = { value, secret: true, scope: 'org' };
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
} else if (project) {
|
|
94
|
+
// Project context: show merged view (org + project, project takes precedence)
|
|
95
|
+
// First, get project's org ID and fetch org env
|
|
96
|
+
const projectData = await tui.spinner('Fetching variables', () => {
|
|
97
|
+
return projectGet(apiClient, { id: project.projectId, mask: false });
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
const projectEnv = projectData.env || {};
|
|
101
|
+
const projectSecrets = projectData.secrets || {};
|
|
102
|
+
|
|
103
|
+
// Try to get org-level env (may fail if no access, that's ok)
|
|
104
|
+
let orgEnv: Record<string, string> = {};
|
|
105
|
+
let orgSecrets: Record<string, string> = {};
|
|
106
|
+
|
|
107
|
+
if (projectData.orgId) {
|
|
108
|
+
try {
|
|
109
|
+
const orgData = await orgEnvGet(apiClient, { id: projectData.orgId, mask: false });
|
|
110
|
+
orgEnv = orgData.env || {};
|
|
111
|
+
orgSecrets = orgData.secrets || {};
|
|
112
|
+
} catch {
|
|
113
|
+
// Ignore errors - user may not have org env access
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
// Add org-level first (will be overridden by project-level)
|
|
118
|
+
if (showEnv) {
|
|
119
|
+
for (const [key, value] of Object.entries(orgEnv)) {
|
|
120
|
+
result[key] = { value, secret: false, scope: 'org' };
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
if (showSecrets) {
|
|
124
|
+
for (const [key, value] of Object.entries(orgSecrets)) {
|
|
125
|
+
result[key] = { value, secret: true, scope: 'org' };
|
|
126
|
+
}
|
|
61
127
|
}
|
|
62
|
-
}
|
|
63
128
|
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
129
|
+
// Add project-level (overrides org-level)
|
|
130
|
+
if (showEnv) {
|
|
131
|
+
for (const [key, value] of Object.entries(projectEnv)) {
|
|
132
|
+
result[key] = { value, secret: false, scope: 'project' };
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
if (showSecrets) {
|
|
136
|
+
for (const [key, value] of Object.entries(projectSecrets)) {
|
|
137
|
+
result[key] = { value, secret: true, scope: 'project' };
|
|
138
|
+
}
|
|
67
139
|
}
|
|
140
|
+
} else {
|
|
141
|
+
tui.fatal(
|
|
142
|
+
'Project context required. Run from a project directory or use --org for organization scope.'
|
|
143
|
+
);
|
|
68
144
|
}
|
|
69
145
|
|
|
70
146
|
// Skip TUI output in JSON mode
|
|
@@ -74,11 +150,17 @@ export const listSubcommand = createSubcommand({
|
|
|
74
150
|
} else {
|
|
75
151
|
tui.newline();
|
|
76
152
|
|
|
77
|
-
const
|
|
78
|
-
const
|
|
153
|
+
const projectCount = Object.values(result).filter((v) => v.scope === 'project').length;
|
|
154
|
+
const orgCount = Object.values(result).filter((v) => v.scope === 'org').length;
|
|
155
|
+
const secretCount = Object.values(result).filter((v) => v.secret).length;
|
|
156
|
+
const envCount = Object.values(result).filter((v) => !v.secret).length;
|
|
157
|
+
|
|
79
158
|
const parts: string[] = [];
|
|
80
159
|
if (envCount > 0) parts.push(`${envCount} env`);
|
|
81
160
|
if (secretCount > 0) parts.push(`${secretCount} secret${secretCount !== 1 ? 's' : ''}`);
|
|
161
|
+
if (!useOrgScope && projectCount > 0 && orgCount > 0) {
|
|
162
|
+
parts.push(`${projectCount} project, ${orgCount} org`);
|
|
163
|
+
}
|
|
82
164
|
|
|
83
165
|
tui.info(`Variables (${parts.join(', ')}):`);
|
|
84
166
|
tui.newline();
|
|
@@ -87,10 +169,13 @@ export const listSubcommand = createSubcommand({
|
|
|
87
169
|
const shouldMask = opts?.mask !== false;
|
|
88
170
|
|
|
89
171
|
for (const key of sortedKeys) {
|
|
90
|
-
const { value, secret } = result[key];
|
|
172
|
+
const { value, secret, scope } = result[key];
|
|
91
173
|
const displayValue = shouldMask && secret ? tui.maskSecret(value) : value;
|
|
92
174
|
const typeIndicator = secret ? ' [secret]' : '';
|
|
93
|
-
|
|
175
|
+
const scopeIndicator = !useOrgScope ? ` [${scope}]` : '';
|
|
176
|
+
console.log(
|
|
177
|
+
`${tui.bold(key)}=${displayValue}${tui.muted(typeIndicator + scopeIndicator)}`
|
|
178
|
+
);
|
|
94
179
|
}
|
|
95
180
|
}
|
|
96
181
|
}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import type { APIClient } from '../../../api';
|
|
2
|
+
import type { Config } from '../../../types';
|
|
3
|
+
import * as tui from '../../../tui';
|
|
4
|
+
import { listOrganizations } from '@agentuity/server';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Resolves the organization ID for org-scoped env operations.
|
|
8
|
+
*
|
|
9
|
+
* @param apiClient - The API client
|
|
10
|
+
* @param config - The CLI config (may be null)
|
|
11
|
+
* @param orgOption - The --org option value (true for default/prompt, or explicit org ID)
|
|
12
|
+
* @returns The resolved organization ID
|
|
13
|
+
*/
|
|
14
|
+
export async function resolveOrgId(
|
|
15
|
+
apiClient: APIClient,
|
|
16
|
+
config: Config | null,
|
|
17
|
+
orgOption: boolean | string
|
|
18
|
+
): Promise<string> {
|
|
19
|
+
// If an explicit org ID was provided (string), use it directly
|
|
20
|
+
if (typeof orgOption === 'string' && orgOption !== 'true') {
|
|
21
|
+
return orgOption;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// Otherwise, we need to select an org
|
|
25
|
+
const orgs = await tui.spinner('Fetching organizations', () => listOrganizations(apiClient));
|
|
26
|
+
|
|
27
|
+
// Use preference if available, otherwise prompt
|
|
28
|
+
const preferredOrgId = config?.preferences?.orgId;
|
|
29
|
+
return tui.selectOrganization(orgs, preferredOrgId);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* Checks if we're operating in org scope based on the --org option.
|
|
34
|
+
*/
|
|
35
|
+
export function isOrgScope(orgOption: boolean | string | undefined): boolean {
|
|
36
|
+
return orgOption === true || (typeof orgOption === 'string' && orgOption.length > 0);
|
|
37
|
+
}
|