Skip to content

Commit

Permalink
Merge pull request #10622 from ndeloof/logs_follow
Browse files Browse the repository at this point in the history
fix `compose -p x logs -f` detect new services started after command
  • Loading branch information
glours authored Jun 2, 2023
2 parents d2aa15c + 0d6b99e commit 7c3fe35
Show file tree
Hide file tree
Showing 5 changed files with 73 additions and 16 deletions.
5 changes: 2 additions & 3 deletions pkg/compose/events.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,6 @@ import (
"github.com/docker/docker/api/types/filters"

"github.com/docker/compose/v2/pkg/api"

"github.com/docker/compose/v2/pkg/utils"
)

Expand Down Expand Up @@ -67,8 +66,8 @@ func (s *composeService) Events(ctx context.Context, projectName string, options
err := options.Consumer(api.Event{
Timestamp: timestamp,
Service: service,
Container: event.ID,
Status: event.Status,
Container: event.Actor.ID,
Status: event.Action,
Attributes: attributes,
})
if err != nil {
Expand Down
15 changes: 4 additions & 11 deletions pkg/compose/logs.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,19 +45,12 @@ func (s *composeService) Logs(
return err
}

project := options.Project
if project == nil {
project, err = s.getProjectWithResources(ctx, containers, projectName)
if err != nil {
return err
}
}

if len(options.Services) == 0 {
options.Services = project.ServiceNames()
if options.Project != nil && len(options.Services) == 0 {
// we run with an explicit compose.yaml, so only consider services defined in this file
options.Services = options.Project.ServiceNames()
containers = containers.filter(isService(options.Services...))
}

containers = containers.filter(isService(options.Services...))
eg, ctx := errgroup.WithContext(ctx)
for _, c := range containers {
c := c
Expand Down
25 changes: 24 additions & 1 deletion pkg/compose/start.go
Original file line number Diff line number Diff line change
Expand Up @@ -155,13 +155,31 @@ func (s *composeService) watchContainers(ctx context.Context, //nolint:gocyclo
required = services
}

// predicate to tell if a container we receive event for should be considered or ignored
ofInterest := func(c moby.Container) bool {
if len(services) > 0 {
// we only watch some services
return utils.Contains(services, c.Labels[api.ServiceLabel])
}
return true
}

// predicate to tell if a container we receive event for should be watched until termination
isRequired := func(c moby.Container) bool {
if len(services) > 0 && len(required) > 0 {
// we only watch some services
return utils.Contains(required, c.Labels[api.ServiceLabel])
}
return true
}

var (
expected []string
watched = map[string]int{}
replaced []string
)
for _, c := range containers {
if utils.Contains(required, c.Labels[api.ServiceLabel]) {
if isRequired(c) {
expected = append(expected, c.ID)
}
watched[c.ID] = 0
Expand Down Expand Up @@ -265,6 +283,11 @@ func (s *composeService) watchContainers(ctx context.Context, //nolint:gocyclo
if utils.Contains(expected, id) {
expected = append(expected, container.ID)
}
} else if ofInterest(container) {
watched[container.ID] = 1
if isRequired(container) {
expected = append(expected, container.ID)
}
}
}
if len(expected) == 0 {
Expand Down
2 changes: 1 addition & 1 deletion pkg/e2e/fixtures/logs-test/compose.yaml
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
services:
ping:
image: alpine
command: ping localhost -c 1
command: ping localhost -c ${REPEAT:-1}
hello:
image: alpine
command: echo hello
42 changes: 42 additions & 0 deletions pkg/e2e/logs_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,10 +17,13 @@
package e2e

import (
"fmt"
"strings"
"testing"
"time"

"gotest.tools/v3/assert"
"gotest.tools/v3/poll"

"gotest.tools/v3/icmd"
)
Expand Down Expand Up @@ -56,3 +59,42 @@ func TestLocalComposeLogs(t *testing.T) {
_ = c.RunDockerComposeCmd(t, "--project-name", projectName, "down")
})
}

func TestLocalComposeLogsFollow(t *testing.T) {
c := NewCLI(t, WithEnv("REPEAT=20"))
const projectName = "compose-e2e-logs"
t.Cleanup(func() {
c.RunDockerComposeCmd(t, "--project-name", projectName, "down")
})

c.RunDockerComposeCmd(t, "-f", "./fixtures/logs-test/compose.yaml", "--project-name", projectName, "up", "-d", "ping")

cmd := c.NewDockerComposeCmd(t, "--project-name", projectName, "logs", "-f")
res := icmd.StartCmd(cmd)
t.Cleanup(func() {
_ = res.Cmd.Process.Kill()
})

expected := fmt.Sprintf("%s-ping-1 ", projectName)
poll.WaitOn(t, expectOutput(res, expected), poll.WithDelay(100*time.Millisecond), poll.WithTimeout(1*time.Second))

c.RunDockerComposeCmd(t, "-f", "./fixtures/logs-test/compose.yaml", "--project-name", projectName, "up", "-d")

expected = fmt.Sprintf("%s-hello-1 ", projectName)
poll.WaitOn(t, expectOutput(res, expected), poll.WithDelay(100*time.Millisecond), poll.WithTimeout(1*time.Second))

c.RunDockerComposeCmd(t, "-f", "./fixtures/logs-test/compose.yaml", "--project-name", projectName, "up", "-d", "--scale", "ping=2", "ping")

expected = fmt.Sprintf("%s-ping-2 ", projectName)
poll.WaitOn(t, expectOutput(res, expected), poll.WithDelay(100*time.Millisecond), poll.WithTimeout(20*time.Second))
}

func expectOutput(res *icmd.Result, expected string) func(t poll.LogT) poll.Result {
return func(t poll.LogT) poll.Result {
if strings.Contains(res.Stdout(), expected) {
return poll.Success()
}
return poll.Continue("condition not met")

}
}

0 comments on commit 7c3fe35

Please sign in to comment.