mirror of https://github.com/docker/compose.git
Attach to container using compose attach
Signed-off-by: Guillaume Tardif <guillaume.tardif@gmail.com>
This commit is contained in:
parent
76f36a69c6
commit
d2cfffafb4
|
@ -19,13 +19,15 @@ package compose
|
|||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"io"
|
||||
"os"
|
||||
|
||||
"github.com/compose-spec/compose-go/types"
|
||||
"github.com/docker/compose-cli/api/compose"
|
||||
convert "github.com/docker/compose-cli/local/moby"
|
||||
"github.com/docker/compose-cli/utils"
|
||||
apitypes "github.com/docker/docker/api/types"
|
||||
"github.com/docker/docker/api/types/filters"
|
||||
"golang.org/x/sync/errgroup"
|
||||
|
||||
moby "github.com/docker/docker/pkg/stringid"
|
||||
)
|
||||
|
||||
|
@ -65,52 +67,33 @@ func (s *composeService) RunOneOffContainer(ctx context.Context, project *types.
|
|||
return containerID, s.apiClient.ContainerStart(ctx, containerID, apitypes.ContainerStartOptions{})
|
||||
}
|
||||
|
||||
cnx, err := s.apiClient.ContainerAttach(ctx, containerID, apitypes.ContainerAttachOptions{
|
||||
Stream: true,
|
||||
Stdin: true,
|
||||
Stdout: true,
|
||||
Stderr: true,
|
||||
Logs: true,
|
||||
containers, err := s.apiClient.ContainerList(ctx, apitypes.ContainerListOptions{
|
||||
Filters: filters.NewArgs(
|
||||
projectFilter(project.Name),
|
||||
),
|
||||
All: true,
|
||||
})
|
||||
if err != nil {
|
||||
return containerID, err
|
||||
return "", err
|
||||
}
|
||||
var oneoffContainer apitypes.Container
|
||||
for _, container := range containers {
|
||||
if utils.StringContains(container.Names, "/"+containerID) {
|
||||
oneoffContainer = container
|
||||
}
|
||||
}
|
||||
eg := errgroup.Group{}
|
||||
eg.Go(func() error {
|
||||
return s.attachContainerStreams(ctx, oneoffContainer, true, os.Stdin, os.Stdout)
|
||||
})
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
defer cnx.Close()
|
||||
|
||||
stdout := convert.ContainerStdout{HijackedResponse: cnx}
|
||||
stdin := convert.ContainerStdin{HijackedResponse: cnx}
|
||||
|
||||
readChannel := make(chan error, 10)
|
||||
writeChannel := make(chan error, 10)
|
||||
|
||||
go func() {
|
||||
_, err := io.Copy(os.Stdout, cnx.Reader)
|
||||
readChannel <- err
|
||||
}()
|
||||
|
||||
go func() {
|
||||
_, err := io.Copy(stdin, os.Stdin)
|
||||
writeChannel <- err
|
||||
}()
|
||||
|
||||
go func() {
|
||||
<-ctx.Done()
|
||||
stdout.Close() //nolint:errcheck
|
||||
stdin.Close() //nolint:errcheck
|
||||
}()
|
||||
|
||||
// start container
|
||||
err = s.apiClient.ContainerStart(ctx, containerID, apitypes.ContainerStartOptions{})
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
err = eg.Wait()
|
||||
return containerID, err
|
||||
}
|
||||
|
||||
for {
|
||||
select {
|
||||
case err := <-readChannel:
|
||||
return containerID, err
|
||||
case err := <-writeChannel:
|
||||
return containerID, err
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue