From 2bf926b0b39da35397c82cccf117859c7997b69d Mon Sep 17 00:00:00 2001 From: Andrew Ghostuhin Date: Wed, 30 Oct 2024 19:17:26 +0300 Subject: [PATCH 01/10] feat(nestjs-connectrpc): init --- .pnp.cjs | 292 ++++++++++++++++++ packages/nestjs-connectrpc/README.md | 1 + packages/nestjs-connectrpc/package.json | 53 ++++ packages/nestjs-connectrpc/src/async.utils.ts | 99 ++++++ .../src/connectrpc.constants.ts | 5 + .../src/connectrpc.decorators.ts | 88 ++++++ .../src/connectrpc.interfaces.ts | 74 +++++ .../src/connectrpc.server.ts | 128 ++++++++ .../src/connectrpc.strategy.ts | 72 +++++ .../src/custom-metadata.storage.ts | 25 ++ packages/nestjs-connectrpc/src/index.ts | 4 + .../nestjs-connectrpc/src/router.utils.ts | 117 +++++++ yarn.lock | 161 +++++++++- 13 files changed, 1118 insertions(+), 1 deletion(-) create mode 100644 packages/nestjs-connectrpc/README.md create mode 100644 packages/nestjs-connectrpc/package.json create mode 100644 packages/nestjs-connectrpc/src/async.utils.ts create mode 100644 packages/nestjs-connectrpc/src/connectrpc.constants.ts create mode 100644 packages/nestjs-connectrpc/src/connectrpc.decorators.ts create mode 100644 packages/nestjs-connectrpc/src/connectrpc.interfaces.ts create mode 100644 packages/nestjs-connectrpc/src/connectrpc.server.ts create mode 100644 packages/nestjs-connectrpc/src/connectrpc.strategy.ts create mode 100644 packages/nestjs-connectrpc/src/custom-metadata.storage.ts create mode 100644 packages/nestjs-connectrpc/src/index.ts create mode 100644 packages/nestjs-connectrpc/src/router.utils.ts diff --git a/.pnp.cjs b/.pnp.cjs index 57f6d343..cc4f5464 100755 --- a/.pnp.cjs +++ b/.pnp.cjs @@ -30,6 +30,10 @@ const RAW_RUNTIME_STATE = "name": "@atls/nestjs-batch-queue",\ "reference": "workspace:packages/nestjs-batch-queue"\ },\ + {\ + "name": "@atls/nestjs-connectrpc",\ + "reference": "workspace:packages/nestjs-connectrpc"\ + },\ {\ "name": "@atls/nestjs-cqrs",\ "reference": "workspace:packages/nestjs-cqrs"\ @@ -133,6 +137,7 @@ const RAW_RUNTIME_STATE = ["@atlantis-lab/nestjs-signed-url", ["workspace:packages/nestjs-signed-url"]],\ ["@atls/grpc-keto", ["workspace:packages/nestjs-grpc-keto"]],\ ["@atls/nestjs-batch-queue", ["workspace:packages/nestjs-batch-queue"]],\ + ["@atls/nestjs-connectrpc", ["workspace:packages/nestjs-connectrpc"]],\ ["@atls/nestjs-cqrs", ["workspace:packages/nestjs-cqrs"]],\ ["@atls/nestjs-dataloader", ["workspace:packages/nestjs-dataloader"]],\ ["@atls/nestjs-external-renderer", ["workspace:packages/nestjs-external-renderer"]],\ @@ -539,6 +544,24 @@ const RAW_RUNTIME_STATE = "linkType": "SOFT"\ }]\ ]],\ + ["@atls/nestjs-connectrpc", [\ + ["workspace:packages/nestjs-connectrpc", {\ + "packageLocation": "./packages/nestjs-connectrpc/",\ + "packageDependencies": [\ + ["@atls/nestjs-connectrpc", "workspace:packages/nestjs-connectrpc"],\ + ["@bufbuild/protobuf", "npm:1.10.0"],\ + ["@connectrpc/connect", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1"],\ + ["@connectrpc/connect-node", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1"],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["reflect-metadata", "npm:0.2.2"],\ + ["rxjs", "npm:7.8.1"]\ + ],\ + "linkType": "SOFT"\ + }]\ + ]],\ ["@atls/nestjs-cqrs", [\ ["workspace:packages/nestjs-cqrs", {\ "packageLocation": "./packages/nestjs-cqrs/",\ @@ -3270,6 +3293,64 @@ const RAW_RUNTIME_STATE = "linkType": "HARD"\ }]\ ]],\ + ["@bufbuild/protobuf", [\ + ["npm:1.10.0", {\ + "packageLocation": "../.yarn/berry/cache/@bufbuild-protobuf-npm-1.10.0-7f066cde74-10c0.zip/node_modules/@bufbuild/protobuf/",\ + "packageDependencies": [\ + ["@bufbuild/protobuf", "npm:1.10.0"]\ + ],\ + "linkType": "HARD"\ + }]\ + ]],\ + ["@connectrpc/connect", [\ + ["npm:1.6.1", {\ + "packageLocation": "../.yarn/berry/cache/@connectrpc-connect-npm-1.6.1-d31c4e5a29-10c0.zip/node_modules/@connectrpc/connect/",\ + "packageDependencies": [\ + ["@connectrpc/connect", "npm:1.6.1"]\ + ],\ + "linkType": "SOFT"\ + }],\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1", {\ + "packageLocation": "./.yarn/__virtual__/@connectrpc-connect-virtual-5e9157bdee/2/.yarn/berry/cache/@connectrpc-connect-npm-1.6.1-d31c4e5a29-10c0.zip/node_modules/@connectrpc/connect/",\ + "packageDependencies": [\ + ["@connectrpc/connect", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1"],\ + ["@bufbuild/protobuf", "npm:1.10.0"],\ + ["@types/bufbuild__protobuf", null]\ + ],\ + "packagePeers": [\ + "@bufbuild/protobuf",\ + "@types/bufbuild__protobuf"\ + ],\ + "linkType": "HARD"\ + }]\ + ]],\ + ["@connectrpc/connect-node", [\ + ["npm:1.6.1", {\ + "packageLocation": "../.yarn/berry/cache/@connectrpc-connect-node-npm-1.6.1-c3083f9671-10c0.zip/node_modules/@connectrpc/connect-node/",\ + "packageDependencies": [\ + ["@connectrpc/connect-node", "npm:1.6.1"]\ + ],\ + "linkType": "SOFT"\ + }],\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1", {\ + "packageLocation": "./.yarn/__virtual__/@connectrpc-connect-node-virtual-c17e9794fb/2/.yarn/berry/cache/@connectrpc-connect-node-npm-1.6.1-c3083f9671-10c0.zip/node_modules/@connectrpc/connect-node/",\ + "packageDependencies": [\ + ["@connectrpc/connect-node", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1"],\ + ["@bufbuild/protobuf", "npm:1.10.0"],\ + ["@connectrpc/connect", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1"],\ + ["@types/bufbuild__protobuf", null],\ + ["@types/connectrpc__connect", null],\ + ["undici", "npm:5.28.4"]\ + ],\ + "packagePeers": [\ + "@bufbuild/protobuf",\ + "@connectrpc/connect",\ + "@types/bufbuild__protobuf",\ + "@types/connectrpc__connect"\ + ],\ + "linkType": "HARD"\ + }]\ + ]],\ ["@emotion/css-prettifier", [\ ["npm:1.1.4", {\ "packageLocation": "../.yarn/berry/cache/@emotion-css-prettifier-npm-1.1.4-849a301a6c-10c0.zip/node_modules/@emotion/css-prettifier/",\ @@ -6542,6 +6623,34 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3", {\ + "packageLocation": "./.yarn/__virtual__/@nestjs-common-virtual-2765291d0b/2/.yarn/berry/cache/@nestjs-common-npm-10.4.3-c8baed1848-10c0.zip/node_modules/@nestjs/common/",\ + "packageDependencies": [\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@types/class-transformer", null],\ + ["@types/class-validator", null],\ + ["@types/reflect-metadata", null],\ + ["@types/rxjs", null],\ + ["class-transformer", null],\ + ["class-validator", null],\ + ["iterare", "npm:1.2.1"],\ + ["reflect-metadata", "npm:0.2.2"],\ + ["rxjs", "npm:7.8.1"],\ + ["tslib", "npm:2.7.0"],\ + ["uid", "npm:2.0.2"]\ + ],\ + "packagePeers": [\ + "@types/class-transformer",\ + "@types/class-validator",\ + "@types/reflect-metadata",\ + "@types/rxjs",\ + "class-transformer",\ + "class-validator",\ + "reflect-metadata",\ + "rxjs"\ + ],\ + "linkType": "HARD"\ + }],\ ["virtual:77887786a24289fa840c9acd370d634accbe79bcf317ecf5401844ffff73b8a593879dd9cce463873637e6414a631dfdb1a2473704bf332d823bcfffac8c2469#npm:10.4.1", {\ "packageLocation": "./.yarn/__virtual__/@nestjs-common-virtual-d13c8c526b/2/.yarn/berry/cache/@nestjs-common-npm-10.4.1-940734b1b1-10c0.zip/node_modules/@nestjs/common/",\ "packageDependencies": [\ @@ -7007,6 +7116,45 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3", {\ + "packageLocation": "./.yarn/unplugged/@nestjs-core-virtual-e9a099626b/node_modules/@nestjs/core/",\ + "packageDependencies": [\ + ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["@nestjs/websockets", null],\ + ["@nuxtjs/opencollective", "npm:0.3.2"],\ + ["@types/nestjs__common", null],\ + ["@types/nestjs__microservices", null],\ + ["@types/nestjs__platform-express", null],\ + ["@types/nestjs__websockets", null],\ + ["@types/reflect-metadata", null],\ + ["@types/rxjs", null],\ + ["fast-safe-stringify", "npm:2.1.1"],\ + ["iterare", "npm:1.2.1"],\ + ["path-to-regexp", "npm:3.3.0"],\ + ["reflect-metadata", "npm:0.2.2"],\ + ["rxjs", "npm:7.8.1"],\ + ["tslib", "npm:2.7.0"],\ + ["uid", "npm:2.0.2"]\ + ],\ + "packagePeers": [\ + "@nestjs/common",\ + "@nestjs/microservices",\ + "@nestjs/platform-express",\ + "@nestjs/websockets",\ + "@types/nestjs__common",\ + "@types/nestjs__microservices",\ + "@types/nestjs__platform-express",\ + "@types/nestjs__websockets",\ + "@types/reflect-metadata",\ + "@types/rxjs",\ + "reflect-metadata",\ + "rxjs"\ + ],\ + "linkType": "HARD"\ + }],\ ["virtual:77887786a24289fa840c9acd370d634accbe79bcf317ecf5401844ffff73b8a593879dd9cce463873637e6414a631dfdb1a2473704bf332d823bcfffac8c2469#npm:10.4.1", {\ "packageLocation": "./.yarn/unplugged/@nestjs-core-virtual-fd48581fb6/node_modules/@nestjs/core/",\ "packageDependencies": [\ @@ -7438,6 +7586,13 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "SOFT"\ }],\ + ["npm:10.4.6", {\ + "packageLocation": "../.yarn/berry/cache/@nestjs-microservices-npm-10.4.6-87a14e792d-10c0.zip/node_modules/@nestjs/microservices/",\ + "packageDependencies": [\ + ["@nestjs/microservices", "npm:10.4.6"]\ + ],\ + "linkType": "SOFT"\ + }],\ ["virtual:0bc3504934a30fca5775db1c73a91864628f7965ab6dd1fddc21cd16addcd821c36580c256d09baa307a7c821a87e6afc774b3c7c84ecf4ca215862f2bb0a0f3#npm:10.4.1", {\ "packageLocation": "./.yarn/__virtual__/@nestjs-microservices-virtual-3b097837b9/2/.yarn/berry/cache/@nestjs-microservices-npm-10.4.1-709407ada4-10c0.zip/node_modules/@nestjs/microservices/",\ "packageDependencies": [\ @@ -7816,6 +7971,69 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6", {\ + "packageLocation": "./.yarn/__virtual__/@nestjs-microservices-virtual-a1a759d5fb/2/.yarn/berry/cache/@nestjs-microservices-npm-10.4.6-87a14e792d-10c0.zip/node_modules/@nestjs/microservices/",\ + "packageDependencies": [\ + ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["@grpc/grpc-js", null],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/websockets", null],\ + ["@types/amqp-connection-manager", null],\ + ["@types/amqplib", null],\ + ["@types/cache-manager", null],\ + ["@types/grpc__grpc-js", null],\ + ["@types/ioredis", null],\ + ["@types/kafkajs", null],\ + ["@types/mqtt", null],\ + ["@types/nats", null],\ + ["@types/nestjs__common", null],\ + ["@types/nestjs__core", null],\ + ["@types/nestjs__websockets", null],\ + ["@types/reflect-metadata", null],\ + ["@types/rxjs", null],\ + ["amqp-connection-manager", null],\ + ["amqplib", null],\ + ["cache-manager", null],\ + ["ioredis", null],\ + ["iterare", "npm:1.2.1"],\ + ["kafkajs", null],\ + ["mqtt", null],\ + ["nats", null],\ + ["reflect-metadata", "npm:0.2.2"],\ + ["rxjs", "npm:7.8.1"],\ + ["tslib", "npm:2.7.0"]\ + ],\ + "packagePeers": [\ + "@grpc/grpc-js",\ + "@nestjs/common",\ + "@nestjs/core",\ + "@nestjs/websockets",\ + "@types/amqp-connection-manager",\ + "@types/amqplib",\ + "@types/cache-manager",\ + "@types/grpc__grpc-js",\ + "@types/ioredis",\ + "@types/kafkajs",\ + "@types/mqtt",\ + "@types/nats",\ + "@types/nestjs__common",\ + "@types/nestjs__core",\ + "@types/nestjs__websockets",\ + "@types/reflect-metadata",\ + "@types/rxjs",\ + "amqp-connection-manager",\ + "amqplib",\ + "cache-manager",\ + "ioredis",\ + "kafkajs",\ + "mqtt",\ + "nats",\ + "reflect-metadata",\ + "rxjs"\ + ],\ + "linkType": "HARD"\ + }],\ ["virtual:77887786a24289fa840c9acd370d634accbe79bcf317ecf5401844ffff73b8a593879dd9cce463873637e6414a631dfdb1a2473704bf332d823bcfffac8c2469#npm:10.4.1", {\ "packageLocation": "./.yarn/__virtual__/@nestjs-microservices-virtual-f7a1c4c51f/2/.yarn/berry/cache/@nestjs-microservices-npm-10.4.1-709407ada4-10c0.zip/node_modules/@nestjs/microservices/",\ "packageDependencies": [\ @@ -8077,6 +8295,13 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "SOFT"\ }],\ + ["npm:10.4.6", {\ + "packageLocation": "../.yarn/berry/cache/@nestjs-platform-express-npm-10.4.6-eebcfdc0f9-10c0.zip/node_modules/@nestjs/platform-express/",\ + "packageDependencies": [\ + ["@nestjs/platform-express", "npm:10.4.6"]\ + ],\ + "linkType": "SOFT"\ + }],\ ["virtual:095bace236d8ecd40b8814fd9ba4cdf2895bf2e2db003ebed3d7dbb04dd5c36db15a52335ba85a0463393568d3c011d855728282a1b03750c53f85aa4912f9c8#npm:10.4.1", {\ "packageLocation": "./.yarn/__virtual__/@nestjs-platform-express-virtual-bc0fce6de1/2/.yarn/berry/cache/@nestjs-platform-express-npm-10.4.1-76944971fd-10c0.zip/node_modules/@nestjs/platform-express/",\ "packageDependencies": [\ @@ -8253,6 +8478,28 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6", {\ + "packageLocation": "./.yarn/__virtual__/@nestjs-platform-express-virtual-1cbdbccd71/2/.yarn/berry/cache/@nestjs-platform-express-npm-10.4.6-eebcfdc0f9-10c0.zip/node_modules/@nestjs/platform-express/",\ + "packageDependencies": [\ + ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@types/nestjs__common", null],\ + ["@types/nestjs__core", null],\ + ["body-parser", "npm:1.20.3"],\ + ["cors", "npm:2.8.5"],\ + ["express", "npm:4.21.1"],\ + ["multer", "npm:1.4.4-lts.1"],\ + ["tslib", "npm:2.7.0"]\ + ],\ + "packagePeers": [\ + "@nestjs/common",\ + "@nestjs/core",\ + "@types/nestjs__common",\ + "@types/nestjs__core"\ + ],\ + "linkType": "HARD"\ + }],\ ["virtual:8bec40052f12d81e744cef5439ddeb1bd4ee060e942425ae026f8b5aed6129af0faec331a05cddfcc04441a2829beb6b34db019f2a7591ff3c9c1776de7944d5#npm:10.4.1", {\ "packageLocation": "./.yarn/__virtual__/@nestjs-platform-express-virtual-ccaf9dc4c1/2/.yarn/berry/cache/@nestjs-platform-express-npm-10.4.1-76944971fd-10c0.zip/node_modules/@nestjs/platform-express/",\ "packageDependencies": [\ @@ -13490,6 +13737,13 @@ const RAW_RUNTIME_STATE = ["cookie", "npm:0.6.0"]\ ],\ "linkType": "HARD"\ + }],\ + ["npm:0.7.1", {\ + "packageLocation": "../.yarn/berry/cache/cookie-npm-0.7.1-f01524ff99-10c0.zip/node_modules/cookie/",\ + "packageDependencies": [\ + ["cookie", "npm:0.7.1"]\ + ],\ + "linkType": "HARD"\ }]\ ]],\ ["cookie-signature", [\ @@ -15216,6 +15470,44 @@ const RAW_RUNTIME_STATE = ["vary", "npm:1.1.2"]\ ],\ "linkType": "HARD"\ + }],\ + ["npm:4.21.1", {\ + "packageLocation": "../.yarn/berry/cache/express-npm-4.21.1-f1cd48000b-10c0.zip/node_modules/express/",\ + "packageDependencies": [\ + ["express", "npm:4.21.1"],\ + ["accepts", "npm:1.3.8"],\ + ["array-flatten", "npm:1.1.1"],\ + ["body-parser", "npm:1.20.3"],\ + ["content-disposition", "npm:0.5.4"],\ + ["content-type", "npm:1.0.5"],\ + ["cookie", "npm:0.7.1"],\ + ["cookie-signature", "npm:1.0.6"],\ + ["debug", "virtual:c7b184cd14c02e3ce555ab1875e60cf5033c617e17d82c4c02ea822101d3c817f48bf25a766b4d4335742dc5c9c14c2e88a57ed955a56c4ad0613899f82f5618#npm:2.6.9"],\ + ["depd", "npm:2.0.0"],\ + ["encodeurl", "npm:2.0.0"],\ + ["escape-html", "npm:1.0.3"],\ + ["etag", "npm:1.8.1"],\ + ["finalhandler", "npm:1.3.1"],\ + ["fresh", "npm:0.5.2"],\ + ["http-errors", "npm:2.0.0"],\ + ["merge-descriptors", "npm:1.0.3"],\ + ["methods", "npm:1.1.2"],\ + ["on-finished", "npm:2.4.1"],\ + ["parseurl", "npm:1.3.3"],\ + ["path-to-regexp", "npm:0.1.10"],\ + ["proxy-addr", "npm:2.0.7"],\ + ["qs", "npm:6.13.0"],\ + ["range-parser", "npm:1.2.1"],\ + ["safe-buffer", "npm:5.2.1"],\ + ["send", "npm:0.19.0"],\ + ["serve-static", "npm:1.16.2"],\ + ["setprototypeof", "npm:1.2.0"],\ + ["statuses", "npm:2.0.1"],\ + ["type-is", "npm:1.6.18"],\ + ["utils-merge", "npm:1.0.1"],\ + ["vary", "npm:1.1.2"]\ + ],\ + "linkType": "HARD"\ }]\ ]],\ ["extend", [\ diff --git a/packages/nestjs-connectrpc/README.md b/packages/nestjs-connectrpc/README.md new file mode 100644 index 00000000..9af49270 --- /dev/null +++ b/packages/nestjs-connectrpc/README.md @@ -0,0 +1 @@ +ConnectRpc migration @wolfcoded/nestjs-bufconnect diff --git a/packages/nestjs-connectrpc/package.json b/packages/nestjs-connectrpc/package.json new file mode 100644 index 00000000..9c8370b7 --- /dev/null +++ b/packages/nestjs-connectrpc/package.json @@ -0,0 +1,53 @@ +{ + "name": "@atls/nestjs-connectrpc", + "version": "0.0.0", + "license": "BSD-3-Clause", + "type": "module", + "exports": { + "./package.json": "./package.json", + ".": "./src/index.ts" + }, + "main": "src/index.ts", + "files": [ + "dist" + ], + "scripts": { + "build": "yarn library build", + "prepack": "yarn run build", + "postpack": "rm -rf dist" + }, + "devDependencies": { + "@bufbuild/protobuf": "^1.10.0", + "@connectrpc/connect": "^1.5.0", + "@connectrpc/connect-node": "^1.5.0", + "@nestjs/common": "^10.0.5", + "@nestjs/core": "^10.0.5", + "@nestjs/microservices": "^10.2.4", + "@nestjs/platform-express": "^10.2.4", + "reflect-metadata": "^0.2.2", + "rxjs": "^7.8.1" + }, + "peerDependencies": { + "@bufbuild/protobuf": "^1", + "@connectrpc/connect": "^1", + "@connectrpc/connect-node": "^1", + "@nestjs/common": "^10", + "@nestjs/core": "^10", + "@nestjs/microservices": "^10", + "@nestjs/platform-express": "^10", + "reflect-metadata": "^0.2", + "rxjs": "^7" + }, + "publishConfig": { + "exports": { + "./package.json": "./package.json", + ".": { + "import": "./dist/index.js", + "types": "./dist/index.d.ts", + "default": "./dist/index.js" + } + }, + "main": "dist/index.js", + "typings": "dist/index.d.ts" + } +} diff --git a/packages/nestjs-connectrpc/src/async.utils.ts b/packages/nestjs-connectrpc/src/async.utils.ts new file mode 100644 index 00000000..1941ef67 --- /dev/null +++ b/packages/nestjs-connectrpc/src/async.utils.ts @@ -0,0 +1,99 @@ +import type { ResultOrDeferred } from './connectrpc.interfaces.js' + +import { Observable } from 'rxjs' + +export function isAsyncGenerator(input: unknown): input is AsyncGenerator { + return typeof input === 'object' && input !== null && Symbol.asyncIterator in input +} + +export async function* observableToAsyncGenerator(observable: Observable): AsyncGenerator { + const queue: Array = [] + let didComplete = false + let error: unknown = null + + const subscriber = observable.subscribe({ + next: (value) => { + queue.push(value) + }, + error: (innerError) => { + error = innerError + didComplete = true + }, + complete: () => { + didComplete = true + }, + }) + + try { + while (!didComplete || queue.length > 0) { + if (queue.length > 0) { + const item = queue.shift() + if (item !== undefined) { + yield item + } + } else { + // eslint-disable-next-line no-await-in-loop, no-promise-executor-return + await new Promise((resolve) => setTimeout(resolve, 10)) + } + } + + if (error) { + throw new Error(String(error)) + } + } finally { + subscriber.unsubscribe() + } +} + +export const isObservable = (object: unknown): object is Observable => + object instanceof Observable + +export const hasSubscribe = (object: unknown): object is { subscribe: () => void } => + typeof object === 'object' && + object !== null && + typeof (object as { subscribe?: () => void }).subscribe === 'function' + +export const hasToPromise = (object: unknown): object is { toPromise: () => Promise } => + typeof object === 'object' && + object !== null && + typeof (object as { toPromise?: () => Promise }).toPromise === 'function' + +export const transformToObservable = (resultOrDeferred: ResultOrDeferred): Observable => { + if (isObservable(resultOrDeferred)) { + return resultOrDeferred + } + if (hasSubscribe(resultOrDeferred)) { + return new Observable(() => { + resultOrDeferred.subscribe() + }) + } + if (hasToPromise(resultOrDeferred)) { + return new Observable((subscriber) => { + resultOrDeferred + .toPromise() + .then((response: T) => { + subscriber.next(response) + subscriber.complete() + }) + .catch((error: unknown) => { + subscriber.error(error) + }) + }) + } + return new Observable((subscriber) => { + subscriber.next(resultOrDeferred) + subscriber.complete() + }) +} + +export async function* toAsyncGenerator( + input: AsyncGenerator | Observable +): AsyncGenerator { + if (isObservable(input)) { + yield* observableToAsyncGenerator(input) + } else if (isAsyncGenerator(input)) { + yield* input + } else { + throw new Error('Unsupported input type. Expected an Observable or an AsyncGenerator.') + } +} diff --git a/packages/nestjs-connectrpc/src/connectrpc.constants.ts b/packages/nestjs-connectrpc/src/connectrpc.constants.ts new file mode 100644 index 00000000..a393c8cb --- /dev/null +++ b/packages/nestjs-connectrpc/src/connectrpc.constants.ts @@ -0,0 +1,5 @@ +export const METHOD_DECORATOR_KEY = Symbol('METHOD_DECORATOR_KEY') + +export const STREAM_METHOD_DECORATOR_KEY = Symbol('STREAM_METHOD_DECORATOR_KEY') + +export const CONNECTRPC_TRANSPORT = Symbol('CONNECTRPC_TRANSPORT') diff --git a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts new file mode 100644 index 00000000..8f9ef197 --- /dev/null +++ b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts @@ -0,0 +1,88 @@ +import type { ServiceType } from '@bufbuild/protobuf' + +import type { ConstructorWithPrototype } from './connectrpc.interfaces.js' +import type { FunctionPropertyDescriptor } from './connectrpc.interfaces.js' +import type { MethodKey } from './connectrpc.interfaces.js' +import type { MethodKeys } from './connectrpc.interfaces.js' + +import { MessagePattern } from '@nestjs/microservices' + +import { CONNECTRPC_TRANSPORT } from './connectrpc.constants.js' +import { METHOD_DECORATOR_KEY } from './connectrpc.constants.js' +import { STREAM_METHOD_DECORATOR_KEY } from './connectrpc.constants.js' +import { MethodType } from './connectrpc.interfaces.js' +import { CustomMetadataStore } from './custom-metadata.storage.js' +import { createConnectRpcMethodMetadata } from './router.utils.js' + +function isFunctionPropertyDescriptor( + descriptor: PropertyDescriptor | undefined +): descriptor is FunctionPropertyDescriptor { + return descriptor !== undefined && typeof descriptor.value === 'function' +} + +export const ConnectRpcService = (serviceName: ServiceType): ClassDecorator => + (target: ConstructorWithPrototype): void => { + const processMethodKey = (methodImpl: MethodKey): void => { + const functionName = methodImpl.key + const { methodType } = methodImpl + + const descriptor = Object.getOwnPropertyDescriptor(target.prototype, functionName) + + if (isFunctionPropertyDescriptor(descriptor)) { + const metadata = createConnectRpcMethodMetadata( + descriptor.value, + functionName, + serviceName.typeName, + functionName, + methodType + ) + + const customMetadataStore = CustomMetadataStore.getInstance() + customMetadataStore.set(serviceName.typeName, serviceName) + + MessagePattern(metadata, CONNECTRPC_TRANSPORT)(target.prototype, functionName, descriptor) + } + } + + const unaryMethodKeys: MethodKeys = Reflect.getMetadata(METHOD_DECORATOR_KEY, target) || [] + const streamMethodKeys: MethodKeys = + Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target) || [] + + unaryMethodKeys.forEach((methodImpl) => { + processMethodKey(methodImpl) + }) + streamMethodKeys.forEach((methodImpl) => { + processMethodKey(methodImpl) + }) + } + +export const ConnectRpcMethod = (): MethodDecorator => (target: object, key: string | symbol) => { + const metadata: MethodKey = { + key: key.toString(), + methodType: MethodType.NO_STREAMING, + } + + const existingMethods: Set = + Reflect.getMetadata(METHOD_DECORATOR_KEY, target.constructor) || new Set() + + if (!existingMethods.has(metadata)) { + existingMethods.add(metadata) + Reflect.defineMetadata(METHOD_DECORATOR_KEY, existingMethods, target.constructor) + } +} + +export const ConnectRpcStreamMethod = (): MethodDecorator => + (target: object, key: string | symbol) => { + const metadata: MethodKey = { + key: key.toString(), + methodType: MethodType.RX_STREAMING, + } + + const existingMethods: Set = + Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target.constructor) || new Set() + + if (!existingMethods.has(metadata)) { + existingMethods.add(metadata) + Reflect.defineMetadata(STREAM_METHOD_DECORATOR_KEY, existingMethods, target.constructor) + } + } diff --git a/packages/nestjs-connectrpc/src/connectrpc.interfaces.ts b/packages/nestjs-connectrpc/src/connectrpc.interfaces.ts new file mode 100644 index 00000000..d1f56cb7 --- /dev/null +++ b/packages/nestjs-connectrpc/src/connectrpc.interfaces.ts @@ -0,0 +1,74 @@ +import type * as http from 'http' +import type * as http2 from 'http2' +import type * as https from 'https' +import type { ConnectRouterOptions } from '@connectrpc/connect' +import type { Observable } from 'rxjs' + +export interface ConnectRpcPattern { + service: string + rpc: string + streaming: MethodType +} + +export enum MethodType { + NO_STREAMING = 'no_stream', + RX_STREAMING = 'rx_stream', +} + +export enum ServerProtocol { + HTTP = 'http', + HTTPS = 'https', + HTTP2 = 'http2', + HTTP2_INSECURE = 'http2_insecure', +} + +export interface BaseServerOptions { + port: number + connectOptions?: ConnectRouterOptions + callback?: () => void +} + +export interface HttpOptions extends BaseServerOptions { + protocol: ServerProtocol.HTTP + serverOptions?: http.ServerOptions +} + +export interface HttpsOptions extends BaseServerOptions { + protocol: ServerProtocol.HTTPS + serverOptions: https.ServerOptions +} + +export interface Http2Options extends BaseServerOptions { + protocol: ServerProtocol.HTTP2 + serverOptions: http2.SecureServerOptions +} + +export interface Http2InsecureOptions extends BaseServerOptions { + protocol: ServerProtocol.HTTP2_INSECURE + serverOptions?: http2.ServerOptions +} + +export type ServerTypeOptions = Http2InsecureOptions | Http2Options | HttpOptions | HttpsOptions + +export type ServerInstance = http.Server | http2.Http2Server | https.Server | null + +export interface ConstructorWithPrototype { + prototype: Record +} + +export interface MethodKey { + key: string + methodType: MethodType +} + +export type MethodKeys = Array + +export interface FunctionPropertyDescriptor extends PropertyDescriptor { + value: (...arguments_: Array) => never +} + +export type ResultOrDeferred = + | Observable + | T + | { subscribe: () => void } + | { toPromise: () => Promise } diff --git a/packages/nestjs-connectrpc/src/connectrpc.server.ts b/packages/nestjs-connectrpc/src/connectrpc.server.ts new file mode 100644 index 00000000..ea193441 --- /dev/null +++ b/packages/nestjs-connectrpc/src/connectrpc.server.ts @@ -0,0 +1,128 @@ +import type { ConnectRouter } from '@connectrpc/connect' + +import type { Http2InsecureOptions } from './connectrpc.interfaces.js' +import type { Http2Options } from './connectrpc.interfaces.js' +import type { ServerTypeOptions } from './connectrpc.interfaces.js' +import type { HttpsOptions } from './connectrpc.interfaces.js' +import type { HttpOptions } from './connectrpc.interfaces.js' +import type { ServerInstance } from './connectrpc.interfaces.js' + +import { connectNodeAdapter } from '@connectrpc/connect-node' +import * as http from 'http' +import * as http2 from 'http2' +import * as https from 'https' + +import { ServerProtocol } from './connectrpc.interfaces.js' + +export class HTTPServer { + private serverPrivate: ServerInstance = null + + constructor( + private readonly options: ServerTypeOptions, + private readonly router: (router: ConnectRouter) => void + ) {} + + set server(value: http.Server | http2.Http2Server | https.Server | null) { + this.serverPrivate = value + } + + get server(): http.Server | http2.Http2Server | https.Server | null { + return this.serverPrivate + } + + async listen(): Promise { + return new Promise((resolve, reject) => { + this.startServer(resolve, reject) + }) + } + + createHttpServer(): http.Server { + const { serverOptions = {}, connectOptions = {} } = this.options as HttpOptions + + return http.createServer( + serverOptions, + connectNodeAdapter({ + ...connectOptions, + routes: this.router, + }) + ) + } + + createHttpsServer(): https.Server { + const { serverOptions = {}, connectOptions = {} } = this.options as HttpsOptions + + return https.createServer( + serverOptions, + connectNodeAdapter({ ...connectOptions, routes: this.router }) + ) + } + + createHttp2Server(): http2.Http2Server { + const { serverOptions = {}, connectOptions = {} } = this.options as Http2Options + + return http2.createSecureServer( + serverOptions, + connectNodeAdapter({ ...connectOptions, routes: this.router }) + ) + } + + createHttp2InsecureServer(): http2.Http2Server { + const { serverOptions = {}, connectOptions = {} } = this.options as Http2InsecureOptions + + return http2.createServer( + serverOptions, + connectNodeAdapter({ ...connectOptions, routes: this.router }) + ) + } + + startServer(resolve: () => void, reject: (error: Error) => void): void { + try { + switch (this.options.protocol) { + case ServerProtocol.HTTP: { + this.server = this.createHttpServer() + break + } + case ServerProtocol.HTTPS: { + this.server = this.createHttpsServer() + break + } + case ServerProtocol.HTTP2: { + this.server = this.createHttp2Server() + break + } + case ServerProtocol.HTTP2_INSECURE: { + this.server = this.createHttp2InsecureServer() + break + } + default: { + throw new Error('Invalid protocol option') + } + } + + this.server.listen(this.options.port, () => { + if (this.options.callback) this.options.callback() + resolve() + }) + } catch (error) { + if (error instanceof Error) { + reject(error) + } else { + reject(new Error('Unknown error occurred')) + } + } + } + + async close(callback?: () => void): Promise { + return new Promise((resolve, reject) => { + if (this.server === null) { + reject(new Error('Server is not running')) + } else { + this.server.close(() => { + this.server = null + if (callback) callback() + resolve() + }) + } + }) + } +} diff --git a/packages/nestjs-connectrpc/src/connectrpc.strategy.ts b/packages/nestjs-connectrpc/src/connectrpc.strategy.ts new file mode 100644 index 00000000..c30651a1 --- /dev/null +++ b/packages/nestjs-connectrpc/src/connectrpc.strategy.ts @@ -0,0 +1,72 @@ +import type { ConnectRouter } from '@connectrpc/connect' +import type { CustomTransportStrategy } from '@nestjs/microservices' +import type { MessageHandler } from '@nestjs/microservices' + +import type { ServerTypeOptions } from './connectrpc.interfaces.js' + +import { Server } from '@nestjs/microservices' +import { isString } from '@nestjs/common/utils/shared.utils.js' + +import { HTTPServer } from './connectrpc.server.js' +import { CustomMetadataStore } from './custom-metadata.storage.js' +import { addServicesToRouter } from './router.utils.js' +import { createServiceHandlersMap } from './router.utils.js' + +export class ConnectRpcServer extends Server implements CustomTransportStrategy { + private readonly customMetadataStore: CustomMetadataStore | null = null + + private server: HTTPServer | null = null + + private readonly options: ServerTypeOptions + + constructor(options: ServerTypeOptions) { + super() + this.customMetadataStore = CustomMetadataStore.getInstance() + this.options = options + } + + async listen( + callback: (error?: unknown, ...optionalParameters: Array) => void + ): Promise { + try { + const router = this.buildRouter() + this.server = new HTTPServer(this.options, router) + + await this.server.listen() + + callback() + } catch (error) { + callback(error) + } + } + + public async close(): Promise { + await this.server?.close() + } + + public override addHandler( + pattern: unknown, + callback: MessageHandler, + isEventHandler = false + ): void { + const route = isString(pattern) ? pattern : JSON.stringify(pattern) + if (isEventHandler) { + const modifiedCallback = callback + modifiedCallback.isEventHandler = true + this.messageHandlers.set(route, modifiedCallback) + } + this.messageHandlers.set(route, callback) + } + + buildRouter() { + return (router: ConnectRouter): void => { + if (this.customMetadataStore) { + const serviceHandlersMap = createServiceHandlersMap( + this.getHandlers(), + this.customMetadataStore + ) + addServicesToRouter(router, serviceHandlersMap, this.customMetadataStore) + } + } + } +} diff --git a/packages/nestjs-connectrpc/src/custom-metadata.storage.ts b/packages/nestjs-connectrpc/src/custom-metadata.storage.ts new file mode 100644 index 00000000..93b4ae9d --- /dev/null +++ b/packages/nestjs-connectrpc/src/custom-metadata.storage.ts @@ -0,0 +1,25 @@ +import type { ServiceType } from '@bufbuild/protobuf' + +export class CustomMetadataStore { + private static instance: CustomMetadataStore + + private customMetadata: Map = new Map() + + // eslint-disable-next-line @typescript-eslint/no-empty-function + private constructor() {} + + public static getInstance(): CustomMetadataStore { + if (!CustomMetadataStore.instance) { + CustomMetadataStore.instance = new CustomMetadataStore() + } + return CustomMetadataStore.instance + } + + set(key: string, value: ServiceType): void { + this.customMetadata.set(key, value) + } + + get(key: string): ServiceType | undefined { + return this.customMetadata.get(key) ?? undefined + } +} diff --git a/packages/nestjs-connectrpc/src/index.ts b/packages/nestjs-connectrpc/src/index.ts new file mode 100644 index 00000000..395e5d22 --- /dev/null +++ b/packages/nestjs-connectrpc/src/index.ts @@ -0,0 +1,4 @@ +export * from './connectrpc.decorators.js' +export * from './connectrpc.interfaces.js' +export * from './connectrpc.strategy.js' +export * from './connectrpc.server.js' diff --git a/packages/nestjs-connectrpc/src/router.utils.ts b/packages/nestjs-connectrpc/src/router.utils.ts new file mode 100644 index 00000000..3482590f --- /dev/null +++ b/packages/nestjs-connectrpc/src/router.utils.ts @@ -0,0 +1,117 @@ +import type { ServiceType } from '@bufbuild/protobuf' +import type { ConnectRouter } from '@connectrpc/connect' +import type { ServiceImpl } from '@connectrpc/connect' +import type { MessageHandler } from '@nestjs/microservices' +import type { Observable } from 'rxjs' + +import type { ConnectRpcPattern } from './connectrpc.interfaces.js' +import type { CustomMetadataStore } from './custom-metadata.storage.js' + +import { lastValueFrom } from 'rxjs' + +import { MethodType } from './connectrpc.interfaces.js' +import { toAsyncGenerator } from './async.utils.js' +import { transformToObservable } from './async.utils.js' + +export const createPattern = (service: string, methodName: string, streaming: MethodType): string => + JSON.stringify({ + service, + rpc: methodName, + streaming, + } as ConnectRpcPattern) + +export const addServicesToRouter = ( + router: ConnectRouter, + serviceHandlersMap: Record>>, + customMetadataStore: CustomMetadataStore +): void => { + Object.keys(serviceHandlersMap).forEach((serviceName) => { + const service = customMetadataStore.get(serviceName) + if (service) { + router.service(service, serviceHandlersMap[serviceName]) + } + }) +} + +export const createServiceHandlersMap = ( + handlers: Map, + customMetadataStore: CustomMetadataStore +): Record>> => { + const serviceHandlersMap: Record>> = {} + + handlers.forEach((handlerMetadata, pattern) => { + const parsedPattern = JSON.parse(pattern) + + if (handlerMetadata) { + const service = customMetadataStore.get(parsedPattern.service as string) + const methodProto = service?.methods[parsedPattern.rpc] + + if (service && methodProto) { + if (!serviceHandlersMap[parsedPattern.service]) { + serviceHandlersMap[parsedPattern.service] = {} + } + + switch (parsedPattern.streaming) { + case MethodType.NO_STREAMING: { + serviceHandlersMap[parsedPattern.service][parsedPattern.rpc] = async ( + request: unknown, + context: unknown + ): Promise => { + const result = handlerMetadata(request, context) + const resultOrDeferred = await result + return lastValueFrom(transformToObservable(resultOrDeferred)) + } + break + } + case MethodType.RX_STREAMING: { + serviceHandlersMap[parsedPattern.service][parsedPattern.rpc] = + async function* rxStreamingHandler( + request: unknown, + context: unknown + ): AsyncGenerator { + const result = handlerMetadata(request, context) + const streamOrValue = await result + yield* toAsyncGenerator( + streamOrValue as AsyncGenerator | Observable + ) + } + break + } + default: { + throw new Error('Invalid streaming type') + } + } + } + } + }) + + return serviceHandlersMap +} + +export const createConnectRpcMethodMetadata = ( + target: object, + key: string | symbol, + service: string | undefined, + method: string | undefined, + streaming = MethodType.NO_STREAMING +): { + service: string + rpc: string | undefined + streaming: MethodType +} => { + const capitalizeFirstLetter = (input: string): string => + input.charAt(0).toUpperCase() + input.slice(1) + + if (!service) { + const { name } = target.constructor + return { + service: name, + rpc: capitalizeFirstLetter(key as string), + streaming, + } + } + if (service && !method) { + return { service, rpc: capitalizeFirstLetter(key as string), streaming } + } + return { service, rpc: method, streaming } +} diff --git a/yarn.lock b/yarn.lock index e92d81f9..1d2381d7 100644 --- a/yarn.lock +++ b/yarn.lock @@ -325,6 +325,32 @@ __metadata: languageName: unknown linkType: soft +"@atls/nestjs-connectrpc@workspace:packages/nestjs-connectrpc": + version: 0.0.0-use.local + resolution: "@atls/nestjs-connectrpc@workspace:packages/nestjs-connectrpc" + dependencies: + "@bufbuild/protobuf": "npm:^1.10.0" + "@connectrpc/connect": "npm:^1.5.0" + "@connectrpc/connect-node": "npm:^1.5.0" + "@nestjs/common": "npm:^10.0.5" + "@nestjs/core": "npm:^10.0.5" + "@nestjs/microservices": "npm:^10.2.4" + "@nestjs/platform-express": "npm:^10.2.4" + reflect-metadata: "npm:^0.2.2" + rxjs: "npm:^7.8.1" + peerDependencies: + "@bufbuild/protobuf": ^1 + "@connectrpc/connect": ^1 + "@connectrpc/connect-node": ^1 + "@nestjs/common": ^10 + "@nestjs/core": ^10 + "@nestjs/microservices": ^10 + "@nestjs/platform-express": ^10 + reflect-metadata: ^0.2 + rxjs: ^7 + languageName: unknown + linkType: soft + "@atls/nestjs-cqrs@workspace:packages/nestjs-cqrs": version: 0.0.0-use.local resolution: "@atls/nestjs-cqrs@workspace:packages/nestjs-cqrs" @@ -2113,6 +2139,34 @@ __metadata: languageName: node linkType: hard +"@bufbuild/protobuf@npm:^1.10.0": + version: 1.10.0 + resolution: "@bufbuild/protobuf@npm:1.10.0" + checksum: 10c0/5487b9c2e63846d0e3bde4d025cc77ae44a22166a5d6c184df0da5581e1ab6d66dd115af0ccad814576dcd011bb1b93989fb0ac1eb4ae452979bb8b186693ba0 + languageName: node + linkType: hard + +"@connectrpc/connect-node@npm:^1.5.0": + version: 1.6.1 + resolution: "@connectrpc/connect-node@npm:1.6.1" + dependencies: + undici: "npm:^5.28.4" + peerDependencies: + "@bufbuild/protobuf": ^1.10.0 + "@connectrpc/connect": 1.6.1 + checksum: 10c0/9891bbbe5ec155d16141e378c120dd6d4c47e1517656d4676aca762d70426a9eb3d9ec92595a7cfc4f5cbe40ff5be572d0c3d9010058107854e7f62ee05fb46e + languageName: node + linkType: hard + +"@connectrpc/connect@npm:^1.5.0": + version: 1.6.1 + resolution: "@connectrpc/connect@npm:1.6.1" + peerDependencies: + "@bufbuild/protobuf": ^1.10.0 + checksum: 10c0/35c6fd3e33c3a1ff9dce230b059ecd7991ef0dc60c16fb898e5c46b930a01077ac0b34d53d6742cc8ed079f20f8eacc7c77a8620aeec9efaf68950494f387011 + languageName: node + linkType: hard + "@emotion/css-prettifier@npm:^1.1.4": version: 1.1.4 resolution: "@emotion/css-prettifier@npm:1.1.4" @@ -4446,6 +4500,49 @@ __metadata: languageName: node linkType: hard +"@nestjs/microservices@npm:^10.2.4": + version: 10.4.6 + resolution: "@nestjs/microservices@npm:10.4.6" + dependencies: + iterare: "npm:1.2.1" + tslib: "npm:2.7.0" + peerDependencies: + "@grpc/grpc-js": "*" + "@nestjs/common": ^10.0.0 + "@nestjs/core": ^10.0.0 + "@nestjs/websockets": ^10.0.0 + amqp-connection-manager: "*" + amqplib: "*" + cache-manager: "*" + ioredis: "*" + kafkajs: "*" + mqtt: "*" + nats: "*" + reflect-metadata: ^0.1.12 || ^0.2.0 + rxjs: ^7.1.0 + peerDependenciesMeta: + "@grpc/grpc-js": + optional: true + "@nestjs/websockets": + optional: true + amqp-connection-manager: + optional: true + amqplib: + optional: true + cache-manager: + optional: true + ioredis: + optional: true + kafkajs: + optional: true + mqtt: + optional: true + nats: + optional: true + checksum: 10c0/c341a566599323048a321539e57e07656c376896330fc5c24979d21469e9603bbbaa37ca8fff1255077957d8db821f985b730bce85e5c645261d9b3853d79ea5 + languageName: node + linkType: hard + "@nestjs/platform-express@npm:*": version: 10.4.1 resolution: "@nestjs/platform-express@npm:10.4.1" @@ -4462,6 +4559,22 @@ __metadata: languageName: node linkType: hard +"@nestjs/platform-express@npm:^10.2.4": + version: 10.4.6 + resolution: "@nestjs/platform-express@npm:10.4.6" + dependencies: + body-parser: "npm:1.20.3" + cors: "npm:2.8.5" + express: "npm:4.21.1" + multer: "npm:1.4.4-lts.1" + tslib: "npm:2.7.0" + peerDependencies: + "@nestjs/common": ^10.0.0 + "@nestjs/core": ^10.0.0 + checksum: 10c0/1d2f9d913a0b4a066aaa783810d0d029a9f201243f8ac52d396c8361a19b73d44e05d8d656388af61696784bdfff1cb8dd711bb0b1765496c6bf40b410284081 + languageName: node + linkType: hard + "@nestjs/testing@npm:10.4.1": version: 10.4.1 resolution: "@nestjs/testing@npm:10.4.1" @@ -8414,6 +8527,13 @@ __metadata: languageName: node linkType: hard +"cookie@npm:0.7.1": + version: 0.7.1 + resolution: "cookie@npm:0.7.1" + checksum: 10c0/5de60c67a410e7c8dc8a46a4b72eb0fe925871d057c9a5d2c0e8145c4270a4f81076de83410c4d397179744b478e33cd80ccbcc457abf40a9409ad27dcd21dde + languageName: node + linkType: hard + "cookiejar@npm:^2.1.4": version: 2.1.4 resolution: "cookiejar@npm:2.1.4" @@ -9814,6 +9934,45 @@ __metadata: languageName: node linkType: hard +"express@npm:4.21.1": + version: 4.21.1 + resolution: "express@npm:4.21.1" + dependencies: + accepts: "npm:~1.3.8" + array-flatten: "npm:1.1.1" + body-parser: "npm:1.20.3" + content-disposition: "npm:0.5.4" + content-type: "npm:~1.0.4" + cookie: "npm:0.7.1" + cookie-signature: "npm:1.0.6" + debug: "npm:2.6.9" + depd: "npm:2.0.0" + encodeurl: "npm:~2.0.0" + escape-html: "npm:~1.0.3" + etag: "npm:~1.8.1" + finalhandler: "npm:1.3.1" + fresh: "npm:0.5.2" + http-errors: "npm:2.0.0" + merge-descriptors: "npm:1.0.3" + methods: "npm:~1.1.2" + on-finished: "npm:2.4.1" + parseurl: "npm:~1.3.3" + path-to-regexp: "npm:0.1.10" + proxy-addr: "npm:~2.0.7" + qs: "npm:6.13.0" + range-parser: "npm:~1.2.1" + safe-buffer: "npm:5.2.1" + send: "npm:0.19.0" + serve-static: "npm:1.16.2" + setprototypeof: "npm:1.2.0" + statuses: "npm:2.0.1" + type-is: "npm:~1.6.18" + utils-merge: "npm:1.0.1" + vary: "npm:~1.1.2" + checksum: 10c0/0c287867e5f6129d3def1edd9b63103a53c40d4dc8628839d4b6827e35eb8f0de5a4656f9d85f4457eba584f9871ebb2ad26c750b36bd75d9bbb8bcebdc4892c + languageName: node + linkType: hard + "extend@npm:^3.0.2": version: 3.0.2 resolution: "extend@npm:3.0.2" @@ -13990,7 +14149,7 @@ __metadata: languageName: node linkType: hard -"reflect-metadata@npm:0.2.2, reflect-metadata@npm:^0.2.1": +"reflect-metadata@npm:0.2.2, reflect-metadata@npm:^0.2.1, reflect-metadata@npm:^0.2.2": version: 0.2.2 resolution: "reflect-metadata@npm:0.2.2" checksum: 10c0/1cd93a15ea291e420204955544637c264c216e7aac527470e393d54b4bb075f10a17e60d8168ec96600c7e0b9fcc0cb0bb6e91c3fbf5b0d8c9056f04e6ac1ec2 From 317226c7698a64f7299287e05e814467b88121b6 Mon Sep 17 00:00:00 2001 From: Andrew Ghostuhin Date: Wed, 30 Oct 2024 20:58:37 +0300 Subject: [PATCH 02/10] refactor(nestjs-connectrpc): utils --- packages/nestjs-connectrpc/src/async.utils.ts | 99 ------------- .../src/connectrpc.decorators.ts | 2 +- .../src/connectrpc.strategy.ts | 4 +- .../nestjs-connectrpc/src/router.utils.ts | 117 --------------- .../src/utils/async.utils.ts | 135 ++++++++++++++++++ .../src/utils/router.utils.ts | 115 +++++++++++++++ 6 files changed, 253 insertions(+), 219 deletions(-) delete mode 100644 packages/nestjs-connectrpc/src/async.utils.ts delete mode 100644 packages/nestjs-connectrpc/src/router.utils.ts create mode 100644 packages/nestjs-connectrpc/src/utils/async.utils.ts create mode 100644 packages/nestjs-connectrpc/src/utils/router.utils.ts diff --git a/packages/nestjs-connectrpc/src/async.utils.ts b/packages/nestjs-connectrpc/src/async.utils.ts deleted file mode 100644 index 1941ef67..00000000 --- a/packages/nestjs-connectrpc/src/async.utils.ts +++ /dev/null @@ -1,99 +0,0 @@ -import type { ResultOrDeferred } from './connectrpc.interfaces.js' - -import { Observable } from 'rxjs' - -export function isAsyncGenerator(input: unknown): input is AsyncGenerator { - return typeof input === 'object' && input !== null && Symbol.asyncIterator in input -} - -export async function* observableToAsyncGenerator(observable: Observable): AsyncGenerator { - const queue: Array = [] - let didComplete = false - let error: unknown = null - - const subscriber = observable.subscribe({ - next: (value) => { - queue.push(value) - }, - error: (innerError) => { - error = innerError - didComplete = true - }, - complete: () => { - didComplete = true - }, - }) - - try { - while (!didComplete || queue.length > 0) { - if (queue.length > 0) { - const item = queue.shift() - if (item !== undefined) { - yield item - } - } else { - // eslint-disable-next-line no-await-in-loop, no-promise-executor-return - await new Promise((resolve) => setTimeout(resolve, 10)) - } - } - - if (error) { - throw new Error(String(error)) - } - } finally { - subscriber.unsubscribe() - } -} - -export const isObservable = (object: unknown): object is Observable => - object instanceof Observable - -export const hasSubscribe = (object: unknown): object is { subscribe: () => void } => - typeof object === 'object' && - object !== null && - typeof (object as { subscribe?: () => void }).subscribe === 'function' - -export const hasToPromise = (object: unknown): object is { toPromise: () => Promise } => - typeof object === 'object' && - object !== null && - typeof (object as { toPromise?: () => Promise }).toPromise === 'function' - -export const transformToObservable = (resultOrDeferred: ResultOrDeferred): Observable => { - if (isObservable(resultOrDeferred)) { - return resultOrDeferred - } - if (hasSubscribe(resultOrDeferred)) { - return new Observable(() => { - resultOrDeferred.subscribe() - }) - } - if (hasToPromise(resultOrDeferred)) { - return new Observable((subscriber) => { - resultOrDeferred - .toPromise() - .then((response: T) => { - subscriber.next(response) - subscriber.complete() - }) - .catch((error: unknown) => { - subscriber.error(error) - }) - }) - } - return new Observable((subscriber) => { - subscriber.next(resultOrDeferred) - subscriber.complete() - }) -} - -export async function* toAsyncGenerator( - input: AsyncGenerator | Observable -): AsyncGenerator { - if (isObservable(input)) { - yield* observableToAsyncGenerator(input) - } else if (isAsyncGenerator(input)) { - yield* input - } else { - throw new Error('Unsupported input type. Expected an Observable or an AsyncGenerator.') - } -} diff --git a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts index 8f9ef197..765ae868 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts @@ -12,7 +12,7 @@ import { METHOD_DECORATOR_KEY } from './connectrpc.constants.js' import { STREAM_METHOD_DECORATOR_KEY } from './connectrpc.constants.js' import { MethodType } from './connectrpc.interfaces.js' import { CustomMetadataStore } from './custom-metadata.storage.js' -import { createConnectRpcMethodMetadata } from './router.utils.js' +import { createConnectRpcMethodMetadata } from './utils/router.utils.js' function isFunctionPropertyDescriptor( descriptor: PropertyDescriptor | undefined diff --git a/packages/nestjs-connectrpc/src/connectrpc.strategy.ts b/packages/nestjs-connectrpc/src/connectrpc.strategy.ts index c30651a1..037cd28a 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.strategy.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.strategy.ts @@ -9,8 +9,8 @@ import { isString } from '@nestjs/common/utils/shared.utils. import { HTTPServer } from './connectrpc.server.js' import { CustomMetadataStore } from './custom-metadata.storage.js' -import { addServicesToRouter } from './router.utils.js' -import { createServiceHandlersMap } from './router.utils.js' +import { addServicesToRouter } from './utils/router.utils.js' +import { createServiceHandlersMap } from './utils/router.utils.js' export class ConnectRpcServer extends Server implements CustomTransportStrategy { private readonly customMetadataStore: CustomMetadataStore | null = null diff --git a/packages/nestjs-connectrpc/src/router.utils.ts b/packages/nestjs-connectrpc/src/router.utils.ts deleted file mode 100644 index 3482590f..00000000 --- a/packages/nestjs-connectrpc/src/router.utils.ts +++ /dev/null @@ -1,117 +0,0 @@ -import type { ServiceType } from '@bufbuild/protobuf' -import type { ConnectRouter } from '@connectrpc/connect' -import type { ServiceImpl } from '@connectrpc/connect' -import type { MessageHandler } from '@nestjs/microservices' -import type { Observable } from 'rxjs' - -import type { ConnectRpcPattern } from './connectrpc.interfaces.js' -import type { CustomMetadataStore } from './custom-metadata.storage.js' - -import { lastValueFrom } from 'rxjs' - -import { MethodType } from './connectrpc.interfaces.js' -import { toAsyncGenerator } from './async.utils.js' -import { transformToObservable } from './async.utils.js' - -export const createPattern = (service: string, methodName: string, streaming: MethodType): string => - JSON.stringify({ - service, - rpc: methodName, - streaming, - } as ConnectRpcPattern) - -export const addServicesToRouter = ( - router: ConnectRouter, - serviceHandlersMap: Record>>, - customMetadataStore: CustomMetadataStore -): void => { - Object.keys(serviceHandlersMap).forEach((serviceName) => { - const service = customMetadataStore.get(serviceName) - if (service) { - router.service(service, serviceHandlersMap[serviceName]) - } - }) -} - -export const createServiceHandlersMap = ( - handlers: Map, - customMetadataStore: CustomMetadataStore -): Record>> => { - const serviceHandlersMap: Record>> = {} - - handlers.forEach((handlerMetadata, pattern) => { - const parsedPattern = JSON.parse(pattern) - - if (handlerMetadata) { - const service = customMetadataStore.get(parsedPattern.service as string) - const methodProto = service?.methods[parsedPattern.rpc] - - if (service && methodProto) { - if (!serviceHandlersMap[parsedPattern.service]) { - serviceHandlersMap[parsedPattern.service] = {} - } - - switch (parsedPattern.streaming) { - case MethodType.NO_STREAMING: { - serviceHandlersMap[parsedPattern.service][parsedPattern.rpc] = async ( - request: unknown, - context: unknown - ): Promise => { - const result = handlerMetadata(request, context) - const resultOrDeferred = await result - return lastValueFrom(transformToObservable(resultOrDeferred)) - } - break - } - case MethodType.RX_STREAMING: { - serviceHandlersMap[parsedPattern.service][parsedPattern.rpc] = - async function* rxStreamingHandler( - request: unknown, - context: unknown - ): AsyncGenerator { - const result = handlerMetadata(request, context) - const streamOrValue = await result - yield* toAsyncGenerator( - streamOrValue as AsyncGenerator | Observable - ) - } - break - } - default: { - throw new Error('Invalid streaming type') - } - } - } - } - }) - - return serviceHandlersMap -} - -export const createConnectRpcMethodMetadata = ( - target: object, - key: string | symbol, - service: string | undefined, - method: string | undefined, - streaming = MethodType.NO_STREAMING -): { - service: string - rpc: string | undefined - streaming: MethodType -} => { - const capitalizeFirstLetter = (input: string): string => - input.charAt(0).toUpperCase() + input.slice(1) - - if (!service) { - const { name } = target.constructor - return { - service: name, - rpc: capitalizeFirstLetter(key as string), - streaming, - } - } - if (service && !method) { - return { service, rpc: capitalizeFirstLetter(key as string), streaming } - } - return { service, rpc: method, streaming } -} diff --git a/packages/nestjs-connectrpc/src/utils/async.utils.ts b/packages/nestjs-connectrpc/src/utils/async.utils.ts new file mode 100644 index 00000000..49a49154 --- /dev/null +++ b/packages/nestjs-connectrpc/src/utils/async.utils.ts @@ -0,0 +1,135 @@ +import type { ResultOrDeferred } from '../connectrpc.interfaces.js' +import { Observable, from, Subject } from 'rxjs' +import { lastValueFrom } from 'rxjs' + +/** + * Type guard to check if a given input is an AsyncGenerator. + * @param {unknown} input - The input to check. + * @returns {boolean} - True if the input is an AsyncGenerator, otherwise false. + */ +export function isAsyncGenerator(input: unknown): input is AsyncGenerator { + return typeof input === 'object' && input !== null && Symbol.asyncIterator in input +} + +/** + * Converts an Observable to an AsyncGenerator, yielding items emitted by the Observable. + * @param {Observable} observable - The Observable to convert. + * @returns {AsyncGenerator} - An AsyncGenerator that yields each emitted value from the Observable. + */ +export async function* observableToAsyncGenerator(observable: Observable): AsyncGenerator { + const queue = new Subject() + let didComplete = false + + const subscriber = observable.subscribe({ + next: (value) => queue.next(value), + error: (error) => queue.error(error), + complete: () => { + didComplete = true + queue.complete() + }, + }) + + try { + for await (const item of asyncIterator(queue)) { + yield item + } + } finally { + subscriber.unsubscribe() + } +} + +/** + * Utility function to create an async iterator for a Subject. + * @param {Subject} subject - The Subject to create an iterator from. + * @returns {AsyncIterableIterator} - The async iterator. + */ +async function* asyncIterator(subject: Subject): AsyncIterableIterator { + const values: T[] = [] + const nextValue = () => + new Promise((resolve, reject) => { + subject.subscribe({ + next: (val) => resolve(val), + error: (err) => reject(err), + complete: () => resolve(null as any), + }) + }) + + while (true) { + const item = await nextValue() + if (item === null) return + yield item + } +} + +/** + * Type guard to check if a given object is an Observable. + * @param {unknown} object - The object to check. + * @returns {boolean} - True if the object is an Observable, otherwise false. + */ +export const isObservable = (object: unknown): object is Observable => + object instanceof Observable + +/** + * Type guard to check if a given object has a subscribe method. + * @param {unknown} object - The object to check. + * @returns {boolean} - True if the object has a subscribe method, otherwise false. + */ +export const hasSubscribe = (object: unknown): object is { subscribe: () => void } => + typeof object === 'object' && + object !== null && + typeof (object as { subscribe?: () => void }).subscribe === 'function' + +/** + * Type guard to check if a given object has a toPromise method. + * @param {unknown} object - The object to check. + * @returns {boolean} - True if the object has a toPromise method, otherwise false. + */ +export const hasToPromise = (object: unknown): object is { toPromise: () => Promise } => + typeof object === 'object' && + object !== null && + typeof (object as { toPromise?: () => Promise }).toPromise === 'function' + +/** + * Converts various types to an Observable, supporting objects that are Observables, have a subscribe or toPromise method. + * @param {ResultOrDeferred} resultOrDeferred - The result or deferred value to convert. + * @returns {Observable} - An Observable that emits the result or deferred value. + */ +export const transformToObservable = (resultOrDeferred: ResultOrDeferred): Observable => { + if (isObservable(resultOrDeferred)) { + return resultOrDeferred as Observable + } + if (hasSubscribe(resultOrDeferred)) { + return new Observable((subscriber) => { + (resultOrDeferred as any).subscribe({ + next: (value: any) => subscriber.next(value as T), + error: (error: any) => subscriber.error(error), + complete: () => subscriber.complete(), + }) + }) + } + if (hasToPromise(resultOrDeferred)) { + return from(lastValueFrom(resultOrDeferred as any)) as Observable + } + return new Observable((subscriber) => { + subscriber.next(resultOrDeferred as T) + subscriber.complete() + }) +} + +/** + * Converts either an AsyncGenerator or an Observable to an AsyncGenerator. + * @param {AsyncGenerator | Observable} input - The input to convert. + * @returns {AsyncGenerator} - An AsyncGenerator that yields values from the input. + * @throws {TypeError} - If the input is not an AsyncGenerator or Observable. + */ +export async function* toAsyncGenerator( + input: AsyncGenerator | Observable +): AsyncGenerator { + if (isObservable(input)) { + yield* observableToAsyncGenerator(input) + } else if (isAsyncGenerator(input)) { + yield* input + } else { + throw new TypeError('Unsupported input type. Expected an Observable or an AsyncGenerator.') + } +} diff --git a/packages/nestjs-connectrpc/src/utils/router.utils.ts b/packages/nestjs-connectrpc/src/utils/router.utils.ts new file mode 100644 index 00000000..2c6dd21e --- /dev/null +++ b/packages/nestjs-connectrpc/src/utils/router.utils.ts @@ -0,0 +1,115 @@ +import type { ServiceType } from '@bufbuild/protobuf' +import type { ConnectRouter } from '@connectrpc/connect' +import type { ServiceImpl } from '@connectrpc/connect' +import type { MessageHandler } from '@nestjs/microservices' +import type { Observable } from 'rxjs' + +import type { ConnectRpcPattern } from '../connectrpc.interfaces.js' +import type { CustomMetadataStore } from '../custom-metadata.storage.js' + +import { lastValueFrom } from 'rxjs' +import { MethodType } from '../connectrpc.interfaces.js' +import { toAsyncGenerator } from './async.utils.js' +import { transformToObservable } from './async.utils.js' + +/** + * Creates a JSON string pattern for a given service, method, and streaming type. + * @param {string} service - The name of the service. + * @param {string} methodName - The name of the method to call. + * @param {MethodType} streaming - The streaming type (NO_STREAMING or RX_STREAMING). + * @returns {string} - The JSON string pattern for the RPC method. + */ +export const createPattern = (service: string, methodName: string, streaming: MethodType): string => + JSON.stringify({ service, rpc: methodName, streaming } as ConnectRpcPattern) + +/** + * Registers services and their handlers to the provided ConnectRouter instance. + * @param {ConnectRouter} router - The ConnectRouter instance to configure. + * @param {Record>>} serviceHandlersMap - + * A map of services and their respective handler implementations. + * @param {CustomMetadataStore} customMetadataStore - The metadata store containing service configurations. + */ +export const addServicesToRouter = ( + router: ConnectRouter, + serviceHandlersMap: Record>>, + customMetadataStore: CustomMetadataStore +): void => { + for (const serviceName of Object.keys(serviceHandlersMap)) { + const service = customMetadataStore.get(serviceName) + if (service) { + router.service(service, serviceHandlersMap[serviceName]) + } + } +} + +/** + * Generates a map of service handlers with support for synchronous and asynchronous streaming. + * @param {Map} handlers - The map of message patterns to their respective handlers. + * @param {CustomMetadataStore} customMetadataStore - The metadata store with service configurations. + * @returns {Record>>} - A map of service names to their handlers. + */ +export const createServiceHandlersMap = ( + handlers: Map, + customMetadataStore: CustomMetadataStore +): Record>> => { + const serviceHandlersMap: Record>> = {} + + handlers.forEach((handlerMetadata, pattern) => { + const parsedPattern = JSON.parse(pattern) as ConnectRpcPattern + const { service, rpc, streaming } = parsedPattern + const serviceMetadata = customMetadataStore.get(service) + + if (serviceMetadata) { + const methodProto = serviceMetadata.methods[rpc] + if (methodProto) { + serviceHandlersMap[service] ??= {} + + switch (streaming) { + case MethodType.NO_STREAMING: + serviceHandlersMap[service][rpc] = async (request: unknown, context: unknown) => { + const resultOrDeferred = await handlerMetadata(request, context) + return lastValueFrom(transformToObservable(resultOrDeferred)) + } + break + + case MethodType.RX_STREAMING: + serviceHandlersMap[service][rpc] = async function* (request: unknown, context: unknown) { + const streamOrValue = await handlerMetadata(request, context) + yield* toAsyncGenerator(streamOrValue as Observable | AsyncGenerator) + } + break + + default: + throw new Error(`Unsupported streaming type: ${streaming}`) + } + } + } + }) + + return serviceHandlersMap +} + +/** + * Creates metadata for a Connect RPC method, setting defaults when necessary. + * @param {object} target - The target object (usually a service class) for metadata association. + * @param {string | symbol} key - The name of the method or property. + * @param {string} [service] - Optional service name, defaults to the class name of the target. + * @param {string} [method] - Optional method name; defaults to the capitalized key name. + * @param {MethodType} [streaming=MethodType.NO_STREAMING] - The streaming type, defaulting to NO_STREAMING. + * @returns {{ service: string; rpc: string; streaming: MethodType }} - Metadata with service, RPC method, and streaming type. + */ +export const createConnectRpcMethodMetadata = ( + target: object, + key: string | symbol, + service?: string, + method?: string, + streaming: MethodType = MethodType.NO_STREAMING +): { service: string; rpc: string; streaming: MethodType } => { + const capitalizeFirstLetter = (input: string): string => + input.charAt(0).toUpperCase() + input.slice(1) + + const serviceName = service ?? target.constructor.name + const rpcMethodName = method ?? capitalizeFirstLetter(String(key)) + + return { service: serviceName, rpc: rpcMethodName, streaming } +} From 80fab754fe411b55035fbccaefdb62dc49753cef Mon Sep 17 00:00:00 2001 From: Andrew Ghostuhin Date: Wed, 30 Oct 2024 21:14:13 +0300 Subject: [PATCH 03/10] refactor(nestjs-connectrpc): decorators --- .../src/connectrpc.decorators.ts | 83 +++++++++++-------- .../src/utils/async.utils.ts | 11 ++- .../src/utils/router.utils.ts | 30 ++++--- 3 files changed, 74 insertions(+), 50 deletions(-) diff --git a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts index 765ae868..08150d4b 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts @@ -1,34 +1,51 @@ -import type { ServiceType } from '@bufbuild/protobuf' +import type { ServiceType } from '@bufbuild/protobuf' -import type { ConstructorWithPrototype } from './connectrpc.interfaces.js' -import type { FunctionPropertyDescriptor } from './connectrpc.interfaces.js' -import type { MethodKey } from './connectrpc.interfaces.js' -import type { MethodKeys } from './connectrpc.interfaces.js' +import type { ConstructorWithPrototype } from './connectrpc.interfaces.js' -import { MessagePattern } from '@nestjs/microservices' +import type { FunctionPropertyDescriptor } from './connectrpc.interfaces.js' -import { CONNECTRPC_TRANSPORT } from './connectrpc.constants.js' -import { METHOD_DECORATOR_KEY } from './connectrpc.constants.js' -import { STREAM_METHOD_DECORATOR_KEY } from './connectrpc.constants.js' -import { MethodType } from './connectrpc.interfaces.js' -import { CustomMetadataStore } from './custom-metadata.storage.js' -import { createConnectRpcMethodMetadata } from './utils/router.utils.js' +import type { MethodKey } from './connectrpc.interfaces.js' +import type { MethodKeys } from './connectrpc.interfaces.js' + +import { MessagePattern } from '@nestjs/microservices' + +import { CONNECTRPC_TRANSPORT } from './connectrpc.constants.js' +import { METHOD_DECORATOR_KEY } from './connectrpc.constants.js' +import { STREAM_METHOD_DECORATOR_KEY } from './connectrpc.constants.js' +import { MethodType } from './connectrpc.interfaces.js' +import { CustomMetadataStore } from './custom-metadata.storage.js' +import { createConnectRpcMethodMetadata } from './utils/router.utils.js' + +/** + * Type guard to check if a given descriptor is a function property descriptor. + * @param {PropertyDescriptor | undefined} descriptor - The descriptor to check. + * @returns {descriptor is FunctionPropertyDescriptor} - True if descriptor is for a function. + */ function isFunctionPropertyDescriptor( descriptor: PropertyDescriptor | undefined ): descriptor is FunctionPropertyDescriptor { return descriptor !== undefined && typeof descriptor.value === 'function' } +/** + * ConnectRpcService decorator to register RPC services and their handlers. + * @param {ServiceType} serviceName - The service type from protobuf. + * @returns {ClassDecorator} - Class decorator function. + */ export const ConnectRpcService = (serviceName: ServiceType): ClassDecorator => (target: ConstructorWithPrototype): void => { - const processMethodKey = (methodImpl: MethodKey): void => { - const functionName = methodImpl.key - const { methodType } = methodImpl + // Получаем все зарегистрированные методы и объединяем их + const unaryMethodKeys: MethodKeys = Reflect.getMetadata(METHOD_DECORATOR_KEY, target) || [] + const streamMethodKeys: MethodKeys = + Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target) || [] + const allMethodKeys = [...unaryMethodKeys, ...streamMethodKeys] as MethodKey[] + allMethodKeys.forEach((methodImpl) => { + const { key: functionName, methodType } = methodImpl const descriptor = Object.getOwnPropertyDescriptor(target.prototype, functionName) - if (isFunctionPropertyDescriptor(descriptor)) { + if (descriptor && isFunctionPropertyDescriptor(descriptor)) { const metadata = createConnectRpcMethodMetadata( descriptor.value, functionName, @@ -37,33 +54,25 @@ export const ConnectRpcService = (serviceName: ServiceType): ClassDecorator => methodType ) - const customMetadataStore = CustomMetadataStore.getInstance() - customMetadataStore.set(serviceName.typeName, serviceName) - + CustomMetadataStore.getInstance().set(serviceName.typeName, serviceName) MessagePattern(metadata, CONNECTRPC_TRANSPORT)(target.prototype, functionName, descriptor) } - } - - const unaryMethodKeys: MethodKeys = Reflect.getMetadata(METHOD_DECORATOR_KEY, target) || [] - const streamMethodKeys: MethodKeys = - Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target) || [] - - unaryMethodKeys.forEach((methodImpl) => { - processMethodKey(methodImpl) - }) - streamMethodKeys.forEach((methodImpl) => { - processMethodKey(methodImpl) }) } +/** + * Decorator for unary RPC methods. + * Registers the method as a unary RPC with no streaming. + * @returns {MethodDecorator} - Method decorator function. + */ export const ConnectRpcMethod = (): MethodDecorator => (target: object, key: string | symbol) => { const metadata: MethodKey = { key: key.toString(), methodType: MethodType.NO_STREAMING, } - const existingMethods: Set = - Reflect.getMetadata(METHOD_DECORATOR_KEY, target.constructor) || new Set() + const existingMethods = + (Reflect.getMetadata(METHOD_DECORATOR_KEY, target.constructor) as Set) || new Set() if (!existingMethods.has(metadata)) { existingMethods.add(metadata) @@ -71,6 +80,11 @@ export const ConnectRpcMethod = (): MethodDecorator => (target: object, key: str } } +/** + * Decorator for streaming RPC methods. + * Registers the method as a streaming RPC with RX_STREAMING type. + * @returns {MethodDecorator} - Method decorator function. + */ export const ConnectRpcStreamMethod = (): MethodDecorator => (target: object, key: string | symbol) => { const metadata: MethodKey = { @@ -78,8 +92,9 @@ export const ConnectRpcStreamMethod = (): MethodDecorator => methodType: MethodType.RX_STREAMING, } - const existingMethods: Set = - Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target.constructor) || new Set() + const existingMethods = + (Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target.constructor) as Set) || + new Set() if (!existingMethods.has(metadata)) { existingMethods.add(metadata) diff --git a/packages/nestjs-connectrpc/src/utils/async.utils.ts b/packages/nestjs-connectrpc/src/utils/async.utils.ts index 49a49154..0b2b614d 100644 --- a/packages/nestjs-connectrpc/src/utils/async.utils.ts +++ b/packages/nestjs-connectrpc/src/utils/async.utils.ts @@ -1,6 +1,9 @@ -import type { ResultOrDeferred } from '../connectrpc.interfaces.js' -import { Observable, from, Subject } from 'rxjs' -import { lastValueFrom } from 'rxjs' +import type { ResultOrDeferred } from '../connectrpc.interfaces.js' + +import { Observable } from 'rxjs' +import { from } from 'rxjs' +import { Subject } from 'rxjs' +import { lastValueFrom } from 'rxjs' /** * Type guard to check if a given input is an AsyncGenerator. @@ -100,7 +103,7 @@ export const transformToObservable = (resultOrDeferred: ResultOrDeferred): } if (hasSubscribe(resultOrDeferred)) { return new Observable((subscriber) => { - (resultOrDeferred as any).subscribe({ + ;(resultOrDeferred as any).subscribe({ next: (value: any) => subscriber.next(value as T), error: (error: any) => subscriber.error(error), complete: () => subscriber.complete(), diff --git a/packages/nestjs-connectrpc/src/utils/router.utils.ts b/packages/nestjs-connectrpc/src/utils/router.utils.ts index 2c6dd21e..9652f026 100644 --- a/packages/nestjs-connectrpc/src/utils/router.utils.ts +++ b/packages/nestjs-connectrpc/src/utils/router.utils.ts @@ -1,16 +1,17 @@ -import type { ServiceType } from '@bufbuild/protobuf' -import type { ConnectRouter } from '@connectrpc/connect' -import type { ServiceImpl } from '@connectrpc/connect' -import type { MessageHandler } from '@nestjs/microservices' -import type { Observable } from 'rxjs' +import type { ServiceType } from '@bufbuild/protobuf' +import type { ConnectRouter } from '@connectrpc/connect' +import type { ServiceImpl } from '@connectrpc/connect' +import type { MessageHandler } from '@nestjs/microservices' +import type { Observable } from 'rxjs' -import type { ConnectRpcPattern } from '../connectrpc.interfaces.js' +import type { ConnectRpcPattern } from '../connectrpc.interfaces.js' import type { CustomMetadataStore } from '../custom-metadata.storage.js' -import { lastValueFrom } from 'rxjs' -import { MethodType } from '../connectrpc.interfaces.js' -import { toAsyncGenerator } from './async.utils.js' -import { transformToObservable } from './async.utils.js' +import { lastValueFrom } from 'rxjs' + +import { MethodType } from '../connectrpc.interfaces.js' +import { toAsyncGenerator } from './async.utils.js' +import { transformToObservable } from './async.utils.js' /** * Creates a JSON string pattern for a given service, method, and streaming type. @@ -73,9 +74,14 @@ export const createServiceHandlersMap = ( break case MethodType.RX_STREAMING: - serviceHandlersMap[service][rpc] = async function* (request: unknown, context: unknown) { + serviceHandlersMap[service][rpc] = async function* ( + request: unknown, + context: unknown + ) { const streamOrValue = await handlerMetadata(request, context) - yield* toAsyncGenerator(streamOrValue as Observable | AsyncGenerator) + yield* toAsyncGenerator( + streamOrValue as Observable | AsyncGenerator + ) } break From 8feabe5322881fd3af1daba7d1d9efb02c1f5aca Mon Sep 17 00:00:00 2001 From: OsirisAnubis Date: Wed, 30 Oct 2024 22:59:49 +0300 Subject: [PATCH 04/10] style: fix lint --- .../src/connectrpc.decorators.ts | 29 +++--- .../src/utils/async.utils.ts | 93 +++++++++++-------- .../src/utils/router.utils.ts | 15 +-- 3 files changed, 75 insertions(+), 62 deletions(-) diff --git a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts index 08150d4b..f6be297e 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts @@ -1,21 +1,18 @@ -import type { ServiceType } from '@bufbuild/protobuf' +import type { ServiceType } from '@bufbuild/protobuf' -import type { ConstructorWithPrototype } from './connectrpc.interfaces.js' +import type { ConstructorWithPrototype } from './connectrpc.interfaces.js' +import type { FunctionPropertyDescriptor } from './connectrpc.interfaces.js' +import type { MethodKey } from './connectrpc.interfaces.js' +import type { MethodKeys } from './connectrpc.interfaces.js' -import type { FunctionPropertyDescriptor } from './connectrpc.interfaces.js' +import { MessagePattern } from '@nestjs/microservices' -import type { MethodKey } from './connectrpc.interfaces.js' - -import type { MethodKeys } from './connectrpc.interfaces.js' - -import { MessagePattern } from '@nestjs/microservices' - -import { CONNECTRPC_TRANSPORT } from './connectrpc.constants.js' -import { METHOD_DECORATOR_KEY } from './connectrpc.constants.js' -import { STREAM_METHOD_DECORATOR_KEY } from './connectrpc.constants.js' -import { MethodType } from './connectrpc.interfaces.js' -import { CustomMetadataStore } from './custom-metadata.storage.js' -import { createConnectRpcMethodMetadata } from './utils/router.utils.js' +import { CONNECTRPC_TRANSPORT } from './connectrpc.constants.js' +import { METHOD_DECORATOR_KEY } from './connectrpc.constants.js' +import { STREAM_METHOD_DECORATOR_KEY } from './connectrpc.constants.js' +import { MethodType } from './connectrpc.interfaces.js' +import { CustomMetadataStore } from './custom-metadata.storage.js' +import { createConnectRpcMethodMetadata } from './utils/router.utils.js' /** * Type guard to check if a given descriptor is a function property descriptor. @@ -39,7 +36,7 @@ export const ConnectRpcService = (serviceName: ServiceType): ClassDecorator => const unaryMethodKeys: MethodKeys = Reflect.getMetadata(METHOD_DECORATOR_KEY, target) || [] const streamMethodKeys: MethodKeys = Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target) || [] - const allMethodKeys = [...unaryMethodKeys, ...streamMethodKeys] as MethodKey[] + const allMethodKeys = [...unaryMethodKeys, ...streamMethodKeys] as Array allMethodKeys.forEach((methodImpl) => { const { key: functionName, methodType } = methodImpl diff --git a/packages/nestjs-connectrpc/src/utils/async.utils.ts b/packages/nestjs-connectrpc/src/utils/async.utils.ts index 0b2b614d..b20a560f 100644 --- a/packages/nestjs-connectrpc/src/utils/async.utils.ts +++ b/packages/nestjs-connectrpc/src/utils/async.utils.ts @@ -1,9 +1,9 @@ -import type { ResultOrDeferred } from '../connectrpc.interfaces.js' +import type { ResultOrDeferred } from '../connectrpc.interfaces.js' -import { Observable } from 'rxjs' -import { from } from 'rxjs' -import { Subject } from 'rxjs' -import { lastValueFrom } from 'rxjs' +import { Observable } from 'rxjs' +import { Subject } from 'rxjs' +import { from } from 'rxjs' +import { lastValueFrom } from 'rxjs' /** * Type guard to check if a given input is an AsyncGenerator. @@ -14,6 +14,35 @@ export function isAsyncGenerator(input: unknown): input is AsyncGenerator return typeof input === 'object' && input !== null && Symbol.asyncIterator in input } +/** + * Utility function to create an async iterator for a Subject. + * @param {Subject} subject - The Subject to create an iterator from. + * @returns {AsyncIterableIterator} - The async iterator. + */ +async function* asyncIterator(subject: Subject): AsyncIterableIterator { + const nextValue = async (): Promise => + new Promise((resolve, reject) => { + subject.subscribe({ + next: (val) => { + resolve(val) + }, + error: (err) => { + reject(err) + }, + complete: () => { + resolve(null as PromiseLike | T) + }, + }) + }) + + while (true) { + // eslint-disable-next-line no-await-in-loop + const item = await nextValue() + if (item === null) return + yield item + } +} + /** * Converts an Observable to an AsyncGenerator, yielding items emitted by the Observable. * @param {Observable} observable - The Observable to convert. @@ -21,13 +50,15 @@ export function isAsyncGenerator(input: unknown): input is AsyncGenerator */ export async function* observableToAsyncGenerator(observable: Observable): AsyncGenerator { const queue = new Subject() - let didComplete = false const subscriber = observable.subscribe({ - next: (value) => queue.next(value), - error: (error) => queue.error(error), + next: (value) => { + queue.next(value) + }, + error: (error) => { + queue.error(error) + }, complete: () => { - didComplete = true queue.complete() }, }) @@ -41,29 +72,6 @@ export async function* observableToAsyncGenerator(observable: Observable): } } -/** - * Utility function to create an async iterator for a Subject. - * @param {Subject} subject - The Subject to create an iterator from. - * @returns {AsyncIterableIterator} - The async iterator. - */ -async function* asyncIterator(subject: Subject): AsyncIterableIterator { - const values: T[] = [] - const nextValue = () => - new Promise((resolve, reject) => { - subject.subscribe({ - next: (val) => resolve(val), - error: (err) => reject(err), - complete: () => resolve(null as any), - }) - }) - - while (true) { - const item = await nextValue() - if (item === null) return - yield item - } -} - /** * Type guard to check if a given object is an Observable. * @param {unknown} object - The object to check. @@ -103,18 +111,25 @@ export const transformToObservable = (resultOrDeferred: ResultOrDeferred): } if (hasSubscribe(resultOrDeferred)) { return new Observable((subscriber) => { - ;(resultOrDeferred as any).subscribe({ - next: (value: any) => subscriber.next(value as T), - error: (error: any) => subscriber.error(error), - complete: () => subscriber.complete(), + // @ts-expect-error + resultOrDeferred.subscribe({ + next: (value: any) => { + subscriber.next(value as T) + }, + error: (error: any) => { + subscriber.error(error) + }, + complete: () => { + subscriber.complete() + }, }) }) } if (hasToPromise(resultOrDeferred)) { - return from(lastValueFrom(resultOrDeferred as any)) as Observable + return from(lastValueFrom(resultOrDeferred as Observable)) } - return new Observable((subscriber) => { - subscriber.next(resultOrDeferred as T) + return new Observable((subscriber) => { + subscriber.next(resultOrDeferred) subscriber.complete() }) } diff --git a/packages/nestjs-connectrpc/src/utils/router.utils.ts b/packages/nestjs-connectrpc/src/utils/router.utils.ts index 9652f026..a0fea720 100644 --- a/packages/nestjs-connectrpc/src/utils/router.utils.ts +++ b/packages/nestjs-connectrpc/src/utils/router.utils.ts @@ -67,26 +67,27 @@ export const createServiceHandlersMap = ( switch (streaming) { case MethodType.NO_STREAMING: - serviceHandlersMap[service][rpc] = async (request: unknown, context: unknown) => { + serviceHandlersMap[service][rpc] = async ( + request: unknown, + context: unknown + ): Promise => { const resultOrDeferred = await handlerMetadata(request, context) return lastValueFrom(transformToObservable(resultOrDeferred)) } break case MethodType.RX_STREAMING: - serviceHandlersMap[service][rpc] = async function* ( + serviceHandlersMap[service][rpc] = async function* handleStream( request: unknown, context: unknown - ) { + ): AsyncGenerator { const streamOrValue = await handlerMetadata(request, context) - yield* toAsyncGenerator( - streamOrValue as Observable | AsyncGenerator - ) + yield* toAsyncGenerator(streamOrValue as AsyncGenerator | Observable) } break default: - throw new Error(`Unsupported streaming type: ${streaming}`) + throw new Error(`Unsupported streaming type: ${streaming as string}`) } } } From e30cb290f04dc165e1b994b1985e89c0772bec12 Mon Sep 17 00:00:00 2001 From: OsirisAnubis Date: Thu, 31 Oct 2024 23:33:34 +0300 Subject: [PATCH 05/10] refactor: reduce nesting to improve code readability --- .../src/connectrpc.decorators.ts | 40 ++++++------- .../src/connectrpc.strategy.ts | 13 ++--- .../src/utils/router.utils.ts | 58 +++++++++---------- 3 files changed, 54 insertions(+), 57 deletions(-) diff --git a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts index f6be297e..0a96f8ff 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts @@ -42,18 +42,18 @@ export const ConnectRpcService = (serviceName: ServiceType): ClassDecorator => const { key: functionName, methodType } = methodImpl const descriptor = Object.getOwnPropertyDescriptor(target.prototype, functionName) - if (descriptor && isFunctionPropertyDescriptor(descriptor)) { - const metadata = createConnectRpcMethodMetadata( - descriptor.value, - functionName, - serviceName.typeName, - functionName, - methodType - ) - - CustomMetadataStore.getInstance().set(serviceName.typeName, serviceName) - MessagePattern(metadata, CONNECTRPC_TRANSPORT)(target.prototype, functionName, descriptor) - } + if (!descriptor || !isFunctionPropertyDescriptor(descriptor)) return + + const metadata = createConnectRpcMethodMetadata( + descriptor.value, + functionName, + serviceName.typeName, + functionName, + methodType + ) + + CustomMetadataStore.getInstance().set(serviceName.typeName, serviceName) + MessagePattern(metadata, CONNECTRPC_TRANSPORT)(target.prototype, functionName, descriptor) }) } @@ -71,10 +71,10 @@ export const ConnectRpcMethod = (): MethodDecorator => (target: object, key: str const existingMethods = (Reflect.getMetadata(METHOD_DECORATOR_KEY, target.constructor) as Set) || new Set() - if (!existingMethods.has(metadata)) { - existingMethods.add(metadata) - Reflect.defineMetadata(METHOD_DECORATOR_KEY, existingMethods, target.constructor) - } + if (existingMethods.has(metadata)) return + + existingMethods.add(metadata) + Reflect.defineMetadata(METHOD_DECORATOR_KEY, existingMethods, target.constructor) } /** @@ -93,8 +93,8 @@ export const ConnectRpcStreamMethod = (): MethodDecorator => (Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target.constructor) as Set) || new Set() - if (!existingMethods.has(metadata)) { - existingMethods.add(metadata) - Reflect.defineMetadata(STREAM_METHOD_DECORATOR_KEY, existingMethods, target.constructor) - } + if (existingMethods.has(metadata)) return + + existingMethods.add(metadata) + Reflect.defineMetadata(STREAM_METHOD_DECORATOR_KEY, existingMethods, target.constructor) } diff --git a/packages/nestjs-connectrpc/src/connectrpc.strategy.ts b/packages/nestjs-connectrpc/src/connectrpc.strategy.ts index 037cd28a..bd16d5b9 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.strategy.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.strategy.ts @@ -60,13 +60,12 @@ export class ConnectRpcServer extends Server implements CustomTransportStrategy buildRouter() { return (router: ConnectRouter): void => { - if (this.customMetadataStore) { - const serviceHandlersMap = createServiceHandlersMap( - this.getHandlers(), - this.customMetadataStore - ) - addServicesToRouter(router, serviceHandlersMap, this.customMetadataStore) - } + if (!this.customMetadataStore) return + const serviceHandlersMap = createServiceHandlersMap( + this.getHandlers(), + this.customMetadataStore + ) + addServicesToRouter(router, serviceHandlersMap, this.customMetadataStore) } } } diff --git a/packages/nestjs-connectrpc/src/utils/router.utils.ts b/packages/nestjs-connectrpc/src/utils/router.utils.ts index a0fea720..03e4373b 100644 --- a/packages/nestjs-connectrpc/src/utils/router.utils.ts +++ b/packages/nestjs-connectrpc/src/utils/router.utils.ts @@ -37,9 +37,9 @@ export const addServicesToRouter = ( ): void => { for (const serviceName of Object.keys(serviceHandlersMap)) { const service = customMetadataStore.get(serviceName) - if (service) { - router.service(service, serviceHandlersMap[serviceName]) - } + // eslint-disable-next-line no-continue + if (!service) continue + router.service(service, serviceHandlersMap[serviceName]) } } @@ -60,36 +60,34 @@ export const createServiceHandlersMap = ( const { service, rpc, streaming } = parsedPattern const serviceMetadata = customMetadataStore.get(service) - if (serviceMetadata) { - const methodProto = serviceMetadata.methods[rpc] - if (methodProto) { - serviceHandlersMap[service] ??= {} - - switch (streaming) { - case MethodType.NO_STREAMING: - serviceHandlersMap[service][rpc] = async ( - request: unknown, - context: unknown - ): Promise => { - const resultOrDeferred = await handlerMetadata(request, context) - return lastValueFrom(transformToObservable(resultOrDeferred)) - } - break + if (!serviceMetadata) return + const methodProto = serviceMetadata.methods[rpc] + if (!methodProto) return + serviceHandlersMap[service] ??= {} - case MethodType.RX_STREAMING: - serviceHandlersMap[service][rpc] = async function* handleStream( - request: unknown, - context: unknown - ): AsyncGenerator { - const streamOrValue = await handlerMetadata(request, context) - yield* toAsyncGenerator(streamOrValue as AsyncGenerator | Observable) - } - break + switch (streaming) { + case MethodType.NO_STREAMING: + serviceHandlersMap[service][rpc] = async ( + request: unknown, + context: unknown + ): Promise => { + const resultOrDeferred = await handlerMetadata(request, context) + return lastValueFrom(transformToObservable(resultOrDeferred)) + } + break - default: - throw new Error(`Unsupported streaming type: ${streaming as string}`) + case MethodType.RX_STREAMING: + serviceHandlersMap[service][rpc] = async function* handleStream( + request: unknown, + context: unknown + ): AsyncGenerator { + const streamOrValue = await handlerMetadata(request, context) + yield* toAsyncGenerator(streamOrValue as AsyncGenerator | Observable) } - } + break + + default: + throw new Error(`Unsupported streaming type: ${streaming as string}`) } }) From 6abcbac5fd92eee81ee128ea93b293cca3d082a2 Mon Sep 17 00:00:00 2001 From: OsirisAnubis Date: Thu, 31 Oct 2024 23:48:34 +0300 Subject: [PATCH 06/10] refactor: add utility function AddMethodMetadata --- .../src/connectrpc.decorators.ts | 41 +++++++++---------- 1 file changed, 19 insertions(+), 22 deletions(-) diff --git a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts index 0a96f8ff..6a189953 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.decorators.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.decorators.ts @@ -57,24 +57,33 @@ export const ConnectRpcService = (serviceName: ServiceType): ClassDecorator => }) } -/** - * Decorator for unary RPC methods. - * Registers the method as a unary RPC with no streaming. - * @returns {MethodDecorator} - Method decorator function. - */ -export const ConnectRpcMethod = (): MethodDecorator => (target: object, key: string | symbol) => { +export const AddMethodMetadata = ( + target: object, + key: string | symbol, + methodType: MethodType, + metadataKey: string | symbol +): void => { const metadata: MethodKey = { key: key.toString(), - methodType: MethodType.NO_STREAMING, + methodType, } const existingMethods = - (Reflect.getMetadata(METHOD_DECORATOR_KEY, target.constructor) as Set) || new Set() + (Reflect.getMetadata(metadataKey, target.constructor) as Set) || new Set() if (existingMethods.has(metadata)) return existingMethods.add(metadata) - Reflect.defineMetadata(METHOD_DECORATOR_KEY, existingMethods, target.constructor) + Reflect.defineMetadata(metadataKey, existingMethods, target.constructor) +} + +/** + * Decorator for unary RPC methods. + * Registers the method as a unary RPC with no streaming. + * @returns {MethodDecorator} - Method decorator function. + */ +export const ConnectRpcMethod = (): MethodDecorator => (target: object, key: string | symbol) => { + AddMethodMetadata(target, key, MethodType.NO_STREAMING, METHOD_DECORATOR_KEY) } /** @@ -84,17 +93,5 @@ export const ConnectRpcMethod = (): MethodDecorator => (target: object, key: str */ export const ConnectRpcStreamMethod = (): MethodDecorator => (target: object, key: string | symbol) => { - const metadata: MethodKey = { - key: key.toString(), - methodType: MethodType.RX_STREAMING, - } - - const existingMethods = - (Reflect.getMetadata(STREAM_METHOD_DECORATOR_KEY, target.constructor) as Set) || - new Set() - - if (existingMethods.has(metadata)) return - - existingMethods.add(metadata) - Reflect.defineMetadata(STREAM_METHOD_DECORATOR_KEY, existingMethods, target.constructor) + AddMethodMetadata(target, key, MethodType.RX_STREAMING, STREAM_METHOD_DECORATOR_KEY) } From cab1e74bf7f367d8fc91404122aa7c6f20b6b04c Mon Sep 17 00:00:00 2001 From: OsirisAnubis Date: Thu, 31 Oct 2024 23:57:09 +0300 Subject: [PATCH 07/10] refactor: method startServer return promise --- .../src/connectrpc.server.ts | 19 ++++++------------- 1 file changed, 6 insertions(+), 13 deletions(-) diff --git a/packages/nestjs-connectrpc/src/connectrpc.server.ts b/packages/nestjs-connectrpc/src/connectrpc.server.ts index ea193441..c119d08a 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.server.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.server.ts @@ -31,9 +31,7 @@ export class HTTPServer { } async listen(): Promise { - return new Promise((resolve, reject) => { - this.startServer(resolve, reject) - }) + await this.startServer() } createHttpServer(): http.Server { @@ -75,8 +73,8 @@ export class HTTPServer { ) } - startServer(resolve: () => void, reject: (error: Error) => void): void { - try { + async startServer(): Promise { + return new Promise((resolve, reject) => { switch (this.options.protocol) { case ServerProtocol.HTTP: { this.server = this.createHttpServer() @@ -95,7 +93,8 @@ export class HTTPServer { break } default: { - throw new Error('Invalid protocol option') + reject(new Error('Invalid protocol option')) + return } } @@ -103,13 +102,7 @@ export class HTTPServer { if (this.options.callback) this.options.callback() resolve() }) - } catch (error) { - if (error instanceof Error) { - reject(error) - } else { - reject(new Error('Unknown error occurred')) - } - } + }) } async close(callback?: () => void): Promise { From db17755d1baf832e93815cc45dd18f974439c2c0 Mon Sep 17 00:00:00 2001 From: OsirisAnubis Date: Fri, 1 Nov 2024 00:04:30 +0300 Subject: [PATCH 08/10] perf: added a return to avoid calling the set twice --- packages/nestjs-connectrpc/src/connectrpc.strategy.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/nestjs-connectrpc/src/connectrpc.strategy.ts b/packages/nestjs-connectrpc/src/connectrpc.strategy.ts index bd16d5b9..a0c50944 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.strategy.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.strategy.ts @@ -54,6 +54,7 @@ export class ConnectRpcServer extends Server implements CustomTransportStrategy const modifiedCallback = callback modifiedCallback.isEventHandler = true this.messageHandlers.set(route, modifiedCallback) + return } this.messageHandlers.set(route, callback) } From 92da2f459ac66272295de6f8c3a98ef94a978fe2 Mon Sep 17 00:00:00 2001 From: OsirisAnubis Date: Tue, 5 Nov 2024 10:34:35 +0300 Subject: [PATCH 09/10] style: remove async from startServer method --- packages/nestjs-connectrpc/src/connectrpc.server.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/packages/nestjs-connectrpc/src/connectrpc.server.ts b/packages/nestjs-connectrpc/src/connectrpc.server.ts index c119d08a..c6e8e7c5 100644 --- a/packages/nestjs-connectrpc/src/connectrpc.server.ts +++ b/packages/nestjs-connectrpc/src/connectrpc.server.ts @@ -73,7 +73,8 @@ export class HTTPServer { ) } - async startServer(): Promise { + // eslint-disable-next-line @typescript-eslint/promise-function-async + startServer(): Promise { return new Promise((resolve, reject) => { switch (this.options.protocol) { case ServerProtocol.HTTP: { From e76b8ddf50a98e436330f8219b350ab3a63288ed Mon Sep 17 00:00:00 2001 From: OsirisAnubis Date: Tue, 5 Nov 2024 10:45:34 +0300 Subject: [PATCH 10/10] chore: dependency versions are fixed --- .pnp.cjs | 158 +++++++++++++--------- packages/nestjs-connectrpc/package.json | 18 +-- yarn.lock | 170 +++++++++++++++++------- 3 files changed, 222 insertions(+), 124 deletions(-) diff --git a/.pnp.cjs b/.pnp.cjs index 76162882..710763df 100755 --- a/.pnp.cjs +++ b/.pnp.cjs @@ -561,10 +561,10 @@ const RAW_RUNTIME_STATE = ["@bufbuild/protobuf", "npm:1.10.0"],\ ["@connectrpc/connect", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1"],\ ["@connectrpc/connect-node", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:1.6.1"],\ - ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ - ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ - ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ - ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ + ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ + ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.2.4"],\ + ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.2.4"],\ ["reflect-metadata", "npm:0.2.2"],\ ["rxjs", "npm:7.8.1"]\ ],\ @@ -7266,10 +7266,10 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ - ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3", {\ - "packageLocation": "./.yarn/__virtual__/@nestjs-common-virtual-2765291d0b/2/.yarn/berry/cache/@nestjs-common-npm-10.4.3-c8baed1848-10c0.zip/node_modules/@nestjs/common/",\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5", {\ + "packageLocation": "./.yarn/__virtual__/@nestjs-common-virtual-80e1232acf/2/.yarn/berry/cache/@nestjs-common-npm-10.0.5-0a7f781d1a-10c0.zip/node_modules/@nestjs/common/",\ "packageDependencies": [\ - ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ ["@types/class-transformer", null],\ ["@types/class-validator", null],\ ["@types/reflect-metadata", null],\ @@ -7279,7 +7279,7 @@ const RAW_RUNTIME_STATE = ["iterare", "npm:1.2.1"],\ ["reflect-metadata", "npm:0.2.2"],\ ["rxjs", "npm:7.8.1"],\ - ["tslib", "npm:2.7.0"],\ + ["tslib", "npm:2.6.0"],\ ["uid", "npm:2.0.2"]\ ],\ "packagePeers": [\ @@ -7381,7 +7381,7 @@ const RAW_RUNTIME_STATE = ]],\ ["@nestjs/core", [\ ["npm:10.0.5", {\ - "packageLocation": "./.yarn/unplugged/@nestjs-core-virtual-684e49535a/node_modules/@nestjs/core/",\ + "packageLocation": "./.yarn/unplugged/@nestjs-core-virtual-4ff5a55b98/node_modules/@nestjs/core/",\ "packageDependencies": [\ ["@nestjs/core", "npm:10.0.5"]\ ],\ @@ -7766,13 +7766,13 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ - ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3", {\ - "packageLocation": "./.yarn/unplugged/@nestjs-core-virtual-e9a099626b/node_modules/@nestjs/core/",\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5", {\ + "packageLocation": "./.yarn/unplugged/@nestjs-core-virtual-4ff5a55b98/node_modules/@nestjs/core/",\ "packageDependencies": [\ - ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ - ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ - ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ - ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ + ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.2.4"],\ + ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.2.4"],\ ["@nestjs/websockets", null],\ ["@nuxtjs/opencollective", "npm:0.3.2"],\ ["@types/nestjs__common", null],\ @@ -7783,10 +7783,10 @@ const RAW_RUNTIME_STATE = ["@types/rxjs", null],\ ["fast-safe-stringify", "npm:2.1.1"],\ ["iterare", "npm:1.2.1"],\ - ["path-to-regexp", "npm:3.3.0"],\ + ["path-to-regexp", "npm:3.2.0"],\ ["reflect-metadata", "npm:0.2.2"],\ ["rxjs", "npm:7.8.1"],\ - ["tslib", "npm:2.7.0"],\ + ["tslib", "npm:2.6.0"],\ ["uid", "npm:2.0.2"]\ ],\ "packagePeers": [\ @@ -8229,6 +8229,13 @@ const RAW_RUNTIME_STATE = }]\ ]],\ ["@nestjs/microservices", [\ + ["npm:10.2.4", {\ + "packageLocation": "../.yarn/berry/cache/@nestjs-microservices-npm-10.2.4-36046d0a39-10c0.zip/node_modules/@nestjs/microservices/",\ + "packageDependencies": [\ + ["@nestjs/microservices", "npm:10.2.4"]\ + ],\ + "linkType": "SOFT"\ + }],\ ["npm:10.4.1", {\ "packageLocation": "../.yarn/berry/cache/@nestjs-microservices-npm-10.4.1-709407ada4-10c0.zip/node_modules/@nestjs/microservices/",\ "packageDependencies": [\ @@ -8621,13 +8628,13 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ - ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6", {\ - "packageLocation": "./.yarn/__virtual__/@nestjs-microservices-virtual-a1a759d5fb/2/.yarn/berry/cache/@nestjs-microservices-npm-10.4.6-87a14e792d-10c0.zip/node_modules/@nestjs/microservices/",\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.2.4", {\ + "packageLocation": "./.yarn/__virtual__/@nestjs-microservices-virtual-40bd19dfd0/2/.yarn/berry/cache/@nestjs-microservices-npm-10.2.4-36046d0a39-10c0.zip/node_modules/@nestjs/microservices/",\ "packageDependencies": [\ - ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ + ["@nestjs/microservices", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.2.4"],\ ["@grpc/grpc-js", null],\ - ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ - ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ + ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ ["@nestjs/websockets", null],\ ["@types/amqp-connection-manager", null],\ ["@types/amqplib", null],\ @@ -8652,7 +8659,7 @@ const RAW_RUNTIME_STATE = ["nats", null],\ ["reflect-metadata", "npm:0.2.2"],\ ["rxjs", "npm:7.8.1"],\ - ["tslib", "npm:2.7.0"]\ + ["tslib", "npm:2.6.2"]\ ],\ "packagePeers": [\ "@grpc/grpc-js",\ @@ -8938,17 +8945,17 @@ const RAW_RUNTIME_STATE = }]\ ]],\ ["@nestjs/platform-express", [\ - ["npm:10.4.1", {\ - "packageLocation": "../.yarn/berry/cache/@nestjs-platform-express-npm-10.4.1-76944971fd-10c0.zip/node_modules/@nestjs/platform-express/",\ + ["npm:10.2.4", {\ + "packageLocation": "../.yarn/berry/cache/@nestjs-platform-express-npm-10.2.4-9288a94935-10c0.zip/node_modules/@nestjs/platform-express/",\ "packageDependencies": [\ - ["@nestjs/platform-express", "npm:10.4.1"]\ + ["@nestjs/platform-express", "npm:10.2.4"]\ ],\ "linkType": "SOFT"\ }],\ - ["npm:10.4.6", {\ - "packageLocation": "../.yarn/berry/cache/@nestjs-platform-express-npm-10.4.6-eebcfdc0f9-10c0.zip/node_modules/@nestjs/platform-express/",\ + ["npm:10.4.1", {\ + "packageLocation": "../.yarn/berry/cache/@nestjs-platform-express-npm-10.4.1-76944971fd-10c0.zip/node_modules/@nestjs/platform-express/",\ "packageDependencies": [\ - ["@nestjs/platform-express", "npm:10.4.6"]\ + ["@nestjs/platform-express", "npm:10.4.1"]\ ],\ "linkType": "SOFT"\ }],\ @@ -9106,19 +9113,19 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ - ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6", {\ - "packageLocation": "./.yarn/__virtual__/@nestjs-platform-express-virtual-1cbdbccd71/2/.yarn/berry/cache/@nestjs-platform-express-npm-10.4.6-eebcfdc0f9-10c0.zip/node_modules/@nestjs/platform-express/",\ + ["virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.2.4", {\ + "packageLocation": "./.yarn/__virtual__/@nestjs-platform-express-virtual-4ffd205e87/2/.yarn/berry/cache/@nestjs-platform-express-npm-10.2.4-9288a94935-10c0.zip/node_modules/@nestjs/platform-express/",\ "packageDependencies": [\ - ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.6"],\ - ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ - ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.4.3"],\ + ["@nestjs/platform-express", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.2.4"],\ + ["@nestjs/common", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ + ["@nestjs/core", "virtual:648c68c35811325b322f076db385dc59f68f30bef81445d78151df0a3ea0d4065e47b943cf5d70cd0f1373bda966ec3fc1243f9ee65c9dabb966db8a62ec7194#npm:10.0.5"],\ ["@types/nestjs__common", null],\ ["@types/nestjs__core", null],\ - ["body-parser", "npm:1.20.3"],\ + ["body-parser", "npm:1.20.2"],\ ["cors", "npm:2.8.5"],\ - ["express", "npm:4.21.1"],\ + ["express", "npm:4.18.2"],\ ["multer", "npm:1.4.4-lts.1"],\ - ["tslib", "npm:2.7.0"]\ + ["tslib", "npm:2.6.2"]\ ],\ "packagePeers": [\ "@nestjs/common",\ @@ -13706,6 +13713,25 @@ const RAW_RUNTIME_STATE = }]\ ]],\ ["body-parser", [\ + ["npm:1.20.1", {\ + "packageLocation": "../.yarn/berry/cache/body-parser-npm-1.20.1-759fd14db9-10c0.zip/node_modules/body-parser/",\ + "packageDependencies": [\ + ["body-parser", "npm:1.20.1"],\ + ["bytes", "npm:3.1.2"],\ + ["content-type", "npm:1.0.5"],\ + ["debug", "virtual:c7b184cd14c02e3ce555ab1875e60cf5033c617e17d82c4c02ea822101d3c817f48bf25a766b4d4335742dc5c9c14c2e88a57ed955a56c4ad0613899f82f5618#npm:2.6.9"],\ + ["depd", "npm:2.0.0"],\ + ["destroy", "npm:1.2.0"],\ + ["http-errors", "npm:2.0.0"],\ + ["iconv-lite", "npm:0.4.24"],\ + ["on-finished", "npm:2.4.1"],\ + ["qs", "npm:6.11.0"],\ + ["raw-body", "npm:2.5.1"],\ + ["type-is", "npm:1.6.18"],\ + ["unpipe", "npm:1.0.0"]\ + ],\ + "linkType": "HARD"\ + }],\ ["npm:1.20.2", {\ "packageLocation": "../.yarn/berry/cache/body-parser-npm-1.20.2-44738662cf-10c0.zip/node_modules/body-parser/",\ "packageDependencies": [\ @@ -14470,13 +14496,6 @@ const RAW_RUNTIME_STATE = ["cookie", "npm:0.6.0"]\ ],\ "linkType": "HARD"\ - }],\ - ["npm:0.7.1", {\ - "packageLocation": "../.yarn/berry/cache/cookie-npm-0.7.1-f01524ff99-10c0.zip/node_modules/cookie/",\ - "packageDependencies": [\ - ["cookie", "npm:0.7.1"]\ - ],\ - "linkType": "HARD"\ }]\ ]],\ ["cookie-signature", [\ @@ -16142,16 +16161,16 @@ const RAW_RUNTIME_STATE = }]\ ]],\ ["express", [\ - ["npm:4.19.2", {\ - "packageLocation": "../.yarn/berry/cache/express-npm-4.19.2-f81334a22a-10c0.zip/node_modules/express/",\ + ["npm:4.18.2", {\ + "packageLocation": "../.yarn/berry/cache/express-npm-4.18.2-bb15ff679a-10c0.zip/node_modules/express/",\ "packageDependencies": [\ - ["express", "npm:4.19.2"],\ + ["express", "npm:4.18.2"],\ ["accepts", "npm:1.3.8"],\ ["array-flatten", "npm:1.1.1"],\ - ["body-parser", "npm:1.20.2"],\ + ["body-parser", "npm:1.20.1"],\ ["content-disposition", "npm:0.5.4"],\ ["content-type", "npm:1.0.5"],\ - ["cookie", "npm:0.6.0"],\ + ["cookie", "npm:0.5.0"],\ ["cookie-signature", "npm:1.0.6"],\ ["debug", "virtual:c7b184cd14c02e3ce555ab1875e60cf5033c617e17d82c4c02ea822101d3c817f48bf25a766b4d4335742dc5c9c14c2e88a57ed955a56c4ad0613899f82f5618#npm:2.6.9"],\ ["depd", "npm:2.0.0"],\ @@ -16180,36 +16199,36 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ - ["npm:4.21.0", {\ - "packageLocation": "../.yarn/berry/cache/express-npm-4.21.0-377d90d8f4-10c0.zip/node_modules/express/",\ + ["npm:4.19.2", {\ + "packageLocation": "../.yarn/berry/cache/express-npm-4.19.2-f81334a22a-10c0.zip/node_modules/express/",\ "packageDependencies": [\ - ["express", "npm:4.21.0"],\ + ["express", "npm:4.19.2"],\ ["accepts", "npm:1.3.8"],\ ["array-flatten", "npm:1.1.1"],\ - ["body-parser", "npm:1.20.3"],\ + ["body-parser", "npm:1.20.2"],\ ["content-disposition", "npm:0.5.4"],\ ["content-type", "npm:1.0.5"],\ ["cookie", "npm:0.6.0"],\ ["cookie-signature", "npm:1.0.6"],\ ["debug", "virtual:c7b184cd14c02e3ce555ab1875e60cf5033c617e17d82c4c02ea822101d3c817f48bf25a766b4d4335742dc5c9c14c2e88a57ed955a56c4ad0613899f82f5618#npm:2.6.9"],\ ["depd", "npm:2.0.0"],\ - ["encodeurl", "npm:2.0.0"],\ + ["encodeurl", "npm:1.0.2"],\ ["escape-html", "npm:1.0.3"],\ ["etag", "npm:1.8.1"],\ - ["finalhandler", "npm:1.3.1"],\ + ["finalhandler", "npm:1.2.0"],\ ["fresh", "npm:0.5.2"],\ ["http-errors", "npm:2.0.0"],\ - ["merge-descriptors", "npm:1.0.3"],\ + ["merge-descriptors", "npm:1.0.1"],\ ["methods", "npm:1.1.2"],\ ["on-finished", "npm:2.4.1"],\ ["parseurl", "npm:1.3.3"],\ - ["path-to-regexp", "npm:0.1.10"],\ + ["path-to-regexp", "npm:0.1.7"],\ ["proxy-addr", "npm:2.0.7"],\ - ["qs", "npm:6.13.0"],\ + ["qs", "npm:6.11.0"],\ ["range-parser", "npm:1.2.1"],\ ["safe-buffer", "npm:5.2.1"],\ - ["send", "npm:0.19.0"],\ - ["serve-static", "npm:1.16.2"],\ + ["send", "npm:0.18.0"],\ + ["serve-static", "npm:1.15.0"],\ ["setprototypeof", "npm:1.2.0"],\ ["statuses", "npm:2.0.1"],\ ["type-is", "npm:1.6.18"],\ @@ -16218,16 +16237,16 @@ const RAW_RUNTIME_STATE = ],\ "linkType": "HARD"\ }],\ - ["npm:4.21.1", {\ - "packageLocation": "../.yarn/berry/cache/express-npm-4.21.1-f1cd48000b-10c0.zip/node_modules/express/",\ + ["npm:4.21.0", {\ + "packageLocation": "../.yarn/berry/cache/express-npm-4.21.0-377d90d8f4-10c0.zip/node_modules/express/",\ "packageDependencies": [\ - ["express", "npm:4.21.1"],\ + ["express", "npm:4.21.0"],\ ["accepts", "npm:1.3.8"],\ ["array-flatten", "npm:1.1.1"],\ ["body-parser", "npm:1.20.3"],\ ["content-disposition", "npm:0.5.4"],\ ["content-type", "npm:1.0.5"],\ - ["cookie", "npm:0.7.1"],\ + ["cookie", "npm:0.6.0"],\ ["cookie-signature", "npm:1.0.6"],\ ["debug", "virtual:c7b184cd14c02e3ce555ab1875e60cf5033c617e17d82c4c02ea822101d3c817f48bf25a766b4d4335742dc5c9c14c2e88a57ed955a56c4ad0613899f82f5618#npm:2.6.9"],\ ["depd", "npm:2.0.0"],\ @@ -21011,6 +21030,17 @@ const RAW_RUNTIME_STATE = }]\ ]],\ ["raw-body", [\ + ["npm:2.5.1", {\ + "packageLocation": "../.yarn/berry/cache/raw-body-npm-2.5.1-9dd1d9fff9-10c0.zip/node_modules/raw-body/",\ + "packageDependencies": [\ + ["raw-body", "npm:2.5.1"],\ + ["bytes", "npm:3.1.2"],\ + ["http-errors", "npm:2.0.0"],\ + ["iconv-lite", "npm:0.4.24"],\ + ["unpipe", "npm:1.0.0"]\ + ],\ + "linkType": "HARD"\ + }],\ ["npm:2.5.2", {\ "packageLocation": "../.yarn/berry/cache/raw-body-npm-2.5.2-5cb9dfebc1-10c0.zip/node_modules/raw-body/",\ "packageDependencies": [\ diff --git a/packages/nestjs-connectrpc/package.json b/packages/nestjs-connectrpc/package.json index 9c8370b7..aad01eaa 100644 --- a/packages/nestjs-connectrpc/package.json +++ b/packages/nestjs-connectrpc/package.json @@ -17,15 +17,15 @@ "postpack": "rm -rf dist" }, "devDependencies": { - "@bufbuild/protobuf": "^1.10.0", - "@connectrpc/connect": "^1.5.0", - "@connectrpc/connect-node": "^1.5.0", - "@nestjs/common": "^10.0.5", - "@nestjs/core": "^10.0.5", - "@nestjs/microservices": "^10.2.4", - "@nestjs/platform-express": "^10.2.4", - "reflect-metadata": "^0.2.2", - "rxjs": "^7.8.1" + "@bufbuild/protobuf": "1.10.0", + "@connectrpc/connect": "1.6.1", + "@connectrpc/connect-node": "1.6.1", + "@nestjs/common": "10.0.5", + "@nestjs/core": "10.0.5", + "@nestjs/microservices": "10.2.4", + "@nestjs/platform-express": "10.2.4", + "reflect-metadata": "0.2.2", + "rxjs": "7.8.1" }, "peerDependencies": { "@bufbuild/protobuf": "^1", diff --git a/yarn.lock b/yarn.lock index 9ade440b..fad4c2b9 100644 --- a/yarn.lock +++ b/yarn.lock @@ -339,15 +339,15 @@ __metadata: version: 0.0.0-use.local resolution: "@atls/nestjs-connectrpc@workspace:packages/nestjs-connectrpc" dependencies: - "@bufbuild/protobuf": "npm:^1.10.0" - "@connectrpc/connect": "npm:^1.5.0" - "@connectrpc/connect-node": "npm:^1.5.0" - "@nestjs/common": "npm:^10.0.5" - "@nestjs/core": "npm:^10.0.5" - "@nestjs/microservices": "npm:^10.2.4" - "@nestjs/platform-express": "npm:^10.2.4" - reflect-metadata: "npm:^0.2.2" - rxjs: "npm:^7.8.1" + "@bufbuild/protobuf": "npm:1.10.0" + "@connectrpc/connect": "npm:1.6.1" + "@connectrpc/connect-node": "npm:1.6.1" + "@nestjs/common": "npm:10.0.5" + "@nestjs/core": "npm:10.0.5" + "@nestjs/microservices": "npm:10.2.4" + "@nestjs/platform-express": "npm:10.2.4" + reflect-metadata: "npm:0.2.2" + rxjs: "npm:7.8.1" peerDependencies: "@bufbuild/protobuf": ^1 "@connectrpc/connect": ^1 @@ -2150,14 +2150,14 @@ __metadata: languageName: node linkType: hard -"@bufbuild/protobuf@npm:^1.10.0": +"@bufbuild/protobuf@npm:1.10.0": version: 1.10.0 resolution: "@bufbuild/protobuf@npm:1.10.0" checksum: 10c0/5487b9c2e63846d0e3bde4d025cc77ae44a22166a5d6c184df0da5581e1ab6d66dd115af0ccad814576dcd011bb1b93989fb0ac1eb4ae452979bb8b186693ba0 languageName: node linkType: hard -"@connectrpc/connect-node@npm:^1.5.0": +"@connectrpc/connect-node@npm:1.6.1": version: 1.6.1 resolution: "@connectrpc/connect-node@npm:1.6.1" dependencies: @@ -2169,7 +2169,7 @@ __metadata: languageName: node linkType: hard -"@connectrpc/connect@npm:^1.5.0": +"@connectrpc/connect@npm:1.6.1": version: 1.6.1 resolution: "@connectrpc/connect@npm:1.6.1" peerDependencies: @@ -4807,6 +4807,49 @@ __metadata: languageName: node linkType: hard +"@nestjs/microservices@npm:10.2.4": + version: 10.2.4 + resolution: "@nestjs/microservices@npm:10.2.4" + dependencies: + iterare: "npm:1.2.1" + tslib: "npm:2.6.2" + peerDependencies: + "@grpc/grpc-js": "*" + "@nestjs/common": ^10.0.0 + "@nestjs/core": ^10.0.0 + "@nestjs/websockets": ^10.0.0 + amqp-connection-manager: "*" + amqplib: "*" + cache-manager: "*" + ioredis: "*" + kafkajs: "*" + mqtt: "*" + nats: "*" + reflect-metadata: ^0.1.12 + rxjs: ^7.1.0 + peerDependenciesMeta: + "@grpc/grpc-js": + optional: true + "@nestjs/websockets": + optional: true + amqp-connection-manager: + optional: true + amqplib: + optional: true + cache-manager: + optional: true + ioredis: + optional: true + kafkajs: + optional: true + mqtt: + optional: true + nats: + optional: true + checksum: 10c0/ef25bace3239d5afa6e19976794aadb5f8f0ee9a3eb1ebd2ddf670b385bacd81d707465086e18d134d1dad9042baef157482efa5645d6cff202ec998a75793c3 + languageName: node + linkType: hard + "@nestjs/microservices@npm:10.4.1": version: 10.4.1 resolution: "@nestjs/microservices@npm:10.4.1" @@ -4850,7 +4893,7 @@ __metadata: languageName: node linkType: hard -"@nestjs/microservices@npm:10.4.6, @nestjs/microservices@npm:^10.2.4": +"@nestjs/microservices@npm:10.4.6": version: 10.4.6 resolution: "@nestjs/microservices@npm:10.4.6" dependencies: @@ -4909,19 +4952,19 @@ __metadata: languageName: node linkType: hard -"@nestjs/platform-express@npm:^10.2.4": - version: 10.4.6 - resolution: "@nestjs/platform-express@npm:10.4.6" +"@nestjs/platform-express@npm:10.2.4": + version: 10.2.4 + resolution: "@nestjs/platform-express@npm:10.2.4" dependencies: - body-parser: "npm:1.20.3" + body-parser: "npm:1.20.2" cors: "npm:2.8.5" - express: "npm:4.21.1" + express: "npm:4.18.2" multer: "npm:1.4.4-lts.1" - tslib: "npm:2.7.0" + tslib: "npm:2.6.2" peerDependencies: "@nestjs/common": ^10.0.0 "@nestjs/core": ^10.0.0 - checksum: 10c0/1d2f9d913a0b4a066aaa783810d0d029a9f201243f8ac52d396c8361a19b73d44e05d8d656388af61696784bdfff1cb8dd711bb0b1765496c6bf40b410284081 + checksum: 10c0/b94a6e0899d80e506aa93d4ea9b80501a5888269d418308c9212efa65901257abec696b0c80cef941e77fff80a005843f28c6ec75d7a7bd3951811e91d7e91c8 languageName: node linkType: hard @@ -8264,6 +8307,26 @@ __metadata: languageName: node linkType: hard +"body-parser@npm:1.20.1": + version: 1.20.1 + resolution: "body-parser@npm:1.20.1" + dependencies: + bytes: "npm:3.1.2" + content-type: "npm:~1.0.4" + debug: "npm:2.6.9" + depd: "npm:2.0.0" + destroy: "npm:1.2.0" + http-errors: "npm:2.0.0" + iconv-lite: "npm:0.4.24" + on-finished: "npm:2.4.1" + qs: "npm:6.11.0" + raw-body: "npm:2.5.1" + type-is: "npm:~1.6.18" + unpipe: "npm:1.0.0" + checksum: 10c0/a202d493e2c10a33fb7413dac7d2f713be579c4b88343cd814b6df7a38e5af1901fc31044e04de176db56b16d9772aa25a7723f64478c20f4d91b1ac223bf3b8 + languageName: node + linkType: hard + "body-parser@npm:1.20.2": version: 1.20.2 resolution: "body-parser@npm:1.20.2" @@ -8964,13 +9027,6 @@ __metadata: languageName: node linkType: hard -"cookie@npm:0.7.1": - version: 0.7.1 - resolution: "cookie@npm:0.7.1" - checksum: 10c0/5de60c67a410e7c8dc8a46a4b72eb0fe925871d057c9a5d2c0e8145c4270a4f81076de83410c4d397179744b478e33cd80ccbcc457abf40a9409ad27dcd21dde - languageName: node - linkType: hard - "cookiejar@npm:^2.1.4": version: 2.1.4 resolution: "cookiejar@npm:2.1.4" @@ -10307,16 +10363,16 @@ __metadata: languageName: node linkType: hard -"express@npm:4.19.2": - version: 4.19.2 - resolution: "express@npm:4.19.2" +"express@npm:4.18.2": + version: 4.18.2 + resolution: "express@npm:4.18.2" dependencies: accepts: "npm:~1.3.8" array-flatten: "npm:1.1.1" - body-parser: "npm:1.20.2" + body-parser: "npm:1.20.1" content-disposition: "npm:0.5.4" content-type: "npm:~1.0.4" - cookie: "npm:0.6.0" + cookie: "npm:0.5.0" cookie-signature: "npm:1.0.6" debug: "npm:2.6.9" depd: "npm:2.0.0" @@ -10342,59 +10398,59 @@ __metadata: type-is: "npm:~1.6.18" utils-merge: "npm:1.0.1" vary: "npm:~1.1.2" - checksum: 10c0/e82e2662ea9971c1407aea9fc3c16d6b963e55e3830cd0ef5e00b533feda8b770af4e3be630488ef8a752d7c75c4fcefb15892868eeaafe7353cb9e3e269fdcb + checksum: 10c0/75af556306b9241bc1d7bdd40c9744b516c38ce50ae3210658efcbf96e3aed4ab83b3432f06215eae5610c123bc4136957dc06e50dfc50b7d4d775af56c4c59c languageName: node linkType: hard -"express@npm:4.21.0, express@npm:^4.17.1": - version: 4.21.0 - resolution: "express@npm:4.21.0" +"express@npm:4.19.2": + version: 4.19.2 + resolution: "express@npm:4.19.2" dependencies: accepts: "npm:~1.3.8" array-flatten: "npm:1.1.1" - body-parser: "npm:1.20.3" + body-parser: "npm:1.20.2" content-disposition: "npm:0.5.4" content-type: "npm:~1.0.4" cookie: "npm:0.6.0" cookie-signature: "npm:1.0.6" debug: "npm:2.6.9" depd: "npm:2.0.0" - encodeurl: "npm:~2.0.0" + encodeurl: "npm:~1.0.2" escape-html: "npm:~1.0.3" etag: "npm:~1.8.1" - finalhandler: "npm:1.3.1" + finalhandler: "npm:1.2.0" fresh: "npm:0.5.2" http-errors: "npm:2.0.0" - merge-descriptors: "npm:1.0.3" + merge-descriptors: "npm:1.0.1" methods: "npm:~1.1.2" on-finished: "npm:2.4.1" parseurl: "npm:~1.3.3" - path-to-regexp: "npm:0.1.10" + path-to-regexp: "npm:0.1.7" proxy-addr: "npm:~2.0.7" - qs: "npm:6.13.0" + qs: "npm:6.11.0" range-parser: "npm:~1.2.1" safe-buffer: "npm:5.2.1" - send: "npm:0.19.0" - serve-static: "npm:1.16.2" + send: "npm:0.18.0" + serve-static: "npm:1.15.0" setprototypeof: "npm:1.2.0" statuses: "npm:2.0.1" type-is: "npm:~1.6.18" utils-merge: "npm:1.0.1" vary: "npm:~1.1.2" - checksum: 10c0/4cf7ca328f3fdeb720f30ccb2ea7708bfa7d345f9cc460b64a82bf1b2c91e5b5852ba15a9a11b2a165d6089acf83457fc477dc904d59cd71ed34c7a91762c6cc + checksum: 10c0/e82e2662ea9971c1407aea9fc3c16d6b963e55e3830cd0ef5e00b533feda8b770af4e3be630488ef8a752d7c75c4fcefb15892868eeaafe7353cb9e3e269fdcb languageName: node linkType: hard -"express@npm:4.21.1": - version: 4.21.1 - resolution: "express@npm:4.21.1" +"express@npm:4.21.0, express@npm:^4.17.1": + version: 4.21.0 + resolution: "express@npm:4.21.0" dependencies: accepts: "npm:~1.3.8" array-flatten: "npm:1.1.1" body-parser: "npm:1.20.3" content-disposition: "npm:0.5.4" content-type: "npm:~1.0.4" - cookie: "npm:0.7.1" + cookie: "npm:0.6.0" cookie-signature: "npm:1.0.6" debug: "npm:2.6.9" depd: "npm:2.0.0" @@ -10420,7 +10476,7 @@ __metadata: type-is: "npm:~1.6.18" utils-merge: "npm:1.0.1" vary: "npm:~1.1.2" - checksum: 10c0/0c287867e5f6129d3def1edd9b63103a53c40d4dc8628839d4b6827e35eb8f0de5a4656f9d85f4457eba584f9871ebb2ad26c750b36bd75d9bbb8bcebdc4892c + checksum: 10c0/4cf7ca328f3fdeb720f30ccb2ea7708bfa7d345f9cc460b64a82bf1b2c91e5b5852ba15a9a11b2a165d6089acf83457fc477dc904d59cd71ed34c7a91762c6cc languageName: node linkType: hard @@ -14519,6 +14575,18 @@ __metadata: languageName: node linkType: hard +"raw-body@npm:2.5.1": + version: 2.5.1 + resolution: "raw-body@npm:2.5.1" + dependencies: + bytes: "npm:3.1.2" + http-errors: "npm:2.0.0" + iconv-lite: "npm:0.4.24" + unpipe: "npm:1.0.0" + checksum: 10c0/5dad5a3a64a023b894ad7ab4e5c7c1ce34d3497fc7138d02f8c88a3781e68d8a55aa7d4fd3a458616fa8647cc228be314a1c03fb430a07521de78b32c4dd09d2 + languageName: node + linkType: hard + "raw-body@npm:2.5.2": version: 2.5.2 resolution: "raw-body@npm:2.5.2" @@ -14692,7 +14760,7 @@ __metadata: languageName: node linkType: hard -"reflect-metadata@npm:0.2.2, reflect-metadata@npm:^0.2.1, reflect-metadata@npm:^0.2.2": +"reflect-metadata@npm:0.2.2, reflect-metadata@npm:^0.2.1": version: 0.2.2 resolution: "reflect-metadata@npm:0.2.2" checksum: 10c0/1cd93a15ea291e420204955544637c264c216e7aac527470e393d54b4bb075f10a17e60d8168ec96600c7e0b9fcc0cb0bb6e91c3fbf5b0d8c9056f04e6ac1ec2