| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158 |
- /*
- 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"
- "github.com/compose-spec/compose-go/types"
- moby "github.com/docker/docker/api/types"
- "github.com/pkg/errors"
- "golang.org/x/sync/errgroup"
- "github.com/docker/compose/v2/pkg/api"
- "github.com/docker/compose/v2/pkg/progress"
- )
- func (s *composeService) Start(ctx context.Context, project *types.Project, options api.StartOptions) error {
- return progress.Run(ctx, func(ctx context.Context) error {
- return s.start(ctx, project, options, nil)
- })
- }
- func (s *composeService) start(ctx context.Context, project *types.Project, options api.StartOptions, listener api.ContainerEventListener) error {
- if len(options.AttachTo) == 0 {
- options.AttachTo = project.ServiceNames()
- }
- eg, ctx := errgroup.WithContext(ctx)
- if listener != nil {
- attached, err := s.attach(ctx, project, listener, options.AttachTo)
- if err != nil {
- return err
- }
- eg.Go(func() error {
- return s.watchContainers(context.Background(), project.Name, options.AttachTo, listener, attached, func(container moby.Container) error {
- return s.attachContainer(ctx, container, listener, project)
- })
- })
- }
- err := InDependencyOrder(ctx, project, func(c context.Context, name string) error {
- service, err := project.GetService(name)
- if err != nil {
- return err
- }
- return s.startService(ctx, project, service)
- })
- if err != nil {
- return err
- }
- if options.Wait {
- depends := types.DependsOnConfig{}
- for _, s := range project.Services {
- depends[s.Name] = types.ServiceDependency{
- Condition: ServiceConditionRuningOrHealthy,
- }
- }
- err = s.waitDependencies(ctx, project, depends)
- if err != nil {
- return err
- }
- }
- return eg.Wait()
- }
- type containerWatchFn func(container moby.Container) error
- // watchContainers uses engine events to capture container start/die and notify ContainerEventListener
- func (s *composeService) watchContainers(ctx context.Context, projectName string, services []string, listener api.ContainerEventListener, containers Containers, onStart containerWatchFn) error {
- watched := map[string]int{}
- for _, c := range containers {
- watched[c.ID] = 0
- }
- ctx, stop := context.WithCancel(ctx)
- err := s.Events(ctx, projectName, api.EventsOptions{
- Services: services,
- Consumer: func(event api.Event) error {
- inspected, err := s.apiClient.ContainerInspect(ctx, event.Container)
- if err != nil {
- return err
- }
- container := moby.Container{
- ID: inspected.ID,
- Names: []string{inspected.Name},
- Labels: inspected.Config.Labels,
- }
- name := getContainerNameWithoutProject(container)
- if event.Status == "die" {
- restarted := watched[container.ID]
- watched[container.ID] = restarted + 1
- // Container terminated.
- willRestart := inspected.HostConfig.RestartPolicy.MaximumRetryCount > restarted
- listener(api.ContainerEvent{
- Type: api.ContainerEventExit,
- Container: name,
- Service: container.Labels[api.ServiceLabel],
- ExitCode: inspected.State.ExitCode,
- Restarting: willRestart,
- })
- if !willRestart {
- // we're done with this one
- delete(watched, container.ID)
- }
- if len(watched) == 0 {
- // all project containers stopped, we're done
- stop()
- }
- return nil
- }
- if event.Status == "start" {
- count, ok := watched[container.ID]
- mustAttach := ok && count > 0 // Container restarted, need to re-attach
- if !ok {
- // A new container has just been added to service by scale
- watched[container.ID] = 0
- mustAttach = true
- }
- if mustAttach {
- // Container restarted, need to re-attach
- err := onStart(container)
- if err != nil {
- return err
- }
- }
- }
- return nil
- },
- })
- if errors.Is(ctx.Err(), context.Canceled) {
- return nil
- }
- return err
- }
|