latchkey 2.7.2 → 2.8.0
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/README.md +49 -5
- package/dist/scripts/cryptFile.js +2 -2
- package/dist/scripts/cryptFile.js.map +1 -1
- package/dist/scripts/recordBrowserSession.js +3 -2
- package/dist/scripts/recordBrowserSession.js.map +1 -1
- package/dist/src/cli.js +5 -4
- package/dist/src/cli.js.map +1 -1
- package/dist/src/cliCommands.d.ts.map +1 -1
- package/dist/src/cliCommands.js +44 -6
- package/dist/src/cliCommands.js.map +1 -1
- package/dist/src/config.d.ts +29 -0
- package/dist/src/config.d.ts.map +1 -1
- package/dist/src/config.js +45 -0
- package/dist/src/config.js.map +1 -1
- package/dist/src/encryptedStorage.d.ts +9 -25
- package/dist/src/encryptedStorage.d.ts.map +1 -1
- package/dist/src/encryptedStorage.js +9 -52
- package/dist/src/encryptedStorage.js.map +1 -1
- package/dist/src/encryption.d.ts +45 -0
- package/dist/src/encryption.d.ts.map +1 -1
- package/dist/src/encryption.js +69 -0
- package/dist/src/encryption.js.map +1 -1
- package/dist/src/gateway/client.d.ts +12 -2
- package/dist/src/gateway/client.d.ts.map +1 -1
- package/dist/src/gateway/client.js +31 -4
- package/dist/src/gateway/client.js.map +1 -1
- package/dist/src/gateway/gatewayEndpoint.d.ts +11 -0
- package/dist/src/gateway/gatewayEndpoint.d.ts.map +1 -1
- package/dist/src/gateway/gatewayEndpoint.js +40 -4
- package/dist/src/gateway/gatewayEndpoint.js.map +1 -1
- package/dist/src/gateway/password.d.ts +16 -0
- package/dist/src/gateway/password.d.ts.map +1 -0
- package/dist/src/gateway/password.js +24 -0
- package/dist/src/gateway/password.js.map +1 -0
- package/dist/src/gateway/permissionPointer.d.ts +56 -0
- package/dist/src/gateway/permissionPointer.d.ts.map +1 -0
- package/dist/src/gateway/permissionPointer.js +171 -0
- package/dist/src/gateway/permissionPointer.js.map +1 -0
- package/dist/src/gateway/permissionsOverride.d.ts +56 -0
- package/dist/src/gateway/permissionsOverride.d.ts.map +1 -0
- package/dist/src/gateway/permissionsOverride.js +157 -0
- package/dist/src/gateway/permissionsOverride.js.map +1 -0
- package/dist/src/gateway/server.d.ts.map +1 -1
- package/dist/src/gateway/server.js +34 -1
- package/dist/src/gateway/server.js.map +1 -1
- package/dist/src/index.d.ts +2 -2
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +2 -2
- package/dist/src/index.js.map +1 -1
- package/dist/src/oauthUtils.d.ts +11 -2
- package/dist/src/oauthUtils.d.ts.map +1 -1
- package/dist/src/oauthUtils.js +25 -4
- package/dist/src/oauthUtils.js.map +1 -1
- package/dist/src/serviceRegistry.d.ts.map +1 -1
- package/dist/src/serviceRegistry.js +2 -1
- package/dist/src/serviceRegistry.js.map +1 -1
- package/dist/src/services/index.d.ts +1 -0
- package/dist/src/services/index.d.ts.map +1 -1
- package/dist/src/services/index.js +1 -0
- package/dist/src/services/index.js.map +1 -1
- package/dist/src/services/notion-mcp.d.ts +29 -0
- package/dist/src/services/notion-mcp.d.ts.map +1 -0
- package/dist/src/services/notion-mcp.js +156 -0
- package/dist/src/services/notion-mcp.js.map +1 -0
- package/dist/src/services/notion.d.ts +1 -1
- package/dist/src/services/notion.d.ts.map +1 -1
- package/dist/src/services/notion.js +5 -4
- package/dist/src/services/notion.js.map +1 -1
- package/dist/src/version.d.ts +1 -1
- package/dist/src/version.js +1 -1
- package/dist/tests/apiCredentialStore.test.js +2 -2
- package/dist/tests/apiCredentialStore.test.js.map +1 -1
- package/dist/tests/cli.test.js +95 -53
- package/dist/tests/cli.test.js.map +1 -1
- package/dist/tests/config.test.js +37 -0
- package/dist/tests/config.test.js.map +1 -1
- package/dist/tests/encryptedStorage.test.js +19 -39
- package/dist/tests/encryptedStorage.test.js.map +1 -1
- package/dist/tests/encryptedStorageKeyGeneration.test.js +2 -1
- package/dist/tests/encryptedStorageKeyGeneration.test.js.map +1 -1
- package/dist/tests/gateway.test.js +170 -7
- package/dist/tests/gateway.test.js.map +1 -1
- package/dist/tests/gatewayClient.test.js +74 -0
- package/dist/tests/gatewayClient.test.js.map +1 -1
- package/dist/tests/latchkeyEndpoint.test.js +7 -6
- package/dist/tests/latchkeyEndpoint.test.js.map +1 -1
- package/dist/tests/migrations.test.js +2 -2
- package/dist/tests/migrations.test.js.map +1 -1
- package/dist/tests/oauthUtils.test.d.ts +2 -0
- package/dist/tests/oauthUtils.test.d.ts.map +1 -0
- package/dist/tests/oauthUtils.test.js +63 -0
- package/dist/tests/oauthUtils.test.js.map +1 -0
- package/dist/tests/permissionPointer.test.d.ts +2 -0
- package/dist/tests/permissionPointer.test.d.ts.map +1 -0
- package/dist/tests/permissionPointer.test.js +152 -0
- package/dist/tests/permissionPointer.test.js.map +1 -0
- package/dist/tests/permissionsOverride.test.d.ts +2 -0
- package/dist/tests/permissionsOverride.test.d.ts.map +1 -0
- package/dist/tests/permissionsOverride.test.js +136 -0
- package/dist/tests/permissionsOverride.test.js.map +1 -0
- package/dist/tests/resolveEncryptionKey.test.d.ts +2 -0
- package/dist/tests/resolveEncryptionKey.test.d.ts.map +1 -0
- package/dist/tests/resolveEncryptionKey.test.js +26 -0
- package/dist/tests/resolveEncryptionKey.test.js.map +1 -0
- package/dist/tests/sharedOperations.test.js +34 -50
- package/dist/tests/sharedOperations.test.js.map +1 -1
- package/package.json +2 -2
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"oauthUtils.test.d.ts","sourceRoot":"","sources":["../../tests/oauthUtils.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import { describe, it, expect, vi, afterEach } from 'vitest';
|
|
2
|
+
import { startOAuthCallbackServer, generateCodeVerifier, generateCodeChallenge, exchangeCodeForTokens, refreshAccessToken, OAuthTokenExchangeError, OAuthCallbackServerTimeoutError, } from '../src/oauthUtils.js';
|
|
3
|
+
import * as curl from '../src/curl.js';
|
|
4
|
+
afterEach(() => {
|
|
5
|
+
vi.restoreAllMocks();
|
|
6
|
+
});
|
|
7
|
+
void startOAuthCallbackServer;
|
|
8
|
+
void OAuthTokenExchangeError;
|
|
9
|
+
void OAuthCallbackServerTimeoutError;
|
|
10
|
+
describe('startOAuthCallbackServer', () => {
|
|
11
|
+
it.todo('add tests');
|
|
12
|
+
});
|
|
13
|
+
describe('generateCodeVerifier', () => {
|
|
14
|
+
it('just runs', () => {
|
|
15
|
+
generateCodeVerifier();
|
|
16
|
+
});
|
|
17
|
+
});
|
|
18
|
+
describe('generateCodeChallenge', () => {
|
|
19
|
+
it('just runs', () => {
|
|
20
|
+
generateCodeChallenge('dBjftJeZ4CVP-mB92K27uhbUJU1p1r_wW1gFWFOEjXk');
|
|
21
|
+
});
|
|
22
|
+
});
|
|
23
|
+
describe('exchangeCodeForTokens', () => {
|
|
24
|
+
it('with PKCE: includes code_verifier in body', () => {
|
|
25
|
+
const spy = vi.spyOn(curl, 'runCaptured').mockImplementation(() => {
|
|
26
|
+
throw new Error('STOP');
|
|
27
|
+
});
|
|
28
|
+
expect(() => exchangeCodeForTokens('https://api.notion.com/v1/oauth/token', 'auth-code-abc123', 'test-client-id', 'test-client-secret', 'http://localhost:12345/oauth2callback', 'dBjftJeZ4CVP-mB92K27uhbUJU1p1r_wW1gFWFOEjXk')).toThrow('STOP');
|
|
29
|
+
const args = spy.mock.calls[0][0];
|
|
30
|
+
const body = args[args.indexOf('-d') + 1];
|
|
31
|
+
expect(body).toBe('code=auth-code-abc123&client_id=test-client-id&redirect_uri=http%3A%2F%2Flocalhost%3A12345%2Foauth2callback&grant_type=authorization_code&client_secret=test-client-secret&code_verifier=dBjftJeZ4CVP-mB92K27uhbUJU1p1r_wW1gFWFOEjXk');
|
|
32
|
+
});
|
|
33
|
+
it('without PKCE: omits code_verifier from body', () => {
|
|
34
|
+
const spy = vi.spyOn(curl, 'runCaptured').mockImplementation(() => {
|
|
35
|
+
throw new Error('STOP');
|
|
36
|
+
});
|
|
37
|
+
expect(() => exchangeCodeForTokens('https://api.notion.com/v1/oauth/token', 'auth-code-abc123', 'test-client-id', 'test-client-secret', 'http://localhost:12345/oauth2callback')).toThrow('STOP');
|
|
38
|
+
const args = spy.mock.calls[0][0];
|
|
39
|
+
const body = args[args.indexOf('-d') + 1];
|
|
40
|
+
expect(body).toBe('code=auth-code-abc123&client_id=test-client-id&redirect_uri=http%3A%2F%2Flocalhost%3A12345%2Foauth2callback&grant_type=authorization_code&client_secret=test-client-secret');
|
|
41
|
+
});
|
|
42
|
+
});
|
|
43
|
+
describe('refreshAccessToken', () => {
|
|
44
|
+
it('confidential client: includes client_secret in body', () => {
|
|
45
|
+
const spy = vi.spyOn(curl, 'runCaptured').mockImplementation(() => {
|
|
46
|
+
throw new Error('STOP');
|
|
47
|
+
});
|
|
48
|
+
expect(() => refreshAccessToken('https://api.notion.com/v1/oauth/token', 'refresh-token-xyz789', 'test-client-id', 'test-client-secret')).toThrow('STOP');
|
|
49
|
+
const args = spy.mock.calls[0][0];
|
|
50
|
+
const body = args[args.indexOf('-d') + 1];
|
|
51
|
+
expect(body).toBe('refresh_token=refresh-token-xyz789&client_id=test-client-id&grant_type=refresh_token&client_secret=test-client-secret');
|
|
52
|
+
});
|
|
53
|
+
it('public client: omits client_secret from body', () => {
|
|
54
|
+
const spy = vi.spyOn(curl, 'runCaptured').mockImplementation(() => {
|
|
55
|
+
throw new Error('STOP');
|
|
56
|
+
});
|
|
57
|
+
expect(() => refreshAccessToken('https://api.notion.com/v1/oauth/token', 'refresh-token-xyz789', 'test-client-id', '')).toThrow('STOP');
|
|
58
|
+
const args = spy.mock.calls[0][0];
|
|
59
|
+
const body = args[args.indexOf('-d') + 1];
|
|
60
|
+
expect(body).toBe('refresh_token=refresh-token-xyz789&client_id=test-client-id&grant_type=refresh_token');
|
|
61
|
+
});
|
|
62
|
+
});
|
|
63
|
+
//# sourceMappingURL=oauthUtils.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"oauthUtils.test.js","sourceRoot":"","sources":["../../tests/oauthUtils.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"permissionPointer.test.d.ts","sourceRoot":"","sources":["../../tests/permissionPointer.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { createHmac } from 'node:crypto';
|
|
3
|
+
import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from 'node:fs';
|
|
4
|
+
import { join } from 'node:path';
|
|
5
|
+
import { tmpdir } from 'node:os';
|
|
6
|
+
import { createPermissionPointerJwt, derivePermissionPointerSigningKey, InvalidPermissionPointerError, PERMISSION_POINTER_HEADER, PermissionPointerFileMissingError, resolvePermissionPointer, verifyPermissionPointerJwt, } from '../src/gateway/permissionPointer.js';
|
|
7
|
+
const TEST_ENCRYPTION_KEY = 'dGVzdGtleXRlc3RrZXl0ZXN0a2V5dGVzdGtleXRlc3Q=';
|
|
8
|
+
const OTHER_ENCRYPTION_KEY = 'b3RoZXJrZXlvdGhlcmtleW90aGVya2V5b3RoZXJrZXk=';
|
|
9
|
+
describe('PERMISSION_POINTER_HEADER', () => {
|
|
10
|
+
it('is the lowercase header name', () => {
|
|
11
|
+
expect(PERMISSION_POINTER_HEADER).toBe('x-latchkey-gateway-permission-pointer');
|
|
12
|
+
});
|
|
13
|
+
});
|
|
14
|
+
describe('derivePermissionPointerSigningKey', () => {
|
|
15
|
+
it('produces a deterministic 32-byte key', () => {
|
|
16
|
+
const a = derivePermissionPointerSigningKey(TEST_ENCRYPTION_KEY);
|
|
17
|
+
const b = derivePermissionPointerSigningKey(TEST_ENCRYPTION_KEY);
|
|
18
|
+
expect(a.length).toBe(32);
|
|
19
|
+
expect(a.equals(b)).toBe(true);
|
|
20
|
+
});
|
|
21
|
+
it('produces different keys for different master keys', () => {
|
|
22
|
+
const a = derivePermissionPointerSigningKey(TEST_ENCRYPTION_KEY);
|
|
23
|
+
const b = derivePermissionPointerSigningKey(OTHER_ENCRYPTION_KEY);
|
|
24
|
+
expect(a.equals(b)).toBe(false);
|
|
25
|
+
});
|
|
26
|
+
it('does not equal the encryption key itself', () => {
|
|
27
|
+
const derived = derivePermissionPointerSigningKey(TEST_ENCRYPTION_KEY);
|
|
28
|
+
const master = Buffer.from(TEST_ENCRYPTION_KEY, 'base64');
|
|
29
|
+
expect(derived.equals(master)).toBe(false);
|
|
30
|
+
});
|
|
31
|
+
});
|
|
32
|
+
describe('createPermissionPointerJwt / verifyPermissionPointerJwt', () => {
|
|
33
|
+
const signingKey = derivePermissionPointerSigningKey(TEST_ENCRYPTION_KEY);
|
|
34
|
+
it('round-trips an absolute path', () => {
|
|
35
|
+
const token = createPermissionPointerJwt('/etc/latchkey/permissions.json', signingKey);
|
|
36
|
+
const payload = verifyPermissionPointerJwt(token, signingKey);
|
|
37
|
+
expect(payload).toEqual({ permissionsConfig: '/etc/latchkey/permissions.json' });
|
|
38
|
+
});
|
|
39
|
+
it('produces a three-segment JWT', () => {
|
|
40
|
+
const token = createPermissionPointerJwt('/x.json', signingKey);
|
|
41
|
+
expect(token.split('.')).toHaveLength(3);
|
|
42
|
+
});
|
|
43
|
+
it('uses HS256/JWT in the header', () => {
|
|
44
|
+
const token = createPermissionPointerJwt('/x.json', signingKey);
|
|
45
|
+
const header = token.split('.')[0];
|
|
46
|
+
const json = Buffer.from(header.replace(/-/g, '+').replace(/_/g, '/'), 'base64').toString('utf-8');
|
|
47
|
+
expect(JSON.parse(json)).toEqual({ alg: 'HS256', typ: 'JWT' });
|
|
48
|
+
});
|
|
49
|
+
it('payload contains only the permissionsConfig field', () => {
|
|
50
|
+
const token = createPermissionPointerJwt('/x.json', signingKey);
|
|
51
|
+
const payload = token.split('.')[1];
|
|
52
|
+
const json = Buffer.from(payload.replace(/-/g, '+').replace(/_/g, '/'), 'base64').toString('utf-8');
|
|
53
|
+
expect(JSON.parse(json)).toEqual({ permissionsConfig: '/x.json' });
|
|
54
|
+
});
|
|
55
|
+
it('rejects creation for non-absolute paths', () => {
|
|
56
|
+
expect(() => createPermissionPointerJwt('relative/path.json', signingKey)).toThrow(InvalidPermissionPointerError);
|
|
57
|
+
});
|
|
58
|
+
it('rejects tokens signed with a different key', () => {
|
|
59
|
+
const otherKey = derivePermissionPointerSigningKey(OTHER_ENCRYPTION_KEY);
|
|
60
|
+
const token = createPermissionPointerJwt('/x.json', otherKey);
|
|
61
|
+
expect(() => verifyPermissionPointerJwt(token, signingKey)).toThrow(InvalidPermissionPointerError);
|
|
62
|
+
});
|
|
63
|
+
it('rejects tokens with a tampered payload', () => {
|
|
64
|
+
const token = createPermissionPointerJwt('/x.json', signingKey);
|
|
65
|
+
const [header, , signature] = token.split('.');
|
|
66
|
+
const tamperedPayload = Buffer.from(JSON.stringify({ permissionsConfig: '/y.json' }), 'utf-8')
|
|
67
|
+
.toString('base64')
|
|
68
|
+
.replace(/\+/g, '-')
|
|
69
|
+
.replace(/\//g, '_')
|
|
70
|
+
.replace(/=+$/, '');
|
|
71
|
+
const tampered = `${header}.${tamperedPayload}.${signature}`;
|
|
72
|
+
expect(() => verifyPermissionPointerJwt(tampered, signingKey)).toThrow(InvalidPermissionPointerError);
|
|
73
|
+
});
|
|
74
|
+
it('rejects tokens that do not have three segments', () => {
|
|
75
|
+
expect(() => verifyPermissionPointerJwt('a.b', signingKey)).toThrow(InvalidPermissionPointerError);
|
|
76
|
+
expect(() => verifyPermissionPointerJwt('a.b.c.d', signingKey)).toThrow(InvalidPermissionPointerError);
|
|
77
|
+
});
|
|
78
|
+
function base64Url(value) {
|
|
79
|
+
return Buffer.from(value, 'utf-8')
|
|
80
|
+
.toString('base64')
|
|
81
|
+
.replace(/\+/g, '-')
|
|
82
|
+
.replace(/\//g, '_')
|
|
83
|
+
.replace(/=+$/, '');
|
|
84
|
+
}
|
|
85
|
+
function buildSignedToken(headerJson, payloadJson) {
|
|
86
|
+
const headerSegment = base64Url(headerJson);
|
|
87
|
+
const payloadSegment = base64Url(payloadJson);
|
|
88
|
+
const signature = createHmac('sha256', signingKey)
|
|
89
|
+
.update(`${headerSegment}.${payloadSegment}`)
|
|
90
|
+
.digest()
|
|
91
|
+
.toString('base64')
|
|
92
|
+
.replace(/\+/g, '-')
|
|
93
|
+
.replace(/\//g, '_')
|
|
94
|
+
.replace(/=+$/, '');
|
|
95
|
+
return `${headerSegment}.${payloadSegment}.${signature}`;
|
|
96
|
+
}
|
|
97
|
+
it('rejects tokens whose payload is not valid JSON', () => {
|
|
98
|
+
// Sign a payload that is base64url of "not-json" so we hit the JSON parse
|
|
99
|
+
// error rather than the signature mismatch error first.
|
|
100
|
+
const headerSegment = base64Url(JSON.stringify({ alg: 'HS256', typ: 'JWT' }));
|
|
101
|
+
const payloadSegment = base64Url('not-json');
|
|
102
|
+
const signature = createHmac('sha256', signingKey)
|
|
103
|
+
.update(`${headerSegment}.${payloadSegment}`)
|
|
104
|
+
.digest()
|
|
105
|
+
.toString('base64')
|
|
106
|
+
.replace(/\+/g, '-')
|
|
107
|
+
.replace(/\//g, '_')
|
|
108
|
+
.replace(/=+$/, '');
|
|
109
|
+
const token = `${headerSegment}.${payloadSegment}.${signature}`;
|
|
110
|
+
expect(() => verifyPermissionPointerJwt(token, signingKey)).toThrow(InvalidPermissionPointerError);
|
|
111
|
+
});
|
|
112
|
+
it('rejects payloads without permissionsConfig', () => {
|
|
113
|
+
const token = buildSignedToken(JSON.stringify({ alg: 'HS256', typ: 'JWT' }), JSON.stringify({ other: '/x.json' }));
|
|
114
|
+
expect(() => verifyPermissionPointerJwt(token, signingKey)).toThrow(InvalidPermissionPointerError);
|
|
115
|
+
});
|
|
116
|
+
it('rejects payloads whose permissionsConfig is not absolute', () => {
|
|
117
|
+
const token = buildSignedToken(JSON.stringify({ alg: 'HS256', typ: 'JWT' }), JSON.stringify({ permissionsConfig: 'relative.json' }));
|
|
118
|
+
expect(() => verifyPermissionPointerJwt(token, signingKey)).toThrow(InvalidPermissionPointerError);
|
|
119
|
+
});
|
|
120
|
+
it('rejects headers with the wrong algorithm', () => {
|
|
121
|
+
const token = buildSignedToken(JSON.stringify({ alg: 'none', typ: 'JWT' }), JSON.stringify({ permissionsConfig: '/x.json' }));
|
|
122
|
+
expect(() => verifyPermissionPointerJwt(token, signingKey)).toThrow(InvalidPermissionPointerError);
|
|
123
|
+
});
|
|
124
|
+
});
|
|
125
|
+
describe('resolvePermissionPointer', () => {
|
|
126
|
+
const signingKey = derivePermissionPointerSigningKey(TEST_ENCRYPTION_KEY);
|
|
127
|
+
let tempDir;
|
|
128
|
+
beforeEach(() => {
|
|
129
|
+
tempDir = mkdtempSync(join(tmpdir(), 'latchkey-pp-test-'));
|
|
130
|
+
});
|
|
131
|
+
afterEach(() => {
|
|
132
|
+
rmSync(tempDir, { recursive: true, force: true });
|
|
133
|
+
});
|
|
134
|
+
it('returns the path when the file exists', () => {
|
|
135
|
+
const path = join(tempDir, 'permissions.json');
|
|
136
|
+
writeFileSync(path, '{}');
|
|
137
|
+
const token = createPermissionPointerJwt(path, signingKey);
|
|
138
|
+
expect(resolvePermissionPointer(token, signingKey)).toBe(path);
|
|
139
|
+
});
|
|
140
|
+
it('throws PermissionPointerFileMissingError when the file is absent', () => {
|
|
141
|
+
const path = join(tempDir, 'does-not-exist.json');
|
|
142
|
+
const token = createPermissionPointerJwt(path, signingKey);
|
|
143
|
+
expect(() => resolvePermissionPointer(token, signingKey)).toThrow(PermissionPointerFileMissingError);
|
|
144
|
+
});
|
|
145
|
+
it('throws PermissionPointerFileMissingError when the path is a directory', () => {
|
|
146
|
+
const path = join(tempDir, 'subdir');
|
|
147
|
+
mkdirSync(path);
|
|
148
|
+
const token = createPermissionPointerJwt(path, signingKey);
|
|
149
|
+
expect(() => resolvePermissionPointer(token, signingKey)).toThrow(PermissionPointerFileMissingError);
|
|
150
|
+
});
|
|
151
|
+
});
|
|
152
|
+
//# sourceMappingURL=permissionPointer.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"permissionPointer.test.js","sourceRoot":"","sources":["../../tests/permissionPointer.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"permissionsOverride.test.d.ts","sourceRoot":"","sources":["../../tests/permissionsOverride.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { createHmac } from 'node:crypto';
|
|
3
|
+
import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from 'node:fs';
|
|
4
|
+
import { join } from 'node:path';
|
|
5
|
+
import { tmpdir } from 'node:os';
|
|
6
|
+
import { createPermissionsOverrideJwt, derivePermissionsOverrideSigningKey, InvalidPermissionsOverrideError, PERMISSIONS_OVERRIDE_HEADER, PermissionsOverrideFileMissingError, resolvePermissionsOverride, verifyPermissionsOverrideJwt, } from '../src/gateway/permissionsOverride.js';
|
|
7
|
+
const TEST_ENCRYPTION_KEY = 'dGVzdGtleXRlc3RrZXl0ZXN0a2V5dGVzdGtleXRlc3Q=';
|
|
8
|
+
const OTHER_ENCRYPTION_KEY = 'b3RoZXJrZXlvdGhlcmtleW90aGVya2V5b3RoZXJrZXk=';
|
|
9
|
+
describe('PERMISSIONS_OVERRIDE_HEADER', () => {
|
|
10
|
+
it('is the lowercase header name', () => {
|
|
11
|
+
expect(PERMISSIONS_OVERRIDE_HEADER).toBe('x-latchkey-gateway-permissions-override');
|
|
12
|
+
});
|
|
13
|
+
});
|
|
14
|
+
describe('derivePermissionsOverrideSigningKey', () => {
|
|
15
|
+
it('produces a deterministic 32-byte key', () => {
|
|
16
|
+
const a = derivePermissionsOverrideSigningKey(TEST_ENCRYPTION_KEY);
|
|
17
|
+
const b = derivePermissionsOverrideSigningKey(TEST_ENCRYPTION_KEY);
|
|
18
|
+
expect(a.length).toBe(32);
|
|
19
|
+
expect(a.equals(b)).toBe(true);
|
|
20
|
+
});
|
|
21
|
+
it('produces different keys for different master keys', () => {
|
|
22
|
+
const a = derivePermissionsOverrideSigningKey(TEST_ENCRYPTION_KEY);
|
|
23
|
+
const b = derivePermissionsOverrideSigningKey(OTHER_ENCRYPTION_KEY);
|
|
24
|
+
expect(a.equals(b)).toBe(false);
|
|
25
|
+
});
|
|
26
|
+
it('does not equal the encryption key itself', () => {
|
|
27
|
+
const derived = derivePermissionsOverrideSigningKey(TEST_ENCRYPTION_KEY);
|
|
28
|
+
const master = Buffer.from(TEST_ENCRYPTION_KEY, 'base64');
|
|
29
|
+
expect(derived.equals(master)).toBe(false);
|
|
30
|
+
});
|
|
31
|
+
});
|
|
32
|
+
describe('createPermissionsOverrideJwt / verifyPermissionsOverrideJwt', () => {
|
|
33
|
+
const signingKey = derivePermissionsOverrideSigningKey(TEST_ENCRYPTION_KEY);
|
|
34
|
+
it('round-trips an absolute path', () => {
|
|
35
|
+
const token = createPermissionsOverrideJwt('/etc/latchkey/permissions.json', signingKey);
|
|
36
|
+
const payload = verifyPermissionsOverrideJwt(token, signingKey);
|
|
37
|
+
expect(payload).toEqual({ permissionsConfig: '/etc/latchkey/permissions.json' });
|
|
38
|
+
});
|
|
39
|
+
it('produces a three-segment JWT', () => {
|
|
40
|
+
const token = createPermissionsOverrideJwt('/x.json', signingKey);
|
|
41
|
+
expect(token.split('.')).toHaveLength(3);
|
|
42
|
+
});
|
|
43
|
+
it('uses HS256/JWT in the header', () => {
|
|
44
|
+
const token = createPermissionsOverrideJwt('/x.json', signingKey);
|
|
45
|
+
const header = token.split('.')[0];
|
|
46
|
+
const json = Buffer.from(header, 'base64url').toString('utf-8');
|
|
47
|
+
expect(JSON.parse(json)).toEqual({ alg: 'HS256', typ: 'JWT' });
|
|
48
|
+
});
|
|
49
|
+
it('payload contains only the permissionsConfig field', () => {
|
|
50
|
+
const token = createPermissionsOverrideJwt('/x.json', signingKey);
|
|
51
|
+
const payload = token.split('.')[1];
|
|
52
|
+
const json = Buffer.from(payload, 'base64url').toString('utf-8');
|
|
53
|
+
expect(JSON.parse(json)).toEqual({ permissionsConfig: '/x.json' });
|
|
54
|
+
});
|
|
55
|
+
it('rejects creation for non-absolute paths', () => {
|
|
56
|
+
expect(() => createPermissionsOverrideJwt('relative/path.json', signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
57
|
+
});
|
|
58
|
+
it('rejects tokens signed with a different key', () => {
|
|
59
|
+
const otherKey = derivePermissionsOverrideSigningKey(OTHER_ENCRYPTION_KEY);
|
|
60
|
+
const token = createPermissionsOverrideJwt('/x.json', otherKey);
|
|
61
|
+
expect(() => verifyPermissionsOverrideJwt(token, signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
62
|
+
});
|
|
63
|
+
it('rejects tokens with a tampered payload', () => {
|
|
64
|
+
const token = createPermissionsOverrideJwt('/x.json', signingKey);
|
|
65
|
+
const [header, , signature] = token.split('.');
|
|
66
|
+
const tamperedPayload = Buffer.from(JSON.stringify({ permissionsConfig: '/y.json' }), 'utf-8').toString('base64url');
|
|
67
|
+
const tampered = `${header}.${tamperedPayload}.${signature}`;
|
|
68
|
+
expect(() => verifyPermissionsOverrideJwt(tampered, signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
69
|
+
});
|
|
70
|
+
it('rejects tokens that do not have three segments', () => {
|
|
71
|
+
expect(() => verifyPermissionsOverrideJwt('a.b', signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
72
|
+
expect(() => verifyPermissionsOverrideJwt('a.b.c.d', signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
73
|
+
});
|
|
74
|
+
function base64Url(value) {
|
|
75
|
+
return Buffer.from(value, 'utf-8').toString('base64url');
|
|
76
|
+
}
|
|
77
|
+
function buildSignedToken(headerJson, payloadJson) {
|
|
78
|
+
const headerSegment = base64Url(headerJson);
|
|
79
|
+
const payloadSegment = base64Url(payloadJson);
|
|
80
|
+
const signature = createHmac('sha256', signingKey)
|
|
81
|
+
.update(`${headerSegment}.${payloadSegment}`)
|
|
82
|
+
.digest('base64url');
|
|
83
|
+
return `${headerSegment}.${payloadSegment}.${signature}`;
|
|
84
|
+
}
|
|
85
|
+
it('rejects tokens whose payload is not valid JSON', () => {
|
|
86
|
+
// Sign a payload that is base64url of "not-json" so we hit the JSON parse
|
|
87
|
+
// error rather than the signature mismatch error first.
|
|
88
|
+
const headerSegment = base64Url(JSON.stringify({ alg: 'HS256', typ: 'JWT' }));
|
|
89
|
+
const payloadSegment = base64Url('not-json');
|
|
90
|
+
const signature = createHmac('sha256', signingKey)
|
|
91
|
+
.update(`${headerSegment}.${payloadSegment}`)
|
|
92
|
+
.digest('base64url');
|
|
93
|
+
const token = `${headerSegment}.${payloadSegment}.${signature}`;
|
|
94
|
+
expect(() => verifyPermissionsOverrideJwt(token, signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
95
|
+
});
|
|
96
|
+
it('rejects payloads without permissionsConfig', () => {
|
|
97
|
+
const token = buildSignedToken(JSON.stringify({ alg: 'HS256', typ: 'JWT' }), JSON.stringify({ other: '/x.json' }));
|
|
98
|
+
expect(() => verifyPermissionsOverrideJwt(token, signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
99
|
+
});
|
|
100
|
+
it('rejects payloads whose permissionsConfig is not absolute', () => {
|
|
101
|
+
const token = buildSignedToken(JSON.stringify({ alg: 'HS256', typ: 'JWT' }), JSON.stringify({ permissionsConfig: 'relative.json' }));
|
|
102
|
+
expect(() => verifyPermissionsOverrideJwt(token, signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
103
|
+
});
|
|
104
|
+
it('rejects headers with the wrong algorithm', () => {
|
|
105
|
+
const token = buildSignedToken(JSON.stringify({ alg: 'none', typ: 'JWT' }), JSON.stringify({ permissionsConfig: '/x.json' }));
|
|
106
|
+
expect(() => verifyPermissionsOverrideJwt(token, signingKey)).toThrow(InvalidPermissionsOverrideError);
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
describe('resolvePermissionsOverride', () => {
|
|
110
|
+
const signingKey = derivePermissionsOverrideSigningKey(TEST_ENCRYPTION_KEY);
|
|
111
|
+
let tempDir;
|
|
112
|
+
beforeEach(() => {
|
|
113
|
+
tempDir = mkdtempSync(join(tmpdir(), 'latchkey-pp-test-'));
|
|
114
|
+
});
|
|
115
|
+
afterEach(() => {
|
|
116
|
+
rmSync(tempDir, { recursive: true, force: true });
|
|
117
|
+
});
|
|
118
|
+
it('returns the path when the file exists', () => {
|
|
119
|
+
const path = join(tempDir, 'permissions.json');
|
|
120
|
+
writeFileSync(path, '{}');
|
|
121
|
+
const token = createPermissionsOverrideJwt(path, signingKey);
|
|
122
|
+
expect(resolvePermissionsOverride(token, signingKey)).toBe(path);
|
|
123
|
+
});
|
|
124
|
+
it('throws PermissionsOverrideFileMissingError when the file is absent', () => {
|
|
125
|
+
const path = join(tempDir, 'does-not-exist.json');
|
|
126
|
+
const token = createPermissionsOverrideJwt(path, signingKey);
|
|
127
|
+
expect(() => resolvePermissionsOverride(token, signingKey)).toThrow(PermissionsOverrideFileMissingError);
|
|
128
|
+
});
|
|
129
|
+
it('throws PermissionsOverrideFileMissingError when the path is a directory', () => {
|
|
130
|
+
const path = join(tempDir, 'subdir');
|
|
131
|
+
mkdirSync(path);
|
|
132
|
+
const token = createPermissionsOverrideJwt(path, signingKey);
|
|
133
|
+
expect(() => resolvePermissionsOverride(token, signingKey)).toThrow(PermissionsOverrideFileMissingError);
|
|
134
|
+
});
|
|
135
|
+
});
|
|
136
|
+
//# sourceMappingURL=permissionsOverride.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"permissionsOverride.test.js","sourceRoot":"","sources":["../../tests/permissionsOverride.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACrE,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AACzC,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AACxE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EACL,4BAA4B,EAC5B,mCAAmC,EACnC,+BAA+B,EAC/B,2BAA2B,EAC3B,mCAAmC,EACnC,0BAA0B,EAC1B,4BAA4B,GAC7B,MAAM,uCAAuC,CAAC;AAE/C,MAAM,mBAAmB,GAAG,8CAA8C,CAAC;AAC3E,MAAM,oBAAoB,GAAG,8CAA8C,CAAC;AAE5E,QAAQ,CAAC,6BAA6B,EAAE,GAAG,EAAE;IAC3C,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,CAAC,2BAA2B,CAAC,CAAC,IAAI,CAAC,yCAAyC,CAAC,CAAC;IACtF,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,qCAAqC,EAAE,GAAG,EAAE;IACnD,EAAE,CAAC,sCAAsC,EAAE,GAAG,EAAE;QAC9C,MAAM,CAAC,GAAG,mCAAmC,CAAC,mBAAmB,CAAC,CAAC;QACnE,MAAM,CAAC,GAAG,mCAAmC,CAAC,mBAAmB,CAAC,CAAC;QACnE,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC1B,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mDAAmD,EAAE,GAAG,EAAE;QAC3D,MAAM,CAAC,GAAG,mCAAmC,CAAC,mBAAmB,CAAC,CAAC;QACnE,MAAM,CAAC,GAAG,mCAAmC,CAAC,oBAAoB,CAAC,CAAC;QACpE,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAClC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,OAAO,GAAG,mCAAmC,CAAC,mBAAmB,CAAC,CAAC;QACzE,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,mBAAmB,EAAE,QAAQ,CAAC,CAAC;QAC1D,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC7C,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,6DAA6D,EAAE,GAAG,EAAE;IAC3E,MAAM,UAAU,GAAG,mCAAmC,CAAC,mBAAmB,CAAC,CAAC;IAE5E,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,KAAK,GAAG,4BAA4B,CAAC,gCAAgC,EAAE,UAAU,CAAC,CAAC;QACzF,MAAM,OAAO,GAAG,4BAA4B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC;QAChE,MAAM,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC,EAAE,iBAAiB,EAAE,gCAAgC,EAAE,CAAC,CAAC;IACnF,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,KAAK,GAAG,4BAA4B,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC;QAClE,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;IAC3C,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,GAAG,EAAE;QACtC,MAAM,KAAK,GAAG,4BAA4B,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC;QAClE,MAAM,MAAM,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAE,CAAC;QACpC,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QAChE,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;IACjE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mDAAmD,EAAE,GAAG,EAAE;QAC3D,MAAM,KAAK,GAAG,4BAA4B,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC;QAClE,MAAM,OAAO,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAE,CAAC;QACrC,MAAM,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,WAAW,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC;QACjE,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,EAAE,iBAAiB,EAAE,SAAS,EAAE,CAAC,CAAC;IACrE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,yCAAyC,EAAE,GAAG,EAAE;QACjD,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,oBAAoB,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CAClF,+BAA+B,CAChC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;QACpD,MAAM,QAAQ,GAAG,mCAAmC,CAAC,oBAAoB,CAAC,CAAC;QAC3E,MAAM,KAAK,GAAG,4BAA4B,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;QAChE,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACnE,+BAA+B,CAChC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wCAAwC,EAAE,GAAG,EAAE;QAChD,MAAM,KAAK,GAAG,4BAA4B,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC;QAClE,MAAM,CAAC,MAAM,EAAE,AAAD,EAAG,SAAS,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAA6B,CAAC;QAC3E,MAAM,eAAe,GAAG,MAAM,CAAC,IAAI,CACjC,IAAI,CAAC,SAAS,CAAC,EAAE,iBAAiB,EAAE,SAAS,EAAE,CAAC,EAChD,OAAO,CACR,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;QACxB,MAAM,QAAQ,GAAG,GAAG,MAAM,IAAI,eAAe,IAAI,SAAS,EAAE,CAAC;QAC7D,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,QAAQ,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACtE,+BAA+B,CAChC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACnE,+BAA+B,CAChC,CAAC;QACF,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACvE,+BAA+B,CAChC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,SAAS,SAAS,CAAC,KAAa;QAC9B,OAAO,MAAM,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;IAC3D,CAAC;IAED,SAAS,gBAAgB,CAAC,UAAkB,EAAE,WAAmB;QAC/D,MAAM,aAAa,GAAG,SAAS,CAAC,UAAU,CAAC,CAAC;QAC5C,MAAM,cAAc,GAAG,SAAS,CAAC,WAAW,CAAC,CAAC;QAC9C,MAAM,SAAS,GAAG,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC;aAC/C,MAAM,CAAC,GAAG,aAAa,IAAI,cAAc,EAAE,CAAC;aAC5C,MAAM,CAAC,WAAW,CAAC,CAAC;QACvB,OAAO,GAAG,aAAa,IAAI,cAAc,IAAI,SAAS,EAAE,CAAC;IAC3D,CAAC;IAED,EAAE,CAAC,gDAAgD,EAAE,GAAG,EAAE;QACxD,0EAA0E;QAC1E,wDAAwD;QACxD,MAAM,aAAa,GAAG,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;QAC9E,MAAM,cAAc,GAAG,SAAS,CAAC,UAAU,CAAC,CAAC;QAC7C,MAAM,SAAS,GAAG,UAAU,CAAC,QAAQ,EAAE,UAAU,CAAC;aAC/C,MAAM,CAAC,GAAG,aAAa,IAAI,cAAc,EAAE,CAAC;aAC5C,MAAM,CAAC,WAAW,CAAC,CAAC;QACvB,MAAM,KAAK,GAAG,GAAG,aAAa,IAAI,cAAc,IAAI,SAAS,EAAE,CAAC;QAChE,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACnE,+BAA+B,CAChC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4CAA4C,EAAE,GAAG,EAAE;QACpD,MAAM,KAAK,GAAG,gBAAgB,CAC5B,IAAI,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,EAC5C,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,SAAS,EAAE,CAAC,CACrC,CAAC;QACF,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACnE,+BAA+B,CAChC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0DAA0D,EAAE,GAAG,EAAE;QAClE,MAAM,KAAK,GAAG,gBAAgB,CAC5B,IAAI,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,EAC5C,IAAI,CAAC,SAAS,CAAC,EAAE,iBAAiB,EAAE,eAAe,EAAE,CAAC,CACvD,CAAC;QACF,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACnE,+BAA+B,CAChC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,KAAK,GAAG,gBAAgB,CAC5B,IAAI,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,MAAM,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,EAC3C,IAAI,CAAC,SAAS,CAAC,EAAE,iBAAiB,EAAE,SAAS,EAAE,CAAC,CACjD,CAAC;QACF,MAAM,CAAC,GAAG,EAAE,CAAC,4BAA4B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACnE,+BAA+B,CAChC,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,4BAA4B,EAAE,GAAG,EAAE;IAC1C,MAAM,UAAU,GAAG,mCAAmC,CAAC,mBAAmB,CAAC,CAAC;IAC5E,IAAI,OAAe,CAAC;IAEpB,UAAU,CAAC,GAAG,EAAE;QACd,OAAO,GAAG,WAAW,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,mBAAmB,CAAC,CAAC,CAAC;IAC7D,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,MAAM,CAAC,OAAO,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;IACpD,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uCAAuC,EAAE,GAAG,EAAE;QAC/C,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,EAAE,kBAAkB,CAAC,CAAC;QAC/C,aAAa,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;QAC1B,MAAM,KAAK,GAAG,4BAA4B,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;QAC7D,MAAM,CAAC,0BAA0B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACnE,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oEAAoE,EAAE,GAAG,EAAE;QAC5E,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,EAAE,qBAAqB,CAAC,CAAC;QAClD,MAAM,KAAK,GAAG,4BAA4B,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;QAC7D,MAAM,CAAC,GAAG,EAAE,CAAC,0BAA0B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACjE,mCAAmC,CACpC,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,yEAAyE,EAAE,GAAG,EAAE;QACjF,MAAM,IAAI,GAAG,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;QACrC,SAAS,CAAC,IAAI,CAAC,CAAC;QAChB,MAAM,KAAK,GAAG,4BAA4B,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;QAC7D,MAAM,CAAC,GAAG,EAAE,CAAC,0BAA0B,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,OAAO,CACjE,mCAAmC,CACpC,CAAC;IACJ,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resolveEncryptionKey.test.d.ts","sourceRoot":"","sources":["../../tests/resolveEncryptionKey.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { describe, it, expect, vi } from 'vitest';
|
|
2
|
+
import { EncryptionKeyLostError, generateKey, resolveEncryptionKey } from '../src/encryption.js';
|
|
3
|
+
vi.mock('../src/keychain.js', async (importOriginal) => {
|
|
4
|
+
const original = await importOriginal();
|
|
5
|
+
return {
|
|
6
|
+
...original,
|
|
7
|
+
retrieveFromKeychain: () => Promise.resolve(null),
|
|
8
|
+
storeInKeychain: () => Promise.resolve(undefined),
|
|
9
|
+
};
|
|
10
|
+
});
|
|
11
|
+
describe('resolveEncryptionKey', () => {
|
|
12
|
+
it('returns the override verbatim and does not touch the keychain', async () => {
|
|
13
|
+
const override = generateKey();
|
|
14
|
+
await expect(resolveEncryptionKey({ encryptionKeyOverride: override })).resolves.toBe(override);
|
|
15
|
+
});
|
|
16
|
+
it('throws EncryptionKeyLostError when allowKeyGeneration is false and keychain has no key', async () => {
|
|
17
|
+
await expect(resolveEncryptionKey({ allowKeyGeneration: false })).rejects.toThrow(EncryptionKeyLostError);
|
|
18
|
+
});
|
|
19
|
+
it('generates a new key when allowKeyGeneration is true and keychain has no key', async () => {
|
|
20
|
+
await expect(resolveEncryptionKey({ allowKeyGeneration: true })).resolves.toMatch(/.+/);
|
|
21
|
+
});
|
|
22
|
+
it('generates a new key when allowKeyGeneration is unset and keychain has no key', async () => {
|
|
23
|
+
await expect(resolveEncryptionKey({})).resolves.toMatch(/.+/);
|
|
24
|
+
});
|
|
25
|
+
});
|
|
26
|
+
//# sourceMappingURL=resolveEncryptionKey.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"resolveEncryptionKey.test.js","sourceRoot":"","sources":["../../tests/resolveEncryptionKey.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,MAAM,QAAQ,CAAC;AAClD,OAAO,EAAE,sBAAsB,EAAE,WAAW,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAEjG,EAAE,CAAC,IAAI,CAAC,oBAAoB,EAAE,KAAK,EAAE,cAAc,EAAE,EAAE;IACrD,MAAM,QAAQ,GAAG,MAAM,cAAc,EAAuC,CAAC;IAC7E,OAAO;QACL,GAAG,QAAQ;QACX,oBAAoB,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC;QACjD,eAAe,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,OAAO,CAAC,SAAS,CAAC;KAClD,CAAC;AACJ,CAAC,CAAC,CAAC;AAEH,QAAQ,CAAC,sBAAsB,EAAE,GAAG,EAAE;IACpC,EAAE,CAAC,+DAA+D,EAAE,KAAK,IAAI,EAAE;QAC7E,MAAM,QAAQ,GAAG,WAAW,EAAE,CAAC;QAC/B,MAAM,MAAM,CAAC,oBAAoB,CAAC,EAAE,qBAAqB,EAAE,QAAQ,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IAClG,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wFAAwF,EAAE,KAAK,IAAI,EAAE;QACtG,MAAM,MAAM,CAAC,oBAAoB,CAAC,EAAE,kBAAkB,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAC/E,sBAAsB,CACvB,CAAC;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6EAA6E,EAAE,KAAK,IAAI,EAAE;QAC3F,MAAM,MAAM,CAAC,oBAAoB,CAAC,EAAE,kBAAkB,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAC1F,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8EAA8E,EAAE,KAAK,IAAI,EAAE;QAC5F,MAAM,MAAM,CAAC,oBAAoB,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;IAChE,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
|