mirror of https://github.com/grpc/grpc-node.git
236 lines
6.9 KiB
TypeScript
236 lines
6.9 KiB
TypeScript
/*
|
|
* Copyright 2019 gRPC authors.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
*/
|
|
|
|
import { ChannelCredentials } from './channel-credentials';
|
|
import { ChannelOptions } from './channel-options';
|
|
import { Client } from './client';
|
|
import { UntypedServiceImplementation } from './server';
|
|
|
|
export interface Serialize<T> {
|
|
(value: T): Buffer;
|
|
}
|
|
|
|
export interface Deserialize<T> {
|
|
(bytes: Buffer): T;
|
|
}
|
|
|
|
export interface ClientMethodDefinition<RequestType, ResponseType> {
|
|
path: string;
|
|
requestStream: boolean;
|
|
responseStream: boolean;
|
|
requestSerialize: Serialize<RequestType>;
|
|
responseDeserialize: Deserialize<ResponseType>;
|
|
originalName?: string;
|
|
}
|
|
|
|
export interface ServerMethodDefinition<RequestType, ResponseType> {
|
|
path: string;
|
|
requestStream: boolean;
|
|
responseStream: boolean;
|
|
responseSerialize: Serialize<ResponseType>;
|
|
requestDeserialize: Deserialize<RequestType>;
|
|
originalName?: string;
|
|
}
|
|
|
|
export interface MethodDefinition<RequestType, ResponseType>
|
|
extends ClientMethodDefinition<RequestType, ResponseType>,
|
|
ServerMethodDefinition<RequestType, ResponseType> {}
|
|
|
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
export type ServiceDefinition<
|
|
ImplementationType = UntypedServiceImplementation
|
|
> = {
|
|
readonly [index in keyof ImplementationType]: MethodDefinition<any, any>;
|
|
};
|
|
/* eslint-enable @typescript-eslint/no-explicit-any */
|
|
|
|
export interface ProtobufTypeDefinition {
|
|
format: string;
|
|
type: object;
|
|
fileDescriptorProtos: Buffer[];
|
|
}
|
|
|
|
export interface PackageDefinition {
|
|
[index: string]: ServiceDefinition | ProtobufTypeDefinition;
|
|
}
|
|
|
|
/**
|
|
* Map with short names for each of the requester maker functions. Used in
|
|
* makeClientConstructor
|
|
* @private
|
|
*/
|
|
const requesterFuncs = {
|
|
unary: Client.prototype.makeUnaryRequest,
|
|
server_stream: Client.prototype.makeServerStreamRequest,
|
|
client_stream: Client.prototype.makeClientStreamRequest,
|
|
bidi: Client.prototype.makeBidiStreamRequest,
|
|
};
|
|
|
|
export interface ServiceClient extends Client {
|
|
[methodName: string]: Function;
|
|
}
|
|
|
|
export interface ServiceClientConstructor {
|
|
new (
|
|
address: string,
|
|
credentials: ChannelCredentials,
|
|
options?: Partial<ChannelOptions>
|
|
): ServiceClient;
|
|
service: ServiceDefinition;
|
|
}
|
|
|
|
/**
|
|
* Returns true, if given key is included in the blacklisted
|
|
* keys.
|
|
* @param key key for check, string.
|
|
*/
|
|
function isPrototypePolluted(key: string): Boolean {
|
|
return ['__proto__', 'prototype', 'constructor'].includes(key);
|
|
}
|
|
|
|
/**
|
|
* Creates a constructor for a client with the given methods, as specified in
|
|
* the methods argument. The resulting class will have an instance method for
|
|
* each method in the service, which is a partial application of one of the
|
|
* [Client]{@link grpc.Client} request methods, depending on `requestSerialize`
|
|
* and `responseSerialize`, with the `method`, `serialize`, and `deserialize`
|
|
* arguments predefined.
|
|
* @param methods An object mapping method names to
|
|
* method attributes
|
|
* @param serviceName The fully qualified name of the service
|
|
* @param classOptions An options object.
|
|
* @return New client constructor, which is a subclass of
|
|
* {@link grpc.Client}, and has the same arguments as that constructor.
|
|
*/
|
|
export function makeClientConstructor(
|
|
methods: ServiceDefinition,
|
|
serviceName: string,
|
|
classOptions?: {}
|
|
): ServiceClientConstructor {
|
|
if (!classOptions) {
|
|
classOptions = {};
|
|
}
|
|
|
|
class ServiceClientImpl extends Client implements ServiceClient {
|
|
static service: ServiceDefinition;
|
|
[methodName: string]: Function;
|
|
}
|
|
|
|
Object.keys(methods).forEach((name) => {
|
|
if (isPrototypePolluted(name)) {
|
|
return;
|
|
}
|
|
const attrs = methods[name];
|
|
let methodType: keyof typeof requesterFuncs;
|
|
// TODO(murgatroid99): Verify that we don't need this anymore
|
|
if (typeof name === 'string' && name.charAt(0) === '$') {
|
|
throw new Error('Method names cannot start with $');
|
|
}
|
|
if (attrs.requestStream) {
|
|
if (attrs.responseStream) {
|
|
methodType = 'bidi';
|
|
} else {
|
|
methodType = 'client_stream';
|
|
}
|
|
} else {
|
|
if (attrs.responseStream) {
|
|
methodType = 'server_stream';
|
|
} else {
|
|
methodType = 'unary';
|
|
}
|
|
}
|
|
const serialize = attrs.requestSerialize;
|
|
const deserialize = attrs.responseDeserialize;
|
|
const methodFunc = partial(
|
|
requesterFuncs[methodType],
|
|
attrs.path,
|
|
serialize,
|
|
deserialize
|
|
);
|
|
ServiceClientImpl.prototype[name] = methodFunc;
|
|
// Associate all provided attributes with the method
|
|
Object.assign(ServiceClientImpl.prototype[name], attrs);
|
|
if (attrs.originalName && !isPrototypePolluted(attrs.originalName)) {
|
|
ServiceClientImpl.prototype[attrs.originalName] =
|
|
ServiceClientImpl.prototype[name];
|
|
}
|
|
});
|
|
|
|
ServiceClientImpl.service = methods;
|
|
|
|
return ServiceClientImpl;
|
|
}
|
|
|
|
function partial(
|
|
fn: Function,
|
|
path: string,
|
|
serialize: Function,
|
|
deserialize: Function
|
|
): Function {
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
return function (this: any, ...args: any[]) {
|
|
return fn.call(this, path, serialize, deserialize, ...args);
|
|
};
|
|
}
|
|
|
|
export interface GrpcObject {
|
|
[index: string]:
|
|
| GrpcObject
|
|
| ServiceClientConstructor
|
|
| ProtobufTypeDefinition;
|
|
}
|
|
|
|
function isProtobufTypeDefinition(
|
|
obj: ServiceDefinition | ProtobufTypeDefinition
|
|
): obj is ProtobufTypeDefinition {
|
|
return 'format' in obj;
|
|
}
|
|
|
|
/**
|
|
* Load a gRPC package definition as a gRPC object hierarchy.
|
|
* @param packageDef The package definition object.
|
|
* @return The resulting gRPC object.
|
|
*/
|
|
export function loadPackageDefinition(
|
|
packageDef: PackageDefinition
|
|
): GrpcObject {
|
|
const result: GrpcObject = {};
|
|
for (const serviceFqn in packageDef) {
|
|
if (Object.prototype.hasOwnProperty.call(packageDef, serviceFqn)) {
|
|
const service = packageDef[serviceFqn];
|
|
const nameComponents = serviceFqn.split('.');
|
|
if (nameComponents.some((comp: string) => isPrototypePolluted(comp))) {
|
|
continue;
|
|
}
|
|
const serviceName = nameComponents[nameComponents.length - 1];
|
|
let current = result;
|
|
for (const packageName of nameComponents.slice(0, -1)) {
|
|
if (!current[packageName]) {
|
|
current[packageName] = {};
|
|
}
|
|
current = current[packageName] as GrpcObject;
|
|
}
|
|
if (isProtobufTypeDefinition(service)) {
|
|
current[serviceName] = service;
|
|
} else {
|
|
current[serviceName] = makeClientConstructor(service, serviceName, {});
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|