@prosopo/database 3.6.6 → 3.13.8

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 (57) hide show
  1. package/.turbo/turbo-build$colon$cjs.log +17 -13
  2. package/.turbo/turbo-build$colon$tsc.log +18 -15
  3. package/.turbo/turbo-build.log +18 -14
  4. package/CHANGELOG.md +612 -0
  5. package/dist/base/mongo.d.ts +1 -1
  6. package/dist/base/mongo.d.ts.map +1 -1
  7. package/dist/base/mongo.js +5 -2
  8. package/dist/base/mongo.js.map +1 -1
  9. package/dist/base/mongoMemory.d.ts +1 -1
  10. package/dist/cjs/base/mongo.cjs +6 -3
  11. package/dist/cjs/databases/captcha.cjs +2 -1
  12. package/dist/cjs/databases/centralDbStreamer.cjs +136 -0
  13. package/dist/cjs/databases/index.cjs +2 -0
  14. package/dist/cjs/databases/provider.cjs +687 -162
  15. package/dist/cjs/index.cjs +4 -0
  16. package/dist/cjs/redisCache.cjs +388 -0
  17. package/dist/databases/captcha.d.ts +1 -1
  18. package/dist/databases/captcha.d.ts.map +1 -1
  19. package/dist/databases/captcha.js +2 -1
  20. package/dist/databases/captcha.js.map +1 -1
  21. package/dist/databases/centralDbStreamer.d.ts +19 -0
  22. package/dist/databases/centralDbStreamer.d.ts.map +1 -0
  23. package/dist/databases/centralDbStreamer.js +136 -0
  24. package/dist/databases/centralDbStreamer.js.map +1 -0
  25. package/dist/databases/client.d.ts +1 -1
  26. package/dist/databases/client.d.ts.map +1 -1
  27. package/dist/databases/client.js.map +1 -1
  28. package/dist/databases/index.d.ts +1 -0
  29. package/dist/databases/index.d.ts.map +1 -1
  30. package/dist/databases/index.js +2 -0
  31. package/dist/databases/index.js.map +1 -1
  32. package/dist/databases/provider.d.ts +45 -14
  33. package/dist/databases/provider.d.ts.map +1 -1
  34. package/dist/databases/provider.js +688 -163
  35. package/dist/databases/provider.js.map +1 -1
  36. package/dist/index.d.ts +1 -0
  37. package/dist/index.d.ts.map +1 -1
  38. package/dist/index.js +5 -1
  39. package/dist/index.js.map +1 -1
  40. package/dist/redisCache.d.ts +31 -0
  41. package/dist/redisCache.d.ts.map +1 -0
  42. package/dist/redisCache.js +388 -0
  43. package/dist/redisCache.js.map +1 -0
  44. package/dist/tests/integration/ipInfoPersistence.integration.test.d.ts +2 -0
  45. package/dist/tests/integration/ipInfoPersistence.integration.test.d.ts.map +1 -0
  46. package/dist/tests/integration/ipInfoPersistence.integration.test.js +243 -0
  47. package/dist/tests/integration/ipInfoPersistence.integration.test.js.map +1 -0
  48. package/dist/tests/unit/captchaLabel.unit.test.d.ts +2 -0
  49. package/dist/tests/unit/captchaLabel.unit.test.d.ts.map +1 -0
  50. package/dist/tests/unit/captchaLabel.unit.test.js +41 -0
  51. package/dist/tests/unit/captchaLabel.unit.test.js.map +1 -0
  52. package/dist/tests/unit/databases/centralDbStreamer.unit.test.d.ts +2 -0
  53. package/dist/tests/unit/databases/centralDbStreamer.unit.test.d.ts.map +1 -0
  54. package/dist/tests/unit/databases/centralDbStreamer.unit.test.js +221 -0
  55. package/dist/tests/unit/databases/centralDbStreamer.unit.test.js.map +1 -0
  56. package/package.json +12 -9
  57. package/vite.test.config.ts +18 -0
