Skip to content

Commit e97d96b

Browse files
author
Jim Minter
committed
Work around docker race condition when running build post commit hooks.
1 parent f1dd3c7 commit e97d96b

File tree

4 files changed

+38
-10
lines changed

4 files changed

+38
-10
lines changed

pkg/build/builder/common.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -150,11 +150,11 @@ func execPostCommitHook(client DockerClient, postCommitSpec api.BuildPostCommitS
150150
Memory: limits.MemoryLimitBytes,
151151
MemorySwap: limits.MemorySwap,
152152
},
153-
}, docker.LogsOptions{
153+
}, docker.AttachToContainerOptions{
154154
// Stream logs to stdout and stderr.
155155
OutputStream: os.Stdout,
156156
ErrorStream: os.Stderr,
157-
Follow: true,
157+
Stream: true,
158158
Stdout: true,
159159
Stderr: true,
160160
})

pkg/build/builder/dockerutil.go

+31-6
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ var (
3939
// DockerClient is an interface to the Docker client that contains
4040
// the methods used by the common builder
4141
type DockerClient interface {
42+
AttachToContainerNonBlocking(opts docker.AttachToContainerOptions) (docker.CloseWaiter, error)
4243
BuildImage(opts docker.BuildImageOptions) error
4344
PushImage(opts docker.PushImageOptions, auth docker.AuthConfiguration) error
4445
RemoveImage(name string) error
@@ -169,7 +170,7 @@ func tagImage(dockerClient DockerClient, image, name string) error {
169170
// dockerRun mimics the 'docker run --rm' CLI command. It uses the Docker Remote
170171
// API to create and start a container and stream its logs. The container is
171172
// removed after it terminates.
172-
func dockerRun(client DockerClient, createOpts docker.CreateContainerOptions, logsOpts docker.LogsOptions) error {
173+
func dockerRun(client DockerClient, createOpts docker.CreateContainerOptions, attachOpts docker.AttachToContainerOptions) error {
173174
// Create a new container.
174175
glog.V(4).Infof("Creating container with options {Name:%q Config:%+v HostConfig:%+v} ...", createOpts.Name, createOpts.Config, createOpts.HostConfig)
175176
c, err := client.CreateContainer(createOpts)
@@ -188,17 +189,41 @@ func dockerRun(client DockerClient, createOpts docker.CreateContainerOptions, lo
188189
}
189190
}
190191
startWaitContainer := func() error {
192+
// Changed to use attach call instead of logs call to stream stdout/stderr
193+
// during execution to avoid race condition
194+
// https://github.com/docker/docker/issues/31323 .
195+
// Using attach call is also racy in docker versions which don't carry
196+
// https://github.com/docker/docker/pull/30446 .
197+
// In RHEL, docker >= 1.12.6-10.el7.x86_64 should be OK.
198+
199+
// Attach to the container.
200+
success := make(chan struct{})
201+
attachOpts.Container = c.ID
202+
attachOpts.Success = success
203+
glog.V(4).Infof("Attaching to container %q with options %+v ...", containerName, attachOpts)
204+
wc, err := client.AttachToContainerNonBlocking(attachOpts)
205+
if err != nil {
206+
return fmt.Errorf("attach container %q: %v", containerName, err)
207+
}
208+
defer wc.Close()
209+
210+
select {
211+
case <-success:
212+
close(success)
213+
case <-time.After(120 * time.Second):
214+
return fmt.Errorf("attach container %q: timeout waiting for success signal", containerName)
215+
}
216+
191217
// Start the container.
192218
glog.V(4).Infof("Starting container %q ...", containerName)
193219
if err := client.StartContainer(c.ID, nil); err != nil {
194220
return fmt.Errorf("start container %q: %v", containerName, err)
195221
}
196222

197-
// Stream container logs.
198-
logsOpts.Container = c.ID
199-
glog.V(4).Infof("Streaming logs of container %q with options %+v ...", containerName, logsOpts)
200-
if err := client.Logs(logsOpts); err != nil {
201-
return fmt.Errorf("streaming logs of %q: %v", containerName, err)
223+
// Wait for streaming to finish.
224+
glog.V(4).Infof("Waiting for streaming to finish ...")
225+
if err := wc.Wait(); err != nil {
226+
return fmt.Errorf("container %q streaming: %v", containerName, err)
202227
}
203228

204229
// Return an error if the exit code of the container is non-zero.

pkg/build/builder/dockerutil_test.go

+3
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,9 @@ func (d *FakeDocker) WaitContainer(id string) (int, error) {
9494
func (d *FakeDocker) Logs(opts docker.LogsOptions) error {
9595
return nil
9696
}
97+
func (d *FakeDocker) AttachToContainerNonBlocking(opts docker.AttachToContainerOptions) (docker.CloseWaiter, error) {
98+
return nil, nil
99+
}
97100
func (d *FakeDocker) TagImage(name string, opts docker.TagImageOptions) error {
98101
d.callLog = append(d.callLog, methodCall{"TagImage", []interface{}{name, opts}})
99102
return nil

test/extended/util/framework.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -592,9 +592,9 @@ func WaitForAnImageStream(client client.ImageStreamInterface,
592592
}
593593

594594
// WaitForAnImageStreamTag waits until an image stream with given name has non-empty history for given tag.
595-
// Defaults to waiting for 60 seconds
595+
// Defaults to waiting for 300 seconds
596596
func WaitForAnImageStreamTag(oc *CLI, namespace, name, tag string) error {
597-
return TimedWaitForAnImageStreamTag(oc, namespace, name, tag, time.Second*60)
597+
return TimedWaitForAnImageStreamTag(oc, namespace, name, tag, time.Second*300)
598598
}
599599

600600
// TimedWaitForAnImageStreamTag waits until an image stream with given name has non-empty history for given tag.

0 commit comments

Comments
 (0)