@@ -3,9 +3,6 @@ package run
3
3
import (
4
4
"bytes"
5
5
"fmt"
6
- "io"
7
- "io/ioutil"
8
- "sync"
9
6
10
7
docker "github.com/fsouza/go-dockerclient"
11
8
"github.com/golang/glog"
@@ -37,7 +34,6 @@ type Runner struct {
37
34
client * docker.Client
38
35
config * docker.Config
39
36
hostConfig * docker.HostConfig
40
- input io.Reader
41
37
removeContainer bool
42
38
}
43
39
@@ -119,14 +115,6 @@ func (h *Runner) DiscardContainer() *Runner {
119
115
return h
120
116
}
121
117
122
- // Input sets an input stream for the Docker run command
123
- func (h * Runner ) Input (reader io.Reader ) * Runner {
124
- h .config .OpenStdin = true
125
- h .config .StdinOnce = true
126
- h .input = reader
127
- return h
128
- }
129
-
130
118
// Start starts the container as a daemon and returns
131
119
func (h * Runner ) Start () (string , error ) {
132
120
id , err := h .Create ()
@@ -138,22 +126,13 @@ func (h *Runner) Start() (string, error) {
138
126
139
127
// Output starts the container, waits for it to finish and returns its output
140
128
func (h * Runner ) Output () (string , string , int , error ) {
141
- stdOut , errOut := & bytes.Buffer {}, & bytes.Buffer {}
142
- rc , err := h .runWithOutput (h .input , stdOut , errOut )
143
- return stdOut .String (), errOut .String (), rc , err
144
- }
145
-
146
- // CombinedOutput is the same as Output, except both output and error streams
147
- // are combined into one.
148
- func (h * Runner ) CombinedOutput () (string , int , error ) {
149
- out := & bytes.Buffer {}
150
- rc , err := h .runWithOutput (h .input , out , out )
151
- return out .String (), rc , err
129
+ return h .runWithOutput ()
152
130
}
153
131
154
132
// Run executes the container and waits until it completes
155
133
func (h * Runner ) Run () (int , error ) {
156
- return h .runWithOutput (h .input , ioutil .Discard , ioutil .Discard )
134
+ _ , _ , rc , err := h .runWithOutput ()
135
+ return rc , err
157
136
}
158
137
159
138
func (h * Runner ) Create () (string , error ) {
@@ -179,10 +158,10 @@ func (h *Runner) startContainer(id string) error {
179
158
return nil
180
159
}
181
160
182
- func (h * Runner ) runWithOutput (stdIn io. Reader , stdOut , stdErr io. Writer ) ( int , error ) {
161
+ func (h * Runner ) runWithOutput () ( string , string , int , error ) {
183
162
id , err := h .Create ()
184
163
if err != nil {
185
- return 0 , err
164
+ return "" , "" , 0 , err
186
165
}
187
166
if h .removeContainer {
188
167
defer func () {
@@ -192,64 +171,48 @@ func (h *Runner) runWithOutput(stdIn io.Reader, stdOut, stdErr io.Writer) (int,
192
171
}
193
172
}()
194
173
}
195
- logOut , logErr := & bytes.Buffer {}, & bytes.Buffer {}
196
- outStream := io .MultiWriter (stdOut , logOut )
197
- errStream := io .MultiWriter (stdErr , logErr )
198
- attached := make (chan struct {})
199
- attachErr := make (chan error )
200
- streamingWG := & sync.WaitGroup {}
201
- streamingWG .Add (1 )
202
- go func () {
203
- glog .V (5 ).Infof ("Attaching to container %q" , id )
204
- err = h .client .AttachToContainer (docker.AttachToContainerOptions {
205
- Container : id ,
206
- Logs : true ,
207
- Stream : true ,
208
- Stdout : true ,
209
- Stderr : true ,
210
- Stdin : stdIn != nil ,
211
- OutputStream : outStream ,
212
- ErrorStream : errStream ,
213
- InputStream : stdIn ,
214
- Success : attached ,
215
- })
216
- if err != nil {
217
- glog .V (2 ).Infof ("Error occurred while attaching: %v" , err )
218
- attachErr <- err
219
- }
220
- streamingWG .Done ()
221
- glog .V (5 ).Infof ("Done attaching to container %q" , id )
222
- }()
223
-
224
- select {
225
- case <- attached :
226
- glog .V (5 ).Infof ("Attach is successful." )
227
- case err = <- attachErr :
228
- return 0 , err
229
- }
174
+
230
175
glog .V (5 ).Infof ("Starting container %q" , id )
231
176
err = h .startContainer (id )
232
177
if err != nil {
233
178
glog .V (2 ).Infof ("Error occurred starting container %q: %v" , id , err )
234
- return 0 , err
179
+ return "" , "" , 0 , err
235
180
}
236
- glog .V (5 ).Infof ("signaling attached channel" )
237
- attached <- struct {}{}
181
+
238
182
glog .V (5 ).Infof ("Waiting for container %q" , id )
239
183
rc , err := h .client .WaitContainer (id )
240
184
if err != nil {
241
- glog .V (2 ).Infof ("Error occurred waiting for container %q: %v, rc = %d" , id , err , rc )
185
+ glog .V (2 ).Infof ("Error occurred waiting for container %q: %v" , id , err )
186
+ return "" , "" , 0 , err
187
+ }
188
+ glog .V (5 ).Infof ("Done waiting for container %q, rc=%d" , id , rc )
189
+
190
+ stdOut , stdErr := & bytes.Buffer {}, & bytes.Buffer {}
191
+
192
+ // changed to only reading logs after execution instead of streaming
193
+ // stdout/stderr to avoid race condition in (at least) docker 1.10-1.14-dev:
194
+ // https://github.com/docker/docker/issues/29285
195
+ glog .V (5 ).Infof ("Reading logs from container %q" , id )
196
+ err = h .client .Logs (docker.LogsOptions {
197
+ Container : id ,
198
+ Stdout : true ,
199
+ Stderr : true ,
200
+ OutputStream : stdOut ,
201
+ ErrorStream : stdErr ,
202
+ })
203
+ if err != nil {
204
+ glog .V (2 ).Infof ("Error occurred while reading logs: %v" , err )
205
+ return "" , "" , 0 , err
242
206
}
243
- glog .V (5 ).Infof ("Done waiting for container %q, rc=%d. Waiting for attach routine" , id , rc )
244
- streamingWG .Wait ()
245
- glog .V (5 ).Infof ("Done waiting for attach routine" )
246
- glog .V (5 ).Infof ("Stdout:\n %s" , logOut .String ())
247
- glog .V (5 ).Infof ("Stderr:\n %s" , logErr .String ())
207
+ glog .V (5 ).Infof ("Done reading logs from container %q" , id )
208
+
209
+ glog .V (5 ).Infof ("Stdout:\n %s" , stdOut .String ())
210
+ glog .V (5 ).Infof ("Stderr:\n %s" , stdErr .String ())
248
211
if rc != 0 || err != nil {
249
- return rc , newRunError (rc , err , logOut .Bytes (), logErr .Bytes (), h .config )
212
+ return stdOut . String (), stdErr . String (), rc , newRunError (rc , err , stdOut .Bytes (), stdErr .Bytes (), h .config )
250
213
}
251
214
glog .V (4 ).Infof ("Container run successful\n " )
252
- return 0 , nil
215
+ return stdOut . String (), stdErr . String (), rc , nil
253
216
}
254
217
255
218
// printConfig prints out the relevant parts of a container's Docker config
0 commit comments