-
Notifications
You must be signed in to change notification settings - Fork 36
/
AsioGrpcProtobufGenerator.cmake
234 lines (203 loc) · 10.8 KB
/
AsioGrpcProtobufGenerator.cmake
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
# Copyright 2024 Dennis Hezel
#
# 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.
#[[===
# /* [asio_grpc_protobuf_generate] */
In the same directory that called `find_package(asio-grpc)` the following CMake function will be made available.
It can be used to generate Protobuf/gRPC source files from `.proto` schemas.<br>
If you are using [cmake-format](https://github.com/cheshirekow/cmake_format) then you can copy the `asio_grpc_protobuf_generate` section from
[cmake-format.yaml](https://github.com/Tradias/asio-grpc/blob/v3.4.1/cmake-format.yaml#L2-L14) to get proper formatting.
```cmake
asio_grpc_protobuf_generate(PROTOS <proto_file1> [<proto_file2>...]
[OUT_DIR <output_directory>]
[OUT_VAR <output_variable>]
[TARGET <target>]
[USAGE_REQUIREMENT PRIVATE|PUBLIC|INTERFACE]
[IMPORT_DIRS <directories>...]
[EXTRA_ARGS <arguments>...]
[GENERATE_GRPC]
[GENERATE_DESCRIPTORS]
[GENERATE_MOCK_CODE])
```
__PROTOS__: Input `.proto` schema files.<br>
<br>
__OUT_DIR__: Generated files output directory. Default: `CMAKE_CURRENT_BINARY_DIR`.<br>
<br>
__OUT_VAR__: Variable to define with generated source files.<br>
<br>
__TARGET__: Add generated source files to target.<br>
<br>
__USAGE_REQUIREMENT__: How to add sources to `<target>`: `PRIVATE`, `PUBLIC`, `INTERFACE`. Default: `PRIVATE`.<br>
<br>
__IMPORT_DIRS__: Import directories to be added to the protoc command line. If unspecified then the directory of each .proto file will be used.<br>
<br>
__EXTRA_ARGS__: Additional protoc command line arguments.<br>
<br>
__GENERATE_GRPC__: Generate gRPC files (.grpc.pb.h and .grpc.pb.cc).<br>
<br>
__GENERATE_DESCRIPTORS__: Generate descriptor files named `<proto_file_base_name>.desc`.<br>
<br>
__GENERATE_MOCK_CODE__: Generate gRPC client stub mock files named `_mock.grpc.pb.h`.
# /* [asio_grpc_protobuf_generate] */
===]]
function(asio_grpc_protobuf_generate)
include(CMakeParseArguments)
set(_asio_grpc_options GENERATE_GRPC GENERATE_DESCRIPTORS GENERATE_MOCK_CODE)
set(_asio_grpc_singleargs OUT_VAR OUT_DIR TARGET USAGE_REQUIREMENT)
set(_asio_grpc_multiargs PROTOS IMPORT_DIRS EXTRA_ARGS)
cmake_parse_arguments(asio_grpc_protobuf_generate "${_asio_grpc_options}" "${_asio_grpc_singleargs}"
"${_asio_grpc_multiargs}" "${ARGN}")
if(asio_grpc_protobuf_generate_UNPARSED_ARGUMENTS)
message(
AUTHOR_WARNING
"asio_grpc_protobuf_generate unknown argument: ${asio_grpc_protobuf_generate_UNPARSED_ARGUMENTS}")
endif()
if(asio_grpc_protobuf_generate_KEYWORDS_MISSING_VALUES)
message(
AUTHOR_WARNING
"asio_grpc_protobuf_generate missing values for: ${asio_grpc_protobuf_generate_KEYWORDS_MISSING_VALUES}"
)
endif()
if(NOT asio_grpc_protobuf_generate_PROTOS)
message(SEND_ERROR "asio_grpc_protobuf_generate called without any proto files: PROTOS")
return()
endif()
if(NOT asio_grpc_protobuf_generate_OUT_VAR AND NOT asio_grpc_protobuf_generate_TARGET)
message(SEND_ERROR "asio_grpc_protobuf_generate called without a target or output variable: TARGET or OUT_VAR")
return()
endif()
if(asio_grpc_protobuf_generate_TARGET AND NOT TARGET ${asio_grpc_protobuf_generate_TARGET})
message(
SEND_ERROR
"asio_grpc_protobuf_generate argument passed to TARGET is not a target: ${asio_grpc_protobuf_generate_TARGET}"
)
return()
endif()
if(asio_grpc_protobuf_generate_GENERATE_MOCK_CODE AND NOT asio_grpc_protobuf_generate_GENERATE_GRPC)
message(SEND_ERROR "asio_grpc_protobuf_generate argument GENERATE_MOCK_CODE requires GENERATE_GRPC")
return()
endif()
if(NOT asio_grpc_protobuf_generate_OUT_DIR)
set(asio_grpc_protobuf_generate_OUT_DIR "${CMAKE_CURRENT_BINARY_DIR}")
endif()
set(_asio_grpc_generated_extensions ".pb.cc" ".pb.h")
if(asio_grpc_protobuf_generate_GENERATE_GRPC)
list(APPEND _asio_grpc_generated_extensions ".grpc.pb.cc" ".grpc.pb.h")
if(asio_grpc_protobuf_generate_GENERATE_MOCK_CODE)
list(APPEND _asio_grpc_generated_extensions "_mock.grpc.pb.h")
endif()
endif()
if(NOT asio_grpc_protobuf_generate_USAGE_REQUIREMENT)
set(asio_grpc_protobuf_generate_USAGE_REQUIREMENT "PRIVATE")
endif()
if(NOT asio_grpc_protobuf_generate_IMPORT_DIRS)
# Create an include path for each file specified
foreach(_asio_grpc_file ${asio_grpc_protobuf_generate_PROTOS})
get_filename_component(_asio_grpc_abs_file "${_asio_grpc_file}" ABSOLUTE)
get_filename_component(_asio_grpc_abs_path "${_asio_grpc_abs_file}" PATH)
list(APPEND asio_grpc_protobuf_generate_IMPORT_DIRS "${_asio_grpc_abs_path}")
endforeach()
endif()
# Add all explicitly specified import directory to the include path
foreach(_asio_grpc_dir ${asio_grpc_protobuf_generate_IMPORT_DIRS})
get_filename_component(_asio_grpc_abs_dir "${_asio_grpc_dir}" ABSOLUTE)
list(FIND _asio_grpc_protobuf_include_args "${_asio_grpc_abs_dir}" _asio_grpc_contains_already)
if(${_asio_grpc_contains_already} EQUAL -1)
list(APPEND _asio_grpc_protobuf_include_args -I "${_asio_grpc_abs_dir}")
endif()
endforeach()
set(_asio_grpc_abs_input_files)
set(_asio_grpc_generated_srcs)
foreach(_asio_grpc_proto ${asio_grpc_protobuf_generate_PROTOS})
get_filename_component(_asio_grpc_abs_file "${_asio_grpc_proto}" ABSOLUTE)
# Get .proto base name
get_filename_component(_asio_grpc_full_name ${_asio_grpc_abs_file} NAME)
string(FIND "${_asio_grpc_full_name}" "." _asio_grpc_file_last_ext_pos REVERSE)
string(SUBSTRING "${_asio_grpc_full_name}" 0 ${_asio_grpc_file_last_ext_pos} _asio_grpc_basename)
list(APPEND _asio_grpc_abs_input_files "${_asio_grpc_abs_file}")
# Compute generated file output directory by checking that the .proto file is not in a parent directory of all
# import directories.
get_filename_component(_asio_grpc_abs_dir ${_asio_grpc_abs_file} DIRECTORY)
set(_asio_grpc_suitable_include_found off)
foreach(_asio_grpc_dir ${_asio_grpc_protobuf_include_args})
if(NOT _asio_grpc_dir STREQUAL "-I")
file(RELATIVE_PATH _asio_grpc_rel_out_dir ${_asio_grpc_dir} ${_asio_grpc_abs_dir})
string(FIND "${_asio_grpc_rel_out_dir}" "../" _asio_grpc_is_in_parent_folder)
if(NOT ${_asio_grpc_is_in_parent_folder} EQUAL 0)
set(_asio_grpc_suitable_include_found on)
break()
endif()
endif()
endforeach()
if(NOT _asio_grpc_suitable_include_found)
string(REPLACE ";" " " _asio_grpc_pretty_import_dirs "${asio_grpc_protobuf_generate_IMPORT_DIRS}")
message(
SEND_ERROR
"None of the IMPORT_DIRS passed to asio_grpc_protobuf_generate contain the proto file: \"${_asio_grpc_abs_file}\".\nIMPORT_DIRS: ${_asio_grpc_pretty_import_dirs}"
)
return()
endif()
set(_asio_grpc_actual_out_dir "${asio_grpc_protobuf_generate_OUT_DIR}/${_asio_grpc_rel_out_dir}")
# Collect generated files
foreach(_asio_grpc_ext ${_asio_grpc_generated_extensions})
list(APPEND _asio_grpc_generated_srcs
"${_asio_grpc_actual_out_dir}/${_asio_grpc_basename}${_asio_grpc_ext}")
endforeach()
if(asio_grpc_protobuf_generate_GENERATE_DESCRIPTORS)
set(_asio_grpc_descriptor_file "${_asio_grpc_actual_out_dir}/${_asio_grpc_basename}.desc")
set(_asio_grpc_descriptor_command "--descriptor_set_out=${_asio_grpc_descriptor_file}")
list(APPEND _asio_grpc_generated_srcs "${_asio_grpc_descriptor_file}")
endif()
endforeach()
# Run protoc
set(_asio_grpc_command_arguments --cpp_out "${asio_grpc_protobuf_generate_OUT_DIR}"
"${_asio_grpc_descriptor_command}" "${_asio_grpc_protobuf_include_args}")
if(asio_grpc_protobuf_generate_GENERATE_GRPC)
if(asio_grpc_protobuf_generate_GENERATE_MOCK_CODE)
set(_asio_grpc_generate_mock_code "generate_mock_code=true:")
endif()
list(APPEND _asio_grpc_command_arguments --grpc_out
"${_asio_grpc_generate_mock_code}${asio_grpc_protobuf_generate_OUT_DIR}"
"--plugin=protoc-gen-grpc=$<TARGET_FILE:gRPC::grpc_cpp_plugin>")
endif()
list(APPEND _asio_grpc_command_arguments ${asio_grpc_protobuf_generate_EXTRA_ARGS} ${_asio_grpc_abs_input_files})
string(REPLACE ";" " " _asio_grpc_pretty_command_arguments "${_asio_grpc_command_arguments}")
add_custom_command(
OUTPUT ${_asio_grpc_generated_srcs}
COMMAND "${CMAKE_COMMAND}" "-E" "make_directory" "${asio_grpc_protobuf_generate_OUT_DIR}"
COMMAND protobuf::protoc ${_asio_grpc_command_arguments}
DEPENDS protobuf::protoc ${_asio_grpc_abs_input_files}
COMMENT "protoc ${_asio_grpc_pretty_command_arguments}"
VERBATIM)
set_source_files_properties(${_asio_grpc_generated_srcs} PROPERTIES SKIP_UNITY_BUILD_INCLUSION on)
if(asio_grpc_protobuf_generate_TARGET)
if("${asio_grpc_protobuf_generate_USAGE_REQUIREMENT}" STREQUAL "INTERFACE")
target_sources(${asio_grpc_protobuf_generate_TARGET} INTERFACE ${_asio_grpc_generated_srcs})
else()
target_sources(${asio_grpc_protobuf_generate_TARGET} PRIVATE ${_asio_grpc_generated_srcs})
endif()
if("${asio_grpc_protobuf_generate_USAGE_REQUIREMENT}" STREQUAL "PUBLIC")
target_include_directories(${asio_grpc_protobuf_generate_TARGET}
PUBLIC "$<BUILD_INTERFACE:${asio_grpc_protobuf_generate_OUT_DIR}>")
else()
target_include_directories(
${asio_grpc_protobuf_generate_TARGET} "${asio_grpc_protobuf_generate_USAGE_REQUIREMENT}"
"${asio_grpc_protobuf_generate_OUT_DIR}")
endif()
endif()
if(asio_grpc_protobuf_generate_OUT_VAR)
set(${asio_grpc_protobuf_generate_OUT_VAR}
${_asio_grpc_generated_srcs}
PARENT_SCOPE)
endif()
endfunction()