|
|
|
package monitor
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
"text/tabwriter"
|
|
|
|
|
|
|
|
"github.com/containerd/console"
|
|
|
|
"github.com/docker/buildx/controller/control"
|
|
|
|
controllerapi "github.com/docker/buildx/controller/pb"
|
|
|
|
"github.com/docker/buildx/util/ioset"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
"golang.org/x/term"
|
|
|
|
)
|
|
|
|
|
|
|
|
const helpMessage = `
|
|
|
|
Available commands are:
|
|
|
|
reload reloads the context and build it.
|
|
|
|
rollback re-runs the interactive container with initial rootfs contents.
|
|
|
|
list list buildx sessions.
|
|
|
|
attach attach to a buildx server.
|
|
|
|
disconnect disconnect a client from a buildx server. Specific session ID can be specified an arg.
|
|
|
|
kill kill buildx server.
|
|
|
|
exit exits monitor.
|
|
|
|
help shows this message.
|
|
|
|
`
|
|
|
|
|
|
|
|
// RunMonitor provides an interactive session for running and managing containers via specified IO.
|
|
|
|
func RunMonitor(ctx context.Context, curRef string, options controllerapi.BuildOptions, invokeConfig controllerapi.ContainerConfig, c control.BuildxController, progressMode string, stdin io.ReadCloser, stdout io.WriteCloser, stderr console.File) error {
|
|
|
|
defer func() {
|
|
|
|
if err := c.Disconnect(ctx, curRef); err != nil {
|
|
|
|
logrus.Warnf("disconnect error: %v", err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
monitorIn, monitorOut := ioset.Pipe()
|
|
|
|
defer func() {
|
|
|
|
monitorIn.Close()
|
|
|
|
}()
|
|
|
|
monitorEnableCh := make(chan struct{})
|
|
|
|
monitorDisableCh := make(chan struct{})
|
|
|
|
monitorOutCtx := ioset.MuxOut{
|
|
|
|
Out: monitorOut,
|
|
|
|
EnableHook: func() { monitorEnableCh <- struct{}{} },
|
|
|
|
DisableHook: func() { monitorDisableCh <- struct{}{} },
|
|
|
|
}
|
|
|
|
|
|
|
|
containerIn, containerOut := ioset.Pipe()
|
|
|
|
defer func() {
|
|
|
|
containerIn.Close()
|
|
|
|
}()
|
|
|
|
containerOutCtx := ioset.MuxOut{
|
|
|
|
Out: containerOut,
|
|
|
|
// send newline to hopefully get the prompt; TODO: better UI (e.g. reprinting the last line)
|
|
|
|
EnableHook: func() { containerOut.Stdin.Write([]byte("\n")) },
|
|
|
|
DisableHook: func() {},
|
|
|
|
}
|
|
|
|
|
|
|
|
invokeForwarder := ioset.NewForwarder()
|
|
|
|
invokeForwarder.SetIn(&containerIn)
|
|
|
|
m := &monitor{
|
|
|
|
invokeIO: invokeForwarder,
|
|
|
|
muxIO: ioset.NewMuxIO(ioset.In{
|
|
|
|
Stdin: io.NopCloser(stdin),
|
|
|
|
Stdout: nopCloser{stdout},
|
|
|
|
Stderr: nopCloser{stderr},
|
|
|
|
}, []ioset.MuxOut{monitorOutCtx, containerOutCtx}, 1, func(prev int, res int) string {
|
|
|
|
if prev == 0 && res == 0 {
|
|
|
|
// No toggle happened because container I/O isn't enabled.
|
|
|
|
return "No running interactive containers. You can start one by issuing rollback command\n"
|
|
|
|
}
|
|
|
|
return "Switched IO\n"
|
|
|
|
}),
|
|
|
|
invokeFunc: func(ctx context.Context, ref string, in io.ReadCloser, out io.WriteCloser, err io.WriteCloser) error {
|
|
|
|
return c.Invoke(ctx, ref, invokeConfig, in, out, err)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
// Start container automatically
|
|
|
|
fmt.Fprintf(stdout, "Launching interactive container. Press Ctrl-a-c to switch to monitor console\n")
|
|
|
|
m.rollback(ctx, curRef)
|
|
|
|
|
|
|
|
// Serve monitor commands
|
|
|
|
monitorForwarder := ioset.NewForwarder()
|
|
|
|
monitorForwarder.SetIn(&monitorIn)
|
|
|
|
for {
|
|
|
|
<-monitorEnableCh
|
|
|
|
in, out := ioset.Pipe()
|
|
|
|
monitorForwarder.SetOut(&out)
|
|
|
|
doneCh, errCh := make(chan struct{}), make(chan error)
|
|
|
|
go func() {
|
|
|
|
defer close(doneCh)
|
|
|
|
defer in.Close()
|
|
|
|
go func() {
|
|
|
|
<-ctx.Done()
|
|
|
|
in.Close()
|
|
|
|
}()
|
|
|
|
t := term.NewTerminal(readWriter{in.Stdin, in.Stdout}, "(buildx) ")
|
|
|
|
for {
|
|
|
|
l, err := t.ReadLine()
|
|
|
|
if err != nil {
|
|
|
|
if err != io.EOF {
|
|
|
|
errCh <- err
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
args := strings.Fields(l) // TODO: use shlex
|
|
|
|
if len(args) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
switch args[0] {
|
|
|
|
case "":
|
|
|
|
// nop
|
|
|
|
case "reload":
|
|
|
|
if curRef != "" {
|
|
|
|
if err := c.Disconnect(ctx, curRef); err != nil {
|
|
|
|
fmt.Println("disconnect error", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ref, err := c.Build(ctx, options, nil, stdout, stderr, progressMode) // TODO: support stdin, hold build ref
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("failed to reload: %v\n", err)
|
|
|
|
} else {
|
|
|
|
curRef = ref
|
|
|
|
// rollback the running container with the new result
|
|
|
|
m.rollback(ctx, curRef)
|
|
|
|
fmt.Fprint(stdout, "Interactive container was restarted. Press Ctrl-a-c to switch to the new container\n")
|
|
|
|
}
|
|
|
|
case "rollback":
|
|
|
|
m.rollback(ctx, curRef)
|
|
|
|
fmt.Fprint(stdout, "Interactive container was restarted. Press Ctrl-a-c to switch to the new container\n")
|
|
|
|
case "list":
|
|
|
|
refs, err := c.List(ctx)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("failed to list: %v\n", err)
|
|
|
|
}
|
|
|
|
sort.Strings(refs)
|
|
|
|
tw := tabwriter.NewWriter(stdout, 1, 8, 1, '\t', 0)
|
|
|
|
fmt.Fprintln(tw, "ID\tCURRENT_SESSION")
|
|
|
|
for _, k := range refs {
|
|
|
|
fmt.Fprintf(tw, "%-20s\t%v\n", k, k == curRef)
|
|
|
|
}
|
|
|
|
tw.Flush()
|
|
|
|
case "disconnect":
|
|
|
|
target := curRef
|
|
|
|
if len(args) >= 2 {
|
|
|
|
target = args[1]
|
|
|
|
}
|
|
|
|
if err := c.Disconnect(ctx, target); err != nil {
|
|
|
|
fmt.Println("disconnect error", err)
|
|
|
|
}
|
|
|
|
case "kill":
|
|
|
|
if err := c.Kill(ctx); err != nil {
|
|
|
|
fmt.Printf("failed to kill: %v\n", err)
|
|
|
|
}
|
|
|
|
case "attach":
|
|
|
|
if len(args) < 2 {
|
|
|
|
fmt.Println("attach: server name must be passed")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
ref := args[1]
|
|
|
|
m.rollback(ctx, ref)
|
|
|
|
curRef = ref
|
|
|
|
case "exit":
|
|
|
|
return
|
|
|
|
case "help":
|
|
|
|
fmt.Fprint(stdout, helpMessage)
|
|
|
|
default:
|
|
|
|
fmt.Printf("unknown command: %q\n", l)
|
|
|
|
fmt.Fprint(stdout, helpMessage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
select {
|
|
|
|
case <-doneCh:
|
|
|
|
if m.curInvokeCancel != nil {
|
|
|
|
m.curInvokeCancel()
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
case err := <-errCh:
|
|
|
|
if m.curInvokeCancel != nil {
|
|
|
|
m.curInvokeCancel()
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
case <-monitorDisableCh:
|
|
|
|
}
|
|
|
|
monitorForwarder.SetOut(nil)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type readWriter struct {
|
|
|
|
io.Reader
|
|
|
|
io.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
type monitor struct {
|
|
|
|
muxIO *ioset.MuxIO
|
|
|
|
invokeIO *ioset.Forwarder
|
|
|
|
invokeFunc func(context.Context, string, io.ReadCloser, io.WriteCloser, io.WriteCloser) error
|
|
|
|
curInvokeCancel func()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *monitor) rollback(ctx context.Context, ref string) {
|
|
|
|
if m.curInvokeCancel != nil {
|
|
|
|
m.curInvokeCancel() // Finish the running container if exists
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
// Start a new container
|
|
|
|
if err := m.invoke(ctx, ref); err != nil {
|
|
|
|
logrus.Debugf("invoke error: %v", err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *monitor) invoke(ctx context.Context, ref string) error {
|
|
|
|
m.muxIO.Enable(1)
|
|
|
|
defer m.muxIO.Disable(1)
|
|
|
|
invokeCtx, invokeCancel := context.WithCancel(ctx)
|
|
|
|
|
|
|
|
containerIn, containerOut := ioset.Pipe()
|
|
|
|
m.invokeIO.SetOut(&containerOut)
|
|
|
|
waitInvokeDoneCh := make(chan struct{})
|
|
|
|
var cancelOnce sync.Once
|
|
|
|
curInvokeCancel := func() {
|
|
|
|
cancelOnce.Do(func() {
|
|
|
|
containerIn.Close()
|
|
|
|
m.invokeIO.SetOut(nil)
|
|
|
|
invokeCancel()
|
|
|
|
})
|
|
|
|
<-waitInvokeDoneCh
|
|
|
|
}
|
|
|
|
defer curInvokeCancel()
|
|
|
|
m.curInvokeCancel = curInvokeCancel
|
|
|
|
|
|
|
|
err := m.invokeFunc(invokeCtx, ref, containerIn.Stdin, containerIn.Stdout, containerIn.Stderr)
|
|
|
|
close(waitInvokeDoneCh)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
type nopCloser struct {
|
|
|
|
io.Writer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c nopCloser) Close() error { return nil }
|