From 511919eb9ab3c3ce11e28b2d509a605d07771d71 Mon Sep 17 00:00:00 2001 From: Chenfeng Bao Date: Tue, 23 Jul 2024 17:22:38 -0400 Subject: [PATCH] use ANSI escape codes to control all console text colours --- src/D2L.Bmx/ConsoleWriter.cs | 30 ++++++++++++++++++++++++++++++ src/D2L.Bmx/OktaAuthenticator.cs | 9 ++++----- src/D2L.Bmx/Program.cs | 12 +++++------- src/D2L.Bmx/UpdateChecker.cs | 25 ++----------------------- 4 files changed, 41 insertions(+), 35 deletions(-) diff --git a/src/D2L.Bmx/ConsoleWriter.cs b/src/D2L.Bmx/ConsoleWriter.cs index c516e47f..04b5cb82 100644 --- a/src/D2L.Bmx/ConsoleWriter.cs +++ b/src/D2L.Bmx/ConsoleWriter.cs @@ -2,6 +2,9 @@ namespace D2L.Bmx; internal interface IConsoleWriter { void WriteParameter( string description, string value, ParameterSource source ); + void WriteUpdateMessage( string text ); + void WriteWarning( string text ); + void WriteError( string text ); } // We use ANSI escape codes to control colours, because .NET's `Console.ForegroundColor` only targets stdout, @@ -21,4 +24,31 @@ void IConsoleWriter.WriteParameter( string description, string value, ParameterS // source: grey / bright black Console.Error.WriteLine( $"\x1b[0m{description}: \x1b[96m{value} \x1b[90m(from {source})\x1b[0m" ); } + + void IConsoleWriter.WriteUpdateMessage( string text ) { + if( _noColor || !VirtualTerminal.TryEnableOnStderr() ) { + Console.Error.WriteLine( text ); + } + string[] lines = text.Split( '\n' ); + int maxLineLength = lines.Max( l => l.Length ); + foreach( string line in lines ) { + Console.Error.WriteLine( $"\x1b[0m\x1b[30;47m{line.PadRight( maxLineLength )}\x1b[0m" ); + } + } + + void IConsoleWriter.WriteWarning( string text ) { + if( _noColor || !VirtualTerminal.TryEnableOnStderr() ) { + Console.Error.WriteLine( text ); + } + // bright yellow - 93 + Console.Error.WriteLine( $"\x1b[0m\x1b[93m{text}\x1b[0m" ); + } + + void IConsoleWriter.WriteError( string text ) { + if( _noColor || !VirtualTerminal.TryEnableOnStderr() ) { + Console.Error.WriteLine( text ); + } + // bright red - 91 + Console.Error.WriteLine( $"\x1b[0m\x1b[91m{text}\x1b[0m" ); + } } diff --git a/src/D2L.Bmx/OktaAuthenticator.cs b/src/D2L.Bmx/OktaAuthenticator.cs index b82ae026..8d480164 100644 --- a/src/D2L.Bmx/OktaAuthenticator.cs +++ b/src/D2L.Bmx/OktaAuthenticator.cs @@ -91,11 +91,10 @@ mfaFactor is OktaMfaQuestionFactor // Security question factor is a static value if( File.Exists( BmxPaths.CONFIG_FILE_NAME ) ) { CacheOktaSession( user, org, sessionResp.Id, sessionResp.ExpiresAt ); } else { - Console.ResetColor(); - Console.ForegroundColor = ConsoleColor.Yellow; - Console.Error.WriteLine( "No config file found. Your Okta session will not be cached. " + - "Consider running `bmx configure` if you own this machine." ); - Console.ResetColor(); + consoleWriter.WriteWarning( + "No config file found. Your Okta session will not be cached. " + + "Consider running `bmx configure` if you own this machine." + ); } return new AuthenticatedOktaApi( Org: org, User: user, Api: oktaApi ); } diff --git a/src/D2L.Bmx/Program.cs b/src/D2L.Bmx/Program.cs index 1b4b8065..3ae8fa3f 100644 --- a/src/D2L.Bmx/Program.cs +++ b/src/D2L.Bmx/Program.cs @@ -246,7 +246,7 @@ } if( context.ParseResult.CommandResult.Command != updateCommand ) { - var updateChecker = new UpdateChecker( new GitHubClient(), new VersionProvider() ); + var updateChecker = new UpdateChecker( new GitHubClient(), new VersionProvider(), new ConsoleWriter() ); await updateChecker.CheckForUpdatesAsync(); } @@ -258,17 +258,15 @@ order: MiddlewareOrder.ExceptionHandler + 1 ) .UseExceptionHandler( ( exception, context ) => { - Console.ResetColor(); - Console.ForegroundColor = ConsoleColor.Red; + IConsoleWriter consoleWriter = new ConsoleWriter(); if( exception is BmxException ) { - Console.Error.WriteLine( exception.Message ); + consoleWriter.WriteError( exception.Message ); } else { - Console.Error.WriteLine( "BMX exited with unexpected internal error" ); + consoleWriter.WriteError( "BMX exited with unexpected internal error" ); } if( Environment.GetEnvironmentVariable( "BMX_DEBUG" ) == "1" ) { - Console.Error.WriteLine( exception ); + consoleWriter.WriteError( exception.ToString() ); } - Console.ResetColor(); } ) .Build() .InvokeAsync( args ); diff --git a/src/D2L.Bmx/UpdateChecker.cs b/src/D2L.Bmx/UpdateChecker.cs index bb04c41d..2703a5b4 100644 --- a/src/D2L.Bmx/UpdateChecker.cs +++ b/src/D2L.Bmx/UpdateChecker.cs @@ -5,7 +5,7 @@ namespace D2L.Bmx; -internal class UpdateChecker( IGitHubClient github, IVersionProvider versionProvider ) { +internal class UpdateChecker( IGitHubClient github, IVersionProvider versionProvider, IConsoleWriter consoleWriter ) { public async Task CheckForUpdatesAsync() { try { var updateCheckCache = GetUpdateCheckCacheOrNull(); @@ -29,7 +29,7 @@ public async Task CheckForUpdatesAsync() { Version? localVersion = versionProvider.GetAssemblyVersion(); if( latestVersion > localVersion ) { string? displayVersion = versionProvider.GetInformationalVersion() ?? localVersion.ToString(); - DisplayUpdateMessage( + consoleWriter.WriteUpdateMessage( $""" A new BMX release is available: v{latestVersion} (current: {displayVersion}) Run "bmx update" now to upgrade. @@ -41,27 +41,6 @@ public async Task CheckForUpdatesAsync() { } } - private static void DisplayUpdateMessage( string message ) { - var originalBackgroundColor = Console.BackgroundColor; - var originalForegroundColor = Console.ForegroundColor; - - Console.BackgroundColor = ConsoleColor.Gray; - Console.ForegroundColor = ConsoleColor.Black; - - string[] lines = message.Split( "\n" ); - int consoleWidth = Console.WindowWidth; - - foreach( string line in lines ) { - Console.Error.Write( line.PadRight( consoleWidth ) ); - Console.Error.WriteLine(); - } - - Console.BackgroundColor = originalBackgroundColor; - Console.ForegroundColor = originalForegroundColor; - Console.ResetColor(); - Console.Error.WriteLine(); - } - private static void SetUpdateCheckCache( Version version ) { var cache = new UpdateCheckCache( VersionName: version,