@agent-relay/sdk 4.0.1 → 4.0.3

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.
Files changed (94) hide show
  1. package/bin/agent-relay-broker-darwin-arm64 +0 -0
  2. package/bin/agent-relay-broker-darwin-x64 +0 -0
  3. package/bin/agent-relay-broker-linux-arm64 +0 -0
  4. package/bin/agent-relay-broker-linux-x64 +0 -0
  5. package/dist/client.d.ts +3 -10
  6. package/dist/client.d.ts.map +1 -1
  7. package/dist/client.js +2 -0
  8. package/dist/client.js.map +1 -1
  9. package/dist/provisioner/__tests__/audit.test.d.ts +2 -0
  10. package/dist/provisioner/__tests__/audit.test.d.ts.map +1 -0
  11. package/dist/provisioner/__tests__/audit.test.js +45 -0
  12. package/dist/provisioner/__tests__/audit.test.js.map +1 -0
  13. package/dist/provisioner/__tests__/compiler.test.d.ts +2 -0
  14. package/dist/provisioner/__tests__/compiler.test.d.ts.map +1 -0
  15. package/dist/provisioner/__tests__/compiler.test.js +345 -0
  16. package/dist/provisioner/__tests__/compiler.test.js.map +1 -0
  17. package/dist/provisioner/__tests__/presets.test.d.ts +2 -0
  18. package/dist/provisioner/__tests__/presets.test.d.ts.map +1 -0
  19. package/dist/provisioner/__tests__/presets.test.js +23 -0
  20. package/dist/provisioner/__tests__/presets.test.js.map +1 -0
  21. package/dist/provisioner/__tests__/seeder.test.d.ts +2 -0
  22. package/dist/provisioner/__tests__/seeder.test.d.ts.map +1 -0
  23. package/dist/provisioner/__tests__/seeder.test.js +224 -0
  24. package/dist/provisioner/__tests__/seeder.test.js.map +1 -0
  25. package/dist/provisioner/__tests__/tar-seeder.test.d.ts +2 -0
  26. package/dist/provisioner/__tests__/tar-seeder.test.d.ts.map +1 -0
  27. package/dist/provisioner/__tests__/tar-seeder.test.js +191 -0
  28. package/dist/provisioner/__tests__/tar-seeder.test.js.map +1 -0
  29. package/dist/provisioner/__tests__/token-factory.test.d.ts +2 -0
  30. package/dist/provisioner/__tests__/token-factory.test.d.ts.map +1 -0
  31. package/dist/provisioner/__tests__/token-factory.test.js +127 -0
  32. package/dist/provisioner/__tests__/token-factory.test.js.map +1 -0
  33. package/dist/provisioner/__tests__/token.test.d.ts +2 -0
  34. package/dist/provisioner/__tests__/token.test.d.ts.map +1 -0
  35. package/dist/provisioner/__tests__/token.test.js +44 -0
  36. package/dist/provisioner/__tests__/token.test.js.map +1 -0
  37. package/dist/provisioner/audit.d.ts +19 -0
  38. package/dist/provisioner/audit.d.ts.map +1 -0
  39. package/dist/provisioner/audit.js +74 -0
  40. package/dist/provisioner/audit.js.map +1 -0
  41. package/dist/provisioner/compiler.d.ts +23 -0
  42. package/dist/provisioner/compiler.d.ts.map +1 -0
  43. package/dist/provisioner/compiler.js +355 -0
  44. package/dist/provisioner/compiler.js.map +1 -0
  45. package/dist/provisioner/index.d.ts +9 -0
  46. package/dist/provisioner/index.d.ts.map +1 -0
  47. package/dist/provisioner/index.js +266 -0
  48. package/dist/provisioner/index.js.map +1 -0
  49. package/dist/provisioner/mount.d.ts +14 -0
  50. package/dist/provisioner/mount.d.ts.map +1 -0
  51. package/dist/provisioner/mount.js +329 -0
  52. package/dist/provisioner/mount.js.map +1 -0
  53. package/dist/provisioner/seeder.d.ts +17 -0
  54. package/dist/provisioner/seeder.d.ts.map +1 -0
  55. package/dist/provisioner/seeder.js +419 -0
  56. package/dist/provisioner/seeder.js.map +1 -0
  57. package/dist/provisioner/token.d.ts +38 -0
  58. package/dist/provisioner/token.d.ts.map +1 -0
  59. package/dist/provisioner/token.js +74 -0
  60. package/dist/provisioner/token.js.map +1 -0
  61. package/dist/provisioner/types.d.ts +133 -0
  62. package/dist/provisioner/types.d.ts.map +1 -0
  63. package/dist/provisioner/types.js +2 -0
  64. package/dist/provisioner/types.js.map +1 -0
  65. package/dist/relay.d.ts +6 -0
  66. package/dist/relay.d.ts.map +1 -1
  67. package/dist/relay.js +17 -5
  68. package/dist/relay.js.map +1 -1
  69. package/dist/types.d.ts +9 -0
  70. package/dist/types.d.ts.map +1 -1
  71. package/dist/workflows/__tests__/e2e-permissions.test.d.ts +2 -0
  72. package/dist/workflows/__tests__/e2e-permissions.test.d.ts.map +1 -0
  73. package/dist/workflows/__tests__/e2e-permissions.test.js +331 -0
  74. package/dist/workflows/__tests__/e2e-permissions.test.js.map +1 -0
  75. package/dist/workflows/__tests__/permission-types.test.d.ts +2 -0
  76. package/dist/workflows/__tests__/permission-types.test.d.ts.map +1 -0
  77. package/dist/workflows/__tests__/permission-types.test.js +124 -0
  78. package/dist/workflows/__tests__/permission-types.test.js.map +1 -0
  79. package/dist/workflows/__tests__/permissions-integration.test.d.ts +2 -0
  80. package/dist/workflows/__tests__/permissions-integration.test.d.ts.map +1 -0
  81. package/dist/workflows/__tests__/permissions-integration.test.js +526 -0
  82. package/dist/workflows/__tests__/permissions-integration.test.js.map +1 -0
  83. package/dist/workflows/dry-run-format.d.ts.map +1 -1
  84. package/dist/workflows/dry-run-format.js +8 -0
  85. package/dist/workflows/dry-run-format.js.map +1 -1
  86. package/dist/workflows/runner.d.ts +14 -0
  87. package/dist/workflows/runner.d.ts.map +1 -1
  88. package/dist/workflows/runner.js +455 -6
  89. package/dist/workflows/runner.js.map +1 -1
  90. package/dist/workflows/types.d.ts +190 -0
  91. package/dist/workflows/types.d.ts.map +1 -1
  92. package/dist/workflows/types.js +29 -0
  93. package/dist/workflows/types.js.map +1 -1
  94. package/package.json +6 -2
