# Copyright 2014 The Bazel Authors. All rights reserved. # # 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. ZIP_PATH = "/usr/bin/zip" py_file_types = FileType([".py"]) def collect_transitive_sources(ctx): source_files = set(order="compile") for dep in ctx.attr.deps: source_files += dep.transitive_py_files source_files += py_file_types.filter(ctx.files.srcs) return source_files def py_library_impl(ctx): transitive_sources = collect_transitive_sources(ctx) return struct( files = set(), transitive_py_files = transitive_sources) def py_binary_impl(ctx): main_file = py_file_types.filter(ctx.files.srcs)[0] transitive_sources = collect_transitive_sources(ctx) deploy_zip = ctx.outputs.deploy_zip deploy_zip_nomain = ctx.new_file(ctx.bin_dir, deploy_zip, ".nomain.zip") # This is not very scalable, because we just construct a huge string instead # of using a nested set. We need to do it this way because Skylark currently # does not support actions with non-artifact executables but with an # argument list (instead of just a single command) command = " ".join([ZIP_PATH, " -q ", deploy_zip_nomain.path, " ".join([f.path for f in transitive_sources])]) ctx.action( inputs = list(transitive_sources), outputs = [ deploy_zip_nomain ], mnemonic = "PyZip", command = command, use_default_shell_env = False) dirs = [f.path[:f.path.rfind('/')] for f in transitive_sources] outdir = deploy_zip.path + ".out" # Add __init__.py files and the __main__.py driver. main_cmd = ("mkdir -p %s && " % outdir + " cp %s %s/__main__.py && " % (main_file.path, outdir) + " cp %s %s/main.zip && " % (deploy_zip_nomain.path, outdir) + " (cd %s && " % outdir + " mkdir -p %s && " % " ".join(dirs) + " find . -type d -exec touch -t 198001010000 '{}'/__init__.py ';' && " + " chmod +w main.zip && " + " %s -quR main.zip $(find . -type f ) ) && " % (ZIP_PATH) + " mv %s/main.zip %s " % (outdir, deploy_zip.path)) ctx.action( inputs = [ deploy_zip_nomain, main_file ], outputs = [ deploy_zip ], mnemonic = "PyZipMain", command = main_cmd) executable = ctx.outputs.executable ctx.action( inputs = [ deploy_zip, ], outputs = [ executable, ], command = "echo '#!/usr/bin/env python' | cat - %s > %s" % ( deploy_zip.path, executable.path)) runfiles_files = transitive_sources + [executable] runfiles = ctx.runfiles(transitive_files = runfiles_files, collect_default = True) files_to_build = set([executable]) return struct(files = files_to_build, runfiles = runfiles) py_srcs_attr = attr.label_list(allow_files = py_file_types) py_deps_attr = attr.label_list( providers = ["transitive_py_files"], allow_files = False) py_attrs = { "srcs": py_srcs_attr, "deps": py_deps_attr } py_library = rule( py_library_impl, attrs = py_attrs) py_binary_outputs = { "deploy_zip": "%{name}.zip" } py_binary = rule( py_binary_impl, executable = True, attrs = py_attrs, outputs = py_binary_outputs) py_test = rule( py_binary_impl, test = True, executable = True, attrs = py_attrs, outputs = py_binary_outputs)