@@ -0,0 +1,388 @@
1
+ const BIGINT_TAG = "__bigint__:";
2
+ const bigIntReplacer = (_key, value) => "bigint" === typeof value ? `${BIGINT_TAG}${value.toString()}` : value;
3
+ const bigIntReviver = (_key, value) => "string" === typeof value && value.startsWith(BIGINT_TAG) ? BigInt(value.slice(BIGINT_TAG.length)) : value;
4
+ const SESSION_KEY_PATTERNS = [
5
+ "cache:session:*",
6
+ "writeq:session:*",
7
+ "writeq:session:pending"
8
+ ];
9
+ class RedisWriteQueue {
10
+ constructor(connection, logger) {
11
+ this.flushTimer = null;
12
+ this.flushCallback = null;
13
+ this.connection = connection;
14
+ this.logger = logger;
15
+ }
16
+ async getClient() {
17
+ if (!this.connection.isReady()) {
18
+ return null;
19
+ }
20
+ try {
21
+ return await this.connection.getClient();
22
+ } catch {
23
+ return null;
24
+ }
25
+ }
26
+ // ── Session read cache ──────────────────────────────────────────────
27
+ /**
28
+ * Cache a session record in Redis for fast lookups, reducing MongoDB reads.
29
+ */
30
+ async cacheSession(sessionId, sessionData, ttlSeconds = 86400) {
31
+ const client = await this.getClient();
32
+ if (!client) {
33
+ return false;
34
+ }
35
+ try {
36
+ const key = `cache:session:${sessionId}`;
37
+ await client.set(key, JSON.stringify(sessionData, bigIntReplacer), {
38
+ EX: ttlSeconds
39
+ });
40
+ return true;
41
+ } catch (error) {
42
+ this.logger.warn(() => ({
43
+ msg: "Failed to cache session in Redis",
44
+ err: error,
45
+ sessionId
46
+ }));
47
+ return false;
48
+ }
49
+ }
50
+ /**
51
+ * Retrieve a cached session record from Redis.
52
+ */
53
+ async getCachedSession(sessionId) {
54
+ const client = await this.getClient();
55
+ if (!client) {
56
+ return null;
57
+ }
58
+ try {
59
+ const key = `cache:session:${sessionId}`;
60
+ const data = await client.get(key);
61
+ if (!data) {
62
+ return null;
63
+ }
64
+ return JSON.parse(data, bigIntReviver);
65
+ } catch (error) {
66
+ this.logger.warn(() => ({
67
+ msg: "Failed to get cached session from Redis",
68
+ err: error,
69
+ sessionId
70
+ }));
71
+ return null;
72
+ }
73
+ }
74
+ /**
75
+ * Patch the cached session in place — read the existing entry, shallow-
76
+ * merge the provided updates, and write it back. No-op on cache miss so
77
+ * the next read repopulates from Mongo. Mirrors the field-level Mongo
78
+ * patch done by `db.updateSessionRecord`; callers invoke both to keep
79
+ * Redis consistent with the DB after every in-request write.
80
+ */
81
+ async patchCachedSession(sessionId, updates, ttlSeconds = 86400) {
82
+ const existing = await this.getCachedSession(sessionId);
83
+ if (!existing) return false;
84
+ return this.cacheSession(
85
+ sessionId,
86
+ { ...existing, ...updates, lastUpdatedTimestamp: /* @__PURE__ */ new Date() },
87
+ ttlSeconds
88
+ );
89
+ }
90
+ /**
91
+ * First-hop-wins SIMD attach on the cache — mirrors the atomic Mongo
92
+ * `$ifNull` pipeline update in `db.recordSessionSimdReadingsIfAbsent`.
93
+ * Reads the cache; if it doesn't already carry `simdReadings`, merges
94
+ * the new readings + stage. No-op on cache miss or when readings are
95
+ * already present.
96
+ */
97
+ async patchCachedSimdReadingsIfAbsent(sessionId, readings, stage, ttlSeconds = 86400) {
98
+ const existing = await this.getCachedSession(sessionId);
99
+ if (!existing) return false;
100
+ if (existing.simdReadings) return false;
101
+ return this.cacheSession(
102
+ sessionId,
103
+ {
104
+ ...existing,
105
+ simdReadings: readings,
106
+ simdReadingsStage: stage,
107
+ lastUpdatedTimestamp: /* @__PURE__ */ new Date()
108
+ },
109
+ ttlSeconds
110
+ );
111
+ }
112
+ /**
113
+ * Invalidate a cached session when it's updated.
114
+ */
115
+ async invalidateCachedSession(sessionId) {
116
+ const client = await this.getClient();
117
+ if (!client) {
118
+ return;
119
+ }
120
+ try {
121
+ await client.del(`cache:session:${sessionId}`);
122
+ } catch (error) {
123
+ this.logger.warn(() => ({
124
+ msg: "Failed to invalidate cached session in Redis",
125
+ err: error,
126
+ sessionId
127
+ }));
128
+ }
129
+ }
130
+ /**
131
+ * Invalidate the hash → sessionId mapping used by /frictionless for
132
+ * dedup. Must be called alongside `invalidateCachedSession` when a
133
+ * session is consumed, otherwise the hash mapping keeps resolving to
134
+ * a dead sessionId for the remainder of its 1-hour TTL.
135
+ */
136
+ async invalidateCachedSessionByHash(userSitekeyIpHash) {
137
+ const client = await this.getClient();
138
+ if (!client) {
139
+ return;
140
+ }
141
+ try {
142
+ await client.del(`cache:session:hash:${userSitekeyIpHash}`);
143
+ } catch (error) {
144
+ this.logger.warn(() => ({
145
+ msg: "Failed to invalidate cached session hash in Redis",
146
+ err: error,
147
+ userSitekeyIpHash
148
+ }));
149
+ }
150
+ }
151
+ // ── Frictionless session deduplication cache ────────────────────────
152
+ /**
153
+ * Cache a session lookup by userSitekeyIpHash for frictionless deduplication.
154
+ * Extends the effective TTL of the cached session, reducing duplicate
155
+ * session creation under concurrent requests for the same user+site+IP.
156
+ */
157
+ async cacheSessionByHash(userSitekeyIpHash, sessionId, ttlSeconds = 3600) {
158
+ const client = await this.getClient();
159
+ if (!client) {
160
+ return false;
161
+ }
162
+ try {
163
+ const key = `cache:session:hash:${userSitekeyIpHash}`;
164
+ await client.set(key, sessionId, { EX: ttlSeconds });
165
+ return true;
166
+ } catch (error) {
167
+ this.logger.warn(() => ({
168
+ msg: "Failed to cache session hash in Redis",
169
+ err: error,
170
+ userSitekeyIpHash
171
+ }));
172
+ return false;
173
+ }
174
+ }
175
+ /**
176
+ * Get cached sessionId by userSitekeyIpHash for frictionless deduplication.
177
+ */
178
+ async getCachedSessionByHash(userSitekeyIpHash) {
179
+ const client = await this.getClient();
180
+ if (!client) {
181
+ return null;
182
+ }
183
+ try {
184
+ const key = `cache:session:hash:${userSitekeyIpHash}`;
185
+ return await client.get(key);
186
+ } catch (error) {
187
+ this.logger.warn(() => ({
188
+ msg: "Failed to get cached session hash from Redis",
189
+ err: error,
190
+ userSitekeyIpHash
191
+ }));
192
+ return null;
193
+ }
194
+ }
195
+ // ── Session write queue ─────────────────────────────────────────────
196
+ /**
197
+ * Queue a session record for batched insertion into MongoDB.
198
+ * The session data is also cached in Redis for immediate reads.
199
+ */
200
+ async queueSessionRecord(sessionId, record, ttlSeconds = 86400) {
201
+ const client = await this.getClient();
202
+ if (!client) {
203
+ return false;
204
+ }
205
+ try {
206
+ const key = `writeq:session:${sessionId}`;
207
+ const serialized = JSON.stringify(record, bigIntReplacer);
208
+ await client.set(key, serialized, { EX: ttlSeconds });
209
+ await client.sAdd("writeq:session:pending", sessionId);
210
+ await client.set(`cache:session:${sessionId}`, serialized, {
211
+ EX: ttlSeconds
212
+ });
213
+ this.triggerEarlyFlushIfNeeded();
214
+ return true;
215
+ } catch (error) {
216
+ this.logger.warn(() => ({
217
+ msg: "Failed to queue session record in Redis",
218
+ err: error,
219
+ sessionId
220
+ }));
221
+ return false;
222
+ }
223
+ }
224
+ /**
225
+ * Get all pending session records ready for batch flush.
226
+ */
227
+ async drainSessionRecords(limit = 500) {
228
+ const client = await this.getClient();
229
+ if (!client) {
230
+ return [];
231
+ }
232
+ try {
233
+ const sessionIds = await client.sMembers("writeq:session:pending");
234
+ const batch = sessionIds.slice(0, limit);
235
+ const results = [];
236
+ for (const sessionId of batch) {
237
+ const key = `writeq:session:${sessionId}`;
238
+ const data = await client.get(key);
239
+ if (data) {
240
+ results.push({
241
+ sessionId,
242
+ record: JSON.parse(data, bigIntReviver)
243
+ });
244
+ }
245
+ await client.sRem("writeq:session:pending", sessionId);
246
+ await client.del(key);
247
+ }
248
+ return results;
249
+ } catch (error) {
250
+ this.logger.warn(() => ({
251
+ msg: "Failed to drain session records from Redis",
252
+ err: error
253
+ }));
254
+ return [];
255
+ }
256
+ }
257
+ // ── Periodic flush ──────────────────────────────────────────────────
258
+ /**
259
+ * Start periodic background flush of queued records.
260
+ * The callback receives this queue instance and should drain + bulk-write records.
261
+ *
262
+ * When the queue depth exceeds `earlyFlushThreshold`, an early flush is
263
+ * triggered on the next `queueSessionRecord` call without waiting for
264
+ * the regular interval.
265
+ */
266
+ startPeriodicFlush(callback, intervalMs = 1e4, earlyFlushThreshold = 50) {
267
+ this.stopPeriodicFlush();
268
+ this.flushCallback = callback;
269
+ this.earlyFlushThreshold = earlyFlushThreshold;
270
+ this.flushTimer = setInterval(() => {
271
+ this.flushCallback?.(this).catch((error) => {
272
+ this.logger.error(() => ({
273
+ msg: "Periodic flush failed",
274
+ err: error
275
+ }));
276
+ });
277
+ }, intervalMs);
278
+ }
279
+ /**
280
+ * Trigger an early flush if the pending queue exceeds the threshold.
281
+ * Called automatically after queueing a record. This avoids large
282
+ * batch build-ups between regular interval flushes.
283
+ */
284
+ triggerEarlyFlushIfNeeded() {
285
+ if (!this.flushCallback || !this.earlyFlushThreshold) return;
286
+ this.getPendingCount().then((count) => {
287
+ if (count >= (this.earlyFlushThreshold ?? 50)) {
288
+ return this.flushCallback?.(this);
289
+ }
290
+ }).catch((error) => {
291
+ this.logger.error(() => ({
292
+ msg: "Early flush failed",
293
+ err: error
294
+ }));
295
+ });
296
+ }
297
+ /** Get the number of pending session records awaiting flush. */
298
+ async getPendingCount() {
299
+ const client = await this.getClient();
300
+ if (!client) return 0;
301
+ try {
302
+ return await client.sCard("writeq:session:pending");
303
+ } catch {
304
+ return 0;
305
+ }
306
+ }
307
+ /**
308
+ * Stop the periodic flush timer.
309
+ */
310
+ stopPeriodicFlush() {
311
+ if (this.flushTimer) {
312
+ clearInterval(this.flushTimer);
313
+ this.flushTimer = null;
314
+ }
315
+ this.flushCallback = null;
316
+ this.earlyFlushThreshold = void 0;
317
+ }
318
+ /**
319
+ * Drop every session-related key from Redis. Intended for provider
320
+ * startup so a fresh process can never inherit stale read-cache,
321
+ * dedup-by-hash, or pending-write-queue entries written by an
322
+ * earlier (possibly crashed) run.
323
+ *
324
+ * Uses SCAN — KEYS would block the Redis server on large keyspaces.
325
+ *
326
+ * Bypasses the `isReady()` fast-path that other methods use because
327
+ * this runs during startup, before the Redis client's async
328
+ * connect() handshake has had time to flip the flag. We await the
329
+ * connection promise directly with a bounded timeout so we don't
330
+ * hang the boot sequence when Redis is unreachable.
331
+ */
332
+ async clearAllSessionRecords(timeoutMs = 5e3) {
333
+ this.logger.info(() => ({
334
+ msg: "Clearing Redis session records at startup"
335
+ }));
336
+ let client;
337
+ try {
338
+ client = await Promise.race([
339
+ this.connection.getClient(),
340
+ new Promise(
341
+ (_, reject) => setTimeout(
342
+ () => reject(new Error("Redis connection timeout")),
343
+ timeoutMs
344
+ )
345
+ )
346
+ ]);
347
+ } catch (error) {
348
+ this.logger.warn(() => ({
349
+ msg: "Skipped Redis session cleanup — Redis not reachable",
350
+ err: error
351
+ }));
352
+ return;
353
+ }
354
+ try {
355
+ let totalDeleted = 0;
356
+ for (const pattern of SESSION_KEY_PATTERNS) {
357
+ for await (const key of client.scanIterator({
358
+ MATCH: pattern,
359
+ COUNT: 500
360
+ })) {
361
+ const keys = Array.isArray(key) ? key : [key];
362
+ if (keys.length > 0) {
363
+ await client.del(keys);
364
+ totalDeleted += keys.length;
365
+ }
366
+ }
367
+ }
368
+ this.logger.info(() => ({
369
+ msg: "Cleared Redis session records at startup",
370
+ data: { totalDeleted }
371
+ }));
372
+ } catch (error) {
373
+ this.logger.warn(() => ({
374
+ msg: "Failed to clear Redis session records at startup",
375
+ err: error
376
+ }));
377
+ }
378
+ }
379
+ /**
380
+ * Check if the underlying Redis connection is ready.
381
+ */
382
+ isReady() {
383
+ return this.connection.isReady();
384
+ }
385
+ }
386
+ export {
387
+ RedisWriteQueue
388
+ };
@@ -0,0 +1 @@
1
+ {"version":3,"file":"redisCache.js","sourceRoot":"","sources":["../src/redisCache.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=ipInfoPersistence.integration.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ipInfoPersistence.integration.test.d.ts","sourceRoot":"","sources":["../../../src/tests/integration/ipInfoPersistence.integration.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,243 @@
1
+ import { LogLevel, getLogger } from "@prosopo/logger";
2
+ import { CaptchaStatus, CaptchaType, IpAddressType, } from "@prosopo/types";
3
+ import { PoWCaptchaRecordSchema, PuzzleCaptchaRecordSchema, SessionRecordSchema, UserCommitmentRecordSchema, } from "@prosopo/types-database";
4
+ import { afterAll, beforeAll, describe, expect, it } from "vitest";
5
+ import { MongoMemoryDatabase } from "../../base/mongoMemory.js";
6
+ const logger = getLogger(LogLevel.enum.error, "ipInfoPersistence.test");
7
+ const validIpInfo = {
8
+ ip: "1.1.1.1",
9
+ isValid: true,
10
+ isVPN: true,
11
+ isTor: false,
12
+ isProxy: true,
13
+ isDatacenter: true,
14
+ isAbuser: true,
15
+ isMobile: false,
16
+ isSatellite: false,
17
+ isCrawler: false,
18
+ countryCode: "DE",
19
+ asnNumber: 12345,
20
+ abuserScore: 0.7,
21
+ companyAbuserScore: 0.4,
22
+ };
23
+ const errorIpInfo = {
24
+ ip: "127.0.0.1",
25
+ isValid: false,
26
+ error: "Non-routable IP address",
27
+ };
28
+ const ipv4Composite = (lower) => ({
29
+ lower,
30
+ type: IpAddressType.v4,
31
+ });
32
+ describe("captcha record ipInfo persistence (MongoMemory roundtrip)", () => {
33
+ let mongoDb;
34
+ let PoWModel;
35
+ let PuzzleModel;
36
+ let CommitmentModel;
37
+ let SessionModel;
38
+ beforeAll(async () => {
39
+ mongoDb = new MongoMemoryDatabase("ignored", "captchastorage", logger);
40
+ await mongoDb.connect();
41
+ if (!mongoDb.connection) {
42
+ throw new Error("MongoMemoryDatabase failed to provide a connection");
43
+ }
44
+ PoWModel = mongoDb.connection.model("PowCaptcha", PoWCaptchaRecordSchema);
45
+ PuzzleModel = mongoDb.connection.model("PuzzleCaptcha", PuzzleCaptchaRecordSchema);
46
+ CommitmentModel = mongoDb.connection.model("Commitment", UserCommitmentRecordSchema);
47
+ SessionModel = mongoDb.connection.model("Session", SessionRecordSchema);
48
+ });
49
+ afterAll(async () => {
50
+ await mongoDb.close();
51
+ });
52
+ it("persists the full IPInfoResponse on a PoW captcha record", async () => {
53
+ await PoWModel.create({
54
+ challenge: "1___2___challenge-pow-valid",
55
+ userAccount: "user1",
56
+ dappAccount: "dapp1",
57
+ requestedAtTimestamp: new Date(),
58
+ ipAddress: ipv4Composite(16843009n),
59
+ headers: { host: "example.com" },
60
+ ja4: "ja4-1",
61
+ result: { status: CaptchaStatus.pending },
62
+ userSubmitted: false,
63
+ serverChecked: false,
64
+ difficulty: 4,
65
+ providerSignature: "sig-1",
66
+ ipInfo: validIpInfo,
67
+ });
68
+ const got = await PoWModel.findOne({
69
+ challenge: "1___2___challenge-pow-valid",
70
+ }).lean();
71
+ expect(got).not.toBeNull();
72
+ expect(got?.ipInfo).toMatchObject(validIpInfo);
73
+ expect(got?.ipInfo?.isValid).toBe(true);
74
+ if (got?.ipInfo?.isValid) {
75
+ expect(got.ipInfo.isVPN).toBe(true);
76
+ expect(got.ipInfo.countryCode).toBe("DE");
77
+ expect(got.ipInfo.asnNumber).toBe(12345);
78
+ }
79
+ });
80
+ it("persists an IPInfoError on a PoW captcha record", async () => {
81
+ await PoWModel.create({
82
+ challenge: "1___2___challenge-pow-error",
83
+ userAccount: "user2",
84
+ dappAccount: "dapp1",
85
+ requestedAtTimestamp: new Date(),
86
+ ipAddress: ipv4Composite(2130706433n),
87
+ headers: { host: "example.com" },
88
+ ja4: "ja4-2",
89
+ result: { status: CaptchaStatus.pending },
90
+ userSubmitted: false,
91
+ serverChecked: false,
92
+ difficulty: 4,
93
+ providerSignature: "sig-2",
94
+ ipInfo: errorIpInfo,
95
+ });
96
+ const got = await PoWModel.findOne({
97
+ challenge: "1___2___challenge-pow-error",
98
+ }).lean();
99
+ expect(got?.ipInfo).toMatchObject(errorIpInfo);
100
+ expect(got?.ipInfo?.isValid).toBe(false);
101
+ });
102
+ it("persists the full IPInfoResponse on a Puzzle captcha record", async () => {
103
+ await PuzzleModel.create({
104
+ challenge: "1___2___challenge-puzzle-valid",
105
+ userAccount: "user3",
106
+ dappAccount: "dapp1",
107
+ requestedAtTimestamp: new Date(),
108
+ ipAddress: ipv4Composite(16843009n),
109
+ headers: { host: "example.com" },
110
+ ja4: "ja4-3",
111
+ result: { status: CaptchaStatus.pending },
112
+ userSubmitted: false,
113
+ serverChecked: false,
114
+ targetX: 100,
115
+ targetY: 100,
116
+ originX: 0,
117
+ originY: 0,
118
+ tolerance: 10,
119
+ providerSignature: "sig-3",
120
+ ipInfo: validIpInfo,
121
+ });
122
+ const got = await PuzzleModel.findOne({
123
+ challenge: "1___2___challenge-puzzle-valid",
124
+ }).lean();
125
+ expect(got?.ipInfo).toMatchObject(validIpInfo);
126
+ });
127
+ it("persists the full IPInfoResponse on a UserCommitment record", async () => {
128
+ await CommitmentModel.create({
129
+ id: "commitment-valid",
130
+ userAccount: "user4",
131
+ dappAccount: "dapp1",
132
+ providerAccount: "provider1",
133
+ datasetId: "dataset1",
134
+ result: { status: CaptchaStatus.pending },
135
+ userSignature: "sig",
136
+ ipAddress: ipv4Composite(16843009n),
137
+ headers: { host: "example.com" },
138
+ ja4: "ja4-4",
139
+ userSubmitted: true,
140
+ serverChecked: false,
141
+ requestedAtTimestamp: new Date(),
142
+ pending: false,
143
+ salt: "salt",
144
+ requestHash: "0xabcdef",
145
+ deadlineTimestamp: new Date(Date.now() + 60_000),
146
+ threshold: 0.5,
147
+ ipInfo: validIpInfo,
148
+ });
149
+ const got = await CommitmentModel.findOne({
150
+ id: "commitment-valid",
151
+ }).lean();
152
+ expect(got?.ipInfo).toMatchObject(validIpInfo);
153
+ });
154
+ it("persists the full IPInfoResponse on a Session record", async () => {
155
+ await SessionModel.create({
156
+ sessionId: "session-valid",
157
+ createdAt: new Date(),
158
+ token: "tok-1",
159
+ score: 0.5,
160
+ threshold: 0.5,
161
+ scoreComponents: { baseScore: 0.5 },
162
+ providerSelectEntropy: 0.1,
163
+ ipAddress: ipv4Composite(16843009n),
164
+ captchaType: CaptchaType.frictionless,
165
+ webView: false,
166
+ iFrame: false,
167
+ decryptedHeadHash: "",
168
+ siteKey: "dapp1",
169
+ ipInfo: validIpInfo,
170
+ });
171
+ const got = await SessionModel.findOne({
172
+ sessionId: "session-valid",
173
+ }).lean();
174
+ expect(got).not.toBeNull();
175
+ expect(got?.ipInfo).toMatchObject(validIpInfo);
176
+ expect(got?.ipInfo?.isValid).toBe(true);
177
+ if (got?.ipInfo?.isValid) {
178
+ expect(got.ipInfo.countryCode).toBe("DE");
179
+ expect(got.ipInfo.isVPN).toBe(true);
180
+ }
181
+ });
182
+ it("persists an IPInfoError on a Session record", async () => {
183
+ await SessionModel.create({
184
+ sessionId: "session-error",
185
+ createdAt: new Date(),
186
+ token: "tok-2",
187
+ score: 0.5,
188
+ threshold: 0.5,
189
+ scoreComponents: { baseScore: 0.5 },
190
+ providerSelectEntropy: 0.1,
191
+ ipAddress: ipv4Composite(2130706433n),
192
+ captchaType: CaptchaType.frictionless,
193
+ webView: false,
194
+ iFrame: false,
195
+ decryptedHeadHash: "",
196
+ siteKey: "dapp1",
197
+ ipInfo: errorIpInfo,
198
+ });
199
+ const got = await SessionModel.findOne({
200
+ sessionId: "session-error",
201
+ }).lean();
202
+ expect(got?.ipInfo).toMatchObject(errorIpInfo);
203
+ expect(got?.ipInfo?.isValid).toBe(false);
204
+ });
205
+ it("backfill query { ipInfo: { $exists: false } } matches records missing ipInfo", async () => {
206
+ await PoWModel.create({
207
+ challenge: "1___2___challenge-pow-with-info",
208
+ userAccount: "user5",
209
+ dappAccount: "dapp1",
210
+ requestedAtTimestamp: new Date(),
211
+ ipAddress: ipv4Composite(16843010n),
212
+ headers: { host: "example.com" },
213
+ ja4: "ja4-5",
214
+ result: { status: CaptchaStatus.pending },
215
+ userSubmitted: false,
216
+ serverChecked: false,
217
+ difficulty: 4,
218
+ providerSignature: "sig-5",
219
+ ipInfo: validIpInfo,
220
+ });
221
+ await PoWModel.create({
222
+ challenge: "1___2___challenge-pow-no-info",
223
+ userAccount: "user6",
224
+ dappAccount: "dapp1",
225
+ requestedAtTimestamp: new Date(),
226
+ ipAddress: ipv4Composite(16843011n),
227
+ headers: { host: "example.com" },
228
+ ja4: "ja4-6",
229
+ result: { status: CaptchaStatus.pending },
230
+ userSubmitted: false,
231
+ serverChecked: false,
232
+ difficulty: 4,
233
+ providerSignature: "sig-6",
234
+ });
235
+ const missing = await PoWModel.find({
236
+ ipInfo: { $exists: false },
237
+ }).lean();
238
+ const challenges = missing.map((r) => r.challenge);
239
+ expect(challenges).toContain("1___2___challenge-pow-no-info");
240
+ expect(challenges).not.toContain("1___2___challenge-pow-with-info");
241
+ });
242
+ });
243
+ //# sourceMappingURL=ipInfoPersistence.integration.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ipInfoPersistence.integration.test.js","sourceRoot":"","sources":["../../../src/tests/integration/ipInfoPersistence.integration.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=captchaLabel.unit.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"captchaLabel.unit.test.d.ts","sourceRoot":"","sources":["../../../src/tests/unit/captchaLabel.unit.test.ts"],"names":[],"mappings":""}