webpeel 0.12.0 → 0.12.2

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 (148) hide show
  1. package/README.md +82 -9
  2. package/dist/cli.js +97 -6
  3. package/dist/cli.js.map +1 -1
  4. package/dist/core/actions.d.ts +28 -0
  5. package/dist/core/actions.d.ts.map +1 -1
  6. package/dist/core/actions.js +60 -0
  7. package/dist/core/actions.js.map +1 -1
  8. package/dist/core/bm25-filter.d.ts +10 -0
  9. package/dist/core/bm25-filter.d.ts.map +1 -1
  10. package/dist/core/bm25-filter.js +40 -0
  11. package/dist/core/bm25-filter.js.map +1 -1
  12. package/dist/core/content-pruner.d.ts +12 -5
  13. package/dist/core/content-pruner.d.ts.map +1 -1
  14. package/dist/core/content-pruner.js +247 -190
  15. package/dist/core/content-pruner.js.map +1 -1
  16. package/dist/core/research.d.ts +67 -0
  17. package/dist/core/research.d.ts.map +1 -0
  18. package/dist/core/research.js +254 -0
  19. package/dist/core/research.js.map +1 -0
  20. package/dist/index.d.ts.map +1 -1
  21. package/dist/index.js +37 -3
  22. package/dist/index.js.map +1 -1
  23. package/dist/mcp/server.js +107 -2
  24. package/dist/mcp/server.js.map +1 -1
  25. package/dist/server/app.d.ts +14 -0
  26. package/dist/server/app.d.ts.map +1 -0
  27. package/dist/server/app.js +189 -0
  28. package/dist/server/app.js.map +1 -0
  29. package/dist/server/auth-store.d.ts +28 -0
  30. package/dist/server/auth-store.d.ts.map +1 -0
  31. package/dist/server/auth-store.js +89 -0
  32. package/dist/server/auth-store.js.map +1 -0
  33. package/dist/server/job-queue.d.ts +93 -0
  34. package/dist/server/job-queue.d.ts.map +1 -0
  35. package/dist/server/job-queue.js +144 -0
  36. package/dist/server/job-queue.js.map +1 -0
  37. package/dist/server/middleware/auth.d.ts +28 -0
  38. package/dist/server/middleware/auth.d.ts.map +1 -0
  39. package/dist/server/middleware/auth.js +183 -0
  40. package/dist/server/middleware/auth.js.map +1 -0
  41. package/dist/server/middleware/rate-limit.d.ts +23 -0
  42. package/dist/server/middleware/rate-limit.d.ts.map +1 -0
  43. package/dist/server/middleware/rate-limit.js +126 -0
  44. package/dist/server/middleware/rate-limit.js.map +1 -0
  45. package/dist/server/middleware/url-validator.d.ts +16 -0
  46. package/dist/server/middleware/url-validator.d.ts.map +1 -0
  47. package/dist/server/middleware/url-validator.js +187 -0
  48. package/dist/server/middleware/url-validator.js.map +1 -0
  49. package/dist/server/pg-auth-store.d.ts +129 -0
  50. package/dist/server/pg-auth-store.d.ts.map +1 -0
  51. package/dist/server/pg-auth-store.js +457 -0
  52. package/dist/server/pg-auth-store.js.map +1 -0
  53. package/dist/server/pg-job-queue.d.ts +60 -0
  54. package/dist/server/pg-job-queue.d.ts.map +1 -0
  55. package/dist/server/pg-job-queue.js +365 -0
  56. package/dist/server/pg-job-queue.js.map +1 -0
  57. package/dist/server/premium/domain-intel.d.ts +17 -0
  58. package/dist/server/premium/domain-intel.d.ts.map +1 -0
  59. package/dist/server/premium/domain-intel.js +134 -0
  60. package/dist/server/premium/domain-intel.js.map +1 -0
  61. package/dist/server/premium/index.d.ts +18 -0
  62. package/dist/server/premium/index.d.ts.map +1 -0
  63. package/dist/server/premium/index.js +36 -0
  64. package/dist/server/premium/index.js.map +1 -0
  65. package/dist/server/premium/swr-cache.d.ts +15 -0
  66. package/dist/server/premium/swr-cache.d.ts.map +1 -0
  67. package/dist/server/premium/swr-cache.js +35 -0
  68. package/dist/server/premium/swr-cache.js.map +1 -0
  69. package/dist/server/routes/activity.d.ts +7 -0
  70. package/dist/server/routes/activity.d.ts.map +1 -0
  71. package/dist/server/routes/activity.js +66 -0
  72. package/dist/server/routes/activity.js.map +1 -0
  73. package/dist/server/routes/agent.d.ts +12 -0
  74. package/dist/server/routes/agent.d.ts.map +1 -0
  75. package/dist/server/routes/agent.js +356 -0
  76. package/dist/server/routes/agent.js.map +1 -0
  77. package/dist/server/routes/answer.d.ts +6 -0
  78. package/dist/server/routes/answer.d.ts.map +1 -0
  79. package/dist/server/routes/answer.js +124 -0
  80. package/dist/server/routes/answer.js.map +1 -0
  81. package/dist/server/routes/batch.d.ts +7 -0
  82. package/dist/server/routes/batch.d.ts.map +1 -0
  83. package/dist/server/routes/batch.js +287 -0
  84. package/dist/server/routes/batch.js.map +1 -0
  85. package/dist/server/routes/cli-usage.d.ts +7 -0
  86. package/dist/server/routes/cli-usage.d.ts.map +1 -0
  87. package/dist/server/routes/cli-usage.js +121 -0
  88. package/dist/server/routes/cli-usage.js.map +1 -0
  89. package/dist/server/routes/compat.d.ts +24 -0
  90. package/dist/server/routes/compat.d.ts.map +1 -0
  91. package/dist/server/routes/compat.js +651 -0
  92. package/dist/server/routes/compat.js.map +1 -0
  93. package/dist/server/routes/extract.d.ts +9 -0
  94. package/dist/server/routes/extract.d.ts.map +1 -0
  95. package/dist/server/routes/extract.js +121 -0
  96. package/dist/server/routes/extract.js.map +1 -0
  97. package/dist/server/routes/fetch.d.ts +7 -0
  98. package/dist/server/routes/fetch.d.ts.map +1 -0
  99. package/dist/server/routes/fetch.js +537 -0
  100. package/dist/server/routes/fetch.js.map +1 -0
  101. package/dist/server/routes/health.d.ts +8 -0
  102. package/dist/server/routes/health.d.ts.map +1 -0
  103. package/dist/server/routes/health.js +36 -0
  104. package/dist/server/routes/health.js.map +1 -0
  105. package/dist/server/routes/jobs.d.ts +8 -0
  106. package/dist/server/routes/jobs.d.ts.map +1 -0
  107. package/dist/server/routes/jobs.js +374 -0
  108. package/dist/server/routes/jobs.js.map +1 -0
  109. package/dist/server/routes/mcp.d.ts +16 -0
  110. package/dist/server/routes/mcp.d.ts.map +1 -0
  111. package/dist/server/routes/mcp.js +475 -0
  112. package/dist/server/routes/mcp.js.map +1 -0
  113. package/dist/server/routes/oauth.d.ts +10 -0
  114. package/dist/server/routes/oauth.d.ts.map +1 -0
  115. package/dist/server/routes/oauth.js +296 -0
  116. package/dist/server/routes/oauth.js.map +1 -0
  117. package/dist/server/routes/screenshot.d.ts +10 -0
  118. package/dist/server/routes/screenshot.d.ts.map +1 -0
  119. package/dist/server/routes/screenshot.js +217 -0
  120. package/dist/server/routes/screenshot.js.map +1 -0
  121. package/dist/server/routes/search.d.ts +7 -0
  122. package/dist/server/routes/search.d.ts.map +1 -0
  123. package/dist/server/routes/search.js +287 -0
  124. package/dist/server/routes/search.js.map +1 -0
  125. package/dist/server/routes/stats.d.ts +7 -0
  126. package/dist/server/routes/stats.d.ts.map +1 -0
  127. package/dist/server/routes/stats.js +65 -0
  128. package/dist/server/routes/stats.js.map +1 -0
  129. package/dist/server/routes/stripe.d.ts +9 -0
  130. package/dist/server/routes/stripe.d.ts.map +1 -0
  131. package/dist/server/routes/stripe.js +233 -0
  132. package/dist/server/routes/stripe.js.map +1 -0
  133. package/dist/server/routes/users.d.ts +9 -0
  134. package/dist/server/routes/users.d.ts.map +1 -0
  135. package/dist/server/routes/users.js +954 -0
  136. package/dist/server/routes/users.js.map +1 -0
  137. package/dist/server/routes/webhooks.d.ts +15 -0
  138. package/dist/server/routes/webhooks.d.ts.map +1 -0
  139. package/dist/server/routes/webhooks.js +73 -0
  140. package/dist/server/routes/webhooks.js.map +1 -0
  141. package/dist/server/sentry.d.ts +14 -0
  142. package/dist/server/sentry.d.ts.map +1 -0
  143. package/dist/server/sentry.js +39 -0
  144. package/dist/server/sentry.js.map +1 -0
  145. package/dist/types.d.ts +13 -0
  146. package/dist/types.d.ts.map +1 -1
  147. package/dist/types.js.map +1 -1
  148. package/package.json +3 -2
