@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.
Files changed (270) hide show
  1. package/dist/app/app.d.ts +11 -0
  2. package/dist/app/app.d.ts.map +1 -0
  3. package/dist/app/app.js +141 -0
  4. package/dist/app/app.js.map +1 -0
  5. package/dist/app/app.routes.d.ts +4 -0
  6. package/dist/app/app.routes.d.ts.map +1 -0
  7. package/dist/app/app.routes.js +22 -0
  8. package/dist/app/app.routes.js.map +1 -0
  9. package/dist/app/app.types.d.ts +15 -0
  10. package/dist/app/app.types.d.ts.map +1 -0
  11. package/dist/app/app.types.js +2 -0
  12. package/dist/app/app.types.js.map +1 -0
  13. package/dist/config/config.d.ts +31 -0
  14. package/dist/config/config.d.ts.map +1 -0
  15. package/dist/config/config.js +93 -0
  16. package/dist/config/config.js.map +1 -0
  17. package/dist/config/config.testing.d.ts +4 -0
  18. package/dist/config/config.testing.d.ts.map +1 -0
  19. package/dist/config/config.testing.js +17 -0
  20. package/dist/config/config.testing.js.map +1 -0
  21. package/dist/container/container.d.ts +38 -0
  22. package/dist/container/container.d.ts.map +1 -0
  23. package/dist/container/container.js +33 -0
  24. package/dist/container/container.js.map +1 -0
  25. package/dist/container/container.test.d.ts +2 -0
  26. package/dist/container/container.test.d.ts.map +1 -0
  27. package/dist/container/container.test.js +65 -0
  28. package/dist/container/container.test.js.map +1 -0
  29. package/dist/exports.d.ts +67 -0
  30. package/dist/exports.d.ts.map +1 -0
  31. package/dist/exports.js +44 -0
  32. package/dist/exports.js.map +1 -0
  33. package/dist/routes/auth/auth.d.ts +4 -0
  34. package/dist/routes/auth/auth.d.ts.map +1 -0
  35. package/dist/routes/auth/auth.js +74 -0
  36. package/dist/routes/auth/auth.js.map +1 -0
  37. package/dist/routes/auth/auth.schemas.d.ts +42 -0
  38. package/dist/routes/auth/auth.schemas.d.ts.map +1 -0
  39. package/dist/routes/auth/auth.schemas.js +34 -0
  40. package/dist/routes/auth/auth.schemas.js.map +1 -0
  41. package/dist/routes/auth/auth.test.d.ts +2 -0
  42. package/dist/routes/auth/auth.test.d.ts.map +1 -0
  43. package/dist/routes/auth/auth.test.js +203 -0
  44. package/dist/routes/auth/auth.test.js.map +1 -0
  45. package/dist/routes/events/events.d.ts +4 -0
  46. package/dist/routes/events/events.d.ts.map +1 -0
  47. package/dist/routes/events/events.js +28 -0
  48. package/dist/routes/events/events.js.map +1 -0
  49. package/dist/routes/identities/identities.d.ts +4 -0
  50. package/dist/routes/identities/identities.d.ts.map +1 -0
  51. package/dist/routes/identities/identities.js +128 -0
  52. package/dist/routes/identities/identities.js.map +1 -0
  53. package/dist/routes/identities/identities.schemas.d.ts +52 -0
  54. package/dist/routes/identities/identities.schemas.d.ts.map +1 -0
  55. package/dist/routes/identities/identities.schemas.js +38 -0
  56. package/dist/routes/identities/identities.schemas.js.map +1 -0
  57. package/dist/routes/identities/identities.test.d.ts +2 -0
  58. package/dist/routes/identities/identities.test.d.ts.map +1 -0
  59. package/dist/routes/identities/identities.test.js +329 -0
  60. package/dist/routes/identities/identities.test.js.map +1 -0
  61. package/dist/routes/models/models.d.ts +4 -0
  62. package/dist/routes/models/models.d.ts.map +1 -0
  63. package/dist/routes/models/models.js +60 -0
  64. package/dist/routes/models/models.js.map +1 -0
  65. package/dist/routes/models/models.schemas.d.ts +16 -0
  66. package/dist/routes/models/models.schemas.d.ts.map +1 -0
  67. package/dist/routes/models/models.schemas.js +9 -0
  68. package/dist/routes/models/models.schemas.js.map +1 -0
  69. package/dist/routes/notifications/notifications.d.ts +4 -0
  70. package/dist/routes/notifications/notifications.d.ts.map +1 -0
  71. package/dist/routes/notifications/notifications.js +226 -0
  72. package/dist/routes/notifications/notifications.js.map +1 -0
  73. package/dist/routes/notifications/notifications.schemas.d.ts +61 -0
  74. package/dist/routes/notifications/notifications.schemas.d.ts.map +1 -0
  75. package/dist/routes/notifications/notifications.schemas.js +44 -0
  76. package/dist/routes/notifications/notifications.schemas.js.map +1 -0
  77. package/dist/routes/notifications/notifications.test.d.ts +2 -0
  78. package/dist/routes/notifications/notifications.test.d.ts.map +1 -0
  79. package/dist/routes/notifications/notifications.test.js +44 -0
  80. package/dist/routes/notifications/notifications.test.js.map +1 -0
  81. package/dist/routes/repos/repos.d.ts +4 -0
  82. package/dist/routes/repos/repos.d.ts.map +1 -0
  83. package/dist/routes/repos/repos.js +142 -0
  84. package/dist/routes/repos/repos.js.map +1 -0
  85. package/dist/routes/repos/repos.schemas.d.ts +47 -0
  86. package/dist/routes/repos/repos.schemas.d.ts.map +1 -0
  87. package/dist/routes/repos/repos.schemas.js +32 -0
  88. package/dist/routes/repos/repos.schemas.js.map +1 -0
  89. package/dist/routes/sessions/sessions.d.ts +4 -0
  90. package/dist/routes/sessions/sessions.d.ts.map +1 -0
  91. package/dist/routes/sessions/sessions.js +299 -0
  92. package/dist/routes/sessions/sessions.js.map +1 -0
  93. package/dist/routes/sessions/sessions.review.d.ts +4 -0
  94. package/dist/routes/sessions/sessions.review.d.ts.map +1 -0
  95. package/dist/routes/sessions/sessions.review.js +294 -0
  96. package/dist/routes/sessions/sessions.review.js.map +1 -0
  97. package/dist/routes/sessions/sessions.review.schemas.d.ts +83 -0
  98. package/dist/routes/sessions/sessions.review.schemas.d.ts.map +1 -0
  99. package/dist/routes/sessions/sessions.review.schemas.js +55 -0
  100. package/dist/routes/sessions/sessions.review.schemas.js.map +1 -0
  101. package/dist/routes/sessions/sessions.schemas.d.ts +81 -0
  102. package/dist/routes/sessions/sessions.schemas.d.ts.map +1 -0
  103. package/dist/routes/sessions/sessions.schemas.js +50 -0
  104. package/dist/routes/sessions/sessions.schemas.js.map +1 -0
  105. package/dist/routes/sessions/sessions.test.d.ts +2 -0
  106. package/dist/routes/sessions/sessions.test.d.ts.map +1 -0
  107. package/dist/routes/sessions/sessions.test.js +297 -0
  108. package/dist/routes/sessions/sessions.test.js.map +1 -0
  109. package/dist/routes/ws/ws.d.ts +4 -0
  110. package/dist/routes/ws/ws.d.ts.map +1 -0
  111. package/dist/routes/ws/ws.js +8 -0
  112. package/dist/routes/ws/ws.js.map +1 -0
  113. package/dist/services/agent/agent.claude.d.ts +4 -0
  114. package/dist/services/agent/agent.claude.d.ts.map +1 -0
  115. package/dist/services/agent/agent.claude.js +141 -0
  116. package/dist/services/agent/agent.claude.js.map +1 -0
  117. package/dist/services/agent/agent.claude.test.d.ts +2 -0
  118. package/dist/services/agent/agent.claude.test.d.ts.map +1 -0
  119. package/dist/services/agent/agent.claude.test.js +43 -0
  120. package/dist/services/agent/agent.claude.test.js.map +1 -0
  121. package/dist/services/agent/agent.d.ts +58 -0
  122. package/dist/services/agent/agent.d.ts.map +1 -0
  123. package/dist/services/agent/agent.errors.d.ts +8 -0
  124. package/dist/services/agent/agent.errors.d.ts.map +1 -0
  125. package/dist/services/agent/agent.errors.js +14 -0
  126. package/dist/services/agent/agent.errors.js.map +1 -0
  127. package/dist/services/agent/agent.js +21 -0
  128. package/dist/services/agent/agent.js.map +1 -0
  129. package/dist/services/agent/agent.queue.d.ts +8 -0
  130. package/dist/services/agent/agent.queue.d.ts.map +1 -0
  131. package/dist/services/agent/agent.queue.js +46 -0
  132. package/dist/services/agent/agent.queue.js.map +1 -0
  133. package/dist/services/auth/auth.d.ts +41 -0
  134. package/dist/services/auth/auth.d.ts.map +1 -0
  135. package/dist/services/auth/auth.errors.d.ts +17 -0
  136. package/dist/services/auth/auth.errors.d.ts.map +1 -0
  137. package/dist/services/auth/auth.errors.js +32 -0
  138. package/dist/services/auth/auth.errors.js.map +1 -0
  139. package/dist/services/auth/auth.js +147 -0
  140. package/dist/services/auth/auth.js.map +1 -0
  141. package/dist/services/auth/auth.test.d.ts +2 -0
  142. package/dist/services/auth/auth.test.d.ts.map +1 -0
  143. package/dist/services/auth/auth.test.js +55 -0
  144. package/dist/services/auth/auth.test.js.map +1 -0
  145. package/dist/services/auth/auth.utils.d.ts +4 -0
  146. package/dist/services/auth/auth.utils.d.ts.map +1 -0
  147. package/dist/services/auth/auth.utils.js +15 -0
  148. package/dist/services/auth/auth.utils.js.map +1 -0
  149. package/dist/services/auth/auth.utils.test.d.ts +2 -0
  150. package/dist/services/auth/auth.utils.test.d.ts.map +1 -0
  151. package/dist/services/auth/auth.utils.test.js +20 -0
  152. package/dist/services/auth/auth.utils.test.js.map +1 -0
  153. package/dist/services/database/database.d.ts +102 -0
  154. package/dist/services/database/database.d.ts.map +1 -0
  155. package/dist/services/database/database.js +41 -0
  156. package/dist/services/database/database.js.map +1 -0
  157. package/dist/services/database/database.migrations.d.ts +4 -0
  158. package/dist/services/database/database.migrations.d.ts.map +1 -0
  159. package/dist/services/database/database.migrations.js +290 -0
  160. package/dist/services/database/database.migrations.js.map +1 -0
  161. package/dist/services/database/database.test.d.ts +2 -0
  162. package/dist/services/database/database.test.d.ts.map +1 -0
  163. package/dist/services/database/database.test.js +60 -0
  164. package/dist/services/database/database.test.js.map +1 -0
  165. package/dist/services/file-review/file-review.d.ts +34 -0
  166. package/dist/services/file-review/file-review.d.ts.map +1 -0
  167. package/dist/services/file-review/file-review.js +71 -0
  168. package/dist/services/file-review/file-review.js.map +1 -0
  169. package/dist/services/file-review/file-review.test.d.ts +2 -0
  170. package/dist/services/file-review/file-review.test.d.ts.map +1 -0
  171. package/dist/services/file-review/file-review.test.js +171 -0
  172. package/dist/services/file-review/file-review.test.js.map +1 -0
  173. package/dist/services/git/git.d.ts +95 -0
  174. package/dist/services/git/git.d.ts.map +1 -0
  175. package/dist/services/git/git.errors.d.ts +20 -0
  176. package/dist/services/git/git.errors.d.ts.map +1 -0
  177. package/dist/services/git/git.errors.js +38 -0
  178. package/dist/services/git/git.errors.js.map +1 -0
  179. package/dist/services/git/git.js +341 -0
  180. package/dist/services/git/git.js.map +1 -0
  181. package/dist/services/git/git.test.d.ts +2 -0
  182. package/dist/services/git/git.test.d.ts.map +1 -0
  183. package/dist/services/git/git.test.js +304 -0
  184. package/dist/services/git/git.test.js.map +1 -0
  185. package/dist/services/identity/identity.d.ts +49 -0
  186. package/dist/services/identity/identity.d.ts.map +1 -0
  187. package/dist/services/identity/identity.errors.d.ts +11 -0
  188. package/dist/services/identity/identity.errors.d.ts.map +1 -0
  189. package/dist/services/identity/identity.errors.js +20 -0
  190. package/dist/services/identity/identity.errors.js.map +1 -0
  191. package/dist/services/identity/identity.js +196 -0
  192. package/dist/services/identity/identity.js.map +1 -0
  193. package/dist/services/identity/identity.test.d.ts +2 -0
  194. package/dist/services/identity/identity.test.d.ts.map +1 -0
  195. package/dist/services/identity/identity.test.js +184 -0
  196. package/dist/services/identity/identity.test.js.map +1 -0
  197. package/dist/services/identity/identity.utils.d.ts +9 -0
  198. package/dist/services/identity/identity.utils.d.ts.map +1 -0
  199. package/dist/services/identity/identity.utils.js +47 -0
  200. package/dist/services/identity/identity.utils.js.map +1 -0
  201. package/dist/services/identity/identity.utils.test.d.ts +2 -0
  202. package/dist/services/identity/identity.utils.test.d.ts.map +1 -0
  203. package/dist/services/identity/identity.utils.test.js +49 -0
  204. package/dist/services/identity/identity.utils.test.js.map +1 -0
  205. package/dist/services/message/message.d.ts +29 -0
  206. package/dist/services/message/message.d.ts.map +1 -0
  207. package/dist/services/message/message.js +85 -0
  208. package/dist/services/message/message.js.map +1 -0
  209. package/dist/services/notification/notification.d.ts +75 -0
  210. package/dist/services/notification/notification.d.ts.map +1 -0
  211. package/dist/services/notification/notification.errors.d.ts +14 -0
  212. package/dist/services/notification/notification.errors.d.ts.map +1 -0
  213. package/dist/services/notification/notification.errors.js +26 -0
  214. package/dist/services/notification/notification.errors.js.map +1 -0
  215. package/dist/services/notification/notification.js +212 -0
  216. package/dist/services/notification/notification.js.map +1 -0
  217. package/dist/services/notification/notification.ntfy.d.ts +12 -0
  218. package/dist/services/notification/notification.ntfy.d.ts.map +1 -0
  219. package/dist/services/notification/notification.ntfy.js +44 -0
  220. package/dist/services/notification/notification.ntfy.js.map +1 -0
  221. package/dist/services/repo/repo.d.ts +46 -0
  222. package/dist/services/repo/repo.d.ts.map +1 -0
  223. package/dist/services/repo/repo.errors.d.ts +11 -0
  224. package/dist/services/repo/repo.errors.d.ts.map +1 -0
  225. package/dist/services/repo/repo.errors.js +20 -0
  226. package/dist/services/repo/repo.errors.js.map +1 -0
  227. package/dist/services/repo/repo.js +138 -0
  228. package/dist/services/repo/repo.js.map +1 -0
  229. package/dist/services/session/session.d.ts +62 -0
  230. package/dist/services/session/session.d.ts.map +1 -0
  231. package/dist/services/session/session.errors.d.ts +11 -0
  232. package/dist/services/session/session.errors.d.ts.map +1 -0
  233. package/dist/services/session/session.errors.js +20 -0
  234. package/dist/services/session/session.errors.js.map +1 -0
  235. package/dist/services/session/session.js +168 -0
  236. package/dist/services/session/session.js.map +1 -0
  237. package/dist/services/session/session.runner.d.ts +8 -0
  238. package/dist/services/session/session.runner.d.ts.map +1 -0
  239. package/dist/services/session/session.runner.js +328 -0
  240. package/dist/services/session/session.runner.js.map +1 -0
  241. package/dist/services/session/session.runner.test.d.ts +2 -0
  242. package/dist/services/session/session.runner.test.d.ts.map +1 -0
  243. package/dist/services/session/session.runner.test.js +143 -0
  244. package/dist/services/session/session.runner.test.js.map +1 -0
  245. package/dist/services/session/session.test.d.ts +2 -0
  246. package/dist/services/session/session.test.d.ts.map +1 -0
  247. package/dist/services/session/session.test.js +146 -0
  248. package/dist/services/session/session.test.js.map +1 -0
  249. package/dist/services/session-event/session-event.d.ts +28 -0
  250. package/dist/services/session-event/session-event.d.ts.map +1 -0
  251. package/dist/services/session-event/session-event.js +86 -0
  252. package/dist/services/session-event/session-event.js.map +1 -0
  253. package/dist/sse/event-bus.d.ts +74 -0
  254. package/dist/sse/event-bus.d.ts.map +1 -0
  255. package/dist/sse/event-bus.js +137 -0
  256. package/dist/sse/event-bus.js.map +1 -0
  257. package/dist/sse/stream.d.ts +20 -0
  258. package/dist/sse/stream.d.ts.map +1 -0
  259. package/dist/sse/stream.js +66 -0
  260. package/dist/sse/stream.js.map +1 -0
  261. package/dist/tsconfig.tsbuildinfo +1 -0
  262. package/dist/utils/crypto.d.ts +4 -0
  263. package/dist/utils/crypto.d.ts.map +1 -0
  264. package/dist/utils/crypto.js +21 -0
  265. package/dist/utils/crypto.js.map +1 -0
  266. package/dist/ws/ws.d.ts +30 -0
  267. package/dist/ws/ws.d.ts.map +1 -0
  268. package/dist/ws/ws.js +137 -0
  269. package/dist/ws/ws.js.map +1 -0
  270. 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,2 @@
1
+ export {};
2
+ //# sourceMappingURL=identity.utils.test.d.ts.map
@@ -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"}