From 58dd096446dcfb66ab3e9e684377f8f47a62213a Mon Sep 17 00:00:00 2001 From: ilslv <47687266+ilslv@users.noreply.github.com> Date: Mon, 13 Dec 2021 17:15:20 +0300 Subject: [PATCH] Fix `writer:: Summarize` ignoring `Coloring` CLI options (#189, #186) - add `cli::Colored` trait for propagating `Coloring` to arbitrary `Writer`s --- CHANGELOG.md | 4 +++- src/cli.rs | 35 +++++++++++++++++++++++++++++++++++ src/writer/basic.rs | 13 ++++++++----- src/writer/out.rs | 11 +++++++++++ src/writer/summarize.rs | 7 ++++++- 5 files changed, 63 insertions(+), 7 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 44eac612..769ab872 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -35,13 +35,15 @@ All user visible changes to `cucumber` crate will be documented in this file. Th - Support for [Cucumber Expressions] via `#[given(expr = ...)]`, `#[when(expr = ...)]` and `#[then(expr = ...)]` syntax. ([#157]) - Support for custom parameters in [Cucumber Expressions] via `#[derive(cucumber::Parameter)]` macro. ([#168]) - Merging tags from `Feature` and `Rule` with `Scenario` when filtering with `--tags` CLI option. ([#166]) -- `writer::AssertNormalized` forcing `Normalized` implementation. ([#182]) +- `writer::AssertNormalized` forcing `Normalized` implementation. ([#182]) +- `cli::Colored` trait for propagating `Coloring` to arbitrary `Writer`s. ([#189]) ### Fixed - Template regex in `Scenario Outline` expansion from `<(\S+)>` to `<([^>\s]+)>`. ([#163]) - Multiple `Examples` in `Scenario Outline`. ([#165], [#164]) - Docstring and name expansion in `Scenario Outline`. ([#178], [#172]) +- `writer::Summarized` ignoring `Coloring` options. ([#189], [#188]) [#147]: /../../pull/147 [#151]: /../../pull/151 diff --git a/src/cli.rs b/src/cli.rs index 64b17b0a..f82ba3f0 100644 --- a/src/cli.rs +++ b/src/cli.rs @@ -31,6 +31,8 @@ use gherkin::tagexpr::TagOperation; use regex::Regex; use structopt::StructOpt; +use crate::writer::Coloring; + // Workaround for overwritten doc-comments. // https://github.com/TeXitoi/structopt/issues/333#issuecomment-712265332 #[cfg_attr( @@ -147,6 +149,20 @@ where pub custom: Custom, } +/// Indication whether a [`Writer`] using CLI options supports colored output. +/// +/// [`Writer`]: crate::Writer +pub trait Colored { + /// Returns [`Coloring`] indicating whether a [`Writer`] using CLI options + /// supports colored output or not. + /// + /// [`Writer`]: crate::Writer + #[must_use] + fn coloring(&self) -> Coloring { + Coloring::Never + } +} + // Workaround for overwritten doc-comments. // https://github.com/TeXitoi/structopt/issues/333#issuecomment-712265332 #[cfg_attr(doc, doc = "Empty CLI options.")] @@ -163,6 +179,8 @@ pub struct Empty { skipped: (), } +impl Colored for Empty {} + // Workaround for overwritten doc-comments. // https://github.com/TeXitoi/structopt/issues/333#issuecomment-712265332 #[cfg_attr( @@ -208,6 +226,8 @@ where // Useful blanket impls: +impl cli::Colored for Cli {} + #[async_trait(?Send)] impl<'val, W, Wr, Val> writer::Arbitrary<'val, W, Val> for CustomWriter where @@ -276,3 +296,18 @@ impl Compose { (left, right) } } + +impl Colored for Compose +where + L: Colored + StructOpt, + R: Colored + StructOpt, +{ + fn coloring(&self) -> Coloring { + // Basically, founds "maximum" `Coloring` of CLI options. + match (self.left.coloring(), self.right.coloring()) { + (Coloring::Always, _) | (_, Coloring::Always) => Coloring::Always, + (Coloring::Auto, _) | (_, Coloring::Auto) => Coloring::Auto, + (Coloring::Never, Coloring::Never) => Coloring::Never, + } + } +} diff --git a/src/writer/basic.rs b/src/writer/basic.rs index 6cda1790..8c7614ac 100644 --- a/src/writer/basic.rs +++ b/src/writer/basic.rs @@ -25,6 +25,7 @@ use regex::CaptureLocations; use structopt::StructOpt; use crate::{ + cli::Colored, event::{self, Info}, parser, writer::{ @@ -54,6 +55,12 @@ pub struct Cli { pub color: Coloring, } +impl Colored for Cli { + fn coloring(&self) -> Coloring { + self.color + } +} + /// Possible policies of a [`console`] output coloring. #[derive(Clone, Copy, Debug)] pub enum Coloring { @@ -221,11 +228,7 @@ impl Basic { if cli.verbose { self.verbose = true; } - match cli.color { - Coloring::Auto => {} - Coloring::Always => self.styles.is_present = true, - Coloring::Never => self.styles.is_present = false, - } + self.styles.apply_coloring(cli.color); } /// Clears last `n` lines if [`Coloring`] is enabled. diff --git a/src/writer/out.rs b/src/writer/out.rs index 28a88b4b..f9c4374e 100644 --- a/src/writer/out.rs +++ b/src/writer/out.rs @@ -15,6 +15,8 @@ use std::{borrow::Cow, io, str}; use console::Style; use derive_more::{Deref, DerefMut, Display, From, Into}; +use super::Coloring; + /// [`Style`]s for terminal output. #[derive(Debug)] pub struct Styles { @@ -58,6 +60,15 @@ impl Styles { Self::default() } + /// Applies the given [`Coloring`] to these [`Styles`]. + pub fn apply_coloring(&mut self, color: Coloring) { + match color { + Coloring::Auto => {} + Coloring::Always => self.is_present = true, + Coloring::Never => self.is_present = false, + } + } + /// If terminal is present colors `input` with [`Styles::ok`] color or /// leaves "as is" otherwise. #[must_use] diff --git a/src/writer/summarize.rs b/src/writer/summarize.rs index 50270d46..e716bbfb 100644 --- a/src/writer/summarize.rs +++ b/src/writer/summarize.rs @@ -17,6 +17,7 @@ use derive_more::Deref; use itertools::Itertools as _; use crate::{ + cli::Colored, event, parser, writer::{self, out::Styles}, Event, World, Writer, @@ -165,6 +166,7 @@ impl Writer for Summarize where W: World, Wr: for<'val> writer::Arbitrary<'val, W, String> + Summarizable, + Wr::Cli: Colored, { type Cli = Wr::Cli; @@ -204,7 +206,10 @@ where if let State::FinishedButNotOutput = self.state { self.state = State::FinishedAndOutput; - self.writer.write(Styles::new().summary(self)).await; + + let mut styles = Styles::new(); + styles.apply_coloring(cli.coloring()); + self.writer.write(styles.summary(self)).await; } } }