blob: 73c396d5564964faba551563e0d92e9e56dd6211 [file] [log] [blame]
Damien Martin-Guillerez76547e52016-01-15 14:01:37 +01001def _GetPath(ctx, path):
2 if ctx.label.workspace_root:
3 return ctx.label.workspace_root + '/' + path
4 else:
5 return path
6
Kristina Chodorow4e7ecde2017-01-25 14:10:56 -05007def _IsNewExternal(ctx):
8 # Bazel 0.4.4 and older have genfiles paths that look like:
9 # bazel-out/local-fastbuild/genfiles/external/repo/foo
10 # After the exec root rearrangement, they look like:
11 # ../repo/bazel-out/local-fastbuild/genfiles/foo
12 return ctx.label.workspace_root.startswith("../")
13
Jisi Liu993fb702015-10-19 17:19:49 -070014def _GenDir(ctx):
Kristina Chodorow4e7ecde2017-01-25 14:10:56 -050015 if _IsNewExternal(ctx):
16 # We are using the fact that Bazel 0.4.4+ provides repository-relative paths
17 # for ctx.genfiles_dir.
18 return ctx.genfiles_dir.path + (
19 "/" + ctx.attr.includes[0] if ctx.attr.includes and ctx.attr.includes[0] else "")
20 # This means that we're either in the old version OR the new version in the local repo.
21 # Either way, appending the source path to the genfiles dir works.
22 return ctx.var["GENDIR"] + "/" + _SourceDir(ctx)
23
24def _SourceDir(ctx):
Jisi Liu53a56be2015-10-20 15:18:20 -070025 if not ctx.attr.includes:
Damien Martin-Guillerez76547e52016-01-15 14:01:37 +010026 return ctx.label.workspace_root
Jisi Liu53a56be2015-10-20 15:18:20 -070027 if not ctx.attr.includes[0]:
Damien Martin-Guillerez76547e52016-01-15 14:01:37 +010028 return _GetPath(ctx, ctx.label.package)
Jisi Liu39362b32015-10-14 17:12:11 -070029 if not ctx.label.package:
Damien Martin-Guillerez76547e52016-01-15 14:01:37 +010030 return _GetPath(ctx, ctx.attr.includes[0])
31 return _GetPath(ctx, ctx.label.package + '/' + ctx.attr.includes[0])
Jisi Liu39362b32015-10-14 17:12:11 -070032
Andreas Bergmeierbbeb9832016-08-15 16:57:30 +020033def _CcHdrs(srcs, use_grpc_plugin=False):
34 ret = [s[:-len(".proto")] + ".pb.h" for s in srcs]
Manjunath Kudlurf5c73632016-02-25 08:50:50 -080035 if use_grpc_plugin:
Andreas Bergmeierbbeb9832016-08-15 16:57:30 +020036 ret += [s[:-len(".proto")] + ".grpc.pb.h" for s in srcs]
Manjunath Kudlurf5c73632016-02-25 08:50:50 -080037 return ret
Jisi Liu39362b32015-10-14 17:12:11 -070038
Andreas Bergmeierbbeb9832016-08-15 16:57:30 +020039def _CcSrcs(srcs, use_grpc_plugin=False):
40 ret = [s[:-len(".proto")] + ".pb.cc" for s in srcs]
41 if use_grpc_plugin:
42 ret += [s[:-len(".proto")] + ".grpc.pb.cc" for s in srcs]
43 return ret
44
45def _CcOuts(srcs, use_grpc_plugin=False):
46 return _CcHdrs(srcs, use_grpc_plugin) + _CcSrcs(srcs, use_grpc_plugin)
47
Jisi Liu993fb702015-10-19 17:19:49 -070048def _PyOuts(srcs):
Jisi Liu125a91b2015-10-14 17:37:39 -070049 return [s[:-len(".proto")] + "_pb2.py" for s in srcs]
Jisi Liu39362b32015-10-14 17:12:11 -070050
David Z. Chen02cd45c2016-05-20 16:49:04 -070051def _RelativeOutputPath(path, include, dest=""):
Jisi Liu993fb702015-10-19 17:19:49 -070052 if include == None:
53 return path
54
55 if not path.startswith(include):
56 fail("Include path %s isn't part of the path %s." % (include, path))
57
58 if include and include[-1] != '/':
59 include = include + '/'
David Z. Chen02cd45c2016-05-20 16:49:04 -070060 if dest and dest[-1] != '/':
61 dest = dest + '/'
Jisi Liu993fb702015-10-19 17:19:49 -070062
63 path = path[len(include):]
David Z. Chen02cd45c2016-05-20 16:49:04 -070064 return dest + path
Jisi Liu993fb702015-10-19 17:19:49 -070065
Jisi Liu9c7d9c02015-10-15 10:51:32 -070066def _proto_gen_impl(ctx):
67 """General implementation for generating protos"""
Jisi Liu39362b32015-10-14 17:12:11 -070068 srcs = ctx.files.srcs
69 deps = []
70 deps += ctx.files.srcs
Kristina Chodorow4e7ecde2017-01-25 14:10:56 -050071 source_dir = _SourceDir(ctx)
Jisi Liu993fb702015-10-19 17:19:49 -070072 gen_dir = _GenDir(ctx)
Kristina Chodorow4e7ecde2017-01-25 14:10:56 -050073 if source_dir:
74 import_flags = ["-I" + source_dir, "-I" + gen_dir]
Jisi Liu53a56be2015-10-20 15:18:20 -070075 else:
76 import_flags = ["-I."]
77
Jisi Liu39362b32015-10-14 17:12:11 -070078 for dep in ctx.attr.deps:
79 import_flags += dep.proto.import_flags
80 deps += dep.proto.deps
81
82 args = []
83 if ctx.attr.gen_cc:
Kristina Chodorow4e7ecde2017-01-25 14:10:56 -050084 args += ["--cpp_out=" + gen_dir]
Jisi Liu39362b32015-10-14 17:12:11 -070085 if ctx.attr.gen_py:
Kristina Chodorow4e7ecde2017-01-25 14:10:56 -050086 args += ["--python_out=" + gen_dir]
Jisi Liu39362b32015-10-14 17:12:11 -070087
Florian Weikertc2b3e702016-10-12 14:03:07 +020088 inputs = srcs + deps
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +090089 if ctx.executable.plugin:
90 plugin = ctx.executable.plugin
91 lang = ctx.attr.plugin_language
92 if not lang and plugin.basename.startswith('protoc-gen-'):
93 lang = plugin.basename[len('protoc-gen-'):]
94 if not lang:
95 fail("cannot infer the target language of plugin", "plugin_language")
96
Kristina Chodorow4e7ecde2017-01-25 14:10:56 -050097 outdir = gen_dir
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +090098 if ctx.attr.plugin_options:
99 outdir = ",".join(ctx.attr.plugin_options) + ":" + outdir
100 args += ["--plugin=protoc-gen-%s=%s" % (lang, plugin.path)]
101 args += ["--%s_out=%s" % (lang, outdir)]
Florian Weikertc2b3e702016-10-12 14:03:07 +0200102 inputs += [plugin]
Manjunath Kudlurf0966a72016-02-22 14:30:43 -0800103
Jisi Liu39362b32015-10-14 17:12:11 -0700104 if args:
105 ctx.action(
Florian Weikertc2b3e702016-10-12 14:03:07 +0200106 inputs=inputs,
Jisi Liu39362b32015-10-14 17:12:11 -0700107 outputs=ctx.outputs.outs,
Jisi Liu125a91b2015-10-14 17:37:39 -0700108 arguments=args + import_flags + [s.path for s in srcs],
Jisi Liu9c7d9c02015-10-15 10:51:32 -0700109 executable=ctx.executable.protoc,
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900110 mnemonic="ProtoCompile",
Jisi Liu39362b32015-10-14 17:12:11 -0700111 )
112
113 return struct(
114 proto=struct(
Jisi Liu125a91b2015-10-14 17:37:39 -0700115 srcs=srcs,
116 import_flags=import_flags,
117 deps=deps,
118 ),
119 )
Jisi Liu39362b32015-10-14 17:12:11 -0700120
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900121proto_gen = rule(
Jisi Liu39362b32015-10-14 17:12:11 -0700122 attrs = {
Jisi Liuee8131a2015-10-14 17:20:05 -0700123 "srcs": attr.label_list(allow_files = True),
124 "deps": attr.label_list(providers = ["proto"]),
Jisi Liu53a56be2015-10-20 15:18:20 -0700125 "includes": attr.string_list(),
Jisi Liuee8131a2015-10-14 17:20:05 -0700126 "protoc": attr.label(
Vladimir Moskvaa86e6d82016-09-09 13:21:35 +0200127 cfg = "host",
Jisi Liuee8131a2015-10-14 17:20:05 -0700128 executable = True,
129 single_file = True,
130 mandatory = True,
131 ),
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900132 "plugin": attr.label(
Vladimir Moskvaa86e6d82016-09-09 13:21:35 +0200133 cfg = "host",
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900134 allow_files = True,
Manjunath Kudlurf0966a72016-02-22 14:30:43 -0800135 executable = True,
Manjunath Kudlurf0966a72016-02-22 14:30:43 -0800136 ),
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900137 "plugin_language": attr.string(),
138 "plugin_options": attr.string_list(),
Jisi Liuee8131a2015-10-14 17:20:05 -0700139 "gen_cc": attr.bool(),
140 "gen_py": attr.bool(),
141 "outs": attr.output_list(),
142 },
143 output_to_genfiles = True,
Jisi Liu9c7d9c02015-10-15 10:51:32 -0700144 implementation = _proto_gen_impl,
Jisi Liu39362b32015-10-14 17:12:11 -0700145)
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900146"""Generates codes from Protocol Buffers definitions.
147
148This rule helps you to implement Skylark macros specific to the target
149language. You should prefer more specific `cc_proto_library `,
150`py_proto_library` and others unless you are adding such wrapper macros.
151
152Args:
153 srcs: Protocol Buffers definition files (.proto) to run the protocol compiler
154 against.
155 deps: a list of dependency labels; must be other proto libraries.
156 includes: a list of include paths to .proto files.
157 protoc: the label of the protocol compiler to generate the sources.
158 plugin: the label of the protocol compiler plugin to be passed to the protocol
159 compiler.
160 plugin_language: the language of the generated sources
161 plugin_options: a list of options to be passed to the plugin
Kristina Chodorow4e7ecde2017-01-25 14:10:56 -0500162 gen_cc: generates C++ sources in addition to the ones from the plugin.
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900163 gen_py: generates Python sources in addition to the ones from the plugin.
164 outs: a list of labels of the expected outputs from the protocol compiler.
165"""
Jisi Liu39362b32015-10-14 17:12:11 -0700166
167def cc_proto_library(
Jisi Liu125a91b2015-10-14 17:37:39 -0700168 name,
169 srcs=[],
Jisi Liu125a91b2015-10-14 17:37:39 -0700170 deps=[],
Jisi Liud8701b52015-10-16 11:44:21 -0700171 cc_libs=[],
Jisi Liu6dac0822015-10-19 14:41:00 -0700172 include=None,
David Z. Chen985c9682016-02-11 18:11:10 -0800173 protoc="//:protoc",
Jisi Liu3101e732015-10-16 12:46:26 -0700174 internal_bootstrap_hack=False,
Manjunath Kudlurf0966a72016-02-22 14:30:43 -0800175 use_grpc_plugin=False,
David Z. Chen985c9682016-02-11 18:11:10 -0800176 default_runtime="//:protobuf",
Jisi Liu125a91b2015-10-14 17:37:39 -0700177 **kargs):
Jisi Liu3101e732015-10-16 12:46:26 -0700178 """Bazel rule to create a C++ protobuf library from proto source files
179
Jisi Liud4bef7d2015-11-02 12:24:32 -0800180 NOTE: the rule is only an internal workaround to generate protos. The
181 interface may change and the rule may be removed when bazel has introduced
182 the native rule.
183
Jisi Liu3101e732015-10-16 12:46:26 -0700184 Args:
185 name: the name of the cc_proto_library.
186 srcs: the .proto files of the cc_proto_library.
187 deps: a list of dependency labels; must be cc_proto_library.
188 cc_libs: a list of other cc_library targets depended by the generated
189 cc_library.
190 include: a string indicating the include path of the .proto files.
191 protoc: the label of the protocol compiler to generate the sources.
192 internal_bootstrap_hack: a flag indicate the cc_proto_library is used only
193 for bootstraping. When it is set to True, no files will be generated.
194 The rule will simply be a provider for .proto files, so that other
195 cc_proto_library can depend on it.
Manjunath Kudlurf0966a72016-02-22 14:30:43 -0800196 use_grpc_plugin: a flag to indicate whether to call the grpc C++ plugin
197 when processing the proto files.
Jisi Liube92ffb2015-10-27 15:11:38 -0700198 default_runtime: the implicitly default runtime which will be depended on by
199 the generated cc_library target.
Jisi Liu3101e732015-10-16 12:46:26 -0700200 **kargs: other keyword arguments that are passed to cc_library.
201
202 """
Jisi Liu39362b32015-10-14 17:12:11 -0700203
Jisi Liu53a56be2015-10-20 15:18:20 -0700204 includes = []
205 if include != None:
206 includes = [include]
207
Jisi Liu39362b32015-10-14 17:12:11 -0700208 if internal_bootstrap_hack:
209 # For pre-checked-in generated files, we add the internal_bootstrap_hack
210 # which will skip the codegen action.
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900211 proto_gen(
Jisi Liu125a91b2015-10-14 17:37:39 -0700212 name=name + "_genproto",
213 srcs=srcs,
Jisi Liud8701b52015-10-16 11:44:21 -0700214 deps=[s + "_genproto" for s in deps],
Jisi Liu53a56be2015-10-20 15:18:20 -0700215 includes=includes,
Jisi Liu125a91b2015-10-14 17:37:39 -0700216 protoc=protoc,
Martin Maly8e0c9a32015-12-04 17:44:58 -0800217 visibility=["//visibility:public"],
Jisi Liu39362b32015-10-14 17:12:11 -0700218 )
219 # An empty cc_library to make rule dependency consistent.
220 native.cc_library(
Jisi Liu125a91b2015-10-14 17:37:39 -0700221 name=name,
Jisi Liud8701b52015-10-16 11:44:21 -0700222 **kargs)
Jisi Liu39362b32015-10-14 17:12:11 -0700223 return
224
Manjunath Kudlurf0966a72016-02-22 14:30:43 -0800225 grpc_cpp_plugin = None
226 if use_grpc_plugin:
227 grpc_cpp_plugin = "//external:grpc_cpp_plugin"
228
Andreas Bergmeierbbeb9832016-08-15 16:57:30 +0200229 gen_srcs = _CcSrcs(srcs, use_grpc_plugin)
230 gen_hdrs = _CcHdrs(srcs, use_grpc_plugin)
231 outs = gen_srcs + gen_hdrs
Manjunath Kudlurf5c73632016-02-25 08:50:50 -0800232
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900233 proto_gen(
Jisi Liu125a91b2015-10-14 17:37:39 -0700234 name=name + "_genproto",
235 srcs=srcs,
Jisi Liud8701b52015-10-16 11:44:21 -0700236 deps=[s + "_genproto" for s in deps],
Jisi Liu53a56be2015-10-20 15:18:20 -0700237 includes=includes,
Jisi Liu125a91b2015-10-14 17:37:39 -0700238 protoc=protoc,
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900239 plugin=grpc_cpp_plugin,
240 plugin_language="grpc",
Jisi Liu125a91b2015-10-14 17:37:39 -0700241 gen_cc=1,
242 outs=outs,
Martin Maly8e0c9a32015-12-04 17:44:58 -0800243 visibility=["//visibility:public"],
Jisi Liu39362b32015-10-14 17:12:11 -0700244 )
245
Jisi Liube92ffb2015-10-27 15:11:38 -0700246 if default_runtime and not default_runtime in cc_libs:
247 cc_libs += [default_runtime]
Manjunath Kudlurf5c73632016-02-25 08:50:50 -0800248 if use_grpc_plugin:
249 cc_libs += ["//external:grpc_lib"]
Jisi Liu6dac0822015-10-19 14:41:00 -0700250
Jisi Liu39362b32015-10-14 17:12:11 -0700251 native.cc_library(
Jisi Liu125a91b2015-10-14 17:37:39 -0700252 name=name,
Andreas Bergmeierbbeb9832016-08-15 16:57:30 +0200253 srcs=gen_srcs,
254 hdrs=gen_hdrs,
Jisi Liud8701b52015-10-16 11:44:21 -0700255 deps=cc_libs + deps,
Jisi Liu6dac0822015-10-19 14:41:00 -0700256 includes=includes,
Jisi Liud8701b52015-10-16 11:44:21 -0700257 **kargs)
Jisi Liu993fb702015-10-19 17:19:49 -0700258
Steven Parkesea188662016-02-25 07:53:19 -0800259def internal_gen_well_known_protos_java(srcs):
260 """Bazel rule to generate the gen_well_known_protos_java genrule
261
262 Args:
263 srcs: the well known protos
264 """
265 root = Label("%s//protobuf_java" % (REPOSITORY_NAME)).workspace_root
cgrushko6fffd4a2017-02-08 12:19:40 -0500266 pkg = PACKAGE_NAME + "/" if PACKAGE_NAME else ""
Steven Parkesea188662016-02-25 07:53:19 -0800267 if root == "":
cgrushko6fffd4a2017-02-08 12:19:40 -0500268 include = " -I%ssrc " % pkg
Steven Parkesea188662016-02-25 07:53:19 -0800269 else:
cgrushko6fffd4a2017-02-08 12:19:40 -0500270 include = " -I%s/%ssrc " % (root, pkg)
Steven Parkesea188662016-02-25 07:53:19 -0800271 native.genrule(
272 name = "gen_well_known_protos_java",
273 srcs = srcs,
274 outs = [
275 "wellknown.srcjar",
276 ],
277 cmd = "$(location :protoc) --java_out=$(@D)/wellknown.jar" +
278 " %s $(SRCS) " % include +
279 " && mv $(@D)/wellknown.jar $(@D)/wellknown.srcjar",
280 tools = [":protoc"],
281 )
282
David Z. Chen02cd45c2016-05-20 16:49:04 -0700283def internal_copied_filegroup(name, srcs, strip_prefix, dest, **kwargs):
284 """Macro to copy files to a different directory and then create a filegroup.
285
286 This is used by the //:protobuf_python py_proto_library target to work around
287 an issue caused by Python source files that are part of the same Python
288 package being in separate directories.
289
290 Args:
291 srcs: The source files to copy and add to the filegroup.
292 strip_prefix: Path to the root of the files to copy.
293 dest: The directory to copy the source files into.
294 **kwargs: extra arguments that will be passesd to the filegroup.
295 """
296 outs = [_RelativeOutputPath(s, strip_prefix, dest) for s in srcs]
297
298 native.genrule(
299 name = name + "_genrule",
300 srcs = srcs,
301 outs = outs,
302 cmd = " && ".join(
303 ["cp $(location %s) $(location %s)" %
304 (s, _RelativeOutputPath(s, strip_prefix, dest)) for s in srcs]),
305 )
306
307 native.filegroup(
308 name = name,
309 srcs = outs,
310 **kwargs)
311
Jisi Liu993fb702015-10-19 17:19:49 -0700312def py_proto_library(
313 name,
314 srcs=[],
315 deps=[],
316 py_libs=[],
317 py_extra_srcs=[],
318 include=None,
David Z. Chen985c9682016-02-11 18:11:10 -0800319 default_runtime="//:protobuf_python",
320 protoc="//:protoc",
Wiktor Tomczak0fa31b22016-11-22 20:18:46 +0100321 use_grpc_plugin=False,
Jisi Liu993fb702015-10-19 17:19:49 -0700322 **kargs):
Jisi Liu7b948cc2015-10-19 17:56:27 -0700323 """Bazel rule to create a Python protobuf library from proto source files
324
Jisi Liud4bef7d2015-11-02 12:24:32 -0800325 NOTE: the rule is only an internal workaround to generate protos. The
326 interface may change and the rule may be removed when bazel has introduced
327 the native rule.
328
Jisi Liu7b948cc2015-10-19 17:56:27 -0700329 Args:
330 name: the name of the py_proto_library.
331 srcs: the .proto files of the py_proto_library.
332 deps: a list of dependency labels; must be py_proto_library.
333 py_libs: a list of other py_library targets depended by the generated
334 py_library.
335 py_extra_srcs: extra source files that will be added to the output
336 py_library. This attribute is used for internal bootstrapping.
337 include: a string indicating the include path of the .proto files.
Jisi Liube92ffb2015-10-27 15:11:38 -0700338 default_runtime: the implicitly default runtime which will be depended on by
339 the generated py_library target.
Jisi Liu7b948cc2015-10-19 17:56:27 -0700340 protoc: the label of the protocol compiler to generate the sources.
Wiktor Tomczak0fa31b22016-11-22 20:18:46 +0100341 use_grpc_plugin: a flag to indicate whether to call the Python C++ plugin
342 when processing the proto files.
Jisi Liu7b948cc2015-10-19 17:56:27 -0700343 **kargs: other keyword arguments that are passed to cc_library.
344
345 """
Jisi Liu993fb702015-10-19 17:19:49 -0700346 outs = _PyOuts(srcs)
Jisi Liu53a56be2015-10-20 15:18:20 -0700347
348 includes = []
349 if include != None:
350 includes = [include]
351
Wiktor Tomczak0fa31b22016-11-22 20:18:46 +0100352 grpc_python_plugin = None
353 if use_grpc_plugin:
354 grpc_python_plugin = "//external:grpc_python_plugin"
355 # Note: Generated grpc code depends on Python grpc module. This dependency
356 # is not explicitly listed in py_libs. Instead, host system is assumed to
357 # have grpc installed.
358
Yuki Yugui Sonoda5977fb02016-06-01 16:23:15 +0900359 proto_gen(
Jisi Liu993fb702015-10-19 17:19:49 -0700360 name=name + "_genproto",
361 srcs=srcs,
362 deps=[s + "_genproto" for s in deps],
Jisi Liu53a56be2015-10-20 15:18:20 -0700363 includes=includes,
Jisi Liu993fb702015-10-19 17:19:49 -0700364 protoc=protoc,
365 gen_py=1,
366 outs=outs,
Martin Maly8e0c9a32015-12-04 17:44:58 -0800367 visibility=["//visibility:public"],
Wiktor Tomczak0fa31b22016-11-22 20:18:46 +0100368 plugin=grpc_python_plugin,
369 plugin_language="grpc"
Jisi Liu993fb702015-10-19 17:19:49 -0700370 )
371
Jisi Liube92ffb2015-10-27 15:11:38 -0700372 if default_runtime and not default_runtime in py_libs + deps:
373 py_libs += [default_runtime]
374
Jisi Liu993fb702015-10-19 17:19:49 -0700375 native.py_library(
376 name=name,
Jisi Liua33fa8e2015-10-20 15:30:44 -0700377 srcs=outs+py_extra_srcs,
378 deps=py_libs+deps,
David Z. Chen985c9682016-02-11 18:11:10 -0800379 imports=includes,
Jisi Liu993fb702015-10-19 17:19:49 -0700380 **kargs)
381
382def internal_protobuf_py_tests(
383 name,
384 modules=[],
385 **kargs):
Jisi Liu7b948cc2015-10-19 17:56:27 -0700386 """Bazel rules to create batch tests for protobuf internal.
387
388 Args:
389 name: the name of the rule.
390 modules: a list of modules for tests. The macro will create a py_test for
391 each of the parameter with the source "google/protobuf/%s.py"
392 kargs: extra parameters that will be passed into the py_test.
393
394 """
Jisi Liu993fb702015-10-19 17:19:49 -0700395 for m in modules:
David Z. Chen985c9682016-02-11 18:11:10 -0800396 s = "python/google/protobuf/internal/%s.py" % m
Jisi Liu993fb702015-10-19 17:19:49 -0700397 native.py_test(
398 name="py_%s" % m,
Jisi Liu7b948cc2015-10-19 17:56:27 -0700399 srcs=[s],
400 main=s,
Jisi Liu993fb702015-10-19 17:19:49 -0700401 **kargs)