@streamr/node 100.0.0-rc.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +78 -0
- package/README.md +54 -0
- package/bin/broker.ts +36 -0
- package/bin/config-wizard.js +4 -0
- package/bin/config-wizard.ts +17 -0
- package/bin/delete-expired-data.ts +41 -0
- package/bin/entry-point.ts +27 -0
- package/configs/development-1.env.json +68 -0
- package/configs/development-2.env.json +53 -0
- package/configs/development-3.env.json +53 -0
- package/configs/development-prod-resend.env.json +21 -0
- package/configs/docker-1.env.json +72 -0
- package/configs/docker-2.env.json +58 -0
- package/configs/docker-3.env.json +58 -0
- package/configuration.md +38 -0
- package/dist/bin/broker.d.ts +2 -0
- package/dist/bin/broker.js +42 -0
- package/dist/bin/broker.js.map +1 -0
- package/dist/bin/config-wizard.d.ts +2 -0
- package/dist/bin/config-wizard.js +22 -0
- package/dist/bin/config-wizard.js.map +1 -0
- package/dist/bin/delete-expired-data.d.ts +2 -0
- package/dist/bin/delete-expired-data.js +44 -0
- package/dist/bin/delete-expired-data.js.map +1 -0
- package/dist/bin/entry-point.d.ts +2 -0
- package/dist/bin/entry-point.js +28 -0
- package/dist/bin/entry-point.js.map +1 -0
- package/dist/package.json +78 -0
- package/dist/src/Plugin.d.ts +33 -0
- package/dist/src/Plugin.js +43 -0
- package/dist/src/Plugin.js.map +1 -0
- package/dist/src/apiAuthentication.d.ts +4 -0
- package/dist/src/apiAuthentication.js +16 -0
- package/dist/src/apiAuthentication.js.map +1 -0
- package/dist/src/broker.d.ts +8 -0
- package/dist/src/broker.js +69 -0
- package/dist/src/broker.js.map +1 -0
- package/dist/src/config/ConfigWizard.d.ts +5 -0
- package/dist/src/config/ConfigWizard.js +466 -0
- package/dist/src/config/ConfigWizard.js.map +1 -0
- package/dist/src/config/config.d.ts +26 -0
- package/dist/src/config/config.js +92 -0
- package/dist/src/config/config.js.map +1 -0
- package/dist/src/config/config.schema.json +86 -0
- package/dist/src/config/definitions.schema.json +35 -0
- package/dist/src/config/migration.d.ts +6 -0
- package/dist/src/config/migration.js +210 -0
- package/dist/src/config/migration.js.map +1 -0
- package/dist/src/config/validateConfig.d.ts +4 -0
- package/dist/src/config/validateConfig.js +40 -0
- package/dist/src/config/validateConfig.js.map +1 -0
- package/dist/src/exports.d.ts +3 -0
- package/dist/src/exports.js +6 -0
- package/dist/src/exports.js.map +1 -0
- package/dist/src/helpers/PayloadFormat.d.ts +19 -0
- package/dist/src/helpers/PayloadFormat.js +85 -0
- package/dist/src/helpers/PayloadFormat.js.map +1 -0
- package/dist/src/helpers/applyPluginClientConfigs.d.ts +3 -0
- package/dist/src/helpers/applyPluginClientConfigs.js +29 -0
- package/dist/src/helpers/applyPluginClientConfigs.js.map +1 -0
- package/dist/src/helpers/fetchOrThrow.d.ts +2 -0
- package/dist/src/helpers/fetchOrThrow.js +20 -0
- package/dist/src/helpers/fetchOrThrow.js.map +1 -0
- package/dist/src/helpers/generateMnemonicFromAddress.d.ts +5 -0
- package/dist/src/helpers/generateMnemonicFromAddress.js +16 -0
- package/dist/src/helpers/generateMnemonicFromAddress.js.map +1 -0
- package/dist/src/helpers/multiply.d.ts +1 -0
- package/dist/src/helpers/multiply.js +10 -0
- package/dist/src/helpers/multiply.js.map +1 -0
- package/dist/src/helpers/parser.d.ts +9 -0
- package/dist/src/helpers/parser.js +62 -0
- package/dist/src/helpers/parser.js.map +1 -0
- package/dist/src/helpers/partitions.d.ts +8 -0
- package/dist/src/helpers/partitions.js +32 -0
- package/dist/src/helpers/partitions.js.map +1 -0
- package/dist/src/helpers/weightedSample.d.ts +16 -0
- package/dist/src/helpers/weightedSample.js +35 -0
- package/dist/src/helpers/weightedSample.js.map +1 -0
- package/dist/src/httpServer.d.ts +16 -0
- package/dist/src/httpServer.js +71 -0
- package/dist/src/httpServer.js.map +1 -0
- package/dist/src/pluginRegistry.d.ts +3 -0
- package/dist/src/pluginRegistry.js +35 -0
- package/dist/src/pluginRegistry.js.map +1 -0
- package/dist/src/plugins/consoleMetrics/ConsoleMetricsPlugin.d.ts +12 -0
- package/dist/src/plugins/consoleMetrics/ConsoleMetricsPlugin.js +36 -0
- package/dist/src/plugins/consoleMetrics/ConsoleMetricsPlugin.js.map +1 -0
- package/dist/src/plugins/consoleMetrics/config.schema.json +18 -0
- package/dist/src/plugins/http/HttpPlugin.d.ts +8 -0
- package/dist/src/plugins/http/HttpPlugin.js +23 -0
- package/dist/src/plugins/http/HttpPlugin.js.map +1 -0
- package/dist/src/plugins/http/config.schema.json +12 -0
- package/dist/src/plugins/http/publishEndpoint.d.ts +3 -0
- package/dist/src/plugins/http/publishEndpoint.js +72 -0
- package/dist/src/plugins/http/publishEndpoint.js.map +1 -0
- package/dist/src/plugins/info/InfoPlugin.d.ts +9 -0
- package/dist/src/plugins/info/InfoPlugin.js +31 -0
- package/dist/src/plugins/info/InfoPlugin.js.map +1 -0
- package/dist/src/plugins/info/config.schema.json +12 -0
- package/dist/src/plugins/mqtt/Bridge.d.ts +40 -0
- package/dist/src/plugins/mqtt/Bridge.js +136 -0
- package/dist/src/plugins/mqtt/Bridge.js.map +1 -0
- package/dist/src/plugins/mqtt/MqttPlugin.d.ts +14 -0
- package/dist/src/plugins/mqtt/MqttPlugin.js +30 -0
- package/dist/src/plugins/mqtt/MqttPlugin.js.map +1 -0
- package/dist/src/plugins/mqtt/MqttServer.d.ts +22 -0
- package/dist/src/plugins/mqtt/MqttServer.js +109 -0
- package/dist/src/plugins/mqtt/MqttServer.js.map +1 -0
- package/dist/src/plugins/mqtt/config.schema.json +26 -0
- package/dist/src/plugins/operator/ConsistentHashRing.d.ts +20 -0
- package/dist/src/plugins/operator/ConsistentHashRing.js +64 -0
- package/dist/src/plugins/operator/ConsistentHashRing.js.map +1 -0
- package/dist/src/plugins/operator/ContractFacade.d.ts +80 -0
- package/dist/src/plugins/operator/ContractFacade.js +364 -0
- package/dist/src/plugins/operator/ContractFacade.js.map +1 -0
- package/dist/src/plugins/operator/MaintainTopologyHelper.d.ts +23 -0
- package/dist/src/plugins/operator/MaintainTopologyHelper.js +75 -0
- package/dist/src/plugins/operator/MaintainTopologyHelper.js.map +1 -0
- package/dist/src/plugins/operator/MaintainTopologyService.d.ts +11 -0
- package/dist/src/plugins/operator/MaintainTopologyService.js +57 -0
- package/dist/src/plugins/operator/MaintainTopologyService.js.map +1 -0
- package/dist/src/plugins/operator/OperatorFleetState.d.ts +33 -0
- package/dist/src/plugins/operator/OperatorFleetState.js +112 -0
- package/dist/src/plugins/operator/OperatorFleetState.js.map +1 -0
- package/dist/src/plugins/operator/OperatorPlugin.d.ts +50 -0
- package/dist/src/plugins/operator/OperatorPlugin.js +159 -0
- package/dist/src/plugins/operator/OperatorPlugin.js.map +1 -0
- package/dist/src/plugins/operator/StreamPartAssignments.d.ts +28 -0
- package/dist/src/plugins/operator/StreamPartAssignments.js +104 -0
- package/dist/src/plugins/operator/StreamPartAssignments.js.map +1 -0
- package/dist/src/plugins/operator/announceNodeToContract.d.ts +3 -0
- package/dist/src/plugins/operator/announceNodeToContract.js +39 -0
- package/dist/src/plugins/operator/announceNodeToContract.js.map +1 -0
- package/dist/src/plugins/operator/announceNodeToStream.d.ts +3 -0
- package/dist/src/plugins/operator/announceNodeToStream.js +25 -0
- package/dist/src/plugins/operator/announceNodeToStream.js.map +1 -0
- package/dist/src/plugins/operator/checkOperatorValueBreach.d.ts +2 -0
- package/dist/src/plugins/operator/checkOperatorValueBreach.js +21 -0
- package/dist/src/plugins/operator/checkOperatorValueBreach.js.map +1 -0
- package/dist/src/plugins/operator/closeExpiredFlags.d.ts +3 -0
- package/dist/src/plugins/operator/closeExpiredFlags.js +24 -0
- package/dist/src/plugins/operator/closeExpiredFlags.js.map +1 -0
- package/dist/src/plugins/operator/config.schema.json +152 -0
- package/dist/src/plugins/operator/createIsLeaderFn.d.ts +4 -0
- package/dist/src/plugins/operator/createIsLeaderFn.js +14 -0
- package/dist/src/plugins/operator/createIsLeaderFn.js.map +1 -0
- package/dist/src/plugins/operator/fetchRedundancyFactor.d.ts +2 -0
- package/dist/src/plugins/operator/fetchRedundancyFactor.js +43 -0
- package/dist/src/plugins/operator/fetchRedundancyFactor.js.map +1 -0
- package/dist/src/plugins/operator/formCoordinationStreamId.d.ts +3 -0
- package/dist/src/plugins/operator/formCoordinationStreamId.js +9 -0
- package/dist/src/plugins/operator/formCoordinationStreamId.js.map +1 -0
- package/dist/src/plugins/operator/heartbeatUtils.d.ts +67 -0
- package/dist/src/plugins/operator/heartbeatUtils.js +26 -0
- package/dist/src/plugins/operator/heartbeatUtils.js.map +1 -0
- package/dist/src/plugins/operator/inspectOverTime.d.ts +22 -0
- package/dist/src/plugins/operator/inspectOverTime.js +146 -0
- package/dist/src/plugins/operator/inspectOverTime.js.map +1 -0
- package/dist/src/plugins/operator/inspectRandomNode.d.ts +8 -0
- package/dist/src/plugins/operator/inspectRandomNode.js +44 -0
- package/dist/src/plugins/operator/inspectRandomNode.js.map +1 -0
- package/dist/src/plugins/operator/inspectionUtils.d.ts +23 -0
- package/dist/src/plugins/operator/inspectionUtils.js +120 -0
- package/dist/src/plugins/operator/inspectionUtils.js.map +1 -0
- package/dist/src/plugins/operator/maintainOperatorValue.d.ts +2 -0
- package/dist/src/plugins/operator/maintainOperatorValue.js +21 -0
- package/dist/src/plugins/operator/maintainOperatorValue.js.map +1 -0
- package/dist/src/plugins/operator/reviewSuspectNode.d.ts +24 -0
- package/dist/src/plugins/operator/reviewSuspectNode.js +56 -0
- package/dist/src/plugins/operator/reviewSuspectNode.js.map +1 -0
- package/dist/src/plugins/storage/Batch.d.ts +51 -0
- package/dist/src/plugins/storage/Batch.js +121 -0
- package/dist/src/plugins/storage/Batch.js.map +1 -0
- package/dist/src/plugins/storage/BatchManager.d.ts +27 -0
- package/dist/src/plugins/storage/BatchManager.js +117 -0
- package/dist/src/plugins/storage/BatchManager.js.map +1 -0
- package/dist/src/plugins/storage/Bucket.d.ts +23 -0
- package/dist/src/plugins/storage/Bucket.js +90 -0
- package/dist/src/plugins/storage/Bucket.js.map +1 -0
- package/dist/src/plugins/storage/BucketManager.d.ts +56 -0
- package/dist/src/plugins/storage/BucketManager.js +306 -0
- package/dist/src/plugins/storage/BucketManager.js.map +1 -0
- package/dist/src/plugins/storage/DataQueryFormat.d.ts +10 -0
- package/dist/src/plugins/storage/DataQueryFormat.js +51 -0
- package/dist/src/plugins/storage/DataQueryFormat.js.map +1 -0
- package/dist/src/plugins/storage/DeleteExpiredCmd.d.ts +28 -0
- package/dist/src/plugins/storage/DeleteExpiredCmd.js +155 -0
- package/dist/src/plugins/storage/DeleteExpiredCmd.js.map +1 -0
- package/dist/src/plugins/storage/SetMembershipSynchronizer.d.ts +33 -0
- package/dist/src/plugins/storage/SetMembershipSynchronizer.js +102 -0
- package/dist/src/plugins/storage/SetMembershipSynchronizer.js.map +1 -0
- package/dist/src/plugins/storage/Storage.d.ts +47 -0
- package/dist/src/plugins/storage/Storage.js +459 -0
- package/dist/src/plugins/storage/Storage.js.map +1 -0
- package/dist/src/plugins/storage/StorageConfig.d.ts +43 -0
- package/dist/src/plugins/storage/StorageConfig.js +83 -0
- package/dist/src/plugins/storage/StorageConfig.js.map +1 -0
- package/dist/src/plugins/storage/StorageEventListener.d.ts +17 -0
- package/dist/src/plugins/storage/StorageEventListener.js +46 -0
- package/dist/src/plugins/storage/StorageEventListener.js.map +1 -0
- package/dist/src/plugins/storage/StoragePlugin.d.ts +32 -0
- package/dist/src/plugins/storage/StoragePlugin.js +103 -0
- package/dist/src/plugins/storage/StoragePlugin.js.map +1 -0
- package/dist/src/plugins/storage/StoragePoller.d.ts +15 -0
- package/dist/src/plugins/storage/StoragePoller.js +47 -0
- package/dist/src/plugins/storage/StoragePoller.js.map +1 -0
- package/dist/src/plugins/storage/config.schema.json +88 -0
- package/dist/src/plugins/storage/dataMetadataEndpoint.d.ts +3 -0
- package/dist/src/plugins/storage/dataMetadataEndpoint.js +35 -0
- package/dist/src/plugins/storage/dataMetadataEndpoint.js.map +1 -0
- package/dist/src/plugins/storage/dataQueryEndpoint.d.ts +6 -0
- package/dist/src/plugins/storage/dataQueryEndpoint.js +181 -0
- package/dist/src/plugins/storage/dataQueryEndpoint.js.map +1 -0
- package/dist/src/plugins/storage/storageConfigEndpoint.d.ts +3 -0
- package/dist/src/plugins/storage/storageConfigEndpoint.js +31 -0
- package/dist/src/plugins/storage/storageConfigEndpoint.js.map +1 -0
- package/dist/src/plugins/subscriber/SubscriberPlugin.d.ts +16 -0
- package/dist/src/plugins/subscriber/SubscriberPlugin.js +22 -0
- package/dist/src/plugins/subscriber/SubscriberPlugin.js.map +1 -0
- package/dist/src/plugins/subscriber/config.schema.json +31 -0
- package/dist/src/plugins/websocket/Connection.d.ts +9 -0
- package/dist/src/plugins/websocket/Connection.js +45 -0
- package/dist/src/plugins/websocket/Connection.js.map +1 -0
- package/dist/src/plugins/websocket/PublishConnection.d.ts +12 -0
- package/dist/src/plugins/websocket/PublishConnection.js +46 -0
- package/dist/src/plugins/websocket/PublishConnection.js.map +1 -0
- package/dist/src/plugins/websocket/SubscribeConnection.d.ts +13 -0
- package/dist/src/plugins/websocket/SubscribeConnection.js +50 -0
- package/dist/src/plugins/websocket/SubscribeConnection.js.map +1 -0
- package/dist/src/plugins/websocket/WebsocketPlugin.d.ts +19 -0
- package/dist/src/plugins/websocket/WebsocketPlugin.js +26 -0
- package/dist/src/plugins/websocket/WebsocketPlugin.js.map +1 -0
- package/dist/src/plugins/websocket/WebsocketServer.d.ts +16 -0
- package/dist/src/plugins/websocket/WebsocketServer.js +132 -0
- package/dist/src/plugins/websocket/WebsocketServer.js.map +1 -0
- package/dist/src/plugins/websocket/config.schema.json +51 -0
- package/package.json +78 -0
- package/plugins.md +318 -0
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.OperatorFleetState = void 0;
|
|
7
|
+
const utils_1 = require("@streamr/utils");
|
|
8
|
+
const eventemitter3_1 = require("eventemitter3");
|
|
9
|
+
const min_1 = __importDefault(require("lodash/min"));
|
|
10
|
+
const once_1 = __importDefault(require("lodash/once"));
|
|
11
|
+
const heartbeatUtils_1 = require("./heartbeatUtils");
|
|
12
|
+
const logger = new utils_1.Logger(module);
|
|
13
|
+
// TODO: add abortSignal support
|
|
14
|
+
class OperatorFleetState extends eventemitter3_1.EventEmitter {
|
|
15
|
+
streamrClient;
|
|
16
|
+
coordinationStreamId;
|
|
17
|
+
timeProvider;
|
|
18
|
+
pruneAgeInMs;
|
|
19
|
+
pruneIntervalInMs;
|
|
20
|
+
heartbeatIntervalInMs;
|
|
21
|
+
latencyExtraInMs;
|
|
22
|
+
warmupPeriodInMs;
|
|
23
|
+
latestHeartbeats = new Map();
|
|
24
|
+
abortController = new AbortController();
|
|
25
|
+
ready = new utils_1.Gate(false);
|
|
26
|
+
subscription;
|
|
27
|
+
static createOperatorFleetStateBuilder(streamrClient, heartbeatIntervalInMs, pruneAgeInMs, pruneIntervalInMs, latencyExtraInMs, warmupPeriodInMs, timeProvider = Date.now) {
|
|
28
|
+
return (coordinationStreamId) => new OperatorFleetState(streamrClient, coordinationStreamId, heartbeatIntervalInMs, pruneAgeInMs, pruneIntervalInMs, latencyExtraInMs, warmupPeriodInMs, timeProvider);
|
|
29
|
+
}
|
|
30
|
+
constructor(streamrClient, coordinationStreamId, heartbeatIntervalInMs, pruneAgeInMs, pruneIntervalInMs, latencyExtraInMs, warmupPeriodInMs, timeProvider = Date.now) {
|
|
31
|
+
super();
|
|
32
|
+
this.streamrClient = streamrClient;
|
|
33
|
+
this.coordinationStreamId = coordinationStreamId;
|
|
34
|
+
this.timeProvider = timeProvider;
|
|
35
|
+
this.pruneAgeInMs = pruneAgeInMs;
|
|
36
|
+
this.pruneIntervalInMs = pruneIntervalInMs;
|
|
37
|
+
this.heartbeatIntervalInMs = heartbeatIntervalInMs;
|
|
38
|
+
this.latencyExtraInMs = latencyExtraInMs;
|
|
39
|
+
this.warmupPeriodInMs = warmupPeriodInMs;
|
|
40
|
+
}
|
|
41
|
+
async start() {
|
|
42
|
+
if (this.subscription !== undefined) {
|
|
43
|
+
throw new Error('already started');
|
|
44
|
+
}
|
|
45
|
+
const startTime = this.timeProvider();
|
|
46
|
+
this.subscription = await this.streamrClient.subscribe(this.coordinationStreamId, (rawContent) => {
|
|
47
|
+
// Ignore messages during warmup period. This is needed because network nodes may propagate old stream messages
|
|
48
|
+
// from cache.
|
|
49
|
+
if ((this.timeProvider() - startTime) < this.warmupPeriodInMs) { // TODO: write test
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
let message;
|
|
53
|
+
try {
|
|
54
|
+
message = heartbeatUtils_1.HeartbeatMessageSchema.parse(rawContent);
|
|
55
|
+
}
|
|
56
|
+
catch (err) {
|
|
57
|
+
logger.warn('Received invalid message in coordination stream', {
|
|
58
|
+
coordinationStreamId: this.coordinationStreamId,
|
|
59
|
+
reason: err?.reason
|
|
60
|
+
});
|
|
61
|
+
return;
|
|
62
|
+
}
|
|
63
|
+
if (message.msgType === 'heartbeat') {
|
|
64
|
+
const nodeId = message.peerDescriptor.nodeId;
|
|
65
|
+
const exists = this.latestHeartbeats.has(nodeId);
|
|
66
|
+
this.latestHeartbeats.set(nodeId, {
|
|
67
|
+
timestamp: this.timeProvider(),
|
|
68
|
+
peerDescriptor: message.peerDescriptor
|
|
69
|
+
});
|
|
70
|
+
if (!exists) {
|
|
71
|
+
this.emit('added', nodeId);
|
|
72
|
+
}
|
|
73
|
+
if (!this.ready.isOpen()) {
|
|
74
|
+
this.launchOpenReadyGateTimer();
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
});
|
|
78
|
+
(0, utils_1.setAbortableInterval)(() => this.pruneOfflineNodes(), this.pruneIntervalInMs, this.abortController.signal);
|
|
79
|
+
}
|
|
80
|
+
async waitUntilReady() {
|
|
81
|
+
return this.ready.waitUntilOpen();
|
|
82
|
+
}
|
|
83
|
+
async destroy() {
|
|
84
|
+
this.abortController.abort();
|
|
85
|
+
await this.subscription?.unsubscribe();
|
|
86
|
+
}
|
|
87
|
+
getLeaderNodeId() {
|
|
88
|
+
return (0, min_1.default)(this.getNodeIds()); // we just need the leader to be consistent
|
|
89
|
+
}
|
|
90
|
+
getNodeIds() {
|
|
91
|
+
return [...this.latestHeartbeats.keys()];
|
|
92
|
+
}
|
|
93
|
+
getPeerDescriptor(nodeId) {
|
|
94
|
+
return this.latestHeartbeats.get(nodeId)?.peerDescriptor;
|
|
95
|
+
}
|
|
96
|
+
launchOpenReadyGateTimer = (0, once_1.default)(() => {
|
|
97
|
+
(0, utils_1.setAbortableTimeout)(() => {
|
|
98
|
+
this.ready.open();
|
|
99
|
+
}, this.heartbeatIntervalInMs + this.latencyExtraInMs, this.abortController.signal);
|
|
100
|
+
});
|
|
101
|
+
pruneOfflineNodes() {
|
|
102
|
+
const now = this.timeProvider();
|
|
103
|
+
for (const [nodeId, { timestamp }] of this.latestHeartbeats) {
|
|
104
|
+
if (now - timestamp >= this.pruneAgeInMs) {
|
|
105
|
+
this.latestHeartbeats.delete(nodeId);
|
|
106
|
+
this.emit('removed', nodeId);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
exports.OperatorFleetState = OperatorFleetState;
|
|
112
|
+
//# sourceMappingURL=OperatorFleetState.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"OperatorFleetState.js","sourceRoot":"","sources":["../../../../src/plugins/operator/OperatorFleetState.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
import { EthereumAddress } from '@streamr/utils';
|
|
2
|
+
import { Schema } from 'ajv';
|
|
3
|
+
import { Overrides } from 'ethers';
|
|
4
|
+
import { StreamrClient, SignerWithProvider } from '@streamr/sdk';
|
|
5
|
+
import { Plugin } from '../../Plugin';
|
|
6
|
+
export interface OperatorPluginConfig {
|
|
7
|
+
operatorContractAddress: string;
|
|
8
|
+
heartbeatUpdateIntervalInMs: number;
|
|
9
|
+
heartbeatTimeoutInMs: number;
|
|
10
|
+
fleetState: {
|
|
11
|
+
pruneAgeInMs: number;
|
|
12
|
+
pruneIntervalInMs: number;
|
|
13
|
+
latencyExtraInMs: number;
|
|
14
|
+
warmupPeriodInMs: number;
|
|
15
|
+
};
|
|
16
|
+
checkOperatorValueBreachIntervalInMs: number;
|
|
17
|
+
announceNodeToContract: {
|
|
18
|
+
pollIntervalInMs: number;
|
|
19
|
+
writeIntervalInMs: number;
|
|
20
|
+
};
|
|
21
|
+
maintainOperatorValue: {
|
|
22
|
+
intervalInMs: number;
|
|
23
|
+
withdrawLimitSafetyFraction: number;
|
|
24
|
+
minSponsorshipEarningsInWithdraw: number;
|
|
25
|
+
maxSponsorshipsInWithdraw: number;
|
|
26
|
+
};
|
|
27
|
+
inspectRandomNode: {
|
|
28
|
+
intervalInMs: number;
|
|
29
|
+
};
|
|
30
|
+
closeExpiredFlags: {
|
|
31
|
+
intervalInMs: number;
|
|
32
|
+
maxAgeInMs: number;
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
export interface OperatorServiceConfig {
|
|
36
|
+
signer: SignerWithProvider;
|
|
37
|
+
operatorContractAddress: EthereumAddress;
|
|
38
|
+
theGraphUrl: string;
|
|
39
|
+
getEthersOverrides: () => Overrides;
|
|
40
|
+
}
|
|
41
|
+
export declare class OperatorPlugin extends Plugin<OperatorPluginConfig> {
|
|
42
|
+
private readonly abortController;
|
|
43
|
+
start(streamrClient: StreamrClient): Promise<void>;
|
|
44
|
+
stop(): Promise<void>;
|
|
45
|
+
getConfigSchema(): Schema;
|
|
46
|
+
getClientConfig(): {
|
|
47
|
+
path: string;
|
|
48
|
+
value: any;
|
|
49
|
+
}[];
|
|
50
|
+
}
|
|
@@ -0,0 +1,159 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.OperatorPlugin = void 0;
|
|
7
|
+
const utils_1 = require("@streamr/utils");
|
|
8
|
+
const Plugin_1 = require("../../Plugin");
|
|
9
|
+
const maintainOperatorValue_1 = require("./maintainOperatorValue");
|
|
10
|
+
const MaintainTopologyService_1 = require("./MaintainTopologyService");
|
|
11
|
+
const OperatorFleetState_1 = require("./OperatorFleetState");
|
|
12
|
+
const config_schema_json_1 = __importDefault(require("./config.schema.json"));
|
|
13
|
+
const createIsLeaderFn_1 = require("./createIsLeaderFn");
|
|
14
|
+
const announceNodeToContract_1 = require("./announceNodeToContract");
|
|
15
|
+
const announceNodeToStream_1 = require("./announceNodeToStream");
|
|
16
|
+
const checkOperatorValueBreach_1 = require("./checkOperatorValueBreach");
|
|
17
|
+
const fetchRedundancyFactor_1 = require("./fetchRedundancyFactor");
|
|
18
|
+
const formCoordinationStreamId_1 = require("./formCoordinationStreamId");
|
|
19
|
+
const StreamPartAssignments_1 = require("./StreamPartAssignments");
|
|
20
|
+
const MaintainTopologyHelper_1 = require("./MaintainTopologyHelper");
|
|
21
|
+
const inspectRandomNode_1 = require("./inspectRandomNode");
|
|
22
|
+
const ContractFacade_1 = require("./ContractFacade");
|
|
23
|
+
const reviewSuspectNode_1 = require("./reviewSuspectNode");
|
|
24
|
+
const closeExpiredFlags_1 = require("./closeExpiredFlags");
|
|
25
|
+
const logger = new utils_1.Logger(module);
|
|
26
|
+
class OperatorPlugin extends Plugin_1.Plugin {
|
|
27
|
+
abortController = new AbortController();
|
|
28
|
+
async start(streamrClient) {
|
|
29
|
+
const signer = await streamrClient.getSigner();
|
|
30
|
+
const nodeId = await streamrClient.getNodeId();
|
|
31
|
+
const operatorContractAddress = (0, utils_1.toEthereumAddress)(this.pluginConfig.operatorContractAddress);
|
|
32
|
+
const serviceConfig = {
|
|
33
|
+
signer,
|
|
34
|
+
operatorContractAddress,
|
|
35
|
+
theGraphUrl: streamrClient.getConfig().contracts.theGraphUrl,
|
|
36
|
+
getEthersOverrides: () => streamrClient.getEthersOverrides()
|
|
37
|
+
};
|
|
38
|
+
const redundancyFactor = await (0, fetchRedundancyFactor_1.fetchRedundancyFactor)(serviceConfig);
|
|
39
|
+
if (redundancyFactor === undefined) {
|
|
40
|
+
throw new Error('Failed to fetch my redundancy factor');
|
|
41
|
+
}
|
|
42
|
+
logger.info('Fetched my redundancy factor', { redundancyFactor });
|
|
43
|
+
const contractFacade = ContractFacade_1.ContractFacade.createInstance(serviceConfig);
|
|
44
|
+
const maintainTopologyHelper = new MaintainTopologyHelper_1.MaintainTopologyHelper(contractFacade);
|
|
45
|
+
const createOperatorFleetState = OperatorFleetState_1.OperatorFleetState.createOperatorFleetStateBuilder(streamrClient, this.pluginConfig.heartbeatUpdateIntervalInMs, this.pluginConfig.fleetState.pruneAgeInMs, this.pluginConfig.fleetState.pruneIntervalInMs, this.pluginConfig.fleetState.latencyExtraInMs, this.pluginConfig.fleetState.warmupPeriodInMs);
|
|
46
|
+
const fleetState = createOperatorFleetState((0, formCoordinationStreamId_1.formCoordinationStreamId)(operatorContractAddress));
|
|
47
|
+
const streamPartAssignments = new StreamPartAssignments_1.StreamPartAssignments(nodeId, redundancyFactor, async (streamId) => {
|
|
48
|
+
const stream = await streamrClient.getStream(streamId);
|
|
49
|
+
return stream.getStreamParts();
|
|
50
|
+
}, fleetState, maintainTopologyHelper);
|
|
51
|
+
// Important: must be created before maintainTopologyHelper#start is invoked!
|
|
52
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
53
|
+
const maintainTopologyService = new MaintainTopologyService_1.MaintainTopologyService(streamrClient, streamPartAssignments);
|
|
54
|
+
await fleetState.start();
|
|
55
|
+
await maintainTopologyHelper.start();
|
|
56
|
+
this.abortController.signal.addEventListener('abort', async () => {
|
|
57
|
+
await fleetState.destroy();
|
|
58
|
+
});
|
|
59
|
+
// start tasks in background so that operations which take significant amount of time (e.g. fleetState.waitUntilReady())
|
|
60
|
+
// don't block the startup of Broker
|
|
61
|
+
setImmediate(async () => {
|
|
62
|
+
(0, utils_1.setAbortableInterval)(() => {
|
|
63
|
+
(async () => {
|
|
64
|
+
await (0, announceNodeToStream_1.announceNodeToStream)((0, utils_1.toEthereumAddress)(this.pluginConfig.operatorContractAddress), streamrClient);
|
|
65
|
+
})();
|
|
66
|
+
}, this.pluginConfig.heartbeatUpdateIntervalInMs, this.abortController.signal);
|
|
67
|
+
await (0, utils_1.scheduleAtInterval)(async () => (0, checkOperatorValueBreach_1.checkOperatorValueBreach)(contractFacade, this.pluginConfig.maintainOperatorValue.minSponsorshipEarningsInWithdraw, this.pluginConfig.maintainOperatorValue.maxSponsorshipsInWithdraw).catch((err) => {
|
|
68
|
+
logger.warn('Encountered error', { err });
|
|
69
|
+
}), this.pluginConfig.checkOperatorValueBreachIntervalInMs, true, this.abortController.signal);
|
|
70
|
+
await fleetState.waitUntilReady();
|
|
71
|
+
const isLeader = await (0, createIsLeaderFn_1.createIsLeaderFn)(streamrClient, fleetState, logger);
|
|
72
|
+
try {
|
|
73
|
+
await (0, utils_1.scheduleAtInterval)(async () => {
|
|
74
|
+
if (isLeader()) {
|
|
75
|
+
await (0, announceNodeToContract_1.announceNodeToContract)(this.pluginConfig.announceNodeToContract.writeIntervalInMs, contractFacade, streamrClient);
|
|
76
|
+
}
|
|
77
|
+
}, this.pluginConfig.announceNodeToContract.pollIntervalInMs, true, this.abortController.signal);
|
|
78
|
+
}
|
|
79
|
+
catch (err) {
|
|
80
|
+
logger.fatal('Encountered fatal error in announceNodeToContract', { err });
|
|
81
|
+
process.exit(1);
|
|
82
|
+
}
|
|
83
|
+
await (0, utils_1.scheduleAtInterval)(async () => {
|
|
84
|
+
if (isLeader()) {
|
|
85
|
+
try {
|
|
86
|
+
await (0, maintainOperatorValue_1.maintainOperatorValue)(this.pluginConfig.maintainOperatorValue.withdrawLimitSafetyFraction, this.pluginConfig.maintainOperatorValue.minSponsorshipEarningsInWithdraw, this.pluginConfig.maintainOperatorValue.maxSponsorshipsInWithdraw, contractFacade);
|
|
87
|
+
}
|
|
88
|
+
catch (err) {
|
|
89
|
+
logger.error('Encountered error while checking earnings', { err });
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}, this.pluginConfig.maintainOperatorValue.intervalInMs, true, this.abortController.signal);
|
|
93
|
+
await (0, utils_1.scheduleAtInterval)(async () => {
|
|
94
|
+
try {
|
|
95
|
+
await (0, inspectRandomNode_1.inspectRandomNode)(operatorContractAddress, contractFacade, streamPartAssignments, streamrClient, this.pluginConfig.heartbeatTimeoutInMs, (operatorContractAddress) => (0, fetchRedundancyFactor_1.fetchRedundancyFactor)({
|
|
96
|
+
operatorContractAddress,
|
|
97
|
+
signer
|
|
98
|
+
}), createOperatorFleetState, this.abortController.signal);
|
|
99
|
+
}
|
|
100
|
+
catch (err) {
|
|
101
|
+
logger.error('Encountered error while inspecting random node', { err });
|
|
102
|
+
}
|
|
103
|
+
}, this.pluginConfig.inspectRandomNode.intervalInMs, false, this.abortController.signal);
|
|
104
|
+
await (0, utils_1.scheduleAtInterval)(async () => {
|
|
105
|
+
try {
|
|
106
|
+
await (0, closeExpiredFlags_1.closeExpiredFlags)(this.pluginConfig.closeExpiredFlags.maxAgeInMs, serviceConfig.operatorContractAddress, contractFacade);
|
|
107
|
+
}
|
|
108
|
+
catch (err) {
|
|
109
|
+
logger.error('Encountered error while closing expired flags', { err });
|
|
110
|
+
}
|
|
111
|
+
}, this.pluginConfig.closeExpiredFlags.intervalInMs, false, this.abortController.signal);
|
|
112
|
+
contractFacade.addReviewRequestListener(async (sponsorshipAddress, targetOperator, partition, votingPeriodStartTimestamp, votingPeriodEndTimestamp) => {
|
|
113
|
+
try {
|
|
114
|
+
if (isLeader()) {
|
|
115
|
+
await (0, reviewSuspectNode_1.reviewSuspectNode)({
|
|
116
|
+
sponsorshipAddress,
|
|
117
|
+
targetOperator,
|
|
118
|
+
partition,
|
|
119
|
+
contractFacade,
|
|
120
|
+
streamrClient,
|
|
121
|
+
createOperatorFleetState,
|
|
122
|
+
getRedundancyFactor: (operatorContractAddress) => (0, fetchRedundancyFactor_1.fetchRedundancyFactor)({
|
|
123
|
+
operatorContractAddress,
|
|
124
|
+
signer
|
|
125
|
+
}),
|
|
126
|
+
maxSleepTime: 5 * 60 * 1000,
|
|
127
|
+
heartbeatTimeoutInMs: this.pluginConfig.heartbeatTimeoutInMs,
|
|
128
|
+
votingPeriod: {
|
|
129
|
+
startTime: votingPeriodStartTimestamp,
|
|
130
|
+
endTime: votingPeriodEndTimestamp
|
|
131
|
+
},
|
|
132
|
+
inspectionIntervalInMs: 8 * 60 * 1000,
|
|
133
|
+
maxInspections: 10,
|
|
134
|
+
abortSignal: this.abortController.signal
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
catch (err) {
|
|
139
|
+
logger.error('Encountered error while processing review request', { err });
|
|
140
|
+
}
|
|
141
|
+
}, this.abortController.signal);
|
|
142
|
+
});
|
|
143
|
+
}
|
|
144
|
+
async stop() {
|
|
145
|
+
this.abortController.abort();
|
|
146
|
+
}
|
|
147
|
+
// eslint-disable-next-line class-methods-use-this
|
|
148
|
+
getConfigSchema() {
|
|
149
|
+
return config_schema_json_1.default;
|
|
150
|
+
}
|
|
151
|
+
// eslint-disable-next-line class-methods-use-this
|
|
152
|
+
getClientConfig() {
|
|
153
|
+
return [{
|
|
154
|
+
path: 'network.node.acceptProxyConnections', value: true
|
|
155
|
+
}];
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
exports.OperatorPlugin = OperatorPlugin;
|
|
159
|
+
//# sourceMappingURL=OperatorPlugin.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"OperatorPlugin.js","sourceRoot":"","sources":["../../../../src/plugins/operator/OperatorPlugin.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { DhtAddress } from '@streamr/dht';
|
|
2
|
+
import { StreamID, StreamPartID } from '@streamr/protocol';
|
|
3
|
+
import EventEmitter3 from 'eventemitter3';
|
|
4
|
+
import { MaintainTopologyHelperEvents } from './MaintainTopologyHelper';
|
|
5
|
+
import { OperatorFleetStateEvents } from './OperatorFleetState';
|
|
6
|
+
export interface StreamPartAssignmentEvents {
|
|
7
|
+
assigned(streamPartId: StreamPartID): void;
|
|
8
|
+
unassigned(streamPartId: StreamPartID): void;
|
|
9
|
+
}
|
|
10
|
+
export declare class StreamPartAssignments extends EventEmitter3<StreamPartAssignmentEvents> {
|
|
11
|
+
private readonly allStreamParts;
|
|
12
|
+
private readonly myStreamParts;
|
|
13
|
+
private readonly concurrencyLimit;
|
|
14
|
+
private readonly consistentHashRing;
|
|
15
|
+
private readonly myNodeId;
|
|
16
|
+
private readonly getStreamParts;
|
|
17
|
+
private readonly operatorFleetState;
|
|
18
|
+
private readonly maintainTopologyHelper;
|
|
19
|
+
constructor(myNodeId: DhtAddress, redundancyFactor: number, getStreamParts: (streamId: StreamID) => Promise<StreamPartID[]>, operatorFleetState: EventEmitter3<OperatorFleetStateEvents>, maintainTopologyHelper: EventEmitter3<MaintainTopologyHelperEvents>);
|
|
20
|
+
getMyStreamParts(): StreamPartID[];
|
|
21
|
+
private nodeAdded;
|
|
22
|
+
private nodeRemoved;
|
|
23
|
+
private streamsStaked;
|
|
24
|
+
private streamUnstaked;
|
|
25
|
+
private recalculateAssignments;
|
|
26
|
+
private getStreamPartIds;
|
|
27
|
+
private concurrencyLimiter;
|
|
28
|
+
}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.StreamPartAssignments = void 0;
|
|
7
|
+
const utils_1 = require("@streamr/utils");
|
|
8
|
+
const eventemitter3_1 = __importDefault(require("eventemitter3"));
|
|
9
|
+
const p_limit_1 = __importDefault(require("p-limit"));
|
|
10
|
+
const ConsistentHashRing_1 = require("./ConsistentHashRing");
|
|
11
|
+
const logger = new utils_1.Logger(module);
|
|
12
|
+
class StreamPartAssignments extends eventemitter3_1.default {
|
|
13
|
+
allStreamParts = new Set();
|
|
14
|
+
myStreamParts = new Set();
|
|
15
|
+
concurrencyLimit = (0, p_limit_1.default)(1);
|
|
16
|
+
consistentHashRing;
|
|
17
|
+
myNodeId;
|
|
18
|
+
getStreamParts;
|
|
19
|
+
operatorFleetState;
|
|
20
|
+
maintainTopologyHelper;
|
|
21
|
+
constructor(myNodeId, redundancyFactor, getStreamParts, operatorFleetState, maintainTopologyHelper) {
|
|
22
|
+
super();
|
|
23
|
+
this.myNodeId = myNodeId;
|
|
24
|
+
this.getStreamParts = getStreamParts;
|
|
25
|
+
this.operatorFleetState = operatorFleetState;
|
|
26
|
+
this.maintainTopologyHelper = maintainTopologyHelper;
|
|
27
|
+
this.consistentHashRing = new ConsistentHashRing_1.ConsistentHashRing(redundancyFactor);
|
|
28
|
+
this.consistentHashRing.add(myNodeId);
|
|
29
|
+
this.operatorFleetState.on('added', this.nodeAdded);
|
|
30
|
+
this.operatorFleetState.on('removed', this.nodeRemoved);
|
|
31
|
+
this.maintainTopologyHelper.on('addStakedStreams', this.streamsStaked);
|
|
32
|
+
this.maintainTopologyHelper.on('removeStakedStream', this.streamUnstaked);
|
|
33
|
+
}
|
|
34
|
+
getMyStreamParts() {
|
|
35
|
+
return Array.from(this.myStreamParts);
|
|
36
|
+
}
|
|
37
|
+
nodeAdded = this.concurrencyLimiter(async (nodeId) => {
|
|
38
|
+
if (nodeId === this.myNodeId) {
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
this.consistentHashRing.add(nodeId);
|
|
42
|
+
this.recalculateAssignments(`nodeAdded:${nodeId}`);
|
|
43
|
+
});
|
|
44
|
+
nodeRemoved = this.concurrencyLimiter(async (nodeId) => {
|
|
45
|
+
if (nodeId === this.myNodeId) {
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
this.consistentHashRing.remove(nodeId);
|
|
49
|
+
this.recalculateAssignments(`nodeRemoved:${nodeId}`);
|
|
50
|
+
});
|
|
51
|
+
streamsStaked = this.concurrencyLimiter(async (streamIds) => {
|
|
52
|
+
const streamPartIds = (await Promise.all(streamIds.map(this.getStreamPartIds))).flat();
|
|
53
|
+
for (const streamPartId of streamPartIds) {
|
|
54
|
+
this.allStreamParts.add(streamPartId);
|
|
55
|
+
}
|
|
56
|
+
// TODO: optimize; calculate efficiently by only considering added stream parts
|
|
57
|
+
this.recalculateAssignments(`streamsStaked:${streamIds.join()}`);
|
|
58
|
+
});
|
|
59
|
+
streamUnstaked = this.concurrencyLimiter(async (streamId) => {
|
|
60
|
+
const streamPartIds = await this.getStreamPartIds(streamId);
|
|
61
|
+
for (const streamPartId of streamPartIds) {
|
|
62
|
+
this.allStreamParts.delete(streamPartId);
|
|
63
|
+
}
|
|
64
|
+
// TODO: optimize; calculate efficiently by only considering removed stream parts
|
|
65
|
+
this.recalculateAssignments(`streamUnstaked:${streamId}`);
|
|
66
|
+
});
|
|
67
|
+
recalculateAssignments(context) {
|
|
68
|
+
const assigned = [];
|
|
69
|
+
const unassigned = [];
|
|
70
|
+
for (const streamPartId of this.allStreamParts) {
|
|
71
|
+
if (this.consistentHashRing.get(streamPartId).includes(this.myNodeId) && !this.myStreamParts.has(streamPartId)) {
|
|
72
|
+
assigned.push(streamPartId);
|
|
73
|
+
this.myStreamParts.add(streamPartId);
|
|
74
|
+
this.emit('assigned', streamPartId);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
for (const streamPartId of this.myStreamParts) {
|
|
78
|
+
if (!this.allStreamParts.has(streamPartId) || !this.consistentHashRing.get(streamPartId).includes(this.myNodeId)) {
|
|
79
|
+
unassigned.push(streamPartId);
|
|
80
|
+
this.myStreamParts.delete(streamPartId);
|
|
81
|
+
this.emit('unassigned', streamPartId);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
logger.info('Recalculate assignments', { assigned, unassigned, context });
|
|
85
|
+
}
|
|
86
|
+
getStreamPartIds = async (streamId) => {
|
|
87
|
+
try {
|
|
88
|
+
return await this.getStreamParts(streamId);
|
|
89
|
+
}
|
|
90
|
+
catch (err) {
|
|
91
|
+
logger.warn('Ignore non-existing stream', { streamId, reason: err?.message });
|
|
92
|
+
return [];
|
|
93
|
+
}
|
|
94
|
+
};
|
|
95
|
+
concurrencyLimiter(fn) {
|
|
96
|
+
return (t) => {
|
|
97
|
+
this.concurrencyLimit(() => fn(t)).catch((err) => {
|
|
98
|
+
logger.warn('Encountered error while processing event', { err });
|
|
99
|
+
});
|
|
100
|
+
};
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
exports.StreamPartAssignments = StreamPartAssignments;
|
|
104
|
+
//# sourceMappingURL=StreamPartAssignments.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"StreamPartAssignments.js","sourceRoot":"","sources":["../../../../src/plugins/operator/StreamPartAssignments.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.announceNodeToContract = void 0;
|
|
4
|
+
const utils_1 = require("@streamr/utils");
|
|
5
|
+
const logger = new utils_1.Logger(module);
|
|
6
|
+
const announceNodeToContract = async (writeIntervalInMs, contractFacade, streamrClient) => {
|
|
7
|
+
if (await isHeartbeatStale(writeIntervalInMs, contractFacade)) {
|
|
8
|
+
await writeHeartbeat(contractFacade, streamrClient);
|
|
9
|
+
}
|
|
10
|
+
};
|
|
11
|
+
exports.announceNodeToContract = announceNodeToContract;
|
|
12
|
+
const isHeartbeatStale = async (writeIntervalInMs, contractFacade) => {
|
|
13
|
+
logger.debug('Polling last heartbeat timestamp', {
|
|
14
|
+
operatorContractAddress: contractFacade.getOperatorContractAddress()
|
|
15
|
+
});
|
|
16
|
+
let lastHeartbeatTs;
|
|
17
|
+
try {
|
|
18
|
+
lastHeartbeatTs = await contractFacade.getTimestampOfLastHeartbeat();
|
|
19
|
+
}
|
|
20
|
+
catch (err) {
|
|
21
|
+
logger.warn('Failed to poll last heartbeat timestamp', { reason: err?.message });
|
|
22
|
+
return false; // we don't know if heartbeat is stale, but we don't want execution to continue
|
|
23
|
+
}
|
|
24
|
+
const stale = lastHeartbeatTs !== undefined ? lastHeartbeatTs + writeIntervalInMs <= Date.now() : true;
|
|
25
|
+
logger.debug('Polled last heartbeat timestamp', { lastHeartbeatTs, stale });
|
|
26
|
+
return stale;
|
|
27
|
+
};
|
|
28
|
+
const writeHeartbeat = async (contractFacade, streamrClient) => {
|
|
29
|
+
logger.info('Write heartbeat');
|
|
30
|
+
try {
|
|
31
|
+
const nodeDescriptor = await streamrClient.getPeerDescriptor();
|
|
32
|
+
await contractFacade.writeHeartbeat(nodeDescriptor);
|
|
33
|
+
logger.debug('Wrote heartbeat', { nodeDescriptor });
|
|
34
|
+
}
|
|
35
|
+
catch (err) {
|
|
36
|
+
logger.warn('Failed to write heartbeat', { reason: err?.message });
|
|
37
|
+
}
|
|
38
|
+
};
|
|
39
|
+
//# sourceMappingURL=announceNodeToContract.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"announceNodeToContract.js","sourceRoot":"","sources":["../../../../src/plugins/operator/announceNodeToContract.ts"],"names":[],"mappings":";;;AAAA,0CAAuC;AAIvC,MAAM,MAAM,GAAG,IAAI,cAAM,CAAC,MAAM,CAAC,CAAA;AAE1B,MAAM,sBAAsB,GAAG,KAAK,EACvC,iBAAyB,EACzB,cAA8B,EAC9B,aAA4B,EACf,EAAE;IACf,IAAI,MAAM,gBAAgB,CAAC,iBAAiB,EAAE,cAAc,CAAC,EAAE,CAAC;QAC5D,MAAM,cAAc,CAAC,cAAc,EAAE,aAAa,CAAC,CAAA;IACvD,CAAC;AACL,CAAC,CAAA;AARY,QAAA,sBAAsB,0BAQlC;AAED,MAAM,gBAAgB,GAAG,KAAK,EAC1B,iBAAyB,EACzB,cAA8B,EACd,EAAE;IAClB,MAAM,CAAC,KAAK,CAAC,kCAAkC,EAAE;QAC7C,uBAAuB,EAAE,cAAc,CAAC,0BAA0B,EAAE;KACvE,CAAC,CAAA;IACF,IAAI,eAAe,CAAA;IACnB,IAAI,CAAC;QACD,eAAe,GAAG,MAAM,cAAc,CAAC,2BAA2B,EAAE,CAAA;IACxE,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACX,MAAM,CAAC,IAAI,CAAC,yCAAyC,EAAE,EAAE,MAAM,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAA;QAChF,OAAO,KAAK,CAAA,CAAC,+EAA+E;IAChG,CAAC;IACD,MAAM,KAAK,GAAG,eAAe,KAAK,SAAS,CAAC,CAAC,CAAC,eAAe,GAAG,iBAAiB,IAAI,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC,IAAI,CAAA;IACtG,MAAM,CAAC,KAAK,CAAC,iCAAiC,EAAE,EAAE,eAAe,EAAE,KAAK,EAAE,CAAC,CAAA;IAC3E,OAAO,KAAK,CAAA;AAChB,CAAC,CAAA;AAED,MAAM,cAAc,GAAG,KAAK,EACxB,cAA8B,EAC9B,aAA4B,EACf,EAAE;IACf,MAAM,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAA;IAC9B,IAAI,CAAC;QACD,MAAM,cAAc,GAAG,MAAM,aAAa,CAAC,iBAAiB,EAAE,CAAA;QAC9D,MAAM,cAAc,CAAC,cAAc,CAAC,cAAc,CAAC,CAAA;QACnD,MAAM,CAAC,KAAK,CAAC,iBAAiB,EAAE,EAAE,cAAc,EAAE,CAAC,CAAA;IACvD,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACX,MAAM,CAAC,IAAI,CAAC,2BAA2B,EAAE,EAAE,MAAM,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAA;IACtE,CAAC;AACL,CAAC,CAAA"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.announceNodeToStream = void 0;
|
|
4
|
+
const utils_1 = require("@streamr/utils");
|
|
5
|
+
const heartbeatUtils_1 = require("./heartbeatUtils");
|
|
6
|
+
const formCoordinationStreamId_1 = require("./formCoordinationStreamId");
|
|
7
|
+
const logger = new utils_1.Logger(module);
|
|
8
|
+
const announceNodeToStream = async (operatorContractAddress, streamrClient) => {
|
|
9
|
+
const coordinationStream = (0, formCoordinationStreamId_1.formCoordinationStreamId)(operatorContractAddress);
|
|
10
|
+
try {
|
|
11
|
+
const peerDescriptor = await streamrClient.getPeerDescriptor();
|
|
12
|
+
await streamrClient.publish(coordinationStream, (0, heartbeatUtils_1.createHeartbeatMessage)(peerDescriptor));
|
|
13
|
+
logger.debug('Published heartbeat to coordination stream', {
|
|
14
|
+
streamId: coordinationStream
|
|
15
|
+
});
|
|
16
|
+
}
|
|
17
|
+
catch (err) {
|
|
18
|
+
logger.warn('Unable to publish to coordination stream', {
|
|
19
|
+
streamId: coordinationStream,
|
|
20
|
+
reason: err?.message
|
|
21
|
+
});
|
|
22
|
+
}
|
|
23
|
+
};
|
|
24
|
+
exports.announceNodeToStream = announceNodeToStream;
|
|
25
|
+
//# sourceMappingURL=announceNodeToStream.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"announceNodeToStream.js","sourceRoot":"","sources":["../../../../src/plugins/operator/announceNodeToStream.ts"],"names":[],"mappings":";;;AAAA,0CAAwD;AAExD,qDAAyD;AACzD,yEAAqE;AAErE,MAAM,MAAM,GAAG,IAAI,cAAM,CAAC,MAAM,CAAC,CAAA;AAE1B,MAAM,oBAAoB,GAAG,KAAK,EACrC,uBAAwC,EACxC,aAA4B,EACf,EAAE;IACf,MAAM,kBAAkB,GAAG,IAAA,mDAAwB,EAAC,uBAAuB,CAAC,CAAA;IAC5E,IAAI,CAAC;QACD,MAAM,cAAc,GAAG,MAAM,aAAa,CAAC,iBAAiB,EAAE,CAAA;QAC9D,MAAM,aAAa,CAAC,OAAO,CAAC,kBAAkB,EAAE,IAAA,uCAAsB,EAAC,cAAc,CAAC,CAAC,CAAA;QACvF,MAAM,CAAC,KAAK,CAAC,4CAA4C,EAAE;YACvD,QAAQ,EAAE,kBAAkB;SAC/B,CAAC,CAAA;IACN,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACX,MAAM,CAAC,IAAI,CAAC,0CAA0C,EAAE;YACpD,QAAQ,EAAE,kBAAkB;YAC5B,MAAM,EAAE,GAAG,EAAE,OAAO;SACvB,CAAC,CAAA;IACN,CAAC;AACL,CAAC,CAAA;AAjBY,QAAA,oBAAoB,wBAiBhC"}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.checkOperatorValueBreach = void 0;
|
|
4
|
+
const utils_1 = require("@streamr/utils");
|
|
5
|
+
const logger = new utils_1.Logger(module);
|
|
6
|
+
const checkOperatorValueBreach = async (contractFacade, minSponsorshipEarningsInWithdraw, maxSponsorshipsInWithdraw) => {
|
|
7
|
+
const targetOperatorAddress = await contractFacade.getRandomOperator();
|
|
8
|
+
if (targetOperatorAddress === undefined) {
|
|
9
|
+
logger.info('No operators found');
|
|
10
|
+
return;
|
|
11
|
+
}
|
|
12
|
+
logger.info('Check other operator\'s earnings for breach', { targetOperatorAddress });
|
|
13
|
+
const { sumDataWei, maxAllowedEarningsDataWei, sponsorshipAddresses } = await contractFacade.getEarningsOf(targetOperatorAddress, minSponsorshipEarningsInWithdraw, maxSponsorshipsInWithdraw);
|
|
14
|
+
logger.trace(` -> is ${sumDataWei} > ${maxAllowedEarningsDataWei}?`);
|
|
15
|
+
if (sumDataWei > maxAllowedEarningsDataWei) {
|
|
16
|
+
logger.info('Withdraw earnings from sponsorships (target operator value in breach)', { targetOperatorAddress, sponsorshipAddresses, sumDataWei, maxAllowedEarningsDataWei });
|
|
17
|
+
await contractFacade.triggerWithdraw(targetOperatorAddress, sponsorshipAddresses);
|
|
18
|
+
}
|
|
19
|
+
};
|
|
20
|
+
exports.checkOperatorValueBreach = checkOperatorValueBreach;
|
|
21
|
+
//# sourceMappingURL=checkOperatorValueBreach.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"checkOperatorValueBreach.js","sourceRoot":"","sources":["../../../../src/plugins/operator/checkOperatorValueBreach.ts"],"names":[],"mappings":";;;AAAA,0CAAuC;AAGvC,MAAM,MAAM,GAAG,IAAI,cAAM,CAAC,MAAM,CAAC,CAAA;AAE1B,MAAM,wBAAwB,GAAG,KAAK,EACzC,cAA8B,EAC9B,gCAAwC,EACxC,yBAAiC,EACpB,EAAE;IACf,MAAM,qBAAqB,GAAG,MAAM,cAAc,CAAC,iBAAiB,EAAE,CAAA;IACtE,IAAI,qBAAqB,KAAK,SAAS,EAAE,CAAC;QACtC,MAAM,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAA;QACjC,OAAM;IACV,CAAC;IACD,MAAM,CAAC,IAAI,CAAC,6CAA6C,EAAE,EAAE,qBAAqB,EAAE,CAAC,CAAA;IACrF,MAAM,EAAE,UAAU,EAAE,yBAAyB,EAAE,oBAAoB,EAAE,GAAG,MAAM,cAAc,CAAC,aAAa,CACtG,qBAAqB,EACrB,gCAAgC,EAChC,yBAAyB,CAC5B,CAAA;IACD,MAAM,CAAC,KAAK,CAAC,UAAU,UAAU,MAAM,yBAAyB,GAAG,CAAC,CAAA;IACpE,IAAI,UAAU,GAAG,yBAAyB,EAAE,CAAC;QACzC,MAAM,CAAC,IAAI,CAAC,uEAAuE,EAC/E,EAAE,qBAAqB,EAAE,oBAAoB,EAAE,UAAU,EAAE,yBAAyB,EAAE,CAAC,CAAA;QAC3F,MAAM,cAAc,CAAC,eAAe,CAAC,qBAAqB,EAAE,oBAAoB,CAAC,CAAA;IACrF,CAAC;AACL,CAAC,CAAA;AAtBY,QAAA,wBAAwB,4BAsBpC"}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.closeExpiredFlags = void 0;
|
|
4
|
+
const utils_1 = require("@streamr/utils");
|
|
5
|
+
const logger = new utils_1.Logger(module);
|
|
6
|
+
const closeExpiredFlags = async (maxAgeInMs, operatorContractAddress, contractFacade) => {
|
|
7
|
+
logger.info('Start');
|
|
8
|
+
const sponsorships = (await contractFacade.getSponsorshipsOfOperator(operatorContractAddress))
|
|
9
|
+
.map((sponsorship) => sponsorship.sponsorshipAddress);
|
|
10
|
+
logger.debug(`Found ${sponsorships.length} sponsorships`);
|
|
11
|
+
if (sponsorships.length === 0) {
|
|
12
|
+
return;
|
|
13
|
+
}
|
|
14
|
+
const flags = await contractFacade.getExpiredFlags(sponsorships, maxAgeInMs);
|
|
15
|
+
logger.debug(`Found ${flags.length} expired flags to close`);
|
|
16
|
+
for (const flag of flags) {
|
|
17
|
+
const operatorAddress = flag.target.id;
|
|
18
|
+
const sponsorship = flag.sponsorship.id;
|
|
19
|
+
logger.info('Close expired flag', { flag });
|
|
20
|
+
await contractFacade.closeFlag(sponsorship, operatorAddress);
|
|
21
|
+
}
|
|
22
|
+
};
|
|
23
|
+
exports.closeExpiredFlags = closeExpiredFlags;
|
|
24
|
+
//# sourceMappingURL=closeExpiredFlags.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"closeExpiredFlags.js","sourceRoot":"","sources":["../../../../src/plugins/operator/closeExpiredFlags.ts"],"names":[],"mappings":";;;AAAA,0CAAwD;AAGxD,MAAM,MAAM,GAAG,IAAI,cAAM,CAAC,MAAM,CAAC,CAAA;AAE1B,MAAM,iBAAiB,GAAG,KAAK,EAClC,UAAkB,EAClB,uBAAwC,EACxC,cAA8B,EACjB,EAAE;IACf,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;IAEpB,MAAM,YAAY,GAAG,CAAC,MAAM,cAAc,CAAC,yBAAyB,CAAC,uBAAuB,CAAC,CAAC;SACzF,GAAG,CAAC,CAAC,WAA8B,EAAE,EAAE,CAAC,WAAW,CAAC,kBAAkB,CAAC,CAAA;IAC5E,MAAM,CAAC,KAAK,CAAC,SAAS,YAAY,CAAC,MAAM,eAAe,CAAC,CAAA;IACzD,IAAI,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC5B,OAAM;IACV,CAAC;IACD,MAAM,KAAK,GAAG,MAAM,cAAc,CAAC,eAAe,CAAC,YAAY,EAAE,UAAU,CAAC,CAAA;IAC5E,MAAM,CAAC,KAAK,CAAC,SAAS,KAAK,CAAC,MAAM,yBAAyB,CAAC,CAAA;IAC5D,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACvB,MAAM,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,EAAE,CAAA;QACtC,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAA;QACvC,MAAM,CAAC,IAAI,CAAC,oBAAoB,EAAE,EAAE,IAAI,EAAE,CAAC,CAAA;QAC3C,MAAM,cAAc,CAAC,SAAS,CAAC,WAAW,EAAE,eAAe,CAAC,CAAA;IAChE,CAAC;AACL,CAAC,CAAA;AArBY,QAAA,iBAAiB,qBAqB7B"}
|