Skip to content

Commit

Permalink
Add a new span_suggestion infrastructure. This lets you edit a snippet
Browse files Browse the repository at this point in the history
of text (perhaps obtained by span_snippet) and then splice that edited
form back into the original file in the form of a suggestion.
  • Loading branch information
nikomatsakis committed Apr 9, 2015
1 parent 7314c1a commit 56ad009
Show file tree
Hide file tree
Showing 2 changed files with 72 additions and 1 deletion.
7 changes: 7 additions & 0 deletions src/librustc/session/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -142,6 +142,13 @@ impl Session {
pub fn span_end_note(&self, sp: Span, msg: &str) {
self.diagnostic().span_end_note(sp, msg)
}

/// Prints out a message with a suggested edit of the code.
///
/// See `diagnostic::RenderSpan::Suggestion` for more information.
pub fn span_suggestion(&self, sp: Span, msg: &str, suggestion: String) {
self.diagnostic().span_suggestion(sp, msg, suggestion)
}
pub fn span_help(&self, sp: Span, msg: &str) {
self.diagnostic().span_help(sp, msg)
}
Expand Down
66 changes: 65 additions & 1 deletion src/libsyntax/diagnostic.rs
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ use codemap;
use diagnostics;

use std::cell::{RefCell, Cell};
use std::cmp;
use std::fmt;
use std::io::prelude::*;
use std::io;
Expand All @@ -28,7 +29,7 @@ use libc;
/// maximum number of lines we will print for each error; arbitrary.
const MAX_LINES: usize = 6;

#[derive(Clone, Copy)]
#[derive(Clone)]
pub enum RenderSpan {
/// A FullSpan renders with both with an initial line for the
/// message, prefixed by file:linenum, followed by a summary of
Expand All @@ -42,6 +43,12 @@ pub enum RenderSpan {
/// of the span).
EndSpan(Span),

/// A suggestion renders with both with an initial line for the
/// message, prefixed by file:linenum, followed by a summary
/// of hypothetical source code, where the `String` is spliced
/// into the lines in place of the code covered by the span.
Suggestion(Span, String),

/// A FileLine renders with just a line for the message prefixed
/// by file:linenum.
FileLine(Span),
Expand All @@ -51,6 +58,7 @@ impl RenderSpan {
fn span(&self) -> Span {
match *self {
FullSpan(s) |
Suggestion(s, _) |
EndSpan(s) |
FileLine(s) =>
s
Expand Down Expand Up @@ -124,6 +132,12 @@ impl SpanHandler {
pub fn span_help(&self, sp: Span, msg: &str) {
self.handler.emit(Some((&self.cm, sp)), msg, Help);
}
/// Prints out a message with a suggested edit of the code.
///
/// See `diagnostic::RenderSpan::Suggestion` for more information.
pub fn span_suggestion(&self, sp: Span, msg: &str, suggestion: String) {
self.handler.custom_emit(&self.cm, Suggestion(sp, suggestion), msg, Help);
}
pub fn fileline_note(&self, sp: Span, msg: &str) {
self.handler.custom_emit(&self.cm, FileLine(sp), msg, Note);
}
Expand Down Expand Up @@ -455,6 +469,9 @@ fn emit(dst: &mut EmitterWriter, cm: &codemap::CodeMap, rsp: RenderSpan,
EndSpan(_) => {
try!(end_highlight_lines(dst, cm, sp, lvl, cm.span_to_lines(sp)));
}
Suggestion(_, ref suggestion) => {
try!(highlight_suggestion(dst, cm, sp, suggestion));
}
FileLine(..) => {
// no source text in this case!
}
Expand All @@ -479,6 +496,53 @@ fn emit(dst: &mut EmitterWriter, cm: &codemap::CodeMap, rsp: RenderSpan,
Ok(())
}

fn highlight_suggestion(err: &mut EmitterWriter,
cm: &codemap::CodeMap,
sp: Span,
suggestion: &str)
-> io::Result<()>
{
let lines = cm.span_to_lines(sp);
assert!(!lines.lines.is_empty());

// To build up the result, we want to take the snippet from the first
// line that precedes the span, prepend that with the suggestion, and
// then append the snippet from the last line that trails the span.
let fm = &lines.file;

let first_line = &lines.lines[0];
let prefix = fm.get_line(first_line.line_index)
.map(|l| &l[..first_line.start_col.0])
.unwrap_or("");

let last_line = lines.lines.last().unwrap();
let suffix = fm.get_line(last_line.line_index)
.map(|l| &l[last_line.end_col.0..])
.unwrap_or("");

let complete = format!("{}{}{}", prefix, suggestion, suffix);

// print the suggestion without any line numbers, but leave
// space for them. This helps with lining up with previous
// snippets from the actual error being reported.
let line_num_space = format!("{0:1$}", "", format!("{}", first_line.line_index+1).len());
let fm = &*lines.file;
let mut lines = complete.lines();
for line in lines.by_ref().take(MAX_LINES) {
try!(write!(&mut err.dst, "{}:{} {}\n", fm.name, line_num_space, line));
}

// if we elided some lines, add an ellipsis
if lines.next().is_some() {
try!(write!(&mut err.dst, "{0:1$} {2} ...\n",
"",
fm.name.len(),
line_num_space));
}

Ok(())
}

fn highlight_lines(err: &mut EmitterWriter,
cm: &codemap::CodeMap,
sp: Span,
Expand Down

0 comments on commit 56ad009

Please sign in to comment.