@symbo.ls/sdk 2.34.35 → 3.1.1

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 +13 -315
  2. package/dist/cjs/config/environment.js +50 -124
  3. package/dist/cjs/index.js +22 -169
  4. package/dist/cjs/services/AIService.js +155 -0
  5. package/dist/cjs/services/AuthService.js +277 -751
  6. package/dist/cjs/services/BaseService.js +6 -158
  7. package/dist/cjs/services/BasedService.js +679 -0
  8. package/dist/cjs/services/SocketIOService.js +309 -0
  9. package/dist/cjs/services/SocketService.js +161 -0
  10. package/dist/cjs/services/SymstoryService.js +485 -0
  11. package/dist/cjs/services/index.js +16 -64
  12. package/dist/cjs/utils/basedQuerys.js +61 -0
  13. package/dist/cjs/utils/permission.js +4 -4
  14. package/dist/cjs/utils/services.js +80 -301
  15. package/dist/cjs/utils/symstoryClient.js +228 -0
  16. package/dist/cjs/utils/validation.js +3 -0
  17. package/dist/esm/config/environment.js +50 -124
  18. package/dist/esm/index.js +11282 -49789
  19. package/dist/esm/services/AIService.js +185 -0
  20. package/dist/esm/services/AuthService.js +358 -1506
  21. package/dist/esm/services/BaseService.js +6 -757
  22. package/dist/esm/services/BasedService.js +4651 -0
  23. package/dist/esm/services/SocketIOService.js +467 -0
  24. package/dist/esm/services/SocketService.js +191 -0
  25. package/dist/esm/services/SymstoryService.js +6849 -0
  26. package/dist/esm/services/index.js +11188 -49352
  27. package/dist/esm/utils/basedQuerys.js +43 -0
  28. package/dist/esm/utils/permission.js +4 -4
  29. package/dist/esm/utils/services.js +80 -301
  30. package/dist/esm/utils/symstoryClient.js +334 -0
  31. package/dist/esm/utils/validation.js +26 -93
  32. package/dist/node/config/environment.js +50 -124
  33. package/dist/node/index.js +26 -202
  34. package/dist/node/services/AIService.js +136 -0
  35. package/dist/node/services/AuthService.js +278 -751
  36. package/dist/node/services/BaseService.js +6 -148
  37. package/dist/node/services/BasedService.js +650 -0
  38. package/dist/node/services/SocketIOService.js +280 -0
  39. package/dist/node/services/SocketService.js +142 -0
  40. package/dist/node/services/SymstoryService.js +456 -0
  41. package/dist/node/services/index.js +16 -64
  42. package/dist/node/utils/basedQuerys.js +42 -0
  43. package/dist/node/utils/permission.js +4 -4
  44. package/dist/node/utils/services.js +80 -301
  45. package/dist/node/utils/symstoryClient.js +199 -0
  46. package/dist/node/utils/validation.js +3 -0
  47. package/package.json +21 -43
  48. package/src/config/environment.js +50 -126
  49. package/src/index.js +24 -208
  50. package/src/services/AIService.js +150 -0
  51. package/src/services/AuthService.js +298 -893
  52. package/src/services/BaseService.js +6 -166
  53. package/src/services/BasedService.js +722 -0
  54. package/src/services/SocketIOService.js +356 -0
  55. package/src/services/SocketService.js +168 -0
  56. package/src/services/SymstoryService.js +563 -0
  57. package/src/services/index.js +13 -80
  58. package/src/utils/basedQuerys.js +41 -0
  59. package/src/utils/permission.js +4 -4
  60. package/src/utils/services.js +83 -325
  61. package/src/utils/symstoryClient.js +218 -0
  62. package/src/utils/validation.js +3 -0
  63. package/dist/cjs/services/AdminService.js +0 -351
  64. package/dist/cjs/services/BranchService.js +0 -484
  65. package/dist/cjs/services/CollabService.js +0 -743
  66. package/dist/cjs/services/DnsService.js +0 -340
  67. package/dist/cjs/services/FeatureFlagService.js +0 -175
  68. package/dist/cjs/services/FileService.js +0 -201
  69. package/dist/cjs/services/IntegrationService.js +0 -538
  70. package/dist/cjs/services/MetricsService.js +0 -62
  71. package/dist/cjs/services/PaymentService.js +0 -271
  72. package/dist/cjs/services/PlanService.js +0 -426
  73. package/dist/cjs/services/ProjectService.js +0 -1207
  74. package/dist/cjs/services/PullRequestService.js +0 -503
  75. package/dist/cjs/services/ScreenshotService.js +0 -304
  76. package/dist/cjs/services/SubscriptionService.js +0 -396
  77. package/dist/cjs/services/TrackingService.js +0 -661
  78. package/dist/cjs/services/WaitlistService.js +0 -148
  79. package/dist/cjs/state/RootStateManager.js +0 -65
  80. package/dist/cjs/state/rootEventBus.js +0 -74
  81. package/dist/cjs/utils/CollabClient.js +0 -223
  82. package/dist/cjs/utils/TokenManager.js +0 -422
  83. package/dist/cjs/utils/changePreprocessor.js +0 -199
  84. package/dist/cjs/utils/jsonDiff.js +0 -145
  85. package/dist/cjs/utils/ordering.js +0 -309
  86. package/dist/esm/services/AdminService.js +0 -1132
  87. package/dist/esm/services/BranchService.js +0 -1265
  88. package/dist/esm/services/CollabService.js +0 -26838
  89. package/dist/esm/services/DnsService.js +0 -1121
  90. package/dist/esm/services/FeatureFlagService.js +0 -956
  91. package/dist/esm/services/FileService.js +0 -982
  92. package/dist/esm/services/IntegrationService.js +0 -1319
  93. package/dist/esm/services/MetricsService.js +0 -843
  94. package/dist/esm/services/PaymentService.js +0 -1052
  95. package/dist/esm/services/PlanService.js +0 -1207
  96. package/dist/esm/services/ProjectService.js +0 -2526
  97. package/dist/esm/services/PullRequestService.js +0 -1284
  98. package/dist/esm/services/ScreenshotService.js +0 -1085
  99. package/dist/esm/services/SubscriptionService.js +0 -1177
  100. package/dist/esm/services/TrackingService.js +0 -18343
  101. package/dist/esm/services/WaitlistService.js +0 -929
  102. package/dist/esm/state/RootStateManager.js +0 -90
  103. package/dist/esm/state/rootEventBus.js +0 -56
  104. package/dist/esm/utils/CollabClient.js +0 -18901
  105. package/dist/esm/utils/TokenManager.js +0 -408
  106. package/dist/esm/utils/changePreprocessor.js +0 -542
  107. package/dist/esm/utils/jsonDiff.js +0 -7011
  108. package/dist/esm/utils/ordering.js +0 -291
  109. package/dist/node/services/AdminService.js +0 -332
  110. package/dist/node/services/BranchService.js +0 -465
  111. package/dist/node/services/CollabService.js +0 -724
  112. package/dist/node/services/DnsService.js +0 -321
  113. package/dist/node/services/FeatureFlagService.js +0 -156
  114. package/dist/node/services/FileService.js +0 -182
  115. package/dist/node/services/IntegrationService.js +0 -519
  116. package/dist/node/services/MetricsService.js +0 -43
  117. package/dist/node/services/PaymentService.js +0 -252
  118. package/dist/node/services/PlanService.js +0 -407
  119. package/dist/node/services/ProjectService.js +0 -1188
  120. package/dist/node/services/PullRequestService.js +0 -484
  121. package/dist/node/services/ScreenshotService.js +0 -285
  122. package/dist/node/services/SubscriptionService.js +0 -377
  123. package/dist/node/services/TrackingService.js +0 -632
  124. package/dist/node/services/WaitlistService.js +0 -129
  125. package/dist/node/state/RootStateManager.js +0 -36
  126. package/dist/node/state/rootEventBus.js +0 -55
  127. package/dist/node/utils/CollabClient.js +0 -194
  128. package/dist/node/utils/TokenManager.js +0 -403
  129. package/dist/node/utils/changePreprocessor.js +0 -180
  130. package/dist/node/utils/jsonDiff.js +0 -116
  131. package/dist/node/utils/ordering.js +0 -290
  132. package/src/services/AdminService.js +0 -374
  133. package/src/services/BranchService.js +0 -536
  134. package/src/services/CollabService.js +0 -900
  135. package/src/services/DnsService.js +0 -366
  136. package/src/services/FeatureFlagService.js +0 -174
  137. package/src/services/FileService.js +0 -213
  138. package/src/services/IntegrationService.js +0 -548
  139. package/src/services/MetricsService.js +0 -40
  140. package/src/services/PaymentService.js +0 -287
  141. package/src/services/PlanService.js +0 -468
  142. package/src/services/ProjectService.js +0 -1366
  143. package/src/services/PullRequestService.js +0 -537
  144. package/src/services/ScreenshotService.js +0 -258
  145. package/src/services/SubscriptionService.js +0 -425
  146. package/src/services/TrackingService.js +0 -853
  147. package/src/services/WaitlistService.js +0 -130
  148. package/src/services/tests/BranchService/createBranch.test.js +0 -153
  149. package/src/services/tests/BranchService/deleteBranch.test.js +0 -173
  150. package/src/services/tests/BranchService/getBranchChanges.test.js +0 -146
  151. package/src/services/tests/BranchService/listBranches.test.js +0 -87
  152. package/src/services/tests/BranchService/mergeBranch.test.js +0 -210
  153. package/src/services/tests/BranchService/publishVersion.test.js +0 -183
  154. package/src/services/tests/BranchService/renameBranch.test.js +0 -240
  155. package/src/services/tests/BranchService/resetBranch.test.js +0 -152
  156. package/src/services/tests/FeatureFlagService/adminFeatureFlags.test.js +0 -67
  157. package/src/services/tests/FeatureFlagService/getFeatureFlags.test.js +0 -75
  158. package/src/services/tests/FileService/createFileFormData.test.js +0 -74
  159. package/src/services/tests/FileService/getFileUrl.test.js +0 -69
  160. package/src/services/tests/FileService/updateProjectIcon.test.js +0 -109
  161. package/src/services/tests/FileService/uploadDocument.test.js +0 -36
  162. package/src/services/tests/FileService/uploadFile.test.js +0 -78
  163. package/src/services/tests/FileService/uploadFileWithValidation.test.js +0 -114
  164. package/src/services/tests/FileService/uploadImage.test.js +0 -36
  165. package/src/services/tests/FileService/uploadMultipleFiles.test.js +0 -111
  166. package/src/services/tests/FileService/validateFile.test.js +0 -63
  167. package/src/services/tests/PlanService/createPlan.test.js +0 -104
  168. package/src/services/tests/PlanService/createPlanWithValidation.test.js +0 -523
  169. package/src/services/tests/PlanService/deletePlan.test.js +0 -92
  170. package/src/services/tests/PlanService/getActivePlans.test.js +0 -123
  171. package/src/services/tests/PlanService/getAdminPlans.test.js +0 -84
  172. package/src/services/tests/PlanService/getPlan.test.js +0 -50
  173. package/src/services/tests/PlanService/getPlanByKey.test.js +0 -109
  174. package/src/services/tests/PlanService/getPlanWithValidation.test.js +0 -85
  175. package/src/services/tests/PlanService/getPlans.test.js +0 -53
  176. package/src/services/tests/PlanService/getPlansByPriceRange.test.js +0 -109
  177. package/src/services/tests/PlanService/getPlansWithValidation.test.js +0 -48
  178. package/src/services/tests/PlanService/initializePlans.test.js +0 -75
  179. package/src/services/tests/PlanService/updatePlan.test.js +0 -111
  180. package/src/services/tests/PlanService/updatePlanWithValidation.test.js +0 -556
  181. package/src/state/RootStateManager.js +0 -76
  182. package/src/state/rootEventBus.js +0 -67
  183. package/src/utils/CollabClient.js +0 -248
  184. package/src/utils/TokenManager.js +0 -479
  185. package/src/utils/changePreprocessor.js +0 -239
  186. package/src/utils/jsonDiff.js +0 -144
  187. package/src/utils/ordering.js +0 -271
