node.js - 您如何为用于 GRPC 的 protobuf 文件生成类型?

标签 node.js typescript protocol-buffers grpc

我正在尝试将 GRPC 与 TypeScript 一起使用,并且我试图确保我设置了所有类型(而不仅仅是添加我自己的映射或使用 any
我已经解决了评论中提到的问题。

import { Server, loadPackageDefinition } from "grpc";
import { loadSync } from "@grpc/proto-loader";

const packageDefinition = loadSync(__dirname + "/protos/artifact.proto");
const artifacts = loadPackageDefinition(packageDefinition).artifacts;

// what are the types here?
function SignedUrlPutObject(call, callback) {
}

const server = new Server();
// There's no ArtifactUpload defined in artifacts
server.addService(artifacts.ArtifactUpload.service, { SignedUrlPutObject })

我尝试的另一种方法是使用 pbjspbts .
"protobuf": "pbjs -t static-module -w commonjs -o protos.js protos/artifact-upload.proto && pbts -o protos.d.ts protos.js",
这生成了typings 文件,但我无法让它与grpc 一起使用。这是我发现的一个 github 问题,可能与 https://github.com/protobufjs/protobuf.js/issues/1381 相关

最佳答案

您可以使用 3 个主要工具:

  • ts-protoc-gen
  • @grpc/proto-loader
  • grpc_tools_node_protoc_ts

  • 我推荐使用 proto-loader:
    npm i @grpc/proto-loader
    
    然后,您可以像这样生成类型:
    ./node_modules/.bin/proto-loader-gen-types --longs=String --enums=String --defaults --oneofs --grpcLib=@grpc/grpc-js --outDir=proto/ proto/*.proto
    
    这是我用于此示例的 proto 文件:
    syntax = "proto3";
    
    package example_package;
    
    message ServerMessage {
      string server_message = 1;
    }
    
    message ClientMessage {
      string client_message = 1;
    }
    
    service Example {
      rpc unaryCall(ClientMessage) returns (ServerMessage) {}
      rpc serverStreamingCall(ClientMessage) returns (stream ServerMessage) {}
      rpc clientStreamingCall(stream ClientMessage) returns (ServerMessage) {}
      rpc bidirectionalStreamingCall(stream ClientMessage) returns (stream ServerMessage) {}
    }
    
    生成类型后,您可以像这样使用它们:
    import * as grpc from '@grpc/grpc-js';
    import * as protoLoader from '@grpc/proto-loader';
    import { ProtoGrpcType } from './proto/example';
    import { ClientMessage } from './proto/example_package/ClientMessage';
    import { ExampleHandlers } from './proto/example_package/Example';
    import { ServerMessage } from './proto/example_package/ServerMessage';
    
    const host = '0.0.0.0:9090';
    
    const exampleServer: ExampleHandlers = {
      unaryCall(
        call: grpc.ServerUnaryCall<ClientMessage, ServerMessage>,
        callback: grpc.sendUnaryData<ServerMessage>
      ) {
        if (call.request) {
          console.log(`(server) Got client message: ${call.request.clientMessage}`);
        }
        callback(null, {
          serverMessage: 'Message from server',
        });
      },
    
      serverStreamingCall(
        call: grpc.ServerWritableStream<ClientMessage, ServerMessage>
      ) {
        call.write({
          serverMessage: 'Message from server',
        });
      },
    
      clientStreamingCall(
        call: grpc.ServerReadableStream<ClientMessage, ServerMessage>
      ) {
        call.on('data', (clientMessage: ClientMessage) => {
          console.log(
            `(server) Got client message: ${clientMessage.clientMessage}`
          );
        });
      },
    
      bidirectionalStreamingCall(
        call: grpc.ServerDuplexStream<ClientMessage, ServerMessage>
      ) {
        call.write({
          serverMessage: 'Message from server',
        });
        call.on('data', (clientMessage: ClientMessage) => {
          console.log(
            `(server) Got client message: ${clientMessage.clientMessage}`
          );
        });
      },
    };
    
    function getServer(): grpc.Server {
      const packageDefinition = protoLoader.loadSync('./proto/example.proto');
      const proto = (grpc.loadPackageDefinition(
        packageDefinition
      ) as unknown) as ProtoGrpcType;
      const server = new grpc.Server();
      server.addService(proto.example_package.Example.service, exampleServer);
      return server;
    }
    
    if (require.main === module) {
      const server = getServer();
      server.bindAsync(
        host,
        grpc.ServerCredentials.createInsecure(),
        (err: Error | null, port: number) => {
          if (err) {
            console.error(`Server error: ${err.message}`);
          } else {
            console.log(`Server bound on port: ${port}`);
            server.start();
          }
        }
      );
    }
    
    我在这里创建了各种如何在 TypeScript 中使用 gRPC 的示例:https://github.com/badsyntax/grpc-js-types

    关于node.js - 您如何为用于 GRPC 的 protobuf 文件生成类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64530776/

    相关文章:

    node.js - 仅使用 gulp 或 webpack 构建此 Node 项目

    c++ - 将 google::protobuf::RepeatedField<float> 转换为 float

    java - 在 Matlab 中使用 Protocol Buffer Java 绑定(bind)

    javascript - 如何使用 vo.js 在 for 循环中产生多个 Promise?

    node.js - 使用不同的模式渲染 mongodb 文档

    javascript - Evernote JS SDK - 能否构造 NoteFilter 来过滤不包含 tagGUID 列表的笔记?

    javascript - 通过 JavaScript 函数调用 Web API

    javascript - 如何将 lint-staged 与 jest --collectCoverageFrom 一起使用

    typescript - VsCode 插件 : Color Coding inline SQL in ts file

    visual-studio-code - VS Code PyLint 错误 E0602( undefined variable )与 ProtoBuf 编译的 Python 结构