@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.
Files changed (238) hide show
  1. package/LICENSE +78 -0
  2. package/README.md +54 -0
  3. package/bin/broker.ts +36 -0
  4. package/bin/config-wizard.js +4 -0
  5. package/bin/config-wizard.ts +17 -0
  6. package/bin/delete-expired-data.ts +41 -0
  7. package/bin/entry-point.ts +27 -0
  8. package/configs/development-1.env.json +68 -0
  9. package/configs/development-2.env.json +53 -0
  10. package/configs/development-3.env.json +53 -0
  11. package/configs/development-prod-resend.env.json +21 -0
  12. package/configs/docker-1.env.json +72 -0
  13. package/configs/docker-2.env.json +58 -0
  14. package/configs/docker-3.env.json +58 -0
  15. package/configuration.md +38 -0
  16. package/dist/bin/broker.d.ts +2 -0
  17. package/dist/bin/broker.js +42 -0
  18. package/dist/bin/broker.js.map +1 -0
  19. package/dist/bin/config-wizard.d.ts +2 -0
  20. package/dist/bin/config-wizard.js +22 -0
  21. package/dist/bin/config-wizard.js.map +1 -0
  22. package/dist/bin/delete-expired-data.d.ts +2 -0
  23. package/dist/bin/delete-expired-data.js +44 -0
  24. package/dist/bin/delete-expired-data.js.map +1 -0
  25. package/dist/bin/entry-point.d.ts +2 -0
  26. package/dist/bin/entry-point.js +28 -0
  27. package/dist/bin/entry-point.js.map +1 -0
  28. package/dist/package.json +78 -0
  29. package/dist/src/Plugin.d.ts +33 -0
  30. package/dist/src/Plugin.js +43 -0
  31. package/dist/src/Plugin.js.map +1 -0
  32. package/dist/src/apiAuthentication.d.ts +4 -0
  33. package/dist/src/apiAuthentication.js +16 -0
  34. package/dist/src/apiAuthentication.js.map +1 -0
  35. package/dist/src/broker.d.ts +8 -0
  36. package/dist/src/broker.js +69 -0
  37. package/dist/src/broker.js.map +1 -0
  38. package/dist/src/config/ConfigWizard.d.ts +5 -0
  39. package/dist/src/config/ConfigWizard.js +466 -0
  40. package/dist/src/config/ConfigWizard.js.map +1 -0
  41. package/dist/src/config/config.d.ts +26 -0
  42. package/dist/src/config/config.js +92 -0
  43. package/dist/src/config/config.js.map +1 -0
  44. package/dist/src/config/config.schema.json +86 -0
  45. package/dist/src/config/definitions.schema.json +35 -0
  46. package/dist/src/config/migration.d.ts +6 -0
  47. package/dist/src/config/migration.js +210 -0
  48. package/dist/src/config/migration.js.map +1 -0
  49. package/dist/src/config/validateConfig.d.ts +4 -0
  50. package/dist/src/config/validateConfig.js +40 -0
  51. package/dist/src/config/validateConfig.js.map +1 -0
  52. package/dist/src/exports.d.ts +3 -0
  53. package/dist/src/exports.js +6 -0
  54. package/dist/src/exports.js.map +1 -0
  55. package/dist/src/helpers/PayloadFormat.d.ts +19 -0
  56. package/dist/src/helpers/PayloadFormat.js +85 -0
  57. package/dist/src/helpers/PayloadFormat.js.map +1 -0
  58. package/dist/src/helpers/applyPluginClientConfigs.d.ts +3 -0
  59. package/dist/src/helpers/applyPluginClientConfigs.js +29 -0
  60. package/dist/src/helpers/applyPluginClientConfigs.js.map +1 -0
  61. package/dist/src/helpers/fetchOrThrow.d.ts +2 -0
  62. package/dist/src/helpers/fetchOrThrow.js +20 -0
  63. package/dist/src/helpers/fetchOrThrow.js.map +1 -0
  64. package/dist/src/helpers/generateMnemonicFromAddress.d.ts +5 -0
  65. package/dist/src/helpers/generateMnemonicFromAddress.js +16 -0
  66. package/dist/src/helpers/generateMnemonicFromAddress.js.map +1 -0
  67. package/dist/src/helpers/multiply.d.ts +1 -0
  68. package/dist/src/helpers/multiply.js +10 -0
  69. package/dist/src/helpers/multiply.js.map +1 -0
  70. package/dist/src/helpers/parser.d.ts +9 -0
  71. package/dist/src/helpers/parser.js +62 -0
  72. package/dist/src/helpers/parser.js.map +1 -0
  73. package/dist/src/helpers/partitions.d.ts +8 -0
  74. package/dist/src/helpers/partitions.js +32 -0
  75. package/dist/src/helpers/partitions.js.map +1 -0
  76. package/dist/src/helpers/weightedSample.d.ts +16 -0
  77. package/dist/src/helpers/weightedSample.js +35 -0
  78. package/dist/src/helpers/weightedSample.js.map +1 -0
  79. package/dist/src/httpServer.d.ts +16 -0
  80. package/dist/src/httpServer.js +71 -0
  81. package/dist/src/httpServer.js.map +1 -0
  82. package/dist/src/pluginRegistry.d.ts +3 -0
  83. package/dist/src/pluginRegistry.js +35 -0
  84. package/dist/src/pluginRegistry.js.map +1 -0
  85. package/dist/src/plugins/consoleMetrics/ConsoleMetricsPlugin.d.ts +12 -0
  86. package/dist/src/plugins/consoleMetrics/ConsoleMetricsPlugin.js +36 -0
  87. package/dist/src/plugins/consoleMetrics/ConsoleMetricsPlugin.js.map +1 -0
  88. package/dist/src/plugins/consoleMetrics/config.schema.json +18 -0
  89. package/dist/src/plugins/http/HttpPlugin.d.ts +8 -0
  90. package/dist/src/plugins/http/HttpPlugin.js +23 -0
  91. package/dist/src/plugins/http/HttpPlugin.js.map +1 -0
  92. package/dist/src/plugins/http/config.schema.json +12 -0
  93. package/dist/src/plugins/http/publishEndpoint.d.ts +3 -0
  94. package/dist/src/plugins/http/publishEndpoint.js +72 -0
  95. package/dist/src/plugins/http/publishEndpoint.js.map +1 -0
  96. package/dist/src/plugins/info/InfoPlugin.d.ts +9 -0
  97. package/dist/src/plugins/info/InfoPlugin.js +31 -0
  98. package/dist/src/plugins/info/InfoPlugin.js.map +1 -0
  99. package/dist/src/plugins/info/config.schema.json +12 -0
  100. package/dist/src/plugins/mqtt/Bridge.d.ts +40 -0
  101. package/dist/src/plugins/mqtt/Bridge.js +136 -0
  102. package/dist/src/plugins/mqtt/Bridge.js.map +1 -0
  103. package/dist/src/plugins/mqtt/MqttPlugin.d.ts +14 -0
  104. package/dist/src/plugins/mqtt/MqttPlugin.js +30 -0
  105. package/dist/src/plugins/mqtt/MqttPlugin.js.map +1 -0
  106. package/dist/src/plugins/mqtt/MqttServer.d.ts +22 -0
  107. package/dist/src/plugins/mqtt/MqttServer.js +109 -0
  108. package/dist/src/plugins/mqtt/MqttServer.js.map +1 -0
  109. package/dist/src/plugins/mqtt/config.schema.json +26 -0
  110. package/dist/src/plugins/operator/ConsistentHashRing.d.ts +20 -0
  111. package/dist/src/plugins/operator/ConsistentHashRing.js +64 -0
  112. package/dist/src/plugins/operator/ConsistentHashRing.js.map +1 -0
  113. package/dist/src/plugins/operator/ContractFacade.d.ts +80 -0
  114. package/dist/src/plugins/operator/ContractFacade.js +364 -0
  115. package/dist/src/plugins/operator/ContractFacade.js.map +1 -0
  116. package/dist/src/plugins/operator/MaintainTopologyHelper.d.ts +23 -0
  117. package/dist/src/plugins/operator/MaintainTopologyHelper.js +75 -0
  118. package/dist/src/plugins/operator/MaintainTopologyHelper.js.map +1 -0
  119. package/dist/src/plugins/operator/MaintainTopologyService.d.ts +11 -0
  120. package/dist/src/plugins/operator/MaintainTopologyService.js +57 -0
  121. package/dist/src/plugins/operator/MaintainTopologyService.js.map +1 -0
  122. package/dist/src/plugins/operator/OperatorFleetState.d.ts +33 -0
  123. package/dist/src/plugins/operator/OperatorFleetState.js +112 -0
  124. package/dist/src/plugins/operator/OperatorFleetState.js.map +1 -0
  125. package/dist/src/plugins/operator/OperatorPlugin.d.ts +50 -0
  126. package/dist/src/plugins/operator/OperatorPlugin.js +159 -0
  127. package/dist/src/plugins/operator/OperatorPlugin.js.map +1 -0
  128. package/dist/src/plugins/operator/StreamPartAssignments.d.ts +28 -0
  129. package/dist/src/plugins/operator/StreamPartAssignments.js +104 -0
  130. package/dist/src/plugins/operator/StreamPartAssignments.js.map +1 -0
  131. package/dist/src/plugins/operator/announceNodeToContract.d.ts +3 -0
  132. package/dist/src/plugins/operator/announceNodeToContract.js +39 -0
  133. package/dist/src/plugins/operator/announceNodeToContract.js.map +1 -0
  134. package/dist/src/plugins/operator/announceNodeToStream.d.ts +3 -0
  135. package/dist/src/plugins/operator/announceNodeToStream.js +25 -0
  136. package/dist/src/plugins/operator/announceNodeToStream.js.map +1 -0
  137. package/dist/src/plugins/operator/checkOperatorValueBreach.d.ts +2 -0
  138. package/dist/src/plugins/operator/checkOperatorValueBreach.js +21 -0
  139. package/dist/src/plugins/operator/checkOperatorValueBreach.js.map +1 -0
  140. package/dist/src/plugins/operator/closeExpiredFlags.d.ts +3 -0
  141. package/dist/src/plugins/operator/closeExpiredFlags.js +24 -0
  142. package/dist/src/plugins/operator/closeExpiredFlags.js.map +1 -0
  143. package/dist/src/plugins/operator/config.schema.json +152 -0
  144. package/dist/src/plugins/operator/createIsLeaderFn.d.ts +4 -0
  145. package/dist/src/plugins/operator/createIsLeaderFn.js +14 -0
  146. package/dist/src/plugins/operator/createIsLeaderFn.js.map +1 -0
  147. package/dist/src/plugins/operator/fetchRedundancyFactor.d.ts +2 -0
  148. package/dist/src/plugins/operator/fetchRedundancyFactor.js +43 -0
  149. package/dist/src/plugins/operator/fetchRedundancyFactor.js.map +1 -0
  150. package/dist/src/plugins/operator/formCoordinationStreamId.d.ts +3 -0
  151. package/dist/src/plugins/operator/formCoordinationStreamId.js +9 -0
  152. package/dist/src/plugins/operator/formCoordinationStreamId.js.map +1 -0
  153. package/dist/src/plugins/operator/heartbeatUtils.d.ts +67 -0
  154. package/dist/src/plugins/operator/heartbeatUtils.js +26 -0
  155. package/dist/src/plugins/operator/heartbeatUtils.js.map +1 -0
  156. package/dist/src/plugins/operator/inspectOverTime.d.ts +22 -0
  157. package/dist/src/plugins/operator/inspectOverTime.js +146 -0
  158. package/dist/src/plugins/operator/inspectOverTime.js.map +1 -0
  159. package/dist/src/plugins/operator/inspectRandomNode.d.ts +8 -0
  160. package/dist/src/plugins/operator/inspectRandomNode.js +44 -0
  161. package/dist/src/plugins/operator/inspectRandomNode.js.map +1 -0
  162. package/dist/src/plugins/operator/inspectionUtils.d.ts +23 -0
  163. package/dist/src/plugins/operator/inspectionUtils.js +120 -0
  164. package/dist/src/plugins/operator/inspectionUtils.js.map +1 -0
  165. package/dist/src/plugins/operator/maintainOperatorValue.d.ts +2 -0
  166. package/dist/src/plugins/operator/maintainOperatorValue.js +21 -0
  167. package/dist/src/plugins/operator/maintainOperatorValue.js.map +1 -0
  168. package/dist/src/plugins/operator/reviewSuspectNode.d.ts +24 -0
  169. package/dist/src/plugins/operator/reviewSuspectNode.js +56 -0
  170. package/dist/src/plugins/operator/reviewSuspectNode.js.map +1 -0
  171. package/dist/src/plugins/storage/Batch.d.ts +51 -0
  172. package/dist/src/plugins/storage/Batch.js +121 -0
  173. package/dist/src/plugins/storage/Batch.js.map +1 -0
  174. package/dist/src/plugins/storage/BatchManager.d.ts +27 -0
  175. package/dist/src/plugins/storage/BatchManager.js +117 -0
  176. package/dist/src/plugins/storage/BatchManager.js.map +1 -0
  177. package/dist/src/plugins/storage/Bucket.d.ts +23 -0
  178. package/dist/src/plugins/storage/Bucket.js +90 -0
  179. package/dist/src/plugins/storage/Bucket.js.map +1 -0
  180. package/dist/src/plugins/storage/BucketManager.d.ts +56 -0
  181. package/dist/src/plugins/storage/BucketManager.js +306 -0
  182. package/dist/src/plugins/storage/BucketManager.js.map +1 -0
  183. package/dist/src/plugins/storage/DataQueryFormat.d.ts +10 -0
  184. package/dist/src/plugins/storage/DataQueryFormat.js +51 -0
  185. package/dist/src/plugins/storage/DataQueryFormat.js.map +1 -0
  186. package/dist/src/plugins/storage/DeleteExpiredCmd.d.ts +28 -0
  187. package/dist/src/plugins/storage/DeleteExpiredCmd.js +155 -0
  188. package/dist/src/plugins/storage/DeleteExpiredCmd.js.map +1 -0
  189. package/dist/src/plugins/storage/SetMembershipSynchronizer.d.ts +33 -0
  190. package/dist/src/plugins/storage/SetMembershipSynchronizer.js +102 -0
  191. package/dist/src/plugins/storage/SetMembershipSynchronizer.js.map +1 -0
  192. package/dist/src/plugins/storage/Storage.d.ts +47 -0
  193. package/dist/src/plugins/storage/Storage.js +459 -0
  194. package/dist/src/plugins/storage/Storage.js.map +1 -0
  195. package/dist/src/plugins/storage/StorageConfig.d.ts +43 -0
  196. package/dist/src/plugins/storage/StorageConfig.js +83 -0
  197. package/dist/src/plugins/storage/StorageConfig.js.map +1 -0
  198. package/dist/src/plugins/storage/StorageEventListener.d.ts +17 -0
  199. package/dist/src/plugins/storage/StorageEventListener.js +46 -0
  200. package/dist/src/plugins/storage/StorageEventListener.js.map +1 -0
  201. package/dist/src/plugins/storage/StoragePlugin.d.ts +32 -0
  202. package/dist/src/plugins/storage/StoragePlugin.js +103 -0
  203. package/dist/src/plugins/storage/StoragePlugin.js.map +1 -0
  204. package/dist/src/plugins/storage/StoragePoller.d.ts +15 -0
  205. package/dist/src/plugins/storage/StoragePoller.js +47 -0
  206. package/dist/src/plugins/storage/StoragePoller.js.map +1 -0
  207. package/dist/src/plugins/storage/config.schema.json +88 -0
  208. package/dist/src/plugins/storage/dataMetadataEndpoint.d.ts +3 -0
  209. package/dist/src/plugins/storage/dataMetadataEndpoint.js +35 -0
  210. package/dist/src/plugins/storage/dataMetadataEndpoint.js.map +1 -0
  211. package/dist/src/plugins/storage/dataQueryEndpoint.d.ts +6 -0
  212. package/dist/src/plugins/storage/dataQueryEndpoint.js +181 -0
  213. package/dist/src/plugins/storage/dataQueryEndpoint.js.map +1 -0
  214. package/dist/src/plugins/storage/storageConfigEndpoint.d.ts +3 -0
  215. package/dist/src/plugins/storage/storageConfigEndpoint.js +31 -0
  216. package/dist/src/plugins/storage/storageConfigEndpoint.js.map +1 -0
  217. package/dist/src/plugins/subscriber/SubscriberPlugin.d.ts +16 -0
  218. package/dist/src/plugins/subscriber/SubscriberPlugin.js +22 -0
  219. package/dist/src/plugins/subscriber/SubscriberPlugin.js.map +1 -0
  220. package/dist/src/plugins/subscriber/config.schema.json +31 -0
  221. package/dist/src/plugins/websocket/Connection.d.ts +9 -0
  222. package/dist/src/plugins/websocket/Connection.js +45 -0
  223. package/dist/src/plugins/websocket/Connection.js.map +1 -0
  224. package/dist/src/plugins/websocket/PublishConnection.d.ts +12 -0
  225. package/dist/src/plugins/websocket/PublishConnection.js +46 -0
  226. package/dist/src/plugins/websocket/PublishConnection.js.map +1 -0
  227. package/dist/src/plugins/websocket/SubscribeConnection.d.ts +13 -0
  228. package/dist/src/plugins/websocket/SubscribeConnection.js +50 -0
  229. package/dist/src/plugins/websocket/SubscribeConnection.js.map +1 -0
  230. package/dist/src/plugins/websocket/WebsocketPlugin.d.ts +19 -0
  231. package/dist/src/plugins/websocket/WebsocketPlugin.js +26 -0
  232. package/dist/src/plugins/websocket/WebsocketPlugin.js.map +1 -0
  233. package/dist/src/plugins/websocket/WebsocketServer.d.ts +16 -0
  234. package/dist/src/plugins/websocket/WebsocketServer.js +132 -0
  235. package/dist/src/plugins/websocket/WebsocketServer.js.map +1 -0
  236. package/dist/src/plugins/websocket/config.schema.json +51 -0
  237. package/package.json +78 -0
  238. 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,3 @@
1
+ import StreamrClient from '@streamr/sdk';
2
+ import { ContractFacade } from './ContractFacade';
3
+ export declare const announceNodeToContract: (writeIntervalInMs: number, contractFacade: ContractFacade, streamrClient: StreamrClient) => Promise<void>;
@@ -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,3 @@
1
+ import { EthereumAddress } from '@streamr/utils';
2
+ import { StreamrClient } from '@streamr/sdk';
3
+ export declare const announceNodeToStream: (operatorContractAddress: EthereumAddress, streamrClient: StreamrClient) => Promise<void>;
@@ -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,2 @@
1
+ import { ContractFacade } from './ContractFacade';
2
+ export declare const checkOperatorValueBreach: (contractFacade: ContractFacade, minSponsorshipEarningsInWithdraw: number, maxSponsorshipsInWithdraw: number) => Promise<void>;
@@ -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,3 @@
1
+ import { EthereumAddress } from '@streamr/utils';
2
+ import { ContractFacade } from './ContractFacade';
3
+ export declare const closeExpiredFlags: (maxAgeInMs: number, operatorContractAddress: EthereumAddress, contractFacade: ContractFacade) => Promise<void>;
@@ -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"}