@xyo-network/sentinel 2.84.7 → 2.84.9
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/browser/MemorySentinel.d.cts +3 -0
- package/dist/browser/MemorySentinel.d.cts.map +1 -1
- package/dist/browser/MemorySentinel.d.mts +3 -0
- package/dist/browser/MemorySentinel.d.mts.map +1 -1
- package/dist/browser/MemorySentinel.d.ts +3 -0
- package/dist/browser/MemorySentinel.d.ts.map +1 -1
- package/dist/browser/SentinelIntervalAutomationWrapper.d.cts +11 -1
- package/dist/browser/SentinelIntervalAutomationWrapper.d.cts.map +1 -1
- package/dist/browser/SentinelIntervalAutomationWrapper.d.mts +11 -1
- package/dist/browser/SentinelIntervalAutomationWrapper.d.mts.map +1 -1
- package/dist/browser/SentinelIntervalAutomationWrapper.d.ts +11 -1
- package/dist/browser/SentinelIntervalAutomationWrapper.d.ts.map +1 -1
- package/dist/browser/SentinelRunner.d.cts +1 -2
- package/dist/browser/SentinelRunner.d.cts.map +1 -1
- package/dist/browser/SentinelRunner.d.mts +1 -2
- package/dist/browser/SentinelRunner.d.mts.map +1 -1
- package/dist/browser/SentinelRunner.d.ts +1 -2
- package/dist/browser/SentinelRunner.d.ts.map +1 -1
- package/dist/browser/index.cjs +132 -98
- package/dist/browser/index.cjs.map +1 -1
- package/dist/browser/index.d.cts +0 -1
- package/dist/browser/index.d.cts.map +1 -1
- package/dist/browser/index.d.mts +0 -1
- package/dist/browser/index.d.mts.map +1 -1
- package/dist/browser/index.d.ts +0 -1
- package/dist/browser/index.d.ts.map +1 -1
- package/dist/browser/index.js +126 -90
- package/dist/browser/index.js.map +1 -1
- package/dist/node/MemorySentinel.d.cts +3 -0
- package/dist/node/MemorySentinel.d.cts.map +1 -1
- package/dist/node/MemorySentinel.d.mts +3 -0
- package/dist/node/MemorySentinel.d.mts.map +1 -1
- package/dist/node/MemorySentinel.d.ts +3 -0
- package/dist/node/MemorySentinel.d.ts.map +1 -1
- package/dist/node/SentinelIntervalAutomationWrapper.d.cts +11 -1
- package/dist/node/SentinelIntervalAutomationWrapper.d.cts.map +1 -1
- package/dist/node/SentinelIntervalAutomationWrapper.d.mts +11 -1
- package/dist/node/SentinelIntervalAutomationWrapper.d.mts.map +1 -1
- package/dist/node/SentinelIntervalAutomationWrapper.d.ts +11 -1
- package/dist/node/SentinelIntervalAutomationWrapper.d.ts.map +1 -1
- package/dist/node/SentinelRunner.d.cts +1 -2
- package/dist/node/SentinelRunner.d.cts.map +1 -1
- package/dist/node/SentinelRunner.d.mts +1 -2
- package/dist/node/SentinelRunner.d.mts.map +1 -1
- package/dist/node/SentinelRunner.d.ts +1 -2
- package/dist/node/SentinelRunner.d.ts.map +1 -1
- package/dist/node/index.cjs +134 -102
- package/dist/node/index.cjs.map +1 -1
- package/dist/node/index.d.cts +0 -1
- package/dist/node/index.d.cts.map +1 -1
- package/dist/node/index.d.mts +0 -1
- package/dist/node/index.d.mts.map +1 -1
- package/dist/node/index.d.ts +0 -1
- package/dist/node/index.d.ts.map +1 -1
- package/dist/node/index.js +128 -90
- package/dist/node/index.js.map +1 -1
- package/package.json +19 -19
- package/src/MemorySentinel.ts +23 -0
- package/src/SentinelIntervalAutomationWrapper.ts +31 -12
- package/src/SentinelRunner.ts +22 -14
- package/src/index.ts +0 -1
- package/dist/browser/Automation.d.cts +0 -71
- package/dist/browser/Automation.d.cts.map +0 -1
- package/dist/browser/Automation.d.mts +0 -71
- package/dist/browser/Automation.d.mts.map +0 -1
- package/dist/browser/Automation.d.ts +0 -71
- package/dist/browser/Automation.d.ts.map +0 -1
- package/dist/node/Automation.d.cts +0 -71
- package/dist/node/Automation.d.cts.map +0 -1
- package/dist/node/Automation.d.mts +0 -71
- package/dist/node/Automation.d.mts.map +0 -1
- package/dist/node/Automation.d.ts +0 -71
- package/dist/node/Automation.d.ts.map +0 -1
- package/src/Automation.ts +0 -55
package/dist/node/index.js
CHANGED
|
@@ -1,10 +1,3 @@
|
|
|
1
|
-
// src/Automation.ts
|
|
2
|
-
import { isPayloadOfSchemaType } from "@xyo-network/payload-model";
|
|
3
|
-
var SentinelAutomationSchema = "network.xyo.automation";
|
|
4
|
-
var SentinelIntervalAutomationSchema = "network.xyo.automation.interval";
|
|
5
|
-
var SentinelEventAutomationSchema = "network.xyo.automation.event";
|
|
6
|
-
var isSentinelIntervalAutomation = isPayloadOfSchemaType(SentinelIntervalAutomationSchema);
|
|
7
|
-
|
|
8
1
|
// src/MemorySentinel.ts
|
|
9
2
|
import { fulfilled, rejected } from "@xylabs/promise";
|
|
10
3
|
import { asDivinerInstance } from "@xyo-network/diviner-model";
|
|
@@ -13,62 +6,13 @@ import {
|
|
|
13
6
|
SentinelConfigSchema
|
|
14
7
|
} from "@xyo-network/sentinel-model";
|
|
15
8
|
import { asWitnessInstance } from "@xyo-network/witness-model";
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
while (index < job.tasks.length) {
|
|
24
|
-
const generatedPayloads = await this.generateResults(job.tasks[index], previousResults, inPayloads);
|
|
25
|
-
previousResults = generatedPayloads;
|
|
26
|
-
index++;
|
|
27
|
-
}
|
|
28
|
-
return Object.values(previousResults).flat();
|
|
29
|
-
}
|
|
30
|
-
async generateResults(tasks, previousResults, inPayloads) {
|
|
31
|
-
const results = await Promise.allSettled(
|
|
32
|
-
tasks == null ? void 0 : tasks.map(async (task) => {
|
|
33
|
-
const witness = asWitnessInstance(task.module);
|
|
34
|
-
const input = task.input ?? false;
|
|
35
|
-
const inPayloadsFound = input === true ? inPayloads : input === false ? [] : this.processPreviousResults(previousResults, await this.inputAddresses(input));
|
|
36
|
-
if (witness) {
|
|
37
|
-
return [witness.address, await witness.observe(inPayloadsFound)];
|
|
38
|
-
}
|
|
39
|
-
const diviner = asDivinerInstance(task.module);
|
|
40
|
-
if (diviner) {
|
|
41
|
-
return [diviner.address, await diviner.divine(inPayloadsFound)];
|
|
42
|
-
}
|
|
43
|
-
throw new Error("Unsupported module type");
|
|
44
|
-
})
|
|
45
|
-
);
|
|
46
|
-
const finalResult = {};
|
|
47
|
-
for (const result of results.filter(fulfilled)) {
|
|
48
|
-
const [address, payloads] = result.value;
|
|
49
|
-
finalResult[address] = finalResult[address] ?? [];
|
|
50
|
-
finalResult[address].push(...payloads);
|
|
51
|
-
}
|
|
52
|
-
if (this.throwErrors) {
|
|
53
|
-
const errors = results.filter(rejected).map((result) => result.reason);
|
|
54
|
-
if (errors.length > 0) {
|
|
55
|
-
throw new Error("At least one module failed");
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
return finalResult;
|
|
59
|
-
}
|
|
60
|
-
async inputAddresses(input) {
|
|
61
|
-
if (Array.isArray(input)) {
|
|
62
|
-
return (await Promise.all(input.map(async (inputItem) => await this.inputAddresses(inputItem)))).flat();
|
|
63
|
-
} else {
|
|
64
|
-
const resolved = await this.resolve(input);
|
|
65
|
-
return resolved ? [resolved.address] : [];
|
|
66
|
-
}
|
|
67
|
-
}
|
|
68
|
-
processPreviousResults(payloads, inputs) {
|
|
69
|
-
return inputs.flatMap((input) => payloads[input] ?? []);
|
|
70
|
-
}
|
|
71
|
-
};
|
|
9
|
+
|
|
10
|
+
// src/SentinelRunner.ts
|
|
11
|
+
import { assertEx } from "@xylabs/assert";
|
|
12
|
+
import { PayloadWrapper as PayloadWrapper2 } from "@xyo-network/payload-wrapper";
|
|
13
|
+
import {
|
|
14
|
+
isSentinelIntervalAutomation
|
|
15
|
+
} from "@xyo-network/sentinel-model";
|
|
72
16
|
|
|
73
17
|
// src/SentinelIntervalAutomationWrapper.ts
|
|
74
18
|
import { PayloadWrapper } from "@xyo-network/payload-wrapper";
|
|
@@ -80,7 +24,8 @@ var SentinelIntervalAutomationWrapper = class extends PayloadWrapper {
|
|
|
80
24
|
const frequency = this.jsonPayload().frequency;
|
|
81
25
|
if (frequency === void 0)
|
|
82
26
|
return Number.POSITIVE_INFINITY;
|
|
83
|
-
|
|
27
|
+
const frequencyUnits = this.jsonPayload().frequencyUnits;
|
|
28
|
+
switch (frequencyUnits ?? "hour") {
|
|
84
29
|
case "second": {
|
|
85
30
|
return frequency * 1e3;
|
|
86
31
|
}
|
|
@@ -93,34 +38,53 @@ var SentinelIntervalAutomationWrapper = class extends PayloadWrapper {
|
|
|
93
38
|
case "day": {
|
|
94
39
|
return frequency * 24 * 60 * 60 * 1e3;
|
|
95
40
|
}
|
|
41
|
+
default: {
|
|
42
|
+
return Number.POSITIVE_INFINITY;
|
|
43
|
+
}
|
|
96
44
|
}
|
|
97
45
|
}
|
|
98
46
|
get remaining() {
|
|
99
|
-
return this.
|
|
47
|
+
return this.jsonPayload().remaining ?? Number.POSITIVE_INFINITY;
|
|
100
48
|
}
|
|
101
49
|
next() {
|
|
102
|
-
|
|
50
|
+
var _a;
|
|
51
|
+
const now = Date.now();
|
|
52
|
+
const previousStart = ((_a = this.jsonPayload()) == null ? void 0 : _a.start) ?? now;
|
|
53
|
+
const start = Math.max(previousStart, now);
|
|
54
|
+
const nextStart = start + this.frequencyMillis;
|
|
55
|
+
this.setStart(nextStart);
|
|
103
56
|
this.consumeRemaining();
|
|
104
57
|
this.checkEnd();
|
|
105
58
|
return this;
|
|
106
59
|
}
|
|
107
60
|
checkEnd() {
|
|
108
61
|
if (this.jsonPayload().start > (this.jsonPayload().end ?? Number.POSITIVE_INFINITY)) {
|
|
109
|
-
this.
|
|
62
|
+
this.setStart(Number.POSITIVE_INFINITY);
|
|
110
63
|
}
|
|
111
64
|
}
|
|
112
65
|
consumeRemaining(count = 1) {
|
|
113
|
-
const remaining = this.remaining - count;
|
|
114
|
-
this.
|
|
115
|
-
if (remaining <= 0)
|
|
116
|
-
this.
|
|
117
|
-
|
|
66
|
+
const remaining = Math.max(this.remaining - count, 0);
|
|
67
|
+
this.setRemaining(remaining);
|
|
68
|
+
if (remaining <= 0)
|
|
69
|
+
this.setStart(Number.POSITIVE_INFINITY);
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Sets the remaining of the wrapped automation
|
|
73
|
+
* @param remaining The remaining time in milliseconds
|
|
74
|
+
*/
|
|
75
|
+
setRemaining(remaining) {
|
|
76
|
+
this.obj.remaining = remaining;
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Sets the start of the wrapped automation
|
|
80
|
+
* @param start The start time in milliseconds
|
|
81
|
+
*/
|
|
82
|
+
setStart(start) {
|
|
83
|
+
this.obj.start = start;
|
|
118
84
|
}
|
|
119
85
|
};
|
|
120
86
|
|
|
121
87
|
// src/SentinelRunner.ts
|
|
122
|
-
import { assertEx } from "@xylabs/assert";
|
|
123
|
-
import { PayloadWrapper as PayloadWrapper2 } from "@xyo-network/payload-wrapper";
|
|
124
88
|
var SentinelRunner = class {
|
|
125
89
|
_automations = {};
|
|
126
90
|
onTriggerResult;
|
|
@@ -168,20 +132,22 @@ var SentinelRunner = class {
|
|
|
168
132
|
await this.start();
|
|
169
133
|
}
|
|
170
134
|
async start() {
|
|
135
|
+
await Promise.resolve();
|
|
171
136
|
assertEx(this.timeoutId === void 0, "Already started");
|
|
172
137
|
const automation = this.next;
|
|
173
138
|
if (isSentinelIntervalAutomation(automation)) {
|
|
174
|
-
const
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
this.timeoutId = setTimeout(
|
|
179
|
-
|
|
180
|
-
this.
|
|
139
|
+
const now = Date.now();
|
|
140
|
+
const start = Math.max(automation.start ?? now, now);
|
|
141
|
+
const delay = Math.max(start - now, 0);
|
|
142
|
+
if (delay < Number.POSITIVE_INFINITY) {
|
|
143
|
+
this.timeoutId = setTimeout(async () => {
|
|
144
|
+
try {
|
|
145
|
+
await this.trigger(automation);
|
|
146
|
+
this.stop();
|
|
147
|
+
} finally {
|
|
181
148
|
await this.start();
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
);
|
|
149
|
+
}
|
|
150
|
+
}, delay);
|
|
185
151
|
}
|
|
186
152
|
}
|
|
187
153
|
}
|
|
@@ -205,7 +171,83 @@ var SentinelRunner = class {
|
|
|
205
171
|
await this.add(wrapper.jsonPayload(), false);
|
|
206
172
|
const triggerResult = await this.sentinel.report();
|
|
207
173
|
(_a = this.onTriggerResult) == null ? void 0 : _a.call(this, triggerResult);
|
|
208
|
-
|
|
174
|
+
}
|
|
175
|
+
};
|
|
176
|
+
|
|
177
|
+
// src/MemorySentinel.ts
|
|
178
|
+
var MemorySentinel = class extends AbstractSentinel {
|
|
179
|
+
static configSchemas = [SentinelConfigSchema];
|
|
180
|
+
runner;
|
|
181
|
+
async reportHandler(inPayloads = []) {
|
|
182
|
+
await this.started("throw");
|
|
183
|
+
const job = await this.jobPromise;
|
|
184
|
+
let index = 0;
|
|
185
|
+
let previousResults = {};
|
|
186
|
+
while (index < job.tasks.length) {
|
|
187
|
+
const generatedPayloads = await this.generateResults(job.tasks[index], previousResults, inPayloads);
|
|
188
|
+
previousResults = generatedPayloads;
|
|
189
|
+
index++;
|
|
190
|
+
}
|
|
191
|
+
return Object.values(previousResults).flat();
|
|
192
|
+
}
|
|
193
|
+
async start(timeout) {
|
|
194
|
+
var _a;
|
|
195
|
+
if (await super.start(timeout)) {
|
|
196
|
+
if ((((_a = this.config.automations) == null ? void 0 : _a.length) ?? 0) > 0) {
|
|
197
|
+
this.runner = new SentinelRunner(this, this.config.automations);
|
|
198
|
+
await this.runner.start();
|
|
199
|
+
}
|
|
200
|
+
return true;
|
|
201
|
+
}
|
|
202
|
+
return false;
|
|
203
|
+
}
|
|
204
|
+
async stop(timeout) {
|
|
205
|
+
if (this.runner) {
|
|
206
|
+
this.runner.stop();
|
|
207
|
+
this.runner = void 0;
|
|
208
|
+
}
|
|
209
|
+
return await super.stop(timeout);
|
|
210
|
+
}
|
|
211
|
+
async generateResults(tasks, previousResults, inPayloads) {
|
|
212
|
+
const results = await Promise.allSettled(
|
|
213
|
+
tasks == null ? void 0 : tasks.map(async (task) => {
|
|
214
|
+
const witness = asWitnessInstance(task.module);
|
|
215
|
+
const input = task.input ?? false;
|
|
216
|
+
const inPayloadsFound = input === true ? inPayloads : input === false ? [] : this.processPreviousResults(previousResults, await this.inputAddresses(input));
|
|
217
|
+
if (witness) {
|
|
218
|
+
return [witness.address, await witness.observe(inPayloadsFound)];
|
|
219
|
+
}
|
|
220
|
+
const diviner = asDivinerInstance(task.module);
|
|
221
|
+
if (diviner) {
|
|
222
|
+
return [diviner.address, await diviner.divine(inPayloadsFound)];
|
|
223
|
+
}
|
|
224
|
+
throw new Error("Unsupported module type");
|
|
225
|
+
})
|
|
226
|
+
);
|
|
227
|
+
const finalResult = {};
|
|
228
|
+
for (const result of results.filter(fulfilled)) {
|
|
229
|
+
const [address, payloads] = result.value;
|
|
230
|
+
finalResult[address] = finalResult[address] ?? [];
|
|
231
|
+
finalResult[address].push(...payloads);
|
|
232
|
+
}
|
|
233
|
+
if (this.throwErrors) {
|
|
234
|
+
const errors = results.filter(rejected).map((result) => result.reason);
|
|
235
|
+
if (errors.length > 0) {
|
|
236
|
+
throw new Error("At least one module failed");
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
return finalResult;
|
|
240
|
+
}
|
|
241
|
+
async inputAddresses(input) {
|
|
242
|
+
if (Array.isArray(input)) {
|
|
243
|
+
return (await Promise.all(input.map(async (inputItem) => await this.inputAddresses(inputItem)))).flat();
|
|
244
|
+
} else {
|
|
245
|
+
const resolved = await this.resolve(input);
|
|
246
|
+
return resolved ? [resolved.address] : [];
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
processPreviousResults(payloads, inputs) {
|
|
250
|
+
return inputs.flatMap((input) => payloads[input] ?? []);
|
|
209
251
|
}
|
|
210
252
|
};
|
|
211
253
|
|
|
@@ -239,12 +281,8 @@ export * from "@xyo-network/sentinel-abstract";
|
|
|
239
281
|
export * from "@xyo-network/sentinel-model";
|
|
240
282
|
export {
|
|
241
283
|
MemorySentinel,
|
|
242
|
-
SentinelAutomationSchema,
|
|
243
|
-
SentinelEventAutomationSchema,
|
|
244
|
-
SentinelIntervalAutomationSchema,
|
|
245
284
|
SentinelIntervalAutomationWrapper,
|
|
246
285
|
SentinelRunner,
|
|
247
|
-
SentinelWrapper
|
|
248
|
-
isSentinelIntervalAutomation
|
|
286
|
+
SentinelWrapper
|
|
249
287
|
};
|
|
250
288
|
//# sourceMappingURL=index.js.map
|
package/dist/node/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/Automation.ts","../../src/MemorySentinel.ts","../../src/SentinelIntervalAutomationWrapper.ts","../../src/SentinelRunner.ts","../../src/Wrapper.ts","../../src/index.ts"],"sourcesContent":["import { isPayloadOfSchemaType, Payload } from '@xyo-network/payload-model'\nimport { Task } from '@xyo-network/sentinel-model'\n\nexport type SentinelAutomationSchema = 'network.xyo.automation'\nexport const SentinelAutomationSchema: SentinelAutomationSchema = 'network.xyo.automation'\n\nexport type SentinelIntervalAutomationSchema = 'network.xyo.automation.interval'\nexport const SentinelIntervalAutomationSchema: SentinelIntervalAutomationSchema = 'network.xyo.automation.interval'\n\nexport type SentinelEventAutomationSchema = 'network.xyo.automation.event'\nexport const SentinelEventAutomationSchema: SentinelEventAutomationSchema = 'network.xyo.automation.event'\n\nexport type SentinelBaseAutomationPayload<T extends Payload> = Payload<\n {\n tasks?: Task[]\n type?: 'interval' | 'event'\n } & T\n>\n\n/** Settings for an Interval Automation */\nexport type SentinelIntervalAutomationPayload = SentinelBaseAutomationPayload<{\n /** Epoch after which any reoccurrence stops */\n end?: number\n\n /** Time between triggers [non-repeating if undefined] */\n frequency?: number\n\n /** Units for frequency field [hour if undefined] */\n frequencyUnits?: 'second' | 'minute' | 'hour' | 'day'\n\n /** Remaining triggers [infinite if undefined] */\n remaining?: number\n\n schema: SentinelIntervalAutomationSchema\n\n /** Epoch of the next trigger */\n start: number\n\n /** The type of automation */\n type: 'interval'\n}>\n\nexport const isSentinelIntervalAutomation = isPayloadOfSchemaType<SentinelIntervalAutomationPayload>(SentinelIntervalAutomationSchema)\n\n/** Settings for an Event Automation */\nexport type SentinelEventAutomationPayload = SentinelBaseAutomationPayload<{\n schema: SentinelEventAutomationSchema\n type: 'event'\n}>\n\n/** Settings for an Automation */\nexport type SentinelAutomationPayload = Payload<\n SentinelIntervalAutomationPayload | SentinelEventAutomationPayload,\n SentinelAutomationSchema | SentinelIntervalAutomationSchema | SentinelEventAutomationSchema\n>\n","import { Address } from '@xylabs/hex'\nimport { fulfilled, rejected } from '@xylabs/promise'\nimport { asDivinerInstance } from '@xyo-network/diviner-model'\nimport { AnyConfigSchema } from '@xyo-network/module-model'\nimport { Payload } from '@xyo-network/payload-model'\nimport { AbstractSentinel } from '@xyo-network/sentinel-abstract'\nimport {\n ResolvedTask,\n SentinelConfig,\n SentinelConfigSchema,\n SentinelInstance,\n SentinelModuleEventData,\n SentinelParams,\n} from '@xyo-network/sentinel-model'\nimport { asWitnessInstance } from '@xyo-network/witness-model'\n\nexport type MemorySentinelParams<TConfig extends AnyConfigSchema<SentinelConfig> = AnyConfigSchema<SentinelConfig>> = SentinelParams<TConfig>\n\nexport class MemorySentinel<\n TParams extends MemorySentinelParams = MemorySentinelParams,\n TEventData extends SentinelModuleEventData<SentinelInstance<TParams>> = SentinelModuleEventData<SentinelInstance<TParams>>,\n> extends AbstractSentinel<TParams, TEventData> {\n static override configSchemas = [SentinelConfigSchema]\n\n async reportHandler(inPayloads: Payload[] = []): Promise<Payload[]> {\n await this.started('throw')\n const job = await this.jobPromise\n\n let index = 0\n let previousResults: Record<Address, Payload[]> = {}\n while (index < job.tasks.length) {\n const generatedPayloads = await this.generateResults(job.tasks[index], previousResults, inPayloads)\n previousResults = generatedPayloads\n index++\n }\n return Object.values(previousResults).flat()\n }\n\n private async generateResults(\n tasks: ResolvedTask[],\n previousResults: Record<Address, Payload[]>,\n inPayloads?: Payload[],\n ): Promise<Record<Address, Payload[]>> {\n const results: PromiseSettledResult<[Address, Payload[]]>[] = await Promise.allSettled(\n tasks?.map(async (task) => {\n const witness = asWitnessInstance(task.module)\n const input = task.input ?? false\n const inPayloadsFound =\n input === true ? inPayloads : input === false ? [] : this.processPreviousResults(previousResults, await this.inputAddresses(input))\n if (witness) {\n return [witness.address, await witness.observe(inPayloadsFound)]\n }\n const diviner = asDivinerInstance(task.module)\n if (diviner) {\n return [diviner.address, await diviner.divine(inPayloadsFound)]\n }\n throw new Error('Unsupported module type')\n }),\n )\n const finalResult: Record<Address, Payload[]> = {}\n for (const result of results.filter(fulfilled)) {\n const [address, payloads] = result.value\n finalResult[address] = finalResult[address] ?? []\n finalResult[address].push(...payloads)\n }\n if (this.throwErrors) {\n const errors = results.filter(rejected).map((result) => result.reason)\n if (errors.length > 0) {\n throw new Error('At least one module failed')\n }\n }\n return finalResult\n }\n\n private async inputAddresses(input: string | string[]): Promise<string[]> {\n if (Array.isArray(input)) {\n return (await Promise.all(input.map(async (inputItem) => await this.inputAddresses(inputItem)))).flat()\n } else {\n const resolved = await this.resolve(input)\n return resolved ? [resolved.address] : []\n }\n }\n\n private processPreviousResults(payloads: Record<string, Payload[]>, inputs: string[]) {\n return inputs.flatMap((input) => payloads[input] ?? [])\n }\n}\n","import { PayloadWrapper } from '@xyo-network/payload-wrapper'\n\nimport { SentinelIntervalAutomationPayload } from './Automation'\n\nexport class SentinelIntervalAutomationWrapper<\n T extends SentinelIntervalAutomationPayload = SentinelIntervalAutomationPayload,\n> extends PayloadWrapper<T> {\n constructor(payload: T) {\n super(payload)\n }\n\n protected get frequencyMillis() {\n const frequency = this.jsonPayload().frequency\n if (frequency === undefined) return Number.POSITIVE_INFINITY\n switch (this.jsonPayload().frequencyUnits ?? 'hour') {\n case 'second': {\n return frequency * 1000\n }\n case 'minute': {\n return frequency * 60 * 1000\n }\n case 'hour': {\n return frequency * 60 * 60 * 1000\n }\n case 'day': {\n return frequency * 24 * 60 * 60 * 1000\n }\n }\n }\n\n protected get remaining() {\n //if remaining is not defined, we assume Infinity\n return this.payload().remaining ?? Number.POSITIVE_INFINITY\n }\n\n next() {\n this.jsonPayload().start = this.jsonPayload().start + this.frequencyMillis\n this.consumeRemaining()\n this.checkEnd()\n return this\n }\n\n protected checkEnd() {\n if (this.jsonPayload().start > (this.jsonPayload().end ?? Number.POSITIVE_INFINITY)) {\n this.jsonPayload().start = Number.POSITIVE_INFINITY\n }\n }\n\n protected consumeRemaining(count = 1) {\n const remaining = this.remaining - count\n this.jsonPayload().remaining = remaining\n\n if (remaining <= 0) {\n this.jsonPayload().start = Number.POSITIVE_INFINITY\n }\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { Payload } from '@xyo-network/payload-model'\nimport { PayloadWrapper } from '@xyo-network/payload-wrapper'\nimport { SentinelInstance } from '@xyo-network/sentinel-model'\n\nimport { isSentinelIntervalAutomation, SentinelAutomationPayload, SentinelIntervalAutomationPayload } from './Automation'\nimport { SentinelIntervalAutomationWrapper } from './SentinelIntervalAutomationWrapper'\n\nexport type OnSentinelRunnerTriggerResult = (result: Payload[]) => void\n\nexport class SentinelRunner {\n protected _automations: Record<string, SentinelAutomationPayload> = {}\n protected onTriggerResult: OnSentinelRunnerTriggerResult | undefined\n protected sentinel: SentinelInstance\n protected timeoutId?: NodeJS.Timeout | string | number\n\n constructor(sentinel: SentinelInstance, automations?: SentinelAutomationPayload[], onTriggerResult?: OnSentinelRunnerTriggerResult) {\n this.sentinel = sentinel\n this.onTriggerResult = onTriggerResult\n if (automations) for (const automation of automations) this.add(automation)\n }\n\n get automations() {\n return this._automations\n }\n\n private get next() {\n // eslint-disable-next-line unicorn/no-array-reduce\n return Object.values(this._automations).reduce<SentinelAutomationPayload | undefined>((previous, current) => {\n if (isSentinelIntervalAutomation(current) && isSentinelIntervalAutomation(previous)) {\n return current.start < (previous?.start ?? Number.POSITIVE_INFINITY) ? current : previous\n }\n return current\n // eslint-disable-next-line unicorn/no-useless-undefined\n }, undefined)\n }\n\n async add(automation: SentinelAutomationPayload, restart = true) {\n const hash = await PayloadWrapper.hashAsync(automation)\n this._automations[hash] = automation\n if (restart) await this.restart()\n return hash\n }\n\n find(hash: string) {\n Object.entries(this._automations).find(([key]) => key === hash)\n }\n\n async remove(hash: string, restart = true) {\n delete this._automations[hash]\n if (restart) await this.restart()\n }\n\n removeAll() {\n this.stop()\n this._automations = {}\n }\n\n async restart() {\n this.stop()\n await this.start()\n }\n\n async start() {\n assertEx(this.timeoutId === undefined, 'Already started')\n const automation = this.next\n if (isSentinelIntervalAutomation(automation)) {\n const delay = automation.start - Date.now()\n if (delay < 0) {\n //automation is due, just do it\n await this.trigger(automation)\n } else {\n this.timeoutId = setTimeout(\n async () => {\n this.timeoutId = undefined\n await this.start()\n },\n delay > 0 ? delay : 0,\n )\n }\n }\n }\n\n stop() {\n if (this.timeoutId) {\n clearTimeout(this.timeoutId)\n this.timeoutId = undefined\n }\n }\n\n async update(hash: string, automation: SentinelAutomationPayload, restart = true) {\n await this.remove(hash, false)\n await this.add(automation, false)\n if (restart) await this.restart()\n }\n\n private async trigger(automation: SentinelIntervalAutomationPayload) {\n const wrapper = new SentinelIntervalAutomationWrapper(automation)\n await this.remove(await wrapper.hashAsync(), false)\n wrapper.next()\n await this.add(wrapper.jsonPayload(), false)\n const triggerResult = await this.sentinel.report()\n this.onTriggerResult?.(triggerResult)\n await this.start()\n }\n}\n","import { ArchivistInstance } from '@xyo-network/archivist'\nimport { constructableModuleWrapper, ModuleWrapper } from '@xyo-network/module-wrapper'\nimport { Payload } from '@xyo-network/payload-model'\nimport {\n isSentinelInstance,\n isSentinelModule,\n SentinelInstance,\n SentinelModule,\n SentinelReportQuery,\n SentinelReportQuerySchema,\n} from '@xyo-network/sentinel-model'\nimport { WitnessInstance } from '@xyo-network/witness-model'\n\nconstructableModuleWrapper()\nexport class SentinelWrapper<TModule extends SentinelModule = SentinelModule>\n extends ModuleWrapper<TModule>\n implements SentinelInstance<TModule['params']>\n{\n static override instanceIdentityCheck = isSentinelInstance\n static override moduleIdentityCheck = isSentinelModule\n static override requiredQueries = [SentinelReportQuerySchema, ...super.requiredQueries]\n\n archivists(): Promise<ArchivistInstance[]> {\n throw new Error('Not supported')\n }\n\n async report(payloads?: Payload[]): Promise<Payload[]> {\n const queryPayload: SentinelReportQuery = { schema: SentinelReportQuerySchema }\n const result = await this.sendQuery(queryPayload, payloads)\n return result\n }\n\n witnesses(): Promise<WitnessInstance[]> {\n throw new Error('Not supported')\n }\n}\n","export * from './Automation'\nexport * from './MemorySentinel'\nexport * from './SentinelIntervalAutomationWrapper'\nexport * from './SentinelRunner'\nexport * from './Wrapper'\nexport * from '@xyo-network/sentinel-abstract'\nexport * from '@xyo-network/sentinel-model'\n"],"mappings":"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","names":["PayloadWrapper","PayloadWrapper"]}
|
|
1
|
+
{"version":3,"sources":["../../src/MemorySentinel.ts","../../src/SentinelRunner.ts","../../src/SentinelIntervalAutomationWrapper.ts","../../src/Wrapper.ts","../../src/index.ts"],"sourcesContent":["import { Address } from '@xylabs/hex'\nimport { fulfilled, rejected } from '@xylabs/promise'\nimport { asDivinerInstance } from '@xyo-network/diviner-model'\nimport { AnyConfigSchema } from '@xyo-network/module-model'\nimport { Payload } from '@xyo-network/payload-model'\nimport { AbstractSentinel } from '@xyo-network/sentinel-abstract'\nimport {\n ResolvedTask,\n SentinelConfig,\n SentinelConfigSchema,\n SentinelInstance,\n SentinelModuleEventData,\n SentinelParams,\n} from '@xyo-network/sentinel-model'\nimport { asWitnessInstance } from '@xyo-network/witness-model'\n\nimport { SentinelRunner } from './SentinelRunner'\n\nexport type MemorySentinelParams<TConfig extends AnyConfigSchema<SentinelConfig> = AnyConfigSchema<SentinelConfig>> = SentinelParams<TConfig>\n\nexport class MemorySentinel<\n TParams extends MemorySentinelParams = MemorySentinelParams,\n TEventData extends SentinelModuleEventData<SentinelInstance<TParams>> = SentinelModuleEventData<SentinelInstance<TParams>>,\n> extends AbstractSentinel<TParams, TEventData> {\n static override configSchemas = [SentinelConfigSchema]\n\n private runner?: SentinelRunner\n\n async reportHandler(inPayloads: Payload[] = []): Promise<Payload[]> {\n await this.started('throw')\n const job = await this.jobPromise\n\n let index = 0\n let previousResults: Record<Address, Payload[]> = {}\n while (index < job.tasks.length) {\n const generatedPayloads = await this.generateResults(job.tasks[index], previousResults, inPayloads)\n previousResults = generatedPayloads\n index++\n }\n return Object.values(previousResults).flat()\n }\n\n override async start(timeout?: number | undefined): Promise<boolean> {\n if (await super.start(timeout)) {\n if ((this.config.automations?.length ?? 0) > 0) {\n this.runner = new SentinelRunner(this, this.config.automations)\n await this.runner.start()\n }\n return true\n }\n return false\n }\n\n override async stop(timeout?: number | undefined): Promise<boolean> {\n if (this.runner) {\n this.runner.stop()\n this.runner = undefined\n }\n return await super.stop(timeout)\n }\n\n private async generateResults(\n tasks: ResolvedTask[],\n previousResults: Record<Address, Payload[]>,\n inPayloads?: Payload[],\n ): Promise<Record<Address, Payload[]>> {\n const results: PromiseSettledResult<[Address, Payload[]]>[] = await Promise.allSettled(\n tasks?.map(async (task) => {\n const witness = asWitnessInstance(task.module)\n const input = task.input ?? false\n const inPayloadsFound =\n input === true ? inPayloads : input === false ? [] : this.processPreviousResults(previousResults, await this.inputAddresses(input))\n if (witness) {\n return [witness.address, await witness.observe(inPayloadsFound)]\n }\n const diviner = asDivinerInstance(task.module)\n if (diviner) {\n return [diviner.address, await diviner.divine(inPayloadsFound)]\n }\n throw new Error('Unsupported module type')\n }),\n )\n const finalResult: Record<Address, Payload[]> = {}\n for (const result of results.filter(fulfilled)) {\n const [address, payloads] = result.value\n finalResult[address] = finalResult[address] ?? []\n finalResult[address].push(...payloads)\n }\n if (this.throwErrors) {\n const errors = results.filter(rejected).map((result) => result.reason)\n if (errors.length > 0) {\n throw new Error('At least one module failed')\n }\n }\n return finalResult\n }\n\n private async inputAddresses(input: string | string[]): Promise<string[]> {\n if (Array.isArray(input)) {\n return (await Promise.all(input.map(async (inputItem) => await this.inputAddresses(inputItem)))).flat()\n } else {\n const resolved = await this.resolve(input)\n return resolved ? [resolved.address] : []\n }\n }\n\n private processPreviousResults(payloads: Record<string, Payload[]>, inputs: string[]) {\n return inputs.flatMap((input) => payloads[input] ?? [])\n }\n}\n","import { assertEx } from '@xylabs/assert'\nimport { Payload } from '@xyo-network/payload-model'\nimport { PayloadWrapper } from '@xyo-network/payload-wrapper'\nimport {\n isSentinelIntervalAutomation,\n SentinelAutomationPayload,\n SentinelInstance,\n SentinelIntervalAutomationPayload,\n} from '@xyo-network/sentinel-model'\n\nimport { SentinelIntervalAutomationWrapper } from './SentinelIntervalAutomationWrapper'\n\nexport type OnSentinelRunnerTriggerResult = (result: Payload[]) => void\n\nexport class SentinelRunner {\n protected _automations: Record<string, SentinelAutomationPayload> = {}\n protected onTriggerResult: OnSentinelRunnerTriggerResult | undefined\n protected sentinel: SentinelInstance\n protected timeoutId?: NodeJS.Timeout | string | number\n\n constructor(sentinel: SentinelInstance, automations?: SentinelAutomationPayload[], onTriggerResult?: OnSentinelRunnerTriggerResult) {\n this.sentinel = sentinel\n this.onTriggerResult = onTriggerResult\n if (automations) for (const automation of automations) this.add(automation)\n }\n\n get automations() {\n return this._automations\n }\n\n private get next() {\n // eslint-disable-next-line unicorn/no-array-reduce\n return Object.values(this._automations).reduce<SentinelAutomationPayload | undefined>((previous, current) => {\n if (isSentinelIntervalAutomation(current) && isSentinelIntervalAutomation(previous)) {\n return current.start < (previous?.start ?? Number.POSITIVE_INFINITY) ? current : previous\n }\n return current\n // eslint-disable-next-line unicorn/no-useless-undefined\n }, undefined)\n }\n\n async add(automation: SentinelAutomationPayload, restart = true) {\n const hash = await PayloadWrapper.hashAsync(automation)\n this._automations[hash] = automation\n if (restart) await this.restart()\n return hash\n }\n\n find(hash: string) {\n Object.entries(this._automations).find(([key]) => key === hash)\n }\n\n async remove(hash: string, restart = true) {\n delete this._automations[hash]\n if (restart) await this.restart()\n }\n\n removeAll() {\n this.stop()\n this._automations = {}\n }\n\n async restart() {\n this.stop()\n await this.start()\n }\n\n async start() {\n // NOTE: Keep async to match module start signature\n await Promise.resolve()\n assertEx(this.timeoutId === undefined, 'Already started')\n const automation = this.next\n if (isSentinelIntervalAutomation(automation)) {\n const now = Date.now()\n const start = Math.max(automation.start ?? now, now)\n const delay = Math.max(start - now, 0)\n if (delay < Number.POSITIVE_INFINITY) {\n this.timeoutId = setTimeout(async () => {\n try {\n // Run the automation\n await this.trigger(automation)\n this.stop()\n } finally {\n // No matter what start the next automation\n await this.start()\n }\n }, delay)\n }\n }\n }\n\n stop() {\n if (this.timeoutId) {\n clearTimeout(this.timeoutId)\n this.timeoutId = undefined\n }\n }\n\n async update(hash: string, automation: SentinelAutomationPayload, restart = true) {\n await this.remove(hash, false)\n await this.add(automation, false)\n if (restart) await this.restart()\n }\n\n private async trigger(automation: SentinelIntervalAutomationPayload) {\n const wrapper = new SentinelIntervalAutomationWrapper(automation)\n await this.remove(await wrapper.hashAsync(), false)\n wrapper.next()\n await this.add(wrapper.jsonPayload(), false)\n const triggerResult = await this.sentinel.report()\n this.onTriggerResult?.(triggerResult)\n // await this.start()\n }\n}\n","import { PayloadWrapper } from '@xyo-network/payload-wrapper'\nimport { SentinelIntervalAutomationPayload } from '@xyo-network/sentinel-model'\n\nexport class SentinelIntervalAutomationWrapper<\n T extends SentinelIntervalAutomationPayload = SentinelIntervalAutomationPayload,\n> extends PayloadWrapper<T> {\n constructor(payload: T) {\n super(payload)\n }\n\n protected get frequencyMillis() {\n const frequency = this.jsonPayload().frequency\n if (frequency === undefined) return Number.POSITIVE_INFINITY\n const frequencyUnits = this.jsonPayload().frequencyUnits\n switch (frequencyUnits ?? 'hour') {\n case 'second': {\n return frequency * 1000\n }\n case 'minute': {\n return frequency * 60 * 1000\n }\n case 'hour': {\n return frequency * 60 * 60 * 1000\n }\n case 'day': {\n return frequency * 24 * 60 * 60 * 1000\n }\n default: {\n return Number.POSITIVE_INFINITY\n }\n }\n }\n\n protected get remaining() {\n return this.jsonPayload().remaining ?? Number.POSITIVE_INFINITY\n }\n\n next() {\n const now = Date.now()\n const previousStart = this.jsonPayload()?.start ?? now\n const start = Math.max(previousStart, now)\n const nextStart = start + this.frequencyMillis\n this.setStart(nextStart)\n this.consumeRemaining()\n this.checkEnd()\n return this\n }\n\n protected checkEnd() {\n if (this.jsonPayload().start > (this.jsonPayload().end ?? Number.POSITIVE_INFINITY)) {\n this.setStart(Number.POSITIVE_INFINITY)\n }\n }\n\n protected consumeRemaining(count = 1) {\n const remaining = Math.max(this.remaining - count, 0)\n this.setRemaining(remaining)\n if (remaining <= 0) this.setStart(Number.POSITIVE_INFINITY)\n }\n\n /**\n * Sets the remaining of the wrapped automation\n * @param remaining The remaining time in milliseconds\n */\n protected setRemaining(remaining: number) {\n this.obj.remaining = remaining\n }\n\n /**\n * Sets the start of the wrapped automation\n * @param start The start time in milliseconds\n */\n protected setStart(start: number) {\n this.obj.start = start\n }\n}\n","import { ArchivistInstance } from '@xyo-network/archivist'\nimport { constructableModuleWrapper, ModuleWrapper } from '@xyo-network/module-wrapper'\nimport { Payload } from '@xyo-network/payload-model'\nimport {\n isSentinelInstance,\n isSentinelModule,\n SentinelInstance,\n SentinelModule,\n SentinelReportQuery,\n SentinelReportQuerySchema,\n} from '@xyo-network/sentinel-model'\nimport { WitnessInstance } from '@xyo-network/witness-model'\n\nconstructableModuleWrapper()\nexport class SentinelWrapper<TModule extends SentinelModule = SentinelModule>\n extends ModuleWrapper<TModule>\n implements SentinelInstance<TModule['params']>\n{\n static override instanceIdentityCheck = isSentinelInstance\n static override moduleIdentityCheck = isSentinelModule\n static override requiredQueries = [SentinelReportQuerySchema, ...super.requiredQueries]\n\n archivists(): Promise<ArchivistInstance[]> {\n throw new Error('Not supported')\n }\n\n async report(payloads?: Payload[]): Promise<Payload[]> {\n const queryPayload: SentinelReportQuery = { schema: SentinelReportQuerySchema }\n const result = await this.sendQuery(queryPayload, payloads)\n return result\n }\n\n witnesses(): Promise<WitnessInstance[]> {\n throw new Error('Not supported')\n }\n}\n","export * from './MemorySentinel'\nexport * from './SentinelIntervalAutomationWrapper'\nexport * from './SentinelRunner'\nexport * from './Wrapper'\nexport * from '@xyo-network/sentinel-abstract'\nexport * from '@xyo-network/sentinel-model'\n"],"mappings":"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","names":["PayloadWrapper","PayloadWrapper"]}
|
package/package.json
CHANGED
|
@@ -13,28 +13,28 @@
|
|
|
13
13
|
"@xylabs/assert": "^2.13.20",
|
|
14
14
|
"@xylabs/hex": "^2.13.20",
|
|
15
15
|
"@xylabs/promise": "^2.13.20",
|
|
16
|
-
"@xyo-network/archivist": "~2.84.
|
|
17
|
-
"@xyo-network/diviner-model": "~2.84.
|
|
18
|
-
"@xyo-network/module-model": "~2.84.
|
|
19
|
-
"@xyo-network/module-wrapper": "~2.84.
|
|
20
|
-
"@xyo-network/payload-model": "~2.84.
|
|
21
|
-
"@xyo-network/payload-wrapper": "~2.84.
|
|
22
|
-
"@xyo-network/sentinel-abstract": "~2.84.
|
|
23
|
-
"@xyo-network/sentinel-model": "~2.84.
|
|
24
|
-
"@xyo-network/witness-model": "~2.84.
|
|
16
|
+
"@xyo-network/archivist": "~2.84.9",
|
|
17
|
+
"@xyo-network/diviner-model": "~2.84.9",
|
|
18
|
+
"@xyo-network/module-model": "~2.84.9",
|
|
19
|
+
"@xyo-network/module-wrapper": "~2.84.9",
|
|
20
|
+
"@xyo-network/payload-model": "~2.84.9",
|
|
21
|
+
"@xyo-network/payload-wrapper": "~2.84.9",
|
|
22
|
+
"@xyo-network/sentinel-abstract": "~2.84.9",
|
|
23
|
+
"@xyo-network/sentinel-model": "~2.84.9",
|
|
24
|
+
"@xyo-network/witness-model": "~2.84.9"
|
|
25
25
|
},
|
|
26
26
|
"description": "Primary SDK for using XYO Protocol 2.0",
|
|
27
27
|
"devDependencies": {
|
|
28
28
|
"@xylabs/delay": "^2.13.20",
|
|
29
|
-
"@xylabs/ts-scripts-yarn3": "^3.2.
|
|
30
|
-
"@xylabs/tsconfig": "^3.2.
|
|
31
|
-
"@xyo-network/abstract-witness": "~2.84.
|
|
32
|
-
"@xyo-network/account": "~2.84.
|
|
33
|
-
"@xyo-network/boundwitness-model": "~2.84.
|
|
34
|
-
"@xyo-network/hash": "~2.84.
|
|
35
|
-
"@xyo-network/id-payload-plugin": "^2.84.
|
|
36
|
-
"@xyo-network/node-memory": "~2.84.
|
|
37
|
-
"@xyo-network/witness-adhoc": "~2.84.
|
|
29
|
+
"@xylabs/ts-scripts-yarn3": "^3.2.25",
|
|
30
|
+
"@xylabs/tsconfig": "^3.2.25",
|
|
31
|
+
"@xyo-network/abstract-witness": "~2.84.9",
|
|
32
|
+
"@xyo-network/account": "~2.84.9",
|
|
33
|
+
"@xyo-network/boundwitness-model": "~2.84.9",
|
|
34
|
+
"@xyo-network/hash": "~2.84.9",
|
|
35
|
+
"@xyo-network/id-payload-plugin": "^2.84.3",
|
|
36
|
+
"@xyo-network/node-memory": "~2.84.9",
|
|
37
|
+
"@xyo-network/witness-adhoc": "~2.84.9",
|
|
38
38
|
"typescript": "^5.3.3"
|
|
39
39
|
},
|
|
40
40
|
"types": "dist/node/index.d.ts",
|
|
@@ -75,6 +75,6 @@
|
|
|
75
75
|
"url": "https://github.com/XYOracleNetwork/sdk-xyo-client-js.git"
|
|
76
76
|
},
|
|
77
77
|
"sideEffects": false,
|
|
78
|
-
"version": "2.84.
|
|
78
|
+
"version": "2.84.9",
|
|
79
79
|
"type": "module"
|
|
80
80
|
}
|
package/src/MemorySentinel.ts
CHANGED
|
@@ -14,6 +14,8 @@ import {
|
|
|
14
14
|
} from '@xyo-network/sentinel-model'
|
|
15
15
|
import { asWitnessInstance } from '@xyo-network/witness-model'
|
|
16
16
|
|
|
17
|
+
import { SentinelRunner } from './SentinelRunner'
|
|
18
|
+
|
|
17
19
|
export type MemorySentinelParams<TConfig extends AnyConfigSchema<SentinelConfig> = AnyConfigSchema<SentinelConfig>> = SentinelParams<TConfig>
|
|
18
20
|
|
|
19
21
|
export class MemorySentinel<
|
|
@@ -22,6 +24,8 @@ export class MemorySentinel<
|
|
|
22
24
|
> extends AbstractSentinel<TParams, TEventData> {
|
|
23
25
|
static override configSchemas = [SentinelConfigSchema]
|
|
24
26
|
|
|
27
|
+
private runner?: SentinelRunner
|
|
28
|
+
|
|
25
29
|
async reportHandler(inPayloads: Payload[] = []): Promise<Payload[]> {
|
|
26
30
|
await this.started('throw')
|
|
27
31
|
const job = await this.jobPromise
|
|
@@ -36,6 +40,25 @@ export class MemorySentinel<
|
|
|
36
40
|
return Object.values(previousResults).flat()
|
|
37
41
|
}
|
|
38
42
|
|
|
43
|
+
override async start(timeout?: number | undefined): Promise<boolean> {
|
|
44
|
+
if (await super.start(timeout)) {
|
|
45
|
+
if ((this.config.automations?.length ?? 0) > 0) {
|
|
46
|
+
this.runner = new SentinelRunner(this, this.config.automations)
|
|
47
|
+
await this.runner.start()
|
|
48
|
+
}
|
|
49
|
+
return true
|
|
50
|
+
}
|
|
51
|
+
return false
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
override async stop(timeout?: number | undefined): Promise<boolean> {
|
|
55
|
+
if (this.runner) {
|
|
56
|
+
this.runner.stop()
|
|
57
|
+
this.runner = undefined
|
|
58
|
+
}
|
|
59
|
+
return await super.stop(timeout)
|
|
60
|
+
}
|
|
61
|
+
|
|
39
62
|
private async generateResults(
|
|
40
63
|
tasks: ResolvedTask[],
|
|
41
64
|
previousResults: Record<Address, Payload[]>,
|
|
@@ -1,6 +1,5 @@
|
|
|
1
1
|
import { PayloadWrapper } from '@xyo-network/payload-wrapper'
|
|
2
|
-
|
|
3
|
-
import { SentinelIntervalAutomationPayload } from './Automation'
|
|
2
|
+
import { SentinelIntervalAutomationPayload } from '@xyo-network/sentinel-model'
|
|
4
3
|
|
|
5
4
|
export class SentinelIntervalAutomationWrapper<
|
|
6
5
|
T extends SentinelIntervalAutomationPayload = SentinelIntervalAutomationPayload,
|
|
@@ -12,7 +11,8 @@ export class SentinelIntervalAutomationWrapper<
|
|
|
12
11
|
protected get frequencyMillis() {
|
|
13
12
|
const frequency = this.jsonPayload().frequency
|
|
14
13
|
if (frequency === undefined) return Number.POSITIVE_INFINITY
|
|
15
|
-
|
|
14
|
+
const frequencyUnits = this.jsonPayload().frequencyUnits
|
|
15
|
+
switch (frequencyUnits ?? 'hour') {
|
|
16
16
|
case 'second': {
|
|
17
17
|
return frequency * 1000
|
|
18
18
|
}
|
|
@@ -25,16 +25,22 @@ export class SentinelIntervalAutomationWrapper<
|
|
|
25
25
|
case 'day': {
|
|
26
26
|
return frequency * 24 * 60 * 60 * 1000
|
|
27
27
|
}
|
|
28
|
+
default: {
|
|
29
|
+
return Number.POSITIVE_INFINITY
|
|
30
|
+
}
|
|
28
31
|
}
|
|
29
32
|
}
|
|
30
33
|
|
|
31
34
|
protected get remaining() {
|
|
32
|
-
|
|
33
|
-
return this.payload().remaining ?? Number.POSITIVE_INFINITY
|
|
35
|
+
return this.jsonPayload().remaining ?? Number.POSITIVE_INFINITY
|
|
34
36
|
}
|
|
35
37
|
|
|
36
38
|
next() {
|
|
37
|
-
|
|
39
|
+
const now = Date.now()
|
|
40
|
+
const previousStart = this.jsonPayload()?.start ?? now
|
|
41
|
+
const start = Math.max(previousStart, now)
|
|
42
|
+
const nextStart = start + this.frequencyMillis
|
|
43
|
+
this.setStart(nextStart)
|
|
38
44
|
this.consumeRemaining()
|
|
39
45
|
this.checkEnd()
|
|
40
46
|
return this
|
|
@@ -42,16 +48,29 @@ export class SentinelIntervalAutomationWrapper<
|
|
|
42
48
|
|
|
43
49
|
protected checkEnd() {
|
|
44
50
|
if (this.jsonPayload().start > (this.jsonPayload().end ?? Number.POSITIVE_INFINITY)) {
|
|
45
|
-
this.
|
|
51
|
+
this.setStart(Number.POSITIVE_INFINITY)
|
|
46
52
|
}
|
|
47
53
|
}
|
|
48
54
|
|
|
49
55
|
protected consumeRemaining(count = 1) {
|
|
50
|
-
const remaining = this.remaining - count
|
|
51
|
-
this.
|
|
56
|
+
const remaining = Math.max(this.remaining - count, 0)
|
|
57
|
+
this.setRemaining(remaining)
|
|
58
|
+
if (remaining <= 0) this.setStart(Number.POSITIVE_INFINITY)
|
|
59
|
+
}
|
|
52
60
|
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
61
|
+
/**
|
|
62
|
+
* Sets the remaining of the wrapped automation
|
|
63
|
+
* @param remaining The remaining time in milliseconds
|
|
64
|
+
*/
|
|
65
|
+
protected setRemaining(remaining: number) {
|
|
66
|
+
this.obj.remaining = remaining
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* Sets the start of the wrapped automation
|
|
71
|
+
* @param start The start time in milliseconds
|
|
72
|
+
*/
|
|
73
|
+
protected setStart(start: number) {
|
|
74
|
+
this.obj.start = start
|
|
56
75
|
}
|
|
57
76
|
}
|
package/src/SentinelRunner.ts
CHANGED
|
@@ -1,9 +1,13 @@
|
|
|
1
1
|
import { assertEx } from '@xylabs/assert'
|
|
2
2
|
import { Payload } from '@xyo-network/payload-model'
|
|
3
3
|
import { PayloadWrapper } from '@xyo-network/payload-wrapper'
|
|
4
|
-
import {
|
|
4
|
+
import {
|
|
5
|
+
isSentinelIntervalAutomation,
|
|
6
|
+
SentinelAutomationPayload,
|
|
7
|
+
SentinelInstance,
|
|
8
|
+
SentinelIntervalAutomationPayload,
|
|
9
|
+
} from '@xyo-network/sentinel-model'
|
|
5
10
|
|
|
6
|
-
import { isSentinelIntervalAutomation, SentinelAutomationPayload, SentinelIntervalAutomationPayload } from './Automation'
|
|
7
11
|
import { SentinelIntervalAutomationWrapper } from './SentinelIntervalAutomationWrapper'
|
|
8
12
|
|
|
9
13
|
export type OnSentinelRunnerTriggerResult = (result: Payload[]) => void
|
|
@@ -62,21 +66,25 @@ export class SentinelRunner {
|
|
|
62
66
|
}
|
|
63
67
|
|
|
64
68
|
async start() {
|
|
69
|
+
// NOTE: Keep async to match module start signature
|
|
70
|
+
await Promise.resolve()
|
|
65
71
|
assertEx(this.timeoutId === undefined, 'Already started')
|
|
66
72
|
const automation = this.next
|
|
67
73
|
if (isSentinelIntervalAutomation(automation)) {
|
|
68
|
-
const
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
this.
|
|
74
|
+
const now = Date.now()
|
|
75
|
+
const start = Math.max(automation.start ?? now, now)
|
|
76
|
+
const delay = Math.max(start - now, 0)
|
|
77
|
+
if (delay < Number.POSITIVE_INFINITY) {
|
|
78
|
+
this.timeoutId = setTimeout(async () => {
|
|
79
|
+
try {
|
|
80
|
+
// Run the automation
|
|
81
|
+
await this.trigger(automation)
|
|
82
|
+
this.stop()
|
|
83
|
+
} finally {
|
|
84
|
+
// No matter what start the next automation
|
|
76
85
|
await this.start()
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
)
|
|
86
|
+
}
|
|
87
|
+
}, delay)
|
|
80
88
|
}
|
|
81
89
|
}
|
|
82
90
|
}
|
|
@@ -101,6 +109,6 @@ export class SentinelRunner {
|
|
|
101
109
|
await this.add(wrapper.jsonPayload(), false)
|
|
102
110
|
const triggerResult = await this.sentinel.report()
|
|
103
111
|
this.onTriggerResult?.(triggerResult)
|
|
104
|
-
await this.start()
|
|
112
|
+
// await this.start()
|
|
105
113
|
}
|
|
106
114
|
}
|
package/src/index.ts
CHANGED