@periodic/vanadium 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +37 -0
- package/LICENSE +21 -0
- package/README.md +846 -0
- package/dist/cjs/adapters/memory/index.js +134 -0
- package/dist/cjs/adapters/memory/index.js.map +1 -0
- package/dist/cjs/adapters/mongodb/index.js +189 -0
- package/dist/cjs/adapters/mongodb/index.js.map +1 -0
- package/dist/cjs/adapters/mongoose/index.js +199 -0
- package/dist/cjs/adapters/mongoose/index.js.map +1 -0
- package/dist/cjs/adapters/postgres/index.js +202 -0
- package/dist/cjs/adapters/postgres/index.js.map +1 -0
- package/dist/cjs/adapters/prisma/index.js +176 -0
- package/dist/cjs/adapters/prisma/index.js.map +1 -0
- package/dist/cjs/adapters/redis/index.js +178 -0
- package/dist/cjs/adapters/redis/index.js.map +1 -0
- package/dist/cjs/cleanup/engine.js +100 -0
- package/dist/cjs/cleanup/engine.js.map +1 -0
- package/dist/cjs/core/concurrencyGuard.js +50 -0
- package/dist/cjs/core/concurrencyGuard.js.map +1 -0
- package/dist/cjs/core/metrics.js +39 -0
- package/dist/cjs/core/metrics.js.map +1 -0
- package/dist/cjs/core/stateMachine.js +46 -0
- package/dist/cjs/core/stateMachine.js.map +1 -0
- package/dist/cjs/errors/index.js +127 -0
- package/dist/cjs/errors/index.js.map +1 -0
- package/dist/cjs/http/express.js +84 -0
- package/dist/cjs/http/express.js.map +1 -0
- package/dist/cjs/http/fastify.js +70 -0
- package/dist/cjs/http/fastify.js.map +1 -0
- package/dist/cjs/idempotency/engine.js +266 -0
- package/dist/cjs/idempotency/engine.js.map +1 -0
- package/dist/cjs/index.js +19 -0
- package/dist/cjs/index.js.map +1 -0
- package/dist/cjs/lock/engine.js +187 -0
- package/dist/cjs/lock/engine.js.map +1 -0
- package/dist/cjs/observability/metrics.js +92 -0
- package/dist/cjs/observability/metrics.js.map +1 -0
- package/dist/cjs/resilience/circuitBreaker.js +129 -0
- package/dist/cjs/resilience/circuitBreaker.js.map +1 -0
- package/dist/cjs/types/index.js +13 -0
- package/dist/cjs/types/index.js.map +1 -0
- package/dist/cjs/utils/crypto.js +64 -0
- package/dist/cjs/utils/crypto.js.map +1 -0
- package/dist/cjs/utils/keys.js +40 -0
- package/dist/cjs/utils/keys.js.map +1 -0
- package/dist/cjs/utils/sleep.js +25 -0
- package/dist/cjs/utils/sleep.js.map +1 -0
- package/dist/esm/adapters/memory/index.js +129 -0
- package/dist/esm/adapters/memory/index.js.map +1 -0
- package/dist/esm/adapters/mongodb/index.js +184 -0
- package/dist/esm/adapters/mongodb/index.js.map +1 -0
- package/dist/esm/adapters/mongoose/index.js +193 -0
- package/dist/esm/adapters/mongoose/index.js.map +1 -0
- package/dist/esm/adapters/postgres/index.js +197 -0
- package/dist/esm/adapters/postgres/index.js.map +1 -0
- package/dist/esm/adapters/prisma/index.js +171 -0
- package/dist/esm/adapters/prisma/index.js.map +1 -0
- package/dist/esm/adapters/redis/index.js +173 -0
- package/dist/esm/adapters/redis/index.js.map +1 -0
- package/dist/esm/cleanup/engine.js +95 -0
- package/dist/esm/cleanup/engine.js.map +1 -0
- package/dist/esm/core/concurrencyGuard.js +46 -0
- package/dist/esm/core/concurrencyGuard.js.map +1 -0
- package/dist/esm/core/metrics.js +35 -0
- package/dist/esm/core/metrics.js.map +1 -0
- package/dist/esm/core/stateMachine.js +40 -0
- package/dist/esm/core/stateMachine.js.map +1 -0
- package/dist/esm/errors/index.js +114 -0
- package/dist/esm/errors/index.js.map +1 -0
- package/dist/esm/http/express.js +81 -0
- package/dist/esm/http/express.js.map +1 -0
- package/dist/esm/http/fastify.js +67 -0
- package/dist/esm/http/fastify.js.map +1 -0
- package/dist/esm/idempotency/engine.js +261 -0
- package/dist/esm/idempotency/engine.js.map +1 -0
- package/dist/esm/index.js +9 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/lock/engine.js +182 -0
- package/dist/esm/lock/engine.js.map +1 -0
- package/dist/esm/observability/metrics.js +89 -0
- package/dist/esm/observability/metrics.js.map +1 -0
- package/dist/esm/resilience/circuitBreaker.js +124 -0
- package/dist/esm/resilience/circuitBreaker.js.map +1 -0
- package/dist/esm/types/index.js +10 -0
- package/dist/esm/types/index.js.map +1 -0
- package/dist/esm/utils/crypto.js +58 -0
- package/dist/esm/utils/crypto.js.map +1 -0
- package/dist/esm/utils/keys.js +35 -0
- package/dist/esm/utils/keys.js.map +1 -0
- package/dist/esm/utils/sleep.js +20 -0
- package/dist/esm/utils/sleep.js.map +1 -0
- package/dist/types/adapters/memory/index.d.ts +49 -0
- package/dist/types/adapters/memory/index.d.ts.map +1 -0
- package/dist/types/adapters/mongodb/index.d.ts +97 -0
- package/dist/types/adapters/mongodb/index.d.ts.map +1 -0
- package/dist/types/adapters/mongoose/index.d.ts +107 -0
- package/dist/types/adapters/mongoose/index.d.ts.map +1 -0
- package/dist/types/adapters/postgres/index.d.ts +85 -0
- package/dist/types/adapters/postgres/index.d.ts.map +1 -0
- package/dist/types/adapters/prisma/index.d.ts +73 -0
- package/dist/types/adapters/prisma/index.d.ts.map +1 -0
- package/dist/types/adapters/redis/index.d.ts +77 -0
- package/dist/types/adapters/redis/index.d.ts.map +1 -0
- package/dist/types/cleanup/engine.d.ts +41 -0
- package/dist/types/cleanup/engine.d.ts.map +1 -0
- package/dist/types/core/concurrencyGuard.d.ts +28 -0
- package/dist/types/core/concurrencyGuard.d.ts.map +1 -0
- package/dist/types/core/metrics.d.ts +13 -0
- package/dist/types/core/metrics.d.ts.map +1 -0
- package/dist/types/core/stateMachine.d.ts +20 -0
- package/dist/types/core/stateMachine.d.ts.map +1 -0
- package/dist/types/errors/index.d.ts +32 -0
- package/dist/types/errors/index.d.ts.map +1 -0
- package/dist/types/http/express.d.ts +50 -0
- package/dist/types/http/express.d.ts.map +1 -0
- package/dist/types/http/fastify.d.ts +48 -0
- package/dist/types/http/fastify.d.ts.map +1 -0
- package/dist/types/idempotency/engine.d.ts +24 -0
- package/dist/types/idempotency/engine.d.ts.map +1 -0
- package/dist/types/index.d.ts +8 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/lock/engine.d.ts +28 -0
- package/dist/types/lock/engine.d.ts.map +1 -0
- package/dist/types/observability/metrics.d.ts +45 -0
- package/dist/types/observability/metrics.d.ts.map +1 -0
- package/dist/types/resilience/circuitBreaker.d.ts +48 -0
- package/dist/types/resilience/circuitBreaker.d.ts.map +1 -0
- package/dist/types/types/index.d.ts +170 -0
- package/dist/types/types/index.d.ts.map +1 -0
- package/dist/types/utils/crypto.d.ts +20 -0
- package/dist/types/utils/crypto.d.ts.map +1 -0
- package/dist/types/utils/keys.d.ts +15 -0
- package/dist/types/utils/keys.d.ts.map +1 -0
- package/dist/types/utils/sleep.d.ts +13 -0
- package/dist/types/utils/sleep.d.ts.map +1 -0
- package/package.json +140 -0
|
@@ -0,0 +1,134 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.MemoryAdapter = void 0;
|
|
4
|
+
exports.createMemoryAdapter = createMemoryAdapter;
|
|
5
|
+
const index_js_1 = require("../../errors/index.js");
|
|
6
|
+
// ─── Memory Storage Adapter ───────────────────────────────────────────────────
|
|
7
|
+
/**
|
|
8
|
+
* Built-in in-memory storage adapter.
|
|
9
|
+
*
|
|
10
|
+
* Features:
|
|
11
|
+
* - O(1) key lookup via Map
|
|
12
|
+
* - Automatic TTL eviction on read
|
|
13
|
+
* - Optional maxKeys with LRU eviction to prevent unbounded growth
|
|
14
|
+
* - No global state — each instance is isolated
|
|
15
|
+
*
|
|
16
|
+
* @example
|
|
17
|
+
* ```ts
|
|
18
|
+
* import { createMemoryAdapter } from '@periodic/vanadium';
|
|
19
|
+
* const adapter = createMemoryAdapter({ maxKeys: 10_000 });
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
class MemoryAdapter {
|
|
23
|
+
constructor(options = {}) {
|
|
24
|
+
this.name = 'memory';
|
|
25
|
+
this.store = new Map();
|
|
26
|
+
this.maxKeys = options.maxKeys ?? Infinity;
|
|
27
|
+
this.clock = options.clock ?? Date.now;
|
|
28
|
+
}
|
|
29
|
+
async get(key) {
|
|
30
|
+
const entry = this.store.get(key);
|
|
31
|
+
if (!entry)
|
|
32
|
+
return null;
|
|
33
|
+
const now = this.clock();
|
|
34
|
+
// Return expired IN_PROGRESS records so callers can attempt takeover
|
|
35
|
+
if (entry.expiresAt !== undefined && now >= entry.expiresAt) {
|
|
36
|
+
if (entry.record.status === 'IN_PROGRESS') {
|
|
37
|
+
entry.lastAccessed = now;
|
|
38
|
+
return { ...entry.record };
|
|
39
|
+
}
|
|
40
|
+
this.store.delete(key);
|
|
41
|
+
return null;
|
|
42
|
+
}
|
|
43
|
+
entry.lastAccessed = now;
|
|
44
|
+
return { ...entry.record };
|
|
45
|
+
}
|
|
46
|
+
async set(key, value, ttlMs) {
|
|
47
|
+
const now = this.clock();
|
|
48
|
+
const expiresAt = ttlMs !== undefined ? now + ttlMs : undefined;
|
|
49
|
+
// Evict LRU entries if at capacity
|
|
50
|
+
if (!this.store.has(key) && this.store.size >= this.maxKeys) {
|
|
51
|
+
this._evictLRU();
|
|
52
|
+
}
|
|
53
|
+
this.store.set(key, {
|
|
54
|
+
record: { ...value },
|
|
55
|
+
expiresAt,
|
|
56
|
+
lastAccessed: now,
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
async delete(key) {
|
|
60
|
+
this.store.delete(key);
|
|
61
|
+
}
|
|
62
|
+
async compareAndSet(key, expectedOwnerToken, newValue, ttlMs) {
|
|
63
|
+
const entry = this.store.get(key);
|
|
64
|
+
if (!entry)
|
|
65
|
+
return false;
|
|
66
|
+
if (entry.record.ownerToken !== expectedOwnerToken)
|
|
67
|
+
return false;
|
|
68
|
+
const now = this.clock();
|
|
69
|
+
const expiresAt = ttlMs !== undefined ? now + ttlMs : entry.expiresAt;
|
|
70
|
+
this.store.set(key, {
|
|
71
|
+
record: { ...newValue },
|
|
72
|
+
expiresAt,
|
|
73
|
+
lastAccessed: now,
|
|
74
|
+
});
|
|
75
|
+
return true;
|
|
76
|
+
}
|
|
77
|
+
// ── Maintenance ───────────────────────────────────────────────────────────
|
|
78
|
+
async setIfAbsent(key, value, ttlMs) {
|
|
79
|
+
const now = this.clock();
|
|
80
|
+
const existing = this.store.get(key);
|
|
81
|
+
if (existing !== undefined) {
|
|
82
|
+
if (existing.expiresAt === undefined || now < existing.expiresAt) {
|
|
83
|
+
return false;
|
|
84
|
+
}
|
|
85
|
+
this.store.delete(key);
|
|
86
|
+
}
|
|
87
|
+
const expiresAt = ttlMs !== undefined ? now + ttlMs : undefined;
|
|
88
|
+
this.store.set(key, {
|
|
89
|
+
record: { ...value },
|
|
90
|
+
expiresAt,
|
|
91
|
+
lastAccessed: now,
|
|
92
|
+
});
|
|
93
|
+
return true;
|
|
94
|
+
}
|
|
95
|
+
/** Evict the single least-recently-accessed entry. */
|
|
96
|
+
_evictLRU() {
|
|
97
|
+
let oldestKey;
|
|
98
|
+
let oldestTime = Infinity;
|
|
99
|
+
for (const [key, entry] of this.store) {
|
|
100
|
+
if (entry.lastAccessed < oldestTime) {
|
|
101
|
+
oldestTime = entry.lastAccessed;
|
|
102
|
+
oldestKey = key;
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
if (oldestKey !== undefined) {
|
|
106
|
+
this.store.delete(oldestKey);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
/** Return current number of stored keys. */
|
|
110
|
+
size() {
|
|
111
|
+
return this.store.size;
|
|
112
|
+
}
|
|
113
|
+
/** Clear all stored entries. */
|
|
114
|
+
clear() {
|
|
115
|
+
this.store.clear();
|
|
116
|
+
}
|
|
117
|
+
/** Expose adapter capabilities. */
|
|
118
|
+
capabilities() {
|
|
119
|
+
return {
|
|
120
|
+
transactions: false,
|
|
121
|
+
cas: true,
|
|
122
|
+
ttl: true,
|
|
123
|
+
advisoryLocks: false,
|
|
124
|
+
};
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
exports.MemoryAdapter = MemoryAdapter;
|
|
128
|
+
// ─── Factory ──────────────────────────────────────────────────────────────────
|
|
129
|
+
function createMemoryAdapter(options) {
|
|
130
|
+
return new MemoryAdapter(options);
|
|
131
|
+
}
|
|
132
|
+
// Suppress unused import warning — this file intentionally imports for type safety
|
|
133
|
+
void index_js_1.createStorageError;
|
|
134
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/adapters/memory/index.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,189 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* MongoDB Storage Adapter for @periodic/vanadium
|
|
4
|
+
*
|
|
5
|
+
* Peer dependency: "mongodb" >= 5.0.0
|
|
6
|
+
* Import: import { createMongoAdapter } from '@periodic/vanadium/adapters/mongodb'
|
|
7
|
+
*
|
|
8
|
+
* Uses the official MongoDB driver (NOT Mongoose).
|
|
9
|
+
* The caller is responsible for providing a connected MongoClient.
|
|
10
|
+
*
|
|
11
|
+
* Required indexes (run once):
|
|
12
|
+
* ```js
|
|
13
|
+
* db.collection('vanadium_records').createIndex({ expiresAt: 1 });
|
|
14
|
+
* db.collection('vanadium_records').createIndex({ status: 1, expiresAt: 1 });
|
|
15
|
+
* ```
|
|
16
|
+
*/
|
|
17
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
18
|
+
exports.MongoAdapter = void 0;
|
|
19
|
+
exports.createMongoAdapter = createMongoAdapter;
|
|
20
|
+
const index_js_1 = require("../../errors/index.js");
|
|
21
|
+
/* eslint-enable @typescript-eslint/no-explicit-any */
|
|
22
|
+
const MONGO_DUPLICATE_KEY_CODE = 11000;
|
|
23
|
+
// ─── MongoDB Adapter ──────────────────────────────────────────────────────────
|
|
24
|
+
class MongoAdapter {
|
|
25
|
+
constructor(options) {
|
|
26
|
+
this.name = 'mongodb';
|
|
27
|
+
if (!options.client || !options.dbName) {
|
|
28
|
+
throw (0, index_js_1.createConfigurationError)('', 'mongodb', 'MongoDB adapter requires a client and dbName.');
|
|
29
|
+
}
|
|
30
|
+
this.db = options.client.db(options.dbName);
|
|
31
|
+
this.collectionName = options.collectionName ?? 'vanadium_records';
|
|
32
|
+
this.useTransactions = options.useTransactions ?? true;
|
|
33
|
+
this.clock = options.clock ?? Date.now;
|
|
34
|
+
}
|
|
35
|
+
get collection() {
|
|
36
|
+
return this.db.collection(this.collectionName);
|
|
37
|
+
}
|
|
38
|
+
// ── Storage Interface ─────────────────────────────────────────────────────
|
|
39
|
+
async get(key) {
|
|
40
|
+
try {
|
|
41
|
+
const doc = (await this.collection.findOne({ _id: key }));
|
|
42
|
+
if (!doc)
|
|
43
|
+
return null;
|
|
44
|
+
return this._docToRecord(doc);
|
|
45
|
+
}
|
|
46
|
+
catch (err) {
|
|
47
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
async set(key, value, ttlMs) {
|
|
51
|
+
const now = this.clock();
|
|
52
|
+
const expiresAt = ttlMs !== undefined ? now + ttlMs : value.expiresAt;
|
|
53
|
+
const doc = {
|
|
54
|
+
_id: key,
|
|
55
|
+
status: value.status,
|
|
56
|
+
result: value.result,
|
|
57
|
+
payloadHash: value.payloadHash,
|
|
58
|
+
ownerToken: value.ownerToken,
|
|
59
|
+
attempts: value.attempts,
|
|
60
|
+
createdAt: value.createdAt,
|
|
61
|
+
updatedAt: value.updatedAt,
|
|
62
|
+
expiresAt,
|
|
63
|
+
};
|
|
64
|
+
try {
|
|
65
|
+
// Try atomic insert first (first-writer-wins)
|
|
66
|
+
await this.collection.insertOne(doc);
|
|
67
|
+
}
|
|
68
|
+
catch (err) {
|
|
69
|
+
const code = err.code;
|
|
70
|
+
if (code !== MONGO_DUPLICATE_KEY_CODE) {
|
|
71
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
72
|
+
}
|
|
73
|
+
// Key exists — update it
|
|
74
|
+
try {
|
|
75
|
+
await this.collection.findOneAndUpdate({ _id: key }, {
|
|
76
|
+
$set: {
|
|
77
|
+
status: value.status,
|
|
78
|
+
result: value.result,
|
|
79
|
+
payloadHash: value.payloadHash,
|
|
80
|
+
ownerToken: value.ownerToken,
|
|
81
|
+
attempts: value.attempts,
|
|
82
|
+
updatedAt: value.updatedAt,
|
|
83
|
+
expiresAt,
|
|
84
|
+
},
|
|
85
|
+
}, { returnDocument: 'after' });
|
|
86
|
+
}
|
|
87
|
+
catch (updateErr) {
|
|
88
|
+
throw (0, index_js_1.createStorageError)(key, this.name, updateErr, this.clock);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
async delete(key) {
|
|
93
|
+
try {
|
|
94
|
+
await this.collection.deleteOne({ _id: key });
|
|
95
|
+
}
|
|
96
|
+
catch (err) {
|
|
97
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
async compareAndSet(key, expectedOwnerToken, newValue, ttlMs) {
|
|
101
|
+
const now = this.clock();
|
|
102
|
+
const expiresAt = ttlMs !== undefined ? now + ttlMs : newValue.expiresAt;
|
|
103
|
+
try {
|
|
104
|
+
const result = await this.collection.findOneAndUpdate({ _id: key, ownerToken: expectedOwnerToken }, {
|
|
105
|
+
$set: {
|
|
106
|
+
status: newValue.status,
|
|
107
|
+
result: newValue.result,
|
|
108
|
+
payloadHash: newValue.payloadHash,
|
|
109
|
+
ownerToken: newValue.ownerToken,
|
|
110
|
+
attempts: newValue.attempts,
|
|
111
|
+
updatedAt: newValue.updatedAt,
|
|
112
|
+
expiresAt,
|
|
113
|
+
},
|
|
114
|
+
}, { returnDocument: 'after' });
|
|
115
|
+
return result !== null;
|
|
116
|
+
}
|
|
117
|
+
catch (err) {
|
|
118
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Atomic expired takeover using conditional findOneAndUpdate.
|
|
123
|
+
* Guarantees only one concurrent caller succeeds.
|
|
124
|
+
*/
|
|
125
|
+
async takeoverExpired(key, newOwnerToken, ttlMs) {
|
|
126
|
+
const now = this.clock();
|
|
127
|
+
try {
|
|
128
|
+
const result = (await this.collection.findOneAndUpdate({
|
|
129
|
+
_id: key,
|
|
130
|
+
status: 'IN_PROGRESS',
|
|
131
|
+
expiresAt: { $lt: now },
|
|
132
|
+
}, {
|
|
133
|
+
$set: {
|
|
134
|
+
ownerToken: newOwnerToken,
|
|
135
|
+
updatedAt: now,
|
|
136
|
+
expiresAt: now + ttlMs,
|
|
137
|
+
},
|
|
138
|
+
$inc: { attempts: 1 },
|
|
139
|
+
}, { returnDocument: 'after' }));
|
|
140
|
+
return result;
|
|
141
|
+
}
|
|
142
|
+
catch (err) {
|
|
143
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
// ── Capabilities ──────────────────────────────────────────────────────────
|
|
147
|
+
capabilities() {
|
|
148
|
+
return {
|
|
149
|
+
transactions: this.useTransactions,
|
|
150
|
+
cas: true,
|
|
151
|
+
ttl: true, // via TTL index
|
|
152
|
+
advisoryLocks: false,
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
// ── Private ───────────────────────────────────────────────────────────────
|
|
156
|
+
_docToRecord(doc) {
|
|
157
|
+
return {
|
|
158
|
+
key: doc._id,
|
|
159
|
+
status: doc.status,
|
|
160
|
+
result: doc.result,
|
|
161
|
+
payloadHash: doc.payloadHash,
|
|
162
|
+
ownerToken: doc.ownerToken,
|
|
163
|
+
attempts: doc.attempts,
|
|
164
|
+
createdAt: doc.createdAt,
|
|
165
|
+
updatedAt: doc.updatedAt,
|
|
166
|
+
expiresAt: doc.expiresAt,
|
|
167
|
+
};
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
exports.MongoAdapter = MongoAdapter;
|
|
171
|
+
// ─── Factory ──────────────────────────────────────────────────────────────────
|
|
172
|
+
/**
|
|
173
|
+
* Create a MongoDB storage adapter.
|
|
174
|
+
*
|
|
175
|
+
* @example
|
|
176
|
+
* ```ts
|
|
177
|
+
* import { MongoClient } from 'mongodb';
|
|
178
|
+
* import { createMongoAdapter } from '@periodic/vanadium/adapters/mongodb';
|
|
179
|
+
*
|
|
180
|
+
* const client = new MongoClient(process.env.MONGO_URL!);
|
|
181
|
+
* await client.connect();
|
|
182
|
+
*
|
|
183
|
+
* const adapter = createMongoAdapter({ client, dbName: 'myapp' });
|
|
184
|
+
* ```
|
|
185
|
+
*/
|
|
186
|
+
function createMongoAdapter(options) {
|
|
187
|
+
return new MongoAdapter(options);
|
|
188
|
+
}
|
|
189
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/adapters/mongodb/index.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;GAcG;;;AAuQH,gDAEC;AAtQD,oDAAqF;AAqBrF,sDAAsD;AAEtD,MAAM,wBAAwB,GAAG,KAAK,CAAC;AAkCvC,iFAAiF;AAEjF,MAAa,YAAY;IAQvB,YAAY,OAA4B;QAPxB,SAAI,GAAG,SAAS,CAAC;QAQ/B,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;YACvC,MAAM,IAAA,mCAAwB,EAC5B,EAAE,EACF,SAAS,EACT,+CAA+C,CAChD,CAAC;QACJ,CAAC;QACD,IAAI,CAAC,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QAC5C,IAAI,CAAC,cAAc,GAAG,OAAO,CAAC,cAAc,IAAI,kBAAkB,CAAC;QACnE,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC,eAAe,IAAI,IAAI,CAAC;QACvD,IAAI,CAAC,KAAK,GAAG,OAAO,CAAC,KAAK,IAAI,IAAI,CAAC,GAAG,CAAC;IACzC,CAAC;IAED,IAAY,UAAU;QACpB,OAAO,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;IACjD,CAAC;IAED,6EAA6E;IAE7E,KAAK,CAAC,GAAG,CAAc,GAAW;QAChC,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,CAAC,MAAM,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAA4B,CAAC;YACrF,IAAI,CAAC,GAAG;gBAAE,OAAO,IAAI,CAAC;YACtB,OAAO,IAAI,CAAC,YAAY,CAAI,GAAG,CAAC,CAAC;QACnC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,IAAA,6BAAkB,EAAC,GAAG,EAAE,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC;IAED,KAAK,CAAC,GAAG,CAAc,GAAW,EAAE,KAAsB,EAAE,KAAc;QACxE,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;QACzB,MAAM,SAAS,GAAG,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,GAAG,GAAG,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC;QAEtE,MAAM,GAAG,GAAqB;YAC5B,GAAG,EAAE,GAAG;YACR,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,SAAS,EAAE,KAAK,CAAC,SAAS;YAC1B,SAAS,EAAE,KAAK,CAAC,SAAS;YAC1B,SAAS;SACV,CAAC;QAEF,IAAI,CAAC;YACH,8CAA8C;YAC9C,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC;QACvC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,IAAI,GAAI,GAAyB,CAAC,IAAI,CAAC;YAC7C,IAAI,IAAI,KAAK,wBAAwB,EAAE,CAAC;gBACtC,MAAM,IAAA,6BAAkB,EAAC,GAAG,EAAE,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;YAC5D,CAAC;YACD,yBAAyB;YACzB,IAAI,CAAC;gBACH,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,CACpC,EAAE,GAAG,EAAE,GAAG,EAAE,EACZ;oBACE,IAAI,EAAE;wBACJ,MAAM,EAAE,KAAK,CAAC,MAAM;wBACpB,MAAM,EAAE,KAAK,CAAC,MAAM;wBACpB,WAAW,EAAE,KAAK,CAAC,WAAW;wBAC9B,UAAU,EAAE,KAAK,CAAC,UAAU;wBAC5B,QAAQ,EAAE,KAAK,CAAC,QAAQ;wBACxB,SAAS,EAAE,KAAK,CAAC,SAAS;wBAC1B,SAAS;qBACV;iBACF,EACD,EAAE,cAAc,EAAE,OAAO,EAAE,CAC5B,CAAC;YACJ,CAAC;YAAC,OAAO,SAAS,EAAE,CAAC;gBACnB,MAAM,IAAA,6BAAkB,EAAC,GAAG,EAAE,IAAI,CAAC,IAAI,EAAE,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;YAClE,CAAC;QACH,CAAC;IACH,CAAC;IAED,KAAK,CAAC,MAAM,CAAC,GAAW;QACtB,IAAI,CAAC;YACH,MAAM,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,CAAC,CAAC;QAChD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,IAAA,6BAAkB,EAAC,GAAG,EAAE,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC;IAED,KAAK,CAAC,aAAa,CACjB,GAAW,EACX,kBAA0B,EAC1B,QAAyB,EACzB,KAAc;QAEd,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;QACzB,MAAM,SAAS,GAAG,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,GAAG,GAAG,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC,SAAS,CAAC;QAEzE,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,CACnD,EAAE,GAAG,EAAE,GAAG,EAAE,UAAU,EAAE,kBAAkB,EAAE,EAC5C;gBACE,IAAI,EAAE;oBACJ,MAAM,EAAE,QAAQ,CAAC,MAAM;oBACvB,MAAM,EAAE,QAAQ,CAAC,MAAM;oBACvB,WAAW,EAAE,QAAQ,CAAC,WAAW;oBACjC,UAAU,EAAE,QAAQ,CAAC,UAAU;oBAC/B,QAAQ,EAAE,QAAQ,CAAC,QAAQ;oBAC3B,SAAS,EAAE,QAAQ,CAAC,SAAS;oBAC7B,SAAS;iBACV;aACF,EACD,EAAE,cAAc,EAAE,OAAO,EAAE,CAC5B,CAAC;YACF,OAAO,MAAM,KAAK,IAAI,CAAC;QACzB,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,IAAA,6BAAkB,EAAC,GAAG,EAAE,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,eAAe,CACnB,GAAW,EACX,aAAqB,EACrB,KAAa;QAEb,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;QACzB,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,CAAC,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,CACpD;gBACE,GAAG,EAAE,GAAG;gBACR,MAAM,EAAE,aAAa;gBACrB,SAAS,EAAE,EAAE,GAAG,EAAE,GAAG,EAAE;aACxB,EACD;gBACE,IAAI,EAAE;oBACJ,UAAU,EAAE,aAAa;oBACzB,SAAS,EAAE,GAAG;oBACd,SAAS,EAAE,GAAG,GAAG,KAAK;iBACvB;gBACD,IAAI,EAAE,EAAE,QAAQ,EAAE,CAAC,EAAE;aACtB,EACD,EAAE,cAAc,EAAE,OAAO,EAAE,CAC5B,CAA4B,CAAC;YAC9B,OAAO,MAAM,CAAC;QAChB,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,IAAA,6BAAkB,EAAC,GAAG,EAAE,IAAI,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC;IAED,6EAA6E;IAE7E,YAAY;QACV,OAAO;YACL,YAAY,EAAE,IAAI,CAAC,eAAe;YAClC,GAAG,EAAE,IAAI;YACT,GAAG,EAAE,IAAI,EAAE,gBAAgB;YAC3B,aAAa,EAAE,KAAK;SACrB,CAAC;IACJ,CAAC;IAED,6EAA6E;IAErE,YAAY,CAAI,GAAqB;QAC3C,OAAO;YACL,GAAG,EAAE,GAAG,CAAC,GAAG;YACZ,MAAM,EAAE,GAAG,CAAC,MAAmC;YAC/C,MAAM,EAAE,GAAG,CAAC,MAAuB;YACnC,WAAW,EAAE,GAAG,CAAC,WAAW;YAC5B,UAAU,EAAE,GAAG,CAAC,UAAU;YAC1B,QAAQ,EAAE,GAAG,CAAC,QAAQ;YACtB,SAAS,EAAE,GAAG,CAAC,SAAS;YACxB,SAAS,EAAE,GAAG,CAAC,SAAS;YACxB,SAAS,EAAE,GAAG,CAAC,SAAS;SACzB,CAAC;IACJ,CAAC;CACF;AAvLD,oCAuLC;AAED,iFAAiF;AAEjF;;;;;;;;;;;;;GAaG;AACH,SAAgB,kBAAkB,CAAC,OAA4B;IAC7D,OAAO,IAAI,YAAY,CAAC,OAAO,CAAC,CAAC;AACnC,CAAC"}
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Mongoose Storage Adapter Wrapper for @periodic/vanadium
|
|
4
|
+
*
|
|
5
|
+
* Peer dependency: "mongoose" >= 7.0.0
|
|
6
|
+
* Import: import { createMongooseAdapter } from '@periodic/vanadium/adapters/mongoose'
|
|
7
|
+
*
|
|
8
|
+
* This is a WRAPPER that preserves identical semantics to the MongoDB native adapter.
|
|
9
|
+
* Mongoose is purely an ORM layer — it does NOT change idempotency behavior.
|
|
10
|
+
*
|
|
11
|
+
* Required Prisma schema:
|
|
12
|
+
* ```prisma
|
|
13
|
+
* model vanadiumRecord {
|
|
14
|
+
* id String @id @map("_id")
|
|
15
|
+
* status String
|
|
16
|
+
* result Json?
|
|
17
|
+
* payloadHash String?
|
|
18
|
+
* ownerToken String?
|
|
19
|
+
* attempts Int @default(0)
|
|
20
|
+
* createdAt BigInt
|
|
21
|
+
* updatedAt BigInt
|
|
22
|
+
* expiresAt BigInt?
|
|
23
|
+
* }
|
|
24
|
+
* ```
|
|
25
|
+
*
|
|
26
|
+
* Required Mongoose schema (use createVanadiumSchema() helper):
|
|
27
|
+
* ```ts
|
|
28
|
+
* import { createVanadiumSchema } from '@periodic/vanadium/adapters/mongoose';
|
|
29
|
+
* const VanadiumRecord = mongoose.model('VanadiumRecord', createVanadiumSchema());
|
|
30
|
+
* ```
|
|
31
|
+
*/
|
|
32
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
33
|
+
exports.MongooseAdapter = void 0;
|
|
34
|
+
exports.createVanadiumSchema = createVanadiumSchema;
|
|
35
|
+
exports.createMongooseAdapter = createMongooseAdapter;
|
|
36
|
+
const index_js_1 = require("../../errors/index.js");
|
|
37
|
+
/* eslint-enable @typescript-eslint/no-explicit-any */
|
|
38
|
+
const MONGOOSE_DUPLICATE_KEY_CODE = 11000;
|
|
39
|
+
// ─── Mongoose Adapter ─────────────────────────────────────────────────────────
|
|
40
|
+
class MongooseAdapter {
|
|
41
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
42
|
+
constructor(options) {
|
|
43
|
+
this.name = 'mongoose';
|
|
44
|
+
if (!options.model) {
|
|
45
|
+
throw (0, index_js_1.createConfigurationError)('', 'mongoose', 'Mongoose adapter requires a model.');
|
|
46
|
+
}
|
|
47
|
+
this.model = options.model;
|
|
48
|
+
this.useTransactions = options.useTransactions ?? true;
|
|
49
|
+
this.clock = options.clock ?? Date.now;
|
|
50
|
+
}
|
|
51
|
+
// ── Storage Interface ─────────────────────────────────────────────────────
|
|
52
|
+
async get(key) {
|
|
53
|
+
try {
|
|
54
|
+
const doc = await this.model.findOne({ _id: key });
|
|
55
|
+
if (!doc)
|
|
56
|
+
return null;
|
|
57
|
+
return this._docToRecord(doc);
|
|
58
|
+
}
|
|
59
|
+
catch (err) {
|
|
60
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
async set(key, value, ttlMs) {
|
|
64
|
+
const now = this.clock();
|
|
65
|
+
const expiresAt = ttlMs !== undefined ? now + ttlMs : value.expiresAt;
|
|
66
|
+
const docData = {
|
|
67
|
+
_id: key,
|
|
68
|
+
status: value.status,
|
|
69
|
+
result: value.result,
|
|
70
|
+
payloadHash: value.payloadHash,
|
|
71
|
+
ownerToken: value.ownerToken,
|
|
72
|
+
attempts: value.attempts,
|
|
73
|
+
createdAt: value.createdAt,
|
|
74
|
+
updatedAt: value.updatedAt,
|
|
75
|
+
expiresAt,
|
|
76
|
+
};
|
|
77
|
+
try {
|
|
78
|
+
await this.model.create(docData);
|
|
79
|
+
}
|
|
80
|
+
catch (err) {
|
|
81
|
+
const code = err.code;
|
|
82
|
+
if (code !== MONGOOSE_DUPLICATE_KEY_CODE) {
|
|
83
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
84
|
+
}
|
|
85
|
+
// Key exists — update
|
|
86
|
+
try {
|
|
87
|
+
await this.model.findOneAndUpdate({ _id: key }, {
|
|
88
|
+
$set: {
|
|
89
|
+
status: value.status,
|
|
90
|
+
result: value.result,
|
|
91
|
+
payloadHash: value.payloadHash,
|
|
92
|
+
ownerToken: value.ownerToken,
|
|
93
|
+
attempts: value.attempts,
|
|
94
|
+
updatedAt: value.updatedAt,
|
|
95
|
+
expiresAt,
|
|
96
|
+
},
|
|
97
|
+
}, { new: true });
|
|
98
|
+
}
|
|
99
|
+
catch (updateErr) {
|
|
100
|
+
throw (0, index_js_1.createStorageError)(key, this.name, updateErr, this.clock);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
async delete(key) {
|
|
105
|
+
try {
|
|
106
|
+
await this.model.deleteOne({ _id: key });
|
|
107
|
+
}
|
|
108
|
+
catch (err) {
|
|
109
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
async compareAndSet(key, expectedOwnerToken, newValue, ttlMs) {
|
|
113
|
+
const now = this.clock();
|
|
114
|
+
const expiresAt = ttlMs !== undefined ? now + ttlMs : newValue.expiresAt;
|
|
115
|
+
try {
|
|
116
|
+
const result = await this.model.findOneAndUpdate({ _id: key, ownerToken: expectedOwnerToken }, {
|
|
117
|
+
$set: {
|
|
118
|
+
status: newValue.status,
|
|
119
|
+
result: newValue.result,
|
|
120
|
+
payloadHash: newValue.payloadHash,
|
|
121
|
+
ownerToken: newValue.ownerToken,
|
|
122
|
+
attempts: newValue.attempts,
|
|
123
|
+
updatedAt: newValue.updatedAt,
|
|
124
|
+
expiresAt,
|
|
125
|
+
},
|
|
126
|
+
}, { new: true });
|
|
127
|
+
return result !== null;
|
|
128
|
+
}
|
|
129
|
+
catch (err) {
|
|
130
|
+
throw (0, index_js_1.createStorageError)(key, this.name, err, this.clock);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
// ── Capabilities ──────────────────────────────────────────────────────────
|
|
134
|
+
capabilities() {
|
|
135
|
+
return {
|
|
136
|
+
transactions: this.useTransactions,
|
|
137
|
+
cas: true,
|
|
138
|
+
ttl: true,
|
|
139
|
+
advisoryLocks: false,
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
// ── Private ───────────────────────────────────────────────────────────────
|
|
143
|
+
_docToRecord(doc) {
|
|
144
|
+
return {
|
|
145
|
+
key: doc._id,
|
|
146
|
+
status: doc.status,
|
|
147
|
+
result: doc.result,
|
|
148
|
+
payloadHash: doc.payloadHash,
|
|
149
|
+
ownerToken: doc.ownerToken,
|
|
150
|
+
attempts: doc.attempts,
|
|
151
|
+
createdAt: doc.createdAt,
|
|
152
|
+
updatedAt: doc.updatedAt,
|
|
153
|
+
expiresAt: doc.expiresAt,
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
exports.MongooseAdapter = MongooseAdapter;
|
|
158
|
+
// ─── Schema Builder ───────────────────────────────────────────────────────────
|
|
159
|
+
/**
|
|
160
|
+
* Build the required Mongoose schema for Vanadium records.
|
|
161
|
+
* Pass the Mongoose Schema constructor from your mongoose import.
|
|
162
|
+
*/
|
|
163
|
+
function createVanadiumSchema(Schema) {
|
|
164
|
+
return new Schema({
|
|
165
|
+
_id: { type: String, required: true },
|
|
166
|
+
status: {
|
|
167
|
+
type: String,
|
|
168
|
+
enum: ['IN_PROGRESS', 'COMPLETED', 'FAILED'],
|
|
169
|
+
required: true,
|
|
170
|
+
},
|
|
171
|
+
result: { type: Schema.Types.Mixed },
|
|
172
|
+
payloadHash: { type: String },
|
|
173
|
+
ownerToken: { type: String },
|
|
174
|
+
attempts: { type: Number, default: 0 },
|
|
175
|
+
createdAt: { type: Number, required: true },
|
|
176
|
+
updatedAt: { type: Number, required: true },
|
|
177
|
+
expiresAt: { type: Number },
|
|
178
|
+
}, {
|
|
179
|
+
versionKey: false,
|
|
180
|
+
strict: true,
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
// ─── Factory ──────────────────────────────────────────────────────────────────
|
|
184
|
+
/**
|
|
185
|
+
* Create a Mongoose storage adapter.
|
|
186
|
+
*
|
|
187
|
+
* @example
|
|
188
|
+
* ```ts
|
|
189
|
+
* import mongoose from 'mongoose';
|
|
190
|
+
* import { createMongooseAdapter, createVanadiumSchema } from '@periodic/vanadium/adapters/mongoose';
|
|
191
|
+
*
|
|
192
|
+
* const VanadiumRecord = mongoose.model('VanadiumRecord', createVanadiumSchema(mongoose.Schema));
|
|
193
|
+
* const adapter = createMongooseAdapter({ model: VanadiumRecord });
|
|
194
|
+
* ```
|
|
195
|
+
*/
|
|
196
|
+
function createMongooseAdapter(options) {
|
|
197
|
+
return new MongooseAdapter(options);
|
|
198
|
+
}
|
|
199
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/adapters/mongoose/index.ts"],"names":[],"mappings":"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"}
|