feat(tvix/eval): implement initial fancy display for warnings
This implements an initial fancy display for warnings emitted by the tvix compiler, using the codemap_diagnostic crate. Each warning variant has an associated message, and optionally an associated annotation for the span displayed to the user. In theory we could get a lot more fancy with the display for specific variants if needed (e.g. re-parse the AST and actually add multiple semantic spans based on context), but this is already a good start. Example: tvix-repl> let toString = https://tvl.fyi; in let inherit toString; in ({}: 42) rec {} warning[W004]: declared variable 'toString' shadows a built-in global! --> [tvix-repl]:1:5 | 1 | let toString = https://tvl.fyi; in let inherit toString; in ({}: 42) rec {} | ^^^^^^^^ variable declared here warning[W001]: URL literal syntax is deprecated, use a quoted string instead --> [tvix-repl]:1:16 | 1 | let toString = https://tvl.fyi; in let inherit toString; in ({}: 42) rec {} | ^^^^^^^^^^^^^^^ warning[W002]: inherited variable already exists with the same value --> [tvix-repl]:1:40 | 1 | let toString = https://tvl.fyi; in let inherit toString; in ({}: 42) rec {} | ^^^^^^^^^^^^^^^^^ warning[W999]: feature not yet implemented in tvix: recursive attribute sets --> [tvix-repl]:1:70 | 1 | let toString = https://tvl.fyi; in let inherit toString; in ({}: 42) rec {} | ^^^^^^ warning[W999]: feature not yet implemented in tvix: closed formals --> [tvix-repl]:1:62 | 1 | let toString = https://tvl.fyi; in let inherit toString; in ({}: 42) rec {} | ^^ warning[W003]: variable 'toString' is declared, but never used: --> [tvix-repl]:1:5 | 1 | let toString = https://tvl.fyi; in let inherit toString; in ({}: 42) rec {} | ^^^^^^^^ variable declared here => 42 :: int These are coloured when output to a terminal. Change-Id: If315648a07e333895db4ae1d0915ee2013806585 Reviewed-on: https://cl.tvl.fyi/c/depot/+/6532 Autosubmit: tazjin <tazjin@tvl.su> Reviewed-by: grfn <grfn@gws.fyi> Tested-by: BuildkiteCI
This commit is contained in:
parent
beb78c7104
commit
4f67cf221a
3 changed files with 94 additions and 9 deletions
|
@ -33,7 +33,7 @@ impl Compiler<'_, '_> {
|
||||||
let span = self.span_for(&node);
|
let span = self.span_for(&node);
|
||||||
self.emit_warning(
|
self.emit_warning(
|
||||||
span,
|
span,
|
||||||
WarningKind::NotImplemented("recursive attribute sets are not yet implemented"),
|
WarningKind::NotImplemented("recursive attribute sets"),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -13,7 +13,7 @@ pub fn interpret(code: &str, location: Option<PathBuf>) -> EvalResult<Value> {
|
||||||
location
|
location
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.map(|p| p.to_string_lossy().to_string())
|
.map(|p| p.to_string_lossy().to_string())
|
||||||
.unwrap_or_else(|| "<repl>".into()),
|
.unwrap_or_else(|| "[tvix-repl]".into()),
|
||||||
code.into(),
|
code.into(),
|
||||||
);
|
);
|
||||||
let codemap = Rc::new(codemap);
|
let codemap = Rc::new(codemap);
|
||||||
|
@ -47,7 +47,7 @@ pub fn interpret(code: &str, location: Option<PathBuf>) -> EvalResult<Value> {
|
||||||
location,
|
location,
|
||||||
&file,
|
&file,
|
||||||
global_builtins(),
|
global_builtins(),
|
||||||
&mut DisassemblingObserver::new(codemap, std::io::stderr()),
|
&mut DisassemblingObserver::new(codemap.clone(), std::io::stderr()),
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
crate::compiler::compile(
|
crate::compiler::compile(
|
||||||
|
@ -60,12 +60,7 @@ pub fn interpret(code: &str, location: Option<PathBuf>) -> EvalResult<Value> {
|
||||||
}?;
|
}?;
|
||||||
|
|
||||||
for warning in result.warnings {
|
for warning in result.warnings {
|
||||||
eprintln!(
|
warning.fancy_format_stderr(&codemap);
|
||||||
"warning: {:?} at `{}`[line {}]",
|
|
||||||
warning.kind,
|
|
||||||
file.source_slice(warning.span),
|
|
||||||
file.find_line(warning.span.low()) + 1
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for error in &result.errors {
|
for error in &result.errors {
|
||||||
|
|
|
@ -1,6 +1,9 @@
|
||||||
//! Implements warnings that are emitted in cases where code passed to
|
//! Implements warnings that are emitted in cases where code passed to
|
||||||
//! Tvix exhibits problems that the user could address.
|
//! Tvix exhibits problems that the user could address.
|
||||||
|
|
||||||
|
use codemap::CodeMap;
|
||||||
|
use codemap_diagnostic::{ColorConfig, Diagnostic, Emitter, Level, SpanLabel, SpanStyle};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum WarningKind {
|
pub enum WarningKind {
|
||||||
DeprecatedLiteralURL,
|
DeprecatedLiteralURL,
|
||||||
|
@ -18,3 +21,90 @@ pub struct EvalWarning {
|
||||||
pub kind: WarningKind,
|
pub kind: WarningKind,
|
||||||
pub span: codemap::Span,
|
pub span: codemap::Span,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl EvalWarning {
|
||||||
|
/// Render a fancy, human-readable output of this warning and
|
||||||
|
/// return it as a String. Note that this version of the output
|
||||||
|
/// does not include any colours or font styles.
|
||||||
|
pub fn fancy_format_str(&self, codemap: &CodeMap) -> String {
|
||||||
|
let mut out = vec![];
|
||||||
|
Emitter::vec(&mut out, Some(codemap)).emit(&[self.diagnostic(codemap)]);
|
||||||
|
String::from_utf8_lossy(&out).to_string()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Render a fancy, human-readable output of this warning and
|
||||||
|
/// print it to stderr. If rendered in a terminal that supports
|
||||||
|
/// colours and font styles, the output will include those.
|
||||||
|
pub fn fancy_format_stderr(&self, codemap: &CodeMap) {
|
||||||
|
Emitter::stderr(ColorConfig::Auto, Some(codemap)).emit(&[self.diagnostic(codemap)]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create the optional span label displayed as an annotation on
|
||||||
|
/// the underlined span of the warning.
|
||||||
|
fn span_label(&self) -> Option<String> {
|
||||||
|
match self.kind {
|
||||||
|
WarningKind::UnusedBinding | WarningKind::ShadowedGlobal(_) => {
|
||||||
|
Some("variable declared here".into())
|
||||||
|
}
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create the primary warning message displayed to users for a
|
||||||
|
/// warning.
|
||||||
|
fn message(&self, codemap: &CodeMap) -> String {
|
||||||
|
match self.kind {
|
||||||
|
WarningKind::DeprecatedLiteralURL => {
|
||||||
|
format!("URL literal syntax is deprecated, use a quoted string instead")
|
||||||
|
}
|
||||||
|
|
||||||
|
WarningKind::UselessInherit => {
|
||||||
|
format!("inherited variable already exists with the same value")
|
||||||
|
}
|
||||||
|
|
||||||
|
WarningKind::UnusedBinding => {
|
||||||
|
let file = codemap.find_file(self.span.low());
|
||||||
|
|
||||||
|
format!(
|
||||||
|
"variable '{}' is declared, but never used:",
|
||||||
|
file.source_slice(self.span)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
WarningKind::ShadowedGlobal(name) => {
|
||||||
|
format!("declared variable '{}' shadows a built-in global!", name)
|
||||||
|
}
|
||||||
|
|
||||||
|
WarningKind::NotImplemented(what) => {
|
||||||
|
format!("feature not yet implemented in tvix: {}", what)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return the unique warning code for this variant which can be
|
||||||
|
/// used to refer users to documentation.
|
||||||
|
fn code(&self) -> &'static str {
|
||||||
|
match self.kind {
|
||||||
|
WarningKind::DeprecatedLiteralURL => "W001",
|
||||||
|
WarningKind::UselessInherit => "W002",
|
||||||
|
WarningKind::UnusedBinding => "W003",
|
||||||
|
WarningKind::ShadowedGlobal(_) => "W004",
|
||||||
|
WarningKind::NotImplemented(_) => "W999",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn diagnostic(&self, codemap: &CodeMap) -> Diagnostic {
|
||||||
|
let span_label = SpanLabel {
|
||||||
|
label: self.span_label(),
|
||||||
|
span: self.span,
|
||||||
|
style: SpanStyle::Primary,
|
||||||
|
};
|
||||||
|
|
||||||
|
Diagnostic {
|
||||||
|
level: Level::Warning,
|
||||||
|
message: self.message(codemap),
|
||||||
|
spans: vec![span_label],
|
||||||
|
code: Some(self.code().into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue