blob: 599ef01b3aee1e0aa33953df7b97ee5be80b61aa [file]
# Copyright 2019 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.
"""Contains the web_package rule.
"""
def html_asset_inject(index_html, action_factory, injector, root_dirs, assets, output):
"""Injects JS and CSS resources into the index.html.
Args:
index_html: The input html file
action_factory: Bazel's actions module from ctx.actions - see https://docs.bazel.build/versions/master/skylark/lib/actions.html
injector: The injector executable
root_dirs: Path prefixes to strip off all assets. Longest wins.
assets: Asset files to inject
output: The output html file
Returns:
The output html file
"""
args = action_factory.args()
args.add(output.path)
args.add(index_html.path)
args.add_all(root_dirs)
args.add("--assets")
args.add_all(assets)
args.use_param_file("%s", use_always = True)
action_factory.run(
inputs = [index_html],
outputs = [output],
executable = injector,
arguments = [args],
)
return output
def move_files(output_name, files, action_factory, assembler, root_paths):
"""Moves files into an output directory
Args:
output_name: The name of the output directory
files: The files to move
action_factory: Bazel's actions module from ctx.actions - see https://docs.bazel.build/versions/master/skylark/lib/actions.html
assembler: The assembler executable
root_paths: Path prefixes to strip off all assets. Longest wins.
Returns:
The output directory tree-artifact
"""
www_dir = action_factory.declare_directory(output_name)
args = action_factory.args()
args.add(www_dir.path)
args.add_all(root_paths)
args.add("--assets")
args.add_all([f.path for f in files])
args.use_param_file("%s", use_always = True)
action_factory.run(
inputs = files,
outputs = [www_dir],
executable = assembler,
arguments = [args],
execution_requirements = {"local": "1"},
)
return depset([www_dir])
def additional_root_paths(ctx):
return ctx.attr.additional_root_paths + [
# package path is the root, including in bin/gen
ctx.label.package,
"/".join([ctx.bin_dir.path, ctx.label.package]),
"/".join([ctx.genfiles_dir.path, ctx.label.package]),
# bazel-bin/gen dirs to absolute paths
ctx.genfiles_dir.path,
ctx.bin_dir.path,
# package re-rooted subdirectory
"/".join([p for p in [ctx.bin_dir.path, ctx.label.package, "_" + ctx.label.name, ctx.label.package] if p]),
]
def _web_package(ctx):
root_paths = additional_root_paths(ctx)
# Create the output file in a re-rooted subdirectory so it doesn't collide with the input file
html = ctx.actions.declare_file("_%s/%s" % (ctx.label.name, ctx.file.index_html.path))
# Move that index file back into place inside the package
populated_index = html_asset_inject(
ctx.file.index_html,
ctx.actions,
ctx.executable._injector,
root_paths,
[f.path for f in ctx.files.assets],
html,
)
package_layout = move_files(
ctx.label.name,
ctx.files.data + ctx.files.assets + [html],
ctx.actions,
ctx.executable._assembler,
root_paths,
)
return [
DefaultInfo(files = package_layout),
]
web_package = rule(
implementation = _web_package,
attrs = {
"additional_root_paths": attr.string_list(
doc = """Path prefixes to strip off all assets, in addition to the current package. Longest wins.""",
),
"assets": attr.label_list(
allow_files = True,
doc = """Files which should be referenced from the index_html""",
),
"data": attr.label_list(
allow_files = True,
doc = """Additional files which should be served on request""",
),
"index_html": attr.label(
allow_single_file = True,
doc = """The entry point of the application""",
),
"_assembler": attr.label(
default = "@build_bazel_rules_nodejs//internal/web_package:assembler",
executable = True,
cfg = "host",
),
"_injector": attr.label(
default = "@build_bazel_rules_nodejs//internal/web_package:injector",
executable = True,
cfg = "host",
),
},
doc = """Assembles a web application from source files.
Injects JS and CSS resources into the index.html.
""",
)