@symbo.ls/sdk 3.1.2 → 3.2.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 (187) hide show
  1. package/README.md +143 -2
  2. package/dist/cjs/config/environment.js +98 -30
  3. package/dist/cjs/index.js +144 -24
  4. package/dist/cjs/services/AdminService.js +351 -0
  5. package/dist/cjs/services/AuthService.js +738 -305
  6. package/dist/cjs/services/BaseService.js +158 -6
  7. package/dist/cjs/services/BranchService.js +484 -0
  8. package/dist/cjs/services/CollabService.js +743 -0
  9. package/dist/cjs/services/DnsService.js +340 -0
  10. package/dist/cjs/services/FeatureFlagService.js +175 -0
  11. package/dist/cjs/services/FileService.js +201 -0
  12. package/dist/cjs/services/IntegrationService.js +538 -0
  13. package/dist/cjs/services/MetricsService.js +62 -0
  14. package/dist/cjs/services/PaymentService.js +271 -0
  15. package/dist/cjs/services/PlanService.js +426 -0
  16. package/dist/cjs/services/ProjectService.js +1207 -0
  17. package/dist/cjs/services/PullRequestService.js +503 -0
  18. package/dist/cjs/services/ScreenshotService.js +304 -0
  19. package/dist/cjs/services/SubscriptionService.js +396 -0
  20. package/dist/cjs/services/TrackingService.js +661 -0
  21. package/dist/cjs/services/WaitlistService.js +148 -0
  22. package/dist/cjs/services/index.js +64 -16
  23. package/dist/cjs/state/RootStateManager.js +65 -0
  24. package/dist/cjs/state/rootEventBus.js +74 -0
  25. package/dist/cjs/utils/CollabClient.js +223 -0
  26. package/dist/cjs/utils/TokenManager.js +78 -30
  27. package/dist/cjs/utils/changePreprocessor.js +199 -0
  28. package/dist/cjs/utils/jsonDiff.js +145 -0
  29. package/dist/cjs/utils/ordering.js +309 -0
  30. package/dist/cjs/utils/services.js +301 -103
  31. package/dist/cjs/utils/validation.js +0 -3
  32. package/dist/esm/config/environment.js +98 -30
  33. package/dist/esm/index.js +49505 -8718
  34. package/dist/esm/services/AdminService.js +1132 -0
  35. package/dist/esm/services/AuthService.js +1493 -386
  36. package/dist/esm/services/BaseService.js +757 -6
  37. package/dist/esm/services/BranchService.js +1265 -0
  38. package/dist/esm/services/CollabService.js +26895 -0
  39. package/dist/esm/services/DnsService.js +1121 -0
  40. package/dist/esm/services/FeatureFlagService.js +956 -0
  41. package/dist/esm/services/FileService.js +982 -0
  42. package/dist/esm/services/IntegrationService.js +1319 -0
  43. package/dist/esm/services/MetricsService.js +843 -0
  44. package/dist/esm/services/PaymentService.js +1052 -0
  45. package/dist/esm/services/PlanService.js +1207 -0
  46. package/dist/esm/services/ProjectService.js +2526 -0
  47. package/dist/esm/services/PullRequestService.js +1284 -0
  48. package/dist/esm/services/ScreenshotService.js +1085 -0
  49. package/dist/esm/services/SubscriptionService.js +1177 -0
  50. package/dist/esm/services/TrackingService.js +18454 -0
  51. package/dist/esm/services/WaitlistService.js +929 -0
  52. package/dist/esm/services/index.js +49062 -8569
  53. package/dist/esm/state/RootStateManager.js +90 -0
  54. package/dist/esm/state/rootEventBus.js +56 -0
  55. package/dist/esm/utils/CollabClient.js +18889 -0
  56. package/dist/esm/utils/TokenManager.js +78 -30
  57. package/dist/esm/utils/changePreprocessor.js +542 -0
  58. package/dist/esm/utils/jsonDiff.js +7011 -0
  59. package/dist/esm/utils/ordering.js +291 -0
  60. package/dist/esm/utils/services.js +301 -103
  61. package/dist/esm/utils/validation.js +116 -50
  62. package/dist/node/config/environment.js +98 -30
  63. package/dist/node/index.js +175 -32
  64. package/dist/node/services/AdminService.js +332 -0
  65. package/dist/node/services/AuthService.js +742 -310
  66. package/dist/node/services/BaseService.js +148 -6
  67. package/dist/node/services/BranchService.js +465 -0
  68. package/dist/node/services/CollabService.js +724 -0
  69. package/dist/node/services/DnsService.js +321 -0
  70. package/dist/node/services/FeatureFlagService.js +156 -0
  71. package/dist/node/services/FileService.js +182 -0
  72. package/dist/node/services/IntegrationService.js +519 -0
  73. package/dist/node/services/MetricsService.js +43 -0
  74. package/dist/node/services/PaymentService.js +252 -0
  75. package/dist/node/services/PlanService.js +407 -0
  76. package/dist/node/services/ProjectService.js +1188 -0
  77. package/dist/node/services/PullRequestService.js +484 -0
  78. package/dist/node/services/ScreenshotService.js +285 -0
  79. package/dist/node/services/SubscriptionService.js +377 -0
  80. package/dist/node/services/TrackingService.js +632 -0
  81. package/dist/node/services/WaitlistService.js +129 -0
  82. package/dist/node/services/index.js +64 -16
  83. package/dist/node/state/RootStateManager.js +36 -0
  84. package/dist/node/state/rootEventBus.js +55 -0
  85. package/dist/node/utils/CollabClient.js +194 -0
  86. package/dist/node/utils/TokenManager.js +78 -30
  87. package/dist/node/utils/changePreprocessor.js +180 -0
  88. package/dist/node/utils/jsonDiff.js +116 -0
  89. package/dist/node/utils/ordering.js +290 -0
  90. package/dist/node/utils/services.js +301 -103
  91. package/dist/node/utils/validation.js +0 -3
  92. package/package.json +39 -21
  93. package/src/config/environment.js +99 -28
  94. package/src/index.js +181 -36
  95. package/src/services/AdminService.js +374 -0
  96. package/src/services/AuthService.js +874 -328
  97. package/src/services/BaseService.js +166 -6
  98. package/src/services/BranchService.js +536 -0
  99. package/src/services/CollabService.js +900 -0
  100. package/src/services/DnsService.js +366 -0
  101. package/src/services/FeatureFlagService.js +174 -0
  102. package/src/services/FileService.js +213 -0
  103. package/src/services/IntegrationService.js +548 -0
  104. package/src/services/MetricsService.js +40 -0
  105. package/src/services/PaymentService.js +287 -0
  106. package/src/services/PlanService.js +468 -0
  107. package/src/services/ProjectService.js +1366 -0
  108. package/src/services/PullRequestService.js +537 -0
  109. package/src/services/ScreenshotService.js +258 -0
  110. package/src/services/SubscriptionService.js +425 -0
  111. package/src/services/TrackingService.js +853 -0
  112. package/src/services/WaitlistService.js +130 -0
  113. package/src/services/index.js +80 -13
  114. package/src/services/tests/BranchService/createBranch.test.js +153 -0
  115. package/src/services/tests/BranchService/deleteBranch.test.js +173 -0
  116. package/src/services/tests/BranchService/getBranchChanges.test.js +146 -0
  117. package/src/services/tests/BranchService/listBranches.test.js +87 -0
  118. package/src/services/tests/BranchService/mergeBranch.test.js +210 -0
  119. package/src/services/tests/BranchService/publishVersion.test.js +183 -0
  120. package/src/services/tests/BranchService/renameBranch.test.js +240 -0
  121. package/src/services/tests/BranchService/resetBranch.test.js +152 -0
  122. package/src/services/tests/FeatureFlagService/adminFeatureFlags.test.js +67 -0
  123. package/src/services/tests/FeatureFlagService/getFeatureFlags.test.js +75 -0
  124. package/src/services/tests/FileService/createFileFormData.test.js +74 -0
  125. package/src/services/tests/FileService/getFileUrl.test.js +69 -0
  126. package/src/services/tests/FileService/updateProjectIcon.test.js +109 -0
  127. package/src/services/tests/FileService/uploadDocument.test.js +36 -0
  128. package/src/services/tests/FileService/uploadFile.test.js +78 -0
  129. package/src/services/tests/FileService/uploadFileWithValidation.test.js +114 -0
  130. package/src/services/tests/FileService/uploadImage.test.js +36 -0
  131. package/src/services/tests/FileService/uploadMultipleFiles.test.js +111 -0
  132. package/src/services/tests/FileService/validateFile.test.js +63 -0
  133. package/src/services/tests/PlanService/createPlan.test.js +104 -0
  134. package/src/services/tests/PlanService/createPlanWithValidation.test.js +523 -0
  135. package/src/services/tests/PlanService/deletePlan.test.js +92 -0
  136. package/src/services/tests/PlanService/getActivePlans.test.js +123 -0
  137. package/src/services/tests/PlanService/getAdminPlans.test.js +84 -0
  138. package/src/services/tests/PlanService/getPlan.test.js +50 -0
  139. package/src/services/tests/PlanService/getPlanByKey.test.js +109 -0
  140. package/src/services/tests/PlanService/getPlanWithValidation.test.js +85 -0
  141. package/src/services/tests/PlanService/getPlans.test.js +53 -0
  142. package/src/services/tests/PlanService/getPlansByPriceRange.test.js +109 -0
  143. package/src/services/tests/PlanService/getPlansWithValidation.test.js +48 -0
  144. package/src/services/tests/PlanService/initializePlans.test.js +75 -0
  145. package/src/services/tests/PlanService/updatePlan.test.js +111 -0
  146. package/src/services/tests/PlanService/updatePlanWithValidation.test.js +556 -0
  147. package/src/state/RootStateManager.js +76 -0
  148. package/src/state/rootEventBus.js +67 -0
  149. package/src/utils/CollabClient.js +248 -0
  150. package/src/utils/TokenManager.js +88 -33
  151. package/src/utils/changePreprocessor.js +239 -0
  152. package/src/utils/jsonDiff.js +144 -0
  153. package/src/utils/ordering.js +271 -0
  154. package/src/utils/services.js +326 -107
  155. package/src/utils/validation.js +0 -3
  156. package/dist/cjs/services/AIService.js +0 -155
  157. package/dist/cjs/services/BasedService.js +0 -1185
  158. package/dist/cjs/services/CoreService.js +0 -1751
  159. package/dist/cjs/services/SocketIOService.js +0 -307
  160. package/dist/cjs/services/SocketService.js +0 -161
  161. package/dist/cjs/services/SymstoryService.js +0 -571
  162. package/dist/cjs/utils/basedQuerys.js +0 -181
  163. package/dist/cjs/utils/symstoryClient.js +0 -259
  164. package/dist/esm/services/AIService.js +0 -185
  165. package/dist/esm/services/BasedService.js +0 -5278
  166. package/dist/esm/services/CoreService.js +0 -2264
  167. package/dist/esm/services/SocketIOService.js +0 -470
  168. package/dist/esm/services/SocketService.js +0 -191
  169. package/dist/esm/services/SymstoryService.js +0 -7041
  170. package/dist/esm/utils/basedQuerys.js +0 -163
  171. package/dist/esm/utils/symstoryClient.js +0 -370
  172. package/dist/node/services/AIService.js +0 -136
  173. package/dist/node/services/BasedService.js +0 -1156
  174. package/dist/node/services/CoreService.js +0 -1722
  175. package/dist/node/services/SocketIOService.js +0 -278
  176. package/dist/node/services/SocketService.js +0 -142
  177. package/dist/node/services/SymstoryService.js +0 -542
  178. package/dist/node/utils/basedQuerys.js +0 -162
  179. package/dist/node/utils/symstoryClient.js +0 -230
  180. package/src/services/AIService.js +0 -150
  181. package/src/services/BasedService.js +0 -1301
  182. package/src/services/CoreService.js +0 -1943
  183. package/src/services/SocketIOService.js +0 -334
  184. package/src/services/SocketService.js +0 -168
  185. package/src/services/SymstoryService.js +0 -649
  186. package/src/utils/basedQuerys.js +0 -164
  187. package/src/utils/symstoryClient.js +0 -252