@@ -0,0 +1,224 @@
1
+ import assert from 'node:assert/strict';
2
+ import { mkdtemp, mkdir, rm, writeFile } from 'node:fs/promises';
3
+ import { tmpdir } from 'node:os';
4
+ import path from 'node:path';
5
+ import test, { afterEach, mock } from 'node:test';
6
+ import { RelayFileClient } from '@relayfile/sdk';
7
+ import { seedAclRules, seedWorkflowAcls, seedWorkspace } from '../seeder.js';
8
+ const originalFetch = globalThis.fetch;
9
+ async function createWorkspace(files) {
10
+ const dir = await mkdtemp(path.join(tmpdir(), 'relay-seeder-'));
11
+ for (const [relativePath, content] of Object.entries(files)) {
12
+ const filePath = path.join(dir, relativePath);
13
+ await mkdir(path.dirname(filePath), { recursive: true });
14
+ await writeFile(filePath, content);
15
+ }
16
+ return {
17
+ dir,
18
+ cleanup: () => rm(dir, { recursive: true, force: true }),
19
+ };
20
+ }
21
+ function createFetchResponse(body, status = 200) {
22
+ return {
23
+ ok: status >= 200 && status < 300,
24
+ status,
25
+ async text() {
26
+ return body;
27
+ },
28
+ };
29
+ }
30
+ function parseFetchBody(fetchMock) {
31
+ assert.equal(fetchMock.mock.calls.length, 1);
32
+ const [, options] = fetchMock.mock.calls[0].arguments;
33
+ assert.equal(typeof options.body, 'string');
34
+ return JSON.parse(options.body);
35
+ }
36
+ afterEach(() => {
37
+ mock.restoreAll();
38
+ globalThis.fetch = originalFetch;
39
+ });
40
+ test('seedWorkspace posts the expected HTTP payload after SDK fallback', async () => {
41
+ const workspace = await createWorkspace({
42
+ 'alpha.txt': 'alpha payload',
43
+ 'binary.bin': Buffer.from([0xff, 0x00, 0x61]),
44
+ '.relay/ignored.txt': 'skip',
45
+ '.git/config': 'skip',
46
+ 'node_modules/pkg/index.js': 'skip',
47
+ 'custom-skip/ignored.txt': 'skip',
48
+ '.relayfile-mount-state.json': '{"skip":true}',
49
+ });
50
+ try {
51
+ mock.method(RelayFileClient.prototype, 'bulkWrite', async () => {
52
+ throw new Error('fall back to HTTP');
53
+ });
54
+ const fetchMock = mock.method(globalThis, 'fetch', async () => createFetchResponse(JSON.stringify({ written: 2, errorCount: 0, errors: [] })));
55
+ const seededCount = await seedWorkspace('https://relay.example///', 'admin-token', ' workspace-123 ', workspace.dir, ['custom-skip']);
56
+ assert.equal(seededCount, 2);
57
+ assert.equal(fetchMock.mock.calls.length, 1);
58
+ const [url, options] = fetchMock.mock.calls[0].arguments;
59
+ assert.equal(url, 'https://relay.example/v1/workspaces/workspace-123/fs/bulk');
60
+ assert.equal(options.method, 'POST');
61
+ assert.deepEqual(options.headers, {
62
+ Authorization: 'Bearer admin-token',
63
+ 'Content-Type': 'application/json',
64
+ 'X-Correlation-Id': options.headers && options.headers['X-Correlation-Id'],
65
+ });
66
+ assert.match(options.headers['X-Correlation-Id'], /^seed-workspace-workspace-123-\d+-0$/u);
67
+ const body = parseFetchBody(fetchMock);
68
+ assert.deepEqual(body.files, [
69
+ {
70
+ path: '/alpha.txt',
71
+ content: 'alpha payload',
72
+ encoding: 'utf-8',
73
+ },
74
+ {
75
+ path: '/binary.bin',
76
+ content: Buffer.from([0xff, 0x00, 0x61]).toString('base64'),
77
+ encoding: 'base64',
78
+ },
79
+ ]);
80
+ }
81
+ finally {
82
+ await workspace.cleanup();
83
+ }
84
+ });
85
+ test('seedAclRules formats ACL files for root and nested directories', async () => {
86
+ mock.method(RelayFileClient.prototype, 'bulkWrite', async () => {
87
+ throw new Error('fall back to HTTP');
88
+ });
89
+ const fetchMock = mock.method(globalThis, 'fetch', async () => createFetchResponse(JSON.stringify({ written: 2, errorCount: 0, errors: [] })));
90
+ await seedAclRules('https://relay.example/', 'acl-token', 'workspace-acl', {
91
+ '/': ['allow:agent:lead:read'],
92
+ '/docs/': ['allow:agent:writer:write', 'deny:agent:reader'],
93
+ });
94
+ const body = parseFetchBody(fetchMock);
95
+ assert.deepEqual(body.files, [
96
+ {
97
+ path: '/.relayfile.acl',
98
+ content: JSON.stringify({
99
+ semantics: { permissions: ['allow:agent:lead:read'] },
100
+ }),
101
+ encoding: 'utf-8',
102
+ },
103
+ {
104
+ path: '/docs/.relayfile.acl',
105
+ content: JSON.stringify({
106
+ semantics: {
107
+ permissions: ['allow:agent:writer:write', 'deny:agent:reader'],
108
+ },
109
+ }),
110
+ encoding: 'utf-8',
111
+ },
112
+ ]);
113
+ });
114
+ test('seedWorkflowAcls merges multiple agents onto shared directories', async () => {
115
+ mock.method(RelayFileClient.prototype, 'bulkWrite', async () => {
116
+ throw new Error('fall back to HTTP');
117
+ });
118
+ const fetchMock = mock.method(globalThis, 'fetch', async () => createFetchResponse(JSON.stringify({ written: 2, errorCount: 0, errors: [] })));
119
+ await seedWorkflowAcls({
120
+ relayfileUrl: 'https://relay.example',
121
+ adminToken: 'workflow-token',
122
+ workspace: 'workflow-merge',
123
+ agents: [
124
+ { name: 'qa-reviewer', acl: { src: ['read'] } },
125
+ { name: 'builder', acl: { 'src\\': ['write'], '/docs/': ['read'] } },
126
+ { name: 'analyst', acl: { docs: ['read'] } },
127
+ ],
128
+ });
129
+ const body = parseFetchBody(fetchMock);
130
+ assert.deepEqual(body.files, [
131
+ {
132
+ path: '/docs/.relayfile.acl',
133
+ content: JSON.stringify({
134
+ semantics: {
135
+ permissions: [
136
+ 'allow:agent:analyst:read',
137
+ 'allow:agent:builder:read',
138
+ 'allow:agent:qa-reviewer:read',
139
+ ],
140
+ },
141
+ }),
142
+ encoding: 'utf-8',
143
+ },
144
+ {
145
+ path: '/src/.relayfile.acl',
146
+ content: JSON.stringify({
147
+ semantics: {
148
+ permissions: [
149
+ 'allow:agent:builder:read',
150
+ 'allow:agent:builder:write',
151
+ 'allow:agent:qa-reviewer:read',
152
+ 'deny:agent:analyst',
153
+ ],
154
+ },
155
+ }),
156
+ encoding: 'utf-8',
157
+ },
158
+ ]);
159
+ });
160
+ test('seedWorkflowAcls unions deny rules for agents missing directory access', async () => {
161
+ mock.method(RelayFileClient.prototype, 'bulkWrite', async () => {
162
+ throw new Error('fall back to HTTP');
163
+ });
164
+ const fetchMock = mock.method(globalThis, 'fetch', async () => createFetchResponse(JSON.stringify({ written: 2, errorCount: 0, errors: [] })));
165
+ await seedWorkflowAcls({
166
+ relayfileUrl: 'https://relay.example',
167
+ adminToken: 'workflow-token',
168
+ workspace: 'workflow-deny',
169
+ agents: [
170
+ { name: 'alpha', acl: { src: ['read'] } },
171
+ { name: 'beta', acl: { docs: ['write'] } },
172
+ ],
173
+ });
174
+ const body = parseFetchBody(fetchMock);
175
+ assert.deepEqual(body.files, [
176
+ {
177
+ path: '/docs/.relayfile.acl',
178
+ content: JSON.stringify({
179
+ semantics: {
180
+ permissions: ['allow:agent:beta:read', 'allow:agent:beta:write', 'deny:agent:alpha'],
181
+ },
182
+ }),
183
+ encoding: 'utf-8',
184
+ },
185
+ {
186
+ path: '/src/.relayfile.acl',
187
+ content: JSON.stringify({
188
+ semantics: {
189
+ permissions: ['allow:agent:alpha:read', 'deny:agent:beta'],
190
+ },
191
+ }),
192
+ encoding: 'utf-8',
193
+ },
194
+ ]);
195
+ });
196
+ test('seedWorkflowAcls is a no-op when there are no ACL directories to seed', async () => {
197
+ const bulkWriteMock = mock.method(RelayFileClient.prototype, 'bulkWrite', async () => {
198
+ throw new Error('bulkWrite should not be called');
199
+ });
200
+ const fetchMock = mock.method(globalThis, 'fetch', async () => {
201
+ throw new Error('fetch should not be called');
202
+ });
203
+ await seedWorkflowAcls({
204
+ relayfileUrl: 'https://relay.example',
205
+ adminToken: 'workflow-token',
206
+ workspace: 'workflow-empty',
207
+ agents: [
208
+ { name: 'builder', acl: {} },
209
+ { name: 'qa-reviewer', acl: {} },
210
+ ],
211
+ });
212
+ assert.equal(bulkWriteMock.mock.calls.length, 0);
213
+ assert.equal(fetchMock.mock.calls.length, 0);
214
+ });
215
+ test('seedAclRules surfaces HTTP failures from the fallback API', async () => {
216
+ mock.method(RelayFileClient.prototype, 'bulkWrite', async () => {
217
+ throw new Error('fall back to HTTP');
218
+ });
219
+ mock.method(globalThis, 'fetch', async () => createFetchResponse('relay unavailable', 503));
220
+ await assert.rejects(seedAclRules('https://relay.example', 'acl-token', 'workspace-http', {
221
+ '/': ['allow:agent:builder:read'],
222
+ }), new Error('failed to seed workspace workspace-http: HTTP 503 relay unavailable'));
223
+ });
224
+ //# sourceMappingURL=seeder.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"seeder.test.js","sourceRoot":"","sources":["../../../src/provisioner/__tests__/seeder.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=tar-seeder.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tar-seeder.test.d.ts","sourceRoot":"","sources":["../../../src/provisioner/__tests__/tar-seeder.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,191 @@
1
+ import fs from 'node:fs';
2
+ import { tmpdir } from 'node:os';
3
+ import path from 'node:path';
4
+ import * as tar from 'tar';
5
+ import { afterEach, describe, expect, it, vi } from 'vitest';
6
+ const bulkWriteMock = vi.hoisted(() => vi.fn());
7
+ const relayFileClientMock = vi.hoisted(() => vi.fn());
8
+ const execSyncMock = vi.hoisted(() => vi.fn());
9
+ vi.mock('@relayfile/sdk', () => ({
10
+ RelayFileClient: relayFileClientMock.mockImplementation(() => ({
11
+ bulkWrite: bulkWriteMock,
12
+ })),
13
+ }));
14
+ vi.mock('node:child_process', () => ({
15
+ execSync: execSyncMock,
16
+ }));
17
+ import { seedWorkspaceTar } from '../seeder.js';
18
+ const tempDirs = [];
19
+ function makeTempDir(prefix) {
20
+ const dir = fs.mkdtempSync(path.join(tmpdir(), prefix));
21
+ tempDirs.push(dir);
22
+ return dir;
23
+ }
24
+ function jsonResponse(payload, status = 200) {
25
+ return new Response(JSON.stringify(payload), {
26
+ status,
27
+ headers: { 'Content-Type': 'application/json' },
28
+ });
29
+ }
30
+ function listRelativeFiles(rootDir, currentDir = rootDir) {
31
+ const files = [];
32
+ const entries = fs.readdirSync(currentDir, { withFileTypes: true });
33
+ for (const entry of entries) {
34
+ const absolutePath = path.join(currentDir, entry.name);
35
+ if (entry.isDirectory()) {
36
+ files.push(...listRelativeFiles(rootDir, absolutePath));
37
+ continue;
38
+ }
39
+ if (entry.isFile()) {
40
+ files.push(path.relative(rootDir, absolutePath).split(path.sep).join('/'));
41
+ }
42
+ }
43
+ return files.sort((left, right) => left.localeCompare(right));
44
+ }
45
+ async function extractTarballEntries(body) {
46
+ const archiveDir = makeTempDir('relay-tar-archive-');
47
+ const extractDir = makeTempDir('relay-tar-extract-');
48
+ const archivePath = path.join(archiveDir, 'seed.tar.gz');
49
+ fs.writeFileSync(archivePath, Buffer.from(body));
50
+ await tar.extract({ file: archivePath, cwd: extractDir, gzip: true });
51
+ return listRelativeFiles(extractDir);
52
+ }
53
+ afterEach(() => {
54
+ bulkWriteMock.mockReset();
55
+ relayFileClientMock.mockClear();
56
+ execSyncMock.mockReset();
57
+ vi.restoreAllMocks();
58
+ vi.unstubAllGlobals();
59
+ for (const dir of tempDirs.splice(0)) {
60
+ fs.rmSync(dir, { recursive: true, force: true });
61
+ }
62
+ });
63
+ describe('seedWorkspaceTar', () => {
64
+ it('creates and uploads a tar.gz to the import endpoint and respects excludeDirs', async () => {
65
+ const projectDir = makeTempDir('relay-seed-project-');
66
+ fs.mkdirSync(path.join(projectDir, 'src'), { recursive: true });
67
+ fs.mkdirSync(path.join(projectDir, 'ignored'), { recursive: true });
68
+ fs.mkdirSync(path.join(projectDir, 'node_modules', 'left-pad'), { recursive: true });
69
+ fs.writeFileSync(path.join(projectDir, 'src', 'hello.txt'), 'hello world\n');
70
+ fs.writeFileSync(path.join(projectDir, 'src', 'data.bin'), Buffer.from([0xff, 0x00, 0xaa]));
71
+ fs.writeFileSync(path.join(projectDir, 'ignored', 'skip.txt'), 'skip me\n');
72
+ fs.writeFileSync(path.join(projectDir, 'node_modules', 'left-pad', 'index.js'), 'module.exports = 1;\n');
73
+ fs.writeFileSync(path.join(projectDir, '.relayfile-mount-state.json'), '{}\n');
74
+ execSyncMock.mockReturnValue([
75
+ 'src/hello.txt',
76
+ 'src/data.bin',
77
+ 'ignored/skip.txt',
78
+ 'node_modules/left-pad/index.js',
79
+ '.relayfile-mount-state.json',
80
+ ].join('\0'));
81
+ const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ imported: 2 }));
82
+ vi.stubGlobal('fetch', fetchMock);
83
+ const imported = await seedWorkspaceTar('https://relayfile.example/', 'token', 'rw_demo', projectDir, [
84
+ 'ignored',
85
+ ]);
86
+ expect(imported).toBe(2);
87
+ expect(fetchMock).toHaveBeenCalledTimes(1);
88
+ const [url, init] = fetchMock.mock.calls[0];
89
+ expect(String(url)).toContain('/v1/workspaces/rw_demo/fs/import');
90
+ expect(init.method).toBe('POST');
91
+ expect(init.headers).toMatchObject({
92
+ Authorization: 'Bearer token',
93
+ 'Content-Type': 'application/gzip',
94
+ 'X-Correlation-Id': expect.stringMatching(/^seed-tar-rw_demo-/),
95
+ });
96
+ expect(init.body).toBeInstanceOf(Uint8Array);
97
+ const entries = await extractTarballEntries(init.body);
98
+ expect(entries).toEqual(expect.arrayContaining(['src/data.bin', 'src/hello.txt']));
99
+ expect(entries).not.toContain('ignored/skip.txt');
100
+ expect(entries).not.toContain('node_modules/left-pad/index.js');
101
+ expect(entries).not.toContain('.relayfile-mount-state.json');
102
+ });
103
+ it('falls back to seedWorkspace when the import endpoint returns 404', async () => {
104
+ const projectDir = makeTempDir('relay-seed-project-');
105
+ fs.mkdirSync(path.join(projectDir, 'src'), { recursive: true });
106
+ fs.writeFileSync(path.join(projectDir, 'src', 'hello.txt'), 'hello fallback\n');
107
+ execSyncMock.mockReturnValue('src/hello.txt\0');
108
+ bulkWriteMock.mockRejectedValue({ status: undefined });
109
+ const fetchMock = vi
110
+ .fn()
111
+ .mockResolvedValueOnce(new Response('missing', { status: 404 }))
112
+ .mockResolvedValueOnce(jsonResponse({ written: 1, errorCount: 0, errors: [] }));
113
+ vi.stubGlobal('fetch', fetchMock);
114
+ const imported = await seedWorkspaceTar('https://relayfile.example/', 'token', 'rw_demo', projectDir, []);
115
+ expect(imported).toBe(1);
116
+ expect(fetchMock).toHaveBeenCalledTimes(2);
117
+ expect(String(fetchMock.mock.calls[0]?.[0])).toContain('/v1/workspaces/rw_demo/fs/import');
118
+ expect(String(fetchMock.mock.calls[1]?.[0])).toContain('/v1/workspaces/rw_demo/fs/bulk');
119
+ const payload = JSON.parse(String(fetchMock.mock.calls[1]?.[1].body));
120
+ expect(payload.files).toEqual([
121
+ { path: '/src/hello.txt', content: 'hello fallback\n', encoding: 'utf-8' },
122
+ ]);
123
+ });
124
+ it('throws on non-404 HTTP errors', async () => {
125
+ const projectDir = makeTempDir('relay-seed-project-');
126
+ fs.mkdirSync(path.join(projectDir, 'src'), { recursive: true });
127
+ fs.writeFileSync(path.join(projectDir, 'src', 'hello.txt'), 'hello\n');
128
+ execSyncMock.mockReturnValue('src/hello.txt\0');
129
+ const fetchMock = vi.fn().mockResolvedValue(new Response('boom', { status: 500 }));
130
+ vi.stubGlobal('fetch', fetchMock);
131
+ await expect(seedWorkspaceTar('https://relayfile.example/', 'token', 'rw_demo', projectDir, [])).rejects.toThrow('tar import failed for workspace rw_demo: HTTP 500 boom');
132
+ });
133
+ it('works for non-git directories via the directory-walk fallback path', async () => {
134
+ const projectDir = makeTempDir('relay-seed-project-');
135
+ fs.mkdirSync(path.join(projectDir, 'src'), { recursive: true });
136
+ fs.mkdirSync(path.join(projectDir, 'nested', 'docs'), { recursive: true });
137
+ fs.mkdirSync(path.join(projectDir, 'custom-ignore'), { recursive: true });
138
+ fs.mkdirSync(path.join(projectDir, 'node_modules', 'left-pad'), { recursive: true });
139
+ fs.writeFileSync(path.join(projectDir, 'src', 'app.ts'), 'export const app = true;\n');
140
+ fs.writeFileSync(path.join(projectDir, 'nested', 'docs', 'readme.md'), '# hello\n');
141
+ fs.writeFileSync(path.join(projectDir, 'custom-ignore', 'skip.txt'), 'skip\n');
142
+ fs.writeFileSync(path.join(projectDir, 'node_modules', 'left-pad', 'index.js'), 'module.exports = 1;\n');
143
+ fs.writeFileSync(path.join(projectDir, '.relayfile-mount-state.json'), '{}\n');
144
+ execSyncMock.mockImplementation(() => {
145
+ throw new Error('not a git repo');
146
+ });
147
+ const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ imported: 2 }));
148
+ vi.stubGlobal('fetch', fetchMock);
149
+ const imported = await seedWorkspaceTar('https://relayfile.example/', 'token', 'rw_demo', projectDir, [
150
+ 'custom-ignore',
151
+ ]);
152
+ expect(imported).toBe(2);
153
+ expect(fetchMock).toHaveBeenCalledTimes(1);
154
+ const [, init] = fetchMock.mock.calls[0];
155
+ const entries = await extractTarballEntries(init.body);
156
+ expect(entries).toEqual(expect.arrayContaining(['nested/docs/readme.md', 'src/app.ts']));
157
+ expect(entries).not.toContain('custom-ignore/skip.txt');
158
+ expect(entries).not.toContain('node_modules/left-pad/index.js');
159
+ expect(entries).not.toContain('.relayfile-mount-state.json');
160
+ });
161
+ it('includes untracked files returned by git ls-files and preserves gitignore filtering', async () => {
162
+ const projectDir = makeTempDir('relay-seed-project-');
163
+ fs.mkdirSync(path.join(projectDir, 'src'), { recursive: true });
164
+ fs.mkdirSync(path.join(projectDir, 'ignored-by-git'), { recursive: true });
165
+ fs.writeFileSync(path.join(projectDir, 'src', 'tracked.ts'), 'export const tracked = true;\n');
166
+ fs.writeFileSync(path.join(projectDir, 'src', 'draft.ts'), 'export const draft = true;\n');
167
+ fs.writeFileSync(path.join(projectDir, 'ignored-by-git', 'skip.txt'), 'skip\n');
168
+ execSyncMock.mockReturnValue(['src/tracked.ts', 'src/draft.ts'].join('\0'));
169
+ const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ imported: 2 }));
170
+ vi.stubGlobal('fetch', fetchMock);
171
+ const imported = await seedWorkspaceTar('https://relayfile.example/', 'token', 'rw_demo', projectDir, []);
172
+ expect(imported).toBe(2);
173
+ expect(execSyncMock).toHaveBeenCalledWith('git ls-files -z --cached --others --exclude-standard', expect.objectContaining({ cwd: path.resolve(projectDir), encoding: 'utf-8' }));
174
+ const [, init] = fetchMock.mock.calls[0];
175
+ const entries = await extractTarballEntries(init.body);
176
+ expect(entries).toEqual(['src/draft.ts', 'src/tracked.ts']);
177
+ expect(entries).not.toContain('ignored-by-git/skip.txt');
178
+ });
179
+ it('does not fall back to a directory walk when git ls-files succeeds with no files', async () => {
180
+ const projectDir = makeTempDir('relay-seed-project-');
181
+ fs.mkdirSync(path.join(projectDir, 'ignored-by-git'), { recursive: true });
182
+ fs.writeFileSync(path.join(projectDir, 'ignored-by-git', 'skip.txt'), 'skip\n');
183
+ execSyncMock.mockReturnValue('');
184
+ const fetchMock = vi.fn().mockResolvedValue(jsonResponse({ imported: 0 }));
185
+ vi.stubGlobal('fetch', fetchMock);
186
+ const imported = await seedWorkspaceTar('https://relayfile.example/', 'token', 'rw_demo', projectDir, []);
187
+ expect(imported).toBe(0);
188
+ expect(fetchMock).not.toHaveBeenCalled();
189
+ });
190
+ });
191
+ //# sourceMappingURL=tar-seeder.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tar-seeder.test.js","sourceRoot":"","sources":["../../../src/provisioner/__tests__/tar-seeder.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=token-factory.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token-factory.test.d.ts","sourceRoot":"","sources":["../../../src/provisioner/__tests__/token-factory.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,127 @@
1
+ import assert from 'node:assert/strict';
2
+ import { createHmac } from 'node:crypto';
3
+ import test from 'node:test';
4
+ import { DEFAULT_ADMIN_AGENT_NAME, DEFAULT_ADMIN_SCOPES, DEFAULT_WORKFLOW_TOKEN_TTL_SECONDS, WorkflowTokenFactory, mintAgentToken, } from '../token.js';
5
+ function decodeJwtPart(value) {
6
+ return JSON.parse(Buffer.from(value, 'base64url').toString('utf8'));
7
+ }
8
+ function decodeJwt(token) {
9
+ const [header, payload, signature] = token.split('.');
10
+ assert.ok(header);
11
+ assert.ok(payload);
12
+ assert.ok(signature);
13
+ return {
14
+ header: decodeJwtPart(header),
15
+ payload: decodeJwtPart(payload),
16
+ signature,
17
+ };
18
+ }
19
+ test('mintAgentToken returns a valid JWT', () => {
20
+ const token = mintAgentToken({
21
+ secret: 'test-secret',
22
+ agentName: 'worker',
23
+ workspace: 'workspace-123',
24
+ scopes: ['relayfile:fs:read:/src/index.ts'],
25
+ });
26
+ const parts = token.split('.');
27
+ const decoded = decodeJwt(token);
28
+ assert.equal(parts.length, 3);
29
+ assert.ok(parts.every((part) => /^[A-Za-z0-9_-]+$/u.test(part)));
30
+ assert.deepEqual(decoded.header, { alg: 'HS256', typ: 'JWT' });
31
+ assert.equal(decoded.payload.sub, 'agent_worker');
32
+ });
33
+ test('mintAgentToken payload contains agent_name, workspace, and scopes', () => {
34
+ const scopes = ['relayfile:fs:read:/src/index.ts', 'relayfile:fs:write:/src/index.ts'];
35
+ const token = mintAgentToken({
36
+ secret: 'test-secret',
37
+ agentName: 'compiler',
38
+ workspace: 'workspace-abc',
39
+ scopes,
40
+ });
41
+ const { payload } = decodeJwt(token);
42
+ assert.equal(payload.agent_name, 'compiler');
43
+ assert.equal(payload.wks, 'workspace-abc');
44
+ assert.equal(payload.workspace_id, 'workspace-abc');
45
+ assert.deepEqual(payload.scopes, scopes);
46
+ });
47
+ test('mintAgentToken defaults expiry to 2 hours', () => {
48
+ const token = mintAgentToken({
49
+ secret: 'test-secret',
50
+ agentName: 'worker',
51
+ workspace: 'workspace-123',
52
+ scopes: [],
53
+ });
54
+ const { payload } = decodeJwt(token);
55
+ assert.equal(payload.exp - payload.iat, DEFAULT_WORKFLOW_TOKEN_TTL_SECONDS);
56
+ assert.equal(payload.exp - payload.iat, 2 * 60 * 60);
57
+ });
58
+ test('mintAgentToken applies a custom TTL', () => {
59
+ const token = mintAgentToken({
60
+ secret: 'test-secret',
61
+ agentName: 'worker',
62
+ workspace: 'workspace-123',
63
+ scopes: [],
64
+ ttlSeconds: 90,
65
+ });
66
+ const { payload } = decodeJwt(token);
67
+ assert.equal(payload.exp - payload.iat, 90);
68
+ });
69
+ test('WorkflowTokenFactory mintAdmin uses the default admin identity and scopes', () => {
70
+ const factory = new WorkflowTokenFactory('test-secret', 'workspace-admin');
71
+ const token = factory.mintAdmin();
72
+ const { payload } = decodeJwt(token);
73
+ assert.equal(payload.agent_name, DEFAULT_ADMIN_AGENT_NAME);
74
+ assert.equal(payload.wks, 'workspace-admin');
75
+ assert.deepEqual(payload.scopes, DEFAULT_ADMIN_SCOPES);
76
+ });
77
+ test('WorkflowTokenFactory getToken returns the token minted for an agent', () => {
78
+ const factory = new WorkflowTokenFactory('test-secret', 'workspace-123');
79
+ const token = factory.mintForAgent('builder', ['relayfile:fs:read:/src/index.ts']);
80
+ assert.equal(factory.getToken('builder'), token);
81
+ });
82
+ test('WorkflowTokenFactory uses its configured TTL when minting agent tokens', () => {
83
+ const factory = new WorkflowTokenFactory('test-secret', 'workspace-123', 45);
84
+ const token = factory.mintForAgent('builder', []);
85
+ const { payload } = decodeJwt(token);
86
+ assert.equal(payload.exp - payload.iat, 45);
87
+ });
88
+ test('mintAgentToken generates a unique JTI per token', () => {
89
+ const first = decodeJwt(mintAgentToken({
90
+ secret: 'test-secret',
91
+ agentName: 'worker',
92
+ workspace: 'workspace-123',
93
+ scopes: [],
94
+ })).payload;
95
+ const second = decodeJwt(mintAgentToken({
96
+ secret: 'test-secret',
97
+ agentName: 'worker',
98
+ workspace: 'workspace-123',
99
+ scopes: [],
100
+ })).payload;
101
+ assert.notEqual(first.jti, second.jti);
102
+ assert.match(first.jti, /^tok-\d+-/u);
103
+ assert.match(second.jti, /^tok-\d+-/u);
104
+ });
105
+ test('mintAgentToken includes the expected audience claims', () => {
106
+ const token = mintAgentToken({
107
+ secret: 'test-secret',
108
+ agentName: 'worker',
109
+ workspace: 'workspace-123',
110
+ scopes: [],
111
+ });
112
+ const { payload } = decodeJwt(token);
113
+ assert.deepEqual(payload.aud, ['relayauth', 'relayfile']);
114
+ });
115
+ test('mintAgentToken signs tokens with HMAC-SHA256', () => {
116
+ const secret = 'test-secret';
117
+ const token = mintAgentToken({
118
+ secret,
119
+ agentName: 'worker',
120
+ workspace: 'workspace-123',
121
+ scopes: ['relayfile:fs:read:/src/index.ts'],
122
+ });
123
+ const [header, payload, signature] = token.split('.');
124
+ const expectedSignature = createHmac('sha256', secret).update(`${header}.${payload}`).digest('base64url');
125
+ assert.equal(signature, expectedSignature);
126
+ });
127
+ //# sourceMappingURL=token-factory.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token-factory.test.js","sourceRoot":"","sources":["../../../src/provisioner/__tests__/token-factory.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=token.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"token.test.d.ts","sourceRoot":"","sources":["../../../src/provisioner/__tests__/token.test.ts"],"names":[],"mappings":""}