2019-04-02 10:48:31 +03:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2022-11-27 21:20:29 +03:00
// SPDX-License-Identifier: MIT
2019-04-02 10:48:31 +03:00
package log
import (
"fmt"
"io"
2019-04-22 23:40:51 +03:00
"reflect"
2019-04-02 10:48:31 +03:00
"strconv"
"strings"
)
const escape = "\033"
// ColorAttribute defines a single SGR Code
type ColorAttribute int
// Base ColorAttributes
const (
Reset ColorAttribute = iota
Bold
Faint
Italic
Underline
BlinkSlow
BlinkRapid
ReverseVideo
Concealed
CrossedOut
)
// Foreground text colors
const (
FgBlack ColorAttribute = iota + 30
FgRed
FgGreen
FgYellow
FgBlue
FgMagenta
FgCyan
FgWhite
)
// Foreground Hi-Intensity text colors
const (
FgHiBlack ColorAttribute = iota + 90
FgHiRed
FgHiGreen
FgHiYellow
FgHiBlue
FgHiMagenta
FgHiCyan
FgHiWhite
)
// Background text colors
const (
BgBlack ColorAttribute = iota + 40
BgRed
BgGreen
BgYellow
BgBlue
BgMagenta
BgCyan
BgWhite
)
// Background Hi-Intensity text colors
const (
BgHiBlack ColorAttribute = iota + 100
BgHiRed
BgHiGreen
BgHiYellow
BgHiBlue
BgHiMagenta
BgHiCyan
BgHiWhite
)
var colorAttributeToString = map [ ColorAttribute ] string {
Reset : "Reset" ,
Bold : "Bold" ,
Faint : "Faint" ,
Italic : "Italic" ,
Underline : "Underline" ,
BlinkSlow : "BlinkSlow" ,
BlinkRapid : "BlinkRapid" ,
ReverseVideo : "ReverseVideo" ,
Concealed : "Concealed" ,
CrossedOut : "CrossedOut" ,
FgBlack : "FgBlack" ,
FgRed : "FgRed" ,
FgGreen : "FgGreen" ,
FgYellow : "FgYellow" ,
FgBlue : "FgBlue" ,
FgMagenta : "FgMagenta" ,
FgCyan : "FgCyan" ,
FgWhite : "FgWhite" ,
FgHiBlack : "FgHiBlack" ,
FgHiRed : "FgHiRed" ,
FgHiGreen : "FgHiGreen" ,
FgHiYellow : "FgHiYellow" ,
FgHiBlue : "FgHiBlue" ,
FgHiMagenta : "FgHiMagenta" ,
FgHiCyan : "FgHiCyan" ,
FgHiWhite : "FgHiWhite" ,
BgBlack : "BgBlack" ,
BgRed : "BgRed" ,
BgGreen : "BgGreen" ,
BgYellow : "BgYellow" ,
BgBlue : "BgBlue" ,
BgMagenta : "BgMagenta" ,
BgCyan : "BgCyan" ,
BgWhite : "BgWhite" ,
BgHiBlack : "BgHiBlack" ,
BgHiRed : "BgHiRed" ,
BgHiGreen : "BgHiGreen" ,
BgHiYellow : "BgHiYellow" ,
BgHiBlue : "BgHiBlue" ,
BgHiMagenta : "BgHiMagenta" ,
BgHiCyan : "BgHiCyan" ,
BgHiWhite : "BgHiWhite" ,
}
func ( c * ColorAttribute ) String ( ) string {
return colorAttributeToString [ * c ]
}
var colorAttributeFromString = map [ string ] ColorAttribute { }
// ColorAttributeFromString will return a ColorAttribute given a string
func ColorAttributeFromString ( from string ) ColorAttribute {
lowerFrom := strings . TrimSpace ( strings . ToLower ( from ) )
return colorAttributeFromString [ lowerFrom ]
}
// ColorString converts a list of ColorAttributes to a color string
func ColorString ( attrs ... ColorAttribute ) string {
return string ( ColorBytes ( attrs ... ) )
}
// ColorBytes converts a list of ColorAttributes to a byte array
func ColorBytes ( attrs ... ColorAttribute ) [ ] byte {
bytes := make ( [ ] byte , 0 , 20 )
bytes = append ( bytes , escape [ 0 ] , '[' )
if len ( attrs ) > 0 {
bytes = append ( bytes , strconv . Itoa ( int ( attrs [ 0 ] ) ) ... )
for _ , a := range attrs [ 1 : ] {
bytes = append ( bytes , ';' )
bytes = append ( bytes , strconv . Itoa ( int ( a ) ) ... )
}
} else {
bytes = append ( bytes , strconv . Itoa ( int ( Bold ) ) ... )
}
bytes = append ( bytes , 'm' )
return bytes
}
2020-10-31 08:36:46 +03:00
var levelToColor = map [ Level ] [ ] byte {
TRACE : ColorBytes ( Bold , FgCyan ) ,
DEBUG : ColorBytes ( Bold , FgBlue ) ,
INFO : ColorBytes ( Bold , FgGreen ) ,
WARN : ColorBytes ( Bold , FgYellow ) ,
ERROR : ColorBytes ( Bold , FgRed ) ,
CRITICAL : ColorBytes ( Bold , BgMagenta ) ,
FATAL : ColorBytes ( Bold , BgRed ) ,
NONE : ColorBytes ( Reset ) ,
2019-04-02 10:48:31 +03:00
}
2022-01-20 20:46:10 +03:00
var (
resetBytes = ColorBytes ( Reset )
fgCyanBytes = ColorBytes ( FgCyan )
fgGreenBytes = ColorBytes ( FgGreen )
fgBoldBytes = ColorBytes ( Bold )
)
2019-04-02 10:48:31 +03:00
type protectedANSIWriterMode int
const (
escapeAll protectedANSIWriterMode = iota
allowColor
removeColor
)
type protectedANSIWriter struct {
w io . Writer
mode protectedANSIWriterMode
}
// Write will protect against unusual characters
func ( c * protectedANSIWriter ) Write ( bytes [ ] byte ) ( int , error ) {
end := len ( bytes )
totalWritten := 0
normalLoop :
for i := 0 ; i < end ; {
lasti := i
if c . mode == escapeAll {
2019-04-22 23:40:51 +03:00
for i < end && ( bytes [ i ] >= ' ' || bytes [ i ] == '\n' || bytes [ i ] == '\t' ) {
2019-04-02 10:48:31 +03:00
i ++
}
} else {
2019-04-22 23:40:51 +03:00
// Allow tabs if we're not escaping everything
for i < end && ( bytes [ i ] >= ' ' || bytes [ i ] == '\t' ) {
2019-04-02 10:48:31 +03:00
i ++
}
}
if i > lasti {
written , err := c . w . Write ( bytes [ lasti : i ] )
2019-06-12 22:41:28 +03:00
totalWritten += written
2019-04-02 10:48:31 +03:00
if err != nil {
return totalWritten , err
}
}
if i >= end {
break
}
// If we're not just escaping all we should prefix all newlines with a \t
if c . mode != escapeAll {
if bytes [ i ] == '\n' {
written , err := c . w . Write ( [ ] byte { '\n' , '\t' } )
if written > 0 {
totalWritten ++
}
if err != nil {
return totalWritten , err
}
i ++
continue normalLoop
}
if bytes [ i ] == escape [ 0 ] && i + 1 < end && bytes [ i + 1 ] == '[' {
for j := i + 2 ; j < end ; j ++ {
if bytes [ j ] >= '0' && bytes [ j ] <= '9' {
continue
}
if bytes [ j ] == ';' {
continue
}
if bytes [ j ] == 'm' {
if c . mode == allowColor {
written , err := c . w . Write ( bytes [ i : j + 1 ] )
2019-06-12 22:41:28 +03:00
totalWritten += written
2019-04-02 10:48:31 +03:00
if err != nil {
return totalWritten , err
}
} else {
totalWritten = j
}
i = j + 1
continue normalLoop
}
break
}
}
}
// Process naughty character
2021-12-20 00:00:22 +03:00
if _ , err := fmt . Fprintf ( c . w , ` \%#03o ` , bytes [ i ] ) ; err != nil {
2019-04-02 10:48:31 +03:00
return totalWritten , err
}
i ++
totalWritten ++
}
return totalWritten , nil
}
2019-04-22 23:40:51 +03:00
// ColorSprintf returns a colored string from a format and arguments
// arguments will be wrapped in ColoredValues to protect against color spoofing
func ColorSprintf ( format string , args ... interface { } ) string {
if len ( args ) > 0 {
v := make ( [ ] interface { } , len ( args ) )
for i := 0 ; i < len ( v ) ; i ++ {
v [ i ] = NewColoredValuePointer ( & args [ i ] )
}
return fmt . Sprintf ( format , v ... )
}
2019-06-12 22:41:28 +03:00
return format
2019-04-22 23:40:51 +03:00
}
// ColorFprintf will write to the provided writer similar to ColorSprintf
func ColorFprintf ( w io . Writer , format string , args ... interface { } ) ( int , error ) {
if len ( args ) > 0 {
v := make ( [ ] interface { } , len ( args ) )
for i := 0 ; i < len ( v ) ; i ++ {
v [ i ] = NewColoredValuePointer ( & args [ i ] )
}
return fmt . Fprintf ( w , format , v ... )
}
2019-06-12 22:41:28 +03:00
return fmt . Fprint ( w , format )
2019-04-22 23:40:51 +03:00
}
// ColorFormatted structs provide their own colored string when formatted with ColorSprintf
type ColorFormatted interface {
// ColorFormat provides the colored representation of the value
ColorFormat ( s fmt . State )
}
var colorFormattedType = reflect . TypeOf ( ( * ColorFormatted ) ( nil ) ) . Elem ( )
2019-04-02 10:48:31 +03:00
// ColoredValue will Color the provided value
type ColoredValue struct {
2019-04-07 03:25:14 +03:00
colorBytes * [ ] byte
resetBytes * [ ] byte
2019-04-02 10:48:31 +03:00
Value * interface { }
}
// NewColoredValue is a helper function to create a ColoredValue from a Value
// If no color is provided it defaults to Bold with standard Reset
// If a ColoredValue is provided it is not changed
func NewColoredValue ( value interface { } , color ... ColorAttribute ) * ColoredValue {
return NewColoredValuePointer ( & value , color ... )
}
// NewColoredValuePointer is a helper function to create a ColoredValue from a Value Pointer
// If no color is provided it defaults to Bold with standard Reset
// If a ColoredValue is provided it is not changed
func NewColoredValuePointer ( value * interface { } , color ... ColorAttribute ) * ColoredValue {
if val , ok := ( * value ) . ( * ColoredValue ) ; ok {
return val
}
if len ( color ) > 0 {
bytes := ColorBytes ( color ... )
return & ColoredValue {
2019-04-07 03:25:14 +03:00
colorBytes : & bytes ,
resetBytes : & resetBytes ,
2019-04-02 10:48:31 +03:00
Value : value ,
}
}
return & ColoredValue {
2019-04-07 03:25:14 +03:00
colorBytes : & fgBoldBytes ,
resetBytes : & resetBytes ,
2019-04-02 10:48:31 +03:00
Value : value ,
}
}
// NewColoredValueBytes creates a value from the provided value with color bytes
// If a ColoredValue is provided it is not changed
func NewColoredValueBytes ( value interface { } , colorBytes * [ ] byte ) * ColoredValue {
if val , ok := value . ( * ColoredValue ) ; ok {
return val
}
return & ColoredValue {
2019-04-07 03:25:14 +03:00
colorBytes : colorBytes ,
resetBytes : & resetBytes ,
2019-04-02 10:48:31 +03:00
Value : & value ,
}
}
2019-04-22 23:40:51 +03:00
// NewColoredIDValue is a helper function to create a ColoredValue from a Value
// The Value will be colored with FgCyan
// If a ColoredValue is provided it is not changed
func NewColoredIDValue ( value interface { } ) * ColoredValue {
2020-07-23 12:26:45 +03:00
return NewColoredValueBytes ( value , & fgCyanBytes )
2019-04-22 23:40:51 +03:00
}
// Format will format the provided value and protect against ANSI color spoofing within the value
// If the wrapped value is ColorFormatted and the format is "%-v" then its ColorString will
// be used. It is presumed that this ColorString is safe.
2019-04-02 10:48:31 +03:00
func ( cv * ColoredValue ) Format ( s fmt . State , c rune ) {
2019-04-22 23:40:51 +03:00
if c == 'v' && s . Flag ( '-' ) {
if val , ok := ( * cv . Value ) . ( ColorFormatted ) ; ok {
val . ColorFormat ( s )
return
}
v := reflect . ValueOf ( * cv . Value )
t := v . Type ( )
if reflect . PtrTo ( t ) . Implements ( colorFormattedType ) {
vp := reflect . New ( t )
vp . Elem ( ) . Set ( v )
val := vp . Interface ( ) . ( ColorFormatted )
val . ColorFormat ( s )
return
}
}
2019-07-23 21:50:39 +03:00
s . Write ( * cv . colorBytes )
2019-04-02 10:48:31 +03:00
fmt . Fprintf ( & protectedANSIWriter { w : s } , fmtString ( s , c ) , * ( cv . Value ) )
2019-07-23 21:50:39 +03:00
s . Write ( * cv . resetBytes )
2019-04-07 03:25:14 +03:00
}
2023-02-04 02:11:48 +03:00
// ColorFormatAsString returns the result of the ColorFormat without the color
func ColorFormatAsString ( colorVal ColorFormatted ) string {
s := new ( strings . Builder )
_ , _ = ColorFprintf ( & protectedANSIWriter { w : s , mode : removeColor } , "%-v" , colorVal )
return s . String ( )
}
2019-04-07 03:25:14 +03:00
// SetColorBytes will allow a user to set the colorBytes of a colored value
func ( cv * ColoredValue ) SetColorBytes ( colorBytes [ ] byte ) {
cv . colorBytes = & colorBytes
}
// SetColorBytesPointer will allow a user to set the colorBytes pointer of a colored value
func ( cv * ColoredValue ) SetColorBytesPointer ( colorBytes * [ ] byte ) {
cv . colorBytes = colorBytes
}
// SetResetBytes will allow a user to set the resetBytes pointer of a colored value
func ( cv * ColoredValue ) SetResetBytes ( resetBytes [ ] byte ) {
cv . resetBytes = & resetBytes
}
// SetResetBytesPointer will allow a user to set the resetBytes pointer of a colored value
func ( cv * ColoredValue ) SetResetBytesPointer ( resetBytes * [ ] byte ) {
cv . resetBytes = resetBytes
2019-04-02 10:48:31 +03:00
}
func fmtString ( s fmt . State , c rune ) string {
var width , precision string
base := make ( [ ] byte , 0 , 8 )
base = append ( base , '%' )
for _ , c := range [ ] byte ( " +-#0" ) {
if s . Flag ( int ( c ) ) {
base = append ( base , c )
}
}
if w , ok := s . Width ( ) ; ok {
width = strconv . Itoa ( w )
}
if p , ok := s . Precision ( ) ; ok {
precision = "." + strconv . Itoa ( p )
}
return fmt . Sprintf ( "%s%s%s%c" , base , width , precision , c )
}
func init ( ) {
for attr , from := range colorAttributeToString {
colorAttributeFromString [ strings . ToLower ( from ) ] = attr
}
}