@@ -0,0 +1,548 @@
1
+ import { BaseService } from './BaseService.js'
2
+
3
+ export class IntegrationService extends BaseService {
4
+ // ==================== INTEGRATION METHODS ====================
5
+
6
+ /**
7
+ * Programmatic auth sanity check (API key based).
8
+ *
9
+ * Mirrors: GET /integrations/whoami (requireApiKey)
10
+ *
11
+ * Assumption: backend reads API key from `x-api-key` header.
12
+ * You can override via `options.headers`.
13
+ */
14
+ async integrationWhoami (apiKey, options = {}) {
15
+ this._requireReady('integrationWhoami')
16
+ if (!apiKey) {
17
+ throw new Error('API key is required')
18
+ }
19
+
20
+ const headers = {
21
+ 'x-api-key': apiKey,
22
+ ...(options.headers || {})
23
+ }
24
+
25
+ try {
26
+ const response = await this._request('/integrations/whoami', {
27
+ method: 'GET',
28
+ headers,
29
+ methodName: 'integrationWhoami'
30
+ })
31
+ if (response.success) {
32
+ return response.data
33
+ }
34
+ throw new Error(response.message)
35
+ } catch (error) {
36
+ throw new Error(`Failed to validate integration API key: ${error.message}`, { cause: error })
37
+ }
38
+ }
39
+
40
+ /**
41
+ * List integrations visible to the user.
42
+ *
43
+ * Mirrors: GET /integrations?orgId=&projectId=
44
+ */
45
+ async listIntegrations (options = {}) {
46
+ this._requireReady('listIntegrations')
47
+
48
+ const { orgId, projectId } = options || {}
49
+ const queryParams = new URLSearchParams()
50
+ if (orgId != null) { queryParams.append('orgId', String(orgId)) }
51
+ if (projectId != null) { queryParams.append('projectId', String(projectId)) }
52
+
53
+ const queryString = queryParams.toString()
54
+ const url = `/integrations${queryString ? `?${queryString}` : ''}`
55
+
56
+ try {
57
+ const response = await this._request(url, {
58
+ method: 'GET',
59
+ methodName: 'listIntegrations'
60
+ })
61
+ if (response.success) {
62
+ return response.data
63
+ }
64
+ throw new Error(response.message)
65
+ } catch (error) {
66
+ throw new Error(`Failed to list integrations: ${error.message}`, { cause: error })
67
+ }
68
+ }
69
+
70
+ /**
71
+ * Create an integration.
72
+ *
73
+ * Mirrors: POST /integrations
74
+ */
75
+ async createIntegration (data = {}) {
76
+ this._requireReady('createIntegration')
77
+ if (!data || typeof data !== 'object') {
78
+ throw new Error('Integration payload is required')
79
+ }
80
+ if (!data.name) {
81
+ throw new Error('Integration name is required')
82
+ }
83
+ if (!data.ownerType) {
84
+ throw new Error('ownerType is required')
85
+ }
86
+
87
+ try {
88
+ const response = await this._request('/integrations', {
89
+ method: 'POST',
90
+ body: JSON.stringify(data),
91
+ methodName: 'createIntegration'
92
+ })
93
+ if (response.success) {
94
+ return response.data
95
+ }
96
+ throw new Error(response.message)
97
+ } catch (error) {
98
+ throw new Error(`Failed to create integration: ${error.message}`, { cause: error })
99
+ }
100
+ }
101
+
102
+ /**
103
+ * Update an integration.
104
+ *
105
+ * Mirrors: PATCH /integrations/:integrationId
106
+ */
107
+ async updateIntegration (integrationId, update = {}) {
108
+ this._requireReady('updateIntegration')
109
+ if (!integrationId) {
110
+ throw new Error('Integration ID is required')
111
+ }
112
+ if (!update || typeof update !== 'object') {
113
+ throw new Error('Update payload is required')
114
+ }
115
+
116
+ try {
117
+ const response = await this._request(`/integrations/${integrationId}`, {
118
+ method: 'PATCH',
119
+ body: JSON.stringify(update),
120
+ methodName: 'updateIntegration'
121
+ })
122
+ if (response.success) {
123
+ return response.data
124
+ }
125
+ throw new Error(response.message)
126
+ } catch (error) {
127
+ throw new Error(`Failed to update integration: ${error.message}`, { cause: error })
128
+ }
129
+ }
130
+
131
+ // ==================== INTEGRATION API KEY METHODS ====================
132
+
133
+ /**
134
+ * Create a new API key for an integration.
135
+ *
136
+ * Mirrors: POST /integrations/:integrationId/api-keys
137
+ */
138
+ async createIntegrationApiKey (integrationId, data = {}) {
139
+ this._requireReady('createIntegrationApiKey')
140
+ if (!integrationId) {
141
+ throw new Error('Integration ID is required')
142
+ }
143
+ if (!data || typeof data !== 'object') {
144
+ throw new Error('API key payload is required')
145
+ }
146
+
147
+ try {
148
+ const response = await this._request(`/integrations/${integrationId}/api-keys`, {
149
+ method: 'POST',
150
+ body: JSON.stringify(data),
151
+ methodName: 'createIntegrationApiKey'
152
+ })
153
+ if (response.success) {
154
+ return response.data
155
+ }
156
+ throw new Error(response.message)
157
+ } catch (error) {
158
+ throw new Error(`Failed to create integration API key: ${error.message}`, { cause: error })
159
+ }
160
+ }
161
+
162
+ /**
163
+ * List API keys for an integration.
164
+ *
165
+ * Mirrors: GET /integrations/:integrationId/api-keys
166
+ */
167
+ async listIntegrationApiKeys (integrationId) {
168
+ this._requireReady('listIntegrationApiKeys')
169
+ if (!integrationId) {
170
+ throw new Error('Integration ID is required')
171
+ }
172
+
173
+ try {
174
+ const response = await this._request(`/integrations/${integrationId}/api-keys`, {
175
+ method: 'GET',
176
+ methodName: 'listIntegrationApiKeys'
177
+ })
178
+ if (response.success) {
179
+ return response.data
180
+ }
181
+ throw new Error(response.message)
182
+ } catch (error) {
183
+ throw new Error(`Failed to list integration API keys: ${error.message}`, { cause: error })
184
+ }
185
+ }
186
+
187
+ /**
188
+ * Revoke an API key for an integration.
189
+ *
190
+ * Mirrors: POST /integrations/:integrationId/api-keys/:keyId/revoke
191
+ */
192
+ async revokeIntegrationApiKey (integrationId, keyId) {
193
+ this._requireReady('revokeIntegrationApiKey')
194
+ if (!integrationId) {
195
+ throw new Error('Integration ID is required')
196
+ }
197
+ if (!keyId) {
198
+ throw new Error('API key ID is required')
199
+ }
200
+
201
+ try {
202
+ const response = await this._request(
203
+ `/integrations/${integrationId}/api-keys/${keyId}/revoke`,
204
+ {
205
+ method: 'POST',
206
+ methodName: 'revokeIntegrationApiKey'
207
+ }
208
+ )
209
+ if (response.success) {
210
+ return response.data
211
+ }
212
+ throw new Error(response.message)
213
+ } catch (error) {
214
+ throw new Error(`Failed to revoke integration API key: ${error.message}`, { cause: error })
215
+ }
216
+ }
217
+
218
+ // ==================== WEBHOOK METHODS ====================
219
+
220
+ /**
221
+ * Create a webhook endpoint for an integration.
222
+ *
223
+ * Mirrors: POST /integrations/:integrationId/webhooks
224
+ */
225
+ async createIntegrationWebhook (integrationId, data = {}) {
226
+ this._requireReady('createIntegrationWebhook')
227
+ if (!integrationId) {
228
+ throw new Error('Integration ID is required')
229
+ }
230
+ if (!data || typeof data !== 'object') {
231
+ throw new Error('Webhook payload is required')
232
+ }
233
+
234
+ try {
235
+ const response = await this._request(`/integrations/${integrationId}/webhooks`, {
236
+ method: 'POST',
237
+ body: JSON.stringify(data),
238
+ methodName: 'createIntegrationWebhook'
239
+ })
240
+ if (response.success) {
241
+ return response.data
242
+ }
243
+ throw new Error(response.message)
244
+ } catch (error) {
245
+ throw new Error(`Failed to create integration webhook: ${error.message}`, { cause: error })
246
+ }
247
+ }
248
+
249
+ /**
250
+ * List webhook endpoints for an integration.
251
+ *
252
+ * Mirrors: GET /integrations/:integrationId/webhooks
253
+ */
254
+ async listIntegrationWebhooks (integrationId) {
255
+ this._requireReady('listIntegrationWebhooks')
256
+ if (!integrationId) {
257
+ throw new Error('Integration ID is required')
258
+ }
259
+
260
+ try {
261
+ const response = await this._request(`/integrations/${integrationId}/webhooks`, {
262
+ method: 'GET',
263
+ methodName: 'listIntegrationWebhooks'
264
+ })
265
+ if (response.success) {
266
+ return response.data
267
+ }
268
+ throw new Error(response.message)
269
+ } catch (error) {
270
+ throw new Error(`Failed to list integration webhooks: ${error.message}`, { cause: error })
271
+ }
272
+ }
273
+
274
+ /**
275
+ * Update a webhook endpoint for an integration.
276
+ *
277
+ * Mirrors: PATCH /integrations/:integrationId/webhooks/:webhookId
278
+ */
279
+ async updateIntegrationWebhook (integrationId, webhookId, update = {}) {
280
+ this._requireReady('updateIntegrationWebhook')
281
+ if (!integrationId) {
282
+ throw new Error('Integration ID is required')
283
+ }
284
+ if (!webhookId) {
285
+ throw new Error('Webhook ID is required')
286
+ }
287
+ if (!update || typeof update !== 'object') {
288
+ throw new Error('Update payload is required')
289
+ }
290
+
291
+ try {
292
+ const response = await this._request(
293
+ `/integrations/${integrationId}/webhooks/${webhookId}`,
294
+ {
295
+ method: 'PATCH',
296
+ body: JSON.stringify(update),
297
+ methodName: 'updateIntegrationWebhook'
298
+ }
299
+ )
300
+ if (response.success) {
301
+ return response.data
302
+ }
303
+ throw new Error(response.message)
304
+ } catch (error) {
305
+ throw new Error(`Failed to update integration webhook: ${error.message}`, { cause: error })
306
+ }
307
+ }
308
+
309
+ /**
310
+ * Delete a webhook endpoint for an integration.
311
+ *
312
+ * Mirrors: DELETE /integrations/:integrationId/webhooks/:webhookId
313
+ */
314
+ async deleteIntegrationWebhook (integrationId, webhookId) {
315
+ this._requireReady('deleteIntegrationWebhook')
316
+ if (!integrationId) {
317
+ throw new Error('Integration ID is required')
318
+ }
319
+ if (!webhookId) {
320
+ throw new Error('Webhook ID is required')
321
+ }
322
+
323
+ try {
324
+ const response = await this._request(
325
+ `/integrations/${integrationId}/webhooks/${webhookId}`,
326
+ {
327
+ method: 'DELETE',
328
+ methodName: 'deleteIntegrationWebhook'
329
+ }
330
+ )
331
+ if (response && response.success) {
332
+ return response.data
333
+ }
334
+ // Some endpoints may return 204; BaseService returns null then.
335
+ if (response == null) {
336
+ return null
337
+ }
338
+ throw new Error(response.message)
339
+ } catch (error) {
340
+ throw new Error(`Failed to delete integration webhook: ${error.message}`, { cause: error })
341
+ }
342
+ }
343
+
344
+ /**
345
+ * List webhook deliveries for an integration webhook.
346
+ *
347
+ * Mirrors: GET /integrations/:integrationId/webhooks/:webhookId/deliveries
348
+ */
349
+ async listIntegrationWebhookDeliveries (integrationId, webhookId, options = {}) {
350
+ this._requireReady('listIntegrationWebhookDeliveries')
351
+ if (!integrationId) {
352
+ throw new Error('Integration ID is required')
353
+ }
354
+ if (!webhookId) {
355
+ throw new Error('Webhook ID is required')
356
+ }
357
+
358
+ const { page, limit, status, includePayload } = options || {}
359
+ const queryParams = new URLSearchParams()
360
+ if (page != null) { queryParams.append('page', String(page)) }
361
+ if (limit != null) { queryParams.append('limit', String(limit)) }
362
+ if (status != null) { queryParams.append('status', String(status)) }
363
+ if (includePayload != null) { queryParams.append('includePayload', String(includePayload)) }
364
+
365
+ const queryString = queryParams.toString()
366
+ const url = `/integrations/${integrationId}/webhooks/${webhookId}/deliveries${queryString ? `?${queryString}` : ''}`
367
+
368
+ try {
369
+ const response = await this._request(url, {
370
+ method: 'GET',
371
+ methodName: 'listIntegrationWebhookDeliveries'
372
+ })
373
+ if (response.success) {
374
+ return response.data
375
+ }
376
+ throw new Error(response.message)
377
+ } catch (error) {
378
+ throw new Error(`Failed to list webhook deliveries: ${error.message}`, { cause: error })
379
+ }
380
+ }
381
+
382
+ /**
383
+ * Replay a webhook delivery.
384
+ *
385
+ * Mirrors: POST /integrations/:integrationId/webhooks/:webhookId/replay
386
+ * Body: { deliveryId }
387
+ */
388
+ async replayIntegrationWebhookDelivery (integrationId, webhookId, deliveryId) {
389
+ this._requireReady('replayIntegrationWebhookDelivery')
390
+ if (!integrationId) {
391
+ throw new Error('Integration ID is required')
392
+ }
393
+ if (!webhookId) {
394
+ throw new Error('Webhook ID is required')
395
+ }
396
+ if (!deliveryId) {
397
+ throw new Error('deliveryId is required')
398
+ }
399
+
400
+ try {
401
+ const response = await this._request(
402
+ `/integrations/${integrationId}/webhooks/${webhookId}/replay`,
403
+ {
404
+ method: 'POST',
405
+ body: JSON.stringify({ deliveryId }),
406
+ methodName: 'replayIntegrationWebhookDelivery'
407
+ }
408
+ )
409
+ if (response.success) {
410
+ return response.data
411
+ }
412
+ throw new Error(response.message)
413
+ } catch (error) {
414
+ throw new Error(`Failed to replay webhook delivery: ${error.message}`, { cause: error })
415
+ }
416
+ }
417
+
418
+ // ==================== CONNECTOR METHODS (GITHUB) ====================
419
+
420
+ /**
421
+ * List GitHub connectors for an integration.
422
+ *
423
+ * Mirrors: GET /integrations/:integrationId/connectors/github
424
+ */
425
+ async listGitHubConnectors (integrationId) {
426
+ this._requireReady('listGitHubConnectors')
427
+ if (!integrationId) {
428
+ throw new Error('Integration ID is required')
429
+ }
430
+
431
+ try {
432
+ const response = await this._request(`/integrations/${integrationId}/connectors/github`, {
433
+ method: 'GET',
434
+ methodName: 'listGitHubConnectors'
435
+ })
436
+ if (response.success) {
437
+ return response.data
438
+ }
439
+ throw new Error(response.message)
440
+ } catch (error) {
441
+ throw new Error(`Failed to list GitHub connectors: ${error.message}`, { cause: error })
442
+ }
443
+ }
444
+
445
+ /**
446
+ * Create a GitHub connector for an integration.
447
+ *
448
+ * Mirrors: POST /integrations/:integrationId/connectors/github
449
+ */
450
+ async createGitHubConnector (integrationId, data = {}) {
451
+ this._requireReady('createGitHubConnector')
452
+ if (!integrationId) {
453
+ throw new Error('Integration ID is required')
454
+ }
455
+ if (!data || typeof data !== 'object') {
456
+ throw new Error('Connector payload is required')
457
+ }
458
+ if (!data.projectId) {
459
+ throw new Error('projectId is required')
460
+ }
461
+ if (!data.repository) {
462
+ throw new Error('repository is required')
463
+ }
464
+
465
+ try {
466
+ const response = await this._request(`/integrations/${integrationId}/connectors/github`, {
467
+ method: 'POST',
468
+ body: JSON.stringify(data),
469
+ methodName: 'createGitHubConnector'
470
+ })
471
+ if (response.success) {
472
+ return response.data
473
+ }
474
+ throw new Error(response.message)
475
+ } catch (error) {
476
+ throw new Error(`Failed to create GitHub connector: ${error.message}`, { cause: error })
477
+ }
478
+ }
479
+
480
+ /**
481
+ * Update a GitHub connector.
482
+ *
483
+ * Mirrors: PATCH /integrations/:integrationId/connectors/github/:connectorId
484
+ */
485
+ async updateGitHubConnector (integrationId, connectorId, update = {}) {
486
+ this._requireReady('updateGitHubConnector')
487
+ if (!integrationId) {
488
+ throw new Error('Integration ID is required')
489
+ }
490
+ if (!connectorId) {
491
+ throw new Error('Connector ID is required')
492
+ }
493
+ if (!update || typeof update !== 'object') {
494
+ throw new Error('Update payload is required')
495
+ }
496
+
497
+ try {
498
+ const response = await this._request(
499
+ `/integrations/${integrationId}/connectors/github/${connectorId}`,
500
+ {
501
+ method: 'PATCH',
502
+ body: JSON.stringify(update),
503
+ methodName: 'updateGitHubConnector'
504
+ }
505
+ )
506
+ if (response.success) {
507
+ return response.data
508
+ }
509
+ throw new Error(response.message)
510
+ } catch (error) {
511
+ throw new Error(`Failed to update GitHub connector: ${error.message}`, { cause: error })
512
+ }
513
+ }
514
+
515
+ /**
516
+ * Delete a GitHub connector.
517
+ *
518
+ * Mirrors: DELETE /integrations/:integrationId/connectors/github/:connectorId
519
+ */
520
+ async deleteGitHubConnector (integrationId, connectorId) {
521
+ this._requireReady('deleteGitHubConnector')
522
+ if (!integrationId) {
523
+ throw new Error('Integration ID is required')
524
+ }
525
+ if (!connectorId) {
526
+ throw new Error('Connector ID is required')
527
+ }
528
+
529
+ try {
530
+ const response = await this._request(
531
+ `/integrations/${integrationId}/connectors/github/${connectorId}`,
532
+ {
533
+ method: 'DELETE',
534
+ methodName: 'deleteGitHubConnector'
535
+ }
536
+ )
537
+ if (response && response.success) {
538
+ return response.data
539
+ }
540
+ if (response == null) {
541
+ return null
542
+ }
543
+ throw new Error(response.message)
544
+ } catch (error) {
545
+ throw new Error(`Failed to delete GitHub connector: ${error.message}`, { cause: error })
546
+ }
547
+ }
548
+ }
@@ -0,0 +1,40 @@
1
+ import { BaseService } from './BaseService.js'
2
+
3
+ export class MetricsService extends BaseService {
4
+ // ==================== METRICS METHODS ====================
5
+
6
+ /**
7
+ * Contribution heat-map stats.
8
+ *
9
+ * Mirrors: GET /metrics/contributions (MetricsController.getContributions)
10
+ */
11
+ async getContributions (options = {}) {
12
+ this._requireReady('getContributions')
13
+
14
+ const { projectId, userId, from, to } = options || {}
15
+ const queryParams = new URLSearchParams()
16
+
17
+ if (projectId != null) { queryParams.append('projectId', String(projectId)) }
18
+ if (userId != null) { queryParams.append('userId', String(userId)) }
19
+ if (from != null) { queryParams.append('from', String(from)) }
20
+ if (to != null) { queryParams.append('to', String(to)) }
21
+
22
+ const queryString = queryParams.toString()
23
+ const url = `/metrics/contributions${queryString ? `?${queryString}` : ''}`
24
+
25
+ try {
26
+ const response = await this._request(url, {
27
+ method: 'GET',
28
+ methodName: 'getContributions'
29
+ })
30
+ if (response.success) {
31
+ return response.data
32
+ }
33
+ throw new Error(response.message)
34
+ } catch (error) {
35
+ throw new Error(`Failed to get contribution stats: ${error.message}`, { cause: error })
36
+ }
37
+ }
38
+ }
39
+
40
+ // End