@ddd-ts/store-inmemory 0.0.0-0.1
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/dist/in-memory.transaction.d.ts +16 -0
- package/dist/in-memory.transaction.d.ts.map +1 -0
- package/dist/in-memory.transaction.js +31 -0
- package/dist/in-memory.transaction.js.map +1 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +15 -0
- package/dist/index.js.map +1 -0
- package/dist/store/in-memory.collection.d.ts +29 -0
- package/dist/store/in-memory.collection.d.ts.map +1 -0
- package/dist/store/in-memory.collection.js +70 -0
- package/dist/store/in-memory.collection.js.map +1 -0
- package/dist/store/in-memory.database.d.ts +61 -0
- package/dist/store/in-memory.database.d.ts.map +1 -0
- package/dist/store/in-memory.database.js +145 -0
- package/dist/store/in-memory.database.js.map +1 -0
- package/dist/store/in-memory.storage.d.ts +10 -0
- package/dist/store/in-memory.storage.d.ts.map +1 -0
- package/dist/store/in-memory.storage.js +41 -0
- package/dist/store/in-memory.storage.js.map +1 -0
- package/dist/store/in-memory.store.d.ts +25 -0
- package/dist/store/in-memory.store.d.ts.map +1 -0
- package/dist/store/in-memory.store.js +73 -0
- package/dist/store/in-memory.store.js.map +1 -0
- package/package.json +20 -0
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
import { TransactionPerformer } from "@ddd-ts/model";
|
|
2
|
+
import { InMemoryDatabase, InMemoryUnderlyingTransaction } from "./store/in-memory.database";
|
|
3
|
+
export declare class InMemoryTransaction {
|
|
4
|
+
readonly transaction: InMemoryUnderlyingTransaction;
|
|
5
|
+
commitListeners: (() => void)[];
|
|
6
|
+
constructor(transaction: InMemoryUnderlyingTransaction);
|
|
7
|
+
onCommit(callback: () => void): void;
|
|
8
|
+
executeCommitListeners(): Promise<void>;
|
|
9
|
+
}
|
|
10
|
+
export declare class InMemoryTransactionPerformer extends TransactionPerformer<InMemoryTransaction> {
|
|
11
|
+
constructor(db: InMemoryDatabase);
|
|
12
|
+
}
|
|
13
|
+
export declare class FakeInMemoryTransactionPerformer extends TransactionPerformer<InMemoryTransaction> {
|
|
14
|
+
constructor();
|
|
15
|
+
}
|
|
16
|
+
//# sourceMappingURL=in-memory.transaction.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.transaction.d.ts","sourceRoot":"","sources":["../src/in-memory.transaction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,oBAAoB,EAAE,MAAM,eAAe,CAAC;AACrD,OAAO,EACL,gBAAgB,EAChB,6BAA6B,EAC9B,MAAM,4BAA4B,CAAC;AAEpC,qBAAa,mBAAmB;aAGF,WAAW,EAAE,6BAA6B;IAFtE,eAAe,EAAE,CAAC,MAAM,IAAI,CAAC,EAAE,CAAM;gBAET,WAAW,EAAE,6BAA6B;IAEtE,QAAQ,CAAC,QAAQ,EAAE,MAAM,IAAI;IAIvB,sBAAsB;CAG7B;AAED,qBAAa,4BAA6B,SAAQ,oBAAoB,CAAC,mBAAmB,CAAC;gBAC7E,EAAE,EAAE,gBAAgB;CAGjC;AAED,qBAAa,gCAAiC,SAAQ,oBAAoB,CAAC,mBAAmB,CAAC;;CAI9F"}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.FakeInMemoryTransactionPerformer = exports.InMemoryTransactionPerformer = exports.InMemoryTransaction = void 0;
|
|
4
|
+
const model_1 = require("@ddd-ts/model");
|
|
5
|
+
class InMemoryTransaction {
|
|
6
|
+
transaction;
|
|
7
|
+
commitListeners = [];
|
|
8
|
+
constructor(transaction) {
|
|
9
|
+
this.transaction = transaction;
|
|
10
|
+
}
|
|
11
|
+
onCommit(callback) {
|
|
12
|
+
this.commitListeners.push(callback);
|
|
13
|
+
}
|
|
14
|
+
async executeCommitListeners() {
|
|
15
|
+
await Promise.all(this.commitListeners.map((cb) => cb()));
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
exports.InMemoryTransaction = InMemoryTransaction;
|
|
19
|
+
class InMemoryTransactionPerformer extends model_1.TransactionPerformer {
|
|
20
|
+
constructor(db) {
|
|
21
|
+
super((effect) => db.transactionally((trx) => effect(trx)));
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
exports.InMemoryTransactionPerformer = InMemoryTransactionPerformer;
|
|
25
|
+
class FakeInMemoryTransactionPerformer extends model_1.TransactionPerformer {
|
|
26
|
+
constructor() {
|
|
27
|
+
super((effect) => effect(new InMemoryTransaction(null)));
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
exports.FakeInMemoryTransactionPerformer = FakeInMemoryTransactionPerformer;
|
|
31
|
+
//# sourceMappingURL=in-memory.transaction.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.transaction.js","sourceRoot":"","sources":["../src/in-memory.transaction.ts"],"names":[],"mappings":";;;AAAA,yCAAqD;AAMrD,MAAa,mBAAmB;IAGF;IAF5B,eAAe,GAAmB,EAAE,CAAC;IAErC,YAA4B,WAA0C;QAA1C,gBAAW,GAAX,WAAW,CAA+B;IAAG,CAAC;IAE1E,QAAQ,CAAC,QAAoB;QAC3B,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;IACtC,CAAC;IAED,KAAK,CAAC,sBAAsB;QAC1B,MAAM,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC;IAC5D,CAAC;CACF;AAZD,kDAYC;AAED,MAAa,4BAA6B,SAAQ,4BAAyC;IACzF,YAAY,EAAoB;QAC9B,KAAK,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE,CAAC,eAAe,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;IAC9D,CAAC;CACF;AAJD,oEAIC;AAED,MAAa,gCAAiC,SAAQ,4BAAyC;IAC7F;QACE,KAAK,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,mBAAmB,CAAC,IAAW,CAAC,CAAC,CAAC,CAAC;IAClE,CAAC;CACF;AAJD,4EAIC"}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
export { InMemoryTransactionPerformer, FakeInMemoryTransactionPerformer, InMemoryTransaction, } from "./in-memory.transaction";
|
|
2
|
+
export { InMemoryDatabase, InMemoryUnderlyingTransaction, CannotReadAfterWrites, TransactionCollidedTooManyTimes, } from "./store/in-memory.database";
|
|
3
|
+
export { InMemoryStore } from "./store/in-memory.store";
|
|
4
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACN,4BAA4B,EAC5B,gCAAgC,EAChC,mBAAmB,GACnB,MAAM,yBAAyB,CAAC;AACjC,OAAO,EACN,gBAAgB,EAChB,6BAA6B,EAC7B,qBAAqB,EACrB,+BAA+B,GAC/B,MAAM,4BAA4B,CAAC;AACpC,OAAO,EAAE,aAAa,EAAE,MAAM,yBAAyB,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.InMemoryStore = exports.TransactionCollidedTooManyTimes = exports.CannotReadAfterWrites = exports.InMemoryUnderlyingTransaction = exports.InMemoryDatabase = exports.InMemoryTransaction = exports.FakeInMemoryTransactionPerformer = exports.InMemoryTransactionPerformer = void 0;
|
|
4
|
+
var in_memory_transaction_1 = require("./in-memory.transaction");
|
|
5
|
+
Object.defineProperty(exports, "InMemoryTransactionPerformer", { enumerable: true, get: function () { return in_memory_transaction_1.InMemoryTransactionPerformer; } });
|
|
6
|
+
Object.defineProperty(exports, "FakeInMemoryTransactionPerformer", { enumerable: true, get: function () { return in_memory_transaction_1.FakeInMemoryTransactionPerformer; } });
|
|
7
|
+
Object.defineProperty(exports, "InMemoryTransaction", { enumerable: true, get: function () { return in_memory_transaction_1.InMemoryTransaction; } });
|
|
8
|
+
var in_memory_database_1 = require("./store/in-memory.database");
|
|
9
|
+
Object.defineProperty(exports, "InMemoryDatabase", { enumerable: true, get: function () { return in_memory_database_1.InMemoryDatabase; } });
|
|
10
|
+
Object.defineProperty(exports, "InMemoryUnderlyingTransaction", { enumerable: true, get: function () { return in_memory_database_1.InMemoryUnderlyingTransaction; } });
|
|
11
|
+
Object.defineProperty(exports, "CannotReadAfterWrites", { enumerable: true, get: function () { return in_memory_database_1.CannotReadAfterWrites; } });
|
|
12
|
+
Object.defineProperty(exports, "TransactionCollidedTooManyTimes", { enumerable: true, get: function () { return in_memory_database_1.TransactionCollidedTooManyTimes; } });
|
|
13
|
+
var in_memory_store_1 = require("./store/in-memory.store");
|
|
14
|
+
Object.defineProperty(exports, "InMemoryStore", { enumerable: true, get: function () { return in_memory_store_1.InMemoryStore; } });
|
|
15
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";;;AAAA,iEAIiC;AAHhC,qIAAA,4BAA4B,OAAA;AAC5B,yIAAA,gCAAgC,OAAA;AAChC,4HAAA,mBAAmB,OAAA;AAEpB,iEAKoC;AAJnC,sHAAA,gBAAgB,OAAA;AAChB,mIAAA,6BAA6B,OAAA;AAC7B,2HAAA,qBAAqB,OAAA;AACrB,qIAAA,+BAA+B,OAAA;AAEhC,2DAAwD;AAA/C,gHAAA,aAAa,OAAA"}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
export declare class Collection {
|
|
2
|
+
private data;
|
|
3
|
+
constructor(data?: Map<string, {
|
|
4
|
+
savedAt: number;
|
|
5
|
+
data: any;
|
|
6
|
+
}>);
|
|
7
|
+
clear(): void;
|
|
8
|
+
getLatestSnapshot(id: string): any;
|
|
9
|
+
clone(): Collection;
|
|
10
|
+
merge(other: Collection): Collection;
|
|
11
|
+
delete(id: string): void;
|
|
12
|
+
getRaw(id: string): {
|
|
13
|
+
savedAt: number;
|
|
14
|
+
data: any;
|
|
15
|
+
} | undefined;
|
|
16
|
+
countAll(): number;
|
|
17
|
+
get(id: string): any;
|
|
18
|
+
getAllRaw(): {
|
|
19
|
+
id: string;
|
|
20
|
+
data: {
|
|
21
|
+
savedAt: number;
|
|
22
|
+
data: any;
|
|
23
|
+
};
|
|
24
|
+
}[];
|
|
25
|
+
getAll(): any[];
|
|
26
|
+
save(id: string, data: any): void;
|
|
27
|
+
toPretty(): string;
|
|
28
|
+
}
|
|
29
|
+
//# sourceMappingURL=in-memory.collection.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.collection.d.ts","sourceRoot":"","sources":["../../src/store/in-memory.collection.ts"],"names":[],"mappings":"AAAA,qBAAa,UAAU;IAEnB,OAAO,CAAC,IAAI;gBAAJ,IAAI,GAAE,GAAG,CAAC,MAAM,EAAE;QAAE,OAAO,EAAE,MAAM,CAAC;QAAC,IAAI,EAAE,GAAG,CAAA;KAAE,CAAa;IAGvE,KAAK;IAIL,iBAAiB,CAAC,EAAE,EAAE,MAAM;IAO5B,KAAK;IAQL,KAAK,CAAC,KAAK,EAAE,UAAU;IAWvB,MAAM,CAAC,EAAE,EAAE,MAAM,GAAG,IAAI;IAIxB,MAAM,CAAC,EAAE,EAAE,MAAM;;;;IAIjB,QAAQ;IAIR,GAAG,CAAC,EAAE,EAAE,MAAM,GAAG,GAAG;IAIpB,SAAS;;;;;;;IAIT,MAAM,IAAI,GAAG,EAAE;IAIf,IAAI,CAAC,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,GAAG,IAAI;IAKjC,QAAQ;CAQT"}
|
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.Collection = void 0;
|
|
4
|
+
class Collection {
|
|
5
|
+
data;
|
|
6
|
+
constructor(data = new Map()) {
|
|
7
|
+
this.data = data;
|
|
8
|
+
}
|
|
9
|
+
clear() {
|
|
10
|
+
this.data.clear();
|
|
11
|
+
}
|
|
12
|
+
getLatestSnapshot(id) {
|
|
13
|
+
const data = [...this.data.values()];
|
|
14
|
+
const sameId = data.filter((d) => d.data.id === id);
|
|
15
|
+
const sorted = sameId.sort((a, b) => b.savedAt - a.savedAt);
|
|
16
|
+
return sorted[0]?.data;
|
|
17
|
+
}
|
|
18
|
+
clone() {
|
|
19
|
+
const clone = new Map();
|
|
20
|
+
for (const [key, value] of this.data) {
|
|
21
|
+
clone.set(key, value);
|
|
22
|
+
}
|
|
23
|
+
return new Collection(clone);
|
|
24
|
+
}
|
|
25
|
+
merge(other) {
|
|
26
|
+
const merge = new Map();
|
|
27
|
+
for (const [key, value] of this.data) {
|
|
28
|
+
merge.set(key, value);
|
|
29
|
+
}
|
|
30
|
+
for (const [key, value] of other.data) {
|
|
31
|
+
merge.set(key, value);
|
|
32
|
+
}
|
|
33
|
+
return new Collection(merge);
|
|
34
|
+
}
|
|
35
|
+
delete(id) {
|
|
36
|
+
this.data.delete(id);
|
|
37
|
+
}
|
|
38
|
+
getRaw(id) {
|
|
39
|
+
return this.data.get(id);
|
|
40
|
+
}
|
|
41
|
+
countAll() {
|
|
42
|
+
return this.data.size;
|
|
43
|
+
}
|
|
44
|
+
get(id) {
|
|
45
|
+
return this.data.get(id)?.data;
|
|
46
|
+
}
|
|
47
|
+
getAllRaw() {
|
|
48
|
+
return [...this.data.entries()].map(([id, data]) => ({ id, data }));
|
|
49
|
+
}
|
|
50
|
+
getAll() {
|
|
51
|
+
return [...this.data.entries()].map(([id, data]) => data.data);
|
|
52
|
+
}
|
|
53
|
+
save(id, data) {
|
|
54
|
+
const now = process.hrtime.bigint();
|
|
55
|
+
this.data.set(id, { savedAt: Number(now), data });
|
|
56
|
+
}
|
|
57
|
+
toPretty() {
|
|
58
|
+
return [...this.data.entries()]
|
|
59
|
+
.map(([id, data]) => `\t\t"${id}": ${JSON.stringify(data.data, replaceBigInt)}`)
|
|
60
|
+
.join(",\n");
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
exports.Collection = Collection;
|
|
64
|
+
function replaceBigInt(key, value) {
|
|
65
|
+
if (typeof value === "bigint") {
|
|
66
|
+
return value.toString();
|
|
67
|
+
}
|
|
68
|
+
return value;
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=in-memory.collection.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.collection.js","sourceRoot":"","sources":["../../src/store/in-memory.collection.ts"],"names":[],"mappings":";;;AAAA,MAAa,UAAU;IAEX;IADV,YACU,OAAoD,IAAI,GAAG,EAAE;QAA7D,SAAI,GAAJ,IAAI,CAAyD;IACnE,CAAC;IAEL,KAAK;QACH,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;IACpB,CAAC;IAED,iBAAiB,CAAC,EAAU;QAC1B,MAAM,IAAI,GAAG,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QACrC,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;QACpD,MAAM,MAAM,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC;QAC5D,OAAO,MAAM,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC;IACzB,CAAC;IAED,KAAK;QACH,MAAM,KAAK,GAAG,IAAI,GAAG,EAAE,CAAC;QACxB,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;YACrC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QACxB,CAAC;QACD,OAAO,IAAI,UAAU,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,KAAK,CAAC,KAAiB;QACrB,MAAM,KAAK,GAAG,IAAI,GAAG,EAAE,CAAC;QACxB,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,IAAI,CAAC,IAAI,EAAE,CAAC;YACrC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QACxB,CAAC;QACD,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC;YACtC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC;QACxB,CAAC;QACD,OAAO,IAAI,UAAU,CAAC,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,MAAM,CAAC,EAAU;QACf,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;IACvB,CAAC;IAED,MAAM,CAAC,EAAU;QACf,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAC1B,CAAC;IAED,QAAQ;QACN,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC;IACxB,CAAC;IAED,GAAG,CAAC,EAAU;QACZ,OAAO,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC;IACjC,CAAC;IAED,SAAS;QACP,OAAO,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,EAAE,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;IACtE,CAAC;IAED,MAAM;QACJ,OAAO,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,IAAI,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACjE,CAAC;IAED,IAAI,CAAC,EAAU,EAAE,IAAS;QACxB,MAAM,GAAG,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;QACpC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,EAAE,OAAO,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC;IACpD,CAAC;IAED,QAAQ;QACN,OAAO,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;aAC5B,GAAG,CACF,CAAC,CAAC,EAAE,EAAE,IAAI,CAAC,EAAE,EAAE,CACb,QAAQ,EAAE,MAAM,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,IAAI,EAAE,aAAa,CAAC,EAAE,CAC7D;aACA,IAAI,CAAC,KAAK,CAAC,CAAC;IACjB,CAAC;CACF;AAxED,gCAwEC;AAED,SAAS,aAAa,CAAC,GAAW,EAAE,KAAU;IAC5C,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;QAC9B,OAAO,KAAK,CAAC,QAAQ,EAAE,CAAC;IAC1B,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC"}
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { InMemoryTransaction } from "..";
|
|
2
|
+
import { Storage } from "./in-memory.storage";
|
|
3
|
+
export declare class CannotReadAfterWrites extends Error {
|
|
4
|
+
constructor();
|
|
5
|
+
}
|
|
6
|
+
export declare class TransactionCollision extends Error {
|
|
7
|
+
constructor();
|
|
8
|
+
}
|
|
9
|
+
export declare class TransactionCollidedTooManyTimes extends Error {
|
|
10
|
+
constructor(tries: number);
|
|
11
|
+
}
|
|
12
|
+
type ReadOperation = {
|
|
13
|
+
type: "read";
|
|
14
|
+
collectionName: string;
|
|
15
|
+
id: string;
|
|
16
|
+
savedAt: number | undefined;
|
|
17
|
+
};
|
|
18
|
+
type WriteOperation = {
|
|
19
|
+
type: "write";
|
|
20
|
+
collectionName: string;
|
|
21
|
+
id: string;
|
|
22
|
+
data: any;
|
|
23
|
+
};
|
|
24
|
+
type DeleteOperation = {
|
|
25
|
+
type: "delete";
|
|
26
|
+
collectionName: string;
|
|
27
|
+
id: string;
|
|
28
|
+
savedAt: number | undefined;
|
|
29
|
+
};
|
|
30
|
+
type TransactionOperation = ReadOperation | WriteOperation | DeleteOperation;
|
|
31
|
+
export declare class InMemoryUnderlyingTransaction {
|
|
32
|
+
readonly operations: TransactionOperation[];
|
|
33
|
+
readonly id: string;
|
|
34
|
+
private ensureNoWrites;
|
|
35
|
+
markRead(collectionName: string, id: string, savedAt: number | undefined): void;
|
|
36
|
+
markWritten(collectionName: string, id: any, data: any): void;
|
|
37
|
+
markDeleted(collectionName: string, id: any, savedAt: number | undefined): void;
|
|
38
|
+
checkConsistency(storage: Storage): boolean;
|
|
39
|
+
}
|
|
40
|
+
export declare class InMemoryDatabase {
|
|
41
|
+
private storage;
|
|
42
|
+
clear(collectionName: string): void;
|
|
43
|
+
load(collectionName: string, id: string, trx?: InMemoryUnderlyingTransaction): any;
|
|
44
|
+
delete(collectionName: string, id: string, trx?: InMemoryUnderlyingTransaction): void;
|
|
45
|
+
countAll(collectionName: string): number;
|
|
46
|
+
loadAll(collectionName: string): {
|
|
47
|
+
id: string;
|
|
48
|
+
data: {
|
|
49
|
+
savedAt: number;
|
|
50
|
+
data: any;
|
|
51
|
+
};
|
|
52
|
+
}[];
|
|
53
|
+
loadLatestSnapshot(id: string): any;
|
|
54
|
+
save(collectionName: string, id: string, data: any, trx?: InMemoryUnderlyingTransaction): void;
|
|
55
|
+
private static transactionTries;
|
|
56
|
+
transactionally(fn: (trx: InMemoryTransaction) => any): Promise<any>;
|
|
57
|
+
private commit;
|
|
58
|
+
print(): void;
|
|
59
|
+
}
|
|
60
|
+
export {};
|
|
61
|
+
//# sourceMappingURL=in-memory.database.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.database.d.ts","sourceRoot":"","sources":["../../src/store/in-memory.database.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,mBAAmB,EAAE,MAAM,IAAI,CAAC;AAEzC,OAAO,EAAE,OAAO,EAAE,MAAM,qBAAqB,CAAC;AAE9C,qBAAa,qBAAsB,SAAQ,KAAK;;CAI/C;AAED,qBAAa,oBAAqB,SAAQ,KAAK;;CAI9C;AAED,qBAAa,+BAAgC,SAAQ,KAAK;gBAC5C,KAAK,EAAE,MAAM;CAG1B;AAED,KAAK,aAAa,GAAG;IACnB,IAAI,EAAE,MAAM,CAAC;IACb,cAAc,EAAE,MAAM,CAAC;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,EAAE,MAAM,GAAG,SAAS,CAAC;CAC7B,CAAC;AAEF,KAAK,cAAc,GAAG;IACpB,IAAI,EAAE,OAAO,CAAC;IACd,cAAc,EAAE,MAAM,CAAC;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,IAAI,EAAE,GAAG,CAAC;CACX,CAAC;AAEF,KAAK,eAAe,GAAG;IACrB,IAAI,EAAE,QAAQ,CAAC;IACf,cAAc,EAAE,MAAM,CAAC;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,EAAE,MAAM,GAAG,SAAS,CAAC;CAC7B,CAAC;AAEF,KAAK,oBAAoB,GAAG,aAAa,GAAG,cAAc,GAAG,eAAe,CAAC;AAE7E,qBAAa,6BAA6B;IACxC,SAAgB,UAAU,EAAE,oBAAoB,EAAE,CAAM;IAExD,SAAgB,EAAE,SAAyC;IAE3D,OAAO,CAAC,cAAc;IAUf,QAAQ,CACb,cAAc,EAAE,MAAM,EACtB,EAAE,EAAE,MAAM,EACV,OAAO,EAAE,MAAM,GAAG,SAAS;IAMtB,WAAW,CAAC,cAAc,EAAE,MAAM,EAAE,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE,GAAG;IAItD,WAAW,CAChB,cAAc,EAAE,MAAM,EACtB,EAAE,EAAE,GAAG,EACP,OAAO,EAAE,MAAM,GAAG,SAAS;IAKtB,gBAAgB,CAAC,OAAO,EAAE,OAAO;CAkBzC;AAED,qBAAa,gBAAgB;IAC3B,OAAO,CAAC,OAAO,CAAiB;IAEhC,KAAK,CAAC,cAAc,EAAE,MAAM;IAI5B,IAAI,CACF,cAAc,EAAE,MAAM,EACtB,EAAE,EAAE,MAAM,EACV,GAAG,CAAC,EAAE,6BAA6B,GAClC,GAAG;IAUN,MAAM,CACJ,cAAc,EAAE,MAAM,EACtB,EAAE,EAAE,MAAM,EACV,GAAG,CAAC,EAAE,6BAA6B,GAClC,IAAI;IASP,QAAQ,CAAC,cAAc,EAAE,MAAM;IAI/B,OAAO,CAAC,cAAc,EAAE,MAAM;;;;;;;IAI9B,kBAAkB,CAAC,EAAE,EAAE,MAAM;IAI7B,IAAI,CACF,cAAc,EAAE,MAAM,EACtB,EAAE,EAAE,MAAM,EACV,IAAI,EAAE,GAAG,EACT,GAAG,CAAC,EAAE,6BAA6B,GAClC,IAAI;IAQP,OAAO,CAAC,MAAM,CAAC,gBAAgB,CAAK;IAE9B,eAAe,CAAC,EAAE,EAAE,CAAC,GAAG,EAAE,mBAAmB,KAAK,GAAG;IA2B3D,OAAO,CAAC,MAAM;IAkBd,KAAK;CAGN"}
|
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.InMemoryDatabase = exports.InMemoryUnderlyingTransaction = exports.TransactionCollidedTooManyTimes = exports.TransactionCollision = exports.CannotReadAfterWrites = void 0;
|
|
4
|
+
const __1 = require("..");
|
|
5
|
+
const in_memory_storage_1 = require("./in-memory.storage");
|
|
6
|
+
class CannotReadAfterWrites extends Error {
|
|
7
|
+
constructor() {
|
|
8
|
+
super("Cannot read after having written into a transaction");
|
|
9
|
+
}
|
|
10
|
+
}
|
|
11
|
+
exports.CannotReadAfterWrites = CannotReadAfterWrites;
|
|
12
|
+
class TransactionCollision extends Error {
|
|
13
|
+
constructor() {
|
|
14
|
+
super("Transaction has collided with other extern writes");
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
exports.TransactionCollision = TransactionCollision;
|
|
18
|
+
class TransactionCollidedTooManyTimes extends Error {
|
|
19
|
+
constructor(tries) {
|
|
20
|
+
super(`Transaction collided too many times (${tries})`);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
exports.TransactionCollidedTooManyTimes = TransactionCollidedTooManyTimes;
|
|
24
|
+
class InMemoryUnderlyingTransaction {
|
|
25
|
+
operations = [];
|
|
26
|
+
id = Math.random().toString().substring(2);
|
|
27
|
+
ensureNoWrites() {
|
|
28
|
+
if (this.operations.some((operation) => operation.type === "write" || operation.type === "delete")) {
|
|
29
|
+
throw new CannotReadAfterWrites();
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
markRead(collectionName, id, savedAt) {
|
|
33
|
+
this.ensureNoWrites();
|
|
34
|
+
this.operations.push({ type: "read", collectionName, id, savedAt });
|
|
35
|
+
}
|
|
36
|
+
markWritten(collectionName, id, data) {
|
|
37
|
+
this.operations.push({ type: "write", collectionName, id, data });
|
|
38
|
+
}
|
|
39
|
+
markDeleted(collectionName, id, savedAt) {
|
|
40
|
+
this.operations.push({ type: "delete", collectionName, id, savedAt });
|
|
41
|
+
}
|
|
42
|
+
checkConsistency(storage) {
|
|
43
|
+
for (const operation of this.operations) {
|
|
44
|
+
if (operation.type !== "read") {
|
|
45
|
+
continue;
|
|
46
|
+
}
|
|
47
|
+
const collection = storage.getCollection(operation.collectionName);
|
|
48
|
+
if (!collection) {
|
|
49
|
+
return false;
|
|
50
|
+
}
|
|
51
|
+
if (operation.savedAt !== collection.getRaw(operation.id)?.savedAt) {
|
|
52
|
+
return false;
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
return true;
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
exports.InMemoryUnderlyingTransaction = InMemoryUnderlyingTransaction;
|
|
59
|
+
class InMemoryDatabase {
|
|
60
|
+
storage = new in_memory_storage_1.Storage();
|
|
61
|
+
clear(collectionName) {
|
|
62
|
+
this.storage.getCollection(collectionName).clear();
|
|
63
|
+
}
|
|
64
|
+
load(collectionName, id, trx) {
|
|
65
|
+
const collection = this.storage.getCollection(collectionName);
|
|
66
|
+
const data = collection.get(id);
|
|
67
|
+
if (trx) {
|
|
68
|
+
const doc = collection.getRaw(id);
|
|
69
|
+
trx.markRead(collectionName, id, doc?.savedAt);
|
|
70
|
+
}
|
|
71
|
+
return data;
|
|
72
|
+
}
|
|
73
|
+
delete(collectionName, id, trx) {
|
|
74
|
+
if (trx) {
|
|
75
|
+
const doc = this.storage.getCollection(collectionName).getRaw(id);
|
|
76
|
+
trx.markDeleted(collectionName, id, doc?.savedAt);
|
|
77
|
+
}
|
|
78
|
+
else {
|
|
79
|
+
this.storage.getCollection(collectionName).delete(id);
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
countAll(collectionName) {
|
|
83
|
+
return this.storage.getCollection(collectionName).countAll();
|
|
84
|
+
}
|
|
85
|
+
loadAll(collectionName) {
|
|
86
|
+
return this.storage.getCollection(collectionName).getAllRaw();
|
|
87
|
+
}
|
|
88
|
+
loadLatestSnapshot(id) {
|
|
89
|
+
return this.storage.getCollection("snapshots").getLatestSnapshot(id);
|
|
90
|
+
}
|
|
91
|
+
save(collectionName, id, data, trx) {
|
|
92
|
+
if (trx) {
|
|
93
|
+
trx.markWritten(collectionName, id, data);
|
|
94
|
+
}
|
|
95
|
+
else {
|
|
96
|
+
this.storage.getCollection(collectionName).save(id, data);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
static transactionTries = 5;
|
|
100
|
+
async transactionally(fn) {
|
|
101
|
+
let trx = new __1.InMemoryTransaction(new InMemoryUnderlyingTransaction());
|
|
102
|
+
let retry = InMemoryDatabase.transactionTries;
|
|
103
|
+
let latestReturnValue = undefined;
|
|
104
|
+
while (retry--) {
|
|
105
|
+
try {
|
|
106
|
+
latestReturnValue = await fn(trx);
|
|
107
|
+
this.commit(trx);
|
|
108
|
+
break;
|
|
109
|
+
}
|
|
110
|
+
catch (error) {
|
|
111
|
+
if (error instanceof TransactionCollision) {
|
|
112
|
+
trx = new __1.InMemoryTransaction(new InMemoryUnderlyingTransaction());
|
|
113
|
+
}
|
|
114
|
+
else {
|
|
115
|
+
throw error;
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
if (retry === -1) {
|
|
120
|
+
throw new TransactionCollidedTooManyTimes(InMemoryDatabase.transactionTries);
|
|
121
|
+
}
|
|
122
|
+
return latestReturnValue;
|
|
123
|
+
}
|
|
124
|
+
commit(trx) {
|
|
125
|
+
if (!trx.transaction.checkConsistency(this.storage)) {
|
|
126
|
+
throw new TransactionCollision();
|
|
127
|
+
}
|
|
128
|
+
for (const operation of trx.transaction.operations) {
|
|
129
|
+
if (operation.type === "read") {
|
|
130
|
+
continue;
|
|
131
|
+
}
|
|
132
|
+
if (operation.type === "write") {
|
|
133
|
+
this.save(operation.collectionName, operation.id, operation.data);
|
|
134
|
+
}
|
|
135
|
+
if (operation.type === "delete") {
|
|
136
|
+
this.delete(operation.collectionName, operation.id);
|
|
137
|
+
}
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
print() {
|
|
141
|
+
console.log(["Database:", this.storage.toPretty()].join("\n"));
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
exports.InMemoryDatabase = InMemoryDatabase;
|
|
145
|
+
//# sourceMappingURL=in-memory.database.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.database.js","sourceRoot":"","sources":["../../src/store/in-memory.database.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { Collection } from "./in-memory.collection";
|
|
2
|
+
export declare class Storage {
|
|
3
|
+
private collections;
|
|
4
|
+
constructor(collections?: Map<string, Collection>);
|
|
5
|
+
clone(): Storage;
|
|
6
|
+
merge(other: Storage): Storage;
|
|
7
|
+
getCollection(collectionName: string): Collection;
|
|
8
|
+
toPretty(): string;
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=in-memory.storage.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.storage.d.ts","sourceRoot":"","sources":["../../src/store/in-memory.storage.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AAEpD,qBAAa,OAAO;IACN,OAAO,CAAC,WAAW;gBAAX,WAAW,GAAE,GAAG,CAAC,MAAM,EAAE,UAAU,CAAa;IAEpE,KAAK;IASL,KAAK,CAAC,KAAK,EAAE,OAAO;IAWpB,aAAa,CAAC,cAAc,EAAE,MAAM,GAAG,UAAU;IAMjD,QAAQ;CAUT"}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.Storage = void 0;
|
|
4
|
+
const in_memory_collection_1 = require("./in-memory.collection");
|
|
5
|
+
class Storage {
|
|
6
|
+
collections;
|
|
7
|
+
constructor(collections = new Map()) {
|
|
8
|
+
this.collections = collections;
|
|
9
|
+
}
|
|
10
|
+
clone() {
|
|
11
|
+
const clone = new Map();
|
|
12
|
+
for (const [collectionName, collection] of this.collections) {
|
|
13
|
+
clone.set(collectionName, collection.clone());
|
|
14
|
+
}
|
|
15
|
+
return new Storage(clone);
|
|
16
|
+
}
|
|
17
|
+
merge(other) {
|
|
18
|
+
const collections = new Map();
|
|
19
|
+
for (const [collectionName, collection] of other.collections) {
|
|
20
|
+
collections.set(collectionName, this.getCollection(collectionName).merge(collection));
|
|
21
|
+
}
|
|
22
|
+
return new Storage(collections);
|
|
23
|
+
}
|
|
24
|
+
getCollection(collectionName) {
|
|
25
|
+
const collection = this.collections.get(collectionName) || new in_memory_collection_1.Collection();
|
|
26
|
+
this.collections.set(collectionName, collection);
|
|
27
|
+
return collection;
|
|
28
|
+
}
|
|
29
|
+
toPretty() {
|
|
30
|
+
return [...this.collections.entries()]
|
|
31
|
+
.map(([collectionName, collection]) => {
|
|
32
|
+
return [
|
|
33
|
+
'Collection: "' + collectionName + '"',
|
|
34
|
+
collection.toPretty(),
|
|
35
|
+
].join("\n");
|
|
36
|
+
})
|
|
37
|
+
.join("\n");
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
exports.Storage = Storage;
|
|
41
|
+
//# sourceMappingURL=in-memory.storage.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.storage.js","sourceRoot":"","sources":["../../src/store/in-memory.storage.ts"],"names":[],"mappings":";;;AAAA,iEAAoD;AAEpD,MAAa,OAAO;IACE;IAApB,YAAoB,cAAuC,IAAI,GAAG,EAAE;QAAhD,gBAAW,GAAX,WAAW,CAAqC;IAAG,CAAC;IAExE,KAAK;QACH,MAAM,KAAK,GAAG,IAAI,GAAG,EAAsB,CAAC;QAC5C,KAAK,MAAM,CAAC,cAAc,EAAE,UAAU,CAAC,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YAC5D,KAAK,CAAC,GAAG,CAAC,cAAc,EAAE,UAAU,CAAC,KAAK,EAAE,CAAC,CAAC;QAChD,CAAC;QAED,OAAO,IAAI,OAAO,CAAC,KAAK,CAAC,CAAC;IAC5B,CAAC;IAED,KAAK,CAAC,KAAc;QAClB,MAAM,WAAW,GAAG,IAAI,GAAG,EAAsB,CAAC;QAClD,KAAK,MAAM,CAAC,cAAc,EAAE,UAAU,CAAC,IAAI,KAAK,CAAC,WAAW,EAAE,CAAC;YAC7D,WAAW,CAAC,GAAG,CACb,cAAc,EACd,IAAI,CAAC,aAAa,CAAC,cAAc,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,CACrD,CAAC;QACJ,CAAC;QACD,OAAO,IAAI,OAAO,CAAC,WAAW,CAAC,CAAC;IAClC,CAAC;IAED,aAAa,CAAC,cAAsB;QAClC,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,cAAc,CAAC,IAAI,IAAI,iCAAU,EAAE,CAAC;QAC5E,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,cAAc,EAAE,UAAU,CAAC,CAAC;QACjD,OAAO,UAAU,CAAC;IACpB,CAAC;IAED,QAAQ;QACN,OAAO,CAAC,GAAG,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;aACnC,GAAG,CAAC,CAAC,CAAC,cAAc,EAAE,UAAU,CAAC,EAAE,EAAE;YACpC,OAAO;gBACL,eAAe,GAAG,cAAc,GAAG,GAAG;gBACtC,UAAU,CAAC,QAAQ,EAAE;aACtB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACf,CAAC,CAAC;aACD,IAAI,CAAC,IAAI,CAAC,CAAC;IAChB,CAAC;CACF;AAvCD,0BAuCC"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { Store, Model } from "@ddd-ts/model";
|
|
2
|
+
import { InMemoryDatabase } from "./in-memory.database";
|
|
3
|
+
import { InMemoryTransaction } from "../in-memory.transaction";
|
|
4
|
+
import { ISerializer } from "@ddd-ts/serialization";
|
|
5
|
+
/**
|
|
6
|
+
* This in memory store is a copy store. It stores a copy of the actual model.
|
|
7
|
+
* It is the recommended inmemory store to use, as it reflects more closely the behaviour of a real store.
|
|
8
|
+
*/
|
|
9
|
+
export declare class InMemoryStore<M extends Model> implements Store<M> {
|
|
10
|
+
readonly collection: string;
|
|
11
|
+
readonly database: InMemoryDatabase;
|
|
12
|
+
readonly serializer: ISerializer<M>;
|
|
13
|
+
constructor(collection: string, database: InMemoryDatabase, serializer: ISerializer<M>);
|
|
14
|
+
private serializeId;
|
|
15
|
+
protected filter(predicate: (model: M) => boolean, trx?: InMemoryTransaction): Promise<M[]>;
|
|
16
|
+
clear(): void;
|
|
17
|
+
save(model: M, trx?: InMemoryTransaction): Promise<void>;
|
|
18
|
+
load(id: M["id"], trx?: InMemoryTransaction): Promise<M | undefined>;
|
|
19
|
+
loadAll(trx?: InMemoryTransaction): Promise<M[]>;
|
|
20
|
+
loadMany(ids: M["id"][], trx?: InMemoryTransaction): Promise<M[]>;
|
|
21
|
+
delete(id: M["id"], trx?: InMemoryTransaction): Promise<void>;
|
|
22
|
+
countAll(): Promise<number>;
|
|
23
|
+
streamAll(): AsyncIterable<M>;
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=in-memory.store.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.store.d.ts","sourceRoot":"","sources":["../../src/store/in-memory.store.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AAC7C,OAAO,EAAE,gBAAgB,EAAE,MAAM,sBAAsB,CAAC;AACxD,OAAO,EAAE,mBAAmB,EAAE,MAAM,0BAA0B,CAAC;AAC/D,OAAO,EAAE,WAAW,EAAa,MAAM,uBAAuB,CAAC;AAC/D;;;GAGG;AACH,qBAAa,aAAa,CAAC,CAAC,SAAS,KAAK,CAAE,YAAW,KAAK,CAAC,CAAC,CAAC;aAE3C,UAAU,EAAE,MAAM;aAClB,QAAQ,EAAE,gBAAgB;aAC1B,UAAU,EAAE,WAAW,CAAC,CAAC,CAAC;gBAF1B,UAAU,EAAE,MAAM,EAClB,QAAQ,EAAE,gBAAgB,EAC1B,UAAU,EAAE,WAAW,CAAC,CAAC,CAAC;IAG5C,OAAO,CAAC,WAAW;cASH,MAAM,CACpB,SAAS,EAAE,CAAC,KAAK,EAAE,CAAC,KAAK,OAAO,EAChC,GAAG,CAAC,EAAE,mBAAmB,GACxB,OAAO,CAAC,CAAC,EAAE,CAAC;IAcf,KAAK;IAIC,IAAI,CAAC,KAAK,EAAE,CAAC,EAAE,GAAG,CAAC,EAAE,mBAAmB,GAAG,OAAO,CAAC,IAAI,CAAC;IASxD,IAAI,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,GAAG,CAAC,EAAE,mBAAmB,GAAG,OAAO,CAAC,CAAC,GAAG,SAAS,CAAC;IAc1E,OAAO,CAAC,GAAG,CAAC,EAAE,mBAAmB,GAAG,OAAO,CAAC,CAAC,EAAE,CAAC;IAW1C,QAAQ,CAAC,GAAG,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,EAAE,GAAG,CAAC,EAAE,mBAAmB,GAAG,OAAO,CAAC,CAAC,EAAE,CAAC;IAKjE,MAAM,CAAC,EAAE,EAAE,CAAC,CAAC,IAAI,CAAC,EAAE,GAAG,CAAC,EAAE,mBAAmB,GAAG,OAAO,CAAC,IAAI,CAAC;IAI7D,QAAQ;IAIP,SAAS,IAAI,aAAa,CAAC,CAAC,CAAC;CAKrC"}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.InMemoryStore = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* This in memory store is a copy store. It stores a copy of the actual model.
|
|
6
|
+
* It is the recommended inmemory store to use, as it reflects more closely the behaviour of a real store.
|
|
7
|
+
*/
|
|
8
|
+
class InMemoryStore {
|
|
9
|
+
collection;
|
|
10
|
+
database;
|
|
11
|
+
serializer;
|
|
12
|
+
constructor(collection, database, serializer) {
|
|
13
|
+
this.collection = collection;
|
|
14
|
+
this.database = database;
|
|
15
|
+
this.serializer = serializer;
|
|
16
|
+
}
|
|
17
|
+
serializeId(id) {
|
|
18
|
+
if (Object.getOwnPropertyNames(id).includes("serialize")) {
|
|
19
|
+
if ("serialize" in id) {
|
|
20
|
+
return id.serialize();
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
return id.toString();
|
|
24
|
+
}
|
|
25
|
+
async filter(predicate, trx) {
|
|
26
|
+
const filtered = await Promise.all(this.database.loadAll(this.collection).map(async (e) => {
|
|
27
|
+
const deserialized = await this.serializer.deserialize(e.data.data);
|
|
28
|
+
if (!predicate(deserialized)) {
|
|
29
|
+
return undefined;
|
|
30
|
+
}
|
|
31
|
+
trx?.transaction.markRead(this.collection, e.id, e.data.savedAt);
|
|
32
|
+
return deserialized;
|
|
33
|
+
}));
|
|
34
|
+
return filtered.filter((e) => Boolean(e));
|
|
35
|
+
}
|
|
36
|
+
clear() {
|
|
37
|
+
this.database.clear(this.collection);
|
|
38
|
+
}
|
|
39
|
+
async save(model, trx) {
|
|
40
|
+
await this.database.save(this.collection, this.serializeId(model.id), await this.serializer.serialize(model), trx?.transaction);
|
|
41
|
+
}
|
|
42
|
+
async load(id, trx) {
|
|
43
|
+
const serialized = await this.database.load(this.collection, id.toString(), trx?.transaction);
|
|
44
|
+
if (!serialized) {
|
|
45
|
+
return undefined;
|
|
46
|
+
}
|
|
47
|
+
return this.serializer.deserialize(serialized);
|
|
48
|
+
}
|
|
49
|
+
loadAll(trx) {
|
|
50
|
+
const serialized = this.database.loadAll(this.collection);
|
|
51
|
+
return Promise.all(serialized.map(async (s) => {
|
|
52
|
+
trx?.transaction.markRead(this.collection, s.id, s.data.savedAt);
|
|
53
|
+
return this.serializer.deserialize(s.data.data);
|
|
54
|
+
}));
|
|
55
|
+
}
|
|
56
|
+
async loadMany(ids, trx) {
|
|
57
|
+
const result = await Promise.all(ids.map((id) => this.load(id, trx)));
|
|
58
|
+
return result.filter((m) => m !== undefined);
|
|
59
|
+
}
|
|
60
|
+
async delete(id, trx) {
|
|
61
|
+
this.database.delete(this.collection, id.toString(), trx?.transaction);
|
|
62
|
+
}
|
|
63
|
+
async countAll() {
|
|
64
|
+
return this.database.countAll(this.collection);
|
|
65
|
+
}
|
|
66
|
+
async *streamAll() {
|
|
67
|
+
for (const item of await this.filter(() => true)) {
|
|
68
|
+
yield item;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
exports.InMemoryStore = InMemoryStore;
|
|
73
|
+
//# sourceMappingURL=in-memory.store.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"in-memory.store.js","sourceRoot":"","sources":["../../src/store/in-memory.store.ts"],"names":[],"mappings":"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"}
|
package/package.json
ADDED
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@ddd-ts/store-inmemory",
|
|
3
|
+
"version": "0.0.0-0.1",
|
|
4
|
+
"main": "dist/index.js",
|
|
5
|
+
"types": "dist/index.d.ts",
|
|
6
|
+
"license": "MIT",
|
|
7
|
+
"scripts": {
|
|
8
|
+
"build": "tsc"
|
|
9
|
+
},
|
|
10
|
+
"dependencies": {
|
|
11
|
+
"@ddd-ts/model": "0.0.0-0.1",
|
|
12
|
+
"@ddd-ts/serialization": "0.0.0-0.1"
|
|
13
|
+
},
|
|
14
|
+
"devDependencies": {
|
|
15
|
+
"@ddd-ts/tools": "0.0.0-0.1",
|
|
16
|
+
"@ddd-ts/types": "0.0.0-0.1",
|
|
17
|
+
"@types/jest": "^29.5.1",
|
|
18
|
+
"@types/node": "^20.12.4"
|
|
19
|
+
}
|
|
20
|
+
}
|