pull logs and events better than aggregate events from multiple channels

Signed-off-by: Nicolas De Loof <nicolas.deloof@gmail.com>
This commit is contained in:
Nicolas De Loof 2021-02-09 16:51:48 +01:00
parent 17c26e81ff
commit a4b003ecfa
No known key found for this signature in database
GPG Key ID: 9858809D6F8F6E7E
10 changed files with 145 additions and 111 deletions

View File

@ -65,10 +65,8 @@ type CreateOptions struct {
// StartOptions group options of the Start API // StartOptions group options of the Start API
type StartOptions struct { type StartOptions struct {
// Attach will attach to container and pipe stdout/stderr to LogConsumer // Attach will attach to service containers and pipe stdout/stderr to channel
Attach LogConsumer Attach chan ContainerEvent
// Listener will get notified on container events
Listener chan ContainerExited
} }
// UpOptions group options of the Up API // UpOptions group options of the Up API
@ -185,11 +183,21 @@ type Stack struct {
// LogConsumer is a callback to process log messages from services // LogConsumer is a callback to process log messages from services
type LogConsumer interface { type LogConsumer interface {
Log(service, container, message string) Log(service, container, message string)
Status(service, container, message string) Status(service, container, msg string)
} }
// ContainerExited let us know a Container exited // ContainerEvent notify an event has been collected on Source container implementing Service
type ContainerExited struct { type ContainerEvent struct {
Service string Type int
Status int Source string
Service string
Line string
ExitCode int
} }
const (
// ContainerEventLog is a ContainerEvent of type log. Line is set
ContainerEventLog = iota
// ContainerEventExit is a ContainerEvent of type exit. ExitCode is set
ContainerEventExit
)

View File

@ -18,14 +18,12 @@ package compose
import ( import (
"context" "context"
"os"
"github.com/spf13/cobra"
"github.com/docker/compose-cli/api/client" "github.com/docker/compose-cli/api/client"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/docker/compose-cli/api/progress" "github.com/docker/compose-cli/api/progress"
"github.com/docker/compose-cli/cli/formatter"
"github.com/spf13/cobra"
) )
type startOptions struct { type startOptions struct {
@ -67,7 +65,23 @@ func runStart(ctx context.Context, opts startOptions, services []string) error {
return err return err
} }
return c.ComposeService().Start(ctx, project, compose.StartOptions{ queue := make(chan compose.ContainerEvent)
Attach: formatter.NewLogConsumer(ctx, os.Stdout), printer := printer{
queue: queue,
}
err = c.ComposeService().Start(ctx, project, compose.StartOptions{
Attach: queue,
}) })
if err != nil {
return err
}
_, err = printer.run(ctx, false, func() error {
ctx := context.Background()
_, err := progress.Run(ctx, func(ctx context.Context) (string, error) {
return "", c.ComposeService().Stop(ctx, project)
})
return err
})
return err
} }

View File

@ -18,11 +18,11 @@ package compose
import ( import (
"context" "context"
"errors"
"fmt" "fmt"
"github.com/sirupsen/logrus"
"os" "os"
"os/signal"
"path/filepath" "path/filepath"
"syscall"
"github.com/docker/compose-cli/api/client" "github.com/docker/compose-cli/api/client"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
@ -31,6 +31,7 @@ import (
"github.com/docker/compose-cli/cli/formatter" "github.com/docker/compose-cli/cli/formatter"
"github.com/compose-spec/compose-go/types" "github.com/compose-spec/compose-go/types"
"github.com/sirupsen/logrus"
"github.com/spf13/cobra" "github.com/spf13/cobra"
) )
@ -151,47 +152,35 @@ func runCreateStart(ctx context.Context, opts upOptions, services []string) erro
return nil return nil
} }
ctx, cancel := context.WithCancel(ctx) queue := make(chan compose.ContainerEvent)
listener := make(chan compose.ContainerExited) printer := printer{
exitCode := make(chan int) queue: queue,
}
stopFunc := func() error {
ctx := context.Background()
_, err := progress.Run(ctx, func(ctx context.Context) (string, error) {
return "", c.ComposeService().Stop(ctx, project)
})
return err
}
signalChan := make(chan os.Signal, 1)
signal.Notify(signalChan, syscall.SIGINT, syscall.SIGTERM)
go func() { go func() {
var aborting bool <-signalChan
for { fmt.Println("Gracefully stopping...")
exit := <-listener stopFunc() // nolint:errcheck
if opts.cascadeStop && !aborting {
aborting = true
cancel()
exitCode <- exit.Status
}
}
}() }()
err = c.ComposeService().Start(ctx, project, compose.StartOptions{ err = c.ComposeService().Start(ctx, project, compose.StartOptions{
Attach: formatter.NewLogConsumer(ctx, os.Stdout), Attach: queue,
Listener: listener,
}) })
if err != nil {
if errors.Is(ctx.Err(), context.Canceled) { return err
select {
case exit := <-exitCode:
fmt.Println("Aborting on container exit...")
err = stop(c, project)
logrus.Error(exit)
// os.Exit(exit)
default:
// cancelled by user
fmt.Println("Gracefully stopping...")
err = stop(c, project)
}
} }
return err
}
func stop(c *client.Client, project *types.Project) error { _, err = printer.run(ctx, opts.cascadeStop, stopFunc)
ctx := context.Background() // FIXME os.Exit
_, err := progress.Run(ctx, func(ctx context.Context) (string, error) {
return "", c.ComposeService().Stop(ctx, project)
})
return err return err
} }
@ -235,3 +224,26 @@ func setup(ctx context.Context, opts composeOptions, services []string) (*client
return c, project, nil return c, project, nil
} }
type printer struct {
queue chan compose.ContainerEvent
}
func (p printer) run(ctx context.Context, cascadeStop bool, stopFn func() error) (int, error) { //nolint:unparam
consumer := formatter.NewLogConsumer(ctx, os.Stdout)
for {
event := <-p.queue
switch event.Type {
case compose.ContainerEventExit:
consumer.Status(event.Service, event.Source, fmt.Sprintf("exited with code %d", event.ExitCode))
if cascadeStop {
fmt.Println("Aborting on container exit...")
err := stopFn()
logrus.Error(event.ExitCode)
return event.ExitCode, err
}
case compose.ContainerEventLog:
consumer.Log(event.Service, event.Source, event.Line)
}
}
}

