1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
use diagnostic::{
    term::{
        emit,
        termcolor::{ColorChoice, StandardStream, WriteColor},
        Config,
    },
    Diagnostic, DiagnosticLevel, TextStorage,
};

use crate::{DuplicateFields, IOError, Validation, VosError, VosErrorKind, VosResult};

impl VosError {
    pub fn error(&mut self) {
        self.level = DiagnosticLevel::Error;
    }
    pub fn warning(&mut self) {
        self.level = DiagnosticLevel::Warning;
    }
    pub fn fatal(&mut self) {
        self.level = DiagnosticLevel::Bug;
    }
}

pub fn eprint<T>(v: &Validation<T>, text: &TextStorage) -> VosResult {
    let config = Config::default();
    let w: StandardStream = StandardStream::stderr(ColorChoice::Always);
    match v {
        Validation::Success { value: _, diagnostics } => {
            for diagnostic in diagnostics {
                diagnostic.eprint(&mut w.lock(), &config, text)?
            }
        }
        Validation::Failure { fatal, diagnostics } => {
            for diagnostic in diagnostics {
                diagnostic.eprint(&mut w.lock(), &config, text)?
            }
            fatal.eprint(&mut w.lock(), &config, text)?
        }
    }
    Ok(())
}

impl VosError {
    pub fn eprint(&self, writer: &mut impl WriteColor, config: &Config, text: &TextStorage) -> VosResult {
        emit(writer, config, text, &self.as_report())?;
        Ok(())
    }

    pub fn as_report(&self) -> Diagnostic {
        match self.kind() {
            VosErrorKind::IOError(e) => e.as_report(self.level),
            VosErrorKind::ParseError(e) => {
                Diagnostic::new(self.level).with_code("A0002").with_message(e).with_primary(anonymous(), 0..0, "")
            }
            VosErrorKind::RuntimeError(e) => {
                Diagnostic::new(self.level).with_code("A0002").with_message(e).with_primary(anonymous(), 0..0, "")
            }
            VosErrorKind::DuplicateFields(e) => e.as_report(self.level),
            VosErrorKind::UnknownError => {
                Diagnostic::new(self.level).with_code("A0002").with_message("UnknownError").with_primary(anonymous(), 0..0, "")
            }
        }
    }
}

impl IOError {
    pub fn as_report(&self, level: DiagnosticLevel) -> Diagnostic {
        let file_id = match TextStorage::canonicalize(&self.source) {
            Ok(o) => o,
            Err(_) => anonymous(),
        };
        Diagnostic::new(level).with_code("E0001").with_message(self.error.to_string()).with_primary(file_id, 0..0, "")
    }
}

impl DuplicateFields {
    pub fn as_report(&self, level: DiagnosticLevel) -> Diagnostic {
        let file_id = match TextStorage::canonicalize(&self.path) {
            Ok(o) => o,
            Err(_) => anonymous(),
        };
        Diagnostic::new(level).with_code("E0001").with_message(&self.lhs.start).with_primary(file_id, 0..0, &self.lhs.end)
    }
}

fn anonymous() -> String {
    "<anonymous>".to_string()
}