blob: 7eda38901038a6eb5f2a1de97d79cb174e03a721 [file] [log] [blame]
# Copyright 2018 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.
# buildifier: disable=module-docstring
load("//rust/private:common.bzl", "rust_common")
load("//rust/private:utils.bzl", "find_toolchain", "get_lib_name", "get_preferred_artifact")
def _rust_doc_test_impl(ctx):
"""The implementation for the `rust_doc_test` rule
Args:
ctx (ctx): The rule's context object
Returns:
list: A list containing a DefaultInfo provider
"""
if ctx.attr.crate and ctx.attr.dep:
fail("{} should only use the `crate` attribute. `dep` is deprecated".format(
ctx.label,
))
crate = ctx.attr.crate or ctx.attr.dep
if not crate:
fail("{} is missing the `crate` attribute".format(ctx.label))
toolchain = find_toolchain(ctx)
crate_info = crate[rust_common.crate_info]
dep_info = crate[rust_common.dep_info]
# Construct rustdoc test command, which will be written to a shell script
# to be executed to run the test.
flags = _build_rustdoc_flags(dep_info, crate_info)
if toolchain.os != "windows":
rust_doc_test = _build_rustdoc_test_bash_script(ctx, toolchain, flags, crate_info)
else:
rust_doc_test = _build_rustdoc_test_batch_script(ctx, toolchain, flags, crate_info)
# The test script compiles the crate and runs it, so it needs both compile and runtime inputs.
compile_inputs = depset(
[crate_info.output] +
[toolchain.rust_doc] +
[toolchain.rustc] +
toolchain.crosstool_files,
transitive = [
crate_info.srcs,
dep_info.transitive_libs,
toolchain.rustc_lib.files,
toolchain.rust_lib.files,
],
)
return [DefaultInfo(
runfiles = ctx.runfiles(
files = compile_inputs.to_list(),
collect_data = True,
),
executable = rust_doc_test,
)]
# TODO: Replace with bazel-skylib's `path.dirname`. This requires addressing some dependency issues or
# generating docs will break.
def _dirname(path_str):
"""Returns the path of the direcotry from a unix path.
Args:
path_str (str): A string representing a unix path
Returns:
str: The parsed directory name of the provided path
"""
return "/".join(path_str.split("/")[:-1])
def _build_rustdoc_flags(dep_info, crate_info):
"""Constructs the rustdoc script used to test `crate`.
Args:
dep_info (DepInfo): The DepInfo provider
crate_info (CrateInfo): The CrateInfo provider
Returns:
list: A list of rustdoc flags (str)
"""
d = dep_info
# nb. Paths must be constructed wrt runfiles, so we construct relative link flags for doctest.
link_flags = []
link_search_flags = []
link_flags.append("--extern=" + crate_info.name + "=" + crate_info.output.short_path)
link_flags += ["--extern=" + c.name + "=" + c.dep.output.short_path for c in d.direct_crates.to_list()]
link_search_flags += ["-Ldependency={}".format(_dirname(c.output.short_path)) for c in d.transitive_crates.to_list()]
# TODO(hlopko): use the more robust logic from rustc.bzl also here, through a reasonable API.
for lib_to_link in dep_info.transitive_noncrates.to_list():
is_static = bool(lib_to_link.static_library or lib_to_link.pic_static_library)
f = get_preferred_artifact(lib_to_link)
if not is_static:
link_flags.append("-ldylib=" + get_lib_name(f))
else:
link_flags.append("-lstatic=" + get_lib_name(f))
link_flags.append("-Lnative={}".format(_dirname(f.short_path)))
link_search_flags.append("-Lnative={}".format(_dirname(f.short_path)))
if crate_info.type == "proc-macro":
link_flags.extend(["--extern", "proc_macro"])
edition_flags = ["--edition={}".format(crate_info.edition)] if crate_info.edition != "2015" else []
return link_search_flags + link_flags + edition_flags
_rustdoc_test_bash_script = """\
#!/usr/bin/env bash
set -e;
{rust_doc} --test \\
{crate_root} \\
--crate-name={crate_name} \\
{flags}
"""
def _build_rustdoc_test_bash_script(ctx, toolchain, flags, crate_info):
"""Generates a helper script for executing a rustdoc test for unix systems
Args:
ctx (ctx): The `rust_doc_test` rule's context object
toolchain (ToolchainInfo): A rustdoc toolchain
flags (list): A list of rustdoc flags (str)
crate_info (CrateInfo): The CrateInfo provider
Returns:
File: An executable containing information for a rustdoc test
"""
rust_doc_test = ctx.actions.declare_file(
ctx.label.name + ".sh",
)
ctx.actions.write(
output = rust_doc_test,
content = _rustdoc_test_bash_script.format(
rust_doc = toolchain.rust_doc.short_path,
crate_root = crate_info.root.path,
crate_name = crate_info.name,
# TODO: Should be possible to do this with ctx.actions.Args, but can't seem to get them as a str and into the template.
flags = " \\\n ".join(flags),
),
is_executable = True,
)
return rust_doc_test
_rustdoc_test_batch_script = """\
{rust_doc} --test ^
{crate_root} ^
--crate-name={crate_name} ^
{flags}
"""
def _build_rustdoc_test_batch_script(ctx, toolchain, flags, crate_info):
"""Generates a helper script for executing a rustdoc test for windows systems
Args:
ctx (ctx): The `rust_doc_test` rule's context object
toolchain (ToolchainInfo): A rustdoc toolchain
flags (list): A list of rustdoc flags (str)
crate_info (CrateInfo): The CrateInfo provider
Returns:
File: An executable containing information for a rustdoc test
"""
rust_doc_test = ctx.actions.declare_file(
ctx.label.name + ".bat",
)
ctx.actions.write(
output = rust_doc_test,
content = _rustdoc_test_batch_script.format(
rust_doc = toolchain.rust_doc.short_path.replace("/", "\\"),
crate_root = crate_info.root.path,
crate_name = crate_info.name,
# TODO: Should be possible to do this with ctx.actions.Args, but can't seem to get them as a str and into the template.
flags = " ^\n ".join(flags),
),
is_executable = True,
)
return rust_doc_test
rust_doc_test = rule(
implementation = _rust_doc_test_impl,
attrs = {
"crate": attr.label(
doc = (
"The label of the target to generate code documentation for.\n" +
"\n" +
"`rust_doc_test` can generate HTML code documentation for the source files of " +
"`rust_library` or `rust_binary` targets."
),
providers = [rust_common.crate_info],
# TODO: Make this attribute mandatory once `dep` is removed
),
"dep": attr.label(
doc = "__deprecated__: use `crate`",
providers = [rust_common.crate_info],
),
},
executable = True,
test = True,
toolchains = [str(Label("//rust:toolchain"))],
incompatible_use_toolchain_transition = True,
doc = """Runs Rust documentation tests.
Example:
Suppose you have the following directory structure for a Rust library crate:
```output
[workspace]/
WORKSPACE
hello_lib/
BUILD
src/
lib.rs
```
To run [documentation tests][doc-test] for the `hello_lib` crate, define a `rust_doc_test` \
target that depends on the `hello_lib` `rust_library` target:
[doc-test]: https://doc.rust-lang.org/book/documentation.html#documentation-as-tests
```python
package(default_visibility = ["//visibility:public"])
load("@rules_rust//rust:rust.bzl", "rust_library", "rust_doc_test")
rust_library(
name = "hello_lib",
srcs = ["src/lib.rs"],
)
rust_doc_test(
name = "hello_lib_doc_test",
crate = ":hello_lib",
)
```
Running `bazel test //hello_lib:hello_lib_doc_test` will run all documentation tests for the `hello_lib` library crate.
""",
)