|  | # 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. | 
|  |  | 
|  | """Rules for performing `rustdoc --test` on Bazel built crates""" | 
|  |  | 
|  | load("//rust/private:common.bzl", "rust_common") | 
|  | load("//rust/private:providers.bzl", "CrateInfo") | 
|  | load("//rust/private:rustdoc.bzl", "rustdoc_compile_action") | 
|  | load("//rust/private:utils.bzl", "dedent", "find_toolchain", "transform_deps") | 
|  |  | 
|  | def _construct_writer_arguments(ctx, test_runner, opt_test_params, action, crate_info): | 
|  | """Construct arguments and environment variables specific to `rustdoc_test_writer`. | 
|  |  | 
|  | This is largely solving for the fact that tests run from a runfiles directory | 
|  | where actions run in an execroot. But it also tracks what environment variables | 
|  | were explicitly added to the action. | 
|  |  | 
|  | Args: | 
|  | ctx (ctx): The rule's context object. | 
|  | test_runner (File): The test_runner output file declared by `rustdoc_test`. | 
|  | opt_test_params (File): An output file we can optionally use to store params for `rustdoc`. | 
|  | action (struct): Action arguments generated by `rustdoc_compile_action`. | 
|  | crate_info (CrateInfo): The provider of the crate who's docs are being tested. | 
|  |  | 
|  | Returns: | 
|  | tuple: A tuple of `rustdoc_test_writer` specific inputs | 
|  | - Args: Arguments for the test writer | 
|  | - dict: Required environment variables | 
|  | """ | 
|  |  | 
|  | writer_args = ctx.actions.args() | 
|  |  | 
|  | # Track the output path where the test writer should write the test | 
|  | writer_args.add("--output={}".format(test_runner.path)) | 
|  |  | 
|  | # Track where the test writer should move "spilled" Args to | 
|  | writer_args.add("--optional_test_params={}".format(opt_test_params.path)) | 
|  |  | 
|  | # Track what environment variables should be written to the test runner | 
|  | writer_args.add("--action_env=DEVELOPER_DIR") | 
|  | writer_args.add("--action_env=PATHEXT") | 
|  | writer_args.add("--action_env=SDKROOT") | 
|  | writer_args.add("--action_env=SYSROOT") | 
|  | for var in action.env.keys(): | 
|  | writer_args.add("--action_env={}".format(var)) | 
|  |  | 
|  | # Since the test runner will be running from a runfiles directory, the | 
|  | # paths originally generated for the build action will not map to any | 
|  | # files. To ensure rustdoc can find the appropriate dependencies, the | 
|  | # file roots are identified and tracked for each dependency so it can be | 
|  | # stripped from the test runner. | 
|  |  | 
|  | # Collect and dedupe all of the file roots in a list before appending | 
|  | # them to args to prevent generating a large amount of identical args | 
|  | roots = [] | 
|  | root = crate_info.output.root.path | 
|  | if not root in roots: | 
|  | roots.append(root) | 
|  | for dep in crate_info.deps.to_list(): | 
|  | dep_crate_info = getattr(dep, "crate_info", None) | 
|  | dep_dep_info = getattr(dep, "dep_info", None) | 
|  | if dep_crate_info: | 
|  | root = dep_crate_info.output.root.path | 
|  | if not root in roots: | 
|  | roots.append(root) | 
|  | if dep_dep_info: | 
|  | for direct_dep in dep_dep_info.direct_crates.to_list(): | 
|  | root = direct_dep.dep.output.root.path | 
|  | if not root in roots: | 
|  | roots.append(root) | 
|  | for transitive_dep in dep_dep_info.transitive_crates.to_list(): | 
|  | root = transitive_dep.output.root.path | 
|  | if not root in roots: | 
|  | roots.append(root) | 
|  |  | 
|  | for root in roots: | 
|  | writer_args.add("--strip_substring={}/".format(root)) | 
|  |  | 
|  | # Indicate that the rustdoc_test args are over. | 
|  | writer_args.add("--") | 
|  |  | 
|  | # Prepare for the process runner to ingest the rest of the arguments | 
|  | # to match the expectations of `rustc_compile_action`. | 
|  | writer_args.add(ctx.executable._process_wrapper.short_path) | 
|  |  | 
|  | return (writer_args, action.env) | 
|  |  | 
|  | 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 | 
|  | """ | 
|  |  | 
|  | toolchain = find_toolchain(ctx) | 
|  |  | 
|  | crate = ctx.attr.crate[rust_common.crate_info] | 
|  | deps = transform_deps(ctx.attr.deps) | 
|  |  | 
|  | crate_info = rust_common.create_crate_info( | 
|  | name = crate.name, | 
|  | type = crate.type, | 
|  | root = crate.root, | 
|  | srcs = crate.srcs, | 
|  | deps = depset(deps, transitive = [crate.deps]), | 
|  | proc_macro_deps = crate.proc_macro_deps, | 
|  | aliases = crate.aliases, | 
|  | output = crate.output, | 
|  | edition = crate.edition, | 
|  | rustc_env = crate.rustc_env, | 
|  | rustc_env_files = crate.rustc_env_files, | 
|  | is_test = True, | 
|  | compile_data = crate.compile_data, | 
|  | compile_data_targets = crate.compile_data_targets, | 
|  | wrapped_crate_type = crate.type, | 
|  | owner = ctx.label, | 
|  | ) | 
|  |  | 
|  | if toolchain.target_os == "windows": | 
|  | test_runner = ctx.actions.declare_file(ctx.label.name + ".rustdoc_test.bat") | 
|  | else: | 
|  | test_runner = ctx.actions.declare_file(ctx.label.name + ".rustdoc_test.sh") | 
|  |  | 
|  | # Bazel will auto-magically spill params to a file, if they are too many for a given OSes shell | 
|  | # (e.g. Windows ~32k, Linux ~2M). The executable script (aka test_runner) that gets generated, | 
|  | # is run from the runfiles, which is separate from the params_file Bazel generates. To handle | 
|  | # this case, we declare our own params file, that the test_writer will populate, if necessary | 
|  | opt_test_params = ctx.actions.declare_file(ctx.label.name + ".rustdoc_opt_params", sibling = test_runner) | 
|  |  | 
|  | # Add the current crate as an extern for the compile action | 
|  | rustdoc_flags = [ | 
|  | "--extern", | 
|  | "{}={}".format(crate_info.name, crate_info.output.short_path), | 
|  | "--test", | 
|  | ] | 
|  |  | 
|  | action = rustdoc_compile_action( | 
|  | ctx = ctx, | 
|  | toolchain = toolchain, | 
|  | crate_info = crate_info, | 
|  | rustdoc_flags = rustdoc_flags, | 
|  | is_test = True, | 
|  | ) | 
|  |  | 
|  | tools = action.tools + [ctx.executable._process_wrapper] | 
|  |  | 
|  | writer_args, env = _construct_writer_arguments( | 
|  | ctx = ctx, | 
|  | test_runner = test_runner, | 
|  | opt_test_params = opt_test_params, | 
|  | action = action, | 
|  | crate_info = crate_info, | 
|  | ) | 
|  |  | 
|  | # Allow writer environment variables to override those from the action. | 
|  | action.env.update(env) | 
|  |  | 
|  | ctx.actions.run( | 
|  | mnemonic = "RustdocTestWriter", | 
|  | progress_message = "Generating Rustdoc test runner for {}".format(ctx.attr.crate.label), | 
|  | executable = ctx.executable._test_writer, | 
|  | inputs = action.inputs, | 
|  | tools = tools, | 
|  | arguments = [writer_args] + action.arguments, | 
|  | env = action.env, | 
|  | outputs = [test_runner, opt_test_params], | 
|  | ) | 
|  |  | 
|  | return [DefaultInfo( | 
|  | files = depset([test_runner]), | 
|  | runfiles = ctx.runfiles(files = tools + [opt_test_params], transitive_files = action.inputs), | 
|  | executable = test_runner, | 
|  | )] | 
|  |  | 
|  | rust_doc_test = rule( | 
|  | implementation = _rust_doc_test_impl, | 
|  | attrs = { | 
|  | "crate": attr.label( | 
|  | doc = ( | 
|  | "The label of the target to generate code documentation for. " + | 
|  | "`rust_doc_test` can generate HTML code documentation for the " + | 
|  | "source files of `rust_library` or `rust_binary` targets." | 
|  | ), | 
|  | providers = [rust_common.crate_info], | 
|  | mandatory = True, | 
|  | ), | 
|  | "deps": attr.label_list( | 
|  | doc = dedent("""\ | 
|  | List of other libraries to be linked to this library target. | 
|  |  | 
|  | These can be either other `rust_library` targets or `cc_library` targets if | 
|  | linking a native library. | 
|  | """), | 
|  | providers = [[CrateInfo], [CcInfo]], | 
|  | ), | 
|  | "_cc_toolchain": attr.label( | 
|  | doc = ( | 
|  | "In order to use find_cc_toolchain, your rule has to depend " + | 
|  | "on C++ toolchain. See @rules_cc//cc:find_cc_toolchain.bzl " + | 
|  | "docs for details." | 
|  | ), | 
|  | default = Label("@bazel_tools//tools/cpp:current_cc_toolchain"), | 
|  | ), | 
|  | "_process_wrapper": attr.label( | 
|  | doc = "A process wrapper for running rustdoc on all platforms", | 
|  | cfg = "exec", | 
|  | default = Label("//util/process_wrapper"), | 
|  | executable = True, | 
|  | ), | 
|  | "_test_writer": attr.label( | 
|  | doc = "A binary used for writing script for use as the test executable.", | 
|  | cfg = "exec", | 
|  | default = Label("//tools/rustdoc:rustdoc_test_writer"), | 
|  | executable = True, | 
|  | ), | 
|  | }, | 
|  | test = True, | 
|  | fragments = ["cpp"], | 
|  | host_fragments = ["cpp"], | 
|  | toolchains = [ | 
|  | str(Label("//rust:toolchain_type")), | 
|  | "@bazel_tools//tools/cpp:toolchain_type", | 
|  | ], | 
|  | incompatible_use_toolchain_transition = True, | 
|  | doc = dedent("""\ | 
|  | 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:defs.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. | 
|  | """), | 
|  | ) |