Skip to content

Latest commit

 

History

History
124 lines (97 loc) · 2.87 KB

README.md

File metadata and controls

124 lines (97 loc) · 2.87 KB

typescript-grpc

a minimalistic annotation-based typescript library to help build grpc servers and clients.

Usage

$ yarn add typescript-grpc protobufjs grpc @grpc/proto-loader --save

See src/example.ts

import debug from 'debug';
import { Type, Field, Message } from 'protobufjs';
import { load } from '@grpc/proto-loader';
import { Server, ServiceDefinition, ServerCredentials } from 'grpc';

import { Service, Method, generateProto, wrapServiceMethods } from 'typescript-grpc';

const log = debug('typescript-grpc:example');

@Type.d()
class Movie extends Message<Movie> {
  @Field.d(1, 'string')
  name: string;

  @Field.d(2, 'int32')
  year: number;

  @Field.d(3, 'float')
  rating: number;

  @Field.d(4, 'string', 'repeated')
  cast: string[];
}

@Type.d()
class MoviesResult extends Message<MoviesResult> {
  @Field.d(1, Movie, 'repeated')
  result: Movie[];
}

@Type.d()
class EmptyRequest extends Message<EmptyRequest> {}

@Type.d()
class SearchByCastInput extends Message<SearchByCastInput> {
  @Field.d(1, 'string')
  castName: string;
}

@Service()
class ExampleService {
  @Method({
    requestType: 'EmptyRequest',
    requestStream: false,
    responseType: 'MoviesResult',
    responseStream: false,
  })
  async getMovies(req: EmptyRequest): Promise<MoviesResult> {
    log('get movies called');
    return new MoviesResult({ result: [] });
  }

  @Method({
    requestType: 'SearchByCastInput',
    requestStream: false,
    responseType: 'Movie',
    responseStream: true,
  })
  searchMoviesByCast(req: SearchByCastInput): Observable<Movie> {
    const movies = [
      {
        cast: ['Tom Cruise', 'Simon Pegg', 'Jeremy Renner'],
        name: 'Mission: Impossible Rogue Nation',
        rating: 0.97,
        year: 2015,
      },
      {
        cast: ['Tom Cruise', 'Simon Pegg', 'Henry Cavill'],
        name: 'Mission: Impossible - Fallout',
        rating: 0.93,
        year: 2018,
      },
      {
        cast: ['Leonardo DiCaprio', 'Jonah Hill', 'Margot Robbie'],
        name: 'The Wolf of Wall Street',
        rating: 0.78,
        year: 2013,
      },
    ];

    return from(movies.filter(movie => movie.cast.indexOf(req.castName) > -1).map(m => new Movie(m)));
}

async function main(): Promise<void> {
  const service = new ExampleService();
  log(service);

  const protoPath = await generateProto('example');

  const packageDefinition = await load(protoPath);

  const server = new Server({
    'grpc.max_send_message_length': -1,
    'grpc.max_receive_message_length': -1,
  });

  server.addService(packageDefinition[service.constructor.name] as ServiceDefinition<any>, wrapServiceMethods(service));
  server.bind('0.0.0.0:50051', ServerCredentials.createInsecure());
  server.start();

  log(`server started`);
}

main();

Features

  • Modern service method handlers via Promise and rxjs streams
  • Auto proto generation using annotations