@samanhappy/mcphub 1.0.5 → 1.0.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (89) hide show
  1. package/dist/betterAuth.js +2 -2
  2. package/dist/betterAuth.js.map +1 -1
  3. package/dist/clients/openapi.js +98 -5
  4. package/dist/clients/openapi.js.map +1 -1
  5. package/dist/controllers/hostedInternalController.js +55 -0
  6. package/dist/controllers/hostedInternalController.js.map +1 -0
  7. package/dist/controllers/serverController.js +1 -1
  8. package/dist/controllers/serverController.js.map +1 -1
  9. package/dist/dao/ServerDao.js +24 -0
  10. package/dist/dao/ServerDao.js.map +1 -1
  11. package/dist/dao/ServerDaoDbImpl.js +11 -0
  12. package/dist/dao/ServerDaoDbImpl.js.map +1 -1
  13. package/dist/db/repositories/ServerRepository.js +16 -0
  14. package/dist/db/repositories/ServerRepository.js.map +1 -1
  15. package/dist/index.js +8 -0
  16. package/dist/index.js.map +1 -1
  17. package/dist/routes/index.js +5 -1
  18. package/dist/routes/index.js.map +1 -1
  19. package/dist/services/betterAuthConfig.js +144 -33
  20. package/dist/services/betterAuthConfig.js.map +1 -1
  21. package/dist/services/hostedAuthService.js +314 -0
  22. package/dist/services/hostedAuthService.js.map +1 -0
  23. package/dist/services/hostedControlPlaneClient.js +90 -0
  24. package/dist/services/hostedControlPlaneClient.js.map +1 -0
  25. package/dist/services/hostedEventSubscriber.js +206 -0
  26. package/dist/services/hostedEventSubscriber.js.map +1 -0
  27. package/dist/services/hostedInternalAuth.js +108 -0
  28. package/dist/services/hostedInternalAuth.js.map +1 -0
  29. package/dist/services/hostedMode.js +4 -0
  30. package/dist/services/hostedMode.js.map +1 -0
  31. package/dist/services/hostedNodeIdentity.js +8 -0
  32. package/dist/services/hostedNodeIdentity.js.map +1 -0
  33. package/dist/services/hostedRuntimeCatalogNames.js +7 -0
  34. package/dist/services/hostedRuntimeCatalogNames.js.map +1 -0
  35. package/dist/services/hostedRuntimeCatalogService.js +31 -0
  36. package/dist/services/hostedRuntimeCatalogService.js.map +1 -0
  37. package/dist/services/mcpService.js +84 -10
  38. package/dist/services/mcpService.js.map +1 -1
  39. package/dist/services/requestContextService.js +12 -0
  40. package/dist/services/requestContextService.js.map +1 -1
  41. package/dist/services/sseService.js +61 -10
  42. package/dist/services/sseService.js.map +1 -1
  43. package/dist/utils/rateLimit.js +4 -0
  44. package/dist/utils/rateLimit.js.map +1 -1
  45. package/frontend/dist/assets/{ActivityPage-Ccbl6DLW.js → ActivityPage-CLXufyzU.js} +2 -2
  46. package/frontend/dist/assets/{ActivityPage-Ccbl6DLW.js.map → ActivityPage-CLXufyzU.js.map} +1 -1
  47. package/frontend/dist/assets/{Dashboard-CPkAALtS.js → Dashboard-YMaaIlIZ.js} +2 -2
  48. package/frontend/dist/assets/{Dashboard-CPkAALtS.js.map → Dashboard-YMaaIlIZ.js.map} +1 -1
  49. package/frontend/dist/assets/{EndpointCopy-BYHUhlpW.js → EndpointCopy-Cpc2WqQz.js} +2 -2
  50. package/frontend/dist/assets/{EndpointCopy-BYHUhlpW.js.map → EndpointCopy-Cpc2WqQz.js.map} +1 -1
  51. package/frontend/dist/assets/{GroupsPage-CJI35G25.js → GroupsPage-DqK5U5rY.js} +2 -2
  52. package/frontend/dist/assets/{GroupsPage-CJI35G25.js.map → GroupsPage-DqK5U5rY.js.map} +1 -1
  53. package/frontend/dist/assets/{LoginPage-BZ0WR09v.js → LoginPage-ByV35kun.js} +2 -2
  54. package/frontend/dist/assets/{LoginPage-BZ0WR09v.js.map → LoginPage-ByV35kun.js.map} +1 -1
  55. package/frontend/dist/assets/{LogsPage-C_U-xXX0.js → LogsPage-BygFVcaV.js} +2 -2
  56. package/frontend/dist/assets/{LogsPage-C_U-xXX0.js.map → LogsPage-BygFVcaV.js.map} +1 -1
  57. package/frontend/dist/assets/{MarketPage-dyC4ocP6.js → MarketPage-Cc7QeqSw.js} +2 -2
  58. package/frontend/dist/assets/{MarketPage-dyC4ocP6.js.map → MarketPage-Cc7QeqSw.js.map} +1 -1
  59. package/frontend/dist/assets/{PromptsPage-gNHl5xn5.js → PromptsPage-DvLm5TW4.js} +2 -2
  60. package/frontend/dist/assets/{PromptsPage-gNHl5xn5.js.map → PromptsPage-DvLm5TW4.js.map} +1 -1
  61. package/frontend/dist/assets/{ResourcesPage-DvgjlTqh.js → ResourcesPage-Dnn1u2fr.js} +2 -2
  62. package/frontend/dist/assets/{ResourcesPage-DvgjlTqh.js.map → ResourcesPage-Dnn1u2fr.js.map} +1 -1
  63. package/frontend/dist/assets/ServersPage-RQCARDfE.js +37 -0
  64. package/frontend/dist/assets/ServersPage-RQCARDfE.js.map +1 -0
  65. package/frontend/dist/assets/{SettingsPage-D66oC6py.js → SettingsPage-CF40R-50.js} +2 -2
  66. package/frontend/dist/assets/{SettingsPage-D66oC6py.js.map → SettingsPage-CF40R-50.js.map} +1 -1
  67. package/frontend/dist/assets/{StatusDot-C7XhRfMO.js → StatusDot-CkF9WMhU.js} +2 -2
  68. package/frontend/dist/assets/{StatusDot-C7XhRfMO.js.map → StatusDot-CkF9WMhU.js.map} +1 -1
  69. package/frontend/dist/assets/{ToggleGroup-_MiJO3n_.js → ToggleGroup-vRqJ9DIB.js} +2 -2
  70. package/frontend/dist/assets/{ToggleGroup-_MiJO3n_.js.map → ToggleGroup-vRqJ9DIB.js.map} +1 -1
  71. package/frontend/dist/assets/{UsersPage-COmrRpdL.js → UsersPage-DBN5qCeO.js} +2 -2
  72. package/frontend/dist/assets/{UsersPage-COmrRpdL.js.map → UsersPage-DBN5qCeO.js.map} +1 -1
  73. package/frontend/dist/assets/{index-D82vzW6B.js → index-C7dQApMG.js} +3 -3
  74. package/frontend/dist/assets/{index-D82vzW6B.js.map → index-C7dQApMG.js.map} +1 -1
  75. package/frontend/dist/assets/{resourceService-D-fKBZtF.js → resourceService-BgjWoJFK.js} +2 -2
  76. package/frontend/dist/assets/{resourceService-D-fKBZtF.js.map → resourceService-BgjWoJFK.js.map} +1 -1
  77. package/frontend/dist/assets/{useServerData-DgGTUuZ8.js → useServerData-CEGj5OU1.js} +2 -2
  78. package/frontend/dist/assets/{useServerData-DgGTUuZ8.js.map → useServerData-CEGj5OU1.js.map} +1 -1
  79. package/frontend/dist/assets/useSettingsData-C2-q9Okf.js +2 -0
  80. package/frontend/dist/assets/{useSettingsData-B3CTLIUT.js.map → useSettingsData-C2-q9Okf.js.map} +1 -1
  81. package/frontend/dist/assets/variableDetection-GTKqOf1F.js +16 -0
  82. package/frontend/dist/assets/variableDetection-GTKqOf1F.js.map +1 -0
  83. package/frontend/dist/index.html +1 -1
  84. package/package.json +2 -1
  85. package/frontend/dist/assets/ServersPage-CKuVFL3O.js +0 -37
  86. package/frontend/dist/assets/ServersPage-CKuVFL3O.js.map +0 -1
  87. package/frontend/dist/assets/useSettingsData-B3CTLIUT.js +0 -2
  88. package/frontend/dist/assets/variableDetection-DsYuiOB_.js +0 -16
  89. package/frontend/dist/assets/variableDetection-DsYuiOB_.js.map +0 -1
