1use std::fmt;
2
3use crate::highlighters::Highlighter;
4use crate::highlighters::MietteHighlighter;
5use crate::protocol::Diagnostic;
6use crate::GraphicalReportHandler;
7use crate::GraphicalTheme;
8use crate::NarratableReportHandler;
9use crate::ReportHandler;
10use crate::ThemeCharacters;
11use crate::ThemeStyles;
12
13#[derive(Copy, Clone, Debug, Eq, PartialEq, Default)]
15pub enum RgbColors {
16 Always,
18 Preferred,
20 #[default]
22 Never,
23}
24
25#[derive(Default, Debug, Clone)]
42pub struct MietteHandlerOpts {
43 pub(crate) linkify: Option<bool>,
44 pub(crate) width: Option<usize>,
45 pub(crate) theme: Option<GraphicalTheme>,
46 pub(crate) force_graphical: Option<bool>,
47 pub(crate) force_narrated: Option<bool>,
48 pub(crate) rgb_colors: RgbColors,
49 pub(crate) color: Option<bool>,
50 pub(crate) unicode: Option<bool>,
51 pub(crate) footer: Option<String>,
52 pub(crate) context_lines: Option<usize>,
53 pub(crate) tab_width: Option<usize>,
54 pub(crate) with_cause_chain: Option<bool>,
55 pub(crate) break_words: Option<bool>,
56 pub(crate) wrap_lines: Option<bool>,
57 pub(crate) word_separator: Option<textwrap::WordSeparator>,
58 pub(crate) word_splitter: Option<textwrap::WordSplitter>,
59 pub(crate) highlighter: Option<MietteHighlighter>,
60 pub(crate) show_related_as_nested: Option<bool>,
61}
62
63impl MietteHandlerOpts {
64 pub fn new() -> Self {
66 Default::default()
67 }
68
69 pub fn terminal_links(mut self, linkify: bool) -> Self {
73 self.linkify = Some(linkify);
74 self
75 }
76
77 pub fn graphical_theme(mut self, theme: GraphicalTheme) -> Self {
82 self.theme = Some(theme);
83 self
84 }
85
86 pub fn with_syntax_highlighting(
104 mut self,
105 highlighter: impl Highlighter + Send + Sync + 'static,
106 ) -> Self {
107 self.highlighter = Some(MietteHighlighter::from(highlighter));
108 self
109 }
110
111 pub fn without_syntax_highlighting(mut self) -> Self {
119 self.highlighter = Some(MietteHighlighter::nocolor());
120 self
121 }
122
123 pub fn width(mut self, width: usize) -> Self {
125 self.width = Some(width);
126 self
127 }
128
129 pub fn wrap_lines(mut self, wrap_lines: bool) -> Self {
135 self.wrap_lines = Some(wrap_lines);
136 self
137 }
138
139 pub fn break_words(mut self, break_words: bool) -> Self {
145 self.break_words = Some(break_words);
146 self
147 }
148 pub fn word_separator(mut self, word_separator: textwrap::WordSeparator) -> Self {
150 self.word_separator = Some(word_separator);
151 self
152 }
153
154 pub fn word_splitter(mut self, word_splitter: textwrap::WordSplitter) -> Self {
156 self.word_splitter = Some(word_splitter);
157 self
158 }
159 pub fn with_cause_chain(mut self) -> Self {
161 self.with_cause_chain = Some(true);
162 self
163 }
164
165 pub fn without_cause_chain(mut self) -> Self {
167 self.with_cause_chain = Some(false);
168 self
169 }
170
171 pub fn show_related_errors_as_siblings(mut self) -> Self {
173 self.show_related_as_nested = Some(false);
174 self
175 }
176
177 pub fn show_related_errors_as_nested(mut self) -> Self {
179 self.show_related_as_nested = Some(true);
180 self
181 }
182
183 pub fn color(mut self, color: bool) -> Self {
193 self.color = Some(color);
194 self
195 }
196
197 pub fn rgb_colors(mut self, color: RgbColors) -> Self {
207 self.rgb_colors = color;
208 self
209 }
210
211 pub fn unicode(mut self, unicode: bool) -> Self {
214 self.unicode = Some(unicode);
215 self
216 }
217
218 pub fn force_graphical(mut self, force: bool) -> Self {
221 self.force_graphical = Some(force);
222 self
223 }
224
225 pub fn force_narrated(mut self, force: bool) -> Self {
227 self.force_narrated = Some(force);
228 self
229 }
230
231 pub fn footer(mut self, footer: String) -> Self {
233 self.footer = Some(footer);
234 self
235 }
236
237 pub fn context_lines(mut self, context_lines: usize) -> Self {
239 self.context_lines = Some(context_lines);
240 self
241 }
242
243 pub fn tab_width(mut self, width: usize) -> Self {
245 self.tab_width = Some(width);
246 self
247 }
248
249 pub fn build(self) -> MietteHandler {
251 let graphical = self.is_graphical();
252 let width = self.get_width();
253 if !graphical {
254 let mut handler = NarratableReportHandler::new();
255 if let Some(footer) = self.footer {
256 handler = handler.with_footer(footer);
257 }
258 if let Some(context_lines) = self.context_lines {
259 handler = handler.with_context_lines(context_lines);
260 }
261 if let Some(with_cause_chain) = self.with_cause_chain {
262 if with_cause_chain {
263 handler = handler.with_cause_chain();
264 } else {
265 handler = handler.without_cause_chain();
266 }
267 }
268 MietteHandler {
269 inner: Box::new(handler),
270 }
271 } else {
272 let linkify = self.use_links();
273 let characters = match self.unicode {
274 Some(true) => ThemeCharacters::unicode(),
275 Some(false) => ThemeCharacters::ascii(),
276 None if syscall::supports_unicode() => ThemeCharacters::unicode(),
277 None => ThemeCharacters::ascii(),
278 };
279 let styles = if self.color == Some(false) {
280 ThemeStyles::none()
281 } else if let Some(color_has_16m) = syscall::supports_color_has_16m() {
282 match self.rgb_colors {
283 RgbColors::Always => ThemeStyles::rgb(),
284 RgbColors::Preferred if color_has_16m => ThemeStyles::rgb(),
285 _ => ThemeStyles::ansi(),
286 }
287 } else if self.color == Some(true) {
288 match self.rgb_colors {
289 RgbColors::Always => ThemeStyles::rgb(),
290 _ => ThemeStyles::ansi(),
291 }
292 } else {
293 ThemeStyles::none()
294 };
295 #[cfg(not(feature = "syntect-highlighter"))]
296 let highlighter = self.highlighter.unwrap_or_else(MietteHighlighter::nocolor);
297 #[cfg(feature = "syntect-highlighter")]
298 let highlighter = if self.color == Some(false) {
299 MietteHighlighter::nocolor()
300 } else if self.color == Some(true) || syscall::supports_color() {
301 match self.highlighter {
302 Some(highlighter) => highlighter,
303 None => match self.rgb_colors {
304 RgbColors::Never => MietteHighlighter::nocolor(),
309 _ => MietteHighlighter::syntect_truecolor(),
310 },
311 }
312 } else {
313 MietteHighlighter::nocolor()
314 };
315 let theme = self.theme.unwrap_or(GraphicalTheme { characters, styles });
316 let mut handler = GraphicalReportHandler::new_themed(theme)
317 .with_width(width)
318 .with_links(linkify);
319 handler.highlighter = highlighter;
320 if let Some(with_cause_chain) = self.with_cause_chain {
321 if with_cause_chain {
322 handler = handler.with_cause_chain();
323 } else {
324 handler = handler.without_cause_chain();
325 }
326 }
327 if let Some(footer) = self.footer {
328 handler = handler.with_footer(footer);
329 }
330 if let Some(context_lines) = self.context_lines {
331 handler = handler.with_context_lines(context_lines);
332 }
333 if let Some(w) = self.tab_width {
334 handler = handler.tab_width(w);
335 }
336 if let Some(b) = self.break_words {
337 handler = handler.with_break_words(b)
338 }
339 if let Some(b) = self.wrap_lines {
340 handler = handler.with_wrap_lines(b)
341 }
342 if let Some(s) = self.word_separator {
343 handler = handler.with_word_separator(s)
344 }
345 if let Some(s) = self.word_splitter {
346 handler = handler.with_word_splitter(s)
347 }
348 if let Some(b) = self.show_related_as_nested {
349 handler = handler.with_show_related_as_nested(b)
350 }
351
352 MietteHandler {
353 inner: Box::new(handler),
354 }
355 }
356 }
357
358 pub(crate) fn is_graphical(&self) -> bool {
359 if let Some(force_narrated) = self.force_narrated {
360 !force_narrated
361 } else if let Some(force_graphical) = self.force_graphical {
362 force_graphical
363 } else if let Ok(env) = std::env::var("NO_GRAPHICS") {
364 env == "0"
365 } else {
366 true
367 }
368 }
369
370 pub(crate) fn use_links(&self) -> bool {
373 if let Some(linkify) = self.linkify {
374 linkify
375 } else {
376 syscall::supports_hyperlinks()
377 }
378 }
379
380 pub(crate) fn get_width(&self) -> usize {
381 self.width
382 .unwrap_or_else(|| syscall::terminal_width().unwrap_or(80))
383 }
384}
385
386#[allow(missing_debug_implementations)]
401pub struct MietteHandler {
402 inner: Box<dyn ReportHandler + Send + Sync>,
403}
404
405impl MietteHandler {
406 pub fn new() -> Self {
408 Default::default()
409 }
410}
411
412impl Default for MietteHandler {
413 fn default() -> Self {
414 MietteHandlerOpts::new().build()
415 }
416}
417
418impl ReportHandler for MietteHandler {
419 fn debug(&self, diagnostic: &(dyn Diagnostic), f: &mut fmt::Formatter<'_>) -> fmt::Result {
420 if f.alternate() {
421 return fmt::Debug::fmt(diagnostic, f);
422 }
423
424 self.inner.debug(diagnostic, f)
425 }
426}
427
428mod syscall {
429 use cfg_if::cfg_if;
430
431 #[inline]
432 pub(super) fn terminal_width() -> Option<usize> {
433 cfg_if! {
434 if #[cfg(any(feature = "fancy-no-syscall", miri))] {
435 None
436 } else {
437 terminal_size::terminal_size().map(|size| size.0 .0 as usize)
438 }
439 }
440 }
441
442 #[inline]
443 pub(super) fn supports_hyperlinks() -> bool {
444 cfg_if! {
445 if #[cfg(feature = "fancy-no-syscall")] {
446 false
447 } else {
448 supports_hyperlinks::on(supports_hyperlinks::Stream::Stderr)
449 }
450 }
451 }
452
453 #[cfg(feature = "syntect-highlighter")]
454 #[inline]
455 pub(super) fn supports_color() -> bool {
456 cfg_if! {
457 if #[cfg(feature = "fancy-no-syscall")] {
458 false
459 } else {
460 supports_color::on(supports_color::Stream::Stderr).is_some()
461 }
462 }
463 }
464
465 #[inline]
466 pub(super) fn supports_color_has_16m() -> Option<bool> {
467 cfg_if! {
468 if #[cfg(feature = "fancy-no-syscall")] {
469 None
470 } else {
471 supports_color::on(supports_color::Stream::Stderr).map(|color| color.has_16m)
472 }
473 }
474 }
475
476 #[inline]
477 pub(super) fn supports_unicode() -> bool {
478 cfg_if! {
479 if #[cfg(feature = "fancy-no-syscall")] {
480 false
481 } else {
482 supports_unicode::on(supports_unicode::Stream::Stderr)
483 }
484 }
485 }
486}