@enbox/dwn-server 0.0.1

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 (247) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +353 -0
  3. package/dist/cjs/index.js +6811 -0
  4. package/dist/cjs/package.json +1 -0
  5. package/dist/esm/src/config.d.ts +55 -0
  6. package/dist/esm/src/config.d.ts.map +1 -0
  7. package/dist/esm/src/config.js +60 -0
  8. package/dist/esm/src/config.js.map +1 -0
  9. package/dist/esm/src/connection/connection-manager.d.ts +25 -0
  10. package/dist/esm/src/connection/connection-manager.d.ts.map +1 -0
  11. package/dist/esm/src/connection/connection-manager.js +26 -0
  12. package/dist/esm/src/connection/connection-manager.js.map +1 -0
  13. package/dist/esm/src/connection/socket-connection.d.ts +65 -0
  14. package/dist/esm/src/connection/socket-connection.d.ts.map +1 -0
  15. package/dist/esm/src/connection/socket-connection.js +180 -0
  16. package/dist/esm/src/connection/socket-connection.js.map +1 -0
  17. package/dist/esm/src/dwn-error.d.ts +29 -0
  18. package/dist/esm/src/dwn-error.d.ts.map +1 -0
  19. package/dist/esm/src/dwn-error.js +36 -0
  20. package/dist/esm/src/dwn-error.js.map +1 -0
  21. package/dist/esm/src/dwn-server.d.ts +60 -0
  22. package/dist/esm/src/dwn-server.d.ts.map +1 -0
  23. package/dist/esm/src/dwn-server.js +130 -0
  24. package/dist/esm/src/dwn-server.js.map +1 -0
  25. package/dist/esm/src/http-api.d.ts +26 -0
  26. package/dist/esm/src/http-api.d.ts.map +1 -0
  27. package/dist/esm/src/http-api.js +474 -0
  28. package/dist/esm/src/http-api.js.map +1 -0
  29. package/dist/esm/src/index.d.ts +7 -0
  30. package/dist/esm/src/index.d.ts.map +1 -0
  31. package/dist/esm/src/index.js +6 -0
  32. package/dist/esm/src/index.js.map +1 -0
  33. package/dist/esm/src/json-rpc-api.d.ts +3 -0
  34. package/dist/esm/src/json-rpc-api.d.ts.map +1 -0
  35. package/dist/esm/src/json-rpc-api.js +8 -0
  36. package/dist/esm/src/json-rpc-api.js.map +1 -0
  37. package/dist/esm/src/json-rpc-handlers/dwn/index.d.ts +2 -0
  38. package/dist/esm/src/json-rpc-handlers/dwn/index.d.ts.map +1 -0
  39. package/dist/esm/src/json-rpc-handlers/dwn/index.js +2 -0
  40. package/dist/esm/src/json-rpc-handlers/dwn/index.js.map +1 -0
  41. package/dist/esm/src/json-rpc-handlers/dwn/process-message.d.ts +3 -0
  42. package/dist/esm/src/json-rpc-handlers/dwn/process-message.d.ts.map +1 -0
  43. package/dist/esm/src/json-rpc-handlers/dwn/process-message.js +73 -0
  44. package/dist/esm/src/json-rpc-handlers/dwn/process-message.js.map +1 -0
  45. package/dist/esm/src/json-rpc-handlers/subscription/close.d.ts +10 -0
  46. package/dist/esm/src/json-rpc-handlers/subscription/close.d.ts.map +1 -0
  47. package/dist/esm/src/json-rpc-handlers/subscription/close.js +39 -0
  48. package/dist/esm/src/json-rpc-handlers/subscription/close.js.map +1 -0
  49. package/dist/esm/src/json-rpc-handlers/subscription/index.d.ts +2 -0
  50. package/dist/esm/src/json-rpc-handlers/subscription/index.d.ts.map +1 -0
  51. package/dist/esm/src/json-rpc-handlers/subscription/index.js +2 -0
  52. package/dist/esm/src/json-rpc-handlers/subscription/index.js.map +1 -0
  53. package/dist/esm/src/json-rpc-socket.d.ts +39 -0
  54. package/dist/esm/src/json-rpc-socket.d.ts.map +1 -0
  55. package/dist/esm/src/json-rpc-socket.js +125 -0
  56. package/dist/esm/src/json-rpc-socket.js.map +1 -0
  57. package/dist/esm/src/lib/http-server-shutdown-handler.d.ts +10 -0
  58. package/dist/esm/src/lib/http-server-shutdown-handler.d.ts.map +1 -0
  59. package/dist/esm/src/lib/http-server-shutdown-handler.js +65 -0
  60. package/dist/esm/src/lib/http-server-shutdown-handler.js.map +1 -0
  61. package/dist/esm/src/lib/json-rpc-router.d.ts +30 -0
  62. package/dist/esm/src/lib/json-rpc-router.d.ts.map +1 -0
  63. package/dist/esm/src/lib/json-rpc-router.js +14 -0
  64. package/dist/esm/src/lib/json-rpc-router.js.map +1 -0
  65. package/dist/esm/src/lib/json-rpc.d.ts +54 -0
  66. package/dist/esm/src/lib/json-rpc.d.ts.map +1 -0
  67. package/dist/esm/src/lib/json-rpc.js +60 -0
  68. package/dist/esm/src/lib/json-rpc.js.map +1 -0
  69. package/dist/esm/src/main.d.ts +3 -0
  70. package/dist/esm/src/main.d.ts.map +1 -0
  71. package/dist/esm/src/main.js +11 -0
  72. package/dist/esm/src/main.js.map +1 -0
  73. package/dist/esm/src/metrics.d.ts +4 -0
  74. package/dist/esm/src/metrics.d.ts.map +1 -0
  75. package/dist/esm/src/metrics.js +13 -0
  76. package/dist/esm/src/metrics.js.map +1 -0
  77. package/dist/esm/src/plugin-loader.d.ts +10 -0
  78. package/dist/esm/src/plugin-loader.d.ts.map +1 -0
  79. package/dist/esm/src/plugin-loader.js +19 -0
  80. package/dist/esm/src/plugin-loader.js.map +1 -0
  81. package/dist/esm/src/process-handlers.d.ts +11 -0
  82. package/dist/esm/src/process-handlers.d.ts.map +1 -0
  83. package/dist/esm/src/process-handlers.js +40 -0
  84. package/dist/esm/src/process-handlers.js.map +1 -0
  85. package/dist/esm/src/registration/proof-of-work-manager.d.ts +93 -0
  86. package/dist/esm/src/registration/proof-of-work-manager.d.ts.map +1 -0
  87. package/dist/esm/src/registration/proof-of-work-manager.js +243 -0
  88. package/dist/esm/src/registration/proof-of-work-manager.js.map +1 -0
  89. package/dist/esm/src/registration/proof-of-work-types.d.ts +8 -0
  90. package/dist/esm/src/registration/proof-of-work-types.d.ts.map +1 -0
  91. package/dist/esm/src/registration/proof-of-work-types.js +2 -0
  92. package/dist/esm/src/registration/proof-of-work-types.js.map +1 -0
  93. package/dist/esm/src/registration/proof-of-work.d.ts +41 -0
  94. package/dist/esm/src/registration/proof-of-work.d.ts.map +1 -0
  95. package/dist/esm/src/registration/proof-of-work.js +69 -0
  96. package/dist/esm/src/registration/proof-of-work.js.map +1 -0
  97. package/dist/esm/src/registration/registration-manager.d.ts +55 -0
  98. package/dist/esm/src/registration/registration-manager.d.ts.map +1 -0
  99. package/dist/esm/src/registration/registration-manager.js +121 -0
  100. package/dist/esm/src/registration/registration-manager.js.map +1 -0
  101. package/dist/esm/src/registration/registration-store.d.ts +24 -0
  102. package/dist/esm/src/registration/registration-store.d.ts.map +1 -0
  103. package/dist/esm/src/registration/registration-store.js +56 -0
  104. package/dist/esm/src/registration/registration-store.js.map +1 -0
  105. package/dist/esm/src/registration/registration-types.d.ts +18 -0
  106. package/dist/esm/src/registration/registration-types.d.ts.map +1 -0
  107. package/dist/esm/src/registration/registration-types.js +2 -0
  108. package/dist/esm/src/registration/registration-types.js.map +1 -0
  109. package/dist/esm/src/storage.d.ts +24 -0
  110. package/dist/esm/src/storage.d.ts.map +1 -0
  111. package/dist/esm/src/storage.js +146 -0
  112. package/dist/esm/src/storage.js.map +1 -0
  113. package/dist/esm/src/web5-connect/sql-ttl-cache.d.ts +39 -0
  114. package/dist/esm/src/web5-connect/sql-ttl-cache.d.ts.map +1 -0
  115. package/dist/esm/src/web5-connect/sql-ttl-cache.js +106 -0
  116. package/dist/esm/src/web5-connect/sql-ttl-cache.js.map +1 -0
  117. package/dist/esm/src/web5-connect/web5-connect-server.d.ts +58 -0
  118. package/dist/esm/src/web5-connect/web5-connect-server.d.ts.map +1 -0
  119. package/dist/esm/src/web5-connect/web5-connect-server.js +77 -0
  120. package/dist/esm/src/web5-connect/web5-connect-server.js.map +1 -0
  121. package/dist/esm/src/ws-api.d.ts +13 -0
  122. package/dist/esm/src/ws-api.d.ts.map +1 -0
  123. package/dist/esm/src/ws-api.js +31 -0
  124. package/dist/esm/src/ws-api.js.map +1 -0
  125. package/dist/esm/tests/common-scenario-validator.d.ts +11 -0
  126. package/dist/esm/tests/common-scenario-validator.d.ts.map +1 -0
  127. package/dist/esm/tests/common-scenario-validator.js +114 -0
  128. package/dist/esm/tests/common-scenario-validator.js.map +1 -0
  129. package/dist/esm/tests/connection/connection-manager.spec.d.ts +2 -0
  130. package/dist/esm/tests/connection/connection-manager.spec.d.ts.map +1 -0
  131. package/dist/esm/tests/connection/connection-manager.spec.js +47 -0
  132. package/dist/esm/tests/connection/connection-manager.spec.js.map +1 -0
  133. package/dist/esm/tests/connection/socket-connection.spec.d.ts +2 -0
  134. package/dist/esm/tests/connection/socket-connection.spec.d.ts.map +1 -0
  135. package/dist/esm/tests/connection/socket-connection.spec.js +125 -0
  136. package/dist/esm/tests/connection/socket-connection.spec.js.map +1 -0
  137. package/dist/esm/tests/cors/http-api.browser.d.ts +2 -0
  138. package/dist/esm/tests/cors/http-api.browser.d.ts.map +1 -0
  139. package/dist/esm/tests/cors/http-api.browser.js +60 -0
  140. package/dist/esm/tests/cors/http-api.browser.js.map +1 -0
  141. package/dist/esm/tests/cors/ping.browser.d.ts +2 -0
  142. package/dist/esm/tests/cors/ping.browser.d.ts.map +1 -0
  143. package/dist/esm/tests/cors/ping.browser.js +7 -0
  144. package/dist/esm/tests/cors/ping.browser.js.map +1 -0
  145. package/dist/esm/tests/dwn-process-message.spec.d.ts +2 -0
  146. package/dist/esm/tests/dwn-process-message.spec.d.ts.map +1 -0
  147. package/dist/esm/tests/dwn-process-message.spec.js +172 -0
  148. package/dist/esm/tests/dwn-process-message.spec.js.map +1 -0
  149. package/dist/esm/tests/dwn-server.spec.d.ts +2 -0
  150. package/dist/esm/tests/dwn-server.spec.d.ts.map +1 -0
  151. package/dist/esm/tests/dwn-server.spec.js +49 -0
  152. package/dist/esm/tests/dwn-server.spec.js.map +1 -0
  153. package/dist/esm/tests/http-api.spec.d.ts +2 -0
  154. package/dist/esm/tests/http-api.spec.d.ts.map +1 -0
  155. package/dist/esm/tests/http-api.spec.js +775 -0
  156. package/dist/esm/tests/http-api.spec.js.map +1 -0
  157. package/dist/esm/tests/json-rpc-socket.spec.d.ts +2 -0
  158. package/dist/esm/tests/json-rpc-socket.spec.d.ts.map +1 -0
  159. package/dist/esm/tests/json-rpc-socket.spec.js +225 -0
  160. package/dist/esm/tests/json-rpc-socket.spec.js.map +1 -0
  161. package/dist/esm/tests/plugins/data-store-sqlite.d.ts +17 -0
  162. package/dist/esm/tests/plugins/data-store-sqlite.d.ts.map +1 -0
  163. package/dist/esm/tests/plugins/data-store-sqlite.js +23 -0
  164. package/dist/esm/tests/plugins/data-store-sqlite.js.map +1 -0
  165. package/dist/esm/tests/plugins/event-log-sqlite.d.ts +17 -0
  166. package/dist/esm/tests/plugins/event-log-sqlite.d.ts.map +1 -0
  167. package/dist/esm/tests/plugins/event-log-sqlite.js +23 -0
  168. package/dist/esm/tests/plugins/event-log-sqlite.js.map +1 -0
  169. package/dist/esm/tests/plugins/event-stream-in-memory.d.ts +17 -0
  170. package/dist/esm/tests/plugins/event-stream-in-memory.d.ts.map +1 -0
  171. package/dist/esm/tests/plugins/event-stream-in-memory.js +21 -0
  172. package/dist/esm/tests/plugins/event-stream-in-memory.js.map +1 -0
  173. package/dist/esm/tests/plugins/message-store-sqlite.d.ts +17 -0
  174. package/dist/esm/tests/plugins/message-store-sqlite.d.ts.map +1 -0
  175. package/dist/esm/tests/plugins/message-store-sqlite.js +23 -0
  176. package/dist/esm/tests/plugins/message-store-sqlite.js.map +1 -0
  177. package/dist/esm/tests/plugins/resumable-task-store-sqlite.d.ts +17 -0
  178. package/dist/esm/tests/plugins/resumable-task-store-sqlite.d.ts.map +1 -0
  179. package/dist/esm/tests/plugins/resumable-task-store-sqlite.js +23 -0
  180. package/dist/esm/tests/plugins/resumable-task-store-sqlite.js.map +1 -0
  181. package/dist/esm/tests/process-handler.spec.d.ts +2 -0
  182. package/dist/esm/tests/process-handler.spec.d.ts.map +1 -0
  183. package/dist/esm/tests/process-handler.spec.js +60 -0
  184. package/dist/esm/tests/process-handler.spec.js.map +1 -0
  185. package/dist/esm/tests/registration/proof-of-work-manager.spec.d.ts +2 -0
  186. package/dist/esm/tests/registration/proof-of-work-manager.spec.d.ts.map +1 -0
  187. package/dist/esm/tests/registration/proof-of-work-manager.spec.js +157 -0
  188. package/dist/esm/tests/registration/proof-of-work-manager.spec.js.map +1 -0
  189. package/dist/esm/tests/rpc-subscribe-close.spec.d.ts +2 -0
  190. package/dist/esm/tests/rpc-subscribe-close.spec.d.ts.map +1 -0
  191. package/dist/esm/tests/rpc-subscribe-close.spec.js +81 -0
  192. package/dist/esm/tests/rpc-subscribe-close.spec.js.map +1 -0
  193. package/dist/esm/tests/scenarios/dynamic-plugin-loading.spec.d.ts +2 -0
  194. package/dist/esm/tests/scenarios/dynamic-plugin-loading.spec.d.ts.map +1 -0
  195. package/dist/esm/tests/scenarios/dynamic-plugin-loading.spec.js +73 -0
  196. package/dist/esm/tests/scenarios/dynamic-plugin-loading.spec.js.map +1 -0
  197. package/dist/esm/tests/scenarios/registration.spec.d.ts +2 -0
  198. package/dist/esm/tests/scenarios/registration.spec.d.ts.map +1 -0
  199. package/dist/esm/tests/scenarios/registration.spec.js +507 -0
  200. package/dist/esm/tests/scenarios/registration.spec.js.map +1 -0
  201. package/dist/esm/tests/scenarios/web5-connect.spec.d.ts +2 -0
  202. package/dist/esm/tests/scenarios/web5-connect.spec.d.ts.map +1 -0
  203. package/dist/esm/tests/scenarios/web5-connect.spec.js +137 -0
  204. package/dist/esm/tests/scenarios/web5-connect.spec.js.map +1 -0
  205. package/dist/esm/tests/test-dwn.d.ts +7 -0
  206. package/dist/esm/tests/test-dwn.d.ts.map +1 -0
  207. package/dist/esm/tests/test-dwn.js +34 -0
  208. package/dist/esm/tests/test-dwn.js.map +1 -0
  209. package/dist/esm/tests/utils.d.ts +46 -0
  210. package/dist/esm/tests/utils.d.ts.map +1 -0
  211. package/dist/esm/tests/utils.js +116 -0
  212. package/dist/esm/tests/utils.js.map +1 -0
  213. package/dist/esm/tests/ws-api.spec.d.ts +2 -0
  214. package/dist/esm/tests/ws-api.spec.d.ts.map +1 -0
  215. package/dist/esm/tests/ws-api.spec.js +327 -0
  216. package/dist/esm/tests/ws-api.spec.js.map +1 -0
  217. package/package.json +119 -0
  218. package/src/config.ts +71 -0
  219. package/src/connection/connection-manager.ts +39 -0
  220. package/src/connection/socket-connection.ts +221 -0
  221. package/src/dwn-error.ts +38 -0
  222. package/src/dwn-server.ts +178 -0
  223. package/src/http-api.ts +541 -0
  224. package/src/index.ts +6 -0
  225. package/src/json-rpc-api.ts +11 -0
  226. package/src/json-rpc-handlers/dwn/index.ts +1 -0
  227. package/src/json-rpc-handlers/dwn/process-message.ts +123 -0
  228. package/src/json-rpc-handlers/subscription/close.ts +59 -0
  229. package/src/json-rpc-handlers/subscription/index.ts +1 -0
  230. package/src/json-rpc-socket.ts +155 -0
  231. package/src/lib/http-server-shutdown-handler.ts +79 -0
  232. package/src/lib/json-rpc-router.ts +52 -0
  233. package/src/lib/json-rpc.ts +126 -0
  234. package/src/main.ts +14 -0
  235. package/src/metrics.ts +14 -0
  236. package/src/plugin-loader.ts +17 -0
  237. package/src/process-handlers.ts +65 -0
  238. package/src/registration/proof-of-work-manager.ts +317 -0
  239. package/src/registration/proof-of-work-types.ts +7 -0
  240. package/src/registration/proof-of-work.ts +100 -0
  241. package/src/registration/registration-manager.ts +153 -0
  242. package/src/registration/registration-store.ts +79 -0
  243. package/src/registration/registration-types.ts +18 -0
  244. package/src/storage.ts +213 -0
  245. package/src/web5-connect/sql-ttl-cache.ts +137 -0
  246. package/src/web5-connect/web5-connect-server.ts +122 -0
  247. package/src/ws-api.ts +45 -0
