nice-grpc
Version:
A Node.js gRPC library that is nice to you
93 lines • 4.22 kB
JavaScript
;
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