|
8 | 8 | // option. This file may not be copied, modified, or distributed
|
9 | 9 | // except according to those terms.
|
10 | 10 |
|
11 |
| -use ast::{self, Block, Ident, NodeId, PatKind, Path}; |
| 11 | +use ast::{self, Block, Ident, LitKind, NodeId, PatKind, Path}; |
12 | 12 | use ast::{MacStmtStyle, StmtKind, ItemKind};
|
13 | 13 | use attr::{self, HasAttrs};
|
14 | 14 | use source_map::{ExpnInfo, MacroBang, MacroAttribute, dummy_spanned, respan};
|
@@ -1535,21 +1535,65 @@ impl<'a, 'b> Folder for InvocationCollector<'a, 'b> {
|
1535 | 1535 | let item = attr::mk_list_item(DUMMY_SP, include_ident, include_info);
|
1536 | 1536 | items.push(dummy_spanned(ast::NestedMetaItemKind::MetaItem(item)));
|
1537 | 1537 | }
|
1538 |
| - Err(ref e) if e.kind() == ErrorKind::InvalidData => { |
1539 |
| - self.cx.span_err( |
1540 |
| - at.span, |
1541 |
| - &format!("{} wasn't a utf-8 file", filename.display()), |
1542 |
| - ); |
1543 |
| - } |
1544 | 1538 | Err(e) => {
|
1545 |
| - self.cx.span_err( |
1546 |
| - at.span, |
1547 |
| - &format!("couldn't read {}: {}", filename.display(), e), |
1548 |
| - ); |
| 1539 | + let lit = it |
| 1540 | + .meta_item() |
| 1541 | + .and_then(|item| item.name_value_literal()) |
| 1542 | + .unwrap(); |
| 1543 | + |
| 1544 | + if e.kind() == ErrorKind::InvalidData { |
| 1545 | + self.cx |
| 1546 | + .struct_span_err( |
| 1547 | + lit.span, |
| 1548 | + &format!("{} wasn't a utf-8 file", filename.display()), |
| 1549 | + ) |
| 1550 | + .span_label(lit.span, "contains invalid utf-8") |
| 1551 | + .emit(); |
| 1552 | + } else { |
| 1553 | + let mut err = self.cx.struct_span_err( |
| 1554 | + lit.span, |
| 1555 | + &format!("couldn't read {}: {}", filename.display(), e), |
| 1556 | + ); |
| 1557 | + err.span_label(lit.span, "couldn't read file"); |
| 1558 | + |
| 1559 | + if e.kind() == ErrorKind::NotFound { |
| 1560 | + err.help("external doc paths are relative to the crate root"); |
| 1561 | + } |
| 1562 | + |
| 1563 | + err.emit(); |
| 1564 | + } |
1549 | 1565 | }
|
1550 | 1566 | }
|
1551 | 1567 | } else {
|
1552 |
| - items.push(noop_fold_meta_list_item(it, self)); |
| 1568 | + let mut err = self.cx.struct_span_err( |
| 1569 | + it.span, |
| 1570 | + &format!("expected path to external documentation"), |
| 1571 | + ); |
| 1572 | + |
| 1573 | + // Check if the user erroneously used `doc(include(...))` syntax. |
| 1574 | + let literal = it.meta_item_list().and_then(|list| { |
| 1575 | + if list.len() == 1 { |
| 1576 | + list[0].literal().map(|literal| &literal.node) |
| 1577 | + } else { |
| 1578 | + None |
| 1579 | + } |
| 1580 | + }); |
| 1581 | + |
| 1582 | + let (path, applicability) = match &literal { |
| 1583 | + Some(LitKind::Str(path, ..)) => { |
| 1584 | + (path.to_string(), Applicability::MachineApplicable) |
| 1585 | + } |
| 1586 | + _ => (String::from("<path>"), Applicability::HasPlaceholders), |
| 1587 | + }; |
| 1588 | + |
| 1589 | + err.span_suggestion_with_applicability( |
| 1590 | + it.span, |
| 1591 | + "provide a file path with `=`", |
| 1592 | + format!("include = \"{}\"", path), |
| 1593 | + applicability, |
| 1594 | + ); |
| 1595 | + |
| 1596 | + err.emit(); |
1553 | 1597 | }
|
1554 | 1598 | }
|
1555 | 1599 |
|
|
0 commit comments