@symbo.ls/sdk 3.2.3 → 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.
- package/README.md +141 -0
- package/dist/cjs/config/environment.js +94 -10
- package/dist/cjs/index.js +152 -12
- package/dist/cjs/services/AdminService.js +351 -0
- package/dist/cjs/services/AuthService.js +738 -305
- package/dist/cjs/services/BaseService.js +158 -6
- package/dist/cjs/services/BranchService.js +484 -0
- package/dist/cjs/services/CollabService.js +439 -116
- package/dist/cjs/services/DnsService.js +340 -0
- package/dist/cjs/services/FeatureFlagService.js +175 -0
- package/dist/cjs/services/FileService.js +201 -0
- package/dist/cjs/services/IntegrationService.js +538 -0
- package/dist/cjs/services/MetricsService.js +62 -0
- package/dist/cjs/services/PaymentService.js +271 -0
- package/dist/cjs/services/PlanService.js +426 -0
- package/dist/cjs/services/ProjectService.js +1207 -0
- package/dist/cjs/services/PullRequestService.js +503 -0
- package/dist/cjs/services/ScreenshotService.js +304 -0
- package/dist/cjs/services/SubscriptionService.js +396 -0
- package/dist/cjs/services/TrackingService.js +661 -0
- package/dist/cjs/services/WaitlistService.js +148 -0
- package/dist/cjs/services/index.js +60 -4
- package/dist/cjs/state/RootStateManager.js +2 -23
- package/dist/cjs/state/rootEventBus.js +9 -0
- package/dist/cjs/utils/CollabClient.js +78 -12
- package/dist/cjs/utils/TokenManager.js +16 -3
- package/dist/cjs/utils/changePreprocessor.js +199 -0
- package/dist/cjs/utils/jsonDiff.js +46 -4
- package/dist/cjs/utils/ordering.js +309 -0
- package/dist/cjs/utils/services.js +285 -128
- package/dist/cjs/utils/validation.js +0 -3
- package/dist/esm/config/environment.js +94 -10
- package/dist/esm/index.js +47862 -18248
- package/dist/esm/services/AdminService.js +1132 -0
- package/dist/esm/services/AuthService.js +1493 -386
- package/dist/esm/services/BaseService.js +757 -6
- package/dist/esm/services/BranchService.js +1265 -0
- package/dist/esm/services/CollabService.js +24956 -16089
- package/dist/esm/services/DnsService.js +1121 -0
- package/dist/esm/services/FeatureFlagService.js +956 -0
- package/dist/esm/services/FileService.js +982 -0
- package/dist/esm/services/IntegrationService.js +1319 -0
- package/dist/esm/services/MetricsService.js +843 -0
- package/dist/esm/services/PaymentService.js +1052 -0
- package/dist/esm/services/PlanService.js +1207 -0
- package/dist/esm/services/ProjectService.js +2526 -0
- package/dist/esm/services/PullRequestService.js +1284 -0
- package/dist/esm/services/ScreenshotService.js +1085 -0
- package/dist/esm/services/SubscriptionService.js +1177 -0
- package/dist/esm/services/TrackingService.js +18454 -0
- package/dist/esm/services/WaitlistService.js +929 -0
- package/dist/esm/services/index.js +47373 -18027
- package/dist/esm/state/RootStateManager.js +11 -23
- package/dist/esm/state/rootEventBus.js +9 -0
- package/dist/esm/utils/CollabClient.js +17526 -16120
- package/dist/esm/utils/TokenManager.js +16 -3
- package/dist/esm/utils/changePreprocessor.js +542 -0
- package/dist/esm/utils/jsonDiff.js +958 -43
- package/dist/esm/utils/ordering.js +291 -0
- package/dist/esm/utils/services.js +285 -128
- package/dist/esm/utils/validation.js +116 -50
- package/dist/node/config/environment.js +94 -10
- package/dist/node/index.js +183 -16
- package/dist/node/services/AdminService.js +332 -0
- package/dist/node/services/AuthService.js +742 -310
- package/dist/node/services/BaseService.js +148 -6
- package/dist/node/services/BranchService.js +465 -0
- package/dist/node/services/CollabService.js +439 -116
- package/dist/node/services/DnsService.js +321 -0
- package/dist/node/services/FeatureFlagService.js +156 -0
- package/dist/node/services/FileService.js +182 -0
- package/dist/node/services/IntegrationService.js +519 -0
- package/dist/node/services/MetricsService.js +43 -0
- package/dist/node/services/PaymentService.js +252 -0
- package/dist/node/services/PlanService.js +407 -0
- package/dist/node/services/ProjectService.js +1188 -0
- package/dist/node/services/PullRequestService.js +484 -0
- package/dist/node/services/ScreenshotService.js +285 -0
- package/dist/node/services/SubscriptionService.js +377 -0
- package/dist/node/services/TrackingService.js +632 -0
- package/dist/node/services/WaitlistService.js +129 -0
- package/dist/node/services/index.js +60 -4
- package/dist/node/state/RootStateManager.js +2 -23
- package/dist/node/state/rootEventBus.js +9 -0
- package/dist/node/utils/CollabClient.js +77 -11
- package/dist/node/utils/TokenManager.js +16 -3
- package/dist/node/utils/changePreprocessor.js +180 -0
- package/dist/node/utils/jsonDiff.js +46 -4
- package/dist/node/utils/ordering.js +290 -0
- package/dist/node/utils/services.js +285 -128
- package/dist/node/utils/validation.js +0 -3
- package/package.json +30 -18
- package/src/config/environment.js +95 -10
- package/src/index.js +190 -23
- package/src/services/AdminService.js +374 -0
- package/src/services/AuthService.js +874 -328
- package/src/services/BaseService.js +166 -6
- package/src/services/BranchService.js +536 -0
- package/src/services/CollabService.js +557 -148
- package/src/services/DnsService.js +366 -0
- package/src/services/FeatureFlagService.js +174 -0
- package/src/services/FileService.js +213 -0
- package/src/services/IntegrationService.js +548 -0
- package/src/services/MetricsService.js +40 -0
- package/src/services/PaymentService.js +287 -0
- package/src/services/PlanService.js +468 -0
- package/src/services/ProjectService.js +1366 -0
- package/src/services/PullRequestService.js +537 -0
- package/src/services/ScreenshotService.js +258 -0
- package/src/services/SubscriptionService.js +425 -0
- package/src/services/TrackingService.js +853 -0
- package/src/services/WaitlistService.js +130 -0
- package/src/services/index.js +79 -5
- package/src/services/tests/BranchService/createBranch.test.js +153 -0
- package/src/services/tests/BranchService/deleteBranch.test.js +173 -0
- package/src/services/tests/BranchService/getBranchChanges.test.js +146 -0
- package/src/services/tests/BranchService/listBranches.test.js +87 -0
- package/src/services/tests/BranchService/mergeBranch.test.js +210 -0
- package/src/services/tests/BranchService/publishVersion.test.js +183 -0
- package/src/services/tests/BranchService/renameBranch.test.js +240 -0
- package/src/services/tests/BranchService/resetBranch.test.js +152 -0
- package/src/services/tests/FeatureFlagService/adminFeatureFlags.test.js +67 -0
- package/src/services/tests/FeatureFlagService/getFeatureFlags.test.js +75 -0
- package/src/services/tests/FileService/createFileFormData.test.js +74 -0
- package/src/services/tests/FileService/getFileUrl.test.js +69 -0
- package/src/services/tests/FileService/updateProjectIcon.test.js +109 -0
- package/src/services/tests/FileService/uploadDocument.test.js +36 -0
- package/src/services/tests/FileService/uploadFile.test.js +78 -0
- package/src/services/tests/FileService/uploadFileWithValidation.test.js +114 -0
- package/src/services/tests/FileService/uploadImage.test.js +36 -0
- package/src/services/tests/FileService/uploadMultipleFiles.test.js +111 -0
- package/src/services/tests/FileService/validateFile.test.js +63 -0
- package/src/services/tests/PlanService/createPlan.test.js +104 -0
- package/src/services/tests/PlanService/createPlanWithValidation.test.js +523 -0
- package/src/services/tests/PlanService/deletePlan.test.js +92 -0
- package/src/services/tests/PlanService/getActivePlans.test.js +123 -0
- package/src/services/tests/PlanService/getAdminPlans.test.js +84 -0
- package/src/services/tests/PlanService/getPlan.test.js +50 -0
- package/src/services/tests/PlanService/getPlanByKey.test.js +109 -0
- package/src/services/tests/PlanService/getPlanWithValidation.test.js +85 -0
- package/src/services/tests/PlanService/getPlans.test.js +53 -0
- package/src/services/tests/PlanService/getPlansByPriceRange.test.js +109 -0
- package/src/services/tests/PlanService/getPlansWithValidation.test.js +48 -0
- package/src/services/tests/PlanService/initializePlans.test.js +75 -0
- package/src/services/tests/PlanService/updatePlan.test.js +111 -0
- package/src/services/tests/PlanService/updatePlanWithValidation.test.js +556 -0
- package/src/state/RootStateManager.js +37 -32
- package/src/state/rootEventBus.js +19 -0
- package/src/utils/CollabClient.js +99 -12
- package/src/utils/TokenManager.js +20 -3
- package/src/utils/changePreprocessor.js +239 -0
- package/src/utils/jsonDiff.js +40 -5
- package/src/utils/ordering.js +271 -0
- package/src/utils/services.js +306 -139
- package/src/utils/validation.js +0 -3
- package/dist/cjs/services/AIService.js +0 -155
- package/dist/cjs/services/BasedService.js +0 -1185
- package/dist/cjs/services/CoreService.js +0 -2295
- package/dist/cjs/services/SocketService.js +0 -309
- package/dist/cjs/services/SymstoryService.js +0 -571
- package/dist/cjs/utils/basedQuerys.js +0 -181
- package/dist/cjs/utils/symstoryClient.js +0 -259
- package/dist/esm/services/AIService.js +0 -185
- package/dist/esm/services/BasedService.js +0 -5262
- package/dist/esm/services/CoreService.js +0 -2827
- package/dist/esm/services/SocketService.js +0 -456
- package/dist/esm/services/SymstoryService.js +0 -7025
- package/dist/esm/utils/basedQuerys.js +0 -163
- package/dist/esm/utils/symstoryClient.js +0 -354
- package/dist/node/services/AIService.js +0 -136
- package/dist/node/services/BasedService.js +0 -1156
- package/dist/node/services/CoreService.js +0 -2266
- package/dist/node/services/SocketService.js +0 -280
- package/dist/node/services/SymstoryService.js +0 -542
- package/dist/node/utils/basedQuerys.js +0 -162
- package/dist/node/utils/symstoryClient.js +0 -230
- package/src/services/AIService.js +0 -150
- package/src/services/BasedService.js +0 -1302
- package/src/services/CoreService.js +0 -2548
- package/src/services/SocketService.js +0 -336
- package/src/services/SymstoryService.js +0 -649
- package/src/utils/basedQuerys.js +0 -164
- package/src/utils/symstoryClient.js +0 -252
|
@@ -0,0 +1,1207 @@
|
|
|
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/PlanService.js
|
|
801
|
+
var PlanService = class extends BaseService {
|
|
802
|
+
// ==================== PLAN METHODS ====================
|
|
803
|
+
/**
|
|
804
|
+
* Get list of public plans (no authentication required)
|
|
805
|
+
*/
|
|
806
|
+
async getPlans() {
|
|
807
|
+
try {
|
|
808
|
+
const response = await this._request("/plans", {
|
|
809
|
+
method: "GET",
|
|
810
|
+
methodName: "getPlans"
|
|
811
|
+
});
|
|
812
|
+
if (response.success) {
|
|
813
|
+
return response.data;
|
|
814
|
+
}
|
|
815
|
+
throw new Error(response.message);
|
|
816
|
+
} catch (error) {
|
|
817
|
+
throw new Error(`Failed to get plans: ${error.message}`, { cause: error });
|
|
818
|
+
}
|
|
819
|
+
}
|
|
820
|
+
/**
|
|
821
|
+
* Get list of public plans with enhanced pricing information (no authentication required)
|
|
822
|
+
*/
|
|
823
|
+
async getPlansWithPricing() {
|
|
824
|
+
try {
|
|
825
|
+
const response = await this._request("/plans/pricing", {
|
|
826
|
+
method: "GET",
|
|
827
|
+
methodName: "getPlansWithPricing"
|
|
828
|
+
});
|
|
829
|
+
if (response.success) {
|
|
830
|
+
return response.data;
|
|
831
|
+
}
|
|
832
|
+
throw new Error(response.message);
|
|
833
|
+
} catch (error) {
|
|
834
|
+
throw new Error(`Failed to get plans with pricing: ${error.message}`, {
|
|
835
|
+
cause: error
|
|
836
|
+
});
|
|
837
|
+
}
|
|
838
|
+
}
|
|
839
|
+
/**
|
|
840
|
+
* Get a specific plan by ID (no authentication required)
|
|
841
|
+
*/
|
|
842
|
+
async getPlan(planId) {
|
|
843
|
+
if (!planId) {
|
|
844
|
+
throw new Error("Plan ID is required");
|
|
845
|
+
}
|
|
846
|
+
try {
|
|
847
|
+
const response = await this._request(`/plans/${planId}`, {
|
|
848
|
+
method: "GET",
|
|
849
|
+
methodName: "getPlan"
|
|
850
|
+
});
|
|
851
|
+
if (response.success) {
|
|
852
|
+
return response.data;
|
|
853
|
+
}
|
|
854
|
+
throw new Error(response.message);
|
|
855
|
+
} catch (error) {
|
|
856
|
+
throw new Error(`Failed to get plan: ${error.message}`, { cause: error });
|
|
857
|
+
}
|
|
858
|
+
}
|
|
859
|
+
// ==================== ADMIN PLAN METHODS ====================
|
|
860
|
+
/**
|
|
861
|
+
* Get all plans including inactive ones (admin only)
|
|
862
|
+
*/
|
|
863
|
+
async getAdminPlans() {
|
|
864
|
+
this._requireReady("getAdminPlans");
|
|
865
|
+
try {
|
|
866
|
+
const response = await this._request("/admin/plans", {
|
|
867
|
+
method: "GET",
|
|
868
|
+
methodName: "getAdminPlans"
|
|
869
|
+
});
|
|
870
|
+
if (response.success) {
|
|
871
|
+
return response.data;
|
|
872
|
+
}
|
|
873
|
+
throw new Error(response.message);
|
|
874
|
+
} catch (error) {
|
|
875
|
+
throw new Error(`Failed to get admin plans: ${error.message}`, {
|
|
876
|
+
cause: error
|
|
877
|
+
});
|
|
878
|
+
}
|
|
879
|
+
}
|
|
880
|
+
/**
|
|
881
|
+
* Create a new plan (admin only)
|
|
882
|
+
*/
|
|
883
|
+
async createPlan(planData) {
|
|
884
|
+
this._requireReady("createPlan");
|
|
885
|
+
if (!planData || typeof planData !== "object") {
|
|
886
|
+
throw new Error("Plan data is required");
|
|
887
|
+
}
|
|
888
|
+
try {
|
|
889
|
+
const response = await this._request("/admin/plans", {
|
|
890
|
+
method: "POST",
|
|
891
|
+
body: JSON.stringify(planData),
|
|
892
|
+
methodName: "createPlan"
|
|
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 create plan: ${error.message}`, {
|
|
900
|
+
cause: error
|
|
901
|
+
});
|
|
902
|
+
}
|
|
903
|
+
}
|
|
904
|
+
/**
|
|
905
|
+
* Update an existing plan (admin only)
|
|
906
|
+
*/
|
|
907
|
+
async updatePlan(planId, planData) {
|
|
908
|
+
this._requireReady("updatePlan");
|
|
909
|
+
if (!planId) {
|
|
910
|
+
throw new Error("Plan ID is required");
|
|
911
|
+
}
|
|
912
|
+
if (!planData || typeof planData !== "object") {
|
|
913
|
+
throw new Error("Plan data is required");
|
|
914
|
+
}
|
|
915
|
+
try {
|
|
916
|
+
const response = await this._request(`/admin/plans/${planId}`, {
|
|
917
|
+
method: "PATCH",
|
|
918
|
+
body: JSON.stringify(planData),
|
|
919
|
+
methodName: "updatePlan"
|
|
920
|
+
});
|
|
921
|
+
if (response.success) {
|
|
922
|
+
return response.data;
|
|
923
|
+
}
|
|
924
|
+
throw new Error(response.message);
|
|
925
|
+
} catch (error) {
|
|
926
|
+
throw new Error(`Failed to update plan: ${error.message}`, {
|
|
927
|
+
cause: error
|
|
928
|
+
});
|
|
929
|
+
}
|
|
930
|
+
}
|
|
931
|
+
/**
|
|
932
|
+
* Delete a plan (soft delete + archive Stripe product) (admin only)
|
|
933
|
+
*/
|
|
934
|
+
async deletePlan(planId) {
|
|
935
|
+
this._requireReady("deletePlan");
|
|
936
|
+
if (!planId) {
|
|
937
|
+
throw new Error("Plan ID is required");
|
|
938
|
+
}
|
|
939
|
+
try {
|
|
940
|
+
const response = await this._request(`/admin/plans/${planId}`, {
|
|
941
|
+
method: "DELETE",
|
|
942
|
+
methodName: "deletePlan"
|
|
943
|
+
});
|
|
944
|
+
if (response.success) {
|
|
945
|
+
return response.data;
|
|
946
|
+
}
|
|
947
|
+
throw new Error(response.message);
|
|
948
|
+
} catch (error) {
|
|
949
|
+
throw new Error(`Failed to delete plan: ${error.message}`, {
|
|
950
|
+
cause: error
|
|
951
|
+
});
|
|
952
|
+
}
|
|
953
|
+
}
|
|
954
|
+
/**
|
|
955
|
+
* Initialize default plans (admin only)
|
|
956
|
+
*/
|
|
957
|
+
async initializePlans() {
|
|
958
|
+
this._requireReady("initializePlans");
|
|
959
|
+
try {
|
|
960
|
+
const response = await this._request("/admin/plans/initialize", {
|
|
961
|
+
method: "POST",
|
|
962
|
+
methodName: "initializePlans"
|
|
963
|
+
});
|
|
964
|
+
if (response.success) {
|
|
965
|
+
return response;
|
|
966
|
+
}
|
|
967
|
+
throw new Error(response.message);
|
|
968
|
+
} catch (error) {
|
|
969
|
+
throw new Error(`Failed to initialize plans: ${error.message}`, {
|
|
970
|
+
cause: error
|
|
971
|
+
});
|
|
972
|
+
}
|
|
973
|
+
}
|
|
974
|
+
// ==================== PLAN HELPER METHODS ====================
|
|
975
|
+
/**
|
|
976
|
+
* Helper method to get plans with validation
|
|
977
|
+
*/
|
|
978
|
+
async getPlansWithValidation() {
|
|
979
|
+
try {
|
|
980
|
+
const plans = await this.getPlans();
|
|
981
|
+
if (!Array.isArray(plans)) {
|
|
982
|
+
throw new Error("Invalid response format: plans should be an array");
|
|
983
|
+
}
|
|
984
|
+
return plans;
|
|
985
|
+
} catch (error) {
|
|
986
|
+
throw new Error(`Failed to get plans with validation: ${error.message}`, {
|
|
987
|
+
cause: error
|
|
988
|
+
});
|
|
989
|
+
}
|
|
990
|
+
}
|
|
991
|
+
/**
|
|
992
|
+
* Helper method to get a plan by ID with validation
|
|
993
|
+
*/
|
|
994
|
+
async getPlanWithValidation(planId) {
|
|
995
|
+
if (!planId || typeof planId !== "string") {
|
|
996
|
+
throw new Error("Plan ID must be a valid string");
|
|
997
|
+
}
|
|
998
|
+
try {
|
|
999
|
+
const plan = await this.getPlan(planId);
|
|
1000
|
+
if (!plan || typeof plan !== "object") {
|
|
1001
|
+
throw new Error("Invalid plan data received");
|
|
1002
|
+
}
|
|
1003
|
+
return plan;
|
|
1004
|
+
} catch (error) {
|
|
1005
|
+
throw new Error(`Failed to get plan with validation: ${error.message}`, {
|
|
1006
|
+
cause: error
|
|
1007
|
+
});
|
|
1008
|
+
}
|
|
1009
|
+
}
|
|
1010
|
+
/**
|
|
1011
|
+
* Helper method to create a plan with validation (admin only)
|
|
1012
|
+
*/
|
|
1013
|
+
async createPlanWithValidation(planData) {
|
|
1014
|
+
if (!planData || typeof planData !== "object") {
|
|
1015
|
+
throw new Error("Plan data must be a valid object");
|
|
1016
|
+
}
|
|
1017
|
+
const requiredFields = ["name", "description"];
|
|
1018
|
+
for (const field of requiredFields) {
|
|
1019
|
+
if (!planData[field]) {
|
|
1020
|
+
throw new Error(`Required field '${field}' is missing`);
|
|
1021
|
+
}
|
|
1022
|
+
}
|
|
1023
|
+
if (Object.hasOwn(planData, "price")) {
|
|
1024
|
+
throw new Error(
|
|
1025
|
+
'Field "price" is no longer supported. Use unified "pricingOptions" with "amount" instead.'
|
|
1026
|
+
);
|
|
1027
|
+
}
|
|
1028
|
+
if (planData.pricingOptions != null) {
|
|
1029
|
+
if (!Array.isArray(planData.pricingOptions) || planData.pricingOptions.length === 0) {
|
|
1030
|
+
throw new Error(
|
|
1031
|
+
"pricingOptions must be a non-empty array when provided"
|
|
1032
|
+
);
|
|
1033
|
+
}
|
|
1034
|
+
const allowedIntervals = /* @__PURE__ */ new Set(["month", "year", "week", "day", null]);
|
|
1035
|
+
planData.pricingOptions.forEach((option, index) => {
|
|
1036
|
+
if (!option || typeof option !== "object") {
|
|
1037
|
+
throw new Error(`Pricing option at index ${index} must be an object`);
|
|
1038
|
+
}
|
|
1039
|
+
const { key, displayName, amount, interval, lookupKey } = option;
|
|
1040
|
+
if (!key || typeof key !== "string") {
|
|
1041
|
+
throw new Error(
|
|
1042
|
+
`Pricing option at index ${index} is missing required field 'key'`
|
|
1043
|
+
);
|
|
1044
|
+
}
|
|
1045
|
+
if (!/^[a-z0-9-]+$/u.test(key)) {
|
|
1046
|
+
throw new Error(
|
|
1047
|
+
`Pricing option key '${key}' must contain only lowercase letters, numbers, and hyphens`
|
|
1048
|
+
);
|
|
1049
|
+
}
|
|
1050
|
+
if (!displayName || typeof displayName !== "string") {
|
|
1051
|
+
throw new Error(
|
|
1052
|
+
`Pricing option '${key}' is missing required field 'displayName'`
|
|
1053
|
+
);
|
|
1054
|
+
}
|
|
1055
|
+
if (typeof amount !== "number" || amount < 0) {
|
|
1056
|
+
throw new Error(
|
|
1057
|
+
`Pricing option '${key}' must have a non-negative numeric 'amount'`
|
|
1058
|
+
);
|
|
1059
|
+
}
|
|
1060
|
+
if (interval !== null && !allowedIntervals.has(interval)) {
|
|
1061
|
+
throw new Error(
|
|
1062
|
+
`Pricing option '${key}' has invalid interval '${interval}'. Allowed: month, year, week, day or null`
|
|
1063
|
+
);
|
|
1064
|
+
}
|
|
1065
|
+
if (!lookupKey || typeof lookupKey !== "string") {
|
|
1066
|
+
throw new Error(
|
|
1067
|
+
`Pricing option '${key}' is missing required field 'lookupKey'`
|
|
1068
|
+
);
|
|
1069
|
+
}
|
|
1070
|
+
});
|
|
1071
|
+
}
|
|
1072
|
+
if (Object.hasOwn(planData, "key") && planData.key == null) {
|
|
1073
|
+
throw new Error("Plan key must be a valid string");
|
|
1074
|
+
}
|
|
1075
|
+
if (planData.key && !/^[a-z0-9-]+$/u.test(planData.key)) {
|
|
1076
|
+
throw new Error(
|
|
1077
|
+
"Plan key must contain only lowercase letters, numbers, and hyphens"
|
|
1078
|
+
);
|
|
1079
|
+
}
|
|
1080
|
+
return await this.createPlan(planData);
|
|
1081
|
+
}
|
|
1082
|
+
/**
|
|
1083
|
+
* Helper method to update a plan with validation (admin only)
|
|
1084
|
+
*/
|
|
1085
|
+
async updatePlanWithValidation(planId, planData) {
|
|
1086
|
+
if (!planId || typeof planId !== "string") {
|
|
1087
|
+
throw new Error("Plan ID must be a valid string");
|
|
1088
|
+
}
|
|
1089
|
+
if (!planData || typeof planData !== "object") {
|
|
1090
|
+
throw new Error("Plan data must be a valid object");
|
|
1091
|
+
}
|
|
1092
|
+
if (Object.hasOwn(planData, "price")) {
|
|
1093
|
+
throw new Error(
|
|
1094
|
+
'Field "price" is no longer supported. Use unified "pricingOptions" with "amount" instead.'
|
|
1095
|
+
);
|
|
1096
|
+
}
|
|
1097
|
+
if (planData.pricingOptions != null) {
|
|
1098
|
+
if (!Array.isArray(planData.pricingOptions) || planData.pricingOptions.length === 0) {
|
|
1099
|
+
throw new Error(
|
|
1100
|
+
"pricingOptions must be a non-empty array when provided"
|
|
1101
|
+
);
|
|
1102
|
+
}
|
|
1103
|
+
const allowedIntervals = /* @__PURE__ */ new Set(["month", "year", "week", "day", null]);
|
|
1104
|
+
planData.pricingOptions.forEach((option, index) => {
|
|
1105
|
+
if (!option || typeof option !== "object") {
|
|
1106
|
+
throw new Error(`Pricing option at index ${index} must be an object`);
|
|
1107
|
+
}
|
|
1108
|
+
const { key, displayName, amount, interval, lookupKey } = option;
|
|
1109
|
+
if (!key || typeof key !== "string") {
|
|
1110
|
+
throw new Error(
|
|
1111
|
+
`Pricing option at index ${index} is missing required field 'key'`
|
|
1112
|
+
);
|
|
1113
|
+
}
|
|
1114
|
+
if (!/^[a-z0-9-]+$/u.test(key)) {
|
|
1115
|
+
throw new Error(
|
|
1116
|
+
`Pricing option key '${key}' must contain only lowercase letters, numbers, and hyphens`
|
|
1117
|
+
);
|
|
1118
|
+
}
|
|
1119
|
+
if (!displayName || typeof displayName !== "string") {
|
|
1120
|
+
throw new Error(
|
|
1121
|
+
`Pricing option '${key}' is missing required field 'displayName'`
|
|
1122
|
+
);
|
|
1123
|
+
}
|
|
1124
|
+
if (typeof amount !== "number" || amount < 0) {
|
|
1125
|
+
throw new Error(
|
|
1126
|
+
`Pricing option '${key}' must have a non-negative numeric 'amount'`
|
|
1127
|
+
);
|
|
1128
|
+
}
|
|
1129
|
+
if (interval !== null && !allowedIntervals.has(interval)) {
|
|
1130
|
+
throw new Error(
|
|
1131
|
+
`Pricing option '${key}' has invalid interval '${interval}'. Allowed: month, year, week, day or null`
|
|
1132
|
+
);
|
|
1133
|
+
}
|
|
1134
|
+
if (!lookupKey || typeof lookupKey !== "string") {
|
|
1135
|
+
throw new Error(
|
|
1136
|
+
`Pricing option '${key}' is missing required field 'lookupKey'`
|
|
1137
|
+
);
|
|
1138
|
+
}
|
|
1139
|
+
});
|
|
1140
|
+
}
|
|
1141
|
+
if (Object.hasOwn(planData, "key") && planData.key == null) {
|
|
1142
|
+
throw new Error("Plan key must be a valid string");
|
|
1143
|
+
}
|
|
1144
|
+
if (planData.key && !/^[a-z0-9-]+$/u.test(planData.key)) {
|
|
1145
|
+
throw new Error(
|
|
1146
|
+
"Plan key must contain only lowercase letters, numbers, and hyphens"
|
|
1147
|
+
);
|
|
1148
|
+
}
|
|
1149
|
+
return await this.updatePlan(planId, planData);
|
|
1150
|
+
}
|
|
1151
|
+
/**
|
|
1152
|
+
* Helper method to get active plans only
|
|
1153
|
+
*/
|
|
1154
|
+
async getActivePlans() {
|
|
1155
|
+
try {
|
|
1156
|
+
const plans = await this.getPlans();
|
|
1157
|
+
return plans.filter(
|
|
1158
|
+
(plan) => plan.status === "active" && plan.isVisible !== false
|
|
1159
|
+
);
|
|
1160
|
+
} catch (error) {
|
|
1161
|
+
throw new Error(`Failed to get active plans: ${error.message}`, {
|
|
1162
|
+
cause: error
|
|
1163
|
+
});
|
|
1164
|
+
}
|
|
1165
|
+
}
|
|
1166
|
+
/**
|
|
1167
|
+
* Helper method to get plans by price range
|
|
1168
|
+
*/
|
|
1169
|
+
async getPlansByPriceRange(minPrice = 0, maxPrice = Infinity) {
|
|
1170
|
+
try {
|
|
1171
|
+
const plans = await this.getPlansWithPricing();
|
|
1172
|
+
return plans.filter((plan) => {
|
|
1173
|
+
var _a, _b, _c;
|
|
1174
|
+
const price = (_c = (_b = (_a = plan == null ? void 0 : plan.pricing) == null ? void 0 : _a.bestPrice) == null ? void 0 : _b.amount) != null ? _c : 0;
|
|
1175
|
+
return price >= minPrice && price <= maxPrice;
|
|
1176
|
+
});
|
|
1177
|
+
} catch (error) {
|
|
1178
|
+
throw new Error(`Failed to get plans by price range: ${error.message}`, {
|
|
1179
|
+
cause: error
|
|
1180
|
+
});
|
|
1181
|
+
}
|
|
1182
|
+
}
|
|
1183
|
+
/**
|
|
1184
|
+
* Helper method to find plan by key
|
|
1185
|
+
*/
|
|
1186
|
+
async getPlanByKey(key) {
|
|
1187
|
+
if (!key) {
|
|
1188
|
+
throw new Error("Plan key is required");
|
|
1189
|
+
}
|
|
1190
|
+
try {
|
|
1191
|
+
const plans = await this.getPlans();
|
|
1192
|
+
const plan = plans.find((p) => p.key === key);
|
|
1193
|
+
if (!plan) {
|
|
1194
|
+
throw new Error(`Plan with key '${key}' not found`);
|
|
1195
|
+
}
|
|
1196
|
+
return plan;
|
|
1197
|
+
} catch (error) {
|
|
1198
|
+
throw new Error(`Failed to get plan by key: ${error.message}`, {
|
|
1199
|
+
cause: error
|
|
1200
|
+
});
|
|
1201
|
+
}
|
|
1202
|
+
}
|
|
1203
|
+
};
|
|
1204
|
+
export {
|
|
1205
|
+
PlanService
|
|
1206
|
+
};
|
|
1207
|
+
// @preserve-env
|