@xyo-network/archivist-storage 2.85.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +165 -0
- package/README.md +13 -0
- package/dist/browser/StorageArchivist.d.cts +39 -0
- package/dist/browser/StorageArchivist.d.cts.map +1 -0
- package/dist/browser/StorageArchivist.d.mts +39 -0
- package/dist/browser/StorageArchivist.d.mts.map +1 -0
- package/dist/browser/StorageArchivist.d.ts +39 -0
- package/dist/browser/StorageArchivist.d.ts.map +1 -0
- package/dist/browser/index.cjs +178 -0
- package/dist/browser/index.cjs.map +1 -0
- package/dist/browser/index.d.cts +2 -0
- package/dist/browser/index.d.cts.map +1 -0
- package/dist/browser/index.d.mts +2 -0
- package/dist/browser/index.d.mts.map +1 -0
- package/dist/browser/index.d.ts +2 -0
- package/dist/browser/index.d.ts.map +1 -0
- package/dist/browser/index.js +147 -0
- package/dist/browser/index.js.map +1 -0
- package/dist/node/StorageArchivist.d.cts +39 -0
- package/dist/node/StorageArchivist.d.cts.map +1 -0
- package/dist/node/StorageArchivist.d.mts +39 -0
- package/dist/node/StorageArchivist.d.mts.map +1 -0
- package/dist/node/StorageArchivist.d.ts +39 -0
- package/dist/node/StorageArchivist.d.ts.map +1 -0
- package/dist/node/index.cjs +197 -0
- package/dist/node/index.cjs.map +1 -0
- package/dist/node/index.d.cts +2 -0
- package/dist/node/index.d.cts.map +1 -0
- package/dist/node/index.d.mts +2 -0
- package/dist/node/index.d.mts.map +1 -0
- package/dist/node/index.d.ts +2 -0
- package/dist/node/index.d.ts.map +1 -0
- package/dist/node/index.js +161 -0
- package/dist/node/index.js.map +1 -0
- package/package.json +77 -0
- package/src/StorageArchivist.ts +191 -0
- package/src/index.ts +1 -0
- package/src/spec/testArchivist.ts +50 -0
- package/typedoc.json +5 -0
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
var __defProp = Object.defineProperty;
|
|
2
|
+
var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
|
|
3
|
+
var __name = (target, value) => __defProp(target, "name", { value, configurable: true });
|
|
4
|
+
var __publicField = (obj, key, value) => {
|
|
5
|
+
__defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
|
|
6
|
+
return value;
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
// src/StorageArchivist.ts
|
|
10
|
+
import { assertEx } from "@xylabs/assert";
|
|
11
|
+
import { compact } from "@xylabs/lodash";
|
|
12
|
+
import { fulfilled } from "@xylabs/promise";
|
|
13
|
+
import { AbstractArchivist } from "@xyo-network/archivist-abstract";
|
|
14
|
+
import { ArchivistAllQuerySchema, ArchivistClearQuerySchema, ArchivistCommitQuerySchema, ArchivistDeleteQuerySchema, ArchivistInsertQuerySchema } from "@xyo-network/archivist-model";
|
|
15
|
+
import { PayloadHasher } from "@xyo-network/hash";
|
|
16
|
+
import { PayloadWrapper } from "@xyo-network/payload-wrapper";
|
|
17
|
+
import store from "store2";
|
|
18
|
+
var StorageArchivistConfigSchema = "network.xyo.archivist.storage.config";
|
|
19
|
+
var _StorageArchivist = class _StorageArchivist extends AbstractArchivist {
|
|
20
|
+
_privateStorage;
|
|
21
|
+
_storage;
|
|
22
|
+
get maxEntries() {
|
|
23
|
+
var _a;
|
|
24
|
+
return ((_a = this.config) == null ? void 0 : _a.maxEntries) ?? 1e3;
|
|
25
|
+
}
|
|
26
|
+
get maxEntrySize() {
|
|
27
|
+
var _a;
|
|
28
|
+
return ((_a = this.config) == null ? void 0 : _a.maxEntrySize) ?? 16e3;
|
|
29
|
+
}
|
|
30
|
+
get namespace() {
|
|
31
|
+
var _a;
|
|
32
|
+
return ((_a = this.config) == null ? void 0 : _a.namespace) ?? "xyo-archivist";
|
|
33
|
+
}
|
|
34
|
+
get persistAccount() {
|
|
35
|
+
var _a;
|
|
36
|
+
return ((_a = this.config) == null ? void 0 : _a.persistAccount) ?? false;
|
|
37
|
+
}
|
|
38
|
+
get queries() {
|
|
39
|
+
return [
|
|
40
|
+
ArchivistAllQuerySchema,
|
|
41
|
+
ArchivistDeleteQuerySchema,
|
|
42
|
+
ArchivistClearQuerySchema,
|
|
43
|
+
ArchivistInsertQuerySchema,
|
|
44
|
+
ArchivistCommitQuerySchema,
|
|
45
|
+
...super.queries
|
|
46
|
+
];
|
|
47
|
+
}
|
|
48
|
+
get type() {
|
|
49
|
+
var _a;
|
|
50
|
+
return ((_a = this.config) == null ? void 0 : _a.type) ?? "local";
|
|
51
|
+
}
|
|
52
|
+
/* This has to be a getter so that it can access it during construction */
|
|
53
|
+
get privateStorage() {
|
|
54
|
+
this._privateStorage = this._storage ?? store[this.type].namespace(`${this.namespace}|private`);
|
|
55
|
+
return this._privateStorage;
|
|
56
|
+
}
|
|
57
|
+
/* This has to be a getter so that it can access it during construction */
|
|
58
|
+
get storage() {
|
|
59
|
+
this._storage = this._storage ?? store[this.type].namespace(this.namespace);
|
|
60
|
+
return this._storage;
|
|
61
|
+
}
|
|
62
|
+
/*override async loadAccount(account?: AccountInstance, persistAccount?: boolean, privateStorage?: StoreBase, _logger?: Logger) {
|
|
63
|
+
if (!this._account) {
|
|
64
|
+
if (persistAccount) {
|
|
65
|
+
const privateKey = privateStorage?.get('privateKey')
|
|
66
|
+
if (privateKey) {
|
|
67
|
+
try {
|
|
68
|
+
this._account = await Account.create({ privateKey })
|
|
69
|
+
return this._account
|
|
70
|
+
} catch (ex) {
|
|
71
|
+
console.error(`Error reading Account from storage [${ex}] - Recreating Account`)
|
|
72
|
+
privateStorage?.remove('privateKey')
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
return await super.loadAccount()
|
|
78
|
+
}*/
|
|
79
|
+
allHandler() {
|
|
80
|
+
var _a;
|
|
81
|
+
(_a = this.logger) == null ? void 0 : _a.log(`this.storage.length: ${this.storage.length}`);
|
|
82
|
+
return Object.entries(this.storage.getAll()).map(([, value]) => value);
|
|
83
|
+
}
|
|
84
|
+
clearHandler() {
|
|
85
|
+
var _a;
|
|
86
|
+
(_a = this.logger) == null ? void 0 : _a.log(`this.storage.length: ${this.storage.length}`);
|
|
87
|
+
this.storage.clear();
|
|
88
|
+
return this.emit("cleared", {
|
|
89
|
+
module: this
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
async commitHandler() {
|
|
93
|
+
var _a, _b;
|
|
94
|
+
(_a = this.logger) == null ? void 0 : _a.log(`this.storage.length: ${this.storage.length}`);
|
|
95
|
+
const payloads = await this.all();
|
|
96
|
+
assertEx(payloads.length > 0, "Nothing to commit");
|
|
97
|
+
const settled = await Promise.allSettled(compact((_b = Object.values((await this.parents()).commit ?? [])) == null ? void 0 : _b.map(async (parent) => {
|
|
98
|
+
var _a2;
|
|
99
|
+
const queryPayload = {
|
|
100
|
+
schema: ArchivistInsertQuerySchema
|
|
101
|
+
};
|
|
102
|
+
const query = await this.bindQuery(queryPayload, payloads);
|
|
103
|
+
return (_a2 = await (parent == null ? void 0 : parent.query(query[0], query[1]))) == null ? void 0 : _a2[0];
|
|
104
|
+
})));
|
|
105
|
+
await this.clear();
|
|
106
|
+
return compact(settled.filter(fulfilled).map((result) => result.value));
|
|
107
|
+
}
|
|
108
|
+
async deleteHandler(hashes) {
|
|
109
|
+
const payloadPairs = await Promise.all((await this.get(hashes)).map(async (payload) => [
|
|
110
|
+
await PayloadHasher.hashAsync(payload),
|
|
111
|
+
payload
|
|
112
|
+
]));
|
|
113
|
+
const deletedPairs = compact(await Promise.all(payloadPairs.map(([hash, payload]) => {
|
|
114
|
+
this.storage.remove(hash);
|
|
115
|
+
return [
|
|
116
|
+
hash,
|
|
117
|
+
payload
|
|
118
|
+
];
|
|
119
|
+
})));
|
|
120
|
+
return deletedPairs.map(([hash]) => hash);
|
|
121
|
+
}
|
|
122
|
+
getHandler(hashes) {
|
|
123
|
+
return compact(hashes.map((hash) => {
|
|
124
|
+
return this.storage.get(hash);
|
|
125
|
+
}));
|
|
126
|
+
}
|
|
127
|
+
async insertHandler(payloads) {
|
|
128
|
+
const resultPayloads = await Promise.all(payloads.map(async (payload) => {
|
|
129
|
+
const wrapper = PayloadWrapper.wrap(payload);
|
|
130
|
+
const hash = await wrapper.hashAsync();
|
|
131
|
+
const value = JSON.stringify(wrapper.payload());
|
|
132
|
+
assertEx(value.length < this.maxEntrySize, `Payload too large [${hash}, ${value.length}]`);
|
|
133
|
+
this.storage.set(hash, wrapper.payload());
|
|
134
|
+
return wrapper.payload();
|
|
135
|
+
}));
|
|
136
|
+
return resultPayloads;
|
|
137
|
+
}
|
|
138
|
+
saveAccount() {
|
|
139
|
+
var _a;
|
|
140
|
+
if (this.persistAccount) {
|
|
141
|
+
const account = this.account;
|
|
142
|
+
(_a = this.logger) == null ? void 0 : _a.log(account.address);
|
|
143
|
+
this.privateStorage.set("privateKey", account.private.hex);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
async startHandler() {
|
|
147
|
+
await super.startHandler();
|
|
148
|
+
this.saveAccount();
|
|
149
|
+
return true;
|
|
150
|
+
}
|
|
151
|
+
};
|
|
152
|
+
__name(_StorageArchivist, "StorageArchivist");
|
|
153
|
+
__publicField(_StorageArchivist, "configSchemas", [
|
|
154
|
+
StorageArchivistConfigSchema
|
|
155
|
+
]);
|
|
156
|
+
var StorageArchivist = _StorageArchivist;
|
|
157
|
+
export {
|
|
158
|
+
StorageArchivist,
|
|
159
|
+
StorageArchivistConfigSchema
|
|
160
|
+
};
|
|
161
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/StorageArchivist.ts"],"sourcesContent":["import { assertEx } from '@xylabs/assert'\nimport { compact } from '@xylabs/lodash'\nimport { fulfilled, Promisable, PromisableArray } from '@xylabs/promise'\nimport { AbstractArchivist } from '@xyo-network/archivist-abstract'\nimport {\n ArchivistAllQuerySchema,\n ArchivistClearQuerySchema,\n ArchivistCommitQuerySchema,\n ArchivistConfig,\n ArchivistDeleteQuerySchema,\n ArchivistInsertQuery,\n ArchivistInsertQuerySchema,\n ArchivistInstance,\n ArchivistModuleEventData,\n ArchivistParams,\n} from '@xyo-network/archivist-model'\nimport { BoundWitness } from '@xyo-network/boundwitness-model'\nimport { PayloadHasher } from '@xyo-network/hash'\nimport { AnyConfigSchema } from '@xyo-network/module-model'\nimport { Payload } from '@xyo-network/payload-model'\nimport { PayloadWrapper } from '@xyo-network/payload-wrapper'\nimport store, { StoreBase } from 'store2'\n\nexport type StorageArchivistConfigSchema = 'network.xyo.archivist.storage.config'\nexport const StorageArchivistConfigSchema: StorageArchivistConfigSchema = 'network.xyo.archivist.storage.config'\n\nexport type StorageArchivistConfig = ArchivistConfig<{\n maxEntries?: number\n maxEntrySize?: number\n namespace?: string\n persistAccount?: boolean\n schema: StorageArchivistConfigSchema\n type?: 'local' | 'session' | 'page'\n}>\n\nexport type StorageArchivistParams = ArchivistParams<AnyConfigSchema<StorageArchivistConfig>>\nexport class StorageArchivist<\n TParams extends StorageArchivistParams = StorageArchivistParams,\n TEventData extends ArchivistModuleEventData = ArchivistModuleEventData,\n >\n extends AbstractArchivist<TParams, TEventData>\n implements ArchivistInstance\n{\n static override configSchemas = [StorageArchivistConfigSchema]\n\n private _privateStorage: StoreBase | undefined\n private _storage: StoreBase | undefined\n\n get maxEntries() {\n return this.config?.maxEntries ?? 1000\n }\n\n get maxEntrySize() {\n return this.config?.maxEntrySize ?? 16_000\n }\n\n get namespace() {\n return this.config?.namespace ?? 'xyo-archivist'\n }\n\n get persistAccount() {\n return this.config?.persistAccount ?? false\n }\n\n override get queries(): string[] {\n return [\n ArchivistAllQuerySchema,\n ArchivistDeleteQuerySchema,\n ArchivistClearQuerySchema,\n ArchivistInsertQuerySchema,\n ArchivistCommitQuerySchema,\n ...super.queries,\n ]\n }\n\n get type() {\n return this.config?.type ?? 'local'\n }\n\n /* This has to be a getter so that it can access it during construction */\n private get privateStorage(): StoreBase {\n this._privateStorage = this._storage ?? store[this.type].namespace(`${this.namespace}|private`)\n return this._privateStorage\n }\n\n /* This has to be a getter so that it can access it during construction */\n private get storage(): StoreBase {\n this._storage = this._storage ?? store[this.type].namespace(this.namespace)\n return this._storage\n }\n\n /*override async loadAccount(account?: AccountInstance, persistAccount?: boolean, privateStorage?: StoreBase, _logger?: Logger) {\n if (!this._account) {\n if (persistAccount) {\n const privateKey = privateStorage?.get('privateKey')\n if (privateKey) {\n try {\n this._account = await Account.create({ privateKey })\n return this._account\n } catch (ex) {\n console.error(`Error reading Account from storage [${ex}] - Recreating Account`)\n privateStorage?.remove('privateKey')\n }\n }\n }\n }\n return await super.loadAccount()\n }*/\n\n protected override allHandler(): PromisableArray<Payload> {\n this.logger?.log(`this.storage.length: ${this.storage.length}`)\n return Object.entries(this.storage.getAll()).map(([, value]) => value)\n }\n\n protected override clearHandler(): void | Promise<void> {\n this.logger?.log(`this.storage.length: ${this.storage.length}`)\n this.storage.clear()\n return this.emit('cleared', { module: this })\n }\n\n protected override async commitHandler(): Promise<BoundWitness[]> {\n this.logger?.log(`this.storage.length: ${this.storage.length}`)\n const payloads = await this.all()\n assertEx(payloads.length > 0, 'Nothing to commit')\n const settled = await Promise.allSettled(\n compact(\n Object.values((await this.parents()).commit ?? [])?.map(async (parent) => {\n const queryPayload: ArchivistInsertQuery = {\n schema: ArchivistInsertQuerySchema,\n }\n const query = await this.bindQuery(queryPayload, payloads)\n return (await parent?.query(query[0], query[1]))?.[0]\n }),\n ),\n )\n // TODO - rather than clear, delete the payloads that come back as successfully inserted\n await this.clear()\n return compact(settled.filter(fulfilled).map((result) => result.value))\n }\n\n protected override async deleteHandler(hashes: string[]): Promise<string[]> {\n const payloadPairs: [string, Payload][] = await Promise.all(\n (await this.get(hashes)).map<Promise<[string, Payload]>>(async (payload) => [await PayloadHasher.hashAsync(payload), payload]),\n )\n const deletedPairs: [string, Payload][] = compact(\n await Promise.all(\n payloadPairs.map<[string, Payload] | undefined>(([hash, payload]) => {\n this.storage.remove(hash)\n return [hash, payload]\n }),\n ),\n )\n return deletedPairs.map(([hash]) => hash)\n }\n\n protected override getHandler(hashes: string[]): Promisable<Payload[]> {\n return compact(\n hashes.map((hash) => {\n return this.storage.get(hash)\n }),\n )\n }\n\n protected override async insertHandler(payloads: Payload[]): Promise<Payload[]> {\n const resultPayloads = await Promise.all(\n payloads.map(async (payload) => {\n const wrapper = PayloadWrapper.wrap(payload)\n const hash = await wrapper.hashAsync()\n const value = JSON.stringify(wrapper.payload())\n assertEx(value.length < this.maxEntrySize, `Payload too large [${hash}, ${value.length}]`)\n this.storage.set(hash, wrapper.payload())\n return wrapper.payload()\n }),\n )\n return resultPayloads\n }\n\n protected saveAccount() {\n if (this.persistAccount) {\n const account = this.account\n this.logger?.log(account.address)\n this.privateStorage.set('privateKey', account.private.hex)\n }\n }\n\n protected override async startHandler() {\n await super.startHandler()\n this.saveAccount()\n return true\n }\n}\n"],"mappings":"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","names":["assertEx","compact","fulfilled","AbstractArchivist","ArchivistAllQuerySchema","ArchivistClearQuerySchema","ArchivistCommitQuerySchema","ArchivistDeleteQuerySchema","ArchivistInsertQuerySchema","PayloadHasher","PayloadWrapper","store","StorageArchivistConfigSchema","StorageArchivist","AbstractArchivist","_privateStorage","_storage","maxEntries","config","maxEntrySize","namespace","persistAccount","queries","ArchivistAllQuerySchema","ArchivistDeleteQuerySchema","ArchivistClearQuerySchema","ArchivistInsertQuerySchema","ArchivistCommitQuerySchema","type","privateStorage","store","storage","allHandler","logger","log","length","Object","entries","getAll","map","value","clearHandler","clear","emit","module","commitHandler","payloads","all","assertEx","settled","Promise","allSettled","compact","values","parents","commit","parent","_a","queryPayload","schema","query","bindQuery","filter","fulfilled","result","deleteHandler","hashes","payloadPairs","get","payload","PayloadHasher","hashAsync","deletedPairs","hash","remove","getHandler","insertHandler","resultPayloads","wrapper","PayloadWrapper","wrap","JSON","stringify","set","saveAccount","account","address","private","hex","startHandler","configSchemas"]}
|
package/package.json
ADDED
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@xyo-network/archivist-storage",
|
|
3
|
+
"author": {
|
|
4
|
+
"email": "support@xyo.network",
|
|
5
|
+
"name": "XYO Development Team",
|
|
6
|
+
"url": "https://xyo.network"
|
|
7
|
+
},
|
|
8
|
+
"bugs": {
|
|
9
|
+
"email": "support@xyo.network",
|
|
10
|
+
"url": "https://github.com/XYOracleNetwork/sdk-xyo-client-js/issues"
|
|
11
|
+
},
|
|
12
|
+
"dependencies": {
|
|
13
|
+
"@xylabs/assert": "^2.13.23",
|
|
14
|
+
"@xylabs/delay": "^2.13.23",
|
|
15
|
+
"@xylabs/lodash": "^2.13.23",
|
|
16
|
+
"@xylabs/promise": "^2.13.23",
|
|
17
|
+
"@xyo-network/archivist-abstract": "~2.85.6",
|
|
18
|
+
"@xyo-network/archivist-model": "~2.85.6",
|
|
19
|
+
"@xyo-network/boundwitness-model": "~2.85.6",
|
|
20
|
+
"@xyo-network/hash": "~2.85.6",
|
|
21
|
+
"@xyo-network/id-payload-plugin": "~2.85.6",
|
|
22
|
+
"@xyo-network/module-model": "~2.85.6",
|
|
23
|
+
"@xyo-network/payload-model": "~2.85.6",
|
|
24
|
+
"@xyo-network/payload-wrapper": "~2.85.6",
|
|
25
|
+
"store2": "^2.14.2"
|
|
26
|
+
},
|
|
27
|
+
"devDependencies": {
|
|
28
|
+
"@xylabs/ts-scripts-yarn3": "^3.2.25",
|
|
29
|
+
"@xylabs/tsconfig": "^3.2.25",
|
|
30
|
+
"@xyo-network/account": "~2.85.6",
|
|
31
|
+
"@xyo-network/archivist-memory": "~2.85.6",
|
|
32
|
+
"@xyo-network/boundwitness-builder": "~2.85.6",
|
|
33
|
+
"@xyo-network/node-memory": "~2.85.6",
|
|
34
|
+
"typescript": "^5.3.3"
|
|
35
|
+
},
|
|
36
|
+
"description": "Primary SDK for using XYO Protocol 2.0",
|
|
37
|
+
"types": "dist/node/index.d.ts",
|
|
38
|
+
"exports": {
|
|
39
|
+
".": {
|
|
40
|
+
"browser": {
|
|
41
|
+
"require": {
|
|
42
|
+
"types": "./dist/browser/index.d.cts",
|
|
43
|
+
"default": "./dist/browser/index.cjs"
|
|
44
|
+
},
|
|
45
|
+
"import": {
|
|
46
|
+
"types": "./dist/browser/index.d.mts",
|
|
47
|
+
"default": "./dist/browser/index.js"
|
|
48
|
+
}
|
|
49
|
+
},
|
|
50
|
+
"node": {
|
|
51
|
+
"require": {
|
|
52
|
+
"types": "./dist/node/index.d.cts",
|
|
53
|
+
"default": "./dist/node/index.cjs"
|
|
54
|
+
},
|
|
55
|
+
"import": {
|
|
56
|
+
"types": "./dist/node/index.d.mts",
|
|
57
|
+
"default": "./dist/node/index.js"
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
},
|
|
61
|
+
"./package.json": "./package.json"
|
|
62
|
+
},
|
|
63
|
+
"main": "dist/node/index.cjs",
|
|
64
|
+
"module": "dist/node/index.js",
|
|
65
|
+
"homepage": "https://xyo.network",
|
|
66
|
+
"license": "LGPL-3.0-only",
|
|
67
|
+
"publishConfig": {
|
|
68
|
+
"access": "public"
|
|
69
|
+
},
|
|
70
|
+
"repository": {
|
|
71
|
+
"type": "git",
|
|
72
|
+
"url": "https://github.com/XYOracleNetwork/sdk-xyo-client-js.git"
|
|
73
|
+
},
|
|
74
|
+
"sideEffects": false,
|
|
75
|
+
"version": "2.85.6",
|
|
76
|
+
"type": "module"
|
|
77
|
+
}
|
|
@@ -0,0 +1,191 @@
|
|
|
1
|
+
import { assertEx } from '@xylabs/assert'
|
|
2
|
+
import { compact } from '@xylabs/lodash'
|
|
3
|
+
import { fulfilled, Promisable, PromisableArray } from '@xylabs/promise'
|
|
4
|
+
import { AbstractArchivist } from '@xyo-network/archivist-abstract'
|
|
5
|
+
import {
|
|
6
|
+
ArchivistAllQuerySchema,
|
|
7
|
+
ArchivistClearQuerySchema,
|
|
8
|
+
ArchivistCommitQuerySchema,
|
|
9
|
+
ArchivistConfig,
|
|
10
|
+
ArchivistDeleteQuerySchema,
|
|
11
|
+
ArchivistInsertQuery,
|
|
12
|
+
ArchivistInsertQuerySchema,
|
|
13
|
+
ArchivistInstance,
|
|
14
|
+
ArchivistModuleEventData,
|
|
15
|
+
ArchivistParams,
|
|
16
|
+
} from '@xyo-network/archivist-model'
|
|
17
|
+
import { BoundWitness } from '@xyo-network/boundwitness-model'
|
|
18
|
+
import { PayloadHasher } from '@xyo-network/hash'
|
|
19
|
+
import { AnyConfigSchema } from '@xyo-network/module-model'
|
|
20
|
+
import { Payload } from '@xyo-network/payload-model'
|
|
21
|
+
import { PayloadWrapper } from '@xyo-network/payload-wrapper'
|
|
22
|
+
import store, { StoreBase } from 'store2'
|
|
23
|
+
|
|
24
|
+
export type StorageArchivistConfigSchema = 'network.xyo.archivist.storage.config'
|
|
25
|
+
export const StorageArchivistConfigSchema: StorageArchivistConfigSchema = 'network.xyo.archivist.storage.config'
|
|
26
|
+
|
|
27
|
+
export type StorageArchivistConfig = ArchivistConfig<{
|
|
28
|
+
maxEntries?: number
|
|
29
|
+
maxEntrySize?: number
|
|
30
|
+
namespace?: string
|
|
31
|
+
persistAccount?: boolean
|
|
32
|
+
schema: StorageArchivistConfigSchema
|
|
33
|
+
type?: 'local' | 'session' | 'page'
|
|
34
|
+
}>
|
|
35
|
+
|
|
36
|
+
export type StorageArchivistParams = ArchivistParams<AnyConfigSchema<StorageArchivistConfig>>
|
|
37
|
+
export class StorageArchivist<
|
|
38
|
+
TParams extends StorageArchivistParams = StorageArchivistParams,
|
|
39
|
+
TEventData extends ArchivistModuleEventData = ArchivistModuleEventData,
|
|
40
|
+
>
|
|
41
|
+
extends AbstractArchivist<TParams, TEventData>
|
|
42
|
+
implements ArchivistInstance
|
|
43
|
+
{
|
|
44
|
+
static override configSchemas = [StorageArchivistConfigSchema]
|
|
45
|
+
|
|
46
|
+
private _privateStorage: StoreBase | undefined
|
|
47
|
+
private _storage: StoreBase | undefined
|
|
48
|
+
|
|
49
|
+
get maxEntries() {
|
|
50
|
+
return this.config?.maxEntries ?? 1000
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
get maxEntrySize() {
|
|
54
|
+
return this.config?.maxEntrySize ?? 16_000
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
get namespace() {
|
|
58
|
+
return this.config?.namespace ?? 'xyo-archivist'
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
get persistAccount() {
|
|
62
|
+
return this.config?.persistAccount ?? false
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
override get queries(): string[] {
|
|
66
|
+
return [
|
|
67
|
+
ArchivistAllQuerySchema,
|
|
68
|
+
ArchivistDeleteQuerySchema,
|
|
69
|
+
ArchivistClearQuerySchema,
|
|
70
|
+
ArchivistInsertQuerySchema,
|
|
71
|
+
ArchivistCommitQuerySchema,
|
|
72
|
+
...super.queries,
|
|
73
|
+
]
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
get type() {
|
|
77
|
+
return this.config?.type ?? 'local'
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
/* This has to be a getter so that it can access it during construction */
|
|
81
|
+
private get privateStorage(): StoreBase {
|
|
82
|
+
this._privateStorage = this._storage ?? store[this.type].namespace(`${this.namespace}|private`)
|
|
83
|
+
return this._privateStorage
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
/* This has to be a getter so that it can access it during construction */
|
|
87
|
+
private get storage(): StoreBase {
|
|
88
|
+
this._storage = this._storage ?? store[this.type].namespace(this.namespace)
|
|
89
|
+
return this._storage
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
/*override async loadAccount(account?: AccountInstance, persistAccount?: boolean, privateStorage?: StoreBase, _logger?: Logger) {
|
|
93
|
+
if (!this._account) {
|
|
94
|
+
if (persistAccount) {
|
|
95
|
+
const privateKey = privateStorage?.get('privateKey')
|
|
96
|
+
if (privateKey) {
|
|
97
|
+
try {
|
|
98
|
+
this._account = await Account.create({ privateKey })
|
|
99
|
+
return this._account
|
|
100
|
+
} catch (ex) {
|
|
101
|
+
console.error(`Error reading Account from storage [${ex}] - Recreating Account`)
|
|
102
|
+
privateStorage?.remove('privateKey')
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
return await super.loadAccount()
|
|
108
|
+
}*/
|
|
109
|
+
|
|
110
|
+
protected override allHandler(): PromisableArray<Payload> {
|
|
111
|
+
this.logger?.log(`this.storage.length: ${this.storage.length}`)
|
|
112
|
+
return Object.entries(this.storage.getAll()).map(([, value]) => value)
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
protected override clearHandler(): void | Promise<void> {
|
|
116
|
+
this.logger?.log(`this.storage.length: ${this.storage.length}`)
|
|
117
|
+
this.storage.clear()
|
|
118
|
+
return this.emit('cleared', { module: this })
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
protected override async commitHandler(): Promise<BoundWitness[]> {
|
|
122
|
+
this.logger?.log(`this.storage.length: ${this.storage.length}`)
|
|
123
|
+
const payloads = await this.all()
|
|
124
|
+
assertEx(payloads.length > 0, 'Nothing to commit')
|
|
125
|
+
const settled = await Promise.allSettled(
|
|
126
|
+
compact(
|
|
127
|
+
Object.values((await this.parents()).commit ?? [])?.map(async (parent) => {
|
|
128
|
+
const queryPayload: ArchivistInsertQuery = {
|
|
129
|
+
schema: ArchivistInsertQuerySchema,
|
|
130
|
+
}
|
|
131
|
+
const query = await this.bindQuery(queryPayload, payloads)
|
|
132
|
+
return (await parent?.query(query[0], query[1]))?.[0]
|
|
133
|
+
}),
|
|
134
|
+
),
|
|
135
|
+
)
|
|
136
|
+
// TODO - rather than clear, delete the payloads that come back as successfully inserted
|
|
137
|
+
await this.clear()
|
|
138
|
+
return compact(settled.filter(fulfilled).map((result) => result.value))
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
protected override async deleteHandler(hashes: string[]): Promise<string[]> {
|
|
142
|
+
const payloadPairs: [string, Payload][] = await Promise.all(
|
|
143
|
+
(await this.get(hashes)).map<Promise<[string, Payload]>>(async (payload) => [await PayloadHasher.hashAsync(payload), payload]),
|
|
144
|
+
)
|
|
145
|
+
const deletedPairs: [string, Payload][] = compact(
|
|
146
|
+
await Promise.all(
|
|
147
|
+
payloadPairs.map<[string, Payload] | undefined>(([hash, payload]) => {
|
|
148
|
+
this.storage.remove(hash)
|
|
149
|
+
return [hash, payload]
|
|
150
|
+
}),
|
|
151
|
+
),
|
|
152
|
+
)
|
|
153
|
+
return deletedPairs.map(([hash]) => hash)
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
protected override getHandler(hashes: string[]): Promisable<Payload[]> {
|
|
157
|
+
return compact(
|
|
158
|
+
hashes.map((hash) => {
|
|
159
|
+
return this.storage.get(hash)
|
|
160
|
+
}),
|
|
161
|
+
)
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
protected override async insertHandler(payloads: Payload[]): Promise<Payload[]> {
|
|
165
|
+
const resultPayloads = await Promise.all(
|
|
166
|
+
payloads.map(async (payload) => {
|
|
167
|
+
const wrapper = PayloadWrapper.wrap(payload)
|
|
168
|
+
const hash = await wrapper.hashAsync()
|
|
169
|
+
const value = JSON.stringify(wrapper.payload())
|
|
170
|
+
assertEx(value.length < this.maxEntrySize, `Payload too large [${hash}, ${value.length}]`)
|
|
171
|
+
this.storage.set(hash, wrapper.payload())
|
|
172
|
+
return wrapper.payload()
|
|
173
|
+
}),
|
|
174
|
+
)
|
|
175
|
+
return resultPayloads
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
protected saveAccount() {
|
|
179
|
+
if (this.persistAccount) {
|
|
180
|
+
const account = this.account
|
|
181
|
+
this.logger?.log(account.address)
|
|
182
|
+
this.privateStorage.set('privateKey', account.private.hex)
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
protected override async startHandler() {
|
|
187
|
+
await super.startHandler()
|
|
188
|
+
this.saveAccount()
|
|
189
|
+
return true
|
|
190
|
+
}
|
|
191
|
+
}
|
package/src/index.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './StorageArchivist'
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @jest-environment jsdom
|
|
3
|
+
*/
|
|
4
|
+
|
|
5
|
+
import { delay } from '@xylabs/delay'
|
|
6
|
+
import { Promisable } from '@xylabs/promise'
|
|
7
|
+
import { ArchivistInstance } from '@xyo-network/archivist-model'
|
|
8
|
+
import { IdSchema } from '@xyo-network/id-payload-plugin'
|
|
9
|
+
import { Payload } from '@xyo-network/payload-model'
|
|
10
|
+
import { PayloadWrapper } from '@xyo-network/payload-wrapper'
|
|
11
|
+
|
|
12
|
+
export const testArchivistRoundTrip = (archivistPromise: Promisable<ArchivistInstance>, name: string) => {
|
|
13
|
+
test(`Archivist RoundTrip [${name}]`, async () => {
|
|
14
|
+
const idPayload: Payload<{ salt: string }> = {
|
|
15
|
+
salt: Date.now().toString(),
|
|
16
|
+
schema: IdSchema,
|
|
17
|
+
}
|
|
18
|
+
const payloadWrapper = PayloadWrapper.wrap(idPayload)
|
|
19
|
+
|
|
20
|
+
const archivist = await archivistPromise
|
|
21
|
+
const insertResult = await archivist.insert([idPayload])
|
|
22
|
+
expect(insertResult).toBeDefined()
|
|
23
|
+
|
|
24
|
+
const getResult = await archivist.get([await payloadWrapper.hashAsync()])
|
|
25
|
+
expect(getResult).toBeDefined()
|
|
26
|
+
expect(getResult.length).toBe(1)
|
|
27
|
+
const gottenPayload = getResult[0]
|
|
28
|
+
if (gottenPayload) {
|
|
29
|
+
const gottenPayloadWrapper = PayloadWrapper.wrap(gottenPayload)
|
|
30
|
+
expect(await gottenPayloadWrapper.hashAsync()).toBe(await payloadWrapper.hashAsync())
|
|
31
|
+
}
|
|
32
|
+
})
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export const testArchivistAll = (archivist: Promisable<ArchivistInstance>, name: string) => {
|
|
36
|
+
test(`Archivist All [${name}]`, async () => {
|
|
37
|
+
const idPayload = {
|
|
38
|
+
salt: Date.now().toString(),
|
|
39
|
+
schema: IdSchema,
|
|
40
|
+
}
|
|
41
|
+
const archivistModule = await archivist
|
|
42
|
+
for (let x = 0; x < 10; x++) {
|
|
43
|
+
await archivistModule.insert([idPayload])
|
|
44
|
+
await delay(10)
|
|
45
|
+
}
|
|
46
|
+
const getResult = await archivistModule.all?.()
|
|
47
|
+
expect(getResult).toBeDefined()
|
|
48
|
+
expect(getResult?.length).toBe(2)
|
|
49
|
+
})
|
|
50
|
+
}
|