| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417 | def _GetPath(ctx, path):  if ctx.label.workspace_root:    return ctx.label.workspace_root + '/' + path  else:    return pathdef _IsNewExternal(ctx):  # Bazel 0.4.4 and older have genfiles paths that look like:  #   bazel-out/local-fastbuild/genfiles/external/repo/foo  # After the exec root rearrangement, they look like:  #   ../repo/bazel-out/local-fastbuild/genfiles/foo  return ctx.label.workspace_root.startswith("../")def _GenDir(ctx):  if _IsNewExternal(ctx):    # We are using the fact that Bazel 0.4.4+ provides repository-relative paths    # for ctx.genfiles_dir.    return ctx.genfiles_dir.path + (        "/" + ctx.attr.includes[0] if ctx.attr.includes and ctx.attr.includes[0] else "")  # This means that we're either in the old version OR the new version in the local repo.  # Either way, appending the source path to the genfiles dir works.  return ctx.var["GENDIR"] + "/" + _SourceDir(ctx)def _SourceDir(ctx):  if not ctx.attr.includes:    return ctx.label.workspace_root  if not ctx.attr.includes[0]:    return _GetPath(ctx, ctx.label.package)  if not ctx.label.package:    return _GetPath(ctx, ctx.attr.includes[0])  return _GetPath(ctx, ctx.label.package + '/' + ctx.attr.includes[0])def _CcHdrs(srcs, use_grpc_plugin=False):  ret = [s[:-len(".proto")] + ".pb.h" for s in srcs]  if use_grpc_plugin:    ret += [s[:-len(".proto")] + ".grpc.pb.h" for s in srcs]  return retdef _CcSrcs(srcs, use_grpc_plugin=False):  ret = [s[:-len(".proto")] + ".pb.cc" for s in srcs]  if use_grpc_plugin:    ret += [s[:-len(".proto")] + ".grpc.pb.cc" for s in srcs]  return retdef _CcOuts(srcs, use_grpc_plugin=False):  return _CcHdrs(srcs, use_grpc_plugin) + _CcSrcs(srcs, use_grpc_plugin)def _PyOuts(srcs, use_grpc_plugin=False):  ret = [s[:-len(".proto")] + "_pb2.py" for s in srcs]  if use_grpc_plugin:    ret += [s[:-len(".proto")] + "_pb2_grpc.py" for s in srcs]  return retdef _RelativeOutputPath(path, include, dest=""):  if include == None:    return path  if not path.startswith(include):    fail("Include path %s isn't part of the path %s." % (include, path))  if include and include[-1] != '/':    include = include + '/'  if dest and dest[-1] != '/':    dest = dest + '/'  path = path[len(include):]  return dest + pathdef _proto_gen_impl(ctx):  """General implementation for generating protos"""  srcs = ctx.files.srcs  deps = []  deps += ctx.files.srcs  source_dir = _SourceDir(ctx)  gen_dir = _GenDir(ctx)  if source_dir:    import_flags = ["-I" + source_dir, "-I" + gen_dir]  else:    import_flags = ["-I."]  for dep in ctx.attr.deps:    import_flags += dep.proto.import_flags    deps += dep.proto.deps  args = []  if ctx.attr.gen_cc:    args += ["--cpp_out=" + gen_dir]  if ctx.attr.gen_py:    args += ["--python_out=" + gen_dir]  inputs = srcs + deps  if ctx.executable.plugin:    plugin = ctx.executable.plugin    lang = ctx.attr.plugin_language    if not lang and plugin.basename.startswith('protoc-gen-'):      lang = plugin.basename[len('protoc-gen-'):]    if not lang:      fail("cannot infer the target language of plugin", "plugin_language")    outdir = gen_dir    if ctx.attr.plugin_options:      outdir = ",".join(ctx.attr.plugin_options) + ":" + outdir    args += ["--plugin=protoc-gen-%s=%s" % (lang, plugin.path)]    args += ["--%s_out=%s" % (lang, outdir)]    inputs += [plugin]  if args:    ctx.action(        inputs=inputs,        outputs=ctx.outputs.outs,        arguments=args + import_flags + [s.path for s in srcs],        executable=ctx.executable.protoc,        mnemonic="ProtoCompile",        use_default_shell_env=True,    )  return struct(      proto=struct(          srcs=srcs,          import_flags=import_flags,          deps=deps,      ),  )proto_gen = rule(    attrs = {        "srcs": attr.label_list(allow_files = True),        "deps": attr.label_list(providers = ["proto"]),        "includes": attr.string_list(),        "protoc": attr.label(            cfg = "host",            executable = True,            single_file = True,            mandatory = True,        ),        "plugin": attr.label(            cfg = "host",            allow_files = True,            executable = True,        ),        "plugin_language": attr.string(),        "plugin_options": attr.string_list(),        "gen_cc": attr.bool(),        "gen_py": attr.bool(),        "outs": attr.output_list(),    },    output_to_genfiles = True,    implementation = _proto_gen_impl,)"""Generates codes from Protocol Buffers definitions.This rule helps you to implement Skylark macros specific to the targetlanguage. You should prefer more specific `cc_proto_library `,`py_proto_library` and others unless you are adding such wrapper macros.Args:  srcs: Protocol Buffers definition files (.proto) to run the protocol compiler    against.  deps: a list of dependency labels; must be other proto libraries.  includes: a list of include paths to .proto files.  protoc: the label of the protocol compiler to generate the sources.  plugin: the label of the protocol compiler plugin to be passed to the protocol    compiler.  plugin_language: the language of the generated sources  plugin_options: a list of options to be passed to the plugin  gen_cc: generates C++ sources in addition to the ones from the plugin.  gen_py: generates Python sources in addition to the ones from the plugin.  outs: a list of labels of the expected outputs from the protocol compiler."""def cc_proto_library(        name,        srcs=[],        deps=[],        cc_libs=[],        include=None,        protoc="@com_google_protobuf//:protoc",        internal_bootstrap_hack=False,        use_grpc_plugin=False,        default_runtime="@com_google_protobuf//:protobuf",        **kargs):  """Bazel rule to create a C++ protobuf library from proto source files  NOTE: the rule is only an internal workaround to generate protos. The  interface may change and the rule may be removed when bazel has introduced  the native rule.  Args:    name: the name of the cc_proto_library.    srcs: the .proto files of the cc_proto_library.    deps: a list of dependency labels; must be cc_proto_library.    cc_libs: a list of other cc_library targets depended by the generated        cc_library.    include: a string indicating the include path of the .proto files.    protoc: the label of the protocol compiler to generate the sources.    internal_bootstrap_hack: a flag indicate the cc_proto_library is used only        for bootstraping. When it is set to True, no files will be generated.        The rule will simply be a provider for .proto files, so that other        cc_proto_library can depend on it.    use_grpc_plugin: a flag to indicate whether to call the grpc C++ plugin        when processing the proto files.    default_runtime: the implicitly default runtime which will be depended on by        the generated cc_library target.    **kargs: other keyword arguments that are passed to cc_library.  """  includes = []  if include != None:    includes = [include]  if internal_bootstrap_hack:    # For pre-checked-in generated files, we add the internal_bootstrap_hack    # which will skip the codegen action.    proto_gen(        name=name + "_genproto",        srcs=srcs,        deps=[s + "_genproto" for s in deps],        includes=includes,        protoc=protoc,        visibility=["//visibility:public"],    )    # An empty cc_library to make rule dependency consistent.    native.cc_library(        name=name,        **kargs)    return  grpc_cpp_plugin = None  if use_grpc_plugin:    grpc_cpp_plugin = "//external:grpc_cpp_plugin"  gen_srcs = _CcSrcs(srcs, use_grpc_plugin)  gen_hdrs = _CcHdrs(srcs, use_grpc_plugin)  outs = gen_srcs + gen_hdrs  proto_gen(      name=name + "_genproto",      srcs=srcs,      deps=[s + "_genproto" for s in deps],      includes=includes,      protoc=protoc,      plugin=grpc_cpp_plugin,      plugin_language="grpc",      gen_cc=1,      outs=outs,      visibility=["//visibility:public"],  )  if default_runtime and not default_runtime in cc_libs:    cc_libs = cc_libs + [default_runtime]  if use_grpc_plugin:    cc_libs = cc_libs + ["//external:grpc_lib"]  native.cc_library(      name=name,      srcs=gen_srcs,      hdrs=gen_hdrs,      deps=cc_libs + deps,      includes=includes,      **kargs)def internal_gen_well_known_protos_java(srcs):  """Bazel rule to generate the gen_well_known_protos_java genrule  Args:    srcs: the well known protos  """  root = Label("%s//protobuf_java" % (REPOSITORY_NAME)).workspace_root  pkg = PACKAGE_NAME + "/" if PACKAGE_NAME else ""  if root == "":    include = " -I%ssrc " % pkg  else:    include = " -I%s/%ssrc " % (root, pkg)  native.genrule(    name = "gen_well_known_protos_java",    srcs = srcs,    outs = [        "wellknown.srcjar",    ],    cmd = "$(location :protoc) --java_out=$(@D)/wellknown.jar" +          " %s $(SRCS) " % include +          " && mv $(@D)/wellknown.jar $(@D)/wellknown.srcjar",    tools = [":protoc"],  )def internal_copied_filegroup(name, srcs, strip_prefix, dest, **kwargs):  """Macro to copy files to a different directory and then create a filegroup.  This is used by the //:protobuf_python py_proto_library target to work around  an issue caused by Python source files that are part of the same Python  package being in separate directories.  Args:    srcs: The source files to copy and add to the filegroup.    strip_prefix: Path to the root of the files to copy.    dest: The directory to copy the source files into.    **kwargs: extra arguments that will be passesd to the filegroup.  """  outs = [_RelativeOutputPath(s, strip_prefix, dest) for s in srcs]  native.genrule(      name = name + "_genrule",      srcs = srcs,      outs = outs,      cmd = " && ".join(          ["cp $(location %s) $(location %s)" %           (s, _RelativeOutputPath(s, strip_prefix, dest)) for s in srcs]),  )  native.filegroup(      name = name,      srcs = outs,      **kwargs)def py_proto_library(        name,        srcs=[],        deps=[],        py_libs=[],        py_extra_srcs=[],        include=None,        default_runtime="@com_google_protobuf//:protobuf_python",        protoc="@com_google_protobuf//:protoc",        use_grpc_plugin=False,        **kargs):  """Bazel rule to create a Python protobuf library from proto source files  NOTE: the rule is only an internal workaround to generate protos. The  interface may change and the rule may be removed when bazel has introduced  the native rule.  Args:    name: the name of the py_proto_library.    srcs: the .proto files of the py_proto_library.    deps: a list of dependency labels; must be py_proto_library.    py_libs: a list of other py_library targets depended by the generated        py_library.    py_extra_srcs: extra source files that will be added to the output        py_library. This attribute is used for internal bootstrapping.    include: a string indicating the include path of the .proto files.    default_runtime: the implicitly default runtime which will be depended on by        the generated py_library target.    protoc: the label of the protocol compiler to generate the sources.    use_grpc_plugin: a flag to indicate whether to call the Python C++ plugin        when processing the proto files.    **kargs: other keyword arguments that are passed to cc_library.  """  outs = _PyOuts(srcs, use_grpc_plugin)  includes = []  if include != None:    includes = [include]  grpc_python_plugin = None  if use_grpc_plugin:    grpc_python_plugin = "//external:grpc_python_plugin"    # Note: Generated grpc code depends on Python grpc module. This dependency    # is not explicitly listed in py_libs. Instead, host system is assumed to    # have grpc installed.  proto_gen(      name=name + "_genproto",      srcs=srcs,      deps=[s + "_genproto" for s in deps],      includes=includes,      protoc=protoc,      gen_py=1,      outs=outs,      visibility=["//visibility:public"],      plugin=grpc_python_plugin,      plugin_language="grpc"  )  if default_runtime and not default_runtime in py_libs + deps:    py_libs = py_libs + [default_runtime]  native.py_library(      name=name,      srcs=outs+py_extra_srcs,      deps=py_libs+deps,      imports=includes,      **kargs)def internal_protobuf_py_tests(    name,    modules=[],    **kargs):  """Bazel rules to create batch tests for protobuf internal.  Args:    name: the name of the rule.    modules: a list of modules for tests. The macro will create a py_test for        each of the parameter with the source "google/protobuf/%s.py"    kargs: extra parameters that will be passed into the py_test.  """  for m in modules:    s = "python/google/protobuf/internal/%s.py" % m    native.py_test(        name="py_%s" % m,        srcs=[s],        main=s,        **kargs)def check_protobuf_required_bazel_version():  """For WORKSPACE files, to check the installed version of bazel.  This ensures bazel supports our approach to proto_library() depending on a  copied filegroup. (Fixed in bazel 0.5.4)  """  expected = apple_common.dotted_version("0.5.4")  current = apple_common.dotted_version(native.bazel_version)  if current.compare_to(expected) < 0:    fail("Bazel must be newer than 0.5.4")
 |