@@ -0,0 +1,59 @@
1
+ import { v4 as uuidv4 } from 'uuid';
2
+
3
+ import { DwnServerErrorCode } from '../../dwn-error.js';
4
+ import type {
5
+ HandlerResponse,
6
+ JsonRpcHandler,
7
+ } from '../../lib/json-rpc-router.js';
8
+
9
+ import type { JsonRpcId, JsonRpcResponse } from '../../lib/json-rpc.js';
10
+ import {
11
+ createJsonRpcErrorResponse,
12
+ createJsonRpcSuccessResponse,
13
+ JsonRpcErrorCodes,
14
+ } from '../../lib/json-rpc.js';
15
+
16
+ /**
17
+ * Closes a subscription tied to a specific `SocketConnection`.
18
+ *
19
+ * @param jsonRpcRequest must include JsonRpcId of the subscription request within a `subscription object`.
20
+ * @param context must include the associated `SocketConnection`.
21
+ *
22
+ */
23
+ export const handleSubscriptionsClose: JsonRpcHandler = async (
24
+ jsonRpcRequest,
25
+ context,
26
+ ) => {
27
+ const requestId = jsonRpcRequest.id ?? uuidv4();
28
+ if (context.socketConnection === undefined) {
29
+ const jsonRpcResponse = createJsonRpcErrorResponse(requestId, JsonRpcErrorCodes.InvalidRequest, 'socket connection does not exist');
30
+ return { jsonRpcResponse };
31
+ }
32
+
33
+ if (jsonRpcRequest.subscription === undefined) {
34
+ const jsonRpcResponse = createJsonRpcErrorResponse(requestId, JsonRpcErrorCodes.InvalidRequest, 'subscribe options do not exist');
35
+ return { jsonRpcResponse };
36
+ }
37
+
38
+ const { socketConnection } = context;
39
+ const { id } = jsonRpcRequest.subscription as { id: JsonRpcId };
40
+
41
+ let jsonRpcResponse:JsonRpcResponse;
42
+ try {
43
+ // closing the subscription and cleaning up the reference within the given connection.
44
+ await socketConnection.closeSubscription(id);
45
+ jsonRpcResponse = createJsonRpcSuccessResponse(requestId, { reply: { status: 200, detail: 'Accepted' } });
46
+ } catch(error) {
47
+ if (error.code === DwnServerErrorCode.ConnectionSubscriptionJsonRpcIdNotFound) {
48
+ jsonRpcResponse = createJsonRpcErrorResponse(requestId, JsonRpcErrorCodes.InvalidParams, `subscription ${id} does not exist.`);
49
+ } else {
50
+ jsonRpcResponse = createJsonRpcErrorResponse(
51
+ requestId,
52
+ JsonRpcErrorCodes.InternalError,
53
+ `unknown subscription close error for ${id}: ${error.message}`
54
+ );
55
+ }
56
+ }
57
+
58
+ return { jsonRpcResponse } as HandlerResponse;
59
+ }
@@ -0,0 +1 @@
1
+ export * from './close.js';
@@ -0,0 +1,155 @@
1
+ import log from 'loglevel';
2
+ import { v4 as uuidv4 } from 'uuid';
3
+ import WebSocket from 'ws';
4
+
5
+ import type { JsonRpcId, JsonRpcRequest, JsonRpcResponse } from "./lib/json-rpc.js";
6
+ import { createJsonRpcSubscriptionRequest } from "./lib/json-rpc.js";
7
+
8
+ // These were arbitrarily chosen, but can be modified via connect options
9
+ const CONNECT_TIMEOUT = 3_000;
10
+ const RESPONSE_TIMEOUT = 30_000;
11
+
12
+ export interface JsonRpcSocketOptions {
13
+ /** socket connection timeout in milliseconds */
14
+ connectTimeout?: number;
15
+ /** response timeout for rpc requests in milliseconds */
16
+ responseTimeout?: number;
17
+ /** optional connection close handler */
18
+ onclose?: () => void;
19
+ /** optional socket error handler */
20
+ onerror?: (error?: any) => void;
21
+ }
22
+
23
+ /**
24
+ * JSON RPC Socket Client for WebSocket request/response and long-running subscriptions.
25
+ */
26
+ export class JsonRpcSocket {
27
+ private constructor(private socket: WebSocket, private responseTimeout: number) {}
28
+
29
+ static async connect(url: string, options: JsonRpcSocketOptions = {}): Promise<JsonRpcSocket> {
30
+ const { connectTimeout = CONNECT_TIMEOUT, responseTimeout = RESPONSE_TIMEOUT, onclose, onerror } = options;
31
+
32
+ const socket = new WebSocket(url);
33
+
34
+ if (!onclose) {
35
+ socket.onclose = ():void => {
36
+ log.info(`JSON RPC Socket close ${url}`);
37
+ };
38
+ } else {
39
+ socket.onclose = onclose;
40
+ }
41
+
42
+ if (!onerror) {
43
+ socket.onerror = (error?: any):void => {
44
+ log.error(`JSON RPC Socket error ${url}`, error);
45
+ };
46
+ } else {
47
+ socket.onerror = onerror;
48
+ }
49
+
50
+ return new Promise<JsonRpcSocket>((resolve, reject) => {
51
+ socket.addEventListener('open', () => {
52
+ resolve(new JsonRpcSocket(socket, responseTimeout));
53
+ });
54
+
55
+ socket.addEventListener('error', (error) => {
56
+ reject(error);
57
+ });
58
+
59
+ setTimeout(() => reject, connectTimeout);
60
+ });
61
+ }
62
+
63
+ close(): void {
64
+ this.socket.close();
65
+ }
66
+
67
+ /**
68
+ * Sends a JSON-RPC request through the socket and waits for a single response.
69
+ */
70
+ async request(request: JsonRpcRequest): Promise<JsonRpcResponse> {
71
+ return new Promise((resolve, reject) => {
72
+ request.id ??= uuidv4();
73
+
74
+ const handleResponse = (event: { data: any }):void => {
75
+ const jsonRpsResponse = JSON.parse(event.data) as JsonRpcResponse;
76
+ if (jsonRpsResponse.id === request.id) {
77
+ // if the incoming response id matches the request id, we will remove the listener and resolve the response
78
+ this.socket.removeEventListener('message', handleResponse);
79
+ return resolve(jsonRpsResponse);
80
+ }
81
+ };
82
+ // subscribe to the listener before sending the request
83
+ this.socket.addEventListener('message', handleResponse);
84
+ this.send(request);
85
+
86
+ // reject this promise if we don't receive any response back within the timeout period
87
+ setTimeout(() => {
88
+ this.socket.removeEventListener('message', handleResponse);
89
+ reject(new Error('request timed out'));
90
+ }, this.responseTimeout);
91
+ });
92
+ }
93
+
94
+ /**
95
+ * Sends a JSON-RPC request through the socket and keeps a listener open to read associated responses as they arrive.
96
+ * Returns a close method to clean up the listener.
97
+ */
98
+ async subscribe(request: JsonRpcRequest, listener: (response: JsonRpcResponse) => void): Promise<{
99
+ response: JsonRpcResponse;
100
+ close?: () => Promise<void>;
101
+ }> {
102
+
103
+ if (!request.method.startsWith('rpc.subscribe.')) {
104
+ throw new Error('subscribe rpc requests must include the `rpc.subscribe` prefix');
105
+ }
106
+
107
+ if (!request.subscription) {
108
+ throw new Error('subscribe rpc requests must include subscribe options');
109
+ }
110
+
111
+ const subscriptionId = request.subscription.id;
112
+ const socketEventListener = (event: { data: any }):void => {
113
+ const jsonRpcResponse = JSON.parse(event.data.toString()) as JsonRpcResponse;
114
+ if (jsonRpcResponse.id === subscriptionId) {
115
+ if (jsonRpcResponse.error !== undefined) {
116
+ // remove the event listener upon receipt of a JSON RPC Error.
117
+ this.socket.removeEventListener('message', socketEventListener);
118
+ this.closeSubscription(subscriptionId);
119
+ }
120
+ listener(jsonRpcResponse);
121
+ }
122
+ };
123
+ this.socket.addEventListener('message', socketEventListener);
124
+
125
+ const response = await this.request(request);
126
+ if (response.error) {
127
+ this.socket.removeEventListener('message', socketEventListener);
128
+ return { response };
129
+ }
130
+
131
+ // clean up listener and create a `rpc.subscribe.close` message to use when closing this JSON RPC subscription
132
+ const close = async (): Promise<void> => {
133
+ this.socket.removeEventListener('message', socketEventListener);
134
+ await this.closeSubscription(subscriptionId);
135
+ };
136
+
137
+ return {
138
+ response,
139
+ close
140
+ };
141
+ }
142
+
143
+ private closeSubscription(id: JsonRpcId): Promise<JsonRpcResponse> {
144
+ const requestId = uuidv4();
145
+ const request = createJsonRpcSubscriptionRequest(requestId, 'rpc.subscribe.close', {}, id);
146
+ return this.request(request);
147
+ }
148
+
149
+ /**
150
+ * Sends a JSON-RPC request through the socket. You must subscribe to a message listener separately to capture the response.
151
+ */
152
+ send(request: JsonRpcRequest):void {
153
+ this.socket.send(JSON.stringify(request));
154
+ }
155
+ }
@@ -0,0 +1,79 @@
1
+ import type { Server } from 'http';
2
+ import type { Socket } from 'net';
3
+
4
+ const SOCKET_IDLE_SYMBOL = Symbol('idle');
5
+
6
+ export class HttpServerShutdownHandler {
7
+ private tcpSockets: { [socketId: number]: Socket };
8
+ private tcpSocketId: number;
9
+ private server: Server;
10
+ private stopping: boolean;
11
+
12
+ constructor(server: Server) {
13
+ this.tcpSockets = {};
14
+ this.tcpSocketId = 1;
15
+ this.server = server;
16
+ this.stopping = false;
17
+
18
+ // This event is emitted when a new TCP stream is established
19
+ this.server.on('connection', (socket) => {
20
+ // set socket to idle. this same socket will be accessible within the `http.on('request', (req, res))` event listener
21
+ // as `request.connection`
22
+ socket[SOCKET_IDLE_SYMBOL] = true;
23
+ const tcpSocketId = this.tcpSocketId++;
24
+ this.tcpSockets[tcpSocketId] = socket;
25
+
26
+ // This event is emitted when a tcp stream is `destroy`ed
27
+ socket.on('close', () => {
28
+ delete this.tcpSockets[tcpSocketId];
29
+ });
30
+ });
31
+
32
+ // Emitted each time there is a request. There may be multiple requests
33
+ // per connection (in the case of HTTP Keep-Alive connections).
34
+ this.server.on('request', (request, response) => {
35
+ const { socket } = request;
36
+
37
+ // set __idle to false because this socket is being used for an incoming request
38
+ socket[SOCKET_IDLE_SYMBOL] = false;
39
+
40
+ // Emitted when the response has been sent. More specifically, this event is emitted
41
+ // when the last segment of the response headers and body have been handed off to the
42
+ // operating system for transmission over the network.
43
+ // It does not imply that the client has received anything yet.
44
+ response.on('finish', () => {
45
+ // set __idle back to true because the socket has finished facilitating a request. This socket may be used again without being
46
+ // destroyed if keep-alive is being leveraged
47
+ socket[SOCKET_IDLE_SYMBOL] = true;
48
+
49
+ if (this.stopping) {
50
+ socket.destroy();
51
+ }
52
+ });
53
+ });
54
+ }
55
+
56
+ stop(callback): void {
57
+ this.stopping = true;
58
+
59
+ // Stops the server from accepting new connections and keeps existing connections. This function is asynchronous,
60
+ // the server is finally closed when all connections are ended and the server emits a 'close' event.
61
+ // The optional callback will be called once the 'close' event occurs.
62
+ // The callback will be called with an Error as its only argument if the server was not open when close is called.
63
+ this.server.close(() => {
64
+ this.tcpSocketId = 0;
65
+ this.stopping = false;
66
+ callback();
67
+ });
68
+
69
+ // close all idle sockets. the remaining sockets facilitating active requests
70
+ // will be closed after they've served responses back.
71
+ for (const tcpSocketId in this.tcpSockets) {
72
+ const socket = this.tcpSockets[tcpSocketId];
73
+
74
+ if (socket[SOCKET_IDLE_SYMBOL]) {
75
+ socket.destroy();
76
+ }
77
+ }
78
+ }
79
+ }
@@ -0,0 +1,52 @@
1
+ import type { Dwn, MessageSubscriptionHandler } from '@enbox/dwn-sdk-js';
2
+
3
+ import type { Readable } from 'node:stream';
4
+
5
+ import type { JsonRpcId, JsonRpcRequest, JsonRpcResponse } from './json-rpc.js';
6
+ import type { SocketConnection } from '../connection/socket-connection.js';
7
+
8
+ export type RequestContext = {
9
+ transport: 'http' | 'ws';
10
+ dwn: Dwn;
11
+ /** the socket connection associated with this request if over sockets */
12
+ socketConnection?: SocketConnection;
13
+ subscriptionRequest?: {
14
+ /** The JsonRpcId of the subscription handler */
15
+ id: JsonRpcId;
16
+ /** The `MessageEvent` handler associated with a subscription request, only used in `ws` requests */
17
+ subscriptionHandler: MessageSubscriptionHandler;
18
+ }
19
+ /** The `Readable` stream associated with a `RecordsWrite` request only used in `http` requests */
20
+ dataStream?: Readable;
21
+ };
22
+
23
+ export type HandlerResponse = {
24
+ jsonRpcResponse: JsonRpcResponse;
25
+ dataStream?: Readable;
26
+ };
27
+
28
+ export type JsonRpcHandler = (
29
+ JsonRpcRequest: JsonRpcRequest,
30
+ context: RequestContext,
31
+ ) => Promise<HandlerResponse>;
32
+
33
+ export class JsonRpcRouter {
34
+ private methodHandlers: { [method: string]: JsonRpcHandler };
35
+
36
+ constructor() {
37
+ this.methodHandlers = {};
38
+ }
39
+
40
+ on(methodName: string, handler: JsonRpcHandler): void {
41
+ this.methodHandlers[methodName] = handler;
42
+ }
43
+
44
+ async handle(
45
+ rpcRequest: JsonRpcRequest,
46
+ context: RequestContext,
47
+ ): Promise<HandlerResponse> {
48
+ const handler = this.methodHandlers[rpcRequest.method];
49
+
50
+ return await handler(rpcRequest, context);
51
+ }
52
+ }
@@ -0,0 +1,126 @@
1
+ export type JsonRpcId = string | number | null;
2
+ export type JsonRpcVersion = '2.0';
3
+
4
+ export interface JsonRpcRequest {
5
+ jsonrpc: JsonRpcVersion;
6
+ id?: JsonRpcId;
7
+ method: string;
8
+ params?: any;
9
+ /** JSON RPC Subscription Extension Parameters */
10
+ subscription?: {
11
+ id: JsonRpcId
12
+ };
13
+ }
14
+
15
+ export interface JsonRpcError {
16
+ code: JsonRpcErrorCodes;
17
+ message: string;
18
+ data?: any;
19
+ }
20
+
21
+ export interface JsonRpcSubscription {
22
+ /** JSON RPC Id of the Subscription Request */
23
+ id: JsonRpcId;
24
+ close: () => Promise<void>;
25
+ }
26
+
27
+ export enum JsonRpcErrorCodes {
28
+ // JSON-RPC 2.0 pre-defined errors
29
+ InvalidRequest = -32600,
30
+ MethodNotFound = -32601,
31
+ InvalidParams = -32602,
32
+ InternalError = -32603,
33
+ ParseError = -32700,
34
+
35
+ /** App defined error equivalent to HTTP Status 400 */
36
+ BadRequest = -50400,
37
+ /** App defined error equivalent to HTTP Status 401 */
38
+ Unauthorized = -50401,
39
+ /** App defined error equivalent to HTTP Status 403 */
40
+ Forbidden = -50403,
41
+ }
42
+
43
+ export type JsonRpcResponse = JsonRpcSuccessResponse | JsonRpcErrorResponse;
44
+
45
+ export interface JsonRpcSuccessResponse {
46
+ jsonrpc: JsonRpcVersion;
47
+ id: JsonRpcId;
48
+ result: any;
49
+ error?: never;
50
+ }
51
+
52
+ export interface JsonRpcErrorResponse {
53
+ jsonrpc: JsonRpcVersion;
54
+ id: JsonRpcId;
55
+ result?: never;
56
+ error: JsonRpcError;
57
+ }
58
+
59
+ export const createJsonRpcErrorResponse = (
60
+ id: JsonRpcId,
61
+ code: JsonRpcErrorCodes,
62
+ message: string,
63
+ data?: any,
64
+ ): JsonRpcErrorResponse => {
65
+ const error: JsonRpcError = { code, message };
66
+ if (data != undefined) {
67
+ error.data = data;
68
+ }
69
+ return {
70
+ jsonrpc: '2.0',
71
+ id,
72
+ error,
73
+ };
74
+ };
75
+
76
+ export const createJsonRpcNotification = (
77
+ method: string,
78
+ params?: any,
79
+ ): JsonRpcRequest => {
80
+ return {
81
+ jsonrpc: '2.0',
82
+ method,
83
+ params,
84
+ };
85
+ };
86
+
87
+ export const createJsonRpcSubscriptionRequest = (
88
+ id: JsonRpcId,
89
+ method: string,
90
+ params?: any,
91
+ subscriptionId?: JsonRpcId
92
+ ): JsonRpcRequest => {
93
+ return {
94
+ jsonrpc: '2.0',
95
+ id,
96
+ method,
97
+ params,
98
+ subscription: {
99
+ id: subscriptionId,
100
+ }
101
+ }
102
+ }
103
+
104
+ export const createJsonRpcRequest = (
105
+ id: JsonRpcId,
106
+ method: string,
107
+ params?: any,
108
+ ): JsonRpcRequest => {
109
+ return {
110
+ jsonrpc: '2.0',
111
+ id,
112
+ method,
113
+ params,
114
+ };
115
+ };
116
+
117
+ export const createJsonRpcSuccessResponse = (
118
+ id: JsonRpcId,
119
+ result?: any,
120
+ ): JsonRpcSuccessResponse => {
121
+ return {
122
+ jsonrpc: '2.0',
123
+ id,
124
+ result: result ?? null,
125
+ };
126
+ };
package/src/main.ts ADDED
@@ -0,0 +1,14 @@
1
+ #!/usr/bin/env node
2
+ // node.js 18 and earlier, needs globalThis.crypto polyfill. needed for dwn-sdk-js
3
+ import { webcrypto } from 'node:crypto';
4
+
5
+ import { DwnServer } from './dwn-server.js';
6
+
7
+ if (!globalThis.crypto) {
8
+ // @ts-ignore
9
+ globalThis.crypto = webcrypto;
10
+ }
11
+
12
+ const dwnServer = new DwnServer();
13
+
14
+ await dwnServer.start();
package/src/metrics.ts ADDED
@@ -0,0 +1,14 @@
1
+ import { Counter, Histogram } from 'prom-client';
2
+
3
+ export const requestCounter = new Counter({
4
+ name: 'dwn_requests_total',
5
+ help: 'all dwn requests processed',
6
+ labelNames: ['method', 'status', 'error'],
7
+ });
8
+
9
+ export const responseHistogram = new Histogram({
10
+ name: 'http_response',
11
+ help: 'response histogram',
12
+ buckets: [0.005, 0.01, 0.025, 0.05, 0.1, 0.25, 0.5, 1, 2.5, 5, 10],
13
+ labelNames: ['route', 'code'],
14
+ });
@@ -0,0 +1,17 @@
1
+ /**
2
+ * A utility class for dynamically loading plugins from file paths.
3
+ */
4
+ export class PluginLoader {
5
+ /**
6
+ * Dynamically loads a plugin from a file path by invoking the argument-less constructor of the default exported class.
7
+ */
8
+ public static async loadPlugin<T>(filePath: string): Promise<T> {
9
+ try {
10
+ const module = await import(filePath);
11
+ const instance: T = new module.default() as T;
12
+ return instance;
13
+ } catch (error) {
14
+ throw new Error(`Failed to load component at ${filePath}: ${error.message}`);
15
+ }
16
+ }
17
+ }
@@ -0,0 +1,65 @@
1
+ import type { DwnServer } from './dwn-server.js';
2
+
3
+ export const gracefulShutdown = async (dwnServer: DwnServer): Promise<void> => {
4
+ await dwnServer.stop();
5
+ console.log('http server stopped.. exiting');
6
+ process.exit(0);
7
+ };
8
+
9
+ export type ProcessHandlers = {
10
+ unhandledRejectionHandler: (reason: any, promise: Promise<any>) => void,
11
+ uncaughtExceptionHandler: (err: Error) => void,
12
+ sigintHandler: () => Promise<void>,
13
+ sigtermHandler: () => Promise<void>
14
+ };
15
+
16
+ export const setProcessHandlers = (dwnServer: DwnServer): ProcessHandlers => {
17
+ const unhandledRejectionHandler = (reason: any, promise: Promise<any>): void => {
18
+ console.error(
19
+ `Unhandled promise rejection. Reason: ${reason}. Promise: ${JSON.stringify(
20
+ promise,
21
+ )}`,
22
+ );
23
+ };
24
+
25
+ const uncaughtExceptionHandler = (err: Error): void => {
26
+ console.error('Uncaught exception:', err.stack || err);
27
+ };
28
+
29
+ const sigintHandler = async (): Promise<void> => {
30
+ console.log('exit signal received [SIGINT]. starting graceful shutdown');
31
+ await gracefulShutdown(dwnServer);
32
+ };
33
+
34
+ const sigtermHandler = async (): Promise<void> => {
35
+ console.log('exit signal received [SIGTERM]. starting graceful shutdown');
36
+ await gracefulShutdown(dwnServer);
37
+ };
38
+
39
+ process.on('unhandledRejection', unhandledRejectionHandler);
40
+ process.on('uncaughtException', uncaughtExceptionHandler);
41
+ process.on('SIGINT', sigintHandler);
42
+ process.on('SIGTERM', sigtermHandler);
43
+
44
+ // Store handlers to be able to remove them later
45
+ return {
46
+ unhandledRejectionHandler,
47
+ uncaughtExceptionHandler,
48
+ sigintHandler,
49
+ sigtermHandler
50
+ };
51
+ };
52
+
53
+ export const removeProcessHandlers = (handlers: ProcessHandlers): void => {
54
+ const {
55
+ unhandledRejectionHandler,
56
+ uncaughtExceptionHandler,
57
+ sigintHandler,
58
+ sigtermHandler
59
+ } = handlers;
60
+
61
+ process.removeListener('unhandledRejection', unhandledRejectionHandler);
62
+ process.removeListener('uncaughtException', uncaughtExceptionHandler);
63
+ process.removeListener('SIGINT', sigintHandler);
64
+ process.removeListener('SIGTERM', sigtermHandler);
65
+ };