Move progress writter into backend(s)

Signed-off-by: Nicolas De Loof <nicolas.deloof@gmail.com>
This commit is contained in:
Nicolas De Loof 2021-06-03 13:57:56 +02:00
parent db30ef76d0
commit 82f35d1bac
No known key found for this signature in database
GPG Key ID: 9858809D6F8F6E7E
23 changed files with 123 additions and 118 deletions

View File

@ -227,8 +227,8 @@ func (cs *aciComposeService) RunOneOffContainer(ctx context.Context, project *ty
return 0, errdefs.ErrNotImplemented return 0, errdefs.ErrNotImplemented
} }
func (cs *aciComposeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) ([]string, error) { func (cs *aciComposeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) error {
return nil, errdefs.ErrNotImplemented return errdefs.ErrNotImplemented
} }
func (cs *aciComposeService) Exec(ctx context.Context, project *types.Project, opts compose.RunOptions) (int, error) { func (cs *aciComposeService) Exec(ctx context.Context, project *types.Project, opts compose.RunOptions) (int, error) {

View File

@ -88,8 +88,8 @@ func (c *composeService) RunOneOffContainer(ctx context.Context, project *types.
return 0, errdefs.ErrNotImplemented return 0, errdefs.ErrNotImplemented
} }
func (c *composeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) ([]string, error) { func (c *composeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) error {
return nil, errdefs.ErrNotImplemented return errdefs.ErrNotImplemented
} }
func (c *composeService) Exec(ctx context.Context, project *types.Project, opts compose.RunOptions) (int, error) { func (c *composeService) Exec(ctx context.Context, project *types.Project, opts compose.RunOptions) (int, error) {

View File

@ -59,7 +59,7 @@ type Service interface {
// RunOneOffContainer creates a service oneoff container and starts its dependencies // RunOneOffContainer creates a service oneoff container and starts its dependencies
RunOneOffContainer(ctx context.Context, project *types.Project, opts RunOptions) (int, error) RunOneOffContainer(ctx context.Context, project *types.Project, opts RunOptions) (int, error)
// Remove executes the equivalent to a `compose rm` // Remove executes the equivalent to a `compose rm`
Remove(ctx context.Context, project *types.Project, options RemoveOptions) ([]string, error) Remove(ctx context.Context, project *types.Project, options RemoveOptions) error
// Exec executes a command in a running service container // Exec executes a command in a running service container
Exec(ctx context.Context, project *types.Project, opts RunOptions) (int, error) Exec(ctx context.Context, project *types.Project, opts RunOptions) (int, error)
// Copy copies a file/folder between a service container and the local filesystem // Copy copies a file/folder between a service container and the local filesystem
@ -169,8 +169,9 @@ type PushOptions struct {
IgnoreFailures bool IgnoreFailures bool
} }
// PullOptions group options of the Push API // PullOptions group options of the Pull API
type PullOptions struct { type PullOptions struct {
Quiet bool
IgnoreFailures bool IgnoreFailures bool
} }

View File

@ -41,7 +41,7 @@ type ServiceProxy struct {
ConvertFn func(ctx context.Context, project *types.Project, options ConvertOptions) ([]byte, error) ConvertFn func(ctx context.Context, project *types.Project, options ConvertOptions) ([]byte, error)
KillFn func(ctx context.Context, project *types.Project, options KillOptions) error KillFn func(ctx context.Context, project *types.Project, options KillOptions) error
RunOneOffContainerFn func(ctx context.Context, project *types.Project, opts RunOptions) (int, error) RunOneOffContainerFn func(ctx context.Context, project *types.Project, opts RunOptions) (int, error)
RemoveFn func(ctx context.Context, project *types.Project, options RemoveOptions) ([]string, error) RemoveFn func(ctx context.Context, project *types.Project, options RemoveOptions) error
ExecFn func(ctx context.Context, project *types.Project, opts RunOptions) (int, error) ExecFn func(ctx context.Context, project *types.Project, opts RunOptions) (int, error)
CopyFn func(ctx context.Context, project *types.Project, opts CopyOptions) error CopyFn func(ctx context.Context, project *types.Project, opts CopyOptions) error
PauseFn func(ctx context.Context, project string, options PauseOptions) error PauseFn func(ctx context.Context, project string, options PauseOptions) error
@ -252,9 +252,9 @@ func (s *ServiceProxy) RunOneOffContainer(ctx context.Context, project *types.Pr
} }
//Remove implements Service interface //Remove implements Service interface
func (s *ServiceProxy) Remove(ctx context.Context, project *types.Project, options RemoveOptions) ([]string, error) { func (s *ServiceProxy) Remove(ctx context.Context, project *types.Project, options RemoveOptions) error {
if s.RemoveFn == nil { if s.RemoveFn == nil {
return nil, errdefs.ErrNotImplemented return errdefs.ErrNotImplemented
} }
for _, i := range s.interceptors { for _, i := range s.interceptors {
i(ctx, project) i(ctx, project)

View File

@ -25,7 +25,6 @@ import (
"github.com/spf13/cobra" "github.com/spf13/cobra"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/docker/compose-cli/api/progress"
) )
type buildOptions struct { type buildOptions struct {
@ -88,7 +87,6 @@ func runBuild(ctx context.Context, backend compose.Service, opts buildOptions, s
return err return err
} }
err = progress.Run(ctx, func(ctx context.Context) error {
return backend.Build(ctx, project, compose.BuildOptions{ return backend.Build(ctx, project, compose.BuildOptions{
Pull: opts.pull, Pull: opts.pull,
Progress: opts.progress, Progress: opts.progress,
@ -96,6 +94,4 @@ func runBuild(ctx context.Context, backend compose.Service, opts buildOptions, s
NoCache: opts.noCache, NoCache: opts.noCache,
Quiet: opts.quiet, Quiet: opts.quiet,
}) })
})
return err
} }

View File

@ -22,7 +22,6 @@ import (
"github.com/spf13/cobra" "github.com/spf13/cobra"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/docker/compose-cli/api/progress"
) )
type pauseOptions struct { type pauseOptions struct {
@ -49,12 +48,9 @@ func runPause(ctx context.Context, backend compose.Service, opts pauseOptions, s
return err return err
} }
err = progress.Run(ctx, func(ctx context.Context) error {
return backend.Pause(ctx, project, compose.PauseOptions{ return backend.Pause(ctx, project, compose.PauseOptions{
Services: services, Services: services,
}) })
})
return err
} }
type unpauseOptions struct { type unpauseOptions struct {
@ -81,9 +77,7 @@ func runUnPause(ctx context.Context, backend compose.Service, opts unpauseOption
return err return err
} }
return progress.Run(ctx, func(ctx context.Context) error {
return backend.UnPause(ctx, project, compose.PauseOptions{ return backend.UnPause(ctx, project, compose.PauseOptions{
Services: services, Services: services,
}) })
})
} }

View File

@ -25,7 +25,6 @@ import (
"github.com/spf13/cobra" "github.com/spf13/cobra"
"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/utils" "github.com/docker/compose-cli/utils"
) )
@ -86,15 +85,8 @@ func runPull(ctx context.Context, backend compose.Service, opts pullOptions, ser
project.Services = enabled project.Services = enabled
} }
apiOpts := compose.PullOptions{ return backend.Pull(ctx, project, compose.PullOptions{
Quiet: opts.quiet,
IgnoreFailures: opts.ignorePullFailures, IgnoreFailures: opts.ignorePullFailures,
}
if opts.quiet {
return backend.Pull(ctx, project, apiOpts)
}
return progress.Run(ctx, func(ctx context.Context) error {
return backend.Pull(ctx, project, apiOpts)
}) })
} }