View File

@ -20,43 +20,13 @@ import (
"context" "context"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/docker/compose-cli/utils"
) )
func (b *ecsAPIService) Logs(ctx context.Context, projectName string, consumer compose.LogConsumer, options compose.LogOptions) error { func (b *ecsAPIService) Logs(ctx context.Context, projectName string, consumer compose.LogConsumer, options compose.LogOptions) error {
if len(options.Services) > 0 { if len(options.Services) > 0 {
consumer = filteredLogConsumer(consumer, options.Services) consumer = utils.FilteredLogConsumer(consumer, options.Services)
} }
err := b.aws.GetLogs(ctx, projectName, consumer.Log, options.Follow) err := b.aws.GetLogs(ctx, projectName, consumer.Log, options.Follow)
return err return err
} }
func filteredLogConsumer(consumer compose.LogConsumer, services []string) compose.LogConsumer {
if len(services) == 0 {
return consumer
}
allowed := map[string]bool{}
for _, s := range services {
allowed[s] = true
}
return &allowListLogConsumer{
allowList: allowed,
delegate: consumer,
}
}
type allowListLogConsumer struct {
allowList map[string]bool
delegate compose.LogConsumer
}
func (a *allowListLogConsumer) Log(service, container, message string) {
if a.allowList[service] {
a.delegate.Log(service, container, message)
}
}
func (a *allowListLogConsumer) Status(service, container, message string) {
if a.allowList[service] {
a.delegate.Status(service, container, message)
}
}

View File

