@morten-olsen/builder-server 0.0.6
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/dist/app/app.d.ts +11 -0
- package/dist/app/app.d.ts.map +1 -0
- package/dist/app/app.js +141 -0
- package/dist/app/app.js.map +1 -0
- package/dist/app/app.routes.d.ts +4 -0
- package/dist/app/app.routes.d.ts.map +1 -0
- package/dist/app/app.routes.js +22 -0
- package/dist/app/app.routes.js.map +1 -0
- package/dist/app/app.types.d.ts +15 -0
- package/dist/app/app.types.d.ts.map +1 -0
- package/dist/app/app.types.js +2 -0
- package/dist/app/app.types.js.map +1 -0
- package/dist/config/config.d.ts +31 -0
- package/dist/config/config.d.ts.map +1 -0
- package/dist/config/config.js +93 -0
- package/dist/config/config.js.map +1 -0
- package/dist/config/config.testing.d.ts +4 -0
- package/dist/config/config.testing.d.ts.map +1 -0
- package/dist/config/config.testing.js +17 -0
- package/dist/config/config.testing.js.map +1 -0
- package/dist/container/container.d.ts +38 -0
- package/dist/container/container.d.ts.map +1 -0
- package/dist/container/container.js +33 -0
- package/dist/container/container.js.map +1 -0
- package/dist/container/container.test.d.ts +2 -0
- package/dist/container/container.test.d.ts.map +1 -0
- package/dist/container/container.test.js +65 -0
- package/dist/container/container.test.js.map +1 -0
- package/dist/exports.d.ts +67 -0
- package/dist/exports.d.ts.map +1 -0
- package/dist/exports.js +44 -0
- package/dist/exports.js.map +1 -0
- package/dist/routes/auth/auth.d.ts +4 -0
- package/dist/routes/auth/auth.d.ts.map +1 -0
- package/dist/routes/auth/auth.js +74 -0
- package/dist/routes/auth/auth.js.map +1 -0
- package/dist/routes/auth/auth.schemas.d.ts +42 -0
- package/dist/routes/auth/auth.schemas.d.ts.map +1 -0
- package/dist/routes/auth/auth.schemas.js +34 -0
- package/dist/routes/auth/auth.schemas.js.map +1 -0
- package/dist/routes/auth/auth.test.d.ts +2 -0
- package/dist/routes/auth/auth.test.d.ts.map +1 -0
- package/dist/routes/auth/auth.test.js +203 -0
- package/dist/routes/auth/auth.test.js.map +1 -0
- package/dist/routes/events/events.d.ts +4 -0
- package/dist/routes/events/events.d.ts.map +1 -0
- package/dist/routes/events/events.js +28 -0
- package/dist/routes/events/events.js.map +1 -0
- package/dist/routes/identities/identities.d.ts +4 -0
- package/dist/routes/identities/identities.d.ts.map +1 -0
- package/dist/routes/identities/identities.js +128 -0
- package/dist/routes/identities/identities.js.map +1 -0
- package/dist/routes/identities/identities.schemas.d.ts +52 -0
- package/dist/routes/identities/identities.schemas.d.ts.map +1 -0
- package/dist/routes/identities/identities.schemas.js +38 -0
- package/dist/routes/identities/identities.schemas.js.map +1 -0
- package/dist/routes/identities/identities.test.d.ts +2 -0
- package/dist/routes/identities/identities.test.d.ts.map +1 -0
- package/dist/routes/identities/identities.test.js +329 -0
- package/dist/routes/identities/identities.test.js.map +1 -0
- package/dist/routes/models/models.d.ts +4 -0
- package/dist/routes/models/models.d.ts.map +1 -0
- package/dist/routes/models/models.js +60 -0
- package/dist/routes/models/models.js.map +1 -0
- package/dist/routes/models/models.schemas.d.ts +16 -0
- package/dist/routes/models/models.schemas.d.ts.map +1 -0
- package/dist/routes/models/models.schemas.js +9 -0
- package/dist/routes/models/models.schemas.js.map +1 -0
- package/dist/routes/notifications/notifications.d.ts +4 -0
- package/dist/routes/notifications/notifications.d.ts.map +1 -0
- package/dist/routes/notifications/notifications.js +226 -0
- package/dist/routes/notifications/notifications.js.map +1 -0
- package/dist/routes/notifications/notifications.schemas.d.ts +61 -0
- package/dist/routes/notifications/notifications.schemas.d.ts.map +1 -0
- package/dist/routes/notifications/notifications.schemas.js +44 -0
- package/dist/routes/notifications/notifications.schemas.js.map +1 -0
- package/dist/routes/notifications/notifications.test.d.ts +2 -0
- package/dist/routes/notifications/notifications.test.d.ts.map +1 -0
- package/dist/routes/notifications/notifications.test.js +44 -0
- package/dist/routes/notifications/notifications.test.js.map +1 -0
- package/dist/routes/repos/repos.d.ts +4 -0
- package/dist/routes/repos/repos.d.ts.map +1 -0
- package/dist/routes/repos/repos.js +142 -0
- package/dist/routes/repos/repos.js.map +1 -0
- package/dist/routes/repos/repos.schemas.d.ts +47 -0
- package/dist/routes/repos/repos.schemas.d.ts.map +1 -0
- package/dist/routes/repos/repos.schemas.js +32 -0
- package/dist/routes/repos/repos.schemas.js.map +1 -0
- package/dist/routes/sessions/sessions.d.ts +4 -0
- package/dist/routes/sessions/sessions.d.ts.map +1 -0
- package/dist/routes/sessions/sessions.js +299 -0
- package/dist/routes/sessions/sessions.js.map +1 -0
- package/dist/routes/sessions/sessions.review.d.ts +4 -0
- package/dist/routes/sessions/sessions.review.d.ts.map +1 -0
- package/dist/routes/sessions/sessions.review.js +294 -0
- package/dist/routes/sessions/sessions.review.js.map +1 -0
- package/dist/routes/sessions/sessions.review.schemas.d.ts +83 -0
- package/dist/routes/sessions/sessions.review.schemas.d.ts.map +1 -0
- package/dist/routes/sessions/sessions.review.schemas.js +55 -0
- package/dist/routes/sessions/sessions.review.schemas.js.map +1 -0
- package/dist/routes/sessions/sessions.schemas.d.ts +81 -0
- package/dist/routes/sessions/sessions.schemas.d.ts.map +1 -0
- package/dist/routes/sessions/sessions.schemas.js +50 -0
- package/dist/routes/sessions/sessions.schemas.js.map +1 -0
- package/dist/routes/sessions/sessions.test.d.ts +2 -0
- package/dist/routes/sessions/sessions.test.d.ts.map +1 -0
- package/dist/routes/sessions/sessions.test.js +297 -0
- package/dist/routes/sessions/sessions.test.js.map +1 -0
- package/dist/routes/ws/ws.d.ts +4 -0
- package/dist/routes/ws/ws.d.ts.map +1 -0
- package/dist/routes/ws/ws.js +8 -0
- package/dist/routes/ws/ws.js.map +1 -0
- package/dist/services/agent/agent.claude.d.ts +4 -0
- package/dist/services/agent/agent.claude.d.ts.map +1 -0
- package/dist/services/agent/agent.claude.js +141 -0
- package/dist/services/agent/agent.claude.js.map +1 -0
- package/dist/services/agent/agent.claude.test.d.ts +2 -0
- package/dist/services/agent/agent.claude.test.d.ts.map +1 -0
- package/dist/services/agent/agent.claude.test.js +43 -0
- package/dist/services/agent/agent.claude.test.js.map +1 -0
- package/dist/services/agent/agent.d.ts +58 -0
- package/dist/services/agent/agent.d.ts.map +1 -0
- package/dist/services/agent/agent.errors.d.ts +8 -0
- package/dist/services/agent/agent.errors.d.ts.map +1 -0
- package/dist/services/agent/agent.errors.js +14 -0
- package/dist/services/agent/agent.errors.js.map +1 -0
- package/dist/services/agent/agent.js +21 -0
- package/dist/services/agent/agent.js.map +1 -0
- package/dist/services/agent/agent.queue.d.ts +8 -0
- package/dist/services/agent/agent.queue.d.ts.map +1 -0
- package/dist/services/agent/agent.queue.js +46 -0
- package/dist/services/agent/agent.queue.js.map +1 -0
- package/dist/services/auth/auth.d.ts +41 -0
- package/dist/services/auth/auth.d.ts.map +1 -0
- package/dist/services/auth/auth.errors.d.ts +17 -0
- package/dist/services/auth/auth.errors.d.ts.map +1 -0
- package/dist/services/auth/auth.errors.js +32 -0
- package/dist/services/auth/auth.errors.js.map +1 -0
- package/dist/services/auth/auth.js +147 -0
- package/dist/services/auth/auth.js.map +1 -0
- package/dist/services/auth/auth.test.d.ts +2 -0
- package/dist/services/auth/auth.test.d.ts.map +1 -0
- package/dist/services/auth/auth.test.js +55 -0
- package/dist/services/auth/auth.test.js.map +1 -0
- package/dist/services/auth/auth.utils.d.ts +4 -0
- package/dist/services/auth/auth.utils.d.ts.map +1 -0
- package/dist/services/auth/auth.utils.js +15 -0
- package/dist/services/auth/auth.utils.js.map +1 -0
- package/dist/services/auth/auth.utils.test.d.ts +2 -0
- package/dist/services/auth/auth.utils.test.d.ts.map +1 -0
- package/dist/services/auth/auth.utils.test.js +20 -0
- package/dist/services/auth/auth.utils.test.js.map +1 -0
- package/dist/services/database/database.d.ts +102 -0
- package/dist/services/database/database.d.ts.map +1 -0
- package/dist/services/database/database.js +41 -0
- package/dist/services/database/database.js.map +1 -0
- package/dist/services/database/database.migrations.d.ts +4 -0
- package/dist/services/database/database.migrations.d.ts.map +1 -0
- package/dist/services/database/database.migrations.js +290 -0
- package/dist/services/database/database.migrations.js.map +1 -0
- package/dist/services/database/database.test.d.ts +2 -0
- package/dist/services/database/database.test.d.ts.map +1 -0
- package/dist/services/database/database.test.js +60 -0
- package/dist/services/database/database.test.js.map +1 -0
- package/dist/services/file-review/file-review.d.ts +34 -0
- package/dist/services/file-review/file-review.d.ts.map +1 -0
- package/dist/services/file-review/file-review.js +71 -0
- package/dist/services/file-review/file-review.js.map +1 -0
- package/dist/services/file-review/file-review.test.d.ts +2 -0
- package/dist/services/file-review/file-review.test.d.ts.map +1 -0
- package/dist/services/file-review/file-review.test.js +171 -0
- package/dist/services/file-review/file-review.test.js.map +1 -0
- package/dist/services/git/git.d.ts +95 -0
- package/dist/services/git/git.d.ts.map +1 -0
- package/dist/services/git/git.errors.d.ts +20 -0
- package/dist/services/git/git.errors.d.ts.map +1 -0
- package/dist/services/git/git.errors.js +38 -0
- package/dist/services/git/git.errors.js.map +1 -0
- package/dist/services/git/git.js +341 -0
- package/dist/services/git/git.js.map +1 -0
- package/dist/services/git/git.test.d.ts +2 -0
- package/dist/services/git/git.test.d.ts.map +1 -0
- package/dist/services/git/git.test.js +304 -0
- package/dist/services/git/git.test.js.map +1 -0
- package/dist/services/identity/identity.d.ts +49 -0
- package/dist/services/identity/identity.d.ts.map +1 -0
- package/dist/services/identity/identity.errors.d.ts +11 -0
- package/dist/services/identity/identity.errors.d.ts.map +1 -0
- package/dist/services/identity/identity.errors.js +20 -0
- package/dist/services/identity/identity.errors.js.map +1 -0
- package/dist/services/identity/identity.js +196 -0
- package/dist/services/identity/identity.js.map +1 -0
- package/dist/services/identity/identity.test.d.ts +2 -0
- package/dist/services/identity/identity.test.d.ts.map +1 -0
- package/dist/services/identity/identity.test.js +184 -0
- package/dist/services/identity/identity.test.js.map +1 -0
- package/dist/services/identity/identity.utils.d.ts +9 -0
- package/dist/services/identity/identity.utils.d.ts.map +1 -0
- package/dist/services/identity/identity.utils.js +47 -0
- package/dist/services/identity/identity.utils.js.map +1 -0
- package/dist/services/identity/identity.utils.test.d.ts +2 -0
- package/dist/services/identity/identity.utils.test.d.ts.map +1 -0
- package/dist/services/identity/identity.utils.test.js +49 -0
- package/dist/services/identity/identity.utils.test.js.map +1 -0
- package/dist/services/message/message.d.ts +29 -0
- package/dist/services/message/message.d.ts.map +1 -0
- package/dist/services/message/message.js +85 -0
- package/dist/services/message/message.js.map +1 -0
- package/dist/services/notification/notification.d.ts +75 -0
- package/dist/services/notification/notification.d.ts.map +1 -0
- package/dist/services/notification/notification.errors.d.ts +14 -0
- package/dist/services/notification/notification.errors.d.ts.map +1 -0
- package/dist/services/notification/notification.errors.js +26 -0
- package/dist/services/notification/notification.errors.js.map +1 -0
- package/dist/services/notification/notification.js +212 -0
- package/dist/services/notification/notification.js.map +1 -0
- package/dist/services/notification/notification.ntfy.d.ts +12 -0
- package/dist/services/notification/notification.ntfy.d.ts.map +1 -0
- package/dist/services/notification/notification.ntfy.js +44 -0
- package/dist/services/notification/notification.ntfy.js.map +1 -0
- package/dist/services/repo/repo.d.ts +46 -0
- package/dist/services/repo/repo.d.ts.map +1 -0
- package/dist/services/repo/repo.errors.d.ts +11 -0
- package/dist/services/repo/repo.errors.d.ts.map +1 -0
- package/dist/services/repo/repo.errors.js +20 -0
- package/dist/services/repo/repo.errors.js.map +1 -0
- package/dist/services/repo/repo.js +138 -0
- package/dist/services/repo/repo.js.map +1 -0
- package/dist/services/session/session.d.ts +62 -0
- package/dist/services/session/session.d.ts.map +1 -0
- package/dist/services/session/session.errors.d.ts +11 -0
- package/dist/services/session/session.errors.d.ts.map +1 -0
- package/dist/services/session/session.errors.js +20 -0
- package/dist/services/session/session.errors.js.map +1 -0
- package/dist/services/session/session.js +168 -0
- package/dist/services/session/session.js.map +1 -0
- package/dist/services/session/session.runner.d.ts +8 -0
- package/dist/services/session/session.runner.d.ts.map +1 -0
- package/dist/services/session/session.runner.js +328 -0
- package/dist/services/session/session.runner.js.map +1 -0
- package/dist/services/session/session.runner.test.d.ts +2 -0
- package/dist/services/session/session.runner.test.d.ts.map +1 -0
- package/dist/services/session/session.runner.test.js +143 -0
- package/dist/services/session/session.runner.test.js.map +1 -0
- package/dist/services/session/session.test.d.ts +2 -0
- package/dist/services/session/session.test.d.ts.map +1 -0
- package/dist/services/session/session.test.js +146 -0
- package/dist/services/session/session.test.js.map +1 -0
- package/dist/services/session-event/session-event.d.ts +28 -0
- package/dist/services/session-event/session-event.d.ts.map +1 -0
- package/dist/services/session-event/session-event.js +86 -0
- package/dist/services/session-event/session-event.js.map +1 -0
- package/dist/sse/event-bus.d.ts +74 -0
- package/dist/sse/event-bus.d.ts.map +1 -0
- package/dist/sse/event-bus.js +137 -0
- package/dist/sse/event-bus.js.map +1 -0
- package/dist/sse/stream.d.ts +20 -0
- package/dist/sse/stream.d.ts.map +1 -0
- package/dist/sse/stream.js +66 -0
- package/dist/sse/stream.js.map +1 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/dist/utils/crypto.d.ts +4 -0
- package/dist/utils/crypto.d.ts.map +1 -0
- package/dist/utils/crypto.js +21 -0
- package/dist/utils/crypto.js.map +1 -0
- package/dist/ws/ws.d.ts +30 -0
- package/dist/ws/ws.d.ts.map +1 -0
- package/dist/ws/ws.js +137 -0
- package/dist/ws/ws.js.map +1 -0
- package/package.json +50 -0
|
@@ -0,0 +1,184 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { createTestConfig } from '../../config/config.testing.js';
|
|
3
|
+
import { Services, destroy } from '../../container/container.js';
|
|
4
|
+
import { AuthService } from '../auth/auth.js';
|
|
5
|
+
import { IdentityNotFoundError } from './identity.errors.js';
|
|
6
|
+
import { IdentityService } from './identity.js';
|
|
7
|
+
describe('IdentityService', () => {
|
|
8
|
+
let services;
|
|
9
|
+
let identity;
|
|
10
|
+
let userId;
|
|
11
|
+
beforeEach(async () => {
|
|
12
|
+
services = new Services(createTestConfig());
|
|
13
|
+
identity = services.get(IdentityService);
|
|
14
|
+
const auth = services.get(AuthService);
|
|
15
|
+
const { user } = await auth.register({ email: 'test@example.com', password: 'password123' });
|
|
16
|
+
userId = user.id;
|
|
17
|
+
});
|
|
18
|
+
afterEach(async () => {
|
|
19
|
+
await services[destroy]();
|
|
20
|
+
});
|
|
21
|
+
describe('create', () => {
|
|
22
|
+
it('creates an identity with generated keys', async () => {
|
|
23
|
+
const result = await identity.create({
|
|
24
|
+
userId,
|
|
25
|
+
name: 'Work',
|
|
26
|
+
gitAuthorName: 'Alice',
|
|
27
|
+
gitAuthorEmail: 'alice@work.com',
|
|
28
|
+
});
|
|
29
|
+
expect(result.id).toBeTruthy();
|
|
30
|
+
expect(result.name).toBe('Work');
|
|
31
|
+
expect(result.gitAuthorName).toBe('Alice');
|
|
32
|
+
expect(result.gitAuthorEmail).toBe('alice@work.com');
|
|
33
|
+
expect(result.publicKey).toMatch(/^ssh-ed25519 /);
|
|
34
|
+
expect(result.userId).toBe(userId);
|
|
35
|
+
});
|
|
36
|
+
it('creates an identity with imported keys', async () => {
|
|
37
|
+
const result = await identity.create({
|
|
38
|
+
userId,
|
|
39
|
+
name: 'Imported',
|
|
40
|
+
gitAuthorName: 'Bob',
|
|
41
|
+
gitAuthorEmail: 'bob@example.com',
|
|
42
|
+
publicKey: 'ssh-ed25519 AAAA...',
|
|
43
|
+
privateKey: 'private-key-data',
|
|
44
|
+
});
|
|
45
|
+
expect(result.publicKey).toBe('ssh-ed25519 AAAA...');
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
describe('list', () => {
|
|
49
|
+
it('returns all identities for the user', async () => {
|
|
50
|
+
await identity.create({
|
|
51
|
+
userId,
|
|
52
|
+
name: 'First',
|
|
53
|
+
gitAuthorName: 'A',
|
|
54
|
+
gitAuthorEmail: 'a@test.com',
|
|
55
|
+
});
|
|
56
|
+
await identity.create({
|
|
57
|
+
userId,
|
|
58
|
+
name: 'Second',
|
|
59
|
+
gitAuthorName: 'B',
|
|
60
|
+
gitAuthorEmail: 'b@test.com',
|
|
61
|
+
});
|
|
62
|
+
const list = await identity.list(userId);
|
|
63
|
+
expect(list).toHaveLength(2);
|
|
64
|
+
const names = list.map((i) => i.name);
|
|
65
|
+
expect(names).toContain('First');
|
|
66
|
+
expect(names).toContain('Second');
|
|
67
|
+
});
|
|
68
|
+
it('returns empty array for user with no identities', async () => {
|
|
69
|
+
const list = await identity.list(userId);
|
|
70
|
+
expect(list).toEqual([]);
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
describe('get', () => {
|
|
74
|
+
it('returns a single identity', async () => {
|
|
75
|
+
const created = await identity.create({
|
|
76
|
+
userId,
|
|
77
|
+
name: 'Test',
|
|
78
|
+
gitAuthorName: 'Alice',
|
|
79
|
+
gitAuthorEmail: 'alice@test.com',
|
|
80
|
+
});
|
|
81
|
+
const result = await identity.get({ userId, identityId: created.id });
|
|
82
|
+
expect(result.id).toBe(created.id);
|
|
83
|
+
expect(result.name).toBe('Test');
|
|
84
|
+
});
|
|
85
|
+
it('throws IdentityNotFoundError for non-existent identity', async () => {
|
|
86
|
+
await expect(identity.get({ userId, identityId: 'non-existent' })).rejects.toThrow(IdentityNotFoundError);
|
|
87
|
+
});
|
|
88
|
+
it('throws IdentityNotFoundError for another user\'s identity', async () => {
|
|
89
|
+
const created = await identity.create({
|
|
90
|
+
userId,
|
|
91
|
+
name: 'Mine',
|
|
92
|
+
gitAuthorName: 'Alice',
|
|
93
|
+
gitAuthorEmail: 'alice@test.com',
|
|
94
|
+
});
|
|
95
|
+
const auth = services.get(AuthService);
|
|
96
|
+
const { user: otherUser } = await auth.register({ email: 'other@example.com', password: 'password123' });
|
|
97
|
+
await expect(identity.get({ userId: otherUser.id, identityId: created.id })).rejects.toThrow(IdentityNotFoundError);
|
|
98
|
+
});
|
|
99
|
+
});
|
|
100
|
+
describe('update', () => {
|
|
101
|
+
it('updates name and git author fields', async () => {
|
|
102
|
+
const created = await identity.create({
|
|
103
|
+
userId,
|
|
104
|
+
name: 'Old',
|
|
105
|
+
gitAuthorName: 'Old Name',
|
|
106
|
+
gitAuthorEmail: 'old@test.com',
|
|
107
|
+
});
|
|
108
|
+
const updated = await identity.update({
|
|
109
|
+
userId,
|
|
110
|
+
identityId: created.id,
|
|
111
|
+
name: 'New',
|
|
112
|
+
gitAuthorName: 'New Name',
|
|
113
|
+
gitAuthorEmail: 'new@test.com',
|
|
114
|
+
});
|
|
115
|
+
expect(updated.name).toBe('New');
|
|
116
|
+
expect(updated.gitAuthorName).toBe('New Name');
|
|
117
|
+
expect(updated.gitAuthorEmail).toBe('new@test.com');
|
|
118
|
+
expect(updated.publicKey).toBe(created.publicKey);
|
|
119
|
+
});
|
|
120
|
+
it('partially updates only provided fields', async () => {
|
|
121
|
+
const created = await identity.create({
|
|
122
|
+
userId,
|
|
123
|
+
name: 'Original',
|
|
124
|
+
gitAuthorName: 'Name',
|
|
125
|
+
gitAuthorEmail: 'email@test.com',
|
|
126
|
+
});
|
|
127
|
+
const updated = await identity.update({
|
|
128
|
+
userId,
|
|
129
|
+
identityId: created.id,
|
|
130
|
+
name: 'Updated',
|
|
131
|
+
});
|
|
132
|
+
expect(updated.name).toBe('Updated');
|
|
133
|
+
expect(updated.gitAuthorName).toBe('Name');
|
|
134
|
+
expect(updated.gitAuthorEmail).toBe('email@test.com');
|
|
135
|
+
});
|
|
136
|
+
it('throws IdentityNotFoundError for non-existent identity', async () => {
|
|
137
|
+
await expect(identity.update({ userId, identityId: 'non-existent', name: 'X' })).rejects.toThrow(IdentityNotFoundError);
|
|
138
|
+
});
|
|
139
|
+
});
|
|
140
|
+
describe('delete', () => {
|
|
141
|
+
it('deletes an identity', async () => {
|
|
142
|
+
const created = await identity.create({
|
|
143
|
+
userId,
|
|
144
|
+
name: 'ToDelete',
|
|
145
|
+
gitAuthorName: 'A',
|
|
146
|
+
gitAuthorEmail: 'a@test.com',
|
|
147
|
+
});
|
|
148
|
+
await identity.delete({ userId, identityId: created.id });
|
|
149
|
+
await expect(identity.get({ userId, identityId: created.id })).rejects.toThrow(IdentityNotFoundError);
|
|
150
|
+
});
|
|
151
|
+
it('throws IdentityNotFoundError when deleting non-existent identity', async () => {
|
|
152
|
+
await expect(identity.delete({ userId, identityId: 'non-existent' })).rejects.toThrow(IdentityNotFoundError);
|
|
153
|
+
});
|
|
154
|
+
});
|
|
155
|
+
describe('getPrivateKey', () => {
|
|
156
|
+
it('decrypts and returns the private key', async () => {
|
|
157
|
+
const created = await identity.create({
|
|
158
|
+
userId,
|
|
159
|
+
name: 'WithKey',
|
|
160
|
+
gitAuthorName: 'Alice',
|
|
161
|
+
gitAuthorEmail: 'alice@test.com',
|
|
162
|
+
});
|
|
163
|
+
const privateKey = await identity.getPrivateKey({ userId, identityId: created.id });
|
|
164
|
+
expect(privateKey).toContain('-----BEGIN OPENSSH PRIVATE KEY-----');
|
|
165
|
+
});
|
|
166
|
+
it('returns imported private key after decryption', async () => {
|
|
167
|
+
await identity.create({
|
|
168
|
+
userId,
|
|
169
|
+
name: 'Imported',
|
|
170
|
+
gitAuthorName: 'Bob',
|
|
171
|
+
gitAuthorEmail: 'bob@test.com',
|
|
172
|
+
publicKey: 'pub-key',
|
|
173
|
+
privateKey: 'my-secret-private-key',
|
|
174
|
+
});
|
|
175
|
+
const list = await identity.list(userId);
|
|
176
|
+
const privateKey = await identity.getPrivateKey({ userId, identityId: list[0].id });
|
|
177
|
+
expect(privateKey).toBe('my-secret-private-key');
|
|
178
|
+
});
|
|
179
|
+
it('throws IdentityNotFoundError for non-existent identity', async () => {
|
|
180
|
+
await expect(identity.getPrivateKey({ userId, identityId: 'non-existent' })).rejects.toThrow(IdentityNotFoundError);
|
|
181
|
+
});
|
|
182
|
+
});
|
|
183
|
+
});
|
|
184
|
+
//# sourceMappingURL=identity.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"identity.test.js","sourceRoot":"","sources":["../../../src/services/identity/identity.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
declare const encryptPrivateKey: (plaintext: string, keyHex: string) => string;
|
|
2
|
+
declare const decryptPrivateKey: (stored: string, keyHex: string) => string;
|
|
3
|
+
declare const generateSshKeyPair: () => {
|
|
4
|
+
publicKey: string;
|
|
5
|
+
privateKey: string;
|
|
6
|
+
};
|
|
7
|
+
declare const derivePublicKeyFromPrivate: (privateKey: string) => string;
|
|
8
|
+
export { encryptPrivateKey, decryptPrivateKey, generateSshKeyPair, derivePublicKeyFromPrivate };
|
|
9
|
+
//# sourceMappingURL=identity.utils.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"identity.utils.d.ts","sourceRoot":"","sources":["../../../src/services/identity/identity.utils.ts"],"names":[],"mappings":"AAUA,QAAA,MAAM,iBAAiB,+CAAU,CAAC;AAClC,QAAA,MAAM,iBAAiB,4CAAU,CAAC;AAElC,QAAA,MAAM,kBAAkB,QAAO;IAAE,SAAS,EAAE,MAAM,CAAC;IAAC,UAAU,EAAE,MAAM,CAAA;CAWrE,CAAC;AAEF,QAAA,MAAM,0BAA0B,GAAI,YAAY,MAAM,KAAG,MAWxD,CAAC;AAEF,OAAO,EAAE,iBAAiB,EAAE,iBAAiB,EAAE,kBAAkB,EAAE,0BAA0B,EAAE,CAAC"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { execSync } from 'node:child_process';
|
|
2
|
+
import { readFileSync, unlinkSync, writeFileSync } from 'node:fs';
|
|
3
|
+
import { tmpdir } from 'node:os';
|
|
4
|
+
import { join } from 'node:path';
|
|
5
|
+
import { randomBytes } from 'node:crypto';
|
|
6
|
+
import { encrypt, decrypt } from '../../utils/crypto.js';
|
|
7
|
+
import { IdentityError } from './identity.errors.js';
|
|
8
|
+
const encryptPrivateKey = encrypt;
|
|
9
|
+
const decryptPrivateKey = decrypt;
|
|
10
|
+
const generateSshKeyPair = () => {
|
|
11
|
+
const keyPath = join(tmpdir(), `builder-keygen-${randomBytes(8).toString('hex')}`);
|
|
12
|
+
try {
|
|
13
|
+
execSync(`ssh-keygen -t ed25519 -f ${keyPath} -N "" -q -C ""`, { stdio: 'pipe' });
|
|
14
|
+
const privateKey = readFileSync(keyPath, 'utf8');
|
|
15
|
+
const publicKey = readFileSync(`${keyPath}.pub`, 'utf8').trim().replace(/\s+$/, '');
|
|
16
|
+
return { publicKey, privateKey };
|
|
17
|
+
}
|
|
18
|
+
finally {
|
|
19
|
+
try {
|
|
20
|
+
unlinkSync(keyPath);
|
|
21
|
+
}
|
|
22
|
+
catch { /* ignore */ }
|
|
23
|
+
try {
|
|
24
|
+
unlinkSync(`${keyPath}.pub`);
|
|
25
|
+
}
|
|
26
|
+
catch { /* ignore */ }
|
|
27
|
+
}
|
|
28
|
+
};
|
|
29
|
+
const derivePublicKeyFromPrivate = (privateKey) => {
|
|
30
|
+
const keyPath = join(tmpdir(), `builder-derive-${randomBytes(8).toString('hex')}`);
|
|
31
|
+
try {
|
|
32
|
+
writeFileSync(keyPath, privateKey, { mode: 0o600 });
|
|
33
|
+
const publicKey = execSync(`ssh-keygen -y -f ${keyPath}`, { stdio: 'pipe' }).toString().trim();
|
|
34
|
+
return publicKey;
|
|
35
|
+
}
|
|
36
|
+
catch {
|
|
37
|
+
throw new IdentityError('Invalid private key format');
|
|
38
|
+
}
|
|
39
|
+
finally {
|
|
40
|
+
try {
|
|
41
|
+
unlinkSync(keyPath);
|
|
42
|
+
}
|
|
43
|
+
catch { /* ignore */ }
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
export { encryptPrivateKey, decryptPrivateKey, generateSshKeyPair, derivePublicKeyFromPrivate };
|
|
47
|
+
//# sourceMappingURL=identity.utils.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"identity.utils.js","sourceRoot":"","sources":["../../../src/services/identity/identity.utils.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,oBAAoB,CAAC;AAC9C,OAAO,EAAE,YAAY,EAAE,UAAU,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAClE,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AACjC,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAE1C,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,uBAAuB,CAAC;AAEzD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AAErD,MAAM,iBAAiB,GAAG,OAAO,CAAC;AAClC,MAAM,iBAAiB,GAAG,OAAO,CAAC;AAElC,MAAM,kBAAkB,GAAG,GAA8C,EAAE;IACzE,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,kBAAkB,WAAW,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IACnF,IAAI,CAAC;QACH,QAAQ,CAAC,4BAA4B,OAAO,iBAAiB,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC;QAClF,MAAM,UAAU,GAAG,YAAY,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QACjD,MAAM,SAAS,GAAG,YAAY,CAAC,GAAG,OAAO,MAAM,EAAE,MAAM,CAAC,CAAC,IAAI,EAAE,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;QACpF,OAAO,EAAE,SAAS,EAAE,UAAU,EAAE,CAAC;IACnC,CAAC;YAAS,CAAC;QACT,IAAI,CAAC;YAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;QACnD,IAAI,CAAC;YAAC,UAAU,CAAC,GAAG,OAAO,MAAM,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;IAC9D,CAAC;AACH,CAAC,CAAC;AAEF,MAAM,0BAA0B,GAAG,CAAC,UAAkB,EAAU,EAAE;IAChE,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,EAAE,EAAE,kBAAkB,WAAW,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IACnF,IAAI,CAAC;QACH,aAAa,CAAC,OAAO,EAAE,UAAU,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;QACpD,MAAM,SAAS,GAAG,QAAQ,CAAC,oBAAoB,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC,IAAI,EAAE,CAAC;QAC/F,OAAO,SAAS,CAAC;IACnB,CAAC;IAAC,MAAM,CAAC;QACP,MAAM,IAAI,aAAa,CAAC,4BAA4B,CAAC,CAAC;IACxD,CAAC;YAAS,CAAC;QACT,IAAI,CAAC;YAAC,UAAU,CAAC,OAAO,CAAC,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;IACrD,CAAC;AACH,CAAC,CAAC;AAEF,OAAO,EAAE,iBAAiB,EAAE,iBAAiB,EAAE,kBAAkB,EAAE,0BAA0B,EAAE,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"identity.utils.test.d.ts","sourceRoot":"","sources":["../../../src/services/identity/identity.utils.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { randomBytes } from 'node:crypto';
|
|
2
|
+
import { describe, it, expect } from 'vitest';
|
|
3
|
+
import { encryptPrivateKey, decryptPrivateKey, generateSshKeyPair } from './identity.utils.js';
|
|
4
|
+
const testKey = 'a'.repeat(64);
|
|
5
|
+
describe('identity utils', () => {
|
|
6
|
+
describe('encryptPrivateKey / decryptPrivateKey', () => {
|
|
7
|
+
it('round-trips plaintext through encrypt and decrypt', () => {
|
|
8
|
+
const plaintext = 'my-secret-private-key-data';
|
|
9
|
+
const encrypted = encryptPrivateKey(plaintext, testKey);
|
|
10
|
+
const decrypted = decryptPrivateKey(encrypted, testKey);
|
|
11
|
+
expect(decrypted).toBe(plaintext);
|
|
12
|
+
});
|
|
13
|
+
it('produces format iv:authTag:ciphertext', () => {
|
|
14
|
+
const encrypted = encryptPrivateKey('test', testKey);
|
|
15
|
+
const parts = encrypted.split(':');
|
|
16
|
+
expect(parts).toHaveLength(3);
|
|
17
|
+
expect(parts[0]).toMatch(/^[0-9a-f]{24}$/); // 12 bytes = 24 hex chars
|
|
18
|
+
expect(parts[1]).toMatch(/^[0-9a-f]{32}$/); // 16 bytes = 32 hex chars
|
|
19
|
+
expect(parts[2]).toMatch(/^[0-9a-f]+$/);
|
|
20
|
+
});
|
|
21
|
+
it('produces different ciphertext each time (random IV)', () => {
|
|
22
|
+
const plaintext = 'same-data';
|
|
23
|
+
const a = encryptPrivateKey(plaintext, testKey);
|
|
24
|
+
const b = encryptPrivateKey(plaintext, testKey);
|
|
25
|
+
expect(a).not.toBe(b);
|
|
26
|
+
expect(decryptPrivateKey(a, testKey)).toBe(plaintext);
|
|
27
|
+
expect(decryptPrivateKey(b, testKey)).toBe(plaintext);
|
|
28
|
+
});
|
|
29
|
+
it('fails to decrypt with a wrong key', () => {
|
|
30
|
+
const encrypted = encryptPrivateKey('secret', testKey);
|
|
31
|
+
const wrongKey = randomBytes(32).toString('hex');
|
|
32
|
+
expect(() => decryptPrivateKey(encrypted, wrongKey)).toThrow();
|
|
33
|
+
});
|
|
34
|
+
});
|
|
35
|
+
describe('generateSshKeyPair', () => {
|
|
36
|
+
it('generates a key pair in OpenSSH format', () => {
|
|
37
|
+
const { publicKey, privateKey } = generateSshKeyPair();
|
|
38
|
+
expect(publicKey).toMatch(/^ssh-ed25519 [A-Za-z0-9+/=]+/);
|
|
39
|
+
expect(privateKey).toContain('-----BEGIN OPENSSH PRIVATE KEY-----');
|
|
40
|
+
});
|
|
41
|
+
it('generates unique key pairs each time', () => {
|
|
42
|
+
const a = generateSshKeyPair();
|
|
43
|
+
const b = generateSshKeyPair();
|
|
44
|
+
expect(a.publicKey).not.toBe(b.publicKey);
|
|
45
|
+
expect(a.privateKey).not.toBe(b.privateKey);
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
//# sourceMappingURL=identity.utils.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"identity.utils.test.js","sourceRoot":"","sources":["../../../src/services/identity/identity.utils.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import type { Services } from '../../container/container.js';
|
|
2
|
+
type Message = {
|
|
3
|
+
id: string;
|
|
4
|
+
sessionId: string;
|
|
5
|
+
role: 'user' | 'assistant';
|
|
6
|
+
content: string;
|
|
7
|
+
commitSha: string | null;
|
|
8
|
+
createdAt: string;
|
|
9
|
+
};
|
|
10
|
+
type CreateMessageInput = {
|
|
11
|
+
sessionId: string;
|
|
12
|
+
role: 'user' | 'assistant';
|
|
13
|
+
content: string;
|
|
14
|
+
commitSha?: string;
|
|
15
|
+
};
|
|
16
|
+
declare class MessageService {
|
|
17
|
+
#private;
|
|
18
|
+
constructor(services: Services);
|
|
19
|
+
create: (input: CreateMessageInput) => Promise<Message>;
|
|
20
|
+
getById: (messageId: string) => Promise<Message>;
|
|
21
|
+
deleteAfter: (input: {
|
|
22
|
+
sessionId: string;
|
|
23
|
+
messageId: string;
|
|
24
|
+
}) => Promise<void>;
|
|
25
|
+
listBySession: (sessionId: string) => Promise<Message[]>;
|
|
26
|
+
}
|
|
27
|
+
export type { Message, CreateMessageInput };
|
|
28
|
+
export { MessageService };
|
|
29
|
+
//# sourceMappingURL=message.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"message.d.ts","sourceRoot":"","sources":["../../../src/services/message/message.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,8BAA8B,CAAC;AAG7D,KAAK,OAAO,GAAG;IACb,EAAE,EAAE,MAAM,CAAC;IACX,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC;IAC3B,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,SAAS,EAAE,MAAM,CAAC;CACnB,CAAC;AAEF,KAAK,kBAAkB,GAAG;IACxB,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,GAAG,WAAW,CAAC;IAC3B,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB,CAAC;AAkBF,cAAM,cAAc;;gBAGN,QAAQ,EAAE,QAAQ;IAQ9B,MAAM,GAAU,OAAO,kBAAkB,KAAG,OAAO,CAAC,OAAO,CAAC,CA0B1D;IAEF,OAAO,GAAU,WAAW,MAAM,KAAG,OAAO,CAAC,OAAO,CAAC,CAcnD;IAEF,WAAW,GAAU,OAAO;QAAE,SAAS,EAAE,MAAM,CAAC;QAAC,SAAS,EAAE,MAAM,CAAA;KAAE,KAAG,OAAO,CAAC,IAAI,CAAC,CAmBlF;IAEF,aAAa,GAAU,WAAW,MAAM,KAAG,OAAO,CAAC,OAAO,EAAE,CAAC,CAW3D;CACH;AAED,YAAY,EAAE,OAAO,EAAE,kBAAkB,EAAE,CAAC;AAC5C,OAAO,EAAE,cAAc,EAAE,CAAC"}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { randomUUID } from 'node:crypto';
|
|
2
|
+
import { DatabaseService } from '../database/database.js';
|
|
3
|
+
const mapRow = (row) => ({
|
|
4
|
+
id: row.id,
|
|
5
|
+
sessionId: row.session_id,
|
|
6
|
+
role: row.role,
|
|
7
|
+
content: row.content,
|
|
8
|
+
commitSha: row.commit_sha,
|
|
9
|
+
createdAt: row.created_at,
|
|
10
|
+
});
|
|
11
|
+
class MessageService {
|
|
12
|
+
#services;
|
|
13
|
+
constructor(services) {
|
|
14
|
+
this.#services = services;
|
|
15
|
+
}
|
|
16
|
+
get #database() {
|
|
17
|
+
return this.#services.get(DatabaseService);
|
|
18
|
+
}
|
|
19
|
+
create = async (input) => {
|
|
20
|
+
const db = await this.#database.getInstance();
|
|
21
|
+
const id = randomUUID();
|
|
22
|
+
const now = new Date().toISOString();
|
|
23
|
+
const commitSha = input.commitSha ?? null;
|
|
24
|
+
await db
|
|
25
|
+
.insertInto('messages')
|
|
26
|
+
.values({
|
|
27
|
+
id,
|
|
28
|
+
session_id: input.sessionId,
|
|
29
|
+
role: input.role,
|
|
30
|
+
content: input.content,
|
|
31
|
+
commit_sha: commitSha,
|
|
32
|
+
created_at: now,
|
|
33
|
+
})
|
|
34
|
+
.execute();
|
|
35
|
+
return {
|
|
36
|
+
id,
|
|
37
|
+
sessionId: input.sessionId,
|
|
38
|
+
role: input.role,
|
|
39
|
+
content: input.content,
|
|
40
|
+
commitSha,
|
|
41
|
+
createdAt: now,
|
|
42
|
+
};
|
|
43
|
+
};
|
|
44
|
+
getById = async (messageId) => {
|
|
45
|
+
const db = await this.#database.getInstance();
|
|
46
|
+
const row = await db
|
|
47
|
+
.selectFrom('messages')
|
|
48
|
+
.selectAll()
|
|
49
|
+
.where('id', '=', messageId)
|
|
50
|
+
.executeTakeFirst();
|
|
51
|
+
if (!row) {
|
|
52
|
+
throw new Error('Message not found');
|
|
53
|
+
}
|
|
54
|
+
return mapRow(row);
|
|
55
|
+
};
|
|
56
|
+
deleteAfter = async (input) => {
|
|
57
|
+
const db = await this.#database.getInstance();
|
|
58
|
+
const target = await db
|
|
59
|
+
.selectFrom('messages')
|
|
60
|
+
.select('created_at')
|
|
61
|
+
.where('id', '=', input.messageId)
|
|
62
|
+
.where('session_id', '=', input.sessionId)
|
|
63
|
+
.executeTakeFirst();
|
|
64
|
+
if (!target) {
|
|
65
|
+
throw new Error('Message not found');
|
|
66
|
+
}
|
|
67
|
+
await db
|
|
68
|
+
.deleteFrom('messages')
|
|
69
|
+
.where('session_id', '=', input.sessionId)
|
|
70
|
+
.where('created_at', '>', target.created_at)
|
|
71
|
+
.execute();
|
|
72
|
+
};
|
|
73
|
+
listBySession = async (sessionId) => {
|
|
74
|
+
const db = await this.#database.getInstance();
|
|
75
|
+
const rows = await db
|
|
76
|
+
.selectFrom('messages')
|
|
77
|
+
.selectAll()
|
|
78
|
+
.where('session_id', '=', sessionId)
|
|
79
|
+
.orderBy('created_at', 'asc')
|
|
80
|
+
.execute();
|
|
81
|
+
return rows.map(mapRow);
|
|
82
|
+
};
|
|
83
|
+
}
|
|
84
|
+
export { MessageService };
|
|
85
|
+
//# sourceMappingURL=message.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"message.js","sourceRoot":"","sources":["../../../src/services/message/message.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import type { z } from 'zod';
|
|
2
|
+
import type { Services } from '../../container/container.js';
|
|
3
|
+
type NotificationLevel = 'info' | 'warning' | 'error';
|
|
4
|
+
type Notification = {
|
|
5
|
+
title: string;
|
|
6
|
+
body: string;
|
|
7
|
+
level: NotificationLevel;
|
|
8
|
+
sessionId?: string;
|
|
9
|
+
tags?: string[];
|
|
10
|
+
};
|
|
11
|
+
type NotificationProvider = {
|
|
12
|
+
name: string;
|
|
13
|
+
configSchema: z.ZodType;
|
|
14
|
+
send: (config: Record<string, unknown>, notification: Notification) => Promise<void>;
|
|
15
|
+
};
|
|
16
|
+
type NotificationChannel = {
|
|
17
|
+
id: string;
|
|
18
|
+
userId: string;
|
|
19
|
+
name: string;
|
|
20
|
+
provider: string;
|
|
21
|
+
enabled: boolean;
|
|
22
|
+
createdAt: string;
|
|
23
|
+
updatedAt: string;
|
|
24
|
+
};
|
|
25
|
+
type CreateChannelInput = {
|
|
26
|
+
userId: string;
|
|
27
|
+
name: string;
|
|
28
|
+
provider: string;
|
|
29
|
+
config: Record<string, unknown>;
|
|
30
|
+
};
|
|
31
|
+
type UpdateChannelInput = {
|
|
32
|
+
userId: string;
|
|
33
|
+
channelId: string;
|
|
34
|
+
name?: string;
|
|
35
|
+
enabled?: boolean;
|
|
36
|
+
config?: Record<string, unknown>;
|
|
37
|
+
};
|
|
38
|
+
declare class NotificationService {
|
|
39
|
+
#private;
|
|
40
|
+
constructor(services: Services);
|
|
41
|
+
registerProvider: (provider: NotificationProvider) => void;
|
|
42
|
+
getProvider: (name: string) => NotificationProvider;
|
|
43
|
+
getProviderNames: () => string[];
|
|
44
|
+
create: (input: CreateChannelInput) => Promise<NotificationChannel>;
|
|
45
|
+
list: (userId: string) => Promise<NotificationChannel[]>;
|
|
46
|
+
get: (input: {
|
|
47
|
+
userId: string;
|
|
48
|
+
channelId: string;
|
|
49
|
+
}) => Promise<NotificationChannel>;
|
|
50
|
+
update: (input: UpdateChannelInput) => Promise<NotificationChannel>;
|
|
51
|
+
delete: (input: {
|
|
52
|
+
userId: string;
|
|
53
|
+
channelId: string;
|
|
54
|
+
}) => Promise<void>;
|
|
55
|
+
dispatch: (userId: string, notification: Notification) => Promise<void>;
|
|
56
|
+
test: (input: {
|
|
57
|
+
userId: string;
|
|
58
|
+
channelId: string;
|
|
59
|
+
}) => Promise<void>;
|
|
60
|
+
getPreferences: (userId: string) => Promise<{
|
|
61
|
+
notificationsEnabled: boolean;
|
|
62
|
+
notificationEvents: string[];
|
|
63
|
+
}>;
|
|
64
|
+
updatePreferences: (input: {
|
|
65
|
+
userId: string;
|
|
66
|
+
notificationsEnabled?: boolean;
|
|
67
|
+
notificationEvents?: string[];
|
|
68
|
+
}) => Promise<{
|
|
69
|
+
notificationsEnabled: boolean;
|
|
70
|
+
notificationEvents: string[];
|
|
71
|
+
}>;
|
|
72
|
+
}
|
|
73
|
+
export type { NotificationLevel, Notification, NotificationProvider, NotificationChannel, CreateChannelInput, UpdateChannelInput, };
|
|
74
|
+
export { NotificationService };
|
|
75
|
+
//# sourceMappingURL=notification.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notification.d.ts","sourceRoot":"","sources":["../../../src/services/notification/notification.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
declare class NotificationError extends Error {
|
|
2
|
+
constructor(message: string);
|
|
3
|
+
}
|
|
4
|
+
declare class NotificationChannelNotFoundError extends NotificationError {
|
|
5
|
+
constructor();
|
|
6
|
+
}
|
|
7
|
+
declare class NotificationProviderNotFoundError extends NotificationError {
|
|
8
|
+
constructor(provider: string);
|
|
9
|
+
}
|
|
10
|
+
declare class NotificationForbiddenError extends NotificationError {
|
|
11
|
+
constructor();
|
|
12
|
+
}
|
|
13
|
+
export { NotificationError, NotificationChannelNotFoundError, NotificationProviderNotFoundError, NotificationForbiddenError, };
|
|
14
|
+
//# sourceMappingURL=notification.errors.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notification.errors.d.ts","sourceRoot":"","sources":["../../../src/services/notification/notification.errors.ts"],"names":[],"mappings":"AAAA,cAAM,iBAAkB,SAAQ,KAAK;gBACvB,OAAO,EAAE,MAAM;CAI5B;AAED,cAAM,gCAAiC,SAAQ,iBAAiB;;CAK/D;AAED,cAAM,iCAAkC,SAAQ,iBAAiB;gBACnD,QAAQ,EAAE,MAAM;CAI7B;AAED,cAAM,0BAA2B,SAAQ,iBAAiB;;CAKzD;AAED,OAAO,EACL,iBAAiB,EACjB,gCAAgC,EAChC,iCAAiC,EACjC,0BAA0B,GAC3B,CAAC"}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
class NotificationError extends Error {
|
|
2
|
+
constructor(message) {
|
|
3
|
+
super(message);
|
|
4
|
+
this.name = 'NotificationError';
|
|
5
|
+
}
|
|
6
|
+
}
|
|
7
|
+
class NotificationChannelNotFoundError extends NotificationError {
|
|
8
|
+
constructor() {
|
|
9
|
+
super('Notification channel not found');
|
|
10
|
+
this.name = 'NotificationChannelNotFoundError';
|
|
11
|
+
}
|
|
12
|
+
}
|
|
13
|
+
class NotificationProviderNotFoundError extends NotificationError {
|
|
14
|
+
constructor(provider) {
|
|
15
|
+
super(`Notification provider not found: ${provider}`);
|
|
16
|
+
this.name = 'NotificationProviderNotFoundError';
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
class NotificationForbiddenError extends NotificationError {
|
|
20
|
+
constructor() {
|
|
21
|
+
super('Forbidden');
|
|
22
|
+
this.name = 'NotificationForbiddenError';
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
export { NotificationError, NotificationChannelNotFoundError, NotificationProviderNotFoundError, NotificationForbiddenError, };
|
|
26
|
+
//# sourceMappingURL=notification.errors.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"notification.errors.js","sourceRoot":"","sources":["../../../src/services/notification/notification.errors.ts"],"names":[],"mappings":"AAAA,MAAM,iBAAkB,SAAQ,KAAK;IACnC,YAAY,OAAe;QACzB,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,IAAI,GAAG,mBAAmB,CAAC;IAClC,CAAC;CACF;AAED,MAAM,gCAAiC,SAAQ,iBAAiB;IAC9D;QACE,KAAK,CAAC,gCAAgC,CAAC,CAAC;QACxC,IAAI,CAAC,IAAI,GAAG,kCAAkC,CAAC;IACjD,CAAC;CACF;AAED,MAAM,iCAAkC,SAAQ,iBAAiB;IAC/D,YAAY,QAAgB;QAC1B,KAAK,CAAC,oCAAoC,QAAQ,EAAE,CAAC,CAAC;QACtD,IAAI,CAAC,IAAI,GAAG,mCAAmC,CAAC;IAClD,CAAC;CACF;AAED,MAAM,0BAA2B,SAAQ,iBAAiB;IACxD;QACE,KAAK,CAAC,WAAW,CAAC,CAAC;QACnB,IAAI,CAAC,IAAI,GAAG,4BAA4B,CAAC;IAC3C,CAAC;CACF;AAED,OAAO,EACL,iBAAiB,EACjB,gCAAgC,EAChC,iCAAiC,EACjC,0BAA0B,GAC3B,CAAC"}
|