View File

@ -22,7 +22,6 @@ import (
"github.com/spf13/cobra" "github.com/spf13/cobra"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/docker/compose-cli/api/progress"
) )
type pushOptions struct { type pushOptions struct {
@ -54,9 +53,7 @@ func runPush(ctx context.Context, backend compose.Service, opts pushOptions, ser
return err return err
} }
return progress.Run(ctx, func(ctx context.Context) error {
return backend.Push(ctx, project, compose.PushOptions{ return backend.Push(ctx, project, compose.PushOptions{
IgnoreFailures: opts.Ignorefailures, IgnoreFailures: opts.Ignorefailures,
}) })
})
} }

View File

@ -18,13 +18,8 @@ package compose
import ( import (
"context" "context"
"fmt"
"strings"
"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/utils/prompt"
"github.com/spf13/cobra" "github.com/spf13/cobra"
) )
@ -69,46 +64,12 @@ func runRemove(ctx context.Context, backend compose.Service, opts removeOptions,
} }
if opts.stop { if opts.stop {
err = progress.Run(ctx, func(ctx context.Context) error {
return backend.Stop(ctx, project, compose.StopOptions{ return backend.Stop(ctx, project, compose.StopOptions{
Services: services, Services: services,
}) })
})
if err != nil {
return err
}
} }
resources, err := backend.Remove(ctx, project, compose.RemoveOptions{ return backend.Remove(ctx, project, compose.RemoveOptions{
DryRun: true,
Services: services, Services: services,
}) })
if err != nil {
return err
}
if len(resources) == 0 {
fmt.Println("No stopped containers")
return nil
}
msg := fmt.Sprintf("Going to remove %s", strings.Join(resources, ", "))
if opts.force {
fmt.Println(msg)
} else {
confirm, err := prompt.User{}.Confirm(msg, false)
if err != nil {
return err
}
if !confirm {
return nil
}
}
return progress.Run(ctx, func(ctx context.Context) error {
_, err := backend.Remove(ctx, project, compose.RemoveOptions{
Volumes: opts.volumes,
Force: opts.force,
})
return err
})
} }

