mirror of
				https://github.com/docker/compose.git
				synced 2025-10-25 01:03:51 +02:00 
			
		
		
		
	Add docker compose wait
				
					
				
			Signed-off-by: Ulysses Souza <ulyssessouza@gmail.com>
This commit is contained in:
		
							parent
							
								
									c496c23071
								
							
						
					
					
						commit
						edd76bfd70
					
				| @ -431,6 +431,7 @@ func RootCommand(streams command.Cli, backend api.Service) *cobra.Command { //no | |||||||
| 		pullCommand(&opts, backend), | 		pullCommand(&opts, backend), | ||||||
| 		createCommand(&opts, backend), | 		createCommand(&opts, backend), | ||||||
| 		copyCommand(&opts, backend), | 		copyCommand(&opts, backend), | ||||||
|  | 		waitCommand(&opts, backend), | ||||||
| 		alphaCommand(&opts, backend), | 		alphaCommand(&opts, backend), | ||||||
| 	) | 	) | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										72
									
								
								cmd/compose/wait.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								cmd/compose/wait.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,72 @@ | |||||||
|  | /* | ||||||
|  |    Copyright 2023 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" | ||||||
|  | 
 | ||||||
|  | 	"github.com/docker/cli/cli" | ||||||
|  | 	"github.com/docker/compose/v2/pkg/api" | ||||||
|  | 	"github.com/spf13/cobra" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type waitOptions struct { | ||||||
|  | 	*ProjectOptions | ||||||
|  | 
 | ||||||
|  | 	services []string | ||||||
|  | 
 | ||||||
|  | 	downProject bool | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func waitCommand(p *ProjectOptions, backend api.Service) *cobra.Command { | ||||||
|  | 	opts := waitOptions{ | ||||||
|  | 		ProjectOptions: p, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	var statusCode int64 | ||||||
|  | 	var err error | ||||||
|  | 	cmd := &cobra.Command{ | ||||||
|  | 		Use:   "wait SERVICE [SERVICE...] [OPTIONS]", | ||||||
|  | 		Short: "Block until the first service container stops", | ||||||
|  | 		Args:  cli.RequiresMinArgs(1), | ||||||
|  | 		RunE: Adapt(func(ctx context.Context, services []string) error { | ||||||
|  | 			opts.services = services | ||||||
|  | 			statusCode, err = runWait(ctx, backend, &opts) | ||||||
|  | 			return err | ||||||
|  | 		}), | ||||||
|  | 		PostRun: func(cmd *cobra.Command, args []string) { | ||||||
|  | 			os.Exit(int(statusCode)) | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	cmd.Flags().BoolVar(&opts.downProject, "down-project", false, "Drops project when the first container stops") | ||||||
|  | 
 | ||||||
|  | 	return cmd | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func runWait(ctx context.Context, backend api.Service, opts *waitOptions) (int64, error) { | ||||||
|  | 	_, name, err := opts.projectOrName() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return 0, err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return backend.Wait(ctx, name, api.WaitOptions{ | ||||||
|  | 		Services:                   opts.services, | ||||||
|  | 		DownProjectOnContainerExit: opts.downProject, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
| @ -33,6 +33,7 @@ Define and run multi-container applications with Docker. | |||||||
| | [`unpause`](compose_unpause.md) | Unpause services                                                        | | | [`unpause`](compose_unpause.md) | Unpause services                                                        | | ||||||
| | [`up`](compose_up.md)           | Create and start containers                                             | | | [`up`](compose_up.md)           | Create and start containers                                             | | ||||||
| | [`version`](compose_version.md) | Show the Docker Compose version information                             | | | [`version`](compose_version.md) | Show the Docker Compose version information                             | | ||||||
|  | | [`wait`](compose_wait.md)       | Block until the first service container stops                           | | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| ### Options | ### Options | ||||||
|  | |||||||
							
								
								
									
										15
									
								
								docs/reference/compose_wait.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								docs/reference/compose_wait.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,15 @@ | |||||||
|  | # docker compose wait | ||||||
|  | 
 | ||||||
|  | <!---MARKER_GEN_START--> | ||||||
|  | Block until the first service container stops | ||||||
|  | 
 | ||||||
|  | ### Options | ||||||
|  | 
 | ||||||
|  | | Name             | Type | Default | Description                                  | | ||||||
|  | |:-----------------|:-----|:--------|:---------------------------------------------| | ||||||
|  | | `--down-project` |      |         | Drops project when the first container stops | | ||||||
|  | | `--dry-run`      |      |         | Execute command in dry run mode              | | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | <!---MARKER_GEN_END--> | ||||||
|  | 
 | ||||||
| @ -171,6 +171,7 @@ cname: | |||||||
|     - docker compose unpause |     - docker compose unpause | ||||||
|     - docker compose up |     - docker compose up | ||||||
|     - docker compose version |     - docker compose version | ||||||
|  |     - docker compose wait | ||||||
| clink: | clink: | ||||||
|     - docker_compose_build.yaml |     - docker_compose_build.yaml | ||||||
|     - docker_compose_config.yaml |     - docker_compose_config.yaml | ||||||
| @ -197,6 +198,7 @@ clink: | |||||||
|     - docker_compose_unpause.yaml |     - docker_compose_unpause.yaml | ||||||
|     - docker_compose_up.yaml |     - docker_compose_up.yaml | ||||||
|     - docker_compose_version.yaml |     - docker_compose_version.yaml | ||||||
|  |     - docker_compose_wait.yaml | ||||||
| options: | options: | ||||||
|     - option: ansi |     - option: ansi | ||||||
|       value_type: string |       value_type: string | ||||||
|  | |||||||
							
								
								
									
										34
									
								
								docs/reference/docker_compose_wait.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								docs/reference/docker_compose_wait.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | command: docker compose wait | ||||||
|  | short: Block until the first service container stops | ||||||
|  | long: Block until the first service container stops | ||||||
|  | usage: docker compose wait SERVICE [SERVICE...] [OPTIONS] | ||||||
|  | pname: docker compose | ||||||
|  | plink: docker_compose.yaml | ||||||
|  | options: | ||||||
|  |     - option: down-project | ||||||
|  |       value_type: bool | ||||||
|  |       default_value: "false" | ||||||
|  |       description: Drops project when the first container stops | ||||||
|  |       deprecated: false | ||||||
|  |       hidden: false | ||||||
|  |       experimental: false | ||||||
|  |       experimentalcli: false | ||||||
|  |       kubernetes: false | ||||||
|  |       swarm: false | ||||||
|  | inherited_options: | ||||||
|  |     - option: dry-run | ||||||
|  |       value_type: bool | ||||||
|  |       default_value: "false" | ||||||
|  |       description: Execute command in dry run mode | ||||||
|  |       deprecated: false | ||||||
|  |       hidden: false | ||||||
|  |       experimental: false | ||||||
|  |       experimentalcli: false | ||||||
|  |       kubernetes: false | ||||||
|  |       swarm: false | ||||||
|  | deprecated: false | ||||||
|  | experimental: false | ||||||
|  | experimentalcli: false | ||||||
|  | kubernetes: false | ||||||
|  | swarm: false | ||||||
|  | 
 | ||||||
| @ -84,6 +84,15 @@ type Service interface { | |||||||
| 	Watch(ctx context.Context, project *types.Project, services []string, options WatchOptions) error | 	Watch(ctx context.Context, project *types.Project, services []string, options WatchOptions) error | ||||||
| 	// Viz generates a graphviz graph of the project services | 	// Viz generates a graphviz graph of the project services | ||||||
| 	Viz(ctx context.Context, project *types.Project, options VizOptions) (string, error) | 	Viz(ctx context.Context, project *types.Project, options VizOptions) (string, error) | ||||||
|  | 	// Wait blocks until at least one of the services' container exits | ||||||
|  | 	Wait(ctx context.Context, projectName string, options WaitOptions) (int64, error) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type WaitOptions struct { | ||||||
|  | 	// Services passed in the command line to be waited | ||||||
|  | 	Services []string | ||||||
|  | 	// Executes a down when a container exits | ||||||
|  | 	DownProjectOnContainerExit bool | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| type VizOptions struct { | type VizOptions struct { | ||||||
|  | |||||||
| @ -54,6 +54,7 @@ type ServiceProxy struct { | |||||||
| 	MaxConcurrencyFn     func(parallel int) | 	MaxConcurrencyFn     func(parallel int) | ||||||
| 	DryRunModeFn         func(ctx context.Context, dryRun bool) (context.Context, error) | 	DryRunModeFn         func(ctx context.Context, dryRun bool) (context.Context, error) | ||||||
| 	VizFn                func(ctx context.Context, project *types.Project, options VizOptions) (string, error) | 	VizFn                func(ctx context.Context, project *types.Project, options VizOptions) (string, error) | ||||||
|  | 	WaitFn               func(ctx context.Context, projectName string, options WaitOptions) (int64, error) | ||||||
| 	interceptors         []Interceptor | 	interceptors         []Interceptor | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -95,6 +96,7 @@ func (s *ServiceProxy) WithService(service Service) *ServiceProxy { | |||||||
| 	s.MaxConcurrencyFn = service.MaxConcurrency | 	s.MaxConcurrencyFn = service.MaxConcurrency | ||||||
| 	s.DryRunModeFn = service.DryRunMode | 	s.DryRunModeFn = service.DryRunMode | ||||||
| 	s.VizFn = service.Viz | 	s.VizFn = service.Viz | ||||||
|  | 	s.WaitFn = service.Wait | ||||||
| 	return s | 	return s | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -325,7 +327,7 @@ func (s *ServiceProxy) Watch(ctx context.Context, project *types.Project, servic | |||||||
| 	return s.WatchFn(ctx, project, services, options) | 	return s.WatchFn(ctx, project, services, options) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Viz implements Viz interface | // Viz implements Service interface | ||||||
| func (s *ServiceProxy) Viz(ctx context.Context, project *types.Project, options VizOptions) (string, error) { | func (s *ServiceProxy) Viz(ctx context.Context, project *types.Project, options VizOptions) (string, error) { | ||||||
| 	if s.VizFn == nil { | 	if s.VizFn == nil { | ||||||
| 		return "", ErrNotImplemented | 		return "", ErrNotImplemented | ||||||
| @ -333,6 +335,14 @@ func (s *ServiceProxy) Viz(ctx context.Context, project *types.Project, options | |||||||
| 	return s.VizFn(ctx, project, options) | 	return s.VizFn(ctx, project, options) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Wait implements Service interface | ||||||
|  | func (s *ServiceProxy) Wait(ctx context.Context, projectName string, options WaitOptions) (int64, error) { | ||||||
|  | 	if s.WaitFn == nil { | ||||||
|  | 		return 0, ErrNotImplemented | ||||||
|  | 	} | ||||||
|  | 	return s.WaitFn(ctx, projectName, options) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func (s *ServiceProxy) MaxConcurrency(i int) { | func (s *ServiceProxy) MaxConcurrency(i int) { | ||||||
| 	s.MaxConcurrencyFn(i) | 	s.MaxConcurrencyFn(i) | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										67
									
								
								pkg/compose/wait.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								pkg/compose/wait.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,67 @@ | |||||||
|  | /* | ||||||
|  |    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" | ||||||
|  | 	"fmt" | ||||||
|  | 
 | ||||||
|  | 	"github.com/docker/compose/v2/pkg/api" | ||||||
|  | 	"golang.org/x/sync/errgroup" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func (s *composeService) Wait(ctx context.Context, projectName string, options api.WaitOptions) (int64, error) { | ||||||
|  | 	containers, err := s.getContainers(ctx, projectName, oneOffInclude, false, options.Services...) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return 0, err | ||||||
|  | 	} | ||||||
|  | 	if len(containers) == 0 { | ||||||
|  | 		return 0, fmt.Errorf("no containers for project %q", projectName) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	eg, waitCtx := errgroup.WithContext(ctx) | ||||||
|  | 	var statusCode int64 | ||||||
|  | 	for _, c := range containers { | ||||||
|  | 		c := c | ||||||
|  | 		eg.Go(func() error { | ||||||
|  | 			var err error | ||||||
|  | 			resultC, errC := s.dockerCli.Client().ContainerWait(waitCtx, c.ID, "") | ||||||
|  | 
 | ||||||
|  | 			select { | ||||||
|  | 			case result := <-resultC: | ||||||
|  | 				fmt.Fprintf(s.dockerCli.Out(), "container %q exited with status code %d\n", c.ID, result.StatusCode) | ||||||
|  | 				statusCode = result.StatusCode | ||||||
|  | 			case err = <-errC: | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			return err | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	err = eg.Wait() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return 42, err // Ignore abort flag in case of error in wait | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if options.DownProjectOnContainerExit { | ||||||
|  | 		return statusCode, s.Down(ctx, projectName, api.DownOptions{ | ||||||
|  | 			RemoveOrphans: true, | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return statusCode, err | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								pkg/e2e/fixtures/wait/compose.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								pkg/e2e/fixtures/wait/compose.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | services: | ||||||
|  |   faster: | ||||||
|  |     image: alpine | ||||||
|  |     command: sleep 2 | ||||||
|  |   slower: | ||||||
|  |     image: alpine | ||||||
|  |     command: sleep 5 | ||||||
|  |   infinity: | ||||||
|  |     image: alpine | ||||||
|  |     command: sleep infinity | ||||||
|  | 
 | ||||||
							
								
								
									
										72
									
								
								pkg/e2e/wait_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								pkg/e2e/wait_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,72 @@ | |||||||
|  | /* | ||||||
|  |    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 e2e | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"strings" | ||||||
|  | 	"testing" | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	"gotest.tools/v3/assert" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestWaitOnFaster(t *testing.T) { | ||||||
|  | 	const projectName = "e2e-wait-faster" | ||||||
|  | 	c := NewParallelCLI(t) | ||||||
|  | 
 | ||||||
|  | 	c.RunDockerComposeCmd(t, "-f", "./fixtures/wait/compose.yaml", "--project-name", projectName, "up", "-d") | ||||||
|  | 	c.RunDockerComposeCmd(t, "--project-name", projectName, "wait", "faster") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestWaitOnSlower(t *testing.T) { | ||||||
|  | 	const projectName = "e2e-wait-slower" | ||||||
|  | 	c := NewParallelCLI(t) | ||||||
|  | 
 | ||||||
|  | 	c.RunDockerComposeCmd(t, "-f", "./fixtures/wait/compose.yaml", "--project-name", projectName, "up", "-d") | ||||||
|  | 	c.RunDockerComposeCmd(t, "--project-name", projectName, "wait", "slower") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestWaitOnInfinity(t *testing.T) { | ||||||
|  | 	const projectName = "e2e-wait-infinity" | ||||||
|  | 	c := NewParallelCLI(t) | ||||||
|  | 
 | ||||||
|  | 	c.RunDockerComposeCmd(t, "-f", "./fixtures/wait/compose.yaml", "--project-name", projectName, "up", "-d") | ||||||
|  | 
 | ||||||
|  | 	finished := make(chan struct{}) | ||||||
|  | 	ticker := time.NewTicker(7 * time.Second) | ||||||
|  | 	go func() { | ||||||
|  | 		c.RunDockerComposeCmd(t, "--project-name", projectName, "wait", "infinity") | ||||||
|  | 		finished <- struct{}{} | ||||||
|  | 	}() | ||||||
|  | 
 | ||||||
|  | 	select { | ||||||
|  | 	case <-finished: | ||||||
|  | 		t.Fatal("wait infinity should not finish") | ||||||
|  | 	case <-ticker.C: | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestWaitAndDrop(t *testing.T) { | ||||||
|  | 	const projectName = "e2e-wait-and-drop" | ||||||
|  | 	c := NewParallelCLI(t) | ||||||
|  | 
 | ||||||
|  | 	c.RunDockerComposeCmd(t, "-f", "./fixtures/wait/compose.yaml", "--project-name", projectName, "up", "-d") | ||||||
|  | 	c.RunDockerComposeCmd(t, "--project-name", projectName, "wait", "--down-project", "faster") | ||||||
|  | 
 | ||||||
|  | 	res := c.RunDockerCmd(t, "ps", "--all") | ||||||
|  | 	assert.Assert(t, !strings.Contains(res.Combined(), projectName), res.Combined()) | ||||||
|  | } | ||||||
| @ -423,6 +423,21 @@ func (mr *MockServiceMockRecorder) Viz(ctx, project, options interface{}) *gomoc | |||||||
| 	return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Viz", reflect.TypeOf((*MockService)(nil).Viz), ctx, project, options) | 	return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Viz", reflect.TypeOf((*MockService)(nil).Viz), ctx, project, options) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Wait mocks base method. | ||||||
|  | func (m *MockService) Wait(ctx context.Context, projectName string, options api.WaitOptions) (int64, error) { | ||||||
|  | 	m.ctrl.T.Helper() | ||||||
|  | 	ret := m.ctrl.Call(m, "Wait", ctx, projectName, options) | ||||||
|  | 	ret0, _ := ret[0].(int64) | ||||||
|  | 	ret1, _ := ret[1].(error) | ||||||
|  | 	return ret0, ret1 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Wait indicates an expected call of Wait. | ||||||
|  | func (mr *MockServiceMockRecorder) Wait(ctx, projectName, options interface{}) *gomock.Call { | ||||||
|  | 	mr.mock.ctrl.T.Helper() | ||||||
|  | 	return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Wait", reflect.TypeOf((*MockService)(nil).Wait), ctx, projectName, options) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // Watch mocks base method. | // Watch mocks base method. | ||||||
| func (m *MockService) Watch(ctx context.Context, project *types.Project, services []string, options api.WatchOptions) error { | func (m *MockService) Watch(ctx context.Context, project *types.Project, services []string, options api.WatchOptions) error { | ||||||
| 	m.ctrl.T.Helper() | 	m.ctrl.T.Helper() | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user