Auto merge of #51384 - QuietMisdreavus:extern-version, r=GuillaumeGomez
rustdoc: add flag to control the html_root_url of dependencies The `--extern-html-root-url` flag in this PR allows one to override links to crates whose docs are not already available locally in the doc bundle. Docs.rs currently uses a version of this to make sure links to other crates go into that crate's docs.rs page. See the included test for intended use, but the idea is as follows: Calling rustdoc with `--extern-html-root-url crate=https://some-url.com` will cause rustdoc to override links that point to that crate to instead be replaced with a link rooted at `https://some-url.com/`. (e.g. for docs.rs this would be `https://docs.rs/crate/0.1.0` or the like.) Cheekily, rustup could use these options to redirect links to std/core/etc to instead point to locally-downloaded docs, if it so desired. Fixes https://github.com/rust-lang/rust/issues/19603
This commit is contained in:
@@ -361,6 +361,21 @@ This flag allows rustdoc to treat your rust code as the given edition. It will c
|
|||||||
the given edition as well. As with `rustc`, the default edition that `rustdoc` will use is `2015`
|
the given edition as well. As with `rustc`, the default edition that `rustdoc` will use is `2015`
|
||||||
(the first edition).
|
(the first edition).
|
||||||
|
|
||||||
|
### `--extern-html-root-url`: control how rustdoc links to non-local crates
|
||||||
|
|
||||||
|
Using this flag looks like this:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
$ rustdoc src/lib.rs -Z unstable-options --extern-html-root-url some-crate=https://example.com/some-crate/1.0.1
|
||||||
|
```
|
||||||
|
|
||||||
|
Ordinarily, when rustdoc wants to link to a type from a different crate, it looks in two places:
|
||||||
|
docs that already exist in the output directory, or the `#![doc(doc_html_root)]` set in the other
|
||||||
|
crate. However, if you want to link to docs that exist in neither of those places, you can use these
|
||||||
|
flags to control that behavior. When the `--extern-html-root-url` flag is given with a name matching
|
||||||
|
one of your dependencies, rustdoc use that URL for those docs. Keep in mind that if those docs exist
|
||||||
|
in the output directory, those local docs will still override this flag.
|
||||||
|
|
||||||
### `-Z force-unstable-if-unmarked`
|
### `-Z force-unstable-if-unmarked`
|
||||||
|
|
||||||
Using this flag looks like this:
|
Using this flag looks like this:
|
||||||
|
|||||||
@@ -479,6 +479,7 @@ pub fn initial_ids() -> Vec<String> {
|
|||||||
|
|
||||||
/// Generates the documentation for `crate` into the directory `dst`
|
/// Generates the documentation for `crate` into the directory `dst`
|
||||||
pub fn run(mut krate: clean::Crate,
|
pub fn run(mut krate: clean::Crate,
|
||||||
|
extern_urls: BTreeMap<String, String>,
|
||||||
external_html: &ExternalHtml,
|
external_html: &ExternalHtml,
|
||||||
playground_url: Option<String>,
|
playground_url: Option<String>,
|
||||||
dst: PathBuf,
|
dst: PathBuf,
|
||||||
@@ -611,8 +612,9 @@ pub fn run(mut krate: clean::Crate,
|
|||||||
},
|
},
|
||||||
_ => PathBuf::new(),
|
_ => PathBuf::new(),
|
||||||
};
|
};
|
||||||
|
let extern_url = extern_urls.get(&e.name).map(|u| &**u);
|
||||||
cache.extern_locations.insert(n, (e.name.clone(), src_root,
|
cache.extern_locations.insert(n, (e.name.clone(), src_root,
|
||||||
extern_location(e, &cx.dst)));
|
extern_location(e, extern_url, &cx.dst)));
|
||||||
|
|
||||||
let did = DefId { krate: n, index: CRATE_DEF_INDEX };
|
let did = DefId { krate: n, index: CRATE_DEF_INDEX };
|
||||||
cache.external_paths.insert(did, (vec![e.name.to_string()], ItemType::Module));
|
cache.external_paths.insert(did, (vec![e.name.to_string()], ItemType::Module));
|
||||||
@@ -1096,13 +1098,23 @@ fn clean_srcpath<F>(src_root: &Path, p: &Path, keep_filename: bool, mut f: F) wh
|
|||||||
|
|
||||||
/// Attempts to find where an external crate is located, given that we're
|
/// Attempts to find where an external crate is located, given that we're
|
||||||
/// rendering in to the specified source destination.
|
/// rendering in to the specified source destination.
|
||||||
fn extern_location(e: &clean::ExternalCrate, dst: &Path) -> ExternalLocation {
|
fn extern_location(e: &clean::ExternalCrate, extern_url: Option<&str>, dst: &Path)
|
||||||
|
-> ExternalLocation
|
||||||
|
{
|
||||||
// See if there's documentation generated into the local directory
|
// See if there's documentation generated into the local directory
|
||||||
let local_location = dst.join(&e.name);
|
let local_location = dst.join(&e.name);
|
||||||
if local_location.is_dir() {
|
if local_location.is_dir() {
|
||||||
return Local;
|
return Local;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(url) = extern_url {
|
||||||
|
let mut url = url.to_string();
|
||||||
|
if !url.ends_with("/") {
|
||||||
|
url.push('/');
|
||||||
|
}
|
||||||
|
return Remote(url);
|
||||||
|
}
|
||||||
|
|
||||||
// Failing that, see if there's an attribute specifying where to find this
|
// Failing that, see if there's an attribute specifying where to find this
|
||||||
// external crate
|
// external crate
|
||||||
e.attrs.lists("doc")
|
e.attrs.lists("doc")
|
||||||
|
|||||||
@@ -162,6 +162,10 @@ fn opts() -> Vec<RustcOptGroup> {
|
|||||||
stable("extern", |o| {
|
stable("extern", |o| {
|
||||||
o.optmulti("", "extern", "pass an --extern to rustc", "NAME=PATH")
|
o.optmulti("", "extern", "pass an --extern to rustc", "NAME=PATH")
|
||||||
}),
|
}),
|
||||||
|
unstable("extern-html-root-url", |o| {
|
||||||
|
o.optmulti("", "extern-html-root-url",
|
||||||
|
"base URL to use for dependencies", "NAME=URL")
|
||||||
|
}),
|
||||||
stable("plugin-path", |o| {
|
stable("plugin-path", |o| {
|
||||||
o.optmulti("", "plugin-path", "removed", "DIR")
|
o.optmulti("", "plugin-path", "removed", "DIR")
|
||||||
}),
|
}),
|
||||||
@@ -453,6 +457,13 @@ fn main_args(args: &[String]) -> isize {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
let extern_urls = match parse_extern_html_roots(&matches) {
|
||||||
|
Ok(ex) => ex,
|
||||||
|
Err(err) => {
|
||||||
|
diag.struct_err(err).emit();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
let test_args = matches.opt_strs("test-args");
|
let test_args = matches.opt_strs("test-args");
|
||||||
let test_args: Vec<String> = test_args.iter()
|
let test_args: Vec<String> = test_args.iter()
|
||||||
@@ -553,7 +564,7 @@ fn main_args(args: &[String]) -> isize {
|
|||||||
info!("going to format");
|
info!("going to format");
|
||||||
match output_format.as_ref().map(|s| &**s) {
|
match output_format.as_ref().map(|s| &**s) {
|
||||||
Some("html") | None => {
|
Some("html") | None => {
|
||||||
html::render::run(krate, &external_html, playground_url,
|
html::render::run(krate, extern_urls, &external_html, playground_url,
|
||||||
output.unwrap_or(PathBuf::from("doc")),
|
output.unwrap_or(PathBuf::from("doc")),
|
||||||
resource_suffix.unwrap_or(String::new()),
|
resource_suffix.unwrap_or(String::new()),
|
||||||
passes.into_iter().collect(),
|
passes.into_iter().collect(),
|
||||||
@@ -612,6 +623,23 @@ fn parse_externs(matches: &getopts::Matches) -> Result<Externs, String> {
|
|||||||
Ok(Externs::new(externs))
|
Ok(Externs::new(externs))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Extracts `--extern-html-root-url` arguments from `matches` and returns a map of crate names to
|
||||||
|
/// the given URLs. If an `--extern-html-root-url` argument was ill-formed, returns an error
|
||||||
|
/// describing the issue.
|
||||||
|
fn parse_extern_html_roots(matches: &getopts::Matches)
|
||||||
|
-> Result<BTreeMap<String, String>, &'static str>
|
||||||
|
{
|
||||||
|
let mut externs = BTreeMap::new();
|
||||||
|
for arg in &matches.opt_strs("extern-html-root-url") {
|
||||||
|
let mut parts = arg.splitn(2, '=');
|
||||||
|
let name = parts.next().ok_or("--extern-html-root-url must not be empty")?;
|
||||||
|
let url = parts.next().ok_or("--extern-html-root-url must be of the form name=url")?;
|
||||||
|
externs.insert(name.to_string(), url.to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(externs)
|
||||||
|
}
|
||||||
|
|
||||||
/// Interprets the input file as a rust source file, passing it through the
|
/// Interprets the input file as a rust source file, passing it through the
|
||||||
/// compiler all the way through the analysis passes. The rustdoc output is then
|
/// compiler all the way through the analysis passes. The rustdoc output is then
|
||||||
/// generated from the cleaned AST of the crate.
|
/// generated from the cleaned AST of the crate.
|
||||||
|
|||||||
18
src/test/rustdoc/extern-html-root-url.rs
Normal file
18
src/test/rustdoc/extern-html-root-url.rs
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
|
||||||
|
// file at the top-level directory of this distribution and at
|
||||||
|
// http://rust-lang.org/COPYRIGHT.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||||
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||||
|
// option. This file may not be copied, modified, or distributed
|
||||||
|
// except according to those terms.
|
||||||
|
|
||||||
|
// ignore-tidy-linelength
|
||||||
|
|
||||||
|
// compile-flags:-Z unstable-options --extern-html-root-url core=https://example.com/core/0.1.0
|
||||||
|
|
||||||
|
// @has extern_html_root_url/index.html
|
||||||
|
// @has - '//a/@href' 'https://example.com/core/0.1.0/core/iter/index.html'
|
||||||
|
#[doc(no_inline)]
|
||||||
|
pub use std::iter;
|
||||||
Reference in New Issue
Block a user