123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 |
- /*
- Copyright 2020 Docker Compose CLI authors
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package formatter
- import (
- "context"
- "fmt"
- "io"
- "strconv"
- "strings"
- "sync"
- "time"
- "github.com/buger/goterm"
- "github.com/docker/compose/v2/pkg/api"
- "github.com/docker/docker/pkg/jsonmessage"
- )
- // LogConsumer consume logs from services and format them
- type logConsumer struct {
- ctx context.Context
- presenters sync.Map // map[string]*presenter
- width int
- stdout io.Writer
- stderr io.Writer
- color bool
- prefix bool
- timestamp bool
- }
- // NewLogConsumer creates a new LogConsumer
- func NewLogConsumer(ctx context.Context, stdout, stderr io.Writer, color, prefix, timestamp bool) api.LogConsumer {
- return &logConsumer{
- ctx: ctx,
- presenters: sync.Map{},
- width: 0,
- stdout: stdout,
- stderr: stderr,
- color: color,
- prefix: prefix,
- timestamp: timestamp,
- }
- }
- func (l *logConsumer) Register(name string) {
- l.register(name)
- }
- func (l *logConsumer) register(name string) *presenter {
- var p *presenter
- root, _, found := strings.Cut(name, " ")
- if found {
- parent := l.getPresenter(root)
- p = &presenter{
- colors: parent.colors,
- name: name,
- prefix: parent.prefix,
- }
- } else {
- cf := monochrome
- if l.color {
- if name == api.WatchLogger {
- cf = makeColorFunc("92")
- } else {
- cf = nextColor()
- }
- }
- p = &presenter{
- colors: cf,
- name: name,
- }
- }
- l.presenters.Store(name, p)
- l.computeWidth()
- if l.prefix {
- l.presenters.Range(func(key, value interface{}) bool {
- p := value.(*presenter)
- p.setPrefix(l.width)
- return true
- })
- }
- return p
- }
- func (l *logConsumer) getPresenter(container string) *presenter {
- p, ok := l.presenters.Load(container)
- if !ok { // should have been registered, but ¯\_(ツ)_/¯
- return l.register(container)
- }
- return p.(*presenter)
- }
- // Log formats a log message as received from name/container
- func (l *logConsumer) Log(container, message string) {
- l.write(l.stdout, container, message)
- }
- // Err formats a log message as received from name/container
- func (l *logConsumer) Err(container, message string) {
- l.write(l.stderr, container, message)
- }
- func (l *logConsumer) write(w io.Writer, container, message string) {
- if l.ctx.Err() != nil {
- return
- }
- if KeyboardManager != nil {
- KeyboardManager.ClearKeyboardInfo()
- }
- p := l.getPresenter(container)
- timestamp := time.Now().Format(jsonmessage.RFC3339NanoFixed)
- for _, line := range strings.Split(message, "\n") {
- if l.timestamp {
- _, _ = fmt.Fprintf(w, "%s%s%s\n", p.prefix, timestamp, line)
- } else {
- _, _ = fmt.Fprintf(w, "%s%s\n", p.prefix, line)
- }
- }
- if KeyboardManager != nil {
- KeyboardManager.PrintKeyboardInfo()
- }
- }
- func (l *logConsumer) Status(container, msg string) {
- p := l.getPresenter(container)
- s := p.colors(fmt.Sprintf("%s%s %s\n", goterm.RESET_LINE, container, msg))
- l.stdout.Write([]byte(s)) //nolint:errcheck
- }
- func (l *logConsumer) computeWidth() {
- width := 0
- l.presenters.Range(func(key, value interface{}) bool {
- p := value.(*presenter)
- if len(p.name) > width {
- width = len(p.name)
- }
- return true
- })
- l.width = width + 1
- }
- type presenter struct {
- colors colorFunc
- name string
- prefix string
- }
- func (p *presenter) setPrefix(width int) {
- if p.name == api.WatchLogger {
- p.prefix = p.colors(strings.Repeat(" ", width) + " ⦿ ")
- return
- }
- p.prefix = p.colors(fmt.Sprintf("%-"+strconv.Itoa(width)+"s | ", p.name))
- }
|