@@ -0,0 +1,314 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ import { getHostedUserState, HostedControlPlaneError, reserveHostedCredit, settleHostedCredit, validateHostedApiKey, } from './hostedControlPlaneClient.js';
3
+ import { isHostedModeEnabled } from './hostedMode.js';
4
+ import { getHostedNodeIdentity } from './hostedNodeIdentity.js';
5
+ import { safeCompare } from '../utils/safeCompare.js';
6
+ const KEY_PREFIX = 'mcphub-sk';
7
+ const API_KEY_PREFIX_CHARS = 12;
8
+ const DEFAULT_CACHE_TTL_SECONDS = 30;
9
+ const DEFAULT_STALE_TTL_MS = 60 * 60 * 1000;
10
+ const MAX_CACHE_ENTRIES = 1000;
11
+ const CACHE_CLEANUP_INTERVAL_MS = 5 * 60 * 1000;
12
+ export class HostedAuthorizationError extends Error {
13
+ constructor(message, code = 'hosted_forbidden') {
14
+ super(message);
15
+ this.code = code;
16
+ this.name = 'HostedAuthorizationError';
17
+ }
18
+ }
19
+ export class HostedAuthUnavailableError extends Error {
20
+ constructor(message) {
21
+ super(message);
22
+ this.name = 'HostedAuthUnavailableError';
23
+ }
24
+ }
25
+ function createVerificationToken(apiKey) {
26
+ return apiKey;
27
+ }
28
+ function matchesVerificationToken(apiKey, verificationToken) {
29
+ return safeCompare(apiKey, verificationToken);
30
+ }
31
+ class MemoryHostedAuthStateCache {
32
+ constructor() {
33
+ this.keyCache = new Map();
34
+ this.userCacheIndex = new Map();
35
+ this.apiKeyIdCacheIndex = new Map();
36
+ }
37
+ get(prefix) {
38
+ return this.keyCache.get(prefix);
39
+ }
40
+ set(prefix, entry) {
41
+ this.remove(prefix);
42
+ this.keyCache.set(prefix, entry);
43
+ this.addIndexedPrefix(this.userCacheIndex, entry.userId, prefix);
44
+ this.addIndexedPrefix(this.apiKeyIdCacheIndex, entry.apiKeyId, prefix);
45
+ this.prune();
46
+ }
47
+ touch(prefix, entry) {
48
+ this.keyCache.delete(prefix);
49
+ this.keyCache.set(prefix, entry);
50
+ }
51
+ remove(prefix) {
52
+ const existing = this.keyCache.get(prefix);
53
+ if (!existing)
54
+ return;
55
+ this.keyCache.delete(prefix);
56
+ this.removeIndexedPrefix(this.userCacheIndex, existing.userId, prefix);
57
+ this.removeIndexedPrefix(this.apiKeyIdCacheIndex, existing.apiKeyId, prefix);
58
+ }
59
+ invalidateUser(userId) {
60
+ const prefixes = this.userCacheIndex.get(userId);
61
+ if (!prefixes)
62
+ return;
63
+ for (const prefix of [...prefixes]) {
64
+ this.remove(prefix);
65
+ }
66
+ }
67
+ invalidateApiKeyId(apiKeyId) {
68
+ const prefixes = this.apiKeyIdCacheIndex.get(apiKeyId);
69
+ if (!prefixes)
70
+ return;
71
+ for (const prefix of [...prefixes]) {
72
+ this.remove(prefix);
73
+ }
74
+ }
75
+ prune(now = Date.now()) {
76
+ for (const [prefix, entry] of this.keyCache.entries()) {
77
+ if (entry.staleUntil <= now) {
78
+ this.remove(prefix);
79
+ }
80
+ }
81
+ while (this.keyCache.size > MAX_CACHE_ENTRIES) {
82
+ const oldestPrefix = this.keyCache.keys().next().value;
83
+ if (!oldestPrefix)
84
+ break;
85
+ this.remove(oldestPrefix);
86
+ }
87
+ }
88
+ size() {
89
+ return this.keyCache.size;
90
+ }
91
+ addIndexedPrefix(index, key, prefix) {
92
+ const prefixes = index.get(key);
93
+ if (prefixes) {
94
+ prefixes.add(prefix);
95
+ return;
96
+ }
97
+ index.set(key, new Set([prefix]));
98
+ }
99
+ removeIndexedPrefix(index, key, prefix) {
100
+ const prefixes = index.get(key);
101
+ if (!prefixes)
102
+ return;
103
+ prefixes.delete(prefix);
104
+ if (prefixes.size === 0) {
105
+ index.delete(key);
106
+ }
107
+ }
108
+ }
109
+ const authStateCache = new MemoryHostedAuthStateCache();
110
+ const cacheCleanupTimer = setInterval(() => {
111
+ authStateCache.prune();
112
+ }, CACHE_CLEANUP_INTERVAL_MS);
113
+ cacheCleanupTimer.unref?.();
114
+ export function isHostedApiKey(value) {
115
+ return Boolean(value?.startsWith(`${KEY_PREFIX}-`));
116
+ }
117
+ function extractApiKeyPrefix(key) {
118
+ const prefix = `${KEY_PREFIX}-`;
119
+ if (!key.startsWith(prefix))
120
+ return null;
121
+ const token = key.slice(prefix.length);
122
+ if (token.length < API_KEY_PREFIX_CHARS)
123
+ return null;
124
+ return token.slice(0, API_KEY_PREFIX_CHARS);
125
+ }
126
+ function projectState(validation, state, apiKey) {
127
+ const matchingKey = state.apiKeys.find((key) => key.id === validation.apiKeyId);
128
+ const ttlSeconds = Math.max(1, state.cacheTtlSeconds || validation.cacheTtlSeconds || DEFAULT_CACHE_TTL_SECONDS);
129
+ const now = Date.now();
130
+ const entry = {
131
+ userId: validation.userId,
132
+ apiKeyId: validation.apiKeyId,
133
+ apiKeyPrefix: validation.prefix,
134
+ verificationToken: createVerificationToken(apiKey),
135
+ scopeSlugs: matchingKey?.scopeSlugs ?? validation.scopeSlugs,
136
+ contentRecordingEnabled: state.contentRecordingEnabled || validation.contentRecordingEnabled,
137
+ subscriptions: state.subscriptions.map((subscription) => ({
138
+ serverSlug: subscription.serverSlug,
139
+ tools: subscription.tools ?? 'all',
140
+ byokCredentialId: subscription.byokCredentialId,
141
+ })),
142
+ expiresAt: now + ttlSeconds * 1000,
143
+ staleUntil: now + DEFAULT_STALE_TTL_MS,
144
+ };
145
+ Object.defineProperty(entry, 'verificationToken', {
146
+ value: entry.verificationToken,
147
+ enumerable: false,
148
+ writable: true,
149
+ configurable: true,
150
+ });
151
+ return entry;
152
+ }
153
+ async function loadFreshContext(apiKey, prefix) {
154
+ const validation = await validateHostedApiKey(apiKey);
155
+ if (!validation.valid || !validation.userId || !validation.apiKeyId || !validation.prefix) {
156
+ authStateCache.remove(prefix);
157
+ return null;
158
+ }
159
+ const state = await getHostedUserState(validation.userId);
160
+ const entry = projectState({
161
+ userId: validation.userId,
162
+ apiKeyId: validation.apiKeyId,
163
+ prefix: validation.prefix,
164
+ scopeSlugs: validation.scopeSlugs,
165
+ contentRecordingEnabled: validation.contentRecordingEnabled,
166
+ cacheTtlSeconds: validation.cacheTtlSeconds,
167
+ }, state, apiKey);
168
+ authStateCache.set(prefix, entry);
169
+ return entry;
170
+ }
171
+ function publicContext(entry) {
172
+ return {
173
+ userId: entry.userId,
174
+ apiKeyId: entry.apiKeyId,
175
+ apiKeyPrefix: entry.apiKeyPrefix,
176
+ scopeSlugs: entry.scopeSlugs,
177
+ contentRecordingEnabled: entry.contentRecordingEnabled,
178
+ subscriptions: entry.subscriptions,
179
+ };
180
+ }
181
+ export async function validateHostedBearer(apiKey) {
182
+ if (!isHostedModeEnabled() || !isHostedApiKey(apiKey)) {
183
+ return null;
184
+ }
185
+ const prefix = extractApiKeyPrefix(apiKey);
186
+ if (!prefix)
187
+ return null;
188
+ authStateCache.prune();
189
+ const cached = authStateCache.get(prefix);
190
+ if (cached &&
191
+ Date.now() < cached.expiresAt &&
192
+ matchesVerificationToken(apiKey, cached.verificationToken)) {
193
+ authStateCache.touch(prefix, cached);
194
+ return publicContext(cached);
195
+ }
196
+ try {
197
+ const fresh = await loadFreshContext(apiKey, prefix);
198
+ return fresh ? publicContext(fresh) : null;
199
+ }
200
+ catch (error) {
201
+ if (cached &&
202
+ Date.now() < cached.staleUntil &&
203
+ matchesVerificationToken(apiKey, cached.verificationToken)) {
204
+ console.warn('[hosted] control plane unavailable, serving stale cached auth state', {
205
+ error: String(error),
206
+ });
207
+ authStateCache.touch(prefix, cached);
208
+ return publicContext(cached);
209
+ }
210
+ if (error instanceof HostedControlPlaneError) {
211
+ throw new HostedAuthUnavailableError(error.message);
212
+ }
213
+ throw error;
214
+ }
215
+ }
216
+ function findSubscription(context, serverSlug) {
217
+ return (context.subscriptions.find((subscription) => subscription.serverSlug === serverSlug) ?? null);
218
+ }
219
+ export function assertHostedToolAllowed(context, serverSlug, toolName) {
220
+ if (!context)
221
+ return;
222
+ if (context.scopeSlugs && !context.scopeSlugs.includes(serverSlug)) {
223
+ throw new HostedAuthorizationError('API key is not scoped for this hosted server');
224
+ }
225
+ const subscription = findSubscription(context, serverSlug);
226
+ if (!subscription) {
227
+ throw new HostedAuthorizationError('User is not subscribed to this hosted server');
228
+ }
229
+ if (subscription.tools !== 'all' && !subscription.tools.includes(toolName)) {
230
+ throw new HostedAuthorizationError('Tool is not enabled in this hosted toolset');
231
+ }
232
+ }
233
+ export function filterHostedTools(context, serverSlug, tools, nameSeparator) {
234
+ if (!context)
235
+ return tools;
236
+ if (context.scopeSlugs && !context.scopeSlugs.includes(serverSlug))
237
+ return [];
238
+ const subscription = findSubscription(context, serverSlug);
239
+ if (!subscription)
240
+ return [];
241
+ if (subscription.tools === 'all')
242
+ return tools;
243
+ const prefix = `${serverSlug}${nameSeparator}`;
244
+ return tools.filter((tool) => {
245
+ const cleanName = tool.name.startsWith(prefix) ? tool.name.slice(prefix.length) : tool.name;
246
+ return subscription.tools !== 'all' && subscription.tools.includes(cleanName);
247
+ });
248
+ }
249
+ export async function reserveHostedToolCall(context, serverSlug, toolName) {
250
+ if (!context)
251
+ return null;
252
+ assertHostedToolAllowed(context, serverSlug, toolName);
253
+ const hubRequestId = randomUUID();
254
+ const reservation = await reserveHostedCredit({
255
+ userId: context.userId,
256
+ apiKeyId: context.apiKeyId,
257
+ serverSlug,
258
+ toolName,
259
+ hubRequestId,
260
+ });
261
+ return {
262
+ reservationId: reservation.reservationId,
263
+ hubRequestId,
264
+ userId: context.userId,
265
+ apiKeyId: context.apiKeyId,
266
+ serverSlug,
267
+ toolName,
268
+ estimatedCostMillicents: reservation.estimatedCostMillicents,
269
+ contentRecordingEnabled: context.contentRecordingEnabled,
270
+ };
271
+ }
272
+ export async function settleHostedToolCall(reservation, input) {
273
+ if (!reservation)
274
+ return;
275
+ try {
276
+ const nodeIdentity = getHostedNodeIdentity();
277
+ await settleHostedCredit({
278
+ reservationId: reservation.reservationId,
279
+ hubEventId: randomUUID(),
280
+ success: input.success,
281
+ latencyMs: input.latencyMs,
282
+ occurredAt: new Date().toISOString(),
283
+ costMillicents: input.success ? reservation.estimatedCostMillicents : 0,
284
+ metadata: {
285
+ ...(input.metadata ?? {}),
286
+ hubRequestId: reservation.hubRequestId,
287
+ hubClusterId: nodeIdentity.clusterId,
288
+ hubNodeId: nodeIdentity.nodeId,
289
+ serverSlug: reservation.serverSlug,
290
+ toolName: reservation.toolName,
291
+ },
292
+ requestContent: reservation.contentRecordingEnabled ? input.requestContent : undefined,
293
+ responseContent: reservation.contentRecordingEnabled ? input.responseContent : undefined,
294
+ });
295
+ }
296
+ catch (error) {
297
+ console.warn('[hosted] failed to settle hosted tool call', {
298
+ reservationId: reservation.reservationId,
299
+ error: String(error),
300
+ });
301
+ }
302
+ }
303
+ export function applyHostedWebhookEvent(event) {
304
+ if (event.type === 'api_key.created' && event.prefix) {
305
+ authStateCache.remove(event.prefix);
306
+ return;
307
+ }
308
+ if (event.type === 'api_key.revoked' && event.keyId) {
309
+ authStateCache.invalidateApiKeyId(event.keyId);
310
+ return;
311
+ }
312
+ authStateCache.invalidateUser(event.userId);
313
+ }
314
+ //# sourceMappingURL=hostedAuthService.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hostedAuthService.js","sourceRoot":"","sources":["../../src/services/hostedAuthService.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,90 @@
1
+ import { REDACTED_SIGNATURE_VALUE, signInternalRequest, SIGNATURE_HEADER, TIMESTAMP_HEADER, } from './hostedInternalAuth.js';
2
+ export class HostedControlPlaneError extends Error {
3
+ constructor(message, status, code) {
4
+ super(message);
5
+ this.status = status;
6
+ this.code = code;
7
+ this.name = 'HostedControlPlaneError';
8
+ }
9
+ }
10
+ function controlPlaneBaseUrl() {
11
+ const raw = process.env.HOSTED_CONTROL_PLANE_URL || process.env.MCPHUB_APP_URL;
12
+ if (!raw) {
13
+ throw new HostedControlPlaneError('HOSTED_CONTROL_PLANE_URL is not configured');
14
+ }
15
+ return raw.replace(/\/+$/, '');
16
+ }
17
+ const VALIDATE_API_KEY_METHOD = 'POST';
18
+ const VALIDATE_API_KEY_PATH = '/api/internal/v1/keys/validate';
19
+ const REDACTED_VALIDATE_API_KEY_SIGNATURE_BODY = `{"apiKey":"${REDACTED_SIGNATURE_VALUE}"}`;
20
+ function createSignedControlPlaneHeaders(method, path, body) {
21
+ const { timestamp, signature } = signInternalRequest(method, path, body);
22
+ return {
23
+ [TIMESTAMP_HEADER]: timestamp,
24
+ [SIGNATURE_HEADER]: signature,
25
+ };
26
+ }
27
+ function createValidateHostedApiKeyAuthHeaders() {
28
+ const { timestamp, signature } = signInternalRequest(VALIDATE_API_KEY_METHOD, VALIDATE_API_KEY_PATH, REDACTED_VALIDATE_API_KEY_SIGNATURE_BODY);
29
+ return {
30
+ [TIMESTAMP_HEADER]: timestamp,
31
+ [SIGNATURE_HEADER]: signature,
32
+ };
33
+ }
34
+ async function parseControlPlaneResponse(response, method, path) {
35
+ const envelope = (await response.json().catch(() => null));
36
+ if (!response.ok || !envelope?.success) {
37
+ throw new HostedControlPlaneError(envelope?.message || `Control plane request failed: ${method} ${path}`, response.status, envelope?.code);
38
+ }
39
+ return envelope.data;
40
+ }
41
+ async function requestControlPlane(method, path, authHeaders, body) {
42
+ const bodyText = body === undefined ? '' : JSON.stringify(body);
43
+ const response = await fetch(`${controlPlaneBaseUrl()}${path}`, {
44
+ method,
45
+ headers: {
46
+ ...authHeaders,
47
+ ...(body === undefined ? {} : { 'Content-Type': 'application/json' }),
48
+ },
49
+ body: body === undefined ? undefined : bodyText,
50
+ signal: AbortSignal.timeout(Number(process.env.HOSTED_CONTROL_PLANE_TIMEOUT_MS || 5000)),
51
+ });
52
+ return parseControlPlaneResponse(response, method, path);
53
+ }
54
+ export async function validateHostedApiKey(apiKey) {
55
+ const body = {
56
+ apiKey,
57
+ };
58
+ const response = await fetch(`${controlPlaneBaseUrl()}${VALIDATE_API_KEY_PATH}`, {
59
+ method: VALIDATE_API_KEY_METHOD,
60
+ headers: {
61
+ ...createValidateHostedApiKeyAuthHeaders(),
62
+ 'Content-Type': 'application/json',
63
+ },
64
+ body: JSON.stringify(body),
65
+ signal: AbortSignal.timeout(Number(process.env.HOSTED_CONTROL_PLANE_TIMEOUT_MS || 5000)),
66
+ });
67
+ return parseControlPlaneResponse(response, VALIDATE_API_KEY_METHOD, VALIDATE_API_KEY_PATH);
68
+ }
69
+ export async function getHostedUserState(userId) {
70
+ const method = 'GET';
71
+ const path = `/api/internal/v1/users/${encodeURIComponent(userId)}/state`;
72
+ return requestControlPlane(method, path, createSignedControlPlaneHeaders(method, path));
73
+ }
74
+ export async function reserveHostedCredit(input) {
75
+ const method = 'POST';
76
+ const path = '/api/internal/v1/credits/reserve';
77
+ return requestControlPlane(method, path, createSignedControlPlaneHeaders(method, path, {
78
+ userId: input.userId,
79
+ apiKeyId: REDACTED_SIGNATURE_VALUE,
80
+ serverSlug: input.serverSlug,
81
+ toolName: input.toolName,
82
+ hubRequestId: input.hubRequestId,
83
+ }), input);
84
+ }
85
+ export async function settleHostedCredit(input) {
86
+ const method = 'POST';
87
+ const path = '/api/internal/v1/credits/settle';
88
+ return requestControlPlane(method, path, createSignedControlPlaneHeaders(method, path, input), input);
89
+ }
90
+ //# sourceMappingURL=hostedControlPlaneClient.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hostedControlPlaneClient.js","sourceRoot":"","sources":["../../src/services/hostedControlPlaneClient.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,206 @@
1
+ import { createClient } from 'redis';
2
+ import { applyHostedWebhookEvent } from './hostedAuthService.js';
3
+ import { isHostedModeEnabled } from './hostedMode.js';
4
+ import { getHostedNodeIdentity } from './hostedNodeIdentity.js';
5
+ const DEFAULT_EVENT_CHANNEL = 'mcphub:hosted-events';
6
+ const INITIAL_CONNECT_TIMEOUT_MS = 5000;
7
+ const VALID_EVENT_TYPES = new Set([
8
+ 'api_key.created',
9
+ 'api_key.revoked',
10
+ 'subscription.added',
11
+ 'subscription.removed',
12
+ 'byok.upserted',
13
+ 'byok.deleted',
14
+ 'user.suspended',
15
+ ]);
16
+ let subscriber = null;
17
+ let connectingClient = null;
18
+ let starting = false;
19
+ let retryTimer = null;
20
+ let lastStartupFailureSignature = null;
21
+ let subscriberEnabled = false;
22
+ function eventChannel() {
23
+ return process.env.HUB_EVENT_CHANNEL || DEFAULT_EVENT_CHANNEL;
24
+ }
25
+ function redisUrl() {
26
+ return process.env.HUB_EVENT_REDIS_URL || null;
27
+ }
28
+ function shouldSubscribeToRedisEvents() {
29
+ if (!isHostedModeEnabled())
30
+ return false;
31
+ if ((process.env.HUB_EVENT_TRANSPORT || '').toLowerCase() === 'webhook')
32
+ return false;
33
+ return Boolean(redisUrl());
34
+ }
35
+ function isHubClusterEvent(value) {
36
+ if (!value || typeof value !== 'object')
37
+ return false;
38
+ const event = value;
39
+ return (typeof event.type === 'string' &&
40
+ VALID_EVENT_TYPES.has(event.type) &&
41
+ typeof event.userId === 'string');
42
+ }
43
+ function clearRetryTimer() {
44
+ if (!retryTimer)
45
+ return;
46
+ clearTimeout(retryTimer);
47
+ retryTimer = null;
48
+ }
49
+ function isTrackedClient(client) {
50
+ return subscriber === client || connectingClient === client;
51
+ }
52
+ function shouldAbortStartup(client) {
53
+ return !subscriberEnabled || connectingClient !== client;
54
+ }
55
+ function scheduleRetry() {
56
+ if (!subscriberEnabled)
57
+ return;
58
+ clearRetryTimer();
59
+ retryTimer = setTimeout(() => {
60
+ retryTimer = null;
61
+ void startHostedEventSubscriber();
62
+ }, 5000);
63
+ retryTimer.unref?.();
64
+ }
65
+ function startupFailureSignature(channel, error) {
66
+ return `${channel}:${String(error)}`;
67
+ }
68
+ export async function startHostedEventSubscriber() {
69
+ if (subscriber || starting || !shouldSubscribeToRedisEvents())
70
+ return;
71
+ const url = redisUrl();
72
+ if (!url)
73
+ return;
74
+ subscriberEnabled = true;
75
+ starting = true;
76
+ const channel = eventChannel();
77
+ const nodeIdentity = getHostedNodeIdentity();
78
+ const client = createClient({
79
+ url,
80
+ socket: {
81
+ connectTimeout: INITIAL_CONNECT_TIMEOUT_MS,
82
+ reconnectStrategy: false,
83
+ },
84
+ });
85
+ connectingClient = client;
86
+ client.on('error', (error) => {
87
+ if (!subscriberEnabled || subscriber !== client) {
88
+ return;
89
+ }
90
+ console.warn('[hosted] Redis event subscriber error', {
91
+ error: String(error),
92
+ channel,
93
+ hubClusterId: nodeIdentity.clusterId,
94
+ hubNodeId: nodeIdentity.nodeId,
95
+ });
96
+ });
97
+ client.on('ready', () => {
98
+ if (!subscriberEnabled || !isTrackedClient(client)) {
99
+ return;
100
+ }
101
+ console.info('[hosted] Redis event subscriber ready', {
102
+ channel,
103
+ hubClusterId: nodeIdentity.clusterId,
104
+ hubNodeId: nodeIdentity.nodeId,
105
+ });
106
+ });
107
+ client.on('end', () => {
108
+ const wasActiveSubscriber = subscriber === client;
109
+ if (!wasActiveSubscriber) {
110
+ return;
111
+ }
112
+ subscriber = null;
113
+ console.warn('[hosted] Redis event subscriber connection ended', {
114
+ channel,
115
+ hubClusterId: nodeIdentity.clusterId,
116
+ hubNodeId: nodeIdentity.nodeId,
117
+ });
118
+ scheduleRetry();
119
+ });
120
+ try {
121
+ await client.connect();
122
+ if (shouldAbortStartup(client)) {
123
+ await client.quit().catch(() => undefined);
124
+ return;
125
+ }
126
+ await client.subscribe(channel, (message) => {
127
+ if (!subscriberEnabled || !isTrackedClient(client)) {
128
+ return;
129
+ }
130
+ try {
131
+ const event = JSON.parse(message);
132
+ if (!isHubClusterEvent(event)) {
133
+ console.warn('[hosted] Ignoring invalid cluster event', { channel });
134
+ return;
135
+ }
136
+ applyHostedWebhookEvent(event);
137
+ console.info('[hosted] Applied cluster event', {
138
+ eventId: event.eventId,
139
+ type: event.type,
140
+ userId: event.userId,
141
+ channel,
142
+ hubClusterId: nodeIdentity.clusterId,
143
+ hubNodeId: nodeIdentity.nodeId,
144
+ });
145
+ }
146
+ catch (error) {
147
+ console.warn('[hosted] Failed to process cluster event', {
148
+ error: String(error),
149
+ channel,
150
+ });
151
+ }
152
+ });
153
+ if (shouldAbortStartup(client)) {
154
+ await client.quit().catch(() => undefined);
155
+ return;
156
+ }
157
+ clearRetryTimer();
158
+ lastStartupFailureSignature = null;
159
+ subscriber = client;
160
+ connectingClient = null;
161
+ }
162
+ catch (error) {
163
+ const shouldHandleFailure = subscriberEnabled && connectingClient === client;
164
+ if (shouldHandleFailure) {
165
+ const failureSignature = startupFailureSignature(channel, error);
166
+ if (lastStartupFailureSignature !== failureSignature) {
167
+ lastStartupFailureSignature = failureSignature;
168
+ console.warn('[hosted] Failed to start Redis event subscriber; local cache TTL remains authoritative', {
169
+ error: String(error),
170
+ channel,
171
+ hubClusterId: nodeIdentity.clusterId,
172
+ hubNodeId: nodeIdentity.nodeId,
173
+ });
174
+ }
175
+ }
176
+ await client.quit().catch(() => undefined);
177
+ if (shouldHandleFailure) {
178
+ scheduleRetry();
179
+ }
180
+ }
181
+ finally {
182
+ if (connectingClient === client) {
183
+ connectingClient = null;
184
+ }
185
+ starting = false;
186
+ }
187
+ }
188
+ export async function stopHostedEventSubscriber() {
189
+ subscriberEnabled = false;
190
+ clearRetryTimer();
191
+ lastStartupFailureSignature = null;
192
+ const client = subscriber || connectingClient;
193
+ subscriber = null;
194
+ connectingClient = null;
195
+ if (!client)
196
+ return;
197
+ try {
198
+ await client.quit();
199
+ }
200
+ catch (error) {
201
+ console.warn('[hosted] Failed to stop Redis event subscriber cleanly', {
202
+ error: String(error),
203
+ });
204
+ }
205
+ }
206
+ //# sourceMappingURL=hostedEventSubscriber.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hostedEventSubscriber.js","sourceRoot":"","sources":["../../src/services/hostedEventSubscriber.ts"],"names":[],"mappings":"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"}