blob: cdb2d9b6baf571f2db48d1749db97e90b209a969 [file] [log] [blame]
# 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.configuration.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)