@ -24,14 +24,13 @@ import (
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
convert "github.com/docker/compose-cli/local/moby" convert "github.com/docker/compose-cli/local/moby"
"github.com/docker/compose-cli/utils"
"github.com/compose-spec/compose-go/types" "github.com/compose-spec/compose-go/types"
moby "github.com/docker/docker/api/types" moby "github.com/docker/docker/api/types"
"github.com/docker/docker/pkg/stdcopy" "github.com/docker/docker/pkg/stdcopy"
) )
func (s *composeService) attach(ctx context.Context, project *types.Project, consumer compose.LogConsumer) (Containers, error) { func (s *composeService) attach(ctx context.Context, project *types.Project, consumer chan compose.ContainerEvent) (Containers, error) {
containers, err := s.getContainers(ctx, project) containers, err := s.getContainers(ctx, project)
if err != nil { if err != nil {
return nil, err return nil, err
@ -52,7 +51,7 @@ func (s *composeService) attach(ctx context.Context, project *types.Project, con
return containers, nil return containers, nil
} }
func (s *composeService) attachContainer(ctx context.Context, container moby.Container, consumer compose.LogConsumer, project *types.Project) error { func (s *composeService) attachContainer(ctx context.Context, container moby.Container, consumer chan compose.ContainerEvent, project *types.Project) error {
serviceName := container.Labels[serviceLabel] serviceName := container.Labels[serviceLabel]
w := getWriter(serviceName, getContainerNameWithoutProject(container), consumer) w := getWriter(serviceName, getContainerNameWithoutProject(container), consumer)

View File

@ -17,6 +17,7 @@
package compose package compose
import ( import (
"bytes"
"context" "context"
"io" "io"
@ -52,6 +53,7 @@ func (s *composeService) Logs(ctx context.Context, projectName string, consumer
} }
eg, ctx := errgroup.WithContext(ctx) eg, ctx := errgroup.WithContext(ctx)
for _, c := range list { for _, c := range list {
c := c
service := c.Labels[serviceLabel] service := c.Labels[serviceLabel]
if ignore(service) { if ignore(service) {
continue continue
@ -73,7 +75,7 @@ func (s *composeService) Logs(ctx context.Context, projectName string, consumer
if err != nil { if err != nil {
return err return err
} }
w := utils.GetWriter(service, container.Name[1:], consumer) w := utils.GetWriter(service, getContainerNameWithoutProject(c), consumer)
if container.Config.Tty { if container.Config.Tty {
_, err = io.Copy(w, r) _, err = io.Copy(w, r)
} else { } else {
@ -84,3 +86,33 @@ func (s *composeService) Logs(ctx context.Context, projectName string, consumer
} }
return eg.Wait() return eg.Wait()
} }
type splitBuffer struct {
service string
container string
consumer chan compose.ContainerEvent
}
// getWriter creates a io.Writer that will actually split by line and format by LogConsumer
func getWriter(service, container string, events chan compose.ContainerEvent) io.Writer {
return splitBuffer{
service: service,
container: container,
consumer: events,
}
}
func (s splitBuffer) Write(b []byte) (n int, err error) {
split := bytes.Split(b, []byte{'\n'})
for _, line := range split {
if len(line) != 0 {
s.consumer <- compose.ContainerEvent{
Type: compose.ContainerEventLog,
Service: s.service,
Source: s.container,
Line: string(line),
}
}
}
return len(b), nil
}

View File

@ -18,13 +18,12 @@ package compose
import ( import (
"context" "context"
"fmt"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/compose-spec/compose-go/types" "github.com/compose-spec/compose-go/types"
"github.com/docker/docker/api/types/container" "github.com/docker/docker/api/types/container"
"golang.org/x/sync/errgroup" "github.com/sirupsen/logrus"
) )
func (s *composeService) Start(ctx context.Context, project *types.Project, options compose.StartOptions) error { func (s *composeService) Start(ctx context.Context, project *types.Project, options compose.StartOptions) error {
@ -35,12 +34,6 @@ func (s *composeService) Start(ctx context.Context, project *types.Project, opti
return err return err
} }
containers = c containers = c
} else {
c, err := s.getContainers(ctx, project)
if err != nil {
return err
}
containers = c
} }
err := InDependencyOrder(ctx, project, func(c context.Context, service types.ServiceConfig) error { err := InDependencyOrder(ctx, project, func(c context.Context, service types.ServiceConfig) error {
@ -54,26 +47,21 @@ func (s *composeService) Start(ctx context.Context, project *types.Project, opti
return nil return nil
} }
eg, ctx := errgroup.WithContext(ctx)
for _, c := range containers { for _, c := range containers {
c := c c := c
eg.Go(func() error { go func() {
statusC, errC := s.apiClient.ContainerWait(ctx, c.ID, container.WaitConditionNotRunning) statusC, errC := s.apiClient.ContainerWait(context.Background(), c.ID, container.WaitConditionNotRunning)
select { select {
case status := <-statusC: case status := <-statusC:
service := c.Labels[serviceLabel] options.Attach <- compose.ContainerEvent{
options.Attach.Status(service, getCanonicalContainerName(c), fmt.Sprintf("exited with code %d", status.StatusCode)) Type: compose.ContainerEventExit,
if options.Listener != nil { Source: getCanonicalContainerName(c),
options.Listener <- compose.ContainerExited{ ExitCode: int(status.StatusCode),
Service: service,
Status: int(status.StatusCode),
}
} }
return nil
case err := <-errC: case err := <-errC:
return err logrus.Warnf("Unexpected API error for %s : %s\n", getCanonicalContainerName(c), err.Error())
} }
}) }()
} }
return eg.Wait() return nil
} }

View File

@ -31,6 +31,8 @@ func TestCascadeStop(t *testing.T) {
res := c.RunDockerCmd("compose", "-f", "./fixtures/cascade-stop-test/compose.yaml", "--project-name", projectName, "up", "--abort-on-container-exit") res := c.RunDockerCmd("compose", "-f", "./fixtures/cascade-stop-test/compose.yaml", "--project-name", projectName, "up", "--abort-on-container-exit")
res.Assert(t, icmd.Expected{Out: `PING localhost (127.0.0.1)`}) res.Assert(t, icmd.Expected{Out: `PING localhost (127.0.0.1)`})
res.Assert(t, icmd.Expected{Out: `ping_1 exited with code 0`}) res.Assert(t, icmd.Expected{Out: `/does_not_exist: No such file or directory`})
res.Assert(t, icmd.Expected{Out: `should_fail_1 exited with code 1`})
res.Assert(t, icmd.Expected{Out: `Aborting on container exit...`}) res.Assert(t, icmd.Expected{Out: `Aborting on container exit...`})
// FIXME res.Assert(t, icmd.Expected{ExitCode: 1})
} }

View File

@ -1,4 +1,7 @@
services: services:
should_fail:
image: busybox:1.27.2
command: ls /does_not_exist
ping: ping:
image: busybox:1.27.2 image: busybox:1.27.2
command: ping localhost -c 1 command: ping localhost

View File

@ -58,6 +58,12 @@ func (a *allowListLogConsumer) Log(service, container, message string) {
} }
} }
func (a *allowListLogConsumer) Status(service, container, message string) {
if a.allowList[service] {
a.delegate.Status(service, container, message)
}
}
type splitBuffer struct { type splitBuffer struct {
service string service string
container string container string