| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 |
- /*
- 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 compose
- import (
- "context"
- "os"
- "testing"
- "time"
- "github.com/compose-spec/compose-go/types"
- "github.com/docker/compose/v2/internal/sync"
- "github.com/docker/compose/v2/pkg/api"
- "github.com/docker/compose/v2/pkg/mocks"
- "github.com/docker/compose/v2/pkg/watch"
- moby "github.com/docker/docker/api/types"
- "github.com/golang/mock/gomock"
- "github.com/jonboulle/clockwork"
- "github.com/stretchr/testify/require"
- "gotest.tools/v3/assert"
- )
- func TestDebounceBatching(t *testing.T) {
- ch := make(chan fileEvent)
- clock := clockwork.NewFakeClock()
- ctx, stop := context.WithCancel(context.Background())
- t.Cleanup(stop)
- eventBatchCh := batchDebounceEvents(ctx, clock, quietPeriod, ch)
- for i := 0; i < 100; i++ {
- var action WatchAction = "a"
- if i%2 == 0 {
- action = "b"
- }
- ch <- fileEvent{Action: action}
- }
- // we sent 100 events + the debouncer
- clock.BlockUntil(101)
- clock.Advance(quietPeriod)
- select {
- case batch := <-eventBatchCh:
- require.ElementsMatch(t, batch, []fileEvent{
- {Action: "a"},
- {Action: "b"},
- })
- case <-time.After(50 * time.Millisecond):
- t.Fatal("timed out waiting for events")
- }
- clock.BlockUntil(1)
- clock.Advance(quietPeriod)
- // there should only be a single batch
- select {
- case batch := <-eventBatchCh:
- t.Fatalf("unexpected events: %v", batch)
- case <-time.After(50 * time.Millisecond):
- // channel is empty
- }
- }
- type testWatcher struct {
- events chan watch.FileEvent
- errors chan error
- }
- func (t testWatcher) Start() error {
- return nil
- }
- func (t testWatcher) Close() error {
- return nil
- }
- func (t testWatcher) Events() chan watch.FileEvent {
- return t.events
- }
- func (t testWatcher) Errors() chan error {
- return t.errors
- }
- func TestWatch_Sync(t *testing.T) {
- mockCtrl := gomock.NewController(t)
- cli := mocks.NewMockCli(mockCtrl)
- cli.EXPECT().Err().Return(os.Stderr).AnyTimes()
- apiClient := mocks.NewMockAPIClient(mockCtrl)
- apiClient.EXPECT().ContainerList(gomock.Any(), gomock.Any()).Return([]moby.Container{
- testContainer("test", "123", false),
- }, nil).AnyTimes()
- cli.EXPECT().Client().Return(apiClient).AnyTimes()
- ctx, cancelFunc := context.WithCancel(context.Background())
- t.Cleanup(cancelFunc)
- proj := types.Project{
- Services: []types.ServiceConfig{
- {
- Name: "test",
- },
- },
- }
- watcher := testWatcher{
- events: make(chan watch.FileEvent),
- errors: make(chan error),
- }
- syncer := newFakeSyncer()
- clock := clockwork.NewFakeClock()
- go func() {
- service := composeService{
- dockerCli: cli,
- clock: clock,
- }
- err := service.watch(ctx, &proj, "test", api.WatchOptions{}, watcher, syncer, []Trigger{
- {
- Path: "/sync",
- Action: "sync",
- Target: "/work",
- Ignore: []string{"ignore"},
- },
- {
- Path: "/rebuild",
- Action: "rebuild",
- },
- })
- assert.NilError(t, err)
- }()
- watcher.Events() <- watch.NewFileEvent("/sync/changed")
- watcher.Events() <- watch.NewFileEvent("/sync/changed/sub")
- clock.BlockUntil(3)
- clock.Advance(quietPeriod)
- select {
- case actual := <-syncer.synced:
- require.ElementsMatch(t, []sync.PathMapping{
- {HostPath: "/sync/changed", ContainerPath: "/work/changed"},
- {HostPath: "/sync/changed/sub", ContainerPath: "/work/changed/sub"},
- }, actual)
- case <-time.After(100 * time.Millisecond):
- t.Error("timeout")
- }
- watcher.Events() <- watch.NewFileEvent("/sync/ignore")
- watcher.Events() <- watch.NewFileEvent("/sync/ignore/sub")
- watcher.Events() <- watch.NewFileEvent("/sync/changed")
- clock.BlockUntil(4)
- clock.Advance(quietPeriod)
- select {
- case actual := <-syncer.synced:
- require.ElementsMatch(t, []sync.PathMapping{
- {HostPath: "/sync/changed", ContainerPath: "/work/changed"},
- }, actual)
- case <-time.After(100 * time.Millisecond):
- t.Error("timed out waiting for events")
- }
- watcher.Events() <- watch.NewFileEvent("/rebuild")
- watcher.Events() <- watch.NewFileEvent("/sync/changed")
- clock.BlockUntil(4)
- clock.Advance(quietPeriod)
- select {
- case batch := <-syncer.synced:
- t.Fatalf("received unexpected events: %v", batch)
- case <-time.After(100 * time.Millisecond):
- // expected
- }
- // TODO: there's not a great way to assert that the rebuild attempt happened
- }
- type fakeSyncer struct {
- synced chan []sync.PathMapping
- }
- func newFakeSyncer() *fakeSyncer {
- return &fakeSyncer{
- synced: make(chan []sync.PathMapping),
- }
- }
- func (f *fakeSyncer) Sync(_ context.Context, _ types.ServiceConfig, paths []sync.PathMapping) error {
- f.synced <- paths
- return nil
- }
|