Attach to container using compose attach

Signed-off-by: Guillaume Tardif <guillaume.tardif@gmail.com>
This commit is contained in:
Guillaume Tardif 2020-12-16 17:40:49 +01:00
parent 76f36a69c6
commit d2cfffafb4
1 changed files with 28 additions and 45 deletions

View File

@ -19,13 +19,15 @@ package compose
import ( import (
"context" "context"
"fmt" "fmt"
"io"
"os" "os"
"github.com/compose-spec/compose-go/types" "github.com/compose-spec/compose-go/types"
"github.com/docker/compose-cli/api/compose" "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" 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" 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{}) return containerID, s.apiClient.ContainerStart(ctx, containerID, apitypes.ContainerStartOptions{})
} }
cnx, err := s.apiClient.ContainerAttach(ctx, containerID, apitypes.ContainerAttachOptions{ containers, err := s.apiClient.ContainerList(ctx, apitypes.ContainerListOptions{
Stream: true, Filters: filters.NewArgs(
Stdin: true, projectFilter(project.Name),
Stdout: true, ),
Stderr: true, All: true,
Logs: true,
}) })
if err != nil { 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{}) err = s.apiClient.ContainerStart(ctx, containerID, apitypes.ContainerStartOptions{})
if err != nil { if err != nil {
return "", err
}
err = eg.Wait()
return containerID, err return containerID, err
} }
for {
select {
case err := <-readChannel:
return containerID, err
case err := <-writeChannel:
return containerID, err
}
}
}