View File

@ -23,7 +23,6 @@ import (
"github.com/spf13/cobra" "github.com/spf13/cobra"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/docker/compose-cli/api/progress"
) )
type restartOptions struct { type restartOptions struct {
@ -55,10 +54,8 @@ func runRestart(ctx context.Context, backend compose.Service, opts restartOption
} }
timeout := time.Duration(opts.timeout) * time.Second timeout := time.Duration(opts.timeout) * time.Second
return progress.Run(ctx, func(ctx context.Context) error {
return backend.Restart(ctx, project, compose.RestartOptions{ return backend.Restart(ctx, project, compose.RestartOptions{
Timeout: &timeout, Timeout: &timeout,
Services: services, Services: services,
}) })
})
} }

View File

@ -20,8 +20,6 @@ import (
"context" "context"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/docker/compose-cli/api/progress"
"github.com/spf13/cobra" "github.com/spf13/cobra"
) )
@ -49,7 +47,5 @@ func runStart(ctx context.Context, backend compose.Service, opts startOptions, s
return err return err
} }
return progress.Run(ctx, func(ctx context.Context) error {
return backend.Start(ctx, project, compose.StartOptions{}) return backend.Start(ctx, project, compose.StartOptions{})
})
} }

View File

@ -23,7 +23,6 @@ import (
"github.com/spf13/cobra" "github.com/spf13/cobra"
"github.com/docker/compose-cli/api/compose" "github.com/docker/compose-cli/api/compose"
"github.com/docker/compose-cli/api/progress"
) )
type stopOptions struct { type stopOptions struct {
@ -63,10 +62,8 @@ func runStop(ctx context.Context, backend compose.Service, opts stopOptions, ser
timeoutValue := time.Duration(opts.timeout) * time.Second timeoutValue := time.Duration(opts.timeout) * time.Second
timeout = &timeoutValue timeout = &timeoutValue
} }
return progress.Run(ctx, func(ctx context.Context) error {
return backend.Stop(ctx, project, compose.StopOptions{ return backend.Stop(ctx, project, compose.StopOptions{
Timeout: timeout, Timeout: timeout,
Services: services, Services: services,
}) })
})
} }

