1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
|
package container // import "github.com/docker/docker/integration/container"
import (
"testing"
"time"
"github.com/docker/docker/api/types"
"github.com/docker/docker/api/types/container"
"github.com/docker/docker/api/types/network"
systemutil "github.com/docker/docker/integration/internal/system"
"github.com/docker/docker/testutil"
"github.com/docker/docker/testutil/daemon"
"gotest.tools/v3/assert"
is "gotest.tools/v3/assert/cmp"
"gotest.tools/v3/poll"
"gotest.tools/v3/skip"
)
func TestAttach(t *testing.T) {
ctx := setupTest(t)
apiClient := testEnv.APIClient()
tests := []struct {
doc string
tty bool
expectedMediaType string
}{
{
doc: "without TTY",
expectedMediaType: types.MediaTypeMultiplexedStream,
},
{
doc: "with TTY",
tty: true,
expectedMediaType: types.MediaTypeRawStream,
},
}
for _, tc := range tests {
tc := tc
t.Run(tc.doc, func(t *testing.T) {
t.Parallel()
ctx := testutil.StartSpan(ctx, t)
resp, err := apiClient.ContainerCreate(ctx,
&container.Config{
Image: "busybox",
Cmd: []string{"echo", "hello"},
Tty: tc.tty,
},
&container.HostConfig{},
&network.NetworkingConfig{},
nil,
"",
)
assert.NilError(t, err)
attach, err := apiClient.ContainerAttach(ctx, resp.ID, container.AttachOptions{
Stdout: true,
Stderr: true,
})
assert.NilError(t, err)
mediaType, ok := attach.MediaType()
assert.Check(t, ok)
assert.Check(t, is.Equal(mediaType, tc.expectedMediaType))
})
}
}
// Regression test for #37182
func TestAttachDisconnectLeak(t *testing.T) {
skip.If(t, testEnv.DaemonInfo.OSType != "linux", "Bug still exists on Windows")
t.Parallel()
ctx := testutil.StartSpan(baseContext, t)
// Use a new daemon to make sure stuff from other tests isn't affecting the
// goroutine count.
d := daemon.New(t)
defer d.Cleanup(t)
d.StartWithBusybox(ctx, t, "--iptables=false")
client := d.NewClientT(t)
resp, err := client.ContainerCreate(ctx,
&container.Config{
Image: "busybox",
Cmd: []string{"/bin/sh", "-c", "while true; usleep 100000; done"},
},
&container.HostConfig{},
&network.NetworkingConfig{},
nil,
"",
)
assert.NilError(t, err)
cID := resp.ID
defer client.ContainerRemove(ctx, cID, container.RemoveOptions{
Force: true,
})
nGoroutines := systemutil.WaitForStableGoroutineCount(ctx, t, client)
attach, err := client.ContainerAttach(ctx, cID, container.AttachOptions{
Stdout: true,
})
assert.NilError(t, err)
defer attach.Close()
poll.WaitOn(t, func(_ poll.LogT) poll.Result {
count := systemutil.WaitForStableGoroutineCount(ctx, t, client)
if count > nGoroutines {
return poll.Success()
}
return poll.Continue("waiting for goroutines to increase from %d, current: %d", nGoroutines, count)
},
poll.WithTimeout(time.Minute),
)
attach.Close()
poll.WaitOn(t, systemutil.CheckGoroutineCount(ctx, client, nGoroutines), poll.WithTimeout(time.Minute))
}
|