UNPKG

nice-grpc

Version:

A Node.js gRPC library that is nice to you

93 lines 4.22 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.createClientStreamingMethod = createClientStreamingMethod; const abort_controller_x_1 = require("abort-controller-x"); const nice_grpc_common_1 = require("nice-grpc-common"); const service_definitions_1 = require("../service-definitions"); const convertMetadata_1 = require("../utils/convertMetadata"); const isAsyncIterable_1 = require("../utils/isAsyncIterable"); const wrapClientError_1 = require("./wrapClientError"); /** @internal */ function createClientStreamingMethod(definition, client, middleware, defaultOptions) { const grpcMethodDefinition = (0, service_definitions_1.toGrpcJsMethodDefinition)(definition); const methodDescriptor = { path: definition.path, requestStream: definition.requestStream, responseStream: definition.responseStream, options: definition.options, }; async function* clientStreamingMethod(request, options) { if (!(0, isAsyncIterable_1.isAsyncIterable)(request)) { throw new Error('A middleware passed invalid request to next(): expected a single message for client streaming method'); } const { metadata = (0, nice_grpc_common_1.Metadata)(), signal = new AbortController().signal, onHeader, onTrailer, } = options; return await (0, abort_controller_x_1.execute)(signal, (resolve, reject) => { const pipeAbortController = new AbortController(); const call = client.makeClientStreamRequest(grpcMethodDefinition.path, grpcMethodDefinition.requestSerialize, grpcMethodDefinition.responseDeserialize, (0, convertMetadata_1.convertMetadataToGrpcJs)(metadata), (err, response) => { pipeAbortController.abort(); if (err != null) { reject((0, wrapClientError_1.wrapClientError)(err, definition.path)); } else { resolve(response); } }); call.on('metadata', metadata => { onHeader?.((0, convertMetadata_1.convertMetadataFromGrpcJs)(metadata)); }); call.on('status', status => { onTrailer?.((0, convertMetadata_1.convertMetadataFromGrpcJs)(status.metadata)); }); pipeRequest(pipeAbortController.signal, request, call).then(() => { call.end(); }, err => { if (!(0, abort_controller_x_1.isAbortError)(err)) { reject(err); call.cancel(); } }); return () => { pipeAbortController.abort(); call.cancel(); }; }); } const method = middleware == null ? clientStreamingMethod : (request, options) => middleware({ method: methodDescriptor, requestStream: true, request, responseStream: false, next: clientStreamingMethod, }, options); return async (request, options) => { const iterable = method(request, { ...defaultOptions, ...options, }); const iterator = iterable[Symbol.asyncIterator](); let result = await iterator.next(); while (true) { if (!result.done) { result = await iterator.throw(new Error('A middleware yielded a message, but expected to only return a message for client streaming method')); continue; } if (result.value == null) { result = await iterator.throw(new Error('A middleware returned void, but expected to return a message for client streaming method')); continue; } return result.value; } }; } async function pipeRequest(signal, request, call) { for await (const item of request) { (0, abort_controller_x_1.throwIfAborted)(signal); const shouldContinue = call.write(item); if (!shouldContinue) { await (0, abort_controller_x_1.waitForEvent)(signal, call, 'drain'); } } } //# sourceMappingURL=createClientStreamingMethod.js.map