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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
|
// Copyright 2016 syzkaller project authors. All rights reserved.
// Use of this source code is governed by Apache 2 LICENSE that can be found in the LICENSE file.
package vmimpl
import (
"bytes"
"context"
"errors"
"io"
"testing"
"time"
"github.com/google/syzkaller/pkg/osutil"
"github.com/stretchr/testify/assert"
)
func TestMerger(t *testing.T) {
tee := new(bytes.Buffer)
merger := NewOutputMerger(tee)
rp1, wp1, err := osutil.LongPipe()
if err != nil {
t.Fatal(err)
}
defer wp1.Close()
merger.Add("pipe1", OutputConsole, rp1)
rp2, wp2, err := osutil.LongPipe()
if err != nil {
t.Fatal(err)
}
defer wp2.Close()
merger.Add("pipe2", OutputConsole, rp2)
wp1.Write([]byte("111"))
select {
case <-merger.Output:
t.Fatalf("merger produced incomplete line")
case <-time.After(10 * time.Millisecond):
}
wp2.Write([]byte("222"))
select {
case <-merger.Output:
t.Fatalf("merger produced incomplete line")
case <-time.After(10 * time.Millisecond):
}
wp1.Write([]byte("333\n444\r"))
got := (<-merger.Output).Data
if want := "111333\n"; string(got) != want {
t.Fatalf("bad line: '%s', want '%s'", got, want)
}
wp2.Write([]byte("555\r\n666\n\r\r777"))
got = (<-merger.Output).Data
if want := "222555\n666\n"; string(got) != want {
t.Fatalf("bad line: '%s', want '%s'", got, want)
}
// We need to robustly read until we get what we want if we want to be safe.
// But for now let's just match what the implementation does.
// The implementation sends everything it read.
wp1.Close()
got = (<-merger.Output).Data
if want := "444\n"; string(got) != want {
t.Fatalf("bad line: '%s', want '%s'", got, want)
}
var merr MergerError
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
if err := <-merger.Errors(ctx); err == nil {
t.Fatalf("merger did not produce an error on pipe close")
} else if !errors.As(err, &merr) || merr.Name != "pipe1" || merr.R != rp1 || merr.Err != io.EOF {
t.Fatalf("merger produced wrong error: %v", err)
}
wp2.Close()
got = (<-merger.Output).Data
if want := "777\n"; string(got) != want {
t.Fatalf("bad line: '%s', want '%s'", got, want)
}
merger.Wait()
want := "111333\n222555\n666\n444\n777\n"
if got := tee.String(); got != want {
t.Fatalf("bad tee: '%s', want '%s'", got, want)
}
}
type brokenReader struct {
err error
}
func (r *brokenReader) Read(p []byte) (int, error) {
return 0, r.err
}
func (r *brokenReader) Close() error { return nil }
func TestMergerErrors(t *testing.T) {
merger := NewOutputMerger(nil)
r1 := &brokenReader{errors.New("foo")}
merger.Add("foo", OutputConsole, r1)
ctx := context.Background()
var merr MergerError
// Add a background reader that will just hang.
rHang, wHang, err := osutil.LongPipe()
if err != nil {
t.Fatal(err)
}
merger.Add("background", OutputConsole, rHang)
err = <-merger.Errors(ctx)
if assert.Error(t, err) {
assert.True(t, errors.As(err, &merr))
assert.Equal(t, "foo", merr.Name)
assert.EqualError(t, merr.Err, "foo")
}
// The error must persist.
err = <-merger.Errors(ctx)
if assert.Error(t, err) {
assert.True(t, errors.As(err, &merr))
assert.Equal(t, "foo", merr.Name)
assert.EqualError(t, merr.Err, "foo")
}
// We re-add the decoder as "foo".
// The previous error should be gone.
r2 := &brokenReader{errors.New("bar")}
merger.Add("foo", OutputConsole, r2)
err = <-merger.Errors(ctx)
if assert.Error(t, err) {
assert.True(t, errors.As(err, &merr))
assert.Equal(t, "foo", merr.Name)
assert.EqualError(t, merr.Err, "bar")
}
wHang.Close()
merger.Wait()
}
|