@bsv/wallet-toolbox 2.1.9 → 2.1.10
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 +11 -0
- package/docs/client.md +305 -156
- package/docs/monitor.md +153 -16
- package/docs/setup.md +4 -8
- package/docs/storage.md +37 -8
- package/docs/wallet.md +305 -156
- package/out/src/Setup.d.ts.map +1 -1
- package/out/src/Setup.js +1 -2
- package/out/src/Setup.js.map +1 -1
- package/out/src/SetupClient.d.ts.map +1 -1
- package/out/src/SetupClient.js +1 -2
- package/out/src/SetupClient.js.map +1 -1
- package/out/src/index.all.d.ts +1 -2
- package/out/src/index.all.d.ts.map +1 -1
- package/out/src/index.all.js +1 -2
- package/out/src/index.all.js.map +1 -1
- package/out/src/monitor/Monitor.d.ts +4 -3
- package/out/src/monitor/Monitor.d.ts.map +1 -1
- package/out/src/monitor/Monitor.js +39 -11
- package/out/src/monitor/Monitor.js.map +1 -1
- package/out/src/monitor/MonitorDaemon.d.ts +2 -1
- package/out/src/monitor/MonitorDaemon.d.ts.map +1 -1
- package/out/src/monitor/MonitorDaemon.js +1 -4
- package/out/src/monitor/MonitorDaemon.js.map +1 -1
- package/out/src/monitor/index.all.d.ts +4 -0
- package/out/src/monitor/index.all.d.ts.map +1 -0
- package/out/src/monitor/index.all.js +43 -0
- package/out/src/monitor/index.all.js.map +1 -0
- package/out/src/monitor/tasks/TaskReviewDoubleSpends.d.ts +26 -0
- package/out/src/monitor/tasks/TaskReviewDoubleSpends.d.ts.map +1 -0
- package/out/src/monitor/tasks/TaskReviewDoubleSpends.js +124 -0
- package/out/src/monitor/tasks/TaskReviewDoubleSpends.js.map +1 -0
- package/out/src/monitor/tasks/TaskReviewProvenTxs.d.ts +34 -0
- package/out/src/monitor/tasks/TaskReviewProvenTxs.d.ts.map +1 -0
- package/out/src/monitor/tasks/TaskReviewProvenTxs.js +131 -0
- package/out/src/monitor/tasks/TaskReviewProvenTxs.js.map +1 -0
- package/out/src/monitor/tasks/TaskReviewUtxos.d.ts +23 -0
- package/out/src/monitor/tasks/TaskReviewUtxos.d.ts.map +1 -0
- package/out/src/monitor/tasks/TaskReviewUtxos.js +71 -0
- package/out/src/monitor/tasks/TaskReviewUtxos.js.map +1 -0
- package/out/src/monitor/tasks/TaskSendWaiting.d.ts +14 -1
- package/out/src/monitor/tasks/TaskSendWaiting.d.ts.map +1 -1
- package/out/src/monitor/tasks/TaskSendWaiting.js +86 -20
- package/out/src/monitor/tasks/TaskSendWaiting.js.map +1 -1
- package/out/src/monitor/tasks/__tests/TaskReviewDoubleSpends.test.d.ts +2 -0
- package/out/src/monitor/tasks/__tests/TaskReviewDoubleSpends.test.d.ts.map +1 -0
- package/out/src/monitor/tasks/__tests/TaskReviewDoubleSpends.test.js +161 -0
- package/out/src/monitor/tasks/__tests/TaskReviewDoubleSpends.test.js.map +1 -0
- package/out/src/monitor/tasks/__tests/TaskReviewProvenTxs.test.d.ts +2 -0
- package/out/src/monitor/tasks/__tests/TaskReviewProvenTxs.test.d.ts.map +1 -0
- package/out/src/monitor/tasks/__tests/TaskReviewProvenTxs.test.js +214 -0
- package/out/src/monitor/tasks/__tests/TaskReviewProvenTxs.test.js.map +1 -0
- package/out/src/monitor/tasks/__tests/TaskReviewUtxos.test.d.ts +2 -0
- package/out/src/monitor/tasks/__tests/TaskReviewUtxos.test.d.ts.map +1 -0
- package/out/src/monitor/tasks/__tests/TaskReviewUtxos.test.js +92 -0
- package/out/src/monitor/tasks/__tests/TaskReviewUtxos.test.js.map +1 -0
- package/out/src/monitor/tasks/__tests/TaskSendWaiting.test.d.ts +2 -0
- package/out/src/monitor/tasks/__tests/TaskSendWaiting.test.d.ts.map +1 -0
- package/out/src/monitor/tasks/__tests/TaskSendWaiting.test.js +139 -0
- package/out/src/monitor/tasks/__tests/TaskSendWaiting.test.js.map +1 -0
- package/out/src/monitor/tasks/index.all.d.ts +19 -0
- package/out/src/monitor/tasks/index.all.d.ts.map +1 -0
- package/out/src/monitor/tasks/index.all.js +35 -0
- package/out/src/monitor/tasks/index.all.js.map +1 -0
- package/out/src/sdk/WalletStorage.interfaces.d.ts +9 -0
- package/out/src/sdk/WalletStorage.interfaces.d.ts.map +1 -1
- package/out/src/services/Services.d.ts.map +1 -1
- package/out/src/services/Services.js +10 -2
- package/out/src/services/Services.js.map +1 -1
- package/out/src/services/__tests/getFiatExchangeRate.test.d.ts +2 -0
- package/out/src/services/__tests/getFiatExchangeRate.test.d.ts.map +1 -0
- package/out/src/services/__tests/getFiatExchangeRate.test.js +156 -0
- package/out/src/services/__tests/getFiatExchangeRate.test.js.map +1 -0
- package/out/src/services/chaintracker/chaintracks/Ingest/BulkIngestorWhatsOnChainCdn.js +1 -1
- package/out/src/services/chaintracker/chaintracks/Ingest/BulkIngestorWhatsOnChainCdn.js.map +1 -1
- package/out/src/services/createDefaultWalletServicesOptions.js +1 -1
- package/out/src/services/createDefaultWalletServicesOptions.js.map +1 -1
- package/out/src/services/providers/__tests/exchangeRates.test.js +4 -0
- package/out/src/services/providers/__tests/exchangeRates.test.js.map +1 -1
- package/out/src/storage/StorageKnex.d.ts +3 -1
- package/out/src/storage/StorageKnex.d.ts.map +1 -1
- package/out/src/storage/StorageKnex.js +26 -5
- package/out/src/storage/StorageKnex.js.map +1 -1
- package/out/src/storage/StorageProvider.d.ts +6 -1
- package/out/src/storage/StorageProvider.d.ts.map +1 -1
- package/out/src/storage/StorageProvider.js +6 -0
- package/out/src/storage/StorageProvider.js.map +1 -1
- package/out/src/storage/StorageReaderWriter.d.ts +2 -1
- package/out/src/storage/StorageReaderWriter.d.ts.map +1 -1
- package/out/src/storage/StorageReaderWriter.js.map +1 -1
- package/out/src/storage/WalletStorageManager.d.ts +7 -0
- package/out/src/storage/WalletStorageManager.d.ts.map +1 -1
- package/out/src/storage/WalletStorageManager.js +33 -2
- package/out/src/storage/WalletStorageManager.js.map +1 -1
- package/out/src/storage/__test/findStaleMerkleRoots.test.d.ts +2 -0
- package/out/src/storage/__test/findStaleMerkleRoots.test.d.ts.map +1 -0
- package/out/src/storage/__test/findStaleMerkleRoots.test.js +41 -0
- package/out/src/storage/__test/findStaleMerkleRoots.test.js.map +1 -0
- package/out/src/storage/__test/findStaleMerkleRootsKnex.test.d.ts +2 -0
- package/out/src/storage/__test/findStaleMerkleRootsKnex.test.d.ts.map +1 -0
- package/out/src/storage/__test/findStaleMerkleRootsKnex.test.js +73 -0
- package/out/src/storage/__test/findStaleMerkleRootsKnex.test.js.map +1 -0
- package/out/src/utility/Format.d.ts.map +1 -1
- package/out/src/utility/Format.js +1 -0
- package/out/src/utility/Format.js.map +1 -1
- package/out/src/utility/index.all.d.ts +1 -0
- package/out/src/utility/index.all.d.ts.map +1 -1
- package/out/src/utility/index.all.js +1 -0
- package/out/src/utility/index.all.js.map +1 -1
- package/package.json +2 -2
|
@@ -0,0 +1,161 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const TaskReviewDoubleSpends_1 = require("../TaskReviewDoubleSpends");
|
|
4
|
+
function makeReq(provenTxReqId, txid, updatedAt) {
|
|
5
|
+
const now = new Date();
|
|
6
|
+
return {
|
|
7
|
+
provenTxReqId,
|
|
8
|
+
created_at: now,
|
|
9
|
+
updated_at: updatedAt,
|
|
10
|
+
txid,
|
|
11
|
+
status: 'doubleSpend',
|
|
12
|
+
history: '{}',
|
|
13
|
+
notify: '{}',
|
|
14
|
+
attempts: 0,
|
|
15
|
+
notified: false
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
function makeMonitor(statusByTxid, reqs, monitorEvents = []) {
|
|
19
|
+
const updateProvenTxReq = jest.fn().mockResolvedValue(undefined);
|
|
20
|
+
const findProvenTxReqs = jest.fn(async ({ paged }) => reqs.slice(paged.offset, paged.offset + paged.limit));
|
|
21
|
+
const findMonitorEvents = jest.fn().mockResolvedValue(monitorEvents);
|
|
22
|
+
const runAsStorageProvider = jest.fn(async (fn) => await fn({ updateProvenTxReq, findMonitorEvents }));
|
|
23
|
+
const logEvent = jest.fn().mockResolvedValue(undefined);
|
|
24
|
+
return {
|
|
25
|
+
monitor: {
|
|
26
|
+
storage: {
|
|
27
|
+
findProvenTxReqs,
|
|
28
|
+
runAsStorageProvider
|
|
29
|
+
},
|
|
30
|
+
services: {
|
|
31
|
+
getStatusForTxids: jest.fn(async (txids) => ({
|
|
32
|
+
results: txids.map(txid => { var _a; return ({ txid, status: (_a = statusByTxid[txid]) !== null && _a !== void 0 ? _a : 'unknown' }); })
|
|
33
|
+
}))
|
|
34
|
+
},
|
|
35
|
+
logEvent
|
|
36
|
+
},
|
|
37
|
+
updateProvenTxReq,
|
|
38
|
+
findProvenTxReqs,
|
|
39
|
+
findMonitorEvents,
|
|
40
|
+
runAsStorageProvider,
|
|
41
|
+
logEvent
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
describe('TaskReviewDoubleSpends', () => {
|
|
45
|
+
afterEach(() => {
|
|
46
|
+
jest.restoreAllMocks();
|
|
47
|
+
});
|
|
48
|
+
test('0 uses the normal cadence after a partial review chunk and stores checkpoint offset after unfails are removed', async () => {
|
|
49
|
+
const now = new Date('2026-01-01T12:00:00.000Z');
|
|
50
|
+
jest.spyOn(Date, 'now').mockReturnValue(now.getTime());
|
|
51
|
+
const reqs = [
|
|
52
|
+
makeReq(1, 'tx1', new Date('2026-01-01T10:30:00.000Z')),
|
|
53
|
+
makeReq(2, 'tx2', new Date('2026-01-01T10:40:00.000Z'))
|
|
54
|
+
];
|
|
55
|
+
const m = makeMonitor({ tx1: 'success', tx2: 'unknown' }, reqs);
|
|
56
|
+
const task = new TaskReviewDoubleSpends_1.TaskReviewDoubleSpends(m.monitor, 0, 100, 60, 60);
|
|
57
|
+
const log = await task.runTask();
|
|
58
|
+
expect(m.findProvenTxReqs).toHaveBeenCalledWith({
|
|
59
|
+
partial: { status: 'doubleSpend' },
|
|
60
|
+
paged: { limit: 100, offset: 0 }
|
|
61
|
+
});
|
|
62
|
+
expect(m.updateProvenTxReq).toHaveBeenCalledWith([1], { status: 'unfail' });
|
|
63
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
64
|
+
expect(log).toContain('"reviewed":2');
|
|
65
|
+
expect(log).toContain('"unfails":1');
|
|
66
|
+
expect(log).toContain('"resumeOffset":0');
|
|
67
|
+
expect(log).toContain('"expectedProvenTxReqId":2');
|
|
68
|
+
expect(log).toContain('unfail 1 tx1 status:success');
|
|
69
|
+
expect(task.triggerNextMsecs).toBe(0);
|
|
70
|
+
});
|
|
71
|
+
test('0a stores the retained req offset in the post-unfail doubleSpend list', async () => {
|
|
72
|
+
const now = new Date('2026-01-01T12:00:00.000Z');
|
|
73
|
+
jest.spyOn(Date, 'now').mockReturnValue(now.getTime());
|
|
74
|
+
const reqs = [
|
|
75
|
+
makeReq(1, 'tx1', new Date('2026-01-01T10:30:00.000Z')),
|
|
76
|
+
makeReq(2, 'tx2', new Date('2026-01-01T10:35:00.000Z')),
|
|
77
|
+
makeReq(3, 'tx3', new Date('2026-01-01T10:40:00.000Z'))
|
|
78
|
+
];
|
|
79
|
+
const m = makeMonitor({ tx1: 'success', tx2: 'unknown', tx3: 'unknown' }, reqs);
|
|
80
|
+
const task = new TaskReviewDoubleSpends_1.TaskReviewDoubleSpends(m.monitor, 0, 100, 60, 60);
|
|
81
|
+
const log = await task.runTask();
|
|
82
|
+
expect(m.updateProvenTxReq).toHaveBeenCalledWith([1], { status: 'unfail' });
|
|
83
|
+
expect(log).toContain('"resumeOffset":1');
|
|
84
|
+
expect(log).toContain('"expectedProvenTxReqId":3');
|
|
85
|
+
});
|
|
86
|
+
test('1 skips reqs newer than the minAgeMinutes cutoff and returns empty when nothing is eligible', async () => {
|
|
87
|
+
const now = new Date('2026-01-01T12:00:00.000Z');
|
|
88
|
+
jest.spyOn(Date, 'now').mockReturnValue(now.getTime());
|
|
89
|
+
const reqs = [makeReq(1, 'tx1', new Date('2026-01-01T11:30:01.000Z'))];
|
|
90
|
+
const m = makeMonitor({ tx1: 'unknown' }, reqs);
|
|
91
|
+
const task = new TaskReviewDoubleSpends_1.TaskReviewDoubleSpends(m.monitor, 0, 100, 60, 60);
|
|
92
|
+
const log = await task.runTask();
|
|
93
|
+
expect(m.updateProvenTxReq).not.toHaveBeenCalled();
|
|
94
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
95
|
+
expect(log).toBe('');
|
|
96
|
+
expect(task.triggerNextMsecs).toBe(0);
|
|
97
|
+
});
|
|
98
|
+
test('1b uses the quick cadence after consuming a full review chunk', async () => {
|
|
99
|
+
const now = new Date('2026-01-01T12:00:00.000Z');
|
|
100
|
+
jest.spyOn(Date, 'now').mockReturnValue(now.getTime());
|
|
101
|
+
const reqs = [
|
|
102
|
+
makeReq(1, 'tx1', new Date('2026-01-01T10:30:00.000Z')),
|
|
103
|
+
makeReq(2, 'tx2', new Date('2026-01-01T10:40:00.000Z'))
|
|
104
|
+
];
|
|
105
|
+
const m = makeMonitor({ tx1: 'success', tx2: 'unknown' }, reqs);
|
|
106
|
+
const task = new TaskReviewDoubleSpends_1.TaskReviewDoubleSpends(m.monitor, 0, 2, 60, 60);
|
|
107
|
+
await task.runTask();
|
|
108
|
+
expect(task.triggerNextMsecs).toBe(60);
|
|
109
|
+
});
|
|
110
|
+
test('2 resumes from the checkpoint offset and advances it', async () => {
|
|
111
|
+
const now = new Date('2026-01-01T12:00:00.000Z');
|
|
112
|
+
jest.spyOn(Date, 'now').mockReturnValue(now.getTime());
|
|
113
|
+
const reqs = [
|
|
114
|
+
makeReq(1, 'tx1', new Date('2026-01-01T10:00:00.000Z')),
|
|
115
|
+
makeReq(2, 'tx2', new Date('2026-01-01T10:05:00.000Z')),
|
|
116
|
+
makeReq(3, 'tx3', new Date('2026-01-01T10:10:00.000Z'))
|
|
117
|
+
];
|
|
118
|
+
const m = makeMonitor({ tx2: 'unknown', tx3: 'success' }, reqs, [
|
|
119
|
+
{ details: JSON.stringify({ resumeOffset: 1, expectedProvenTxReqId: 2 }) }
|
|
120
|
+
]);
|
|
121
|
+
const task = new TaskReviewDoubleSpends_1.TaskReviewDoubleSpends(m.monitor, 0, 2, 60);
|
|
122
|
+
const log = await task.runTask();
|
|
123
|
+
expect(m.findProvenTxReqs).toHaveBeenNthCalledWith(1, {
|
|
124
|
+
partial: { status: 'doubleSpend' },
|
|
125
|
+
paged: { limit: 1, offset: 1 }
|
|
126
|
+
});
|
|
127
|
+
expect(m.findProvenTxReqs).toHaveBeenNthCalledWith(2, {
|
|
128
|
+
partial: { status: 'doubleSpend' },
|
|
129
|
+
paged: { limit: 2, offset: 2 }
|
|
130
|
+
});
|
|
131
|
+
expect(m.updateProvenTxReq).toHaveBeenCalledWith([3], { status: 'unfail' });
|
|
132
|
+
expect(log).toContain('"reviewed":1');
|
|
133
|
+
expect(log).not.toContain('"resumeOffset"');
|
|
134
|
+
expect(log).not.toContain('"expectedProvenTxReqId"');
|
|
135
|
+
});
|
|
136
|
+
test('3 restarts at offset zero when the checkpoint verification id no longer matches', async () => {
|
|
137
|
+
const now = new Date('2026-01-01T12:00:00.000Z');
|
|
138
|
+
jest.spyOn(Date, 'now').mockReturnValue(now.getTime());
|
|
139
|
+
const reqs = [
|
|
140
|
+
makeReq(10, 'tx10', new Date('2026-01-01T10:00:00.000Z')),
|
|
141
|
+
makeReq(11, 'tx11', new Date('2026-01-01T10:05:00.000Z'))
|
|
142
|
+
];
|
|
143
|
+
const m = makeMonitor({ tx10: 'unknown', tx11: 'success' }, reqs, [
|
|
144
|
+
{ details: JSON.stringify({ resumeOffset: 1, expectedProvenTxReqId: 99 }) }
|
|
145
|
+
]);
|
|
146
|
+
const task = new TaskReviewDoubleSpends_1.TaskReviewDoubleSpends(m.monitor, 0, 2, 60);
|
|
147
|
+
const log = await task.runTask();
|
|
148
|
+
expect(m.findProvenTxReqs).toHaveBeenNthCalledWith(1, {
|
|
149
|
+
partial: { status: 'doubleSpend' },
|
|
150
|
+
paged: { limit: 1, offset: 1 }
|
|
151
|
+
});
|
|
152
|
+
expect(m.findProvenTxReqs).toHaveBeenNthCalledWith(2, {
|
|
153
|
+
partial: { status: 'doubleSpend' },
|
|
154
|
+
paged: { limit: 2, offset: 0 }
|
|
155
|
+
});
|
|
156
|
+
expect(m.updateProvenTxReq).toHaveBeenCalledWith([11], { status: 'unfail' });
|
|
157
|
+
expect(log).toContain('"resumeOffset":0');
|
|
158
|
+
expect(log).toContain('"expectedProvenTxReqId":10');
|
|
159
|
+
});
|
|
160
|
+
});
|
|
161
|
+
//# sourceMappingURL=TaskReviewDoubleSpends.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TaskReviewDoubleSpends.test.js","sourceRoot":"","sources":["../../../../../src/monitor/tasks/__tests/TaskReviewDoubleSpends.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TaskReviewProvenTxs.test.d.ts","sourceRoot":"","sources":["../../../../../src/monitor/tasks/__tests/TaskReviewProvenTxs.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,214 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const TaskReviewProvenTxs_1 = require("../TaskReviewProvenTxs");
|
|
4
|
+
const HeightRange_1 = require("../../../services/chaintracker/chaintracks/util/HeightRange");
|
|
5
|
+
function makeMonitor(options) {
|
|
6
|
+
const reviewHeightRange = jest.fn().mockResolvedValue(options.reviewResult || {
|
|
7
|
+
log: '',
|
|
8
|
+
reviewedHeights: 0,
|
|
9
|
+
mismatchedHeights: 0,
|
|
10
|
+
affectedTransactions: 0,
|
|
11
|
+
updatedTransactions: 0
|
|
12
|
+
});
|
|
13
|
+
const findStaleMerkleRoots = jest.fn(async ({ height }) => { var _a, _b; return (_b = (_a = options.staleRootsByHeight) === null || _a === void 0 ? void 0 : _a[height]) !== null && _b !== void 0 ? _b : []; });
|
|
14
|
+
const findMonitorEvents = jest.fn(async () => options.monitorEvents || []);
|
|
15
|
+
const runAsStorageProvider = jest.fn(async (fn) => await fn({ findStaleMerkleRoots }));
|
|
16
|
+
const reproveHeightMerkleRoot = jest.fn(async (height, staleRoot) => {
|
|
17
|
+
var _a;
|
|
18
|
+
return (((_a = options.reproveResultsByHeightRoot) === null || _a === void 0 ? void 0 : _a[`${height}:${staleRoot}`]) || {
|
|
19
|
+
log: ` reproved ${height}:${staleRoot}\n`,
|
|
20
|
+
updated: [],
|
|
21
|
+
unchanged: [],
|
|
22
|
+
unavailable: []
|
|
23
|
+
});
|
|
24
|
+
});
|
|
25
|
+
const logEvent = jest.fn().mockResolvedValue(undefined);
|
|
26
|
+
const chaintracks = {
|
|
27
|
+
currentHeight: jest.fn().mockResolvedValue(options.tipHeight),
|
|
28
|
+
findHeaderForHeight: jest.fn(async (height) => { var _a; return (_a = options.headersByHeight) === null || _a === void 0 ? void 0 : _a[height]; })
|
|
29
|
+
};
|
|
30
|
+
return {
|
|
31
|
+
monitor: {
|
|
32
|
+
storage: {
|
|
33
|
+
runAsStorageProvider,
|
|
34
|
+
reproveHeightMerkleRoot
|
|
35
|
+
},
|
|
36
|
+
chaintracks,
|
|
37
|
+
chaintracksWithEvents: undefined,
|
|
38
|
+
logEvent
|
|
39
|
+
},
|
|
40
|
+
reviewHeightRange,
|
|
41
|
+
findStaleMerkleRoots,
|
|
42
|
+
findMonitorEvents,
|
|
43
|
+
runAsStorageProvider,
|
|
44
|
+
reproveHeightMerkleRoot,
|
|
45
|
+
logEvent,
|
|
46
|
+
chaintracks
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
describe('TaskReviewProvenTxs tests', () => {
|
|
50
|
+
test('0 reviewHeightRange uses findStaleMerkleRoots and reproves only stale roots in the requested range', async () => {
|
|
51
|
+
const m = makeMonitor({
|
|
52
|
+
tipHeight: 120,
|
|
53
|
+
headersByHeight: {
|
|
54
|
+
107: { height: 107, merkleRoot: 'root-107', hash: 'hash-107' },
|
|
55
|
+
108: { height: 108, merkleRoot: 'root-108-new', hash: 'hash-108' }
|
|
56
|
+
},
|
|
57
|
+
staleRootsByHeight: {
|
|
58
|
+
107: [],
|
|
59
|
+
108: ['root-108-old']
|
|
60
|
+
},
|
|
61
|
+
reproveResultsByHeightRoot: {
|
|
62
|
+
'108:root-108-old': {
|
|
63
|
+
log: ' height 108 stale merkleRoot root-108-old with 2 impacted transactions\n',
|
|
64
|
+
updated: [{}],
|
|
65
|
+
unchanged: [{}],
|
|
66
|
+
unavailable: []
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
});
|
|
70
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor, 0, 2, 12);
|
|
71
|
+
const result = await task.reviewHeightRange(new HeightRange_1.HeightRange(107, 108));
|
|
72
|
+
expect(m.findStaleMerkleRoots).toHaveBeenNthCalledWith(1, { height: 107, merkleRoot: 'root-107' });
|
|
73
|
+
expect(m.findStaleMerkleRoots).toHaveBeenNthCalledWith(2, { height: 108, merkleRoot: 'root-108-new' });
|
|
74
|
+
expect(m.reproveHeightMerkleRoot).toHaveBeenCalledTimes(1);
|
|
75
|
+
expect(m.reproveHeightMerkleRoot).toHaveBeenCalledWith(108, 'root-108-old');
|
|
76
|
+
expect(result.reviewedHeights).toBe(2);
|
|
77
|
+
expect(result.mismatchedHeights).toBe(1);
|
|
78
|
+
expect(result.affectedTransactions).toBe(2);
|
|
79
|
+
expect(result.updatedTransactions).toBe(1);
|
|
80
|
+
expect(result.log).toContain('height 108 canonical root-108-new stale root-108-old');
|
|
81
|
+
});
|
|
82
|
+
test('1 reviewHeightRange records unavailable headers and skips empty ranges cleanly', async () => {
|
|
83
|
+
const m = makeMonitor({
|
|
84
|
+
tipHeight: 120,
|
|
85
|
+
headersByHeight: {
|
|
86
|
+
10: undefined
|
|
87
|
+
}
|
|
88
|
+
});
|
|
89
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor, 0, 2, 12);
|
|
90
|
+
const result = await task.reviewHeightRange(new HeightRange_1.HeightRange(10, 10));
|
|
91
|
+
const empty = await task.reviewHeightRange(HeightRange_1.HeightRange.empty);
|
|
92
|
+
expect(m.reproveHeightMerkleRoot).not.toHaveBeenCalled();
|
|
93
|
+
expect(result.reviewedHeights).toBe(1);
|
|
94
|
+
expect(result.log).toContain('height 10 canonical header unavailable');
|
|
95
|
+
expect(empty.reviewedHeights).toBe(0);
|
|
96
|
+
expect(empty.log).toBe('');
|
|
97
|
+
});
|
|
98
|
+
test('2 getLastReviewedHeight skips plain-text events and uses the latest checkpoint event', async () => {
|
|
99
|
+
const m = makeMonitor({
|
|
100
|
+
tipHeight: 120,
|
|
101
|
+
monitorEvents: [
|
|
102
|
+
{ details: 'reviewing heights 10..20 tip=120 minAge=100 maxPerRun=100' },
|
|
103
|
+
{ details: JSON.stringify({ reviewedThroughHeight: 20 }) },
|
|
104
|
+
{ details: JSON.stringify({ reviewedThroughHeight: 7 }) }
|
|
105
|
+
]
|
|
106
|
+
});
|
|
107
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor);
|
|
108
|
+
jest
|
|
109
|
+
.spyOn(task.storage, 'runAsStorageProvider')
|
|
110
|
+
.mockImplementation(async (fn) => await fn({ findMonitorEvents: m.findMonitorEvents }));
|
|
111
|
+
const lastReviewedHeight = await task.getLastReviewedHeight();
|
|
112
|
+
expect(lastReviewedHeight).toBe(20);
|
|
113
|
+
});
|
|
114
|
+
test('3 runTask starts from height 0 on cold start and caps the range by batch size and minimum age', async () => {
|
|
115
|
+
const m = makeMonitor({
|
|
116
|
+
tipHeight: 250,
|
|
117
|
+
reviewResult: {
|
|
118
|
+
log: '',
|
|
119
|
+
reviewedHeights: 100,
|
|
120
|
+
mismatchedHeights: 0,
|
|
121
|
+
affectedTransactions: 0,
|
|
122
|
+
updatedTransactions: 0
|
|
123
|
+
}
|
|
124
|
+
});
|
|
125
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor, 0, 100, 100);
|
|
126
|
+
jest.spyOn(task, 'getLastReviewedHeight').mockResolvedValue(undefined);
|
|
127
|
+
const reviewSpy = jest.spyOn(task, 'reviewHeightRange').mockResolvedValue(m.reviewHeightRange());
|
|
128
|
+
const log = await task.runTask();
|
|
129
|
+
expect(reviewSpy).toHaveBeenCalledWith(new HeightRange_1.HeightRange(0, 99));
|
|
130
|
+
expect(log).toContain('"reviewedThroughHeight":99');
|
|
131
|
+
expect(log).toContain('"minBlockAge":100');
|
|
132
|
+
expect(log).toContain('reviewing heights 0..99 tip=250 minAge=100 maxPerRun=100');
|
|
133
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
134
|
+
});
|
|
135
|
+
test('4 runTask resumes from the last reviewed height plus one', async () => {
|
|
136
|
+
const m = makeMonitor({
|
|
137
|
+
tipHeight: 250,
|
|
138
|
+
reviewResult: {
|
|
139
|
+
log: ' height 121 canonical root-121-new stale root-121-old\n reproved stale root\n',
|
|
140
|
+
reviewedHeights: 21,
|
|
141
|
+
mismatchedHeights: 1,
|
|
142
|
+
affectedTransactions: 1,
|
|
143
|
+
updatedTransactions: 0
|
|
144
|
+
}
|
|
145
|
+
});
|
|
146
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor, 0, 100, 100);
|
|
147
|
+
jest.spyOn(task, 'getLastReviewedHeight').mockResolvedValue(100);
|
|
148
|
+
const reviewSpy = jest.spyOn(task, 'reviewHeightRange').mockResolvedValue(m.reviewHeightRange());
|
|
149
|
+
const log = await task.runTask();
|
|
150
|
+
expect(reviewSpy).toHaveBeenCalledTimes(1);
|
|
151
|
+
expect(reviewSpy).toHaveBeenCalledWith(new HeightRange_1.HeightRange(101, 150));
|
|
152
|
+
expect(log).toContain('"reviewedThroughHeight":150');
|
|
153
|
+
expect(log).toContain('"minBlockAge":100');
|
|
154
|
+
expect(log).toContain('reviewing heights 101..150 tip=250 minAge=100 maxPerRun=100');
|
|
155
|
+
expect(log).toContain('height 121 canonical root-121-new stale root-121-old');
|
|
156
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
157
|
+
});
|
|
158
|
+
test('5 runTask returns review logs even when the range is clean', async () => {
|
|
159
|
+
const m = makeMonitor({
|
|
160
|
+
tipHeight: 250,
|
|
161
|
+
reviewResult: {
|
|
162
|
+
log: '',
|
|
163
|
+
reviewedHeights: 50,
|
|
164
|
+
mismatchedHeights: 0,
|
|
165
|
+
affectedTransactions: 0,
|
|
166
|
+
updatedTransactions: 0
|
|
167
|
+
}
|
|
168
|
+
});
|
|
169
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor, 0, 50, 100);
|
|
170
|
+
jest.spyOn(task, 'getLastReviewedHeight').mockResolvedValue(149);
|
|
171
|
+
const reviewSpy = jest.spyOn(task, 'reviewHeightRange').mockResolvedValue(m.reviewHeightRange());
|
|
172
|
+
const log = await task.runTask();
|
|
173
|
+
expect(reviewSpy).toHaveBeenCalledTimes(1);
|
|
174
|
+
expect(reviewSpy).toHaveBeenCalledWith(new HeightRange_1.HeightRange(150, 150));
|
|
175
|
+
expect(log).toContain('"reviewedThroughHeight":150');
|
|
176
|
+
expect(log).toContain('"minBlockAge":100');
|
|
177
|
+
expect(log).toContain('reviewing heights 150..150 tip=250 minAge=100 maxPerRun=50');
|
|
178
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
179
|
+
});
|
|
180
|
+
test('6 runTask returns early when the chain tip is below the minimum age window', async () => {
|
|
181
|
+
const m = makeMonitor({
|
|
182
|
+
tipHeight: 5
|
|
183
|
+
});
|
|
184
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor, 0, 100, 100);
|
|
185
|
+
jest.spyOn(task, 'getLastReviewedHeight').mockResolvedValue(undefined);
|
|
186
|
+
const log = await task.runTask();
|
|
187
|
+
expect(m.reviewHeightRange).not.toHaveBeenCalled();
|
|
188
|
+
expect(log).toBe('');
|
|
189
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
190
|
+
});
|
|
191
|
+
test('7 runTask returns early when all eligible heights have already been reviewed', async () => {
|
|
192
|
+
const m = makeMonitor({
|
|
193
|
+
tipHeight: 250
|
|
194
|
+
});
|
|
195
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor, 0, 100, 100);
|
|
196
|
+
jest.spyOn(task, 'getLastReviewedHeight').mockResolvedValue(200);
|
|
197
|
+
const log = await task.runTask();
|
|
198
|
+
expect(m.reviewHeightRange).not.toHaveBeenCalled();
|
|
199
|
+
expect(log).toBe('');
|
|
200
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
201
|
+
});
|
|
202
|
+
test('8 runTask returns empty log when the computed range is empty', async () => {
|
|
203
|
+
const m = makeMonitor({
|
|
204
|
+
tipHeight: 250
|
|
205
|
+
});
|
|
206
|
+
const task = new TaskReviewProvenTxs_1.TaskReviewProvenTxs(m.monitor, 0, 0, 100);
|
|
207
|
+
jest.spyOn(task, 'getLastReviewedHeight').mockResolvedValue(undefined);
|
|
208
|
+
const log = await task.runTask();
|
|
209
|
+
expect(m.reviewHeightRange).not.toHaveBeenCalled();
|
|
210
|
+
expect(log).toBe('');
|
|
211
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
212
|
+
});
|
|
213
|
+
});
|
|
214
|
+
//# sourceMappingURL=TaskReviewProvenTxs.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TaskReviewProvenTxs.test.js","sourceRoot":"","sources":["../../../../../src/monitor/tasks/__tests/TaskReviewProvenTxs.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TaskReviewUtxos.test.d.ts","sourceRoot":"","sources":["../../../../../src/monitor/tasks/__tests/TaskReviewUtxos.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const TaskReviewUtxos_1 = require("../TaskReviewUtxos");
|
|
4
|
+
const sdk_1 = require("../../../sdk");
|
|
5
|
+
function makeUser(userId, identityKey = `key-${userId}`) {
|
|
6
|
+
const now = new Date();
|
|
7
|
+
return {
|
|
8
|
+
created_at: now,
|
|
9
|
+
updated_at: now,
|
|
10
|
+
userId,
|
|
11
|
+
identityKey,
|
|
12
|
+
activeStorage: 'storage-key'
|
|
13
|
+
};
|
|
14
|
+
}
|
|
15
|
+
function makeOutput(outpoint, satoshis, spendable) {
|
|
16
|
+
return { outpoint, satoshis, spendable };
|
|
17
|
+
}
|
|
18
|
+
function makeMonitor(users, outputsByUserId) {
|
|
19
|
+
const findUsers = jest.fn().mockResolvedValue(users);
|
|
20
|
+
const listOutputs = jest.fn(async (auth) => {
|
|
21
|
+
var _a;
|
|
22
|
+
const outputs = (_a = outputsByUserId[auth.userId]) !== null && _a !== void 0 ? _a : [];
|
|
23
|
+
return {
|
|
24
|
+
totalOutputs: outputs.length,
|
|
25
|
+
outputs
|
|
26
|
+
};
|
|
27
|
+
});
|
|
28
|
+
const runAsStorageProvider = jest.fn(async (fn) => await fn({ findUsers, listOutputs }));
|
|
29
|
+
const logEvent = jest.fn().mockResolvedValue(undefined);
|
|
30
|
+
return {
|
|
31
|
+
monitor: {
|
|
32
|
+
storage: { runAsStorageProvider },
|
|
33
|
+
logEvent
|
|
34
|
+
},
|
|
35
|
+
findUsers,
|
|
36
|
+
listOutputs,
|
|
37
|
+
runAsStorageProvider,
|
|
38
|
+
logEvent
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
describe('TaskReviewUtxos', () => {
|
|
42
|
+
test('0 reviewByIdentityKey reviews all invalid utxos for the matching user', async () => {
|
|
43
|
+
const users = [makeUser(1), makeUser(2)];
|
|
44
|
+
const m = makeMonitor(users, {
|
|
45
|
+
1: [makeOutput('tx1.0', 50, false)],
|
|
46
|
+
2: []
|
|
47
|
+
});
|
|
48
|
+
const task = new TaskReviewUtxos_1.TaskReviewUtxos(m.monitor);
|
|
49
|
+
const log = await task.reviewByIdentityKey('key-1');
|
|
50
|
+
expect(m.findUsers).toHaveBeenCalledWith({ partial: { identityKey: 'key-1' } });
|
|
51
|
+
expect(m.listOutputs).toHaveBeenCalledWith({ userId: 1, identityKey: 'key-1' }, expect.objectContaining({
|
|
52
|
+
basket: sdk_1.specOpInvalidChange,
|
|
53
|
+
tags: ['release', 'all'],
|
|
54
|
+
tagQueryMode: 'all',
|
|
55
|
+
limit: 0,
|
|
56
|
+
offset: 0
|
|
57
|
+
}));
|
|
58
|
+
expect(m.logEvent).not.toHaveBeenCalled();
|
|
59
|
+
expect(log).toContain('userId 1: 1 spendable utxos updated to unspendable');
|
|
60
|
+
expect(log).toContain('tx1.0 50 now spent');
|
|
61
|
+
});
|
|
62
|
+
test('1 reviewByIdentityKey limits review to invalid change utxos when mode is change', async () => {
|
|
63
|
+
const users = [makeUser(1)];
|
|
64
|
+
const m = makeMonitor(users, { 1: [makeOutput('tx1.0', 50, false)] });
|
|
65
|
+
const task = new TaskReviewUtxos_1.TaskReviewUtxos(m.monitor);
|
|
66
|
+
await task.reviewByIdentityKey('key-1', 'change');
|
|
67
|
+
expect(m.listOutputs).toHaveBeenCalledWith({ userId: 1, identityKey: 'key-1' }, expect.objectContaining({
|
|
68
|
+
tags: ['release']
|
|
69
|
+
}));
|
|
70
|
+
});
|
|
71
|
+
test('2 reviewByIdentityKey returns no-findings summary when the user has no invalid utxos', async () => {
|
|
72
|
+
const users = [makeUser(1)];
|
|
73
|
+
const m = makeMonitor(users, {});
|
|
74
|
+
const task = new TaskReviewUtxos_1.TaskReviewUtxos(m.monitor);
|
|
75
|
+
const log = await task.reviewByIdentityKey('key-1');
|
|
76
|
+
expect(log).toBe('userId 1: no invalid utxos found, key-1\n');
|
|
77
|
+
});
|
|
78
|
+
test('3 reviewByIdentityKey reports when the identity key does not exist', async () => {
|
|
79
|
+
const m = makeMonitor([], {});
|
|
80
|
+
const task = new TaskReviewUtxos_1.TaskReviewUtxos(m.monitor);
|
|
81
|
+
const log = await task.reviewByIdentityKey('missing-key');
|
|
82
|
+
expect(m.listOutputs).not.toHaveBeenCalled();
|
|
83
|
+
expect(log).toBe('identityKey missing-key was not found\n');
|
|
84
|
+
});
|
|
85
|
+
test('4 trigger and runTask are stubbed out', async () => {
|
|
86
|
+
const m = makeMonitor([], {});
|
|
87
|
+
const task = new TaskReviewUtxos_1.TaskReviewUtxos(m.monitor);
|
|
88
|
+
expect(task.trigger(Date.now())).toEqual({ run: false });
|
|
89
|
+
await expect(task.runTask()).resolves.toBe('TaskReviewUtxos is disabled; use reviewByIdentityKey instead.\n');
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
//# sourceMappingURL=TaskReviewUtxos.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TaskReviewUtxos.test.js","sourceRoot":"","sources":["../../../../../src/monitor/tasks/__tests/TaskReviewUtxos.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"TaskSendWaiting.test.d.ts","sourceRoot":"","sources":["../../../../../src/monitor/tasks/__tests/TaskSendWaiting.test.ts"],"names":[],"mappings":""}
|