2019-03-23 13:28:47 +00:00
|
|
|
use std::{fmt, any::Any};
|
2019-03-21 19:13:11 +00:00
|
|
|
|
2019-03-23 13:28:47 +00:00
|
|
|
use ra_syntax::{SyntaxNodePtr, AstPtr, ast};
|
|
|
|
|
|
|
|
use crate::HirFileId;
|
2019-03-23 15:35:14 +00:00
|
|
|
use relative_path::RelativePathBuf;
|
2019-03-23 13:28:47 +00:00
|
|
|
|
2019-03-23 15:35:14 +00:00
|
|
|
/// Diagnostic defines hir API for errors and warnings.
|
|
|
|
///
|
|
|
|
/// It is used as a `dyn` object, which you can downcast to a concrete
|
2019-03-23 17:41:59 +00:00
|
|
|
/// diagnostic. DiagnosticSink are structured, meaning that they include rich
|
|
|
|
/// information which can be used by IDE to create fixes. DiagnosticSink are
|
2019-03-23 15:35:14 +00:00
|
|
|
/// expressed in terms of macro-expanded syntax tree nodes (so, it's a bad idea
|
|
|
|
/// to diagnostic in a salsa value).
|
|
|
|
///
|
|
|
|
/// Internally, various subsystems of hir produce diagnostics specific to a
|
2019-03-23 17:41:59 +00:00
|
|
|
/// subsystem (typically, an `enum`), which are safe to store in salsa but do not
|
2019-03-23 15:35:14 +00:00
|
|
|
/// include source locations. Such internal diagnostic are transformed into an
|
|
|
|
/// instance of `Diagnostic` on demand.
|
2019-03-23 13:28:47 +00:00
|
|
|
pub trait Diagnostic: Any + Send + Sync + fmt::Debug + 'static {
|
2019-03-23 17:41:59 +00:00
|
|
|
fn message(&self) -> String;
|
2019-03-23 13:28:47 +00:00
|
|
|
fn file(&self) -> HirFileId;
|
|
|
|
fn syntax_node(&self) -> SyntaxNodePtr;
|
|
|
|
fn as_any(&self) -> &(Any + Send + 'static);
|
|
|
|
}
|
|
|
|
|
|
|
|
impl dyn Diagnostic {
|
|
|
|
pub fn downcast_ref<D: Diagnostic>(&self) -> Option<&D> {
|
|
|
|
self.as_any().downcast_ref()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Default)]
|
2019-03-23 17:41:59 +00:00
|
|
|
pub struct DiagnosticSink {
|
2019-03-23 13:28:47 +00:00
|
|
|
data: Vec<Box<dyn Diagnostic>>,
|
|
|
|
}
|
|
|
|
|
2019-03-23 17:41:59 +00:00
|
|
|
impl DiagnosticSink {
|
2019-03-23 13:28:47 +00:00
|
|
|
pub fn push(&mut self, d: impl Diagnostic) {
|
|
|
|
self.data.push(Box::new(d))
|
|
|
|
}
|
|
|
|
|
2019-03-23 17:41:59 +00:00
|
|
|
pub fn into_diagnostics(self) -> Vec<Box<dyn Diagnostic>> {
|
|
|
|
self.data
|
2019-03-23 13:28:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct NoSuchField {
|
2019-03-23 15:35:14 +00:00
|
|
|
pub file: HirFileId,
|
|
|
|
pub field: AstPtr<ast::NamedField>,
|
2019-03-23 13:28:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Diagnostic for NoSuchField {
|
2019-03-23 17:41:59 +00:00
|
|
|
fn message(&self) -> String {
|
|
|
|
"no such field".to_string()
|
|
|
|
}
|
2019-03-23 13:28:47 +00:00
|
|
|
fn file(&self) -> HirFileId {
|
|
|
|
self.file
|
|
|
|
}
|
|
|
|
fn syntax_node(&self) -> SyntaxNodePtr {
|
|
|
|
self.field.into()
|
|
|
|
}
|
|
|
|
fn as_any(&self) -> &(Any + Send + 'static) {
|
|
|
|
self
|
|
|
|
}
|
2019-03-21 19:13:11 +00:00
|
|
|
}
|
2019-03-23 15:35:14 +00:00
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct UnresolvedModule {
|
|
|
|
pub file: HirFileId,
|
|
|
|
pub decl: AstPtr<ast::Module>,
|
|
|
|
pub candidate: RelativePathBuf,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Diagnostic for UnresolvedModule {
|
2019-03-23 17:41:59 +00:00
|
|
|
fn message(&self) -> String {
|
|
|
|
"unresolved module".to_string()
|
|
|
|
}
|
2019-03-23 15:35:14 +00:00
|
|
|
fn file(&self) -> HirFileId {
|
|
|
|
self.file
|
|
|
|
}
|
|
|
|
fn syntax_node(&self) -> SyntaxNodePtr {
|
|
|
|
self.decl.into()
|
|
|
|
}
|
|
|
|
fn as_any(&self) -> &(Any + Send + 'static) {
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|