proto-utils.mjs 1.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950
  1. #!/usr/bin/env node
  2. import * as fs from "fs/promises"
  3. import * as path from "path"
  4. import * as grpc from "@grpc/grpc-js"
  5. import * as protoLoader from "@grpc/proto-loader"
  6. const DESCRIPTOR_SET = path.resolve("dist-standalone/proto/descriptor_set.pb")
  7. const typeNameToFQN = new Map()
  8. function addTypeNameToFqn(name, fqn) {
  9. if (typeNameToFQN.has(name)) {
  10. throw new Error(`Proto type ${name} redefined (${fqn}).`)
  11. }
  12. typeNameToFQN.set(name, fqn)
  13. }
  14. // Get the fully qualified name for a proto type, e.g. getFqn('StringRequest') returns 'cline.StringRequest'
  15. export function getFqn(name) {
  16. if (!typeNameToFQN.has(name)) {
  17. throw Error(`No FQN for ${name}`)
  18. }
  19. return typeNameToFQN.get(name)
  20. }
  21. export async function loadServicesFromProtoDescriptor() {
  22. // Load service definitions from descriptor set
  23. const descriptorBuffer = await fs.readFile(DESCRIPTOR_SET)
  24. const packageDefinition = protoLoader.loadFileDescriptorSetFromBuffer(descriptorBuffer)
  25. const proto = grpc.loadPackageDefinition(packageDefinition)
  26. // Extract host services and proto messages from the proto definition
  27. const hostServices = {}
  28. for (const [name, def] of Object.entries(proto.host)) {
  29. if (def && "service" in def) {
  30. hostServices[name] = def
  31. } else {
  32. addTypeNameToFqn(name, `proto.host.${name}`)
  33. }
  34. }
  35. const protobusServices = {}
  36. for (const [name, def] of Object.entries(proto.cline)) {
  37. if (def && "service" in def) {
  38. protobusServices[name] = def
  39. } else {
  40. addTypeNameToFqn(name, `proto.cline.${name}`)
  41. }
  42. }
  43. return { protobusServices, hostServices }
  44. }