@@ -0,0 +1,457 @@
1
+ /**
2
+ * PostgreSQL-backed auth store for production deployments
3
+ * Uses SHA-256 hashing for API keys and tracks WEEKLY usage with burst limits
4
+ */
5
+ import pg from 'pg';
6
+ import crypto from 'crypto';
7
+ const { Pool } = pg;
8
+ // Extra usage cost constants
9
+ const EXTRA_USAGE_RATES = {
10
+ basic: 0.002, // $0.002 per basic fetch
11
+ stealth: 0.01, // $0.01 per stealth fetch
12
+ captcha: 0.02, // $0.02 per CAPTCHA solve
13
+ search: 0.001, // $0.001 per search
14
+ };
15
+ /**
16
+ * PostgreSQL auth store for production
17
+ */
18
+ export class PostgresAuthStore {
19
+ pool;
20
+ constructor(connectionString) {
21
+ const dbUrl = connectionString || process.env.DATABASE_URL;
22
+ if (!dbUrl) {
23
+ throw new Error('DATABASE_URL environment variable is required for PostgresAuthStore');
24
+ }
25
+ this.pool = new Pool({
26
+ connectionString: dbUrl,
27
+ // TLS: enabled when DATABASE_URL contains sslmode=require.
28
+ // Secure by default (rejectUnauthorized: true); set PG_REJECT_UNAUTHORIZED=false
29
+ // only for managed DBs (Render/Neon/Supabase) that use self-signed certs.
30
+ ssl: process.env.DATABASE_URL?.includes('sslmode=require')
31
+ ? { rejectUnauthorized: process.env.PG_REJECT_UNAUTHORIZED !== 'false' }
32
+ : undefined,
33
+ max: 20,
34
+ idleTimeoutMillis: 30000,
35
+ connectionTimeoutMillis: 10000,
36
+ });
37
+ }
38
+ /**
39
+ * Hash API key with SHA-256
40
+ * SECURITY: Never store raw API keys
41
+ */
42
+ hashKey(key) {
43
+ return crypto.createHash('sha256').update(key).digest('hex');
44
+ }
45
+ /**
46
+ * Get current ISO week in YYYY-WXX format (e.g., "2026-W07")
47
+ */
48
+ getCurrentWeek() {
49
+ const now = new Date();
50
+ const year = now.getUTCFullYear();
51
+ const jan4 = new Date(Date.UTC(year, 0, 4));
52
+ const weekNum = Math.ceil(((now.getTime() - jan4.getTime()) / 86400000 + jan4.getUTCDay() + 1) / 7);
53
+ return `${year}-W${String(weekNum).padStart(2, '0')}`;
54
+ }
55
+ /**
56
+ * Get previous ISO week in YYYY-WXX format
57
+ */
58
+ getPreviousWeek() {
59
+ const now = new Date();
60
+ const lastWeek = new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000);
61
+ const year = lastWeek.getUTCFullYear();
62
+ const jan4 = new Date(Date.UTC(year, 0, 4));
63
+ const weekNum = Math.ceil(((lastWeek.getTime() - jan4.getTime()) / 86400000 + jan4.getUTCDay() + 1) / 7);
64
+ return `${year}-W${String(weekNum).padStart(2, '0')}`;
65
+ }
66
+ /**
67
+ * Get next Monday 00:00 UTC (week reset time)
68
+ */
69
+ getWeekResetTime() {
70
+ const now = new Date();
71
+ const dayOfWeek = now.getUTCDay();
72
+ const daysUntilMonday = dayOfWeek === 0 ? 1 : 8 - dayOfWeek;
73
+ const nextMonday = new Date(now);
74
+ nextMonday.setUTCDate(now.getUTCDate() + daysUntilMonday);
75
+ nextMonday.setUTCHours(0, 0, 0, 0);
76
+ return nextMonday;
77
+ }
78
+ /**
79
+ * Get current hour bucket in YYYY-MM-DDTHH format (UTC)
80
+ */
81
+ getCurrentHour() {
82
+ const now = new Date();
83
+ return now.toISOString().substring(0, 13); // "2026-02-12T20"
84
+ }
85
+ /**
86
+ * Get human-readable time until next hour
87
+ */
88
+ getTimeUntilNextHour() {
89
+ const now = new Date();
90
+ const minutesRemaining = 59 - now.getUTCMinutes();
91
+ if (minutesRemaining === 0) {
92
+ return '< 1 min';
93
+ }
94
+ return `${minutesRemaining} min`;
95
+ }
96
+ /**
97
+ * Validate API key and return user info
98
+ * SECURITY: Uses SHA-256 hash comparison, updates last_used_at
99
+ */
100
+ async validateKey(key) {
101
+ if (!key || typeof key !== 'string') {
102
+ return null;
103
+ }
104
+ const keyHash = this.hashKey(key);
105
+ try {
106
+ const result = await this.pool.query(`SELECT
107
+ ak.id,
108
+ ak.user_id,
109
+ ak.key_prefix,
110
+ ak.name,
111
+ u.tier,
112
+ u.rate_limit,
113
+ u.weekly_limit,
114
+ u.burst_limit,
115
+ u.email
116
+ FROM api_keys ak
117
+ JOIN users u ON ak.user_id = u.id
118
+ WHERE ak.key_hash = $1 AND ak.is_active = true`, [keyHash]);
119
+ if (result.rows.length === 0) {
120
+ return null;
121
+ }
122
+ const row = result.rows[0];
123
+ // Update last_used_at (fire and forget, don't wait)
124
+ this.pool.query('UPDATE api_keys SET last_used_at = now() WHERE id = $1', [row.id]).catch(err => console.error('Failed to update last_used_at:', err));
125
+ return {
126
+ key,
127
+ tier: row.tier,
128
+ rateLimit: row.rate_limit,
129
+ accountId: row.user_id,
130
+ createdAt: new Date(),
131
+ };
132
+ }
133
+ catch (error) {
134
+ console.error('Failed to validate API key:', error);
135
+ return null;
136
+ }
137
+ }
138
+ /**
139
+ * Track weekly usage for an API key
140
+ * SECURITY: Uses UPSERT to prevent race conditions
141
+ */
142
+ async trackUsage(key, fetchType) {
143
+ const keyHash = this.hashKey(key);
144
+ const week = this.getCurrentWeek();
145
+ try {
146
+ // Get API key ID and user ID
147
+ const keyResult = await this.pool.query('SELECT id, user_id FROM api_keys WHERE key_hash = $1', [keyHash]);
148
+ if (keyResult.rows.length === 0) {
149
+ return;
150
+ }
151
+ const { id: apiKeyId, user_id: userId } = keyResult.rows[0];
152
+ // Determine which counter to increment
153
+ const columnMap = {
154
+ basic: 'basic_count',
155
+ stealth: 'stealth_count',
156
+ captcha: 'captcha_count',
157
+ search: 'search_count',
158
+ };
159
+ const column = columnMap[fetchType];
160
+ // UPSERT usage record (total_count is GENERATED, don't touch it)
161
+ await this.pool.query(`INSERT INTO weekly_usage (user_id, api_key_id, week, ${column})
162
+ VALUES ($1, $2, $3, 1)
163
+ ON CONFLICT (api_key_id, week)
164
+ DO UPDATE SET
165
+ ${column} = weekly_usage.${column} + 1,
166
+ updated_at = now()`, [userId, apiKeyId, week]);
167
+ }
168
+ catch (error) {
169
+ console.error('Failed to track usage:', error);
170
+ throw error;
171
+ }
172
+ }
173
+ /**
174
+ * Track burst usage (hourly limit)
175
+ */
176
+ async trackBurstUsage(key) {
177
+ const keyHash = this.hashKey(key);
178
+ const hourBucket = this.getCurrentHour();
179
+ try {
180
+ const keyResult = await this.pool.query('SELECT id FROM api_keys WHERE key_hash = $1', [keyHash]);
181
+ if (keyResult.rows.length === 0) {
182
+ return;
183
+ }
184
+ const apiKeyId = keyResult.rows[0].id;
185
+ // UPSERT burst usage
186
+ await this.pool.query(`INSERT INTO burst_usage (api_key_id, hour_bucket, count)
187
+ VALUES ($1, $2, 1)
188
+ ON CONFLICT (api_key_id, hour_bucket)
189
+ DO UPDATE SET
190
+ count = burst_usage.count + 1,
191
+ updated_at = now()`, [apiKeyId, hourBucket]);
192
+ }
193
+ catch (error) {
194
+ console.error('Failed to track burst usage:', error);
195
+ throw error;
196
+ }
197
+ }
198
+ /**
199
+ * Check burst limit (hourly)
200
+ */
201
+ async checkBurstLimit(key) {
202
+ const keyHash = this.hashKey(key);
203
+ const hourBucket = this.getCurrentHour();
204
+ try {
205
+ const result = await this.pool.query(`SELECT
206
+ u.burst_limit,
207
+ COALESCE(bu.count, 0) as count
208
+ FROM api_keys ak
209
+ JOIN users u ON ak.user_id = u.id
210
+ LEFT JOIN burst_usage bu ON bu.api_key_id = ak.id AND bu.hour_bucket = $2
211
+ WHERE ak.key_hash = $1`, [keyHash, hourBucket]);
212
+ if (result.rows.length === 0) {
213
+ return {
214
+ allowed: false,
215
+ burst: {
216
+ hourBucket,
217
+ count: 0,
218
+ limit: 0,
219
+ remaining: 0,
220
+ resetsIn: this.getTimeUntilNextHour(),
221
+ },
222
+ };
223
+ }
224
+ const row = result.rows[0];
225
+ const allowed = row.count < row.burst_limit;
226
+ return {
227
+ allowed,
228
+ burst: {
229
+ hourBucket,
230
+ count: row.count,
231
+ limit: row.burst_limit,
232
+ remaining: Math.max(0, row.burst_limit - row.count),
233
+ resetsIn: this.getTimeUntilNextHour(),
234
+ },
235
+ };
236
+ }
237
+ catch (error) {
238
+ console.error('Failed to check burst limit:', error);
239
+ return {
240
+ allowed: false,
241
+ burst: {
242
+ hourBucket,
243
+ count: 0,
244
+ limit: 0,
245
+ remaining: 0,
246
+ resetsIn: this.getTimeUntilNextHour(),
247
+ },
248
+ };
249
+ }
250
+ }
251
+ /**
252
+ * Get weekly usage info for an API key with rollover calculation
253
+ */
254
+ async getUsage(key) {
255
+ const keyHash = this.hashKey(key);
256
+ const currentWeek = this.getCurrentWeek();
257
+ const previousWeek = this.getPreviousWeek();
258
+ try {
259
+ const result = await this.pool.query(`SELECT
260
+ u.weekly_limit,
261
+ COALESCE(curr.basic_count, 0) as basic_count,
262
+ COALESCE(curr.stealth_count, 0) as stealth_count,
263
+ COALESCE(curr.captcha_count, 0) as captcha_count,
264
+ COALESCE(curr.search_count, 0) as search_count,
265
+ COALESCE(curr.total_count, 0) as current_used,
266
+ COALESCE(prev.total_count, 0) as prev_used,
267
+ COALESCE(curr.rollover_credits, 0) as rollover_credits
268
+ FROM api_keys ak
269
+ JOIN users u ON ak.user_id = u.id
270
+ LEFT JOIN weekly_usage curr ON curr.api_key_id = ak.id AND curr.week = $2
271
+ LEFT JOIN weekly_usage prev ON prev.api_key_id = ak.id AND prev.week = $3
272
+ WHERE ak.key_hash = $1`, [keyHash, currentWeek, previousWeek]);
273
+ if (result.rows.length === 0) {
274
+ return null;
275
+ }
276
+ const row = result.rows[0];
277
+ const weeklyLimit = row.weekly_limit;
278
+ const currentUsed = row.current_used;
279
+ const prevUsed = row.prev_used;
280
+ const rolloverCredits = row.rollover_credits;
281
+ // Calculate rollover: MIN(unused_last_week, weekly_limit)
282
+ const prevUnused = Math.max(0, weeklyLimit - prevUsed);
283
+ const calculatedRollover = Math.min(prevUnused, weeklyLimit);
284
+ // Update rollover if it's the first access this week
285
+ if (rolloverCredits === 0 && calculatedRollover > 0) {
286
+ await this.pool.query(`INSERT INTO weekly_usage (user_id, api_key_id, week, rollover_credits, updated_at)
287
+ SELECT user_id, id, $2, $3, now()
288
+ FROM api_keys WHERE key_hash = $1
289
+ ON CONFLICT (api_key_id, week)
290
+ DO UPDATE SET rollover_credits = $3`, [keyHash, currentWeek, calculatedRollover]);
291
+ }
292
+ const effectiveRollover = rolloverCredits > 0 ? rolloverCredits : calculatedRollover;
293
+ const totalAvailable = weeklyLimit + effectiveRollover;
294
+ const remaining = Math.max(0, totalAvailable - currentUsed);
295
+ const percentUsed = totalAvailable > 0 ? Math.round((currentUsed / totalAvailable) * 100) : 0;
296
+ return {
297
+ week: currentWeek,
298
+ basicCount: row.basic_count,
299
+ stealthCount: row.stealth_count,
300
+ captchaCount: row.captcha_count,
301
+ searchCount: row.search_count,
302
+ totalUsed: currentUsed,
303
+ weeklyLimit,
304
+ rolloverCredits: effectiveRollover,
305
+ totalAvailable,
306
+ remaining,
307
+ percentUsed,
308
+ resetsAt: this.getWeekResetTime().toISOString(),
309
+ };
310
+ }
311
+ catch (error) {
312
+ console.error('Failed to get usage:', error);
313
+ return null;
314
+ }
315
+ }
316
+ /**
317
+ * Check if API key has exceeded weekly limit
318
+ */
319
+ async checkLimit(key) {
320
+ const usage = await this.getUsage(key);
321
+ if (!usage) {
322
+ return { allowed: false };
323
+ }
324
+ const allowed = usage.remaining > 0;
325
+ return { allowed, usage };
326
+ }
327
+ /**
328
+ * Get extra usage info for a user
329
+ */
330
+ async getExtraUsageInfo(key) {
331
+ const keyHash = this.hashKey(key);
332
+ try {
333
+ const result = await this.pool.query(`SELECT
334
+ u.extra_usage_enabled,
335
+ u.extra_usage_balance,
336
+ u.extra_usage_spent,
337
+ u.extra_usage_spending_limit,
338
+ u.auto_reload_enabled,
339
+ u.extra_usage_period_start
340
+ FROM api_keys ak
341
+ JOIN users u ON ak.user_id = u.id
342
+ WHERE ak.key_hash = $1`, [keyHash]);
343
+ if (result.rows.length === 0) {
344
+ return null;
345
+ }
346
+ const row = result.rows[0];
347
+ // Calculate next month reset (1st of next month, 00:00 UTC)
348
+ const now = new Date();
349
+ const nextMonth = new Date(Date.UTC(now.getUTCFullYear(), now.getUTCMonth() + 1, 1, 0, 0, 0, 0));
350
+ const percentUsed = row.extra_usage_spending_limit > 0
351
+ ? Math.round((parseFloat(row.extra_usage_spent) / parseFloat(row.extra_usage_spending_limit)) * 100)
352
+ : 0;
353
+ return {
354
+ enabled: row.extra_usage_enabled,
355
+ balance: parseFloat(row.extra_usage_balance),
356
+ spent: parseFloat(row.extra_usage_spent),
357
+ spendingLimit: parseFloat(row.extra_usage_spending_limit),
358
+ autoReload: row.auto_reload_enabled,
359
+ percentUsed,
360
+ resetsAt: nextMonth.toISOString(),
361
+ };
362
+ }
363
+ catch (error) {
364
+ console.error('Failed to get extra usage info:', error);
365
+ return null;
366
+ }
367
+ }
368
+ /**
369
+ * Check if extra usage can be used
370
+ */
371
+ async canUseExtraUsage(key) {
372
+ const info = await this.getExtraUsageInfo(key);
373
+ if (!info || !info.enabled) {
374
+ return false;
375
+ }
376
+ // Check if under spending limit and has balance
377
+ return info.balance > 0 && info.spent < info.spendingLimit;
378
+ }
379
+ /**
380
+ * Track extra usage and deduct from balance
381
+ */
382
+ async trackExtraUsage(key, fetchType, url, processingTimeMs, statusCode) {
383
+ const keyHash = this.hashKey(key);
384
+ const cost = EXTRA_USAGE_RATES[fetchType];
385
+ try {
386
+ // Get API key and user info
387
+ const keyResult = await this.pool.query(`SELECT
388
+ ak.id as api_key_id,
389
+ ak.user_id,
390
+ u.extra_usage_balance,
391
+ u.extra_usage_spent
392
+ FROM api_keys ak
393
+ JOIN users u ON ak.user_id = u.id
394
+ WHERE ak.key_hash = $1`, [keyHash]);
395
+ if (keyResult.rows.length === 0) {
396
+ return { success: false, cost: 0, newBalance: 0 };
397
+ }
398
+ const { api_key_id, user_id, extra_usage_balance } = keyResult.rows[0];
399
+ const currentBalance = parseFloat(extra_usage_balance);
400
+ if (currentBalance < cost) {
401
+ return { success: false, cost, newBalance: currentBalance };
402
+ }
403
+ // Start transaction
404
+ const client = await this.pool.connect();
405
+ try {
406
+ await client.query('BEGIN');
407
+ // Deduct from balance and add to spent
408
+ const updateResult = await client.query(`UPDATE users
409
+ SET
410
+ extra_usage_balance = extra_usage_balance - $1,
411
+ extra_usage_spent = extra_usage_spent + $1,
412
+ updated_at = now()
413
+ WHERE id = $2
414
+ RETURNING extra_usage_balance`, [cost, user_id]);
415
+ const newBalance = parseFloat(updateResult.rows[0].extra_usage_balance);
416
+ // Log to extra_usage_logs
417
+ await client.query(`INSERT INTO extra_usage_logs
418
+ (user_id, api_key_id, fetch_type, url, cost, processing_time_ms, status_code)
419
+ VALUES ($1, $2, $3, $4, $5, $6, $7)`, [user_id, api_key_id, fetchType, url, cost, processingTimeMs, statusCode]);
420
+ await client.query('COMMIT');
421
+ return { success: true, cost, newBalance };
422
+ }
423
+ catch (error) {
424
+ await client.query('ROLLBACK');
425
+ throw error;
426
+ }
427
+ finally {
428
+ client.release();
429
+ }
430
+ }
431
+ catch (error) {
432
+ console.error('Failed to track extra usage:', error);
433
+ return { success: false, cost, newBalance: 0 };
434
+ }
435
+ }
436
+ /**
437
+ * Generate a cryptographically secure API key
438
+ * Format: wp_live_ + 32 random hex chars (total 40 chars)
439
+ */
440
+ static generateApiKey() {
441
+ const randomBytes = crypto.randomBytes(16).toString('hex');
442
+ return `wp_live_${randomBytes}`;
443
+ }
444
+ /**
445
+ * Get key prefix (first 12 characters for display)
446
+ */
447
+ static getKeyPrefix(key) {
448
+ return key.substring(0, 12);
449
+ }
450
+ /**
451
+ * Close the database pool
452
+ */
453
+ async close() {
454
+ await this.pool.end();
455
+ }
456
+ }
457
+ //# sourceMappingURL=pg-auth-store.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pg-auth-store.js","sourceRoot":"","sources":["../../src/server/pg-auth-store.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,MAAM,IAAI,CAAC;AACpB,OAAO,MAAM,MAAM,QAAQ,CAAC;AAG5B,MAAM,EAAE,IAAI,EAAE,GAAG,EAAE,CAAC;AAmCpB,6BAA6B;AAC7B,MAAM,iBAAiB,GAAG;IACxB,KAAK,EAAE,KAAK,EAAK,yBAAyB;IAC1C,OAAO,EAAE,IAAI,EAAI,0BAA0B;IAC3C,OAAO,EAAE,IAAI,EAAI,0BAA0B;IAC3C,MAAM,EAAE,KAAK,EAAI,oBAAoB;CACtC,CAAC;AAEF;;GAEG;AACH,MAAM,OAAO,iBAAiB;IACpB,IAAI,CAAU;IAEtB,YAAY,gBAAyB;QACnC,MAAM,KAAK,GAAG,gBAAgB,IAAI,OAAO,CAAC,GAAG,CAAC,YAAY,CAAC;QAE3D,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,qEAAqE,CAAC,CAAC;QACzF,CAAC;QAED,IAAI,CAAC,IAAI,GAAG,IAAI,IAAI,CAAC;YACnB,gBAAgB,EAAE,KAAK;YACvB,2DAA2D;YAC3D,iFAAiF;YACjF,0EAA0E;YAC1E,GAAG,EAAE,OAAO,CAAC,GAAG,CAAC,YAAY,EAAE,QAAQ,CAAC,iBAAiB,CAAC;gBACxD,CAAC,CAAC,EAAE,kBAAkB,EAAE,OAAO,CAAC,GAAG,CAAC,sBAAsB,KAAK,OAAO,EAAE;gBACxE,CAAC,CAAC,SAAS;YACb,GAAG,EAAE,EAAE;YACP,iBAAiB,EAAE,KAAK;YACxB,uBAAuB,EAAE,KAAK;SAC/B,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACK,OAAO,CAAC,GAAW;QACzB,OAAO,MAAM,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;IAC/D,CAAC;IAED;;OAEG;IACK,cAAc;QACpB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,IAAI,GAAG,GAAG,CAAC,cAAc,EAAE,CAAC;QAClC,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QAC5C,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC,GAAG,QAAQ,GAAG,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;QACpG,OAAO,GAAG,IAAI,KAAK,MAAM,CAAC,OAAO,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;IACxD,CAAC;IAED;;OAEG;IACK,eAAe;QACrB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,QAAQ,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,OAAO,EAAE,GAAG,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC;QACnE,MAAM,IAAI,GAAG,QAAQ,CAAC,cAAc,EAAE,CAAC;QACvC,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;QAC5C,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC,GAAG,QAAQ,GAAG,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;QACzG,OAAO,GAAG,IAAI,KAAK,MAAM,CAAC,OAAO,CAAC,CAAC,QAAQ,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC;IACxD,CAAC;IAED;;OAEG;IACK,gBAAgB;QACtB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,SAAS,GAAG,GAAG,CAAC,SAAS,EAAE,CAAC;QAClC,MAAM,eAAe,GAAG,SAAS,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC;QAC5D,MAAM,UAAU,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,CAAC;QACjC,UAAU,CAAC,UAAU,CAAC,GAAG,CAAC,UAAU,EAAE,GAAG,eAAe,CAAC,CAAC;QAC1D,UAAU,CAAC,WAAW,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QACnC,OAAO,UAAU,CAAC;IACpB,CAAC;IAED;;OAEG;IACK,cAAc;QACpB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,OAAO,GAAG,CAAC,WAAW,EAAE,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,kBAAkB;IAC/D,CAAC;IAED;;OAEG;IACK,oBAAoB;QAC1B,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,gBAAgB,GAAG,EAAE,GAAG,GAAG,CAAC,aAAa,EAAE,CAAC;QAClD,IAAI,gBAAgB,KAAK,CAAC,EAAE,CAAC;YAC3B,OAAO,SAAS,CAAC;QACnB,CAAC;QACD,OAAO,GAAG,gBAAgB,MAAM,CAAC;IACnC,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,WAAW,CAAC,GAAW;QAC3B,IAAI,CAAC,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;YACpC,OAAO,IAAI,CAAC;QACd,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAElC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CAClC;;;;;;;;;;;;uDAY+C,EAC/C,CAAC,OAAO,CAAC,CACV,CAAC;YAEF,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC7B,OAAO,IAAI,CAAC;YACd,CAAC;YAED,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAE3B,oDAAoD;YACpD,IAAI,CAAC,IAAI,CAAC,KAAK,CACb,wDAAwD,EACxD,CAAC,GAAG,CAAC,EAAE,CAAC,CACT,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,gCAAgC,EAAE,GAAG,CAAC,CAAC,CAAC;YAErE,OAAO;gBACL,GAAG;gBACH,IAAI,EAAE,GAAG,CAAC,IAAI;gBACd,SAAS,EAAE,GAAG,CAAC,UAAU;gBACzB,SAAS,EAAE,GAAG,CAAC,OAAO;gBACtB,SAAS,EAAE,IAAI,IAAI,EAAE;aACtB,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,KAAK,CAAC,CAAC;YACpD,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,UAAU,CACd,GAAW,EACX,SAAqD;QAErD,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAClC,MAAM,IAAI,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;QAEnC,IAAI,CAAC;YACH,6BAA6B;YAC7B,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CACrC,sDAAsD,EACtD,CAAC,OAAO,CAAC,CACV,CAAC;YAEF,IAAI,SAAS,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAChC,OAAO;YACT,CAAC;YAED,MAAM,EAAE,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAE5D,uCAAuC;YACvC,MAAM,SAAS,GAAG;gBAChB,KAAK,EAAE,aAAa;gBACpB,OAAO,EAAE,eAAe;gBACxB,OAAO,EAAE,eAAe;gBACxB,MAAM,EAAE,cAAc;aACvB,CAAC;YAEF,MAAM,MAAM,GAAG,SAAS,CAAC,SAAS,CAAC,CAAC;YAEpC,iEAAiE;YACjE,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CACnB,wDAAwD,MAAM;;;;YAI1D,MAAM,mBAAmB,MAAM;6BACd,EACrB,CAAC,MAAM,EAAE,QAAQ,EAAE,IAAI,CAAC,CACzB,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,wBAAwB,EAAE,KAAK,CAAC,CAAC;YAC/C,MAAM,KAAK,CAAC;QACd,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,eAAe,CAAC,GAAW;QAC/B,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAClC,MAAM,UAAU,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;QAEzC,IAAI,CAAC;YACH,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CACrC,6CAA6C,EAC7C,CAAC,OAAO,CAAC,CACV,CAAC;YAEF,IAAI,SAAS,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAChC,OAAO;YACT,CAAC;YAED,MAAM,QAAQ,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YAEtC,qBAAqB;YACrB,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CACnB;;;;;6BAKqB,EACrB,CAAC,QAAQ,EAAE,UAAU,CAAC,CACvB,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;YACrD,MAAM,KAAK,CAAC;QACd,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,eAAe,CAAC,GAAW;QAC/B,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAClC,MAAM,UAAU,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;QAEzC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CAClC;;;;;;+BAMuB,EACvB,CAAC,OAAO,EAAE,UAAU,CAAC,CACtB,CAAC;YAEF,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC7B,OAAO;oBACL,OAAO,EAAE,KAAK;oBACd,KAAK,EAAE;wBACL,UAAU;wBACV,KAAK,EAAE,CAAC;wBACR,KAAK,EAAE,CAAC;wBACR,SAAS,EAAE,CAAC;wBACZ,QAAQ,EAAE,IAAI,CAAC,oBAAoB,EAAE;qBACtC;iBACF,CAAC;YACJ,CAAC;YAED,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC3B,MAAM,OAAO,GAAG,GAAG,CAAC,KAAK,GAAG,GAAG,CAAC,WAAW,CAAC;YAE5C,OAAO;gBACL,OAAO;gBACP,KAAK,EAAE;oBACL,UAAU;oBACV,KAAK,EAAE,GAAG,CAAC,KAAK;oBAChB,KAAK,EAAE,GAAG,CAAC,WAAW;oBACtB,SAAS,EAAE,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,GAAG,CAAC,WAAW,GAAG,GAAG,CAAC,KAAK,CAAC;oBACnD,QAAQ,EAAE,IAAI,CAAC,oBAAoB,EAAE;iBACtC;aACF,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;YACrD,OAAO;gBACL,OAAO,EAAE,KAAK;gBACd,KAAK,EAAE;oBACL,UAAU;oBACV,KAAK,EAAE,CAAC;oBACR,KAAK,EAAE,CAAC;oBACR,SAAS,EAAE,CAAC;oBACZ,QAAQ,EAAE,IAAI,CAAC,oBAAoB,EAAE;iBACtC;aACF,CAAC;QACJ,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,QAAQ,CAAC,GAAW;QACxB,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAClC,MAAM,WAAW,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;QAC1C,MAAM,YAAY,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QAE5C,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CAClC;;;;;;;;;;;;;+BAauB,EACvB,CAAC,OAAO,EAAE,WAAW,EAAE,YAAY,CAAC,CACrC,CAAC;YAEF,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC7B,OAAO,IAAI,CAAC;YACd,CAAC;YAED,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAC3B,MAAM,WAAW,GAAG,GAAG,CAAC,YAAY,CAAC;YACrC,MAAM,WAAW,GAAG,GAAG,CAAC,YAAY,CAAC;YACrC,MAAM,QAAQ,GAAG,GAAG,CAAC,SAAS,CAAC;YAC/B,MAAM,eAAe,GAAG,GAAG,CAAC,gBAAgB,CAAC;YAE7C,0DAA0D;YAC1D,MAAM,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,WAAW,GAAG,QAAQ,CAAC,CAAC;YACvD,MAAM,kBAAkB,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;YAE7D,qDAAqD;YACrD,IAAI,eAAe,KAAK,CAAC,IAAI,kBAAkB,GAAG,CAAC,EAAE,CAAC;gBACpD,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CACnB;;;;8CAIoC,EACpC,CAAC,OAAO,EAAE,WAAW,EAAE,kBAAkB,CAAC,CAC3C,CAAC;YACJ,CAAC;YAED,MAAM,iBAAiB,GAAG,eAAe,GAAG,CAAC,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,kBAAkB,CAAC;YACrF,MAAM,cAAc,GAAG,WAAW,GAAG,iBAAiB,CAAC;YACvD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,cAAc,GAAG,WAAW,CAAC,CAAC;YAC5D,MAAM,WAAW,GAAG,cAAc,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,WAAW,GAAG,cAAc,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAE9F,OAAO;gBACL,IAAI,EAAE,WAAW;gBACjB,UAAU,EAAE,GAAG,CAAC,WAAW;gBAC3B,YAAY,EAAE,GAAG,CAAC,aAAa;gBAC/B,YAAY,EAAE,GAAG,CAAC,aAAa;gBAC/B,WAAW,EAAE,GAAG,CAAC,YAAY;gBAC7B,SAAS,EAAE,WAAW;gBACtB,WAAW;gBACX,eAAe,EAAE,iBAAiB;gBAClC,cAAc;gBACd,SAAS;gBACT,WAAW;gBACX,QAAQ,EAAE,IAAI,CAAC,gBAAgB,EAAE,CAAC,WAAW,EAAE;aAChD,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,sBAAsB,EAAE,KAAK,CAAC,CAAC;YAC7C,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,UAAU,CAAC,GAAW;QAC1B,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;QAEvC,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;QAC5B,CAAC;QAED,MAAM,OAAO,GAAG,KAAK,CAAC,SAAS,GAAG,CAAC,CAAC;QAEpC,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;IAC5B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,iBAAiB,CAAC,GAAW;QACjC,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAElC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CAClC;;;;;;;;;+BASuB,EACvB,CAAC,OAAO,CAAC,CACV,CAAC;YAEF,IAAI,MAAM,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC7B,OAAO,IAAI,CAAC;YACd,CAAC;YAED,MAAM,GAAG,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAE3B,4DAA4D;YAC5D,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;YACvB,MAAM,SAAS,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,GAAG,CACjC,GAAG,CAAC,cAAc,EAAE,EACpB,GAAG,CAAC,WAAW,EAAE,GAAG,CAAC,EACrB,CAAC,EACD,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CACX,CAAC,CAAC;YAEH,MAAM,WAAW,GAAG,GAAG,CAAC,0BAA0B,GAAG,CAAC;gBACpD,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,UAAU,CAAC,GAAG,CAAC,iBAAiB,CAAC,GAAG,UAAU,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAC,GAAG,GAAG,CAAC;gBACpG,CAAC,CAAC,CAAC,CAAC;YAEN,OAAO;gBACL,OAAO,EAAE,GAAG,CAAC,mBAAmB;gBAChC,OAAO,EAAE,UAAU,CAAC,GAAG,CAAC,mBAAmB,CAAC;gBAC5C,KAAK,EAAE,UAAU,CAAC,GAAG,CAAC,iBAAiB,CAAC;gBACxC,aAAa,EAAE,UAAU,CAAC,GAAG,CAAC,0BAA0B,CAAC;gBACzD,UAAU,EAAE,GAAG,CAAC,mBAAmB;gBACnC,WAAW;gBACX,QAAQ,EAAE,SAAS,CAAC,WAAW,EAAE;aAClC,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,iCAAiC,EAAE,KAAK,CAAC,CAAC;YACxD,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,gBAAgB,CAAC,GAAW;QAChC,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC;QAE/C,IAAI,CAAC,IAAI,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;YAC3B,OAAO,KAAK,CAAC;QACf,CAAC;QAED,gDAAgD;QAChD,OAAO,IAAI,CAAC,OAAO,GAAG,CAAC,IAAI,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC;IAC7D,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,eAAe,CACnB,GAAW,EACX,SAAqD,EACrD,GAAY,EACZ,gBAAyB,EACzB,UAAmB;QAEnB,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QAClC,MAAM,IAAI,GAAG,iBAAiB,CAAC,SAAS,CAAC,CAAC;QAE1C,IAAI,CAAC;YACH,4BAA4B;YAC5B,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,KAAK,CACrC;;;;;;;+BAOuB,EACvB,CAAC,OAAO,CAAC,CACV,CAAC;YAEF,IAAI,SAAS,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAChC,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,CAAC;YACpD,CAAC;YAED,MAAM,EAAE,UAAU,EAAE,OAAO,EAAE,mBAAmB,EAAE,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YACvE,MAAM,cAAc,GAAG,UAAU,CAAC,mBAAmB,CAAC,CAAC;YAEvD,IAAI,cAAc,GAAG,IAAI,EAAE,CAAC;gBAC1B,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE,cAAc,EAAE,CAAC;YAC9D,CAAC;YAED,oBAAoB;YACpB,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;YACzC,IAAI,CAAC;gBACH,MAAM,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;gBAE5B,uCAAuC;gBACvC,MAAM,YAAY,GAAG,MAAM,MAAM,CAAC,KAAK,CACrC;;;;;;wCAM8B,EAC9B,CAAC,IAAI,EAAE,OAAO,CAAC,CAChB,CAAC;gBAEF,MAAM,UAAU,GAAG,UAAU,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC;gBAExE,0BAA0B;gBAC1B,MAAM,MAAM,CAAC,KAAK,CAChB;;8CAEoC,EACpC,CAAC,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,GAAG,EAAE,IAAI,EAAE,gBAAgB,EAAE,UAAU,CAAC,CAC1E,CAAC;gBAEF,MAAM,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;gBAE7B,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC;YAC7C,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,MAAM,MAAM,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;gBAC/B,MAAM,KAAK,CAAC;YACd,CAAC;oBAAS,CAAC;gBACT,MAAM,CAAC,OAAO,EAAE,CAAC;YACnB,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;YACrD,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC,EAAE,CAAC;QACjD,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,MAAM,CAAC,cAAc;QACnB,MAAM,WAAW,GAAG,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAC3D,OAAO,WAAW,WAAW,EAAE,CAAC;IAClC,CAAC;IAED;;OAEG;IACH,MAAM,CAAC,YAAY,CAAC,GAAW;QAC7B,OAAO,GAAG,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IAC9B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,KAAK;QACT,MAAM,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC;IACxB,CAAC;CACF"}
@@ -0,0 +1,60 @@
1
+ /**
2
+ * PostgreSQL-backed job queue for production deployments
3
+ * Uses same Pool pattern as pg-auth-store.ts
4
+ */
5
+ import type { Job, WebhookConfig } from './job-queue.js';
6
+ export declare class PostgresJobQueue {
7
+ private pool;
8
+ private cleanupInterval;
9
+ constructor(connectionString?: string);
10
+ /**
11
+ * Create jobs table if it doesn't exist
12
+ */
13
+ private initTable;
14
+ /**
15
+ * Create a new job
16
+ */
17
+ createJob(type: Job['type'], webhook?: WebhookConfig, ownerId?: string): Promise<Job>;
18
+ /**
19
+ * Get a job by ID
20
+ */
21
+ getJob(id: string): Promise<Job | null>;
22
+ /**
23
+ * Update a job
24
+ */
25
+ updateJob(id: string, update: Partial<Job>): Promise<void>;
26
+ /**
27
+ * Cancel a job
28
+ */
29
+ cancelJob(id: string): Promise<boolean>;
30
+ /**
31
+ * List jobs with optional filters
32
+ */
33
+ listJobs(options?: {
34
+ type?: string;
35
+ status?: string;
36
+ limit?: number;
37
+ ownerId?: string;
38
+ }): Promise<Job[]>;
39
+ /**
40
+ * Remove expired jobs (called periodically)
41
+ */
42
+ private cleanExpired;
43
+ /**
44
+ * Remove old completed/failed jobs (>7 days)
45
+ */
46
+ private cleanupOldJobs;
47
+ /**
48
+ * Map database row to Job object
49
+ */
50
+ private mapRowToJob;
51
+ /**
52
+ * Clean up interval on shutdown
53
+ */
54
+ destroy(): void;
55
+ /**
56
+ * Close the database pool
57
+ */
58
+ close(): Promise<void>;
59
+ }
60
+ //# sourceMappingURL=pg-job-queue.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pg-job-queue.d.ts","sourceRoot":"","sources":["../../src/server/pg-job-queue.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAIH,OAAO,KAAK,EAAE,GAAG,EAAE,aAAa,EAAE,MAAM,gBAAgB,CAAC;AAIzD,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,IAAI,CAAU;IACtB,OAAO,CAAC,eAAe,CAAiB;gBAE5B,gBAAgB,CAAC,EAAE,MAAM;IAiCrC;;OAEG;YACW,SAAS;IA6DvB;;OAEG;IACG,SAAS,CAAC,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC,EAAE,OAAO,CAAC,EAAE,aAAa,EAAE,OAAO,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC;IAqD3F;;OAEG;IACG,MAAM,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,GAAG,GAAG,IAAI,CAAC;IAkB7C;;OAEG;IACG,SAAS,CAAC,EAAE,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,GAAG,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC;IAgFhE;;OAEG;IACG,SAAS,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAqB7C;;OAEG;IACG,QAAQ,CAAC,OAAO,CAAC,EAAE;QACvB,IAAI,CAAC,EAAE,MAAM,CAAC;QACd,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,KAAK,CAAC,EAAE,MAAM,CAAC;QACf,OAAO,CAAC,EAAE,MAAM,CAAC;KAClB,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;IAyClB;;OAEG;YACW,YAAY;IAU1B;;OAEG;YACW,cAAc;IAgB5B;;OAEG;IACH,OAAO,CAAC,WAAW;IA4BnB;;OAEG;IACH,OAAO,IAAI,IAAI;IAIf;;OAEG;IACG,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;CAI7B"}