mirror of
https://github.com/go-task/task.git
synced 2025-02-09 13:47:06 +02:00
116 lines
2.4 KiB
Go
116 lines
2.4 KiB
Go
package output
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/go-task/task/v3/internal/logger"
|
|
"github.com/go-task/task/v3/internal/templater"
|
|
)
|
|
|
|
type Prefixed struct {
|
|
logger *logger.Logger
|
|
seen map[string]uint
|
|
counter *uint
|
|
mutex sync.Mutex
|
|
}
|
|
|
|
func NewPrefixed(logger *logger.Logger) *Prefixed {
|
|
var counter uint
|
|
|
|
return &Prefixed{
|
|
seen: make(map[string]uint),
|
|
counter: &counter,
|
|
logger: logger,
|
|
}
|
|
}
|
|
|
|
func (p *Prefixed) WrapWriter(stdOut, _ io.Writer, prefix string, _ *templater.Cache) (io.Writer, io.Writer, CloseFunc) {
|
|
pw := &prefixWriter{writer: stdOut, prefix: prefix, prefixed: p}
|
|
return pw, pw, func(error) error { return pw.close() }
|
|
}
|
|
|
|
type prefixWriter struct {
|
|
writer io.Writer
|
|
prefixed *Prefixed
|
|
prefix string
|
|
buff bytes.Buffer
|
|
}
|
|
|
|
func (pw *prefixWriter) Write(p []byte) (int, error) {
|
|
n, err := pw.buff.Write(p)
|
|
if err != nil {
|
|
return n, err
|
|
}
|
|
|
|
return n, pw.writeOutputLines(false)
|
|
}
|
|
|
|
func (pw *prefixWriter) close() error {
|
|
return pw.writeOutputLines(true)
|
|
}
|
|
|
|
func (pw *prefixWriter) writeOutputLines(force bool) error {
|
|
for {
|
|
switch line, err := pw.buff.ReadString('\n'); err {
|
|
case nil:
|
|
if err = pw.writeLine(line); err != nil {
|
|
return err
|
|
}
|
|
case io.EOF:
|
|
// if this line was not a complete line, re-add to the buffer
|
|
if !force && !strings.HasSuffix(line, "\n") {
|
|
_, err = pw.buff.WriteString(line)
|
|
return err
|
|
}
|
|
|
|
return pw.writeLine(line)
|
|
default:
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
var PrefixColorSequence = []logger.Color{
|
|
logger.Yellow, logger.Blue, logger.Magenta, logger.Cyan, logger.Green, logger.Red,
|
|
logger.BrightYellow, logger.BrightBlue, logger.BrightMagenta, logger.BrightCyan, logger.BrightGreen, logger.BrightRed,
|
|
}
|
|
|
|
func (pw *prefixWriter) writeLine(line string) error {
|
|
if line == "" {
|
|
return nil
|
|
}
|
|
if !strings.HasSuffix(line, "\n") {
|
|
line += "\n"
|
|
}
|
|
|
|
defer pw.prefixed.mutex.Unlock()
|
|
pw.prefixed.mutex.Lock()
|
|
|
|
idx, ok := pw.prefixed.seen[pw.prefix]
|
|
|
|
if !ok {
|
|
idx = *pw.prefixed.counter
|
|
pw.prefixed.seen[pw.prefix] = idx
|
|
|
|
*pw.prefixed.counter++
|
|
}
|
|
|
|
if _, err := fmt.Fprint(pw.writer, "["); err != nil {
|
|
return nil
|
|
}
|
|
|
|
color := PrefixColorSequence[idx%uint(len(PrefixColorSequence))]
|
|
pw.prefixed.logger.FOutf(pw.writer, color, pw.prefix)
|
|
|
|
if _, err := fmt.Fprint(pw.writer, "] "); err != nil {
|
|
return nil
|
|
}
|
|
|
|
_, err := fmt.Fprint(pw.writer, line)
|
|
return err
|
|
}
|