View File

@ -180,7 +180,7 @@ func (e ecsLocalSimulation) RunOneOffContainer(ctx context.Context, project *typ
return 0, errors.Wrap(errdefs.ErrNotImplemented, "use docker-compose run") return 0, errors.Wrap(errdefs.ErrNotImplemented, "use docker-compose run")
} }
func (e ecsLocalSimulation) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) ([]string, error) { func (e ecsLocalSimulation) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) error {
return e.compose.Remove(ctx, project, options) return e.compose.Remove(ctx, project, options)
} }

View File

@ -29,6 +29,6 @@ func (b *ecsAPIService) RunOneOffContainer(ctx context.Context, project *types.P
return 0, errdefs.ErrNotImplemented return 0, errdefs.ErrNotImplemented
} }
func (b *ecsAPIService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) ([]string, error) { func (b *ecsAPIService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) error {
return nil, errdefs.ErrNotImplemented return errdefs.ErrNotImplemented
} }

View File

@ -248,8 +248,8 @@ func (s *composeService) RunOneOffContainer(ctx context.Context, project *types.
return 0, errdefs.ErrNotImplemented return 0, errdefs.ErrNotImplemented
} }
func (s *composeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) ([]string, error) { func (s *composeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) error {
return nil, errdefs.ErrNotImplemented return errdefs.ErrNotImplemented
} }
// Exec executes a command in a running service container // Exec executes a command in a running service container

View File

