@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,86 @@
1
+ {
2
+ "$id": "config.schema.json",
3
+ "$schema": "http://json-schema.org/draft-07/schema#",
4
+ "description": "Broker configuration format",
5
+ "type": "object",
6
+ "additionalProperties": false,
7
+ "properties": {
8
+ "$schema": {
9
+ "type": "string"
10
+ },
11
+ "client": {
12
+ "type": "object",
13
+ "description": "Client configuration",
14
+ "additionalProperties": true,
15
+ "properties": {
16
+ "auth": {
17
+ "anyOf": [
18
+ {
19
+ "type": "object",
20
+ "properties": {
21
+ "privateKey": {
22
+ "type": "string",
23
+ "pattern": "^(0x)?[a-fA-F0-9]{64}$"
24
+ }
25
+ },
26
+ "required": ["privateKey"]
27
+ },
28
+ {
29
+ "type": "object",
30
+ "properties": {
31
+ "ethereum": {
32
+ "type": "object"
33
+ }
34
+ },
35
+ "required": ["ethereum"]
36
+ }
37
+ ]
38
+ }
39
+ },
40
+ "default": {}
41
+ },
42
+ "plugins": {
43
+ "type": "object",
44
+ "description": "Plugin configurations",
45
+ "additionalProperties": true,
46
+ "default": {}
47
+ },
48
+ "httpServer": {
49
+ "type": [
50
+ "object"
51
+ ],
52
+ "description": "HTTP server configuration",
53
+ "default": {},
54
+ "additionalProperties": false,
55
+ "properties": {
56
+ "port": {
57
+ "$ref": "definitions.schema.json#/definitions/port",
58
+ "description": "Port to start HTTP server on",
59
+ "default": 7171
60
+ },
61
+ "sslCertificate": {
62
+ "description": "Files to use for SSL",
63
+ "type": "object",
64
+ "required": [
65
+ "certFileName",
66
+ "privateKeyFileName"
67
+ ],
68
+ "additionalProperties": false,
69
+ "properties": {
70
+ "certFileName": {
71
+ "type": "string",
72
+ "description": "Path of certificate file"
73
+ },
74
+ "privateKeyFileName": {
75
+ "type": "string",
76
+ "description": "Path of private key file"
77
+ }
78
+ }
79
+ }
80
+ }
81
+ },
82
+ "apiAuthentication": {
83
+ "$ref": "definitions.schema.json#/definitions/apiAuthentication"
84
+ }
85
+ }
86
+ }
@@ -0,0 +1,35 @@
1
+ {
2
+ "$id": "definitions.schema.json",
3
+ "$schema": "http://json-schema.org/draft-07/schema#",
4
+ "definitions": {
5
+ "port": {
6
+ "type": "number",
7
+ "minimum": 0,
8
+ "maximum": 65353
9
+ },
10
+ "apiAuthentication": {
11
+ "type": "object",
12
+ "description": "Plugins should restrict the API access: if an endpoint requires authentication, the user must provide one of the API keys e.g. in a request header",
13
+ "required": [
14
+ "keys"
15
+ ],
16
+ "additionalProperties": false,
17
+ "properties": {
18
+ "keys": {
19
+ "type": "array",
20
+ "items": {
21
+ "type": "string"
22
+ }
23
+ }
24
+ }
25
+ },
26
+ "apiAuthenticationOverride": {
27
+ "description": "Override global API keys",
28
+ "anyOf": [{
29
+ "$ref": "definitions.schema.json#/definitions/apiAuthentication"
30
+ }, {
31
+ "type": "null"
32
+ }]
33
+ }
34
+ }
35
+ }
@@ -0,0 +1,6 @@
1
+ import { ConfigFile } from './config';
2
+ export declare const CURRENT_CONFIGURATION_VERSION = 3;
3
+ export declare const formSchemaUrl: (version: number) => string;
4
+ export declare const needsMigration: (config: any) => boolean;
5
+ export declare const createMigratedConfig: (source: any) => ConfigFile | never;
6
+ export declare const readConfigAndMigrateIfNeeded: (fileName: string | undefined) => ConfigFile | never;
@@ -0,0 +1,210 @@
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.readConfigAndMigrateIfNeeded = exports.createMigratedConfig = exports.needsMigration = exports.formSchemaUrl = exports.CURRENT_CONFIGURATION_VERSION = void 0;
7
+ const fs_1 = __importDefault(require("fs"));
8
+ const path_1 = __importDefault(require("path"));
9
+ const cloneDeep_1 = __importDefault(require("lodash/cloneDeep"));
10
+ const config_1 = require("./config");
11
+ exports.CURRENT_CONFIGURATION_VERSION = 3;
12
+ const formSchemaUrl = (version) => {
13
+ return `https://schema.streamr.network/config-v${version}.schema.json`;
14
+ };
15
+ exports.formSchemaUrl = formSchemaUrl;
16
+ const getVersion = (config) => {
17
+ const PATTERN = /config-v([0-9]+).schema.json$/;
18
+ const schemaUrl = config.$schema;
19
+ if (schemaUrl !== undefined) {
20
+ const match = PATTERN.exec(schemaUrl);
21
+ if (match !== null) {
22
+ return Number(match[1]);
23
+ }
24
+ }
25
+ return undefined;
26
+ };
27
+ // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types
28
+ const needsMigration = (config) => {
29
+ return (getVersion(config) !== undefined) && (getVersion(config) !== exports.CURRENT_CONFIGURATION_VERSION);
30
+ };
31
+ exports.needsMigration = needsMigration;
32
+ const convertV1ToV2 = (source) => {
33
+ const TARGET_VERSION = 2;
34
+ const target = (0, cloneDeep_1.default)(source);
35
+ target.$schema = (0, exports.formSchemaUrl)(TARGET_VERSION);
36
+ const consoleAndPM2IntervalInSeconds = source.plugins.metrics?.consoleAndPM2IntervalInSeconds;
37
+ if ((consoleAndPM2IntervalInSeconds !== undefined) && (consoleAndPM2IntervalInSeconds !== 0)) {
38
+ target.plugins.consoleMetrics = {
39
+ interval: consoleAndPM2IntervalInSeconds
40
+ };
41
+ delete target.plugins.metrics.consoleAndPM2IntervalInSeconds;
42
+ }
43
+ const isMetricsPluginEnabled = (source.plugins.metrics !== undefined) && (source.plugins.metrics.nodeMetrics !== null);
44
+ if (isMetricsPluginEnabled) {
45
+ const streamIdPrefix = source.plugins.metrics.nodeMetrics?.streamIdPrefix;
46
+ if (streamIdPrefix !== undefined) {
47
+ target.client.metrics = {
48
+ periods: [
49
+ {
50
+ duration: 60000,
51
+ streamId: `${streamIdPrefix}/min`
52
+ },
53
+ {
54
+ duration: 3600000,
55
+ streamId: `${streamIdPrefix}/hour`
56
+ },
57
+ {
58
+ duration: 86400000,
59
+ streamId: `${streamIdPrefix}/day`
60
+ }
61
+ ]
62
+ };
63
+ }
64
+ }
65
+ else {
66
+ target.client.metrics = false;
67
+ }
68
+ delete target.plugins.metrics;
69
+ if (target.client?.network?.name !== undefined) {
70
+ delete target.client.network.name;
71
+ }
72
+ if (source.plugins.publishHttp !== undefined) {
73
+ target.plugins.http = source.plugins.publishHttp;
74
+ delete target.plugins.publishHttp;
75
+ }
76
+ const deleteNullProperties = (obj, excludeKeys = []) => {
77
+ const keys = Object.keys(obj);
78
+ for (const key of keys) {
79
+ if ((obj[key] === null) && (!excludeKeys.includes(key))) {
80
+ delete obj[key];
81
+ }
82
+ }
83
+ };
84
+ deleteNullProperties(target);
85
+ if (target.httpServer !== undefined) {
86
+ deleteNullProperties(target.httpServer);
87
+ }
88
+ if (target.plugins.brubeckMiner !== undefined) {
89
+ deleteNullProperties(target.plugins.brubeckMiner, ['stunServerHost']);
90
+ }
91
+ if (target.plugins.mqtt !== undefined) {
92
+ deleteNullProperties(target.plugins.mqtt);
93
+ }
94
+ if (target.plugins.storage !== undefined) {
95
+ deleteNullProperties(target.plugins.storage.cluster);
96
+ }
97
+ if (target.plugins.websocket !== undefined) {
98
+ deleteNullProperties(target.plugins.websocket);
99
+ }
100
+ if ((target.httpServer?.certFileName !== undefined) || (target.httpServer?.privateKeyFileName !== undefined)) {
101
+ target.httpServer.sslCertificate = {
102
+ certFileName: target.httpServer.certFileName,
103
+ privateKeyFileName: target.httpServer.privateKeyFileName
104
+ };
105
+ delete target.httpServer.certFileName;
106
+ delete target.httpServer.privateKeyFileName;
107
+ }
108
+ return target;
109
+ };
110
+ const convertV2ToV3 = (source) => {
111
+ const TARGET_VERSION = 3;
112
+ const target = (0, cloneDeep_1.default)(source);
113
+ target.$schema = (0, exports.formSchemaUrl)(TARGET_VERSION);
114
+ return target;
115
+ };
116
+ // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types
117
+ const createMigratedConfig = (source) => {
118
+ let config = source;
119
+ do {
120
+ const version = getVersion(config);
121
+ if (version === 1) {
122
+ config = convertV1ToV2(config);
123
+ }
124
+ else if (version === 2) {
125
+ config = convertV2ToV3(config);
126
+ }
127
+ else {
128
+ throw new Error(`Unable to migrate the config: version=${version}`);
129
+ }
130
+ } while ((0, exports.needsMigration)(config));
131
+ return config;
132
+ };
133
+ exports.createMigratedConfig = createMigratedConfig;
134
+ /*
135
+ * Creates a backup file name, derived from the given name. Ensures that no file exist with that name.
136
+ * - foobar.ext.backup
137
+ * - or foobar.ext.backup-123
138
+ */
139
+ const formBackupFileName = (originalFileName) => {
140
+ const name = `${originalFileName}.backup`;
141
+ if (!fs_1.default.existsSync(name)) {
142
+ return name;
143
+ }
144
+ else {
145
+ let suffix = 1;
146
+ // eslint-disable-next-line no-constant-condition
147
+ while (true) {
148
+ const suffixedName = `${name}-${suffix}`;
149
+ if (!fs_1.default.existsSync(suffixedName)) {
150
+ return suffixedName;
151
+ }
152
+ suffix++;
153
+ }
154
+ }
155
+ };
156
+ const readConfigAndMigrateIfNeeded = (fileName) => {
157
+ let explicitTargetFile = undefined;
158
+ if (fileName === undefined) {
159
+ const defaultTargetFile = (0, config_1.getDefaultFile)();
160
+ const legacyTargetFile = (0, config_1.getLegacyDefaultFile)();
161
+ fileName = [defaultTargetFile, legacyTargetFile].find((file) => fs_1.default.existsSync(file));
162
+ if (fileName === undefined) {
163
+ /*
164
+ * No config file. Some config options are maybe set with enviroment variables
165
+ * (see overrideConfigToEnvVarsIfGiven function), and others just
166
+ * use the default values (see `default` definitions in config.schema.json)
167
+ */
168
+ return {};
169
+ }
170
+ if (fileName === legacyTargetFile) {
171
+ /*
172
+ * User has not specified the config file location in the command line and we found
173
+ * the file from the legacy default location. We'll write the migrated file to current
174
+ * default location instead of the legacy default location. There is no need to backup
175
+ * the file as we won't overwrite anything.
176
+ */
177
+ explicitTargetFile = defaultTargetFile;
178
+ }
179
+ }
180
+ let content = JSON.parse(fs_1.default.readFileSync(fileName, 'utf8'));
181
+ if ((0, exports.needsMigration)(content)) {
182
+ if (explicitTargetFile === undefined) {
183
+ const backupFile = formBackupFileName(fileName);
184
+ // eslint-disable-next-line no-console
185
+ console.log(`Migrating config ${fileName}, saving backup to ${backupFile}`);
186
+ fs_1.default.copyFileSync(fileName, backupFile);
187
+ content = (0, exports.createMigratedConfig)(content);
188
+ fs_1.default.writeFileSync(fileName, JSON.stringify(content, undefined, 4));
189
+ }
190
+ else {
191
+ const backupFile = formBackupFileName(fileName);
192
+ // eslint-disable-next-line no-console
193
+ console.log(`Migrating config ${fileName} to ${explicitTargetFile} (archiving the original file to ${backupFile})`);
194
+ content = (0, exports.createMigratedConfig)(content);
195
+ const directory = path_1.default.dirname(explicitTargetFile);
196
+ if (!fs_1.default.existsSync(directory)) {
197
+ fs_1.default.mkdirSync(directory, {
198
+ recursive: true
199
+ });
200
+ }
201
+ fs_1.default.writeFileSync(explicitTargetFile, JSON.stringify(content, undefined, 4));
202
+ // read permissions from the source file and set the same permissions to the target file
203
+ fs_1.default.chmodSync(explicitTargetFile, (fs_1.default.statSync(fileName).mode & parseInt('777', 8)).toString(8));
204
+ fs_1.default.renameSync(fileName, backupFile);
205
+ }
206
+ }
207
+ return content;
208
+ };
209
+ exports.readConfigAndMigrateIfNeeded = readConfigAndMigrateIfNeeded;
210
+ //# sourceMappingURL=migration.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"migration.js","sourceRoot":"","sources":["../../../src/config/migration.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,4 @@
1
+ import { Schema } from 'ajv';
2
+ import { StrictConfig } from './config';
3
+ export declare const validateConfig: (data: unknown, schema: Schema, contextName?: string, useDefaults?: boolean) => StrictConfig;
4
+ export declare const isValidConfig: (data: unknown, schema: Schema) => boolean;
@@ -0,0 +1,40 @@
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.isValidConfig = exports.validateConfig = void 0;
7
+ const ajv_1 = __importDefault(require("ajv"));
8
+ const ajv_formats_1 = __importDefault(require("ajv-formats"));
9
+ const definitions_schema_json_1 = __importDefault(require("./definitions.schema.json"));
10
+ const validateConfig = (data, schema, contextName, useDefaults = true) => {
11
+ const ajv = new ajv_1.default({
12
+ useDefaults
13
+ });
14
+ (0, ajv_formats_1.default)(ajv);
15
+ ajv.addFormat('ethereum-address', /^0x[a-zA-Z0-9]{40}$/);
16
+ ajv.addSchema(definitions_schema_json_1.default);
17
+ if (!ajv.validate(schema, data)) {
18
+ const prefix = (contextName !== undefined) ? (contextName + ': ') : '';
19
+ throw new Error(prefix + ajv.errors.map((e) => {
20
+ let text = ajv.errorsText([e], { dataVar: '' }).trim();
21
+ if (e.params.additionalProperty) {
22
+ text += ` (${e.params.additionalProperty})`;
23
+ }
24
+ return text;
25
+ }).join('\n'));
26
+ }
27
+ return data;
28
+ };
29
+ exports.validateConfig = validateConfig;
30
+ const isValidConfig = (data, schema) => {
31
+ try {
32
+ (0, exports.validateConfig)(data, schema, undefined, false);
33
+ return true;
34
+ }
35
+ catch (_e) {
36
+ return false;
37
+ }
38
+ };
39
+ exports.isValidConfig = isValidConfig;
40
+ //# sourceMappingURL=validateConfig.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"validateConfig.js","sourceRoot":"","sources":["../../../src/config/validateConfig.ts"],"names":[],"mappings":";;;;;;AAAA,8CAA8C;AAC9C,8DAAoC;AAEpC,wFAA0D;AAEnD,MAAM,cAAc,GAAG,CAAC,IAAa,EAAE,MAAc,EAAE,WAAoB,EAAE,WAAW,GAAG,IAAI,EAAgB,EAAE;IACpH,MAAM,GAAG,GAAG,IAAI,aAAG,CAAC;QAChB,WAAW;KACd,CAAC,CAAA;IACF,IAAA,qBAAU,EAAC,GAAG,CAAC,CAAA;IACf,GAAG,CAAC,SAAS,CAAC,kBAAkB,EAAE,qBAAqB,CAAC,CAAA;IACxD,GAAG,CAAC,SAAS,CAAC,iCAAkB,CAAC,CAAA;IACjC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,MAAM,EAAE,IAAI,CAAC,EAAE,CAAC;QAC9B,MAAM,MAAM,GAAG,CAAC,WAAW,KAAK,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC,WAAW,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAA;QACtE,MAAM,IAAI,KAAK,CAAC,MAAM,GAAG,GAAG,CAAC,MAAO,CAAC,GAAG,CAAC,CAAC,CAAc,EAAE,EAAE;YACxD,IAAI,IAAI,GAAG,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,OAAO,EAAE,EAAE,EAAE,CAAE,CAAC,IAAI,EAAE,CAAA;YACvD,IAAI,CAAC,CAAC,MAAM,CAAC,kBAAkB,EAAE,CAAC;gBAC9B,IAAI,IAAI,KAAK,CAAC,CAAC,MAAM,CAAC,kBAAkB,GAAG,CAAA;YAC/C,CAAC;YACD,OAAO,IAAI,CAAA;QACf,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAA;IAClB,CAAC;IACD,OAAO,IAAoB,CAAA;AAC/B,CAAC,CAAA;AAlBY,QAAA,cAAc,kBAkB1B;AAEM,MAAM,aAAa,GAAG,CAAC,IAAa,EAAE,MAAc,EAAW,EAAE;IACpE,IAAI,CAAC;QACD,IAAA,sBAAc,EAAC,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,KAAK,CAAC,CAAA;QAC9C,OAAO,IAAI,CAAA;IACf,CAAC;IAAC,OAAO,EAAE,EAAE,CAAC;QACV,OAAO,KAAK,CAAA;IAChB,CAAC;AACL,CAAC,CAAA;AAPY,QAAA,aAAa,iBAOzB"}
@@ -0,0 +1,3 @@
1
+ export { createBroker, Broker } from './broker';
2
+ export { Config } from './config/config';
3
+ export { StreamrClientConfig } from '@streamr/sdk';
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.createBroker = void 0;
4
+ var broker_1 = require("./broker");
5
+ Object.defineProperty(exports, "createBroker", { enumerable: true, get: function () { return broker_1.createBroker; } });
6
+ //# sourceMappingURL=exports.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"exports.js","sourceRoot":"","sources":["../../src/exports.ts"],"names":[],"mappings":";;;AAAA,mCAA+C;AAAtC,sGAAA,YAAY,OAAA"}
@@ -0,0 +1,19 @@
1
+ import { MessageMetadata } from '@streamr/sdk';
2
+ export interface PayloadFormat {
3
+ createMessage: (payload: string) => Message | never;
4
+ createPayload: (content: Record<string, unknown>, metadata?: Metadata) => string | never;
5
+ }
6
+ export interface Message {
7
+ content: Record<string, unknown>;
8
+ metadata: Metadata;
9
+ }
10
+ export type Metadata = Partial<Pick<MessageMetadata, 'timestamp' | 'sequenceNumber' | 'publisherId' | 'msgChainId'>>;
11
+ export declare class PlainPayloadFormat implements PayloadFormat {
12
+ createMessage(payload: string): Message | never;
13
+ createPayload(content: Record<string, unknown>): string | never;
14
+ }
15
+ export declare class MetadataPayloadFormat implements PayloadFormat {
16
+ createMessage(payload: string): Message | never;
17
+ createPayload(content: Record<string, unknown>, metadata?: Metadata): string | never;
18
+ }
19
+ export declare const getPayloadFormat: (metadata: boolean) => PayloadFormat;
@@ -0,0 +1,85 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getPayloadFormat = exports.MetadataPayloadFormat = exports.PlainPayloadFormat = void 0;
4
+ const METADATA_FIELDS = ['timestamp', 'sequenceNumber', 'publisherId', 'msgChainId'];
5
+ const pickProperties = (fields, from) => {
6
+ const result = {};
7
+ fields.forEach((field) => result[field] = from[field]);
8
+ return result;
9
+ };
10
+ const isJavascriptObject = (content) => {
11
+ return (content instanceof Object) && (!(content instanceof Array));
12
+ };
13
+ const assertContent = (content) => {
14
+ if ((content === undefined) || (content === null)) {
15
+ throw new Error('Content missing');
16
+ }
17
+ if (!isJavascriptObject(content)) {
18
+ throw new Error('Content is not an object');
19
+ }
20
+ };
21
+ const assertMetadata = (metadata) => {
22
+ if (!isJavascriptObject(metadata)) {
23
+ throw new Error('Metadata is not an object');
24
+ }
25
+ };
26
+ const parsePayloadJson = (payload) => {
27
+ if (payload.length === 0) {
28
+ throw new Error('Payload missing');
29
+ }
30
+ try {
31
+ return JSON.parse(payload);
32
+ }
33
+ catch (e) {
34
+ throw new Error(`Payload is not a JSON string: ${e.message}`);
35
+ }
36
+ };
37
+ /* eslint-disable class-methods-use-this */
38
+ class PlainPayloadFormat {
39
+ createMessage(payload) {
40
+ const content = parsePayloadJson(payload);
41
+ assertContent(content);
42
+ return {
43
+ content,
44
+ metadata: {}
45
+ };
46
+ }
47
+ createPayload(content) {
48
+ assertContent(content);
49
+ return JSON.stringify(content);
50
+ }
51
+ }
52
+ exports.PlainPayloadFormat = PlainPayloadFormat;
53
+ class MetadataPayloadFormat {
54
+ createMessage(payload) {
55
+ const json = parsePayloadJson(payload);
56
+ const content = json.content;
57
+ assertContent(content);
58
+ let metadata;
59
+ if (json.metadata !== undefined) {
60
+ assertMetadata(json.metadata);
61
+ metadata = pickProperties(METADATA_FIELDS, json.metadata);
62
+ }
63
+ else {
64
+ metadata = {};
65
+ }
66
+ return { content, metadata };
67
+ }
68
+ createPayload(content, metadata) {
69
+ assertContent(content);
70
+ const payload = {
71
+ content
72
+ };
73
+ if (metadata !== undefined) {
74
+ assertMetadata(metadata);
75
+ payload.metadata = pickProperties(METADATA_FIELDS, metadata);
76
+ }
77
+ return JSON.stringify(payload);
78
+ }
79
+ }
80
+ exports.MetadataPayloadFormat = MetadataPayloadFormat;
81
+ const getPayloadFormat = (metadata) => {
82
+ return metadata ? new MetadataPayloadFormat() : new PlainPayloadFormat();
83
+ };
84
+ exports.getPayloadFormat = getPayloadFormat;
85
+ //# sourceMappingURL=PayloadFormat.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"PayloadFormat.js","sourceRoot":"","sources":["../../../src/helpers/PayloadFormat.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Plugin } from '../Plugin';
2
+ import { StrictConfig } from '../config/config';
3
+ export declare const applyPluginClientConfigs: (plugins: Plugin<any>[], clientConfig: StrictConfig['client']) => void;
@@ -0,0 +1,29 @@
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.applyPluginClientConfigs = void 0;
7
+ const get_1 = __importDefault(require("lodash/get"));
8
+ const has_1 = __importDefault(require("lodash/has"));
9
+ const isEqual_1 = __importDefault(require("lodash/isEqual"));
10
+ const isString_1 = __importDefault(require("lodash/isString"));
11
+ const set_1 = __importDefault(require("lodash/set"));
12
+ const applyPluginClientConfigs = (plugins, clientConfig) => {
13
+ for (const plugin of plugins) {
14
+ for (const item of plugin.getClientConfig()) {
15
+ if (!(0, has_1.default)(clientConfig, item.path)) {
16
+ (0, set_1.default)(clientConfig, item.path, item.value);
17
+ }
18
+ else {
19
+ const existingValue = (0, get_1.default)(clientConfig, item.path);
20
+ if (!(0, isEqual_1.default)(item.value, existingValue)) {
21
+ const formattedValue = (0, isString_1.default)(existingValue) ? existingValue : `${JSON.stringify(existingValue)}`;
22
+ throw new Error(`Plugin ${plugin.name} doesn't support client config value "${formattedValue}" in ${item.path}`);
23
+ }
24
+ }
25
+ }
26
+ }
27
+ };
28
+ exports.applyPluginClientConfigs = applyPluginClientConfigs;
29
+ //# sourceMappingURL=applyPluginClientConfigs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"applyPluginClientConfigs.js","sourceRoot":"","sources":["../../../src/helpers/applyPluginClientConfigs.ts"],"names":[],"mappings":";;;;;;AAAA,qDAA4B;AAC5B,qDAA4B;AAC5B,6DAAoC;AACpC,+DAAsC;AACtC,qDAA4B;AAIrB,MAAM,wBAAwB,GAAG,CAAC,OAAsB,EAAE,YAAoC,EAAQ,EAAE;IAC3G,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;QAC3B,KAAK,MAAM,IAAI,IAAI,MAAM,CAAC,eAAe,EAAE,EAAE,CAAC;YAC1C,IAAI,CAAC,IAAA,aAAG,EAAC,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;gBAChC,IAAA,aAAG,EAAC,YAAY,EAAE,IAAI,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAA;YAC5C,CAAC;iBAAM,CAAC;gBACJ,MAAM,aAAa,GAAG,IAAA,aAAG,EAAC,YAAY,EAAE,IAAI,CAAC,IAAI,CAAC,CAAA;gBAClD,IAAI,CAAC,IAAA,iBAAO,EAAC,IAAI,CAAC,KAAK,EAAE,aAAa,CAAC,EAAE,CAAC;oBACtC,MAAM,cAAc,GAAG,IAAA,kBAAQ,EAAC,aAAa,CAAC,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,EAAE,CAAA;oBACnG,MAAM,IAAI,KAAK,CAAC,UAAU,MAAM,CAAC,IAAI,yCAAyC,cAAc,QAAQ,IAAI,CAAC,IAAI,EAAE,CAAC,CAAA;gBACpH,CAAC;YACL,CAAC;QACL,CAAC;IACL,CAAC;AACL,CAAC,CAAA;AAdY,QAAA,wBAAwB,4BAcpC"}
@@ -0,0 +1,2 @@
1
+ import { RequestInit, Response } from 'node-fetch';
2
+ export declare const fetchOrThrow: (url: string, init?: RequestInit) => Promise<Response>;
@@ -0,0 +1,20 @@
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.fetchOrThrow = void 0;
7
+ const node_fetch_1 = __importDefault(require("node-fetch"));
8
+ const utils_1 = require("@streamr/utils");
9
+ const DEFAULT_TIMEOUT = 30 * 1000;
10
+ const fetchOrThrow = async (url, init) => {
11
+ const res = await (0, node_fetch_1.default)(url, (0, utils_1.merge)({ timeout: DEFAULT_TIMEOUT }, init));
12
+ if (res.ok) {
13
+ return res;
14
+ }
15
+ else {
16
+ throw new Error(`request to ${url} failed, response: ${res.status} ${res.statusText}`);
17
+ }
18
+ };
19
+ exports.fetchOrThrow = fetchOrThrow;
20
+ //# sourceMappingURL=fetchOrThrow.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fetchOrThrow.js","sourceRoot":"","sources":["../../../src/helpers/fetchOrThrow.ts"],"names":[],"mappings":";;;;;;AAAA,4DAAyD;AACzD,0CAAsC;AAEtC,MAAM,eAAe,GAAG,EAAE,GAAG,IAAI,CAAA;AAE1B,MAAM,YAAY,GAAG,KAAK,EAAE,GAAW,EAAE,IAAkB,EAAqB,EAAE;IACrF,MAAM,GAAG,GAAG,MAAM,IAAA,oBAAK,EAAC,GAAG,EAAE,IAAA,aAAK,EAAC,EAAE,OAAO,EAAE,eAAe,EAAE,EAAE,IAAI,CAAC,CAAC,CAAA;IACvE,IAAI,GAAG,CAAC,EAAE,EAAE,CAAC;QACT,OAAO,GAAG,CAAA;IACd,CAAC;SAAM,CAAC;QACJ,MAAM,IAAI,KAAK,CAAC,cAAc,GAAG,sBAAsB,GAAG,CAAC,MAAM,IAAI,GAAG,CAAC,UAAU,EAAE,CAAC,CAAA;IAC1F,CAAC;AACL,CAAC,CAAA;AAPY,QAAA,YAAY,gBAOxB"}
@@ -0,0 +1,5 @@
1
+ import { EthereumAddress } from '@streamr/utils';
2
+ /**
3
+ * @param address - valid eth address with leading 0x
4
+ */
5
+ export declare const generateMnemonicFromAddress: (address: EthereumAddress) => string;
@@ -0,0 +1,16 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.generateMnemonicFromAddress = void 0;
4
+ const hdnode_1 = require("@ethersproject/hdnode");
5
+ /**
6
+ * @param address - valid eth address with leading 0x
7
+ */
8
+ const generateMnemonicFromAddress = (address) => {
9
+ return (0, hdnode_1.entropyToMnemonic)(address)
10
+ .split(' ')
11
+ .slice(0, 3)
12
+ .map((word) => word.charAt(0).toUpperCase() + word.slice(1))
13
+ .join(' ');
14
+ };
15
+ exports.generateMnemonicFromAddress = generateMnemonicFromAddress;
16
+ //# sourceMappingURL=generateMnemonicFromAddress.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"generateMnemonicFromAddress.js","sourceRoot":"","sources":["../../../src/helpers/generateMnemonicFromAddress.ts"],"names":[],"mappings":";;;AAAA,kDAAyD;AAGzD;;GAEG;AACI,MAAM,2BAA2B,GAAG,CAAC,OAAwB,EAAU,EAAE;IAC5E,OAAO,IAAA,0BAAiB,EAAC,OAAO,CAAC;SAC5B,KAAK,CAAC,GAAG,CAAC;SACV,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC;SACX,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;SAC3D,IAAI,CAAC,GAAG,CAAC,CAAA;AAClB,CAAC,CAAA;AANY,QAAA,2BAA2B,+BAMvC"}
@@ -0,0 +1 @@
1
+ export declare const multiply: (val1: bigint, val2: number) => bigint;
@@ -0,0 +1,10 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.multiply = void 0;
4
+ // this precision is big enough as we typically handle bigints which are in weis (the smallest denomination of Ethereum)
5
+ const PRECISION = 1e18;
6
+ const multiply = (val1, val2) => {
7
+ return val1 * BigInt(PRECISION * val2) / BigInt(PRECISION);
8
+ };
9
+ exports.multiply = multiply;
10
+ //# sourceMappingURL=multiply.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"multiply.js","sourceRoot":"","sources":["../../../src/helpers/multiply.ts"],"names":[],"mappings":";;;AAAA,wHAAwH;AACxH,MAAM,SAAS,GAAG,IAAI,CAAA;AAEf,MAAM,QAAQ,GAAG,CAAC,IAAY,EAAE,IAAY,EAAU,EAAE;IAC3D,OAAO,IAAI,GAAG,MAAM,CAAC,SAAS,GAAG,IAAI,CAAC,GAAG,MAAM,CAAC,SAAS,CAAC,CAAA;AAC9D,CAAC,CAAA;AAFY,QAAA,QAAQ,YAEpB"}