|
|
|
package progress
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/containerd/console"
|
|
|
|
"github.com/docker/buildx/util/logutil"
|
|
|
|
"github.com/moby/buildkit/client"
|
|
|
|
"github.com/moby/buildkit/util/progress/progressui"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
PrinterModeAuto = "auto"
|
|
|
|
PrinterModeTty = "tty"
|
|
|
|
PrinterModePlain = "plain"
|
|
|
|
PrinterModeQuiet = "quiet"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Printer struct {
|
|
|
|
status chan *client.SolveStatus
|
|
|
|
done <-chan struct{}
|
|
|
|
err error
|
|
|
|
warnings []client.VertexWarning
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Printer) Wait() error {
|
|
|
|
close(p.status)
|
|
|
|
<-p.done
|
|
|
|
return p.err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Printer) Write(s *client.SolveStatus) {
|
|
|
|
p.status <- s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Printer) Warnings() []client.VertexWarning {
|
|
|
|
return p.warnings
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewPrinter(ctx context.Context, w io.Writer, out console.File, mode string) *Printer {
|
|
|
|
statusCh := make(chan *client.SolveStatus)
|
|
|
|
doneCh := make(chan struct{})
|
|
|
|
|
|
|
|
pw := &Printer{
|
|
|
|
status: statusCh,
|
|
|
|
done: doneCh,
|
|
|
|
}
|
|
|
|
|
|
|
|
if v := os.Getenv("BUILDKIT_PROGRESS"); v != "" && mode == PrinterModeAuto {
|
|
|
|
mode = v
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
var c console.Console
|
|
|
|
switch mode {
|
|
|
|
case PrinterModeQuiet:
|
|
|
|
w = ioutil.Discard
|
|
|
|
case PrinterModeAuto, PrinterModeTty:
|
|
|
|
if cons, err := console.ConsoleFromFile(out); err == nil {
|
|
|
|
c = cons
|
|
|
|
}
|
|
|
|
}
|
|
|
|
resumeLogs := logutil.Pause(logrus.StandardLogger())
|
|
|
|
// not using shared context to not disrupt display but let is finish reporting errors
|
|
|
|
pw.warnings, pw.err = progressui.DisplaySolveStatus(ctx, "", c, w, statusCh)
|
|
|
|
resumeLogs()
|
|
|
|
close(doneCh)
|
|
|
|
}()
|
|
|
|
return pw
|
|
|
|
}
|