@ -42,6 +42,12 @@ import (
) )
func (s *composeService) Build(ctx context.Context, project *types.Project, options compose.BuildOptions) error { func (s *composeService) Build(ctx context.Context, project *types.Project, options compose.BuildOptions) error {
return composeprogress.Run(ctx, func(ctx context.Context) error {
return s.build(ctx, project, options)
})
}
func (s *composeService) build(ctx context.Context, project *types.Project, options compose.BuildOptions) error {
opts := map[string]build.Options{} opts := map[string]build.Options{}
imagesToBuild := []string{} imagesToBuild := []string{}
@ -76,7 +82,7 @@ func (s *composeService) Build(ctx context.Context, project *types.Project, opti
} }
} }
_, err := s.build(ctx, project, opts, Containers{}, options.Progress) _, err := s.doBuild(ctx, project, opts, Containers{}, options.Progress)
if err == nil { if err == nil {
if len(imagesToBuild) > 0 && !options.Quiet { if len(imagesToBuild) > 0 && !options.Quiet {
utils.DisplayScanSuggestMsg() utils.DisplayScanSuggestMsg()
@ -100,7 +106,7 @@ func (s *composeService) ensureImagesExists(ctx context.Context, project *types.
if err != nil { if err != nil {
return err return err
} }
builtImages, err := s.build(ctx, project, opts, observedState, mode) builtImages, err := s.doBuild(ctx, project, opts, observedState, mode)
if err != nil { if err != nil {
return err return err
} }
@ -188,7 +194,7 @@ func (s *composeService) getImageDigests(ctx context.Context, project *types.Pro
return images, nil return images, nil
} }
func (s *composeService) build(ctx context.Context, project *types.Project, opts map[string]build.Options, observedState Containers, mode string) (map[string]string, error) { func (s *composeService) doBuild(ctx context.Context, project *types.Project, opts map[string]build.Options, observedState Containers, mode string) (map[string]string, error) {
info, err := s.apiClient.Info(ctx) info, err := s.apiClient.Info(ctx)
if err != nil { if err != nil {
return nil, err return nil, err

View File

@ -27,6 +27,12 @@ import (
) )
func (s *composeService) Pause(ctx context.Context, project string, options compose.PauseOptions) error { func (s *composeService) Pause(ctx context.Context, project string, options compose.PauseOptions) error {
return progress.Run(ctx, func(ctx context.Context) error {
return s.pause(ctx, project, options)
})
}
func (s *composeService) pause(ctx context.Context, project string, options compose.PauseOptions) error {
containers, err := s.getContainers(ctx, project, oneOffExclude, true, options.Services...) containers, err := s.getContainers(ctx, project, oneOffExclude, true, options.Services...)
if err != nil { if err != nil {
return err return err
@ -49,6 +55,12 @@ func (s *composeService) Pause(ctx context.Context, project string, options comp
} }
func (s *composeService) UnPause(ctx context.Context, project string, options compose.PauseOptions) error { func (s *composeService) UnPause(ctx context.Context, project string, options compose.PauseOptions) error {
return progress.Run(ctx, func(ctx context.Context) error {
return s.unPause(ctx, project, options)
})
}
func (s *composeService) unPause(ctx context.Context, project string, options compose.PauseOptions) error {
containers, err := s.getContainers(ctx, project, oneOffExclude, true, options.Services...) containers, err := s.getContainers(ctx, project, oneOffExclude, true, options.Services...)
if err != nil { if err != nil {
return err return err

View File

@ -38,6 +38,15 @@ import (
) )
func (s *composeService) Pull(ctx context.Context, project *types.Project, opts compose.PullOptions) error { func (s *composeService) Pull(ctx context.Context, project *types.Project, opts compose.PullOptions) error {
if opts.Quiet {
return s.pull(ctx, project, opts)
}
return progress.Run(ctx, func(ctx context.Context) error {
return s.pull(ctx, project, opts)
})
}
func (s *composeService) pull(ctx context.Context, project *types.Project, opts compose.PullOptions) error {
info, err := s.apiClient.Info(ctx) info, err := s.apiClient.Info(ctx)
if err != nil { if err != nil {
return err return err

View File

@ -39,6 +39,12 @@ import (
) )
func (s *composeService) Push(ctx context.Context, project *types.Project, options compose.PushOptions) error { func (s *composeService) Push(ctx context.Context, project *types.Project, options compose.PushOptions) error {
return progress.Run(ctx, func(ctx context.Context) error {
return s.push(ctx, project, options)
})
}
func (s *composeService) push(ctx context.Context, project *types.Project, options compose.PushOptions) error {
configFile, err := cliconfig.Load(config.Dir()) configFile, err := cliconfig.Load(config.Dir())
if err != nil { if err != nil {
return err return err

View File

@ -18,17 +18,20 @@ package compose
import ( import (
"context" "context"
"fmt"
"strings"
"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"
status "github.com/docker/compose-cli/local/moby" status "github.com/docker/compose-cli/local/moby"
"github.com/docker/compose-cli/utils/prompt"
"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"
"golang.org/x/sync/errgroup" "golang.org/x/sync/errgroup"
) )
func (s *composeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) ([]string, error) { func (s *composeService) Remove(ctx context.Context, project *types.Project, options compose.RemoveOptions) error {
services := options.Services services := options.Services
if len(services) == 0 { if len(services) == 0 {
services = project.ServiceNames() services = project.ServiceNames()
@ -36,7 +39,7 @@ func (s *composeService) Remove(ctx context.Context, project *types.Project, opt
containers, err := s.getContainers(ctx, project.Name, oneOffInclude, true, services...) containers, err := s.getContainers(ctx, project.Name, oneOffInclude, true, services...)
if err != nil { if err != nil {
return nil, err return err
} }
stoppedContainers := containers.filter(func(c moby.Container) bool { stoppedContainers := containers.filter(func(c moby.Container) bool {
@ -48,18 +51,36 @@ func (s *composeService) Remove(ctx context.Context, project *types.Project, opt
names = append(names, getCanonicalContainerName(c)) names = append(names, getCanonicalContainerName(c))
}) })
if options.DryRun { if len(names) == 0 {
return names, nil fmt.Println("No stopped containers")
return nil
}
msg := fmt.Sprintf("Going to remove %s", strings.Join(names, ", "))
if options.Force {
fmt.Println(msg)
} else {
confirm, err := prompt.User{}.Confirm(msg, false)
if err != nil {
return err
}
if !confirm {
return nil
}
}
return progress.Run(ctx, func(ctx context.Context) error {
return s.remove(ctx, stoppedContainers, options)
})
} }
func (s *composeService) remove(ctx context.Context, containers Containers, options compose.RemoveOptions) error {
w := progress.ContextWriter(ctx) w := progress.ContextWriter(ctx)
eg, ctx := errgroup.WithContext(ctx) eg, ctx := errgroup.WithContext(ctx)
for _, c := range stoppedContainers { for _, c := range containers {
c := c c := c
eg.Go(func() error { eg.Go(func() error {
eventName := getContainerProgressName(c) eventName := getContainerProgressName(c)
w.Event(progress.RemovingEvent(eventName)) w.Event(progress.RemovingEvent(eventName))
err = s.apiClient.ContainerRemove(ctx, c.ID, moby.ContainerRemoveOptions{ err := s.apiClient.ContainerRemove(ctx, c.ID, moby.ContainerRemoveOptions{
RemoveVolumes: options.Volumes, RemoveVolumes: options.Volumes,
Force: options.Force, Force: options.Force,
}) })
@ -69,5 +90,5 @@ func (s *composeService) Remove(ctx context.Context, project *types.Project, opt
return err return err
}) })
} }
return nil, eg.Wait() return eg.Wait()
} }

View File

@ -20,12 +20,19 @@ import (
"context" "context"
"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/utils" "github.com/docker/compose-cli/utils"
"github.com/compose-spec/compose-go/types" "github.com/compose-spec/compose-go/types"
) )
func (s *composeService) Restart(ctx context.Context, project *types.Project, options compose.RestartOptions) error { func (s *composeService) Restart(ctx context.Context, project *types.Project, options compose.RestartOptions) error {
return progress.Run(ctx, func(ctx context.Context) error {
return s.restart(ctx, project, options)
})
}
func (s *composeService) restart(ctx context.Context, project *types.Project, options compose.RestartOptions) error {
ctx, err := s.getUpdatedContainersStateContext(ctx, project.Name) ctx, err := s.getUpdatedContainersStateContext(ctx, project.Name)
if err != nil { if err != nil {
return err return err

View File

@ -20,6 +20,7 @@ import (
"context" "context"
"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/utils" "github.com/docker/compose-cli/utils"
"github.com/compose-spec/compose-go/types" "github.com/compose-spec/compose-go/types"
@ -29,6 +30,12 @@ import (
) )
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 {
return progress.Run(ctx, func(ctx context.Context) error {
return s.start(ctx, project, options)
})
}
func (s *composeService) start(ctx context.Context, project *types.Project, options compose.StartOptions) error {
listener := options.Attach listener := options.Attach
if len(options.Services) == 0 { if len(options.Services) == 0 {
options.Services = project.ServiceNames() options.Services = project.ServiceNames()

View File

@ -26,6 +26,12 @@ import (
) )
func (s *composeService) Stop(ctx context.Context, project *types.Project, options compose.StopOptions) error { func (s *composeService) Stop(ctx context.Context, project *types.Project, options compose.StopOptions) error {
return progress.Run(ctx, func(ctx context.Context) error {
return s.stop(ctx, project, options)
})
}
func (s *composeService) stop(ctx context.Context, project *types.Project, options compose.StopOptions) error {
w := progress.ContextWriter(ctx) w := progress.ContextWriter(ctx)
services := options.Services services := options.Services