feat(ai): vault_path secret resolver (env:/file:)

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
This commit is contained in:
root
2026-06-01 18:04:21 +10:00
parent 5f601c1a3c
commit d0d61575e3
2 changed files with 48 additions and 0 deletions

18
lib/ai/secret.js Normal file
View File

@@ -0,0 +1,18 @@
import { readFileSync } from 'node:fs';
// Resolve a vault_path-style secret reference. v1 supports:
// env:NAME -> process.env.NAME
// file:/path -> file contents (trimmed)
// <raw> -> returned as-is
// Vaultwarden item-id resolution is a future swap (see spec).
export function resolveSecret(spec) {
if (!spec) return null;
if (spec.startsWith('env:')) {
return process.env[spec.slice(4)] ?? null;
}
if (spec.startsWith('file:')) {
try { return readFileSync(spec.slice(5), 'utf8').trim(); }
catch { return null; }
}
return spec;
}

30
tests/ai/secret.test.js Normal file
View File

@@ -0,0 +1,30 @@
import { describe, it, expect, beforeEach } from 'vitest';
import { writeFileSync, mkdtempSync } from 'node:fs';
import { tmpdir } from 'node:os';
import { join } from 'node:path';
import { resolveSecret } from '../../lib/ai/secret.js';
describe('resolveSecret', () => {
beforeEach(() => { delete process.env.__SECRET_TEST; });
it('resolves env: specs', () => {
process.env.__SECRET_TEST = 'sk-from-env';
expect(resolveSecret('env:__SECRET_TEST')).toBe('sk-from-env');
});
it('resolves file: specs (trimmed)', () => {
const dir = mkdtempSync(join(tmpdir(), 'sec-'));
const f = join(dir, 'key');
writeFileSync(f, 'sk-from-file\n');
expect(resolveSecret('file:' + f)).toBe('sk-from-file');
});
it('returns a raw value unchanged', () => {
expect(resolveSecret('sk-raw')).toBe('sk-raw');
});
it('returns null for empty/missing', () => {
expect(resolveSecret('')).toBeNull();
expect(resolveSecret('env:__SECRET_TEST')).toBeNull();
});
});