@@ -1,1177 +0,0 @@
1
- var __defProp = Object.defineProperty;
2
- var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
3
- var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
4
-
5
- // src/config/environment.js
6
- import { isDevelopment } from "@domql/utils";
7
- var CONFIG = {
8
- // Common defaults for all environments
9
- common: {
10
- // NOTE: Google client id for google auth, need to configure URLs for each environment in Google console
11
- googleClientId: "686286207466-bvd2fqs31rlm64fgich7rtpnc8ns2tqg.apps.googleusercontent.com",
12
- // Feature toggles that apply across all environments by default
13
- features: {
14
- newUserOnboarding: true,
15
- betaFeatures: false,
16
- // Tracking is enabled by default unless overridden per environment
17
- trackingEnabled: true
18
- }
19
- },
20
- // Environment-specific configurations
21
- local: {
22
- // local
23
- socketUrl: "http://localhost:8080",
24
- // For socket api
25
- apiUrl: "http://localhost:8080",
26
- // For server api
27
- basedEnv: "development",
28
- // For based api
29
- basedProject: "platform-v2-sm",
30
- // For based api
31
- basedOrg: "symbols",
32
- // For based api
33
- githubClientId: "Ov23liAFrsR0StbAO6PO",
34
- // For github api
35
- grafanaUrl: "",
36
- // For grafana tracing
37
- grafanaAppName: "Symbols Localhost",
38
- // Environment-specific feature toggles (override common)
39
- features: {
40
- // Disable tracking by default on localhost/dev machines
41
- trackingEnabled: false,
42
- // Enable beta features in local dev
43
- betaFeatures: true,
44
- // Preserve common defaults explicitly for local
45
- newUserOnboarding: true
46
- },
47
- typesenseCollectionName: "docs",
48
- typesenseApiKey: "vZya3L2zpq8L6iI5WWMUZJZABvT63VDb",
49
- typesenseHost: "localhost",
50
- typesensePort: "8108",
51
- typesenseProtocol: "http"
52
- },
53
- development: {
54
- socketUrl: "https://dev.api.symbols.app",
55
- apiUrl: "https://dev.api.symbols.app",
56
- githubClientId: "Ov23liHxyWFBxS8f1gnF",
57
- grafanaUrl: "https://faro-collector-prod-us-east-0.grafana.net/collect/7a3ba473cee2025c68513667024316b8",
58
- // For grafana tracing
59
- grafanaAppName: "Symbols Dev",
60
- typesenseCollectionName: "docs",
61
- typesenseApiKey: "awmcVpbWqZi9IUgmvslp1C5LKDU8tMjA",
62
- typesenseHost: "tl2qpnwxev4cjm36p-1.a1.typesense.net",
63
- typesensePort: "443",
64
- typesenseProtocol: "https"
65
- },
66
- testing: {
67
- socketUrl: "https://test.api.symbols.app",
68
- apiUrl: "https://test.api.symbols.app",
69
- basedEnv: "testing",
70
- basedProject: "platform-v2-sm",
71
- basedOrg: "symbols",
72
- githubClientId: "Ov23liHxyWFBxS8f1gnF",
73
- grafanaUrl: "",
74
- // For grafana tracing
75
- grafanaAppName: "Symbols Test",
76
- typesenseCollectionName: "docs",
77
- typesenseApiKey: "awmcVpbWqZi9IUgmvslp1C5LKDU8tMjA",
78
- typesenseHost: "tl2qpnwxev4cjm36p-1.a1.typesense.net",
79
- typesensePort: "443",
80
- typesenseProtocol: "https"
81
- },
82
- upcoming: {
83
- socketUrl: "https://upcoming.api.symbols.app",
84
- apiUrl: "https://upcoming.api.symbols.app",
85
- githubClientId: "Ov23liWF7NvdZ056RV5J",
86
- grafanaUrl: "",
87
- // For grafana tracing
88
- grafanaAppName: "Symbols Upcoming",
89
- typesenseCollectionName: "docs",
90
- typesenseApiKey: "awmcVpbWqZi9IUgmvslp1C5LKDU8tMjA",
91
- typesenseHost: "tl2qpnwxev4cjm36p-1.a1.typesense.net",
92
- typesensePort: "443",
93
- typesenseProtocol: "https"
94
- },
95
- staging: {
96
- socketUrl: "https://staging.api.symbols.app",
97
- apiUrl: "https://staging.api.symbols.app",
98
- basedEnv: "staging",
99
- basedProject: "platform-v2-sm",
100
- basedOrg: "symbols",
101
- githubClientId: "Ov23ligwZDQVD0VfuWNa",
102
- grafanaUrl: "",
103
- // For grafana tracing
104
- grafanaAppName: "Symbols Staging",
105
- typesenseCollectionName: "docs",
106
- typesenseApiKey: "awmcVpbWqZi9IUgmvslp1C5LKDU8tMjA",
107
- typesenseHost: "tl2qpnwxev4cjm36p-1.a1.typesense.net",
108
- typesensePort: "443",
109
- typesenseProtocol: "https"
110
- },
111
- preview: {
112
- socketUrl: "https://api.symbols.app",
113
- apiUrl: "https://api.symbols.app",
114
- basedEnv: "production",
115
- basedProject: "platform-v2-sm",
116
- basedOrg: "symbols",
117
- githubClientId: "Ov23liFAlOEIXtX3dBtR",
118
- grafanaUrl: "https://faro-collector-prod-us-east-0.grafana.net/collect/5c1089f3c3eea4ec5658e05c3f53baae",
119
- // For grafana tracing
120
- grafanaAppName: "Symbols Preview",
121
- typesenseCollectionName: "docs",
122
- typesenseApiKey: "awmcVpbWqZi9IUgmvslp1C5LKDU8tMjA",
123
- typesenseHost: "tl2qpnwxev4cjm36p-1.a1.typesense.net",
124
- typesensePort: "443",
125
- typesenseProtocol: "https"
126
- },
127
- production: {
128
- socketUrl: "https://api.symbols.app",
129
- apiUrl: "https://api.symbols.app",
130
- basedEnv: "production",
131
- basedProject: "platform-v2-sm",
132
- basedOrg: "symbols",
133
- githubClientId: "Ov23liFAlOEIXtX3dBtR",
134
- grafanaUrl: "https://faro-collector-prod-us-east-0.grafana.net/collect/5c1089f3c3eea4ec5658e05c3f53baae",
135
- // For grafana tracing
136
- grafanaAppName: "Symbols",
137
- typesenseCollectionName: "docs",
138
- typesenseApiKey: "awmcVpbWqZi9IUgmvslp1C5LKDU8tMjA",
139
- typesenseHost: "tl2qpnwxev4cjm36p-1.a1.typesense.net",
140
- typesensePort: "443",
141
- typesenseProtocol: "https"
142
- }
143
- };
144
- var getEnvironment = () => {
145
- const env = process.env.SYMBOLS_APP_ENV || process.env.NODE_ENV;
146
- if (!CONFIG[env]) {
147
- throw new Error(`Unknown environment "${env}"`);
148
- }
149
- return env;
150
- };
151
- var getConfig = () => {
152
- try {
153
- const env = getEnvironment();
154
- const envConfig = { ...CONFIG.common, ...CONFIG[env] };
155
- const finalConfig = {
156
- ...envConfig,
157
- // Deep-merge feature flags so env-specific overrides don't drop common defaults
158
- features: {
159
- ...CONFIG.common.features || {},
160
- ...CONFIG[env] && CONFIG[env].features || {}
161
- },
162
- socketUrl: process.env.SYMBOLS_APP_SOCKET_URL || envConfig.socketUrl,
163
- apiUrl: process.env.SYMBOLS_APP_API_URL || envConfig.apiUrl,
164
- basedEnv: process.env.SYMBOLS_APP_BASED_ENV || envConfig.basedEnv,
165
- basedProject: process.env.SYMBOLS_APP_BASED_PROJECT || envConfig.basedProject,
166
- basedOrg: process.env.SYMBOLS_APP_BASED_ORG || envConfig.basedOrg,
167
- githubClientId: process.env.SYMBOLS_APP_GITHUB_CLIENT_ID || envConfig.githubClientId,
168
- grafanaUrl: process.env.SYMBOLS_APP_GRAFANA_URL || envConfig.grafanaUrl,
169
- typesenseCollectionName: process.env.TYPESENSE_COLLECTION_NAME || envConfig.typesenseCollectionName,
170
- typesenseApiKey: process.env.TYPESENSE_API_KEY || envConfig.typesenseApiKey,
171
- typesenseHost: process.env.TYPESENSE_HOST || envConfig.typesenseHost,
172
- typesensePort: process.env.TYPESENSE_PORT || envConfig.typesensePort,
173
- typesenseProtocol: process.env.TYPESENSE_PROTOCOL || envConfig.typesenseProtocol,
174
- isDevelopment: isDevelopment(env),
175
- isTesting: env === "testing",
176
- isStaging: env === "staging",
177
- isPreview: env === "preview",
178
- isProduction: env === "production"
179
- // Store all environment variables for potential future use
180
- };
181
- const requiredFields = [
182
- "socketUrl",
183
- "apiUrl",
184
- "githubClientId",
185
- "googleClientId"
186
- ];
187
- const missingFields = requiredFields.filter((field) => !finalConfig[field]);
188
- if (missingFields.length > 0) {
189
- console.error(
190
- `Missing required configuration: ${missingFields.join(", ")}`
191
- );
192
- }
193
- if (finalConfig.isDevelopment) {
194
- console.warn(
195
- "environment in SDK:",
196
- env || process.env.NODE_ENV || process.env.NODE_ENV
197
- );
198
- console.log(finalConfig);
199
- } else if (global.window) {
200
- global.window.finalConfig = finalConfig;
201
- }
202
- return finalConfig;
203
- } catch (error) {
204
- console.error("Failed to load environment configuration:", error);
205
- return {
206
- ...CONFIG.development
207
- };
208
- }
209
- };
210
- var environment_default = getConfig();
211
-
212
- // src/utils/TokenManager.js
213
- var TokenManager = class {
214
- constructor(options = {}) {
215
- /**
216
- * Memory storage fallback for server-side rendering
217
- */
218
- __publicField(this, "_memoryStorage", {
219
- _data: {},
220
- getItem: (key) => this._memoryStorage._data[key] || null,
221
- setItem: (key, value) => {
222
- this._memoryStorage._data[key] = value;
223
- },
224
- removeItem: (key) => {
225
- delete this._memoryStorage._data[key];
226
- },
227
- clear: () => {
228
- this._memoryStorage._data = {};
229
- }
230
- });
231
- this.config = {
232
- storagePrefix: "symbols_",
233
- storageType: typeof window === "undefined" || process.env.NODE_ENV === "test" || process.env.NODE_ENV === "testing" ? "memory" : "localStorage",
234
- // 'localStorage' | 'sessionStorage' | 'memory'
235
- refreshBuffer: 60 * 1e3,
236
- // Refresh 1 minute before expiry
237
- maxRetries: 3,
238
- apiUrl: options.apiUrl || "/api",
239
- onTokenRefresh: options.onTokenRefresh || null,
240
- onTokenExpired: options.onTokenExpired || null,
241
- onTokenError: options.onTokenError || null,
242
- ...options
243
- };
244
- this.tokens = {
245
- accessToken: null,
246
- refreshToken: null,
247
- expiresAt: null,
248
- expiresIn: null
249
- };
250
- this.refreshPromise = null;
251
- this.refreshTimeout = null;
252
- this.retryCount = 0;
253
- this.loadTokens();
254
- }
255
- /**
256
- * Storage keys
257
- */
258
- get storageKeys() {
259
- return {
260
- accessToken: `${this.config.storagePrefix}access_token`,
261
- refreshToken: `${this.config.storagePrefix}refresh_token`,
262
- expiresAt: `${this.config.storagePrefix}expires_at`,
263
- expiresIn: `${this.config.storagePrefix}expires_in`
264
- };
265
- }
266
- /**
267
- * Get storage instance based on configuration
268
- */
269
- get storage() {
270
- if (typeof window === "undefined") {
271
- return this._memoryStorage;
272
- }
273
- const safeGetStorage = (provider) => {
274
- try {
275
- const storage = provider();
276
- const testKey = `${this.config.storagePrefix}__tm_test__`;
277
- storage.setItem(testKey, "1");
278
- storage.removeItem(testKey);
279
- return storage;
280
- } catch {
281
- return null;
282
- }
283
- };
284
- const localStorageInstance = safeGetStorage(() => window.localStorage);
285
- const sessionStorageInstance = safeGetStorage(() => window.sessionStorage);
286
- switch (this.config.storageType) {
287
- case "sessionStorage":
288
- return sessionStorageInstance || this._memoryStorage;
289
- case "memory":
290
- return this._memoryStorage;
291
- default:
292
- return localStorageInstance || this._memoryStorage;
293
- }
294
- }
295
- /**
296
- * Set tokens and persist to storage
297
- */
298
- setTokens(tokenData) {
299
- const {
300
- access_token: accessToken,
301
- refresh_token: refreshToken,
302
- expires_in: expiresIn,
303
- token_type: tokenType = "Bearer"
304
- } = tokenData;
305
- if (!accessToken) {
306
- throw new Error("Access token is required");
307
- }
308
- const now = Date.now();
309
- const expiresAt = expiresIn ? now + expiresIn * 1e3 : null;
310
- this.tokens = {
311
- accessToken,
312
- refreshToken: refreshToken || this.tokens.refreshToken,
313
- expiresAt,
314
- expiresIn,
315
- tokenType
316
- };
317
- this.saveTokens();
318
- this.scheduleRefresh();
319
- if (this.config.onTokenRefresh) {
320
- this.config.onTokenRefresh(this.tokens);
321
- }
322
- return this.tokens;
323
- }
324
- /**
325
- * Get current access token
326
- */
327
- getAccessToken() {
328
- return this.tokens.accessToken;
329
- }
330
- /**
331
- * Get current refresh token
332
- */
333
- getRefreshToken() {
334
- return this.tokens.refreshToken;
335
- }
336
- /**
337
- * Get authorization header value
338
- */
339
- getAuthHeader() {
340
- const token = this.getAccessToken();
341
- if (!token) {
342
- return null;
343
- }
344
- return `${this.tokens.tokenType || "Bearer"} ${token}`;
345
- }
346
- /**
347
- * Check if access token is valid and not expired
348
- */
349
- isAccessTokenValid() {
350
- if (!this.tokens.accessToken) {
351
- return false;
352
- }
353
- if (!this.tokens.expiresAt) {
354
- return true;
355
- }
356
- const now = Date.now();
357
- const isValid = now < this.tokens.expiresAt - this.config.refreshBuffer;
358
- if (!isValid) {
359
- console.log("[TokenManager] Access token is expired or near expiry:", {
360
- now: new Date(now).toISOString(),
361
- expiresAt: new Date(this.tokens.expiresAt).toISOString(),
362
- refreshBuffer: this.config.refreshBuffer
363
- });
364
- }
365
- return isValid;
366
- }
367
- /**
368
- * Check if access token exists and is not expired (without refresh buffer)
369
- */
370
- isAccessTokenActuallyValid() {
371
- if (!this.tokens.accessToken) {
372
- return false;
373
- }
374
- if (!this.tokens.expiresAt) {
375
- return true;
376
- }
377
- const now = Date.now();
378
- return now < this.tokens.expiresAt;
379
- }
380
- /**
381
- * Check if tokens exist (regardless of expiry)
382
- */
383
- hasTokens() {
384
- return Boolean(this.tokens.accessToken);
385
- }
386
- /**
387
- * Check if refresh token exists
388
- */
389
- hasRefreshToken() {
390
- return Boolean(this.tokens.refreshToken);
391
- }
392
- /**
393
- * Automatically refresh tokens if needed
394
- */
395
- async ensureValidToken() {
396
- if (!this.hasTokens()) {
397
- return null;
398
- }
399
- if (this.isAccessTokenValid()) {
400
- return this.getAccessToken();
401
- }
402
- if (!this.hasRefreshToken()) {
403
- this.clearTokens();
404
- if (this.config.onTokenExpired) {
405
- this.config.onTokenExpired();
406
- }
407
- return null;
408
- }
409
- try {
410
- await this.refreshTokens();
411
- return this.getAccessToken();
412
- } catch (error) {
413
- this.clearTokens();
414
- if (this.config.onTokenError) {
415
- this.config.onTokenError(error);
416
- }
417
- throw error;
418
- }
419
- }
420
- /**
421
- * Refresh access token using refresh token
422
- */
423
- async refreshTokens() {
424
- if (this.refreshPromise) {
425
- return this.refreshPromise;
426
- }
427
- if (!this.hasRefreshToken()) {
428
- throw new Error("No refresh token available");
429
- }
430
- if (this.retryCount >= this.config.maxRetries) {
431
- throw new Error("Max refresh retries exceeded");
432
- }
433
- this.refreshPromise = this._performRefresh();
434
- try {
435
- const result = await this.refreshPromise;
436
- this.retryCount = 0;
437
- return result;
438
- } catch (error) {
439
- this.retryCount++;
440
- throw error;
441
- } finally {
442
- this.refreshPromise = null;
443
- }
444
- }
445
- /**
446
- * Perform the actual token refresh request
447
- */
448
- async _performRefresh() {
449
- var _a;
450
- const refreshToken = this.getRefreshToken();
451
- const response = await fetch(`${this.config.apiUrl}/core/auth/refresh`, {
452
- method: "POST",
453
- headers: {
454
- "Content-Type": "application/json"
455
- },
456
- body: JSON.stringify({ refreshToken })
457
- });
458
- if (!response.ok) {
459
- const errorData = await response.json().catch(() => ({}));
460
- throw new Error(errorData.message || `Token refresh failed: ${response.status}`);
461
- }
462
- const responseData = await response.json();
463
- if (responseData.success && responseData.data && responseData.data.tokens) {
464
- const { tokens } = responseData.data;
465
- const tokenData = {
466
- access_token: tokens.accessToken,
467
- refresh_token: tokens.refreshToken,
468
- expires_in: (_a = tokens.accessTokenExp) == null ? void 0 : _a.expiresIn,
469
- token_type: "Bearer"
470
- };
471
- return this.setTokens(tokenData);
472
- }
473
- return this.setTokens(responseData);
474
- }
475
- /**
476
- * Schedule automatic token refresh
477
- */
478
- scheduleRefresh() {
479
- if (this.refreshTimeout) {
480
- clearTimeout(this.refreshTimeout);
481
- this.refreshTimeout = null;
482
- }
483
- if (!this.tokens.expiresAt || !this.hasRefreshToken()) {
484
- return;
485
- }
486
- const now = Date.now();
487
- const refreshTime = this.tokens.expiresAt - this.config.refreshBuffer;
488
- const delay = Math.max(0, refreshTime - now);
489
- this.refreshTimeout = setTimeout(async () => {
490
- try {
491
- await this.refreshTokens();
492
- } catch (error) {
493
- console.error("Automatic token refresh failed:", error);
494
- if (this.config.onTokenError) {
495
- this.config.onTokenError(error);
496
- }
497
- }
498
- }, delay);
499
- }
500
- /**
501
- * Save tokens to storage
502
- */
503
- saveTokens() {
504
- try {
505
- const { storage } = this;
506
- const keys = this.storageKeys;
507
- if (this.tokens.accessToken) {
508
- storage.setItem(keys.accessToken, this.tokens.accessToken);
509
- }
510
- if (this.tokens.refreshToken) {
511
- storage.setItem(keys.refreshToken, this.tokens.refreshToken);
512
- }
513
- if (this.tokens.expiresAt) {
514
- storage.setItem(keys.expiresAt, this.tokens.expiresAt.toString());
515
- }
516
- if (this.tokens.expiresIn) {
517
- storage.setItem(keys.expiresIn, this.tokens.expiresIn.toString());
518
- }
519
- } catch (error) {
520
- console.error("[TokenManager] Error saving tokens to storage:", error);
521
- }
522
- }
523
- /**
524
- * Load tokens from storage
525
- */
526
- loadTokens() {
527
- try {
528
- const { storage } = this;
529
- const keys = this.storageKeys;
530
- const accessToken = storage.getItem(keys.accessToken);
531
- const refreshToken = storage.getItem(keys.refreshToken);
532
- const expiresAt = storage.getItem(keys.expiresAt);
533
- const expiresIn = storage.getItem(keys.expiresIn);
534
- if (accessToken) {
535
- this.tokens = {
536
- accessToken,
537
- refreshToken,
538
- expiresAt: expiresAt ? parseInt(expiresAt, 10) : null,
539
- expiresIn: expiresIn ? parseInt(expiresIn, 10) : null,
540
- tokenType: "Bearer"
541
- };
542
- this.scheduleRefresh();
543
- }
544
- } catch (error) {
545
- console.error("[TokenManager] Error loading tokens from storage:", error);
546
- this.tokens = {
547
- accessToken: null,
548
- refreshToken: null,
549
- expiresAt: null,
550
- expiresIn: null
551
- };
552
- }
553
- }
554
- /**
555
- * Clear all tokens
556
- */
557
- clearTokens() {
558
- this.tokens = {
559
- accessToken: null,
560
- refreshToken: null,
561
- expiresAt: null,
562
- expiresIn: null
563
- };
564
- const { storage } = this;
565
- const keys = this.storageKeys;
566
- Object.values(keys).forEach((key) => {
567
- storage.removeItem(key);
568
- });
569
- if (this.refreshTimeout) {
570
- clearTimeout(this.refreshTimeout);
571
- this.refreshTimeout = null;
572
- }
573
- this.retryCount = 0;
574
- }
575
- /**
576
- * Get token status information
577
- */
578
- getTokenStatus() {
579
- const hasTokens = this.hasTokens();
580
- const isValid = this.isAccessTokenValid();
581
- const { expiresAt } = this.tokens;
582
- const timeToExpiry = expiresAt ? expiresAt - Date.now() : null;
583
- return {
584
- hasTokens,
585
- isValid,
586
- hasRefreshToken: this.hasRefreshToken(),
587
- expiresAt,
588
- timeToExpiry,
589
- willExpireSoon: timeToExpiry ? timeToExpiry < this.config.refreshBuffer : false
590
- };
591
- }
592
- /**
593
- * Cleanup resources
594
- */
595
- destroy() {
596
- if (this.refreshTimeout) {
597
- clearTimeout(this.refreshTimeout);
598
- this.refreshTimeout = null;
599
- }
600
- this.refreshPromise = null;
601
- }
602
- };
603
- var defaultTokenManager = null;
604
- var getTokenManager = (options) => {
605
- if (!defaultTokenManager) {
606
- defaultTokenManager = new TokenManager(options);
607
- }
608
- return defaultTokenManager;
609
- };
610
-
611
- // src/services/BaseService.js
612
- var BaseService = class {
613
- constructor({ context, options } = {}) {
614
- this._context = context || {};
615
- this._options = options || {};
616
- this._ready = false;
617
- this._error = null;
618
- this._apiUrl = null;
619
- this._tokenManager = null;
620
- }
621
- // Initialize service
622
- init({ context }) {
623
- try {
624
- const { apiUrl } = context || this._context;
625
- this._apiUrl = apiUrl || environment_default.apiUrl;
626
- if (!this._apiUrl) {
627
- throw new Error("Service base URL not configured");
628
- }
629
- this._tokenManager = getTokenManager({
630
- apiUrl: this._apiUrl,
631
- onTokenError: (error) => {
632
- console.error("Token management error:", error);
633
- }
634
- });
635
- this._setReady();
636
- } catch (error) {
637
- this._setError(error);
638
- throw error;
639
- }
640
- }
641
- // Update context
642
- updateContext(context) {
643
- if (context && typeof context === "object") {
644
- Object.assign(this._context, context);
645
- }
646
- }
647
- // Get service status
648
- getStatus() {
649
- return {
650
- ready: this._ready,
651
- error: this._error,
652
- context: { ...this._context }
653
- };
654
- }
655
- // Check if service is ready
656
- isReady() {
657
- return this._ready;
658
- }
659
- // Protected helper methods
660
- _setReady(ready = true) {
661
- this._ready = ready;
662
- this._error = null;
663
- }
664
- _setError(error) {
665
- this._ready = false;
666
- this._error = error;
667
- }
668
- _getTrackingService() {
669
- var _a;
670
- const services = (_a = this._context) == null ? void 0 : _a.services;
671
- const tracking = services == null ? void 0 : services.tracking;
672
- if (!tracking || typeof tracking.trackError !== "function") {
673
- return null;
674
- }
675
- return tracking;
676
- }
677
- _shouldTrackErrors() {
678
- var _a;
679
- const name = (_a = this == null ? void 0 : this.constructor) == null ? void 0 : _a.name;
680
- return name !== "TrackingService";
681
- }
682
- _trackServiceError(error, details = {}) {
683
- var _a;
684
- if (!this._shouldTrackErrors()) {
685
- return;
686
- }
687
- try {
688
- const tracking = this._getTrackingService();
689
- if (!tracking) {
690
- return;
691
- }
692
- const context = {
693
- service: ((_a = this == null ? void 0 : this.constructor) == null ? void 0 : _a.name) || "UnknownService",
694
- apiUrl: this._apiUrl || null,
695
- ...details
696
- };
697
- tracking.trackError(error instanceof Error ? error : new Error(String(error)), context);
698
- } catch {
699
- }
700
- }
701
- _requireAuth() {
702
- if (!this.getAuthToken()) {
703
- throw new Error("Authentication required");
704
- }
705
- }
706
- _requireReady(methodName = "unknown") {
707
- if (!this.isReady()) {
708
- throw new Error(`Service not initialized for method: ${methodName}`);
709
- }
710
- }
711
- // Shared HTTP request method
712
- async _request(endpoint, options = {}) {
713
- const url = `${this._apiUrl}/core${endpoint}`;
714
- const defaultHeaders = {};
715
- if (!(options.body instanceof FormData)) {
716
- defaultHeaders["Content-Type"] = "application/json";
717
- }
718
- if (this._requiresInit(options.methodName) && this._tokenManager) {
719
- try {
720
- const validToken = await this._tokenManager.ensureValidToken();
721
- if (validToken) {
722
- const authHeader = this._tokenManager.getAuthHeader();
723
- if (authHeader) {
724
- defaultHeaders.Authorization = authHeader;
725
- }
726
- }
727
- } catch (error) {
728
- console.warn(
729
- "Token management failed, proceeding without authentication:",
730
- error
731
- );
732
- }
733
- }
734
- try {
735
- const response = await fetch(url, {
736
- ...options,
737
- headers: {
738
- ...defaultHeaders,
739
- ...options.headers
740
- }
741
- });
742
- if (!response.ok) {
743
- let error = {
744
- message: `HTTP ${response.status}: ${response.statusText}`
745
- };
746
- try {
747
- error = await response.json();
748
- } catch {
749
- }
750
- this._trackServiceError(
751
- new Error(error.message || error.error || `HTTP ${response.status}: ${response.statusText}`),
752
- {
753
- endpoint,
754
- methodName: options.methodName,
755
- status: response.status,
756
- statusText: response.statusText
757
- }
758
- );
759
- throw new Error(error.message || error.error || "Request failed", { cause: error });
760
- }
761
- return response.status === 204 ? null : response.json();
762
- } catch (error) {
763
- this._trackServiceError(error, {
764
- endpoint,
765
- methodName: options.methodName
766
- });
767
- throw new Error(`Request failed: ${error.message}`, { cause: error });
768
- }
769
- }
770
- // Helper method to determine if a method requires initialization
771
- _requiresInit(methodName) {
772
- const noInitMethods = /* @__PURE__ */ new Set([
773
- "register",
774
- "login",
775
- "googleAuth",
776
- "googleAuthCallback",
777
- "githubAuth",
778
- "requestPasswordReset",
779
- "confirmPasswordReset",
780
- "confirmRegistration",
781
- "verifyEmail",
782
- "getPlans",
783
- "getPlan",
784
- "listPublicProjects",
785
- "getPublicProject"
786
- ]);
787
- return !noInitMethods.has(methodName);
788
- }
789
- // Cleanup method
790
- destroy() {
791
- if (this._tokenManager) {
792
- this._tokenManager.destroy();
793
- this._tokenManager = null;
794
- }
795
- this._ready = false;
796
- this._setReady(false);
797
- }
798
- };
799
-
800
- // src/services/SubscriptionService.js
801
- var SubscriptionService = class extends BaseService {
802
- // ==================== SUBSCRIPTION METHODS ====================
803
- /**
804
- * Create a new subscription checkout session
805
- */
806
- async createSubscription(subscriptionData) {
807
- this._requireReady("createSubscription");
808
- if (!subscriptionData || typeof subscriptionData !== "object") {
809
- throw new Error("Subscription data is required");
810
- }
811
- const { projectId, planId, pricingKey = "monthly", seats = 1, successUrl, cancelUrl } = subscriptionData;
812
- if (!projectId) {
813
- throw new Error("Project ID is required");
814
- }
815
- if (!planId) {
816
- throw new Error("Plan ID is required");
817
- }
818
- try {
819
- const response = await this._request("/subscriptions", {
820
- method: "POST",
821
- body: JSON.stringify({
822
- projectId,
823
- planId,
824
- pricingKey,
825
- seats,
826
- successUrl,
827
- cancelUrl
828
- }),
829
- methodName: "createSubscription"
830
- });
831
- if (response.success) {
832
- return response.data;
833
- }
834
- throw new Error(response.message);
835
- } catch (error) {
836
- throw new Error(`Failed to create subscription: ${error.message}`, { cause: error });
837
- }
838
- }
839
- /**
840
- * Get subscription status and usage for a project
841
- */
842
- async getProjectStatus(projectId) {
843
- this._requireReady("getProjectStatus");
844
- if (!projectId) {
845
- throw new Error("Project ID is required");
846
- }
847
- try {
848
- const response = await this._request(`/subscriptions/project/${projectId}`, {
849
- method: "GET",
850
- methodName: "getProjectStatus"
851
- });
852
- if (response.success) {
853
- return response.data;
854
- }
855
- throw new Error(response.message);
856
- } catch (error) {
857
- throw new Error(`Failed to get project subscription status: ${error.message}`, { cause: error });
858
- }
859
- }
860
- /**
861
- * Get usage data for a subscription
862
- */
863
- async getUsage(subscriptionId) {
864
- this._requireReady("getUsage");
865
- if (!subscriptionId) {
866
- throw new Error("Subscription ID is required");
867
- }
868
- try {
869
- const response = await this._request(`/subscriptions/${subscriptionId}/usage`, {
870
- method: "GET",
871
- methodName: "getUsage"
872
- });
873
- if (response.success) {
874
- return response.data;
875
- }
876
- throw new Error(response.message);
877
- } catch (error) {
878
- throw new Error(`Failed to get subscription usage: ${error.message}`, { cause: error });
879
- }
880
- }
881
- /**
882
- * Cancel a subscription
883
- */
884
- async cancelSubscription(subscriptionId) {
885
- this._requireReady("cancelSubscription");
886
- if (!subscriptionId) {
887
- throw new Error("Subscription ID is required");
888
- }
889
- try {
890
- const response = await this._request(`/subscriptions/${subscriptionId}/cancel`, {
891
- method: "POST",
892
- methodName: "cancelSubscription"
893
- });
894
- if (response.success) {
895
- return response.data;
896
- }
897
- throw new Error(response.message);
898
- } catch (error) {
899
- throw new Error(`Failed to cancel subscription: ${error.message}`, { cause: error });
900
- }
901
- }
902
- /**
903
- * List invoices for a subscription
904
- */
905
- async listInvoices(subscriptionId, options = {}) {
906
- this._requireReady("listInvoices");
907
- if (!subscriptionId) {
908
- throw new Error("Subscription ID is required");
909
- }
910
- const { page = 1, limit = 20, status, startingAfter } = options;
911
- try {
912
- const queryParams = new URLSearchParams();
913
- if (page) {
914
- queryParams.append("page", page.toString());
915
- }
916
- if (limit) {
917
- queryParams.append("limit", limit.toString());
918
- }
919
- if (status) {
920
- queryParams.append("status", status);
921
- }
922
- if (startingAfter) {
923
- queryParams.append("startingAfter", startingAfter);
924
- }
925
- const queryString = queryParams.toString();
926
- const url = `/subscriptions/${subscriptionId}/invoices${queryString ? `?${queryString}` : ""}`;
927
- const response = await this._request(url, {
928
- method: "GET",
929
- methodName: "listInvoices"
930
- });
931
- if (response.success) {
932
- return response.data;
933
- }
934
- throw new Error(response.message);
935
- } catch (error) {
936
- throw new Error(`Failed to list invoices: ${error.message}`, { cause: error });
937
- }
938
- }
939
- /**
940
- * Get customer portal URL for a subscription
941
- */
942
- async getPortalUrl(subscriptionId, returnUrl) {
943
- this._requireReady("getPortalUrl");
944
- if (!subscriptionId) {
945
- throw new Error("Subscription ID is required");
946
- }
947
- try {
948
- const queryParams = new URLSearchParams();
949
- if (returnUrl) {
950
- queryParams.append("returnUrl", returnUrl);
951
- }
952
- const queryString = queryParams.toString();
953
- const url = `/subscriptions/${subscriptionId}/portal${queryString ? `?${queryString}` : ""}`;
954
- const response = await this._request(url, {
955
- method: "GET",
956
- methodName: "getPortalUrl"
957
- });
958
- if (response.success) {
959
- return response.data;
960
- }
961
- throw new Error(response.message);
962
- } catch (error) {
963
- throw new Error(`Failed to get portal URL: ${error.message}`, { cause: error });
964
- }
965
- }
966
- // ==================== SUBSCRIPTION HELPER METHODS ====================
967
- /**
968
- * Helper method to create subscription with validation
969
- */
970
- async createSubscriptionWithValidation(subscriptionData) {
971
- if (!subscriptionData || typeof subscriptionData !== "object") {
972
- throw new Error("Subscription data must be a valid object");
973
- }
974
- const requiredFields = ["projectId", "planId"];
975
- for (const field of requiredFields) {
976
- if (!subscriptionData[field]) {
977
- throw new Error(`Required field '${field}' is missing`);
978
- }
979
- }
980
- if (subscriptionData.seats != null) {
981
- if (!Number.isInteger(subscriptionData.seats) || subscriptionData.seats < 1) {
982
- throw new Error("Seats must be a positive integer");
983
- }
984
- }
985
- if (subscriptionData.pricingKey && !["monthly", "yearly"].includes(subscriptionData.pricingKey)) {
986
- throw new Error('Pricing key must be either "monthly" or "yearly"');
987
- }
988
- return await this.createSubscription(subscriptionData);
989
- }
990
- /**
991
- * Helper method to check if project has active subscription
992
- */
993
- async hasActiveSubscription(projectId) {
994
- try {
995
- const status = await this.getProjectStatus(projectId);
996
- return status.hasSubscription === true;
997
- } catch (error) {
998
- throw new Error(`Failed to check subscription status: ${error.message}`, { cause: error });
999
- }
1000
- }
1001
- /**
1002
- * Helper method to get subscription details for a project
1003
- */
1004
- async getProjectSubscription(projectId) {
1005
- try {
1006
- const status = await this.getProjectStatus(projectId);
1007
- if (!status.hasSubscription) {
1008
- return null;
1009
- }
1010
- return status.subscription;
1011
- } catch (error) {
1012
- throw new Error(`Failed to get project subscription: ${error.message}`, { cause: error });
1013
- }
1014
- }
1015
- /**
1016
- * Helper method to get usage with limits for a project
1017
- */
1018
- async getProjectUsage(projectId) {
1019
- try {
1020
- const status = await this.getProjectStatus(projectId);
1021
- if (!status.hasSubscription) {
1022
- return null;
1023
- }
1024
- return status.usage;
1025
- } catch (error) {
1026
- throw new Error(`Failed to get project usage: ${error.message}`, { cause: error });
1027
- }
1028
- }
1029
- /**
1030
- * Helper method to get invoices with pagination
1031
- */
1032
- async getInvoicesWithPagination(subscriptionId, options = {}) {
1033
- try {
1034
- const result = await this.listInvoices(subscriptionId, options);
1035
- return {
1036
- invoices: result.data || [],
1037
- pagination: result.pagination || {}
1038
- };
1039
- } catch (error) {
1040
- throw new Error(`Failed to get invoices with pagination: ${error.message}`, { cause: error });
1041
- }
1042
- }
1043
- /**
1044
- * Helper method to check if subscription is active
1045
- */
1046
- async isSubscriptionActive(subscriptionId) {
1047
- try {
1048
- const usage = await this.getUsage(subscriptionId);
1049
- return usage && usage.subscription && usage.subscription.status === "active";
1050
- } catch (error) {
1051
- throw new Error(`Failed to check subscription status: ${error.message}`, { cause: error });
1052
- }
1053
- }
1054
- /**
1055
- * Helper method to get subscription limits
1056
- */
1057
- async getSubscriptionLimits(subscriptionId) {
1058
- try {
1059
- const usage = await this.getUsage(subscriptionId);
1060
- return usage.limits || {};
1061
- } catch (error) {
1062
- throw new Error(`Failed to get subscription limits: ${error.message}`, { cause: error });
1063
- }
1064
- }
1065
- /**
1066
- * Change subscription (unified endpoint for all pricing changes)
1067
- */
1068
- async changeSubscription(changeData) {
1069
- this._requireReady("changeSubscription");
1070
- if (!changeData || typeof changeData !== "object") {
1071
- throw new Error("Change data is required");
1072
- }
1073
- const { subscriptionId, planId, pricingKey, seats = 1, projectId, successUrl, cancelUrl } = changeData;
1074
- if (!subscriptionId) {
1075
- throw new Error("Subscription ID is required");
1076
- }
1077
- if (!pricingKey) {
1078
- throw new Error("Pricing key is required");
1079
- }
1080
- try {
1081
- const response = await this._request(`/subscriptions/${subscriptionId}/change`, {
1082
- method: "POST",
1083
- body: JSON.stringify({
1084
- planId,
1085
- pricingKey,
1086
- seats,
1087
- projectId,
1088
- successUrl,
1089
- cancelUrl
1090
- }),
1091
- methodName: "changeSubscription"
1092
- });
1093
- if (response.success) {
1094
- return response.data;
1095
- }
1096
- throw new Error(response.message);
1097
- } catch (error) {
1098
- throw new Error(`Failed to change subscription: ${error.message}`, { cause: error });
1099
- }
1100
- }
1101
- /**
1102
- * Downgrade subscription to free plan
1103
- */
1104
- async downgrade(downgradeData) {
1105
- this._requireReady("downgrade");
1106
- if (!downgradeData || typeof downgradeData !== "object") {
1107
- throw new Error("Downgrade data is required");
1108
- }
1109
- const { subscriptionId, reason } = downgradeData;
1110
- if (!subscriptionId) {
1111
- throw new Error("Subscription ID is required");
1112
- }
1113
- try {
1114
- const response = await this._request(`/subscriptions/${subscriptionId}/downgrade`, {
1115
- method: "POST",
1116
- body: JSON.stringify({
1117
- reason
1118
- }),
1119
- methodName: "downgrade"
1120
- });
1121
- if (response.success) {
1122
- return response.data;
1123
- }
1124
- throw new Error(response.message);
1125
- } catch (error) {
1126
- throw new Error(`Failed to downgrade subscription: ${error.message}`, { cause: error });
1127
- }
1128
- }
1129
- /**
1130
- * Helper method to change subscription with validation
1131
- */
1132
- async changeSubscriptionWithValidation(changeData) {
1133
- if (!changeData || typeof changeData !== "object") {
1134
- throw new Error("Change data must be a valid object");
1135
- }
1136
- const requiredFields = ["subscriptionId", "pricingKey"];
1137
- for (const field of requiredFields) {
1138
- if (!changeData[field]) {
1139
- throw new Error(`Required field '${field}' is missing`);
1140
- }
1141
- }
1142
- if (changeData.seats != null) {
1143
- if (!Number.isInteger(changeData.seats) || changeData.seats < 1) {
1144
- throw new Error("Seats must be a positive integer");
1145
- }
1146
- }
1147
- if (typeof changeData.subscriptionId !== "string") {
1148
- throw new Error("Subscription ID must be a valid string");
1149
- }
1150
- if (changeData.planId && typeof changeData.planId !== "string") {
1151
- throw new Error("Plan ID must be a valid string");
1152
- }
1153
- return await this.changeSubscription(changeData);
1154
- }
1155
- /**
1156
- * Helper method to downgrade subscription with validation
1157
- */
1158
- async downgradeWithValidation(downgradeData) {
1159
- if (!downgradeData || typeof downgradeData !== "object") {
1160
- throw new Error("Downgrade data must be a valid object");
1161
- }
1162
- if (!downgradeData.subscriptionId) {
1163
- throw new Error("Subscription ID is required");
1164
- }
1165
- if (typeof downgradeData.subscriptionId !== "string") {
1166
- throw new Error("Subscription ID must be a valid string");
1167
- }
1168
- if (downgradeData.reason && typeof downgradeData.reason !== "string") {
1169
- throw new Error("Reason must be a valid string");
1170
- }
1171
- return await this.downgrade(downgradeData);
1172
- }
1173
- };
1174
- export {
1175
- SubscriptionService
1176
- };
1177
- // @preserve-env