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>
1054 lines
35 KiB
TypeScript
1054 lines
35 KiB
TypeScript
import { createHash, createHmac } from "node:crypto";
|
|
import { S3Client } from "@aws-sdk/client-s3";
|
|
import type { DeploymentMode } from "@paperclipai/shared";
|
|
import { unprocessable } from "../errors.js";
|
|
import type {
|
|
PreparedSecretVersion,
|
|
RemoteSecretListResult,
|
|
SecretProviderClientErrorCode,
|
|
SecretProviderHealthCheck,
|
|
SecretProviderModule,
|
|
SecretProviderValidationResult,
|
|
SecretProviderVaultRuntimeConfig,
|
|
SecretProviderWriteContext,
|
|
StoredSecretVersionMaterial,
|
|
} from "./types.js";
|
|
import { SecretProviderClientError } from "./types.js";
|
|
|
|
const AWS_SECRETS_MANAGER_SCHEME = "aws_secrets_manager_v1";
|
|
const DEFAULT_PREFIX = "paperclip";
|
|
const DEFAULT_OWNER_TAG = "paperclip";
|
|
const DEFAULT_VERSION_STAGE = "AWSCURRENT";
|
|
const PAPERCLIP_PENDING_VERSION_STAGE = "PAPERCLIP_PENDING";
|
|
const DEFAULT_DELETE_RECOVERY_WINDOW_DAYS = 30;
|
|
const AWS_SECRETS_MANAGER_REQUEST_TIMEOUT_MS = 30_000;
|
|
const AWS_CREDENTIAL_CACHE_TTL_MS = 5 * 60_000;
|
|
const AWS_CREDENTIAL_EXPIRATION_SKEW_MS = 60_000;
|
|
const AWS_RUNTIME_CREDENTIAL_WARNING =
|
|
"AWS bootstrap credentials must be available to the Paperclip server runtime through the AWS SDK default credential provider chain: IAM role/workload identity, AWS_PROFILE/SSO/shared credentials, web identity, container/instance metadata, or short-lived shell credentials.";
|
|
const AWS_CREDENTIAL_CUSTODY_WARNING =
|
|
"Do not store AWS root credentials or long-lived IAM user access keys in Paperclip company_secrets; the AWS provider bootstrap belongs in deployment infrastructure, the process environment, an AWS profile, or the orchestrator secret store.";
|
|
|
|
interface AwsSecretsManagerMaterial extends StoredSecretVersionMaterial {
|
|
scheme: typeof AWS_SECRETS_MANAGER_SCHEME;
|
|
secretId: string;
|
|
versionId: string | null;
|
|
source: "managed" | "external_reference";
|
|
}
|
|
|
|
interface AwsSecretsManagerConfig {
|
|
region: string;
|
|
endpoint: string;
|
|
deploymentId: string;
|
|
prefix: string;
|
|
kmsKeyId: string | null;
|
|
environmentTag: string;
|
|
providerOwnerTag: string;
|
|
deleteRecoveryWindowDays: number;
|
|
}
|
|
|
|
interface AwsSecretsManagerTag {
|
|
Key: string;
|
|
Value: string;
|
|
}
|
|
|
|
interface AwsSecretsManagerListSecretEntry {
|
|
ARN?: string;
|
|
Name?: string;
|
|
Description?: string;
|
|
KmsKeyId?: string;
|
|
CreatedDate?: string | number | Date;
|
|
LastAccessedDate?: string | number | Date;
|
|
LastChangedDate?: string | number | Date;
|
|
DeletedDate?: string | number | Date;
|
|
Tags?: AwsSecretsManagerTag[];
|
|
}
|
|
|
|
interface AwsCredentialIdentity {
|
|
accessKeyId: string;
|
|
secretAccessKey: string;
|
|
sessionToken?: string;
|
|
}
|
|
|
|
interface CachedAwsCredentialProvider {
|
|
client: S3Client;
|
|
credentials: AwsCredentialIdentity | null;
|
|
expiresAt: number;
|
|
pending: Promise<AwsCredentialIdentity> | null;
|
|
}
|
|
|
|
type ManagedSecretNamespaceContext = Pick<SecretProviderWriteContext, "companyId" | "secretKey">;
|
|
|
|
const awsCredentialProviders = new Map<string, CachedAwsCredentialProvider>();
|
|
|
|
interface AwsSecretsManagerGateway {
|
|
createSecret(input: {
|
|
Name: string;
|
|
SecretString: string;
|
|
KmsKeyId?: string;
|
|
Description?: string;
|
|
Tags: AwsSecretsManagerTag[];
|
|
}): Promise<{
|
|
ARN?: string;
|
|
Name?: string;
|
|
VersionId?: string;
|
|
}>;
|
|
putSecretValue(input: {
|
|
SecretId: string;
|
|
SecretString: string;
|
|
VersionStages?: string[];
|
|
}): Promise<{
|
|
ARN?: string;
|
|
Name?: string;
|
|
VersionId?: string;
|
|
}>;
|
|
getSecretValue(input: {
|
|
SecretId: string;
|
|
VersionId?: string;
|
|
VersionStage?: string;
|
|
}): Promise<{
|
|
SecretString?: string;
|
|
ARN?: string;
|
|
Name?: string;
|
|
VersionId?: string;
|
|
}>;
|
|
deleteSecret(input: {
|
|
SecretId: string;
|
|
RecoveryWindowInDays: number;
|
|
}): Promise<unknown>;
|
|
updateSecretVersionStage?(input: {
|
|
SecretId: string;
|
|
VersionStage: string;
|
|
RemoveFromVersionId?: string;
|
|
MoveToVersionId?: string;
|
|
}): Promise<unknown>;
|
|
listSecrets?(input: {
|
|
MaxResults?: number;
|
|
NextToken?: string;
|
|
Filters?: Array<{
|
|
Key: "all" | "name" | "description" | "tag-key" | "tag-value" | "primary-region" | "owning-service";
|
|
Values: string[];
|
|
}>;
|
|
IncludePlannedDeletion?: boolean;
|
|
}): Promise<{
|
|
SecretList?: AwsSecretsManagerListSecretEntry[];
|
|
NextToken?: string;
|
|
}>;
|
|
}
|
|
|
|
function sha256Hex(value: string): string {
|
|
return createHash("sha256").update(value).digest("hex");
|
|
}
|
|
|
|
function hmac(key: string | Buffer, value: string) {
|
|
return createHmac("sha256", key).update(value).digest();
|
|
}
|
|
|
|
function awsDateParts(now = new Date()) {
|
|
const iso = now.toISOString().replace(/[:-]|\.\d{3}/g, "");
|
|
return {
|
|
amzDate: iso,
|
|
dateStamp: iso.slice(0, 8),
|
|
};
|
|
}
|
|
|
|
function canonicalHeaderValue(value: string) {
|
|
return value.trim().replace(/\s+/g, " ");
|
|
}
|
|
|
|
function signAwsSecretsManagerRequest(input: {
|
|
endpoint: URL;
|
|
region: string;
|
|
operation: string;
|
|
body: string;
|
|
credentials: AwsCredentialIdentity;
|
|
}) {
|
|
const { amzDate, dateStamp } = awsDateParts();
|
|
const payloadHash = sha256Hex(input.body);
|
|
const headers: Record<string, string> = {
|
|
"content-type": "application/x-amz-json-1.1",
|
|
host: input.endpoint.host,
|
|
"x-amz-content-sha256": payloadHash,
|
|
"x-amz-date": amzDate,
|
|
"x-amz-target": `secretsmanager.${input.operation}`,
|
|
};
|
|
if (input.credentials.sessionToken) {
|
|
headers["x-amz-security-token"] = input.credentials.sessionToken;
|
|
}
|
|
|
|
const sortedHeaderNames = Object.keys(headers).sort();
|
|
const canonicalHeaders = sortedHeaderNames
|
|
.map((name) => `${name}:${canonicalHeaderValue(headers[name] ?? "")}\n`)
|
|
.join("");
|
|
const signedHeaders = sortedHeaderNames.join(";");
|
|
const canonicalRequest = [
|
|
"POST",
|
|
input.endpoint.pathname || "/",
|
|
"",
|
|
canonicalHeaders,
|
|
signedHeaders,
|
|
payloadHash,
|
|
].join("\n");
|
|
const credentialScope = `${dateStamp}/${input.region}/secretsmanager/aws4_request`;
|
|
const stringToSign = [
|
|
"AWS4-HMAC-SHA256",
|
|
amzDate,
|
|
credentialScope,
|
|
sha256Hex(canonicalRequest),
|
|
].join("\n");
|
|
const dateKey = hmac(`AWS4${input.credentials.secretAccessKey}`, dateStamp);
|
|
const regionKey = hmac(dateKey, input.region);
|
|
const serviceKey = hmac(regionKey, "secretsmanager");
|
|
const signingKey = hmac(serviceKey, "aws4_request");
|
|
const signature = createHmac("sha256", signingKey).update(stringToSign).digest("hex");
|
|
|
|
return {
|
|
...headers,
|
|
authorization:
|
|
`AWS4-HMAC-SHA256 Credential=${input.credentials.accessKeyId}/${credentialScope}, ` +
|
|
`SignedHeaders=${signedHeaders}, Signature=${signature}`,
|
|
};
|
|
}
|
|
|
|
async function loadAwsCredentials(region: string): Promise<AwsCredentialIdentity> {
|
|
const now = Date.now();
|
|
let cached = awsCredentialProviders.get(region);
|
|
if (!cached) {
|
|
// S3Client is only used as a carrier for the AWS SDK default credential provider chain.
|
|
// No S3 API calls are made here; switch to defaultProvider({ region }) if we add that dependency.
|
|
cached = {
|
|
client: new S3Client({ region }),
|
|
credentials: null,
|
|
expiresAt: 0,
|
|
pending: null,
|
|
};
|
|
awsCredentialProviders.set(region, cached);
|
|
}
|
|
|
|
if (cached.credentials && cached.expiresAt > now) return cached.credentials;
|
|
if (cached.pending) return cached.pending;
|
|
|
|
cached.pending = (async () => {
|
|
const credentialSource = cached.client.config.credentials;
|
|
const credentials = typeof credentialSource === "function"
|
|
? await credentialSource()
|
|
: await credentialSource;
|
|
if (!credentials?.accessKeyId || !credentials.secretAccessKey) {
|
|
throw new Error("AWS SDK default credential provider chain did not return credentials");
|
|
}
|
|
const resolved = {
|
|
accessKeyId: credentials.accessKeyId,
|
|
secretAccessKey: credentials.secretAccessKey,
|
|
sessionToken: credentials.sessionToken,
|
|
};
|
|
const expiration = (credentials as { expiration?: Date }).expiration?.getTime();
|
|
cached.credentials = resolved;
|
|
cached.expiresAt = Math.min(
|
|
now + AWS_CREDENTIAL_CACHE_TTL_MS,
|
|
expiration ? expiration - AWS_CREDENTIAL_EXPIRATION_SKEW_MS : Number.POSITIVE_INFINITY,
|
|
);
|
|
return resolved;
|
|
})().finally(() => {
|
|
if (cached) cached.pending = null;
|
|
});
|
|
|
|
return cached.pending;
|
|
}
|
|
|
|
function configuredAwsSecretsManagerDescriptor() {
|
|
return {
|
|
id: "aws_secrets_manager" as const,
|
|
label: "AWS Secrets Manager",
|
|
requiresExternalRef: false,
|
|
supportsManagedValues: true,
|
|
supportsExternalReferences: true,
|
|
configured: canLoadAwsSecretsManagerConfig(),
|
|
};
|
|
}
|
|
|
|
function canLoadAwsSecretsManagerConfig() {
|
|
return getAwsConfigReadiness().missingConfig.length === 0;
|
|
}
|
|
|
|
function asOptionalNonEmptyString(value: unknown): string | null {
|
|
return typeof value === "string" && value.trim().length > 0 ? value.trim() : null;
|
|
}
|
|
|
|
function readProviderVaultConfig(input: SecretProviderVaultRuntimeConfig): AwsSecretsManagerConfig {
|
|
if (input.provider !== "aws_secrets_manager") {
|
|
throw unprocessable("AWS Secrets Manager provider received a mismatched provider vault");
|
|
}
|
|
if (input.status === "disabled") {
|
|
throw unprocessable("AWS Secrets Manager provider vault is disabled");
|
|
}
|
|
if (input.status === "coming_soon") {
|
|
throw unprocessable("AWS Secrets Manager provider vault runtime is locked while coming soon");
|
|
}
|
|
const region = asOptionalNonEmptyString(input.config.region);
|
|
if (!region) {
|
|
throw unprocessable("AWS Secrets Manager provider vault requires non-secret config: region");
|
|
}
|
|
const recoveryWindowRaw = process.env.PAPERCLIP_SECRETS_AWS_DELETE_RECOVERY_DAYS?.trim();
|
|
const recoveryWindow = recoveryWindowRaw ? Number(recoveryWindowRaw) : DEFAULT_DELETE_RECOVERY_WINDOW_DAYS;
|
|
if (!Number.isFinite(recoveryWindow) || recoveryWindow < 7 || recoveryWindow > 30) {
|
|
throw unprocessable(
|
|
"PAPERCLIP_SECRETS_AWS_DELETE_RECOVERY_DAYS must be an integer between 7 and 30",
|
|
);
|
|
}
|
|
|
|
return {
|
|
region,
|
|
endpoint:
|
|
process.env.PAPERCLIP_SECRETS_AWS_ENDPOINT?.trim() ||
|
|
`https://secretsmanager.${region}.amazonaws.com`,
|
|
deploymentId: sanitizePathSegment(
|
|
asOptionalNonEmptyString(input.config.namespace) ?? input.id,
|
|
),
|
|
prefix: sanitizePathSegment(
|
|
asOptionalNonEmptyString(input.config.secretNamePrefix) || DEFAULT_PREFIX,
|
|
),
|
|
kmsKeyId: asOptionalNonEmptyString(input.config.kmsKeyId),
|
|
environmentTag:
|
|
asOptionalNonEmptyString(input.config.environmentTag) ||
|
|
process.env.NODE_ENV?.trim() ||
|
|
"unknown",
|
|
providerOwnerTag:
|
|
asOptionalNonEmptyString(input.config.ownerTag) || DEFAULT_OWNER_TAG,
|
|
deleteRecoveryWindowDays: recoveryWindow,
|
|
};
|
|
}
|
|
|
|
function getAwsConfigReadiness() {
|
|
const region = (
|
|
process.env.PAPERCLIP_SECRETS_AWS_REGION ??
|
|
process.env.AWS_REGION ??
|
|
process.env.AWS_DEFAULT_REGION
|
|
)?.trim();
|
|
const deploymentId = process.env.PAPERCLIP_SECRETS_AWS_DEPLOYMENT_ID?.trim();
|
|
const kmsKeyId = process.env.PAPERCLIP_SECRETS_AWS_KMS_KEY_ID?.trim();
|
|
const missingConfig: string[] = [];
|
|
|
|
if (!region) {
|
|
missingConfig.push("PAPERCLIP_SECRETS_AWS_REGION or AWS_REGION/AWS_DEFAULT_REGION");
|
|
}
|
|
if (!deploymentId) {
|
|
missingConfig.push("PAPERCLIP_SECRETS_AWS_DEPLOYMENT_ID");
|
|
}
|
|
if (!kmsKeyId) {
|
|
missingConfig.push("PAPERCLIP_SECRETS_AWS_KMS_KEY_ID");
|
|
}
|
|
|
|
return {
|
|
missingConfig,
|
|
region: region || null,
|
|
deploymentId: deploymentId || null,
|
|
kmsKeyConfigured: Boolean(kmsKeyId),
|
|
credentialSources: describeDetectedAwsCredentialSources(),
|
|
};
|
|
}
|
|
|
|
function describeDetectedAwsCredentialSources() {
|
|
const sources: string[] = [];
|
|
if (process.env.AWS_PROFILE?.trim()) sources.push("AWS_PROFILE/shared config");
|
|
if (process.env.AWS_ACCESS_KEY_ID?.trim() && process.env.AWS_SECRET_ACCESS_KEY?.trim()) {
|
|
sources.push("temporary AWS_ACCESS_KEY_ID/AWS_SECRET_ACCESS_KEY environment credentials");
|
|
}
|
|
if (process.env.AWS_WEB_IDENTITY_TOKEN_FILE?.trim() && process.env.AWS_ROLE_ARN?.trim()) {
|
|
sources.push("AWS web identity token");
|
|
}
|
|
if (
|
|
process.env.AWS_CONTAINER_CREDENTIALS_RELATIVE_URI?.trim() ||
|
|
process.env.AWS_CONTAINER_CREDENTIALS_FULL_URI?.trim()
|
|
) {
|
|
sources.push("AWS container credentials endpoint");
|
|
}
|
|
if (process.env.AWS_SHARED_CREDENTIALS_FILE?.trim() || process.env.AWS_CONFIG_FILE?.trim()) {
|
|
sources.push("custom AWS shared credentials/config file");
|
|
}
|
|
return sources;
|
|
}
|
|
|
|
function loadAwsSecretsManagerConfig(): AwsSecretsManagerConfig {
|
|
const readiness = getAwsConfigReadiness();
|
|
const region =
|
|
process.env.PAPERCLIP_SECRETS_AWS_REGION?.trim() ||
|
|
process.env.AWS_REGION?.trim() ||
|
|
process.env.AWS_DEFAULT_REGION?.trim();
|
|
const deploymentId = process.env.PAPERCLIP_SECRETS_AWS_DEPLOYMENT_ID?.trim();
|
|
const kmsKeyId = process.env.PAPERCLIP_SECRETS_AWS_KMS_KEY_ID?.trim();
|
|
|
|
if (readiness.missingConfig.length > 0) {
|
|
throw unprocessable(
|
|
`AWS Secrets Manager provider requires non-secret config: ${readiness.missingConfig.join(", ")}`,
|
|
);
|
|
}
|
|
if (!region) {
|
|
throw unprocessable(
|
|
"AWS Secrets Manager provider requires PAPERCLIP_SECRETS_AWS_REGION or AWS_REGION",
|
|
);
|
|
}
|
|
if (!deploymentId) {
|
|
throw unprocessable(
|
|
"AWS Secrets Manager provider requires PAPERCLIP_SECRETS_AWS_DEPLOYMENT_ID",
|
|
);
|
|
}
|
|
if (!kmsKeyId) {
|
|
throw unprocessable(
|
|
"AWS Secrets Manager provider requires PAPERCLIP_SECRETS_AWS_KMS_KEY_ID",
|
|
);
|
|
}
|
|
|
|
const recoveryWindowRaw = process.env.PAPERCLIP_SECRETS_AWS_DELETE_RECOVERY_DAYS?.trim();
|
|
const recoveryWindow = recoveryWindowRaw ? Number(recoveryWindowRaw) : DEFAULT_DELETE_RECOVERY_WINDOW_DAYS;
|
|
if (!Number.isFinite(recoveryWindow) || recoveryWindow < 7 || recoveryWindow > 30) {
|
|
throw unprocessable(
|
|
"PAPERCLIP_SECRETS_AWS_DELETE_RECOVERY_DAYS must be an integer between 7 and 30",
|
|
);
|
|
}
|
|
|
|
return {
|
|
region,
|
|
endpoint:
|
|
process.env.PAPERCLIP_SECRETS_AWS_ENDPOINT?.trim() ||
|
|
`https://secretsmanager.${region}.amazonaws.com`,
|
|
deploymentId,
|
|
prefix: sanitizePathSegment(process.env.PAPERCLIP_SECRETS_AWS_PREFIX?.trim() || DEFAULT_PREFIX),
|
|
kmsKeyId,
|
|
environmentTag:
|
|
process.env.PAPERCLIP_SECRETS_AWS_ENVIRONMENT?.trim() ||
|
|
process.env.NODE_ENV?.trim() ||
|
|
"unknown",
|
|
providerOwnerTag:
|
|
process.env.PAPERCLIP_SECRETS_AWS_PROVIDER_OWNER?.trim() || DEFAULT_OWNER_TAG,
|
|
deleteRecoveryWindowDays: recoveryWindow,
|
|
};
|
|
}
|
|
|
|
function sanitizePathSegment(input: string) {
|
|
return input
|
|
.trim()
|
|
.replace(/[^A-Za-z0-9/_+=.@-]+/g, "-")
|
|
.replace(/\/+/g, "/")
|
|
.replace(/^\/+|\/+$/g, "");
|
|
}
|
|
|
|
function buildManagedSecretName(
|
|
config: AwsSecretsManagerConfig,
|
|
context: ManagedSecretNamespaceContext | undefined,
|
|
) {
|
|
if (!context) {
|
|
throw unprocessable("AWS Secrets Manager provider requires secret context for managed values");
|
|
}
|
|
return [
|
|
sanitizePathSegment(config.prefix),
|
|
sanitizePathSegment(config.deploymentId),
|
|
sanitizePathSegment(context.companyId),
|
|
sanitizePathSegment(context.secretKey),
|
|
]
|
|
.filter(Boolean)
|
|
.join("/");
|
|
}
|
|
|
|
function buildManagedSecretId(
|
|
config: AwsSecretsManagerConfig,
|
|
context: ManagedSecretNamespaceContext | undefined,
|
|
) {
|
|
return buildManagedSecretName(config, context);
|
|
}
|
|
|
|
function escapeRegExp(value: string) {
|
|
return value.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
|
|
}
|
|
|
|
function extractAwsSecretName(externalRef: string) {
|
|
const trimmed = externalRef.trim();
|
|
const arnMatch = /^arn:[^:]+:secretsmanager:[^:]*:[^:]*:secret:(.+)$/i.exec(trimmed);
|
|
return arnMatch?.[1] ?? trimmed;
|
|
}
|
|
|
|
function isManagedSecretRefForContext(
|
|
config: AwsSecretsManagerConfig,
|
|
context: ManagedSecretNamespaceContext | undefined,
|
|
externalRef: string | null | undefined,
|
|
) {
|
|
if (!externalRef?.trim()) return false;
|
|
const expectedName = buildManagedSecretName(config, context);
|
|
const actualName = extractAwsSecretName(externalRef);
|
|
return new RegExp(`^${escapeRegExp(expectedName)}(?:-[A-Za-z0-9]{6})?$`).test(actualName);
|
|
}
|
|
|
|
function isManagedSecretNamespaceRef(
|
|
config: AwsSecretsManagerConfig,
|
|
externalRef: string | null | undefined,
|
|
) {
|
|
if (!externalRef?.trim()) return false;
|
|
const namespacePrefix = [
|
|
sanitizePathSegment(config.prefix),
|
|
sanitizePathSegment(config.deploymentId),
|
|
]
|
|
.filter(Boolean)
|
|
.join("/");
|
|
if (!namespacePrefix) return false;
|
|
const actualName = extractAwsSecretName(externalRef);
|
|
return actualName === namespacePrefix || actualName.startsWith(`${namespacePrefix}/`);
|
|
}
|
|
|
|
function assertNotManagedNamespaceExternalRef(
|
|
config: AwsSecretsManagerConfig,
|
|
externalRef: string,
|
|
) {
|
|
if (!isManagedSecretNamespaceRef(config, externalRef)) return;
|
|
throw unprocessable(
|
|
"AWS Paperclip-managed namespace secrets cannot be imported as external references",
|
|
);
|
|
}
|
|
|
|
function resolveManagedSecretRef(input: {
|
|
config: AwsSecretsManagerConfig;
|
|
context: ManagedSecretNamespaceContext | undefined;
|
|
externalRefs: Array<string | null | undefined>;
|
|
}) {
|
|
let sawNonEmptyExternalRef = false;
|
|
for (const externalRef of input.externalRefs) {
|
|
if (externalRef?.trim()) {
|
|
sawNonEmptyExternalRef = true;
|
|
}
|
|
if (externalRef?.trim() && isManagedSecretRefForContext(input.config, input.context, externalRef)) {
|
|
return externalRef.trim();
|
|
}
|
|
}
|
|
if (sawNonEmptyExternalRef) {
|
|
throw unprocessable(
|
|
"AWS Secrets Manager managed secret ref drifted outside the derived deployment/company scope",
|
|
);
|
|
}
|
|
return buildManagedSecretId(input.config, input.context);
|
|
}
|
|
|
|
function buildManagedSecretTags(
|
|
config: AwsSecretsManagerConfig,
|
|
context: SecretProviderWriteContext | undefined,
|
|
): AwsSecretsManagerTag[] {
|
|
if (!context) return [];
|
|
return [
|
|
{ Key: "paperclip:managed-by", Value: "paperclip" },
|
|
{ Key: "paperclip:provider-owner", Value: config.providerOwnerTag },
|
|
{ Key: "paperclip:deployment-id", Value: config.deploymentId },
|
|
{ Key: "paperclip:company-id", Value: context.companyId },
|
|
{ Key: "paperclip:secret-key", Value: context.secretKey },
|
|
{ Key: "paperclip:environment", Value: config.environmentTag },
|
|
];
|
|
}
|
|
|
|
function createExternalReferenceMaterial(
|
|
externalRef: string,
|
|
providerVersionRef: string | null,
|
|
): PreparedSecretVersion {
|
|
const normalizedExternalRef = externalRef.trim();
|
|
const normalizedProviderVersionRef = providerVersionRef?.trim() || null;
|
|
const fingerprint = sha256Hex(
|
|
`${AWS_SECRETS_MANAGER_SCHEME}:${normalizedExternalRef}:${normalizedProviderVersionRef ?? ""}`,
|
|
);
|
|
return {
|
|
material: {
|
|
scheme: AWS_SECRETS_MANAGER_SCHEME,
|
|
secretId: normalizedExternalRef,
|
|
versionId: normalizedProviderVersionRef,
|
|
source: "external_reference",
|
|
},
|
|
valueSha256: fingerprint,
|
|
fingerprintSha256: fingerprint,
|
|
externalRef: normalizedExternalRef,
|
|
providerVersionRef: normalizedProviderVersionRef,
|
|
};
|
|
}
|
|
|
|
function createManagedMaterial(secretId: string, versionId: string | null): AwsSecretsManagerMaterial {
|
|
return {
|
|
scheme: AWS_SECRETS_MANAGER_SCHEME,
|
|
secretId,
|
|
versionId,
|
|
source: "managed",
|
|
};
|
|
}
|
|
|
|
function serializeAwsDate(value: string | number | Date | undefined): string | null {
|
|
if (value === undefined) return null;
|
|
const date = value instanceof Date ? value : new Date(value);
|
|
return Number.isNaN(date.getTime()) ? null : date.toISOString();
|
|
}
|
|
|
|
function createRemoteSecretMetadata(entry: AwsSecretsManagerListSecretEntry): Record<string, unknown> {
|
|
return {
|
|
createdDate: serializeAwsDate(entry.CreatedDate),
|
|
lastAccessedDate: serializeAwsDate(entry.LastAccessedDate),
|
|
lastChangedDate: serializeAwsDate(entry.LastChangedDate),
|
|
deletedDate: serializeAwsDate(entry.DeletedDate),
|
|
hasDescription: Boolean(entry.Description),
|
|
hasKmsKey: Boolean(entry.KmsKeyId),
|
|
tagCount: Array.isArray(entry.Tags) ? entry.Tags.length : 0,
|
|
};
|
|
}
|
|
|
|
function asAwsSecretsManagerMaterial(value: StoredSecretVersionMaterial): AwsSecretsManagerMaterial {
|
|
if (
|
|
value &&
|
|
typeof value === "object" &&
|
|
value.scheme === AWS_SECRETS_MANAGER_SCHEME &&
|
|
typeof value.secretId === "string" &&
|
|
(typeof value.versionId === "string" || value.versionId === null) &&
|
|
(value.source === "managed" || value.source === "external_reference")
|
|
) {
|
|
return value as AwsSecretsManagerMaterial;
|
|
}
|
|
throw unprocessable("Invalid AWS Secrets Manager material");
|
|
}
|
|
|
|
function classifyAwsProviderError(message: string): SecretProviderClientErrorCode {
|
|
if (/ResourceExistsException|AlreadyExists/i.test(message)) return "conflict";
|
|
if (/ResourceNotFoundException|NotFound/i.test(message)) return "not_found";
|
|
if (/AccessDeniedException|AccessDenied|UnrecognizedClientException|InvalidClientTokenId|not authorized/i.test(message)) {
|
|
return "access_denied";
|
|
}
|
|
if (/Throttl|TooManyRequests|RequestLimitExceeded|Rate exceeded/i.test(message)) return "throttled";
|
|
if (/ValidationException|InvalidParameter|InvalidRequest/i.test(message)) return "invalid_request";
|
|
if (/fetch failed|ECONN|ENOTFOUND|ETIMEDOUT|network|timeout/i.test(message)) return "provider_unavailable";
|
|
return "provider_error";
|
|
}
|
|
|
|
function awsProviderSafeMessage(code: SecretProviderClientErrorCode): string {
|
|
switch (code) {
|
|
case "access_denied":
|
|
return "AWS Secrets Manager denied the request. Check IAM permissions for this provider vault.";
|
|
case "throttled":
|
|
return "AWS Secrets Manager throttled the request. Wait and try again.";
|
|
case "not_found":
|
|
return "AWS Secrets Manager could not find the requested secret.";
|
|
case "conflict":
|
|
return "AWS Secrets Manager reported that the requested secret already exists.";
|
|
case "invalid_request":
|
|
return "AWS Secrets Manager rejected the request.";
|
|
case "provider_unavailable":
|
|
return "AWS Secrets Manager is unavailable right now.";
|
|
case "provider_error":
|
|
default:
|
|
return "AWS Secrets Manager request failed.";
|
|
}
|
|
}
|
|
|
|
function normalizeAwsError(operation: string, error: unknown): never {
|
|
const rawMessage = error instanceof Error ? error.message : String(error);
|
|
const code = classifyAwsProviderError(rawMessage);
|
|
throw new SecretProviderClientError({
|
|
code,
|
|
provider: "aws_secrets_manager",
|
|
operation,
|
|
message: awsProviderSafeMessage(code),
|
|
rawMessage,
|
|
cause: error,
|
|
});
|
|
}
|
|
|
|
class AwsSecretsManagerJsonGateway implements AwsSecretsManagerGateway {
|
|
private readonly endpoint: URL;
|
|
|
|
constructor(private readonly config: AwsSecretsManagerConfig) {
|
|
this.endpoint = new URL(config.endpoint);
|
|
}
|
|
|
|
createSecret(input: {
|
|
Name: string;
|
|
SecretString: string;
|
|
KmsKeyId?: string;
|
|
Description?: string;
|
|
Tags: AwsSecretsManagerTag[];
|
|
}) {
|
|
return this.call<{
|
|
ARN?: string;
|
|
Name?: string;
|
|
VersionId?: string;
|
|
}>("CreateSecret", input);
|
|
}
|
|
|
|
putSecretValue(input: {
|
|
SecretId: string;
|
|
SecretString: string;
|
|
VersionStages?: string[];
|
|
}) {
|
|
return this.call<{
|
|
ARN?: string;
|
|
Name?: string;
|
|
VersionId?: string;
|
|
}>("PutSecretValue", input);
|
|
}
|
|
|
|
getSecretValue(input: {
|
|
SecretId: string;
|
|
VersionId?: string;
|
|
VersionStage?: string;
|
|
}) {
|
|
return this.call<{
|
|
SecretString?: string;
|
|
ARN?: string;
|
|
Name?: string;
|
|
VersionId?: string;
|
|
}>("GetSecretValue", input);
|
|
}
|
|
|
|
deleteSecret(input: {
|
|
SecretId: string;
|
|
RecoveryWindowInDays: number;
|
|
}) {
|
|
return this.call("DeleteSecret", input);
|
|
}
|
|
|
|
updateSecretVersionStage(input: {
|
|
SecretId: string;
|
|
VersionStage: string;
|
|
RemoveFromVersionId?: string;
|
|
MoveToVersionId?: string;
|
|
}) {
|
|
return this.call("UpdateSecretVersionStage", input);
|
|
}
|
|
|
|
listSecrets(input: {
|
|
MaxResults?: number;
|
|
NextToken?: string;
|
|
Filters?: Array<{
|
|
Key: "all" | "name" | "description" | "tag-key" | "tag-value" | "primary-region" | "owning-service";
|
|
Values: string[];
|
|
}>;
|
|
IncludePlannedDeletion?: boolean;
|
|
}) {
|
|
return this.call<{
|
|
SecretList?: AwsSecretsManagerListSecretEntry[];
|
|
NextToken?: string;
|
|
}>("ListSecrets", input);
|
|
}
|
|
|
|
private async call<T>(operation: string, payload: Record<string, unknown>): Promise<T> {
|
|
const body = JSON.stringify(payload);
|
|
const credentials = await loadAwsCredentials(this.config.region);
|
|
const headers = signAwsSecretsManagerRequest({
|
|
endpoint: this.endpoint,
|
|
region: this.config.region,
|
|
operation,
|
|
body,
|
|
credentials,
|
|
});
|
|
const response = await fetch(this.endpoint, {
|
|
method: "POST",
|
|
headers,
|
|
body,
|
|
signal: AbortSignal.timeout(AWS_SECRETS_MANAGER_REQUEST_TIMEOUT_MS),
|
|
});
|
|
const text = await response.text();
|
|
const parsed = text ? (JSON.parse(text) as Record<string, unknown>) : {};
|
|
|
|
if (!response.ok) {
|
|
const code = String(parsed.__type ?? parsed.code ?? parsed.Code ?? response.statusText ?? "UnknownError");
|
|
const message = String(parsed.message ?? parsed.Message ?? code);
|
|
const rawMessage = `${code}: ${message}`;
|
|
const clientCode = classifyAwsProviderError(rawMessage);
|
|
throw new SecretProviderClientError({
|
|
code: clientCode,
|
|
provider: "aws_secrets_manager",
|
|
operation,
|
|
message: awsProviderSafeMessage(clientCode),
|
|
rawMessage,
|
|
});
|
|
}
|
|
|
|
return parsed as T;
|
|
}
|
|
}
|
|
|
|
export function createAwsSecretsManagerProvider(
|
|
options?: {
|
|
config?: AwsSecretsManagerConfig;
|
|
gateway?: AwsSecretsManagerGateway;
|
|
},
|
|
): SecretProviderModule {
|
|
function resolveConfig(providerConfig?: SecretProviderVaultRuntimeConfig | null) {
|
|
if (providerConfig) return readProviderVaultConfig(providerConfig);
|
|
return options?.config ?? loadAwsSecretsManagerConfig();
|
|
}
|
|
|
|
function resolveGateway(config: AwsSecretsManagerConfig) {
|
|
return options?.gateway ?? new AwsSecretsManagerJsonGateway(config);
|
|
}
|
|
|
|
async function validateConfig(
|
|
input?: {
|
|
deploymentMode?: DeploymentMode;
|
|
strictMode?: boolean;
|
|
providerConfig?: SecretProviderVaultRuntimeConfig | null;
|
|
},
|
|
): Promise<SecretProviderValidationResult> {
|
|
const warnings: string[] = [];
|
|
if (input?.deploymentMode === "authenticated" && input.strictMode !== true) {
|
|
warnings.push("Strict secret mode should be enabled for authenticated deployments");
|
|
}
|
|
const config = resolveConfig(input?.providerConfig);
|
|
if (!config.prefix) {
|
|
warnings.push("PAPERCLIP_SECRETS_AWS_PREFIX should be set to a deployment-scoped prefix");
|
|
}
|
|
return { ok: true, warnings };
|
|
}
|
|
|
|
async function healthCheck(
|
|
input?: {
|
|
deploymentMode?: DeploymentMode;
|
|
strictMode?: boolean;
|
|
providerConfig?: SecretProviderVaultRuntimeConfig | null;
|
|
},
|
|
): Promise<SecretProviderHealthCheck> {
|
|
try {
|
|
const validation = await validateConfig(input);
|
|
const config = resolveConfig(input?.providerConfig);
|
|
const readiness = getAwsConfigReadiness();
|
|
const warnings = [...validation.warnings];
|
|
if (
|
|
process.env.AWS_ACCESS_KEY_ID?.trim() &&
|
|
process.env.AWS_SECRET_ACCESS_KEY?.trim()
|
|
) {
|
|
warnings.push(
|
|
"AWS static environment credentials are visible to this process; use only short-lived shell credentials locally and prefer IAM role/workload identity for hosted deployments.",
|
|
);
|
|
}
|
|
return {
|
|
provider: "aws_secrets_manager",
|
|
status: warnings.length > 0 ? "warn" : "ok",
|
|
message:
|
|
"AWS Secrets Manager provider config is present; AWS credentials are resolved by the server runtime through the AWS SDK default credential provider chain.",
|
|
warnings,
|
|
details: {
|
|
region: config.region,
|
|
prefix: config.prefix,
|
|
deploymentId: config.deploymentId,
|
|
kmsKeyConfigured: Boolean(config.kmsKeyId),
|
|
credentialSource: "AWS SDK default credential provider chain",
|
|
detectedCredentialSources: readiness.credentialSources,
|
|
},
|
|
backupGuidance: [
|
|
"Back up Paperclip metadata separately from AWS-managed secrets.",
|
|
"Restoring access requires the Paperclip database plus the same AWS secret namespace and KMS permissions.",
|
|
],
|
|
};
|
|
} catch (error) {
|
|
const readiness = getAwsConfigReadiness();
|
|
const providerConfigMissing = input?.providerConfig && !asOptionalNonEmptyString(input.providerConfig.config.region)
|
|
? ["region"]
|
|
: [];
|
|
const missingConfig = input?.providerConfig ? providerConfigMissing : readiness.missingConfig;
|
|
return {
|
|
provider: "aws_secrets_manager",
|
|
status: "warn",
|
|
message:
|
|
missingConfig.length > 0
|
|
? `AWS Secrets Manager provider is not ready: missing ${missingConfig.join(", ")}.`
|
|
: error instanceof Error
|
|
? error.message
|
|
: String(error),
|
|
warnings: [
|
|
...(missingConfig.length > 0
|
|
? [`Missing required non-secret AWS provider config: ${missingConfig.join(", ")}.`]
|
|
: []),
|
|
AWS_RUNTIME_CREDENTIAL_WARNING,
|
|
AWS_CREDENTIAL_CUSTODY_WARNING,
|
|
"Managed secret create/rotate/resolve calls will fail until AWS provider configuration is complete.",
|
|
],
|
|
details: {
|
|
missingConfig,
|
|
requiredProviderConfig: input?.providerConfig
|
|
? ["region"]
|
|
: [
|
|
"PAPERCLIP_SECRETS_AWS_REGION or AWS_REGION/AWS_DEFAULT_REGION",
|
|
"PAPERCLIP_SECRETS_AWS_DEPLOYMENT_ID",
|
|
"PAPERCLIP_SECRETS_AWS_KMS_KEY_ID",
|
|
],
|
|
optionalProviderConfig: [
|
|
"PAPERCLIP_SECRETS_AWS_PREFIX",
|
|
"PAPERCLIP_SECRETS_AWS_ENVIRONMENT",
|
|
"PAPERCLIP_SECRETS_AWS_PROVIDER_OWNER",
|
|
"PAPERCLIP_SECRETS_AWS_ENDPOINT",
|
|
"PAPERCLIP_SECRETS_AWS_DELETE_RECOVERY_DAYS",
|
|
],
|
|
credentialSource: "AWS SDK default credential provider chain",
|
|
detectedCredentialSources: readiness.credentialSources,
|
|
},
|
|
};
|
|
}
|
|
}
|
|
|
|
return {
|
|
id: "aws_secrets_manager",
|
|
descriptor() {
|
|
return configuredAwsSecretsManagerDescriptor();
|
|
},
|
|
validateConfig,
|
|
async createSecret(input) {
|
|
const config = resolveConfig(input.providerConfig);
|
|
const gateway = resolveGateway(config);
|
|
const valueSha256 = sha256Hex(input.value);
|
|
const secretId = buildManagedSecretId(config, input.context);
|
|
|
|
try {
|
|
const createInput = {
|
|
Name: secretId,
|
|
SecretString: input.value,
|
|
...(config.kmsKeyId ? { KmsKeyId: config.kmsKeyId } : {}),
|
|
Description: input.context ? `Paperclip secret ${input.context.secretName}` : undefined,
|
|
Tags: buildManagedSecretTags(config, input.context),
|
|
};
|
|
const created = await gateway.createSecret({
|
|
...createInput,
|
|
});
|
|
const normalizedSecretId = created.ARN ?? created.Name ?? secretId;
|
|
return {
|
|
material: createManagedMaterial(normalizedSecretId, created.VersionId ?? null),
|
|
valueSha256,
|
|
fingerprintSha256: valueSha256,
|
|
externalRef: normalizedSecretId,
|
|
providerVersionRef: created.VersionId ?? null,
|
|
};
|
|
} catch (error) {
|
|
normalizeAwsError("createSecret", error);
|
|
}
|
|
},
|
|
async createVersion(input) {
|
|
const config = resolveConfig(input.providerConfig);
|
|
const gateway = resolveGateway(config);
|
|
const valueSha256 = sha256Hex(input.value);
|
|
const secretId = resolveManagedSecretRef({
|
|
config,
|
|
context: input.context,
|
|
externalRefs: [input.externalRef],
|
|
});
|
|
|
|
try {
|
|
const created = await gateway.putSecretValue({
|
|
SecretId: secretId,
|
|
SecretString: input.value,
|
|
VersionStages: [PAPERCLIP_PENDING_VERSION_STAGE],
|
|
});
|
|
const normalizedSecretId = created.ARN ?? created.Name ?? secretId;
|
|
return {
|
|
material: createManagedMaterial(normalizedSecretId, created.VersionId ?? null),
|
|
valueSha256,
|
|
fingerprintSha256: valueSha256,
|
|
externalRef: normalizedSecretId,
|
|
providerVersionRef: created.VersionId ?? null,
|
|
};
|
|
} catch (error) {
|
|
normalizeAwsError("createVersion", error);
|
|
}
|
|
},
|
|
async linkExternalSecret(input) {
|
|
const config = resolveConfig(input.providerConfig);
|
|
assertNotManagedNamespaceExternalRef(config, input.externalRef);
|
|
return createExternalReferenceMaterial(input.externalRef, input.providerVersionRef ?? null);
|
|
},
|
|
async listRemoteSecrets(input): Promise<RemoteSecretListResult> {
|
|
const config = resolveConfig(input.providerConfig);
|
|
const gateway = resolveGateway(config);
|
|
const query = input.query?.trim();
|
|
const pageSize =
|
|
input.pageSize && Number.isFinite(input.pageSize)
|
|
? Math.min(Math.max(Math.trunc(input.pageSize), 1), 100)
|
|
: 50;
|
|
|
|
try {
|
|
if (!gateway.listSecrets) {
|
|
throw new Error("ListSecrets gateway operation is unavailable");
|
|
}
|
|
const listed = await gateway.listSecrets({
|
|
MaxResults: pageSize,
|
|
NextToken: input.nextToken?.trim() || undefined,
|
|
IncludePlannedDeletion: false,
|
|
Filters: query ? [{ Key: "all", Values: [query] }] : undefined,
|
|
});
|
|
return {
|
|
nextToken: listed.NextToken ?? null,
|
|
secrets: (listed.SecretList ?? [])
|
|
.filter((entry) => Boolean(entry.ARN ?? entry.Name))
|
|
.map((entry) => ({
|
|
externalRef: entry.ARN ?? entry.Name ?? "",
|
|
name: entry.Name ?? entry.ARN ?? "",
|
|
providerVersionRef: null,
|
|
metadata: createRemoteSecretMetadata(entry),
|
|
})),
|
|
};
|
|
} catch (error) {
|
|
normalizeAwsError("listSecrets", error);
|
|
}
|
|
},
|
|
async resolveVersion(input) {
|
|
const config = resolveConfig(input.providerConfig);
|
|
const gateway = resolveGateway(config);
|
|
const material = asAwsSecretsManagerMaterial(input.material);
|
|
const secretId =
|
|
material.source === "managed"
|
|
? resolveManagedSecretRef({
|
|
config,
|
|
context: input.context,
|
|
externalRefs: [input.externalRef, material.secretId],
|
|
})
|
|
: (input.externalRef ?? material.secretId);
|
|
|
|
try {
|
|
const resolved = await gateway.getSecretValue({
|
|
SecretId: secretId,
|
|
VersionId: input.providerVersionRef ?? material.versionId ?? undefined,
|
|
VersionStage:
|
|
input.providerVersionRef || material.versionId ? undefined : DEFAULT_VERSION_STAGE,
|
|
});
|
|
if (typeof resolved.SecretString !== "string") {
|
|
throw new Error("SecretString was empty");
|
|
}
|
|
return resolved.SecretString;
|
|
} catch (error) {
|
|
normalizeAwsError("resolveVersion", error);
|
|
}
|
|
},
|
|
async deleteOrArchive(input) {
|
|
const material =
|
|
input.material && typeof input.material === "object"
|
|
? asAwsSecretsManagerMaterial(input.material)
|
|
: null;
|
|
|
|
if (material?.source !== "managed") return;
|
|
|
|
const config = resolveConfig(input.providerConfig);
|
|
const gateway = resolveGateway(config);
|
|
const secretId = resolveManagedSecretRef({
|
|
config,
|
|
context: input.context,
|
|
externalRefs: [input.externalRef, material.secretId],
|
|
});
|
|
|
|
try {
|
|
if (input.mode === "archive") {
|
|
if (material.versionId && gateway.updateSecretVersionStage) {
|
|
await gateway.updateSecretVersionStage({
|
|
SecretId: secretId,
|
|
VersionStage: PAPERCLIP_PENDING_VERSION_STAGE,
|
|
RemoveFromVersionId: material.versionId,
|
|
});
|
|
}
|
|
return;
|
|
}
|
|
await gateway.deleteSecret({
|
|
SecretId: secretId,
|
|
RecoveryWindowInDays: config.deleteRecoveryWindowDays,
|
|
});
|
|
} catch (error) {
|
|
normalizeAwsError(input.mode === "archive" ? "updateSecretVersionStage" : "deleteSecret", error);
|
|
}
|
|
},
|
|
healthCheck,
|
|
};
|
|
}
|
|
|
|
export const awsSecretsManagerProvider = createAwsSecretsManagerProvider();
|