1use crate::source_tracking::SourceMap;
9use crate::source_tracking::filename::FileName;
10use crate::source_tracking::immutable_string::ImmutableString;
11use crate::source_tracking::{fragment::Fragment, source::SourceId};
12use codespan_reporting::diagnostic::Diagnostic as CRDiagnostic;
13use codespan_reporting::diagnostic::Label;
14use codespan_reporting::files::{Error as CRError, Files};
15use codespan_reporting::term::Config;
16use std::sync::Mutex;
17use termcolor::{ColorChoice, StandardStream, WriteColor};
18
19#[cfg(doc)]
20use crate::source_tracking::source::Source;
21
22pub use codespan_reporting::diagnostic::Severity;
24
25pub use codespan_reporting::diagnostic::LabelStyle;
27
28static STDOUT_COLOR_CHOICE: Mutex<ColorChoice> = Mutex::new(ColorChoice::Auto);
30
31pub fn set_stdout_color(c: ColorChoice) {
33 *STDOUT_COLOR_CHOICE.lock().unwrap() = c;
34}
35
36pub fn get_stdout_color() -> ColorChoice {
38 *STDOUT_COLOR_CHOICE.lock().unwrap()
39}
40
41#[derive(Debug)]
44pub struct Diagnostic(pub CRDiagnostic<SourceId>);
45
46impl Diagnostic {
47 pub fn new(severity: Severity) -> Self {
49 Diagnostic(CRDiagnostic::new(severity))
50 }
51
52 #[inline]
54 pub fn bug() -> Self {
55 Self::new(Severity::Bug)
56 }
57
58 #[inline]
60 pub fn error() -> Self {
61 Self::new(Severity::Error)
62 }
63
64 #[inline]
66 pub fn warning() -> Self {
67 Self::new(Severity::Warning)
68 }
69
70 #[inline]
72 pub fn note() -> Self {
73 Self::new(Severity::Note)
74 }
75
76 #[inline]
78 pub fn help() -> Self {
79 Self::new(Severity::Note)
80 }
81
82 pub fn with_code(self, code: impl Into<String>) -> Self {
84 Diagnostic(self.0.with_code(code))
85 }
86
87 pub fn with_message(self, message: impl Into<String>) -> Self {
89 Diagnostic(self.0.with_message(message))
90 }
91
92 pub fn with_notes<I: Into<String>>(mut self, notes: impl IntoIterator<Item = I>) -> Self {
94 self.0.notes.extend(notes.into_iter().map(Into::into));
95 self
96 }
97
98 pub fn with_highlights(mut self, highlights: impl IntoIterator<Item = Highlight>) -> Self {
100 self.0.labels.extend(highlights.into_iter().map(Into::into));
101 self
102 }
103
104 pub fn write(
107 &self,
108 map: &SourceMap,
109 writer: &mut dyn WriteColor,
110 config: &Config,
111 ) -> Result<(), CRError> {
112 codespan_reporting::term::emit(writer, config, map, &self.0)
113 }
114
115 pub fn print(&self, map: &SourceMap) -> Result<(), codespan_reporting::files::Error> {
119 let stream = StandardStream::stdout(get_stdout_color());
120 let mut lock = stream.lock();
121 self.write(map, &mut lock, &Config::default())
122 }
123}
124
125#[derive(Clone, Debug)]
127pub struct Highlight {
128 pub style: LabelStyle,
131 pub fragment: Fragment,
133 pub message: String,
136}
137
138impl Highlight {
139 pub fn primary(fragment: Fragment, message: impl Into<String>) -> Self {
141 Highlight {
142 style: LabelStyle::Primary,
143 fragment,
144 message: message.into(),
145 }
146 }
147
148 pub fn secondary(fragment: Fragment, message: impl Into<String>) -> Self {
150 Highlight {
151 style: LabelStyle::Secondary,
152 fragment,
153 message: message.into(),
154 }
155 }
156
157 pub fn with_message(mut self, message: impl Into<String>) -> Self {
159 self.message = message.into();
160 self
161 }
162}
163
164impl From<Highlight> for Label<SourceId> {
165 fn from(value: Highlight) -> Self {
166 Label {
167 style: value.style,
168 file_id: value.fragment.source.id,
169 range: value.fragment.range,
170 message: value.message,
171 }
172 }
173}
174
175impl<'f> Files<'f> for SourceMap {
176 type FileId = SourceId;
177
178 type Name = FileName;
179
180 type Source = ImmutableString;
181
182 fn name(&'f self, id: Self::FileId) -> Result<Self::Name, codespan_reporting::files::Error> {
183 self.get(id)
184 .map(|source| source.name().clone())
185 .ok_or(CRError::FileMissing)
186 }
187
188 fn source(
189 &'f self,
190 id: Self::FileId,
191 ) -> Result<Self::Source, codespan_reporting::files::Error> {
192 self.get(id)
193 .map(|source| source.source().clone())
194 .ok_or(CRError::FileMissing)
195 }
196
197 fn line_index(
198 &'f self,
199 id: Self::FileId,
200 byte_index: usize,
201 ) -> Result<usize, codespan_reporting::files::Error> {
202 Ok(self
203 .get(id)
204 .ok_or(CRError::FileMissing)?
205 .line_index(byte_index))
206 }
207
208 fn line_range(
209 &'f self,
210 id: Self::FileId,
211 line_index: usize,
212 ) -> Result<std::ops::Range<usize>, codespan_reporting::files::Error> {
213 Ok(self
214 .get(id)
215 .ok_or(CRError::FileMissing)?
216 .get_line(line_index)
217 .range)
218 }
219}