mirror of
https://github.com/kata-containers/kata-containers.git
synced 2025-06-22 13:38:26 +00:00
v2: Open log fifo with RDWR
instead of WRONLY
The container log fifo is opened as `O_WRONLY` now. When the read side of fifo is closed temporarily such as restarting contaienrd, write to `tty.Stdout` will get an EPIPE error and finally cause `io.CopyBuffer` return. Then `ioCopy` closes the tty io and exits. Thus after containerd restarted, the log fifo can't be reopened. The container will be blocked forever after stdout/stderr buffer is full. Opening the log fifo with `RDWR` instead of `WRONLY` avoids the fifo returning EPIPE when the read side is closed, and keeps the fifo open until the reader reopening it. Fixes: #2590 Signed-off-by: Li Yuxuan <liyuxuan04@baidu.com>
This commit is contained in:
parent
6e398f7c71
commit
8e0f891ebc
@ -63,14 +63,14 @@ func newTtyIO(ctx context.Context, stdin, stdout, stderr string, console bool) (
|
|||||||
}
|
}
|
||||||
|
|
||||||
if stdout != "" {
|
if stdout != "" {
|
||||||
outw, err = fifo.OpenFifo(ctx, stdout, syscall.O_WRONLY, 0)
|
outw, err = fifo.OpenFifo(ctx, stdout, syscall.O_RDWR, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if !console && stderr != "" {
|
if !console && stderr != "" {
|
||||||
errw, err = fifo.OpenFifo(ctx, stderr, syscall.O_WRONLY, 0)
|
errw, err = fifo.OpenFifo(ctx, stderr, syscall.O_RDWR, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
91
containerd-shim-v2/stream_test.go
Normal file
91
containerd-shim-v2/stream_test.go
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
// Copyright (c) 2020 Baidu Corporation
|
||||||
|
//
|
||||||
|
// SPDX-License-Identifier: Apache-2.0
|
||||||
|
//
|
||||||
|
|
||||||
|
package containerdshim
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"path/filepath"
|
||||||
|
"syscall"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/containerd/fifo"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNewTtyIOFifoReopen(t *testing.T) {
|
||||||
|
var outr io.ReadWriteCloser
|
||||||
|
var errr io.ReadWriteCloser
|
||||||
|
var tty *ttyIO
|
||||||
|
assert := assert.New(t)
|
||||||
|
ctx := context.TODO()
|
||||||
|
fifoPath, err := ioutil.TempDir(testDir, "fifo-path-")
|
||||||
|
assert.NoError(err)
|
||||||
|
stdout := filepath.Join(fifoPath, "stdout")
|
||||||
|
stderr := filepath.Join(fifoPath, "stderr")
|
||||||
|
|
||||||
|
createReadFifo := func(f string) io.ReadWriteCloser {
|
||||||
|
rf, err := fifo.OpenFifo(ctx, f, syscall.O_RDONLY|syscall.O_CREAT|syscall.O_NONBLOCK, 0700)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
return rf
|
||||||
|
}
|
||||||
|
|
||||||
|
outr = createReadFifo(stdout)
|
||||||
|
defer outr.Close()
|
||||||
|
errr = createReadFifo(stderr)
|
||||||
|
defer errr.Close()
|
||||||
|
tty, err = newTtyIO(ctx, "", stdout, stderr, false)
|
||||||
|
assert.NoError(err)
|
||||||
|
defer tty.close()
|
||||||
|
|
||||||
|
testBytes := []byte("T")
|
||||||
|
checkFifoWrite := func(w io.Writer) {
|
||||||
|
_, err = w.Write(testBytes)
|
||||||
|
assert.NoError(err)
|
||||||
|
}
|
||||||
|
checkFifoRead := func(r io.Reader) {
|
||||||
|
var err error
|
||||||
|
buf := make([]byte, 1)
|
||||||
|
done := make(chan struct{})
|
||||||
|
timer := time.NewTimer(2 * time.Second)
|
||||||
|
go func() {
|
||||||
|
_, err = r.Read(buf)
|
||||||
|
close(done)
|
||||||
|
}()
|
||||||
|
select {
|
||||||
|
case <-done:
|
||||||
|
assert.NoError(err)
|
||||||
|
assert.Equal(buf, testBytes)
|
||||||
|
case <-timer.C:
|
||||||
|
t.Fatal("read fifo timeout")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
checkFifoWrite(tty.Stdout)
|
||||||
|
checkFifoRead(outr)
|
||||||
|
checkFifoWrite(tty.Stderr)
|
||||||
|
checkFifoRead(errr)
|
||||||
|
|
||||||
|
err = outr.Close()
|
||||||
|
assert.NoError(err)
|
||||||
|
err = errr.Close()
|
||||||
|
assert.NoError(err)
|
||||||
|
|
||||||
|
// Make sure that writing to tty fifo will not get `EPIPE`
|
||||||
|
// when the read side is closed
|
||||||
|
checkFifoWrite(tty.Stdout)
|
||||||
|
checkFifoWrite(tty.Stderr)
|
||||||
|
|
||||||
|
// Reopen the fifo
|
||||||
|
outr = createReadFifo(stdout)
|
||||||
|
errr = createReadFifo(stderr)
|
||||||
|
checkFifoRead(outr)
|
||||||
|
checkFifoRead(errr)
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user