forked from farhoodlabs/paperclip
778e775c35
## Thinking Path > - Paperclip orchestrates AI-agent companies and needs secrets handling to work across local development, hosted operators, and governed agent execution. > - The affected subsystem is the company-scoped secrets control plane: database schema, server services/routes, CLI workflows, and the Secrets settings UI. > - The gap was that secrets were local-only and operators could not manage provider vaults or import existing remote references without exposing plaintext. > - This branch adds provider vault configuration plus an AWS Secrets Manager remote-import path while preserving company boundaries, binding context, and audit trails. > - I kept the PR to a single branch PR, removed unrelated lockfile/package drift, rebased the full branch onto the current `public-gh/master`, and addressed fresh Greptile findings. > - The benefit is a reviewable implementation of provider-backed secrets with focused tests covering provider selection, import conflicts, deleted secret reuse, rotation guards, and AWS signing behavior. ## What Changed - Added provider vault support for company secrets, including provider config storage, default vault handling, health checks, binding usage, access events, and remote import preview/commit. - Added an AWS Secrets Manager provider using SigV4 request signing, bounded request timeouts, namespace guardrails, cached runtime credential resolution, and external-reference linking without plaintext reads. - Added Secrets UI surfaces for vault management and remote import, plus CLI/API documentation for setup and operations. - Stabilized routine webhook secret binding paths and SSH environment-driver fixture bindings discovered during verification. - Addressed Greptile and CI findings: no lockfile/package drift, monotonic migration metadata, disabled-vault default races, soft-deleted secret hiding/recreate behavior, remove behavior with disabled vaults, soft-deleted external-reference re-import, non-active rotation guards, managed-secret soft deletion through PATCH, and per-call AWS SDK credential client churn. - Rebased this branch onto `public-gh/master` at `0e1a5828` and force-pushed with lease to keep this as the single PR for the branch. ## Verification - `git fetch public-gh master` - `git rebase public-gh/master` - `git diff --name-only public-gh/master...HEAD | grep '^pnpm-lock\.yaml$' || true` confirmed `pnpm-lock.yaml` is not in the PR diff. - Confirmed migration ordering: master ends at `0081_optimal_dormammu`; this PR adds `0082_dry_vision` and `0083_company_secret_provider_configs`. - Inspected migrations for repeat safety: new tables/indexes use `IF NOT EXISTS`; foreign keys are guarded by `DO $$ ... IF NOT EXISTS`; column additions use `ADD COLUMN IF NOT EXISTS`. - `pnpm -r typecheck` passed before the Greptile follow-up commits. - `pnpm test:run` ran the full stable Vitest path before the Greptile follow-up commits; it completed with 3 timing-related failures under parallel load: `codex-local-execute.test.ts`, `cursor-local-execute.test.ts`, and `environment-service.test.ts`. - `pnpm --filter @paperclipai/server exec vitest run src/__tests__/codex-local-execute.test.ts src/__tests__/cursor-local-execute.test.ts src/__tests__/environment-service.test.ts` passed on targeted rerun (`24/24`). - `pnpm build` passed before the Greptile follow-up commits. Vite reported existing chunk-size/dynamic-import warnings. - After Greptile follow-up commits: `pnpm --filter @paperclipai/server exec vitest run src/__tests__/secrets-service.test.ts` passed (`26/26`). - After Greptile follow-up commits: `pnpm --filter @paperclipai/server exec vitest run src/__tests__/aws-secrets-manager-provider.test.ts src/__tests__/secrets-service.test.ts` passed (`39/39`). - After Greptile follow-up commits: `pnpm --filter @paperclipai/server typecheck` passed. - Captured Storybook screenshots from `ui/storybook-static` for visual review. - Latest PR checks on `5ca3a5cf`: `policy`, serialized server suites 1/4-4/4, `Canary Dry Run`, `e2e`, `security/snyk`, and `Greptile Review` pass; aggregate `verify` is still registering the completed child checks. - Greptile review loop continued through the latest requested pass; all Greptile review threads are resolved and the latest `Greptile Review` check on `5ca3a5cf` passed with 0 comments added. ## Screenshots Before: the provider-vault and remote-import surfaces did not exist on `master`; these are after-state screenshots from the Storybook fixtures.    ## Risks - Migration risk: this adds new secret provider tables and extends existing secret rows. The migrations were checked for monotonic ordering and idempotent guards, but reviewers should still inspect upgrade behavior carefully. - Provider risk: AWS support uses direct SigV4 requests. Automated tests cover signing, request timeouts, vault-config selection, namespace guardrails, pending-version archival, sanitized provider errors, and service-level cleanup paths. A real-vault AWS smoke test remains deployment validation for an operator with AWS credentials rather than an unverified merge blocker in this local branch. - UI risk: the Secrets page and import dialog are large new surfaces; screenshots are included above for reviewer inspection. - Verification risk: the full local stable test command hit parallel-load timing failures, although the exact failed files passed when rerun directly. - Operational risk: remote import intentionally avoids plaintext reads; operators must understand that imported external references resolve at runtime and may fail if AWS permissions change. > For core feature work, check [`ROADMAP.md`](ROADMAP.md) first and discuss it in `#dev` before opening the PR. Feature PRs that overlap with planned core work may need to be redirected — check the roadmap first. See `CONTRIBUTING.md`. ## Model Used - OpenAI Codex, GPT-5 coding agent with local shell/tool use in the Paperclip worktree. Exact context-window size was not exposed by the runtime. ## Checklist - [x] I have included a thinking path that traces from project context to this change - [x] I have specified the model used (with version and capability details) - [x] I have checked ROADMAP.md and confirmed this PR does not duplicate planned core work - [ ] I have run tests locally and they pass - [x] I have added or updated tests where applicable - [x] If this change affects the UI, I have included before/after screenshots - [x] I have updated relevant documentation to reflect my changes - [x] I have considered and documented any risks above - [x] I will address all Greptile and reviewer comments before requesting merge --------- Co-authored-by: Paperclip <noreply@paperclip.ing> Co-authored-by: Claude Sonnet 4.6 <noreply@anthropic.com>
502 lines
17 KiB
TypeScript
502 lines
17 KiB
TypeScript
import { Command } from "commander";
|
|
import pc from "picocolors";
|
|
import type {
|
|
Agent,
|
|
AgentEnvConfig,
|
|
CompanyPortabilityEnvInput,
|
|
CompanyPortabilityExportPreviewResult,
|
|
CompanyPortabilityInclude,
|
|
CompanySecret,
|
|
EnvBinding,
|
|
SecretProvider,
|
|
SecretProviderDescriptor,
|
|
} from "@paperclipai/shared";
|
|
import {
|
|
addCommonClientOptions,
|
|
formatInlineRecord,
|
|
handleCommandError,
|
|
printOutput,
|
|
resolveCommandContext,
|
|
type BaseClientOptions,
|
|
} from "./common.js";
|
|
|
|
interface SecretListOptions extends BaseClientOptions {
|
|
companyId?: string;
|
|
}
|
|
|
|
interface SecretDeclarationsOptions extends BaseClientOptions {
|
|
companyId?: string;
|
|
include?: string;
|
|
kind?: "all" | "secret" | "plain";
|
|
}
|
|
|
|
interface SecretCreateOptions extends BaseClientOptions {
|
|
companyId?: string;
|
|
name?: string;
|
|
key?: string;
|
|
provider?: SecretProvider;
|
|
value?: string;
|
|
valueEnv?: string;
|
|
description?: string;
|
|
}
|
|
|
|
interface SecretLinkOptions extends BaseClientOptions {
|
|
companyId?: string;
|
|
name?: string;
|
|
key?: string;
|
|
provider?: SecretProvider;
|
|
externalRef?: string;
|
|
providerVersionRef?: string;
|
|
description?: string;
|
|
}
|
|
|
|
interface SecretDoctorOptions extends BaseClientOptions {
|
|
companyId?: string;
|
|
}
|
|
|
|
interface SecretMigrateInlineEnvOptions extends BaseClientOptions {
|
|
companyId?: string;
|
|
apply?: boolean;
|
|
}
|
|
|
|
interface SecretProviderHealth {
|
|
provider: SecretProvider;
|
|
status: "ok" | "warn" | "error";
|
|
message: string;
|
|
warnings?: string[];
|
|
backupGuidance?: string[];
|
|
details?: Record<string, unknown>;
|
|
}
|
|
|
|
interface SecretProviderHealthResponse {
|
|
providers: SecretProviderHealth[];
|
|
}
|
|
|
|
export interface InlineSecretMigrationCandidate {
|
|
agentId: string;
|
|
agentName: string;
|
|
envKey: string;
|
|
secretName: string;
|
|
existingSecretId: string | null;
|
|
}
|
|
|
|
const SENSITIVE_ENV_KEY_RE =
|
|
/(^token$|[-_]?token$|api[-_]?key|access[-_]?token|auth(?:_?token)?|authorization|bearer|secret|passwd|password|credential|jwt|private[-_]?key|cookie|connectionstring)/i;
|
|
|
|
const DEFAULT_DECLARATION_INCLUDE: CompanyPortabilityInclude = {
|
|
company: true,
|
|
agents: true,
|
|
projects: true,
|
|
issues: false,
|
|
skills: false,
|
|
};
|
|
|
|
export function parseSecretsInclude(input: string | undefined): CompanyPortabilityInclude {
|
|
if (!input?.trim()) return { ...DEFAULT_DECLARATION_INCLUDE };
|
|
const values = input.split(",").map((part) => part.trim().toLowerCase()).filter(Boolean);
|
|
const include = {
|
|
company: values.includes("company"),
|
|
agents: values.includes("agents"),
|
|
projects: values.includes("projects"),
|
|
issues: values.includes("issues") || values.includes("tasks"),
|
|
skills: values.includes("skills"),
|
|
};
|
|
if (!Object.values(include).some(Boolean)) {
|
|
throw new Error("Invalid --include value. Use one or more of: company,agents,projects,issues,tasks,skills");
|
|
}
|
|
return include;
|
|
}
|
|
|
|
export function isSensitiveEnvKey(key: string): boolean {
|
|
return SENSITIVE_ENV_KEY_RE.test(key);
|
|
}
|
|
|
|
export function toPlainEnvValue(binding: unknown): string | null {
|
|
if (typeof binding === "string") return binding;
|
|
if (typeof binding !== "object" || binding === null || Array.isArray(binding)) return null;
|
|
const record = binding as Record<string, unknown>;
|
|
if (record.type === "plain" && typeof record.value === "string") return record.value;
|
|
return null;
|
|
}
|
|
|
|
export function buildInlineMigrationSecretName(agentId: string, key: string): string {
|
|
return `agent_${agentId.slice(0, 8)}_${key.toLowerCase()}`;
|
|
}
|
|
|
|
export function collectInlineSecretMigrationCandidates(
|
|
agents: Agent[],
|
|
existingSecrets: CompanySecret[],
|
|
): InlineSecretMigrationCandidate[] {
|
|
const secretByName = new Map(existingSecrets.map((secret) => [secret.name, secret]));
|
|
const candidates: InlineSecretMigrationCandidate[] = [];
|
|
|
|
for (const agent of agents) {
|
|
const env = asRecord(agent.adapterConfig.env);
|
|
if (!env) continue;
|
|
for (const [envKey, binding] of Object.entries(env)) {
|
|
if (!isSensitiveEnvKey(envKey)) continue;
|
|
const plain = toPlainEnvValue(binding);
|
|
if (plain === null || plain.trim().length === 0) continue;
|
|
const secretName = buildInlineMigrationSecretName(agent.id, envKey);
|
|
candidates.push({
|
|
agentId: agent.id,
|
|
agentName: agent.name,
|
|
envKey,
|
|
secretName,
|
|
existingSecretId: secretByName.get(secretName)?.id ?? null,
|
|
});
|
|
}
|
|
}
|
|
|
|
return candidates;
|
|
}
|
|
|
|
export function buildMigratedAgentEnv(
|
|
env: Record<string, unknown>,
|
|
secretIdByEnvKey: Map<string, string>,
|
|
): AgentEnvConfig {
|
|
const next: AgentEnvConfig = { ...(env as Record<string, EnvBinding>) };
|
|
for (const [envKey, secretId] of secretIdByEnvKey) {
|
|
next[envKey] = {
|
|
type: "secret_ref",
|
|
secretId,
|
|
version: "latest",
|
|
};
|
|
}
|
|
return next;
|
|
}
|
|
|
|
function asRecord(value: unknown): Record<string, unknown> | null {
|
|
if (typeof value !== "object" || value === null || Array.isArray(value)) return null;
|
|
return value as Record<string, unknown>;
|
|
}
|
|
|
|
function readValueFromOptions(opts: SecretCreateOptions): string {
|
|
if (opts.value !== undefined && opts.valueEnv !== undefined) {
|
|
throw new Error("Use only one of --value or --value-env.");
|
|
}
|
|
if (opts.valueEnv !== undefined) {
|
|
const value = process.env[opts.valueEnv];
|
|
if (!value) throw new Error(`Environment variable ${opts.valueEnv} is empty or unset.`);
|
|
return value;
|
|
}
|
|
if (opts.value !== undefined) return opts.value;
|
|
throw new Error("Secret value is required. Pass --value or --value-env.");
|
|
}
|
|
|
|
function renderDeclaration(input: CompanyPortabilityEnvInput): Record<string, unknown> {
|
|
const scope = input.agentSlug
|
|
? `agent:${input.agentSlug}`
|
|
: input.projectSlug
|
|
? `project:${input.projectSlug}`
|
|
: "company";
|
|
return {
|
|
key: input.key,
|
|
scope,
|
|
kind: input.kind,
|
|
requirement: input.requirement,
|
|
portability: input.portability,
|
|
hasDefault: input.defaultValue !== null && input.defaultValue.length > 0,
|
|
description: input.description,
|
|
};
|
|
}
|
|
|
|
function renderSecret(secret: CompanySecret): Record<string, unknown> {
|
|
return {
|
|
id: secret.id,
|
|
name: secret.name,
|
|
key: secret.key,
|
|
provider: secret.provider,
|
|
status: secret.status,
|
|
managedMode: secret.managedMode,
|
|
latestVersion: secret.latestVersion,
|
|
externalRef: secret.externalRef ? "yes" : "no",
|
|
};
|
|
}
|
|
|
|
function printProviderHealth(rows: SecretProviderHealth[], json: boolean): void {
|
|
if (json) {
|
|
printOutput(rows, { json: true });
|
|
return;
|
|
}
|
|
if (rows.length === 0) {
|
|
printOutput([], { json: false });
|
|
return;
|
|
}
|
|
for (const row of rows) {
|
|
console.log(
|
|
formatInlineRecord({
|
|
id: row.provider,
|
|
status: row.status,
|
|
message: row.message,
|
|
}),
|
|
);
|
|
for (const warning of row.warnings ?? []) {
|
|
console.log(pc.yellow(`warning=${warning}`));
|
|
}
|
|
const missingConfig = asStringArray(row.details?.missingConfig);
|
|
if (missingConfig.length > 0) {
|
|
console.log(pc.dim(`missingConfig=${missingConfig.join(",")}`));
|
|
}
|
|
const credentialSource = typeof row.details?.credentialSource === "string"
|
|
? row.details.credentialSource
|
|
: null;
|
|
if (credentialSource) {
|
|
console.log(pc.dim(`credentialSource=${credentialSource}`));
|
|
}
|
|
const detectedCredentialSources = asStringArray(row.details?.detectedCredentialSources);
|
|
if (detectedCredentialSources.length > 0) {
|
|
console.log(pc.dim(`detectedCredentialSources=${detectedCredentialSources.join(",")}`));
|
|
}
|
|
for (const guidance of row.backupGuidance ?? []) {
|
|
console.log(pc.dim(`backup=${guidance}`));
|
|
}
|
|
}
|
|
}
|
|
|
|
function asStringArray(value: unknown): string[] {
|
|
return Array.isArray(value)
|
|
? value.filter((entry): entry is string => typeof entry === "string" && entry.length > 0)
|
|
: [];
|
|
}
|
|
|
|
async function migrateInlineEnv(opts: SecretMigrateInlineEnvOptions): Promise<void> {
|
|
const ctx = resolveCommandContext(opts, { requireCompany: true });
|
|
const companyId = ctx.companyId!;
|
|
const agents = (await ctx.api.get<Agent[]>(`/api/companies/${companyId}/agents`)) ?? [];
|
|
const secrets = (await ctx.api.get<CompanySecret[]>(`/api/companies/${companyId}/secrets`)) ?? [];
|
|
const candidates = collectInlineSecretMigrationCandidates(agents, secrets);
|
|
|
|
if (!opts.apply) {
|
|
printOutput(
|
|
{
|
|
apply: false,
|
|
agentsToUpdate: new Set(candidates.map((candidate) => candidate.agentId)).size,
|
|
secretsToCreate: candidates.filter((candidate) => !candidate.existingSecretId).length,
|
|
secretsToRotate: candidates.filter((candidate) => candidate.existingSecretId).length,
|
|
candidates,
|
|
},
|
|
{ json: ctx.json },
|
|
);
|
|
if (!ctx.json) {
|
|
console.log(pc.dim("Re-run with --apply to create/rotate secrets and update agent env bindings."));
|
|
}
|
|
return;
|
|
}
|
|
|
|
const createdOrRotated = new Map<string, string>();
|
|
let createdSecrets = 0;
|
|
let rotatedSecrets = 0;
|
|
|
|
for (const candidate of candidates) {
|
|
const agent = agents.find((row) => row.id === candidate.agentId);
|
|
const env = asRecord(agent?.adapterConfig.env);
|
|
const value = env ? toPlainEnvValue(env[candidate.envKey]) : null;
|
|
if (!value) continue;
|
|
|
|
if (candidate.existingSecretId) {
|
|
await ctx.api.post(`/api/secrets/${candidate.existingSecretId}/rotate`, { value });
|
|
createdOrRotated.set(`${candidate.agentId}:${candidate.envKey}`, candidate.existingSecretId);
|
|
rotatedSecrets += 1;
|
|
continue;
|
|
}
|
|
|
|
const created = await ctx.api.post<CompanySecret>(`/api/companies/${companyId}/secrets`, {
|
|
name: candidate.secretName,
|
|
provider: "local_encrypted",
|
|
value,
|
|
description: `Migrated from agent ${candidate.agentId} env ${candidate.envKey}`,
|
|
});
|
|
if (!created) throw new Error(`Secret create returned no data for ${candidate.secretName}`);
|
|
createdOrRotated.set(`${candidate.agentId}:${candidate.envKey}`, created.id);
|
|
createdSecrets += 1;
|
|
}
|
|
|
|
let updatedAgents = 0;
|
|
for (const agent of agents) {
|
|
const env = asRecord(agent.adapterConfig.env);
|
|
if (!env) continue;
|
|
const secretIdByEnvKey = new Map<string, string>();
|
|
for (const [key] of Object.entries(env)) {
|
|
const secretId = createdOrRotated.get(`${agent.id}:${key}`);
|
|
if (secretId) secretIdByEnvKey.set(key, secretId);
|
|
}
|
|
if (secretIdByEnvKey.size === 0) continue;
|
|
const adapterConfig = {
|
|
...agent.adapterConfig,
|
|
env: buildMigratedAgentEnv(env, secretIdByEnvKey),
|
|
};
|
|
await ctx.api.patch(`/api/agents/${agent.id}`, {
|
|
adapterConfig,
|
|
replaceAdapterConfig: true,
|
|
});
|
|
updatedAgents += 1;
|
|
}
|
|
|
|
printOutput(
|
|
{
|
|
apply: true,
|
|
updatedAgents,
|
|
createdSecrets,
|
|
rotatedSecrets,
|
|
},
|
|
{ json: ctx.json },
|
|
);
|
|
}
|
|
|
|
export function registerSecretCommands(program: Command): void {
|
|
const secrets = program.command("secrets").description("Secret declaration and provider operations");
|
|
|
|
addCommonClientOptions(
|
|
secrets
|
|
.command("list")
|
|
.description("List secret metadata for a company")
|
|
.requiredOption("-C, --company-id <id>", "Company ID")
|
|
.action(async (opts: SecretListOptions) => {
|
|
try {
|
|
const ctx = resolveCommandContext(opts, { requireCompany: true });
|
|
const rows = (await ctx.api.get<CompanySecret[]>(`/api/companies/${ctx.companyId}/secrets`)) ?? [];
|
|
printOutput(ctx.json ? rows : rows.map(renderSecret), { json: ctx.json });
|
|
} catch (err) {
|
|
handleCommandError(err);
|
|
}
|
|
}),
|
|
);
|
|
|
|
addCommonClientOptions(
|
|
secrets
|
|
.command("declarations")
|
|
.description("List portable env declarations emitted by company export")
|
|
.requiredOption("-C, --company-id <id>", "Company ID")
|
|
.option("--include <values>", "Comma-separated include set: company,agents,projects,issues,tasks,skills", "company,agents,projects")
|
|
.option("--kind <kind>", "Filter declarations: all | secret | plain", "all")
|
|
.action(async (opts: SecretDeclarationsOptions) => {
|
|
try {
|
|
const ctx = resolveCommandContext(opts, { requireCompany: true });
|
|
const kind = opts.kind ?? "all";
|
|
if (!["all", "secret", "plain"].includes(kind)) {
|
|
throw new Error("Invalid --kind value. Use: all, secret, plain");
|
|
}
|
|
const preview = await ctx.api.post<CompanyPortabilityExportPreviewResult>(
|
|
`/api/companies/${ctx.companyId}/exports/preview`,
|
|
{ include: parseSecretsInclude(opts.include) },
|
|
);
|
|
const declarations = (preview?.manifest.envInputs ?? [])
|
|
.filter((entry) => kind === "all" || entry.kind === kind);
|
|
printOutput(ctx.json ? declarations : declarations.map(renderDeclaration), { json: ctx.json });
|
|
} catch (err) {
|
|
handleCommandError(err);
|
|
}
|
|
}),
|
|
);
|
|
|
|
addCommonClientOptions(
|
|
secrets
|
|
.command("create")
|
|
.description("Create a Paperclip-managed secret")
|
|
.requiredOption("-C, --company-id <id>", "Company ID")
|
|
.requiredOption("--name <name>", "Secret display name")
|
|
.option("--key <key>", "Portable secret key")
|
|
.option("--provider <provider>", "Secret provider id")
|
|
.option("--value <value>", "Secret value")
|
|
.option("--value-env <name>", "Read secret value from an environment variable")
|
|
.option("--description <text>", "Description")
|
|
.action(async (opts: SecretCreateOptions) => {
|
|
try {
|
|
const ctx = resolveCommandContext(opts, { requireCompany: true });
|
|
const created = await ctx.api.post<CompanySecret>(`/api/companies/${ctx.companyId}/secrets`, {
|
|
name: opts.name,
|
|
key: opts.key,
|
|
provider: opts.provider,
|
|
value: readValueFromOptions(opts),
|
|
description: opts.description,
|
|
});
|
|
printOutput(ctx.json ? created : renderSecret(created!), { json: ctx.json });
|
|
} catch (err) {
|
|
handleCommandError(err);
|
|
}
|
|
}),
|
|
);
|
|
|
|
addCommonClientOptions(
|
|
secrets
|
|
.command("link")
|
|
.description("Link an external provider-owned secret without storing its value in Paperclip")
|
|
.requiredOption("-C, --company-id <id>", "Company ID")
|
|
.requiredOption("--name <name>", "Secret display name")
|
|
.requiredOption("--provider <provider>", "Secret provider id")
|
|
.requiredOption("--external-ref <ref>", "Provider secret ARN/name/path/reference")
|
|
.option("--key <key>", "Portable secret key")
|
|
.option("--provider-version-ref <ref>", "Provider version id or label")
|
|
.option("--description <text>", "Description")
|
|
.action(async (opts: SecretLinkOptions) => {
|
|
try {
|
|
const ctx = resolveCommandContext(opts, { requireCompany: true });
|
|
const created = await ctx.api.post<CompanySecret>(`/api/companies/${ctx.companyId}/secrets`, {
|
|
name: opts.name,
|
|
key: opts.key,
|
|
provider: opts.provider,
|
|
managedMode: "external_reference",
|
|
externalRef: opts.externalRef,
|
|
providerVersionRef: opts.providerVersionRef,
|
|
description: opts.description,
|
|
});
|
|
printOutput(ctx.json ? created : renderSecret(created!), { json: ctx.json });
|
|
} catch (err) {
|
|
handleCommandError(err);
|
|
}
|
|
}),
|
|
);
|
|
|
|
addCommonClientOptions(
|
|
secrets
|
|
.command("doctor")
|
|
.description("Run secret provider health checks through the Paperclip API")
|
|
.requiredOption("-C, --company-id <id>", "Company ID")
|
|
.action(async (opts: SecretDoctorOptions) => {
|
|
try {
|
|
const ctx = resolveCommandContext(opts, { requireCompany: true });
|
|
const health = await ctx.api.get<SecretProviderHealthResponse>(
|
|
`/api/companies/${ctx.companyId}/secret-providers/health`,
|
|
);
|
|
printProviderHealth(health?.providers ?? [], ctx.json);
|
|
} catch (err) {
|
|
handleCommandError(err);
|
|
}
|
|
}),
|
|
);
|
|
|
|
addCommonClientOptions(
|
|
secrets
|
|
.command("providers")
|
|
.description("List configured secret provider descriptors")
|
|
.requiredOption("-C, --company-id <id>", "Company ID")
|
|
.action(async (opts: SecretDoctorOptions) => {
|
|
try {
|
|
const ctx = resolveCommandContext(opts, { requireCompany: true });
|
|
const rows = (await ctx.api.get<SecretProviderDescriptor[]>(
|
|
`/api/companies/${ctx.companyId}/secret-providers`,
|
|
)) ?? [];
|
|
printOutput(rows, { json: ctx.json });
|
|
} catch (err) {
|
|
handleCommandError(err);
|
|
}
|
|
}),
|
|
);
|
|
|
|
addCommonClientOptions(
|
|
secrets
|
|
.command("migrate-inline-env")
|
|
.description("Migrate inline sensitive agent env values into secret references")
|
|
.requiredOption("-C, --company-id <id>", "Company ID")
|
|
.option("--apply", "Persist changes; default is a dry run", false)
|
|
.action(async (opts: SecretMigrateInlineEnvOptions) => {
|
|
try {
|
|
await migrateInlineEnv(opts);
|
|
} catch (err) {
|
|
handleCommandError(err);
|
|
}
|
|
}),
|
|
);
|
|
}
|