blob: 30d616045fb3a6e86c1d88013567c9fd6c05ac3f [file] [log] [blame]
# Copyright 2020 The Pigweed 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
#
# https://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.
"""pw_rpc protoc plugin entrypoint to generate code for RPC services."""
import enum
import sys
from google.protobuf.compiler import plugin_pb2
from pw_rpc import codegen_nanopb
from pw_rpc import codegen_raw
class Codegen(enum.Enum):
RAW = 0
NANOPB = 1
def process_proto_request(codegen: Codegen,
req: plugin_pb2.CodeGeneratorRequest,
res: plugin_pb2.CodeGeneratorResponse) -> None:
"""Handles a protoc CodeGeneratorRequest message.
Generates code for the files in the request and writes the output to the
specified CodeGeneratorResponse message.
Args:
req: A CodeGeneratorRequest for a proto compilation.
res: A CodeGeneratorResponse to populate with the plugin's output.
"""
for proto_file in req.proto_file:
if codegen is Codegen.RAW:
output_files = codegen_raw.process_proto_file(proto_file)
elif codegen is Codegen.NANOPB:
output_files = codegen_nanopb.process_proto_file(proto_file)
else:
raise NotImplementedError(f'Unknown codegen type {codegen}')
for output_file in output_files:
fd = res.file.add()
fd.name = output_file.name()
fd.content = output_file.content()
def main(codegen: Codegen) -> int:
"""Protobuf compiler plugin entrypoint.
Reads a CodeGeneratorRequest proto from stdin and writes a
CodeGeneratorResponse to stdout.
"""
data = sys.stdin.buffer.read()
request = plugin_pb2.CodeGeneratorRequest.FromString(data)
response = plugin_pb2.CodeGeneratorResponse()
process_proto_request(codegen, request, response)
# Declare that this plugin supports optional fields in proto3. No proto
# message code is generated, so optional in proto3 is supported trivially.
response.supported_features |= ( # type: ignore[attr-defined]
response.FEATURE_PROTO3_OPTIONAL) # type: ignore[attr-defined]
sys.stdout.buffer.write(response.SerializeToString())
return 0