如何使用Bazel Build从生成的源文件构建静态库

Man*_*ool 4 build bazel

我试图使用Bazel进行以下设置。通过调用“ bazel build”,Python脚本应该生成未知数量的带有随机名称的* .cc文件,然后将这些文件编译到单个静态库(.a文件)中,所有这些都在一个Bazel调用中完成。我尝试了以下操作:一个生成的文件具有固定名称,该文件在genrule()和cc_library规则的srcs中引用。问题是我需要将所有生成的文件构建为库,而不仅仅是具有固定名称的文件。任何想法如何做到这一点?我的BUILD文件:

py_binary(
    name = "sample_script",
    srcs = ["sample_script.py"],
)
genrule(
    name = "sample_genrule",
    tools = [":sample_script"],
    cmd = "$(location :sample_script)",
    outs = ["cpp_output_fixed.cc"], #or shall also the files with random names be defined here?
)
cc_library(
    name = "autolib",
    srcs = ["cpp_output_fixed.cc"],
    #srcs = glob([ #here should all generated .cc files be named
    #    "./*.cc",
    #    "./**/*.cc",
    #    ])+["cpp_output_fixed.cc"], 
)
Run Code Online (Sandbox Code Playgroud)

Python文件sample_script.py:

#!/usr/bin/env python
import hashlib
import time

time_stamp = time.time()

time_1 = str(time_stamp)
time_2 = str(time_stamp + 1)

random_part_1 = hashlib.sha1(time_1).hexdigest()[-4:]
random_part_2 = hashlib.sha1(time_1).hexdigest()[-4:]

fixed_file = "cpp_output_fixed" + ".cc"
file_1 = "cpp_output_" + random_part_1 + ".cc"
file_2 = "cpp_output3_" + random_part_2 + ".cc"

with open(fixed_file, "w") as outfile:
    outfile.write("#include <iostream>"
                   "int main() {"
                   "  std::cout <<'''Hello_world'''    <<std::endl;"
                   "  return 0"
                   "}")

with open(file_1, "w") as outfile:
    outfile.write("#include <iostream>"
                   "int main() {"
                   "  std::cout <<'''Hello_world'''    <<std::endl;"
                   "  return 0"
                   "}")

with open(file_2, "w") as outfile_new:
    outfile_new.write("#include <iostream>"
                   "int main() {"
                   "  std::cout <<'''Hello_world''' <<std::endl;"
                   "  return 0"
                   "}")

print ".cc generation DONE"
Run Code Online (Sandbox Code Playgroud)

hlo*_*pko 7

[大修改,因为我找到了使它起作用的方法:)]

如果您确实需要发出在分析阶段未知的文件,则唯一的方法就是我们内部称为树状工件的方法。您可以将其视为包含仅在执行阶段进行检查的文件的目录。您可以使用ctx.actions.declare_directory从Skylark声明树状工件。

这是一个工作示例。注意3件事:

  • 我们需要在目录名称中添加“ .cc”,以愚弄这是有效输入的C ++规则
  • 生成器需要创建bazel告诉它的目录
  • 您需要使用bazel @ HEAD(或bazel 0.11.0及更高版本)

genccs.bzl:

def _impl(ctx):
  tree = ctx.actions.declare_directory(ctx.attr.name + ".cc")
  ctx.actions.run(
    inputs = [],
    outputs = [ tree ],
    arguments = [ tree.path ],
    progress_message = "Generating cc files into '%s'" % tree.path,
    executable = ctx.executable._tool,
  )

  return [ DefaultInfo(files = depset([ tree ])) ]

genccs = rule(
  implementation = _impl,
  attrs = {
    "_tool": attr.label(
      executable = True,
      cfg = "host",
      allow_files = True,
      default = Label("//:genccs"),
    )
  }
)
Run Code Online (Sandbox Code Playgroud)

建立:

load(":genccs.bzl", "genccs")

genccs(
    name = "gen_tree",
)

cc_library(
    name = "main",
    srcs = [ "gen_tree" ]
)

cc_binary(
    name = "genccs",
    srcs = [ "genccs.cpp" ],
)
Run Code Online (Sandbox Code Playgroud)

genccs.cpp

#include <fstream>
#include <sys/stat.h>
using namespace std;

int main (int argc, char *argv[]) {
  mkdir(argv[1], S_IRWXU);
  ofstream myfile;
  myfile.open(string(argv[1]) + string("/foo.cpp"));
  myfile << "int main() { return 42; }";
  return 0;
}
Run Code Online (Sandbox Code Playgroud)