annotate libgo/go/io/multi.go @ 158:494b0b89df80 default tip

...
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Mon, 25 May 2020 18:13:55 +0900
parents 1830386684a0
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 // Copyright 2010 The Go Authors. All rights reserved.
kono
parents:
diff changeset
2 // Use of this source code is governed by a BSD-style
kono
parents:
diff changeset
3 // license that can be found in the LICENSE file.
kono
parents:
diff changeset
4
kono
parents:
diff changeset
5 package io
kono
parents:
diff changeset
6
kono
parents:
diff changeset
7 type eofReader struct{}
kono
parents:
diff changeset
8
kono
parents:
diff changeset
9 func (eofReader) Read([]byte) (int, error) {
kono
parents:
diff changeset
10 return 0, EOF
kono
parents:
diff changeset
11 }
kono
parents:
diff changeset
12
kono
parents:
diff changeset
13 type multiReader struct {
kono
parents:
diff changeset
14 readers []Reader
kono
parents:
diff changeset
15 }
kono
parents:
diff changeset
16
kono
parents:
diff changeset
17 func (mr *multiReader) Read(p []byte) (n int, err error) {
kono
parents:
diff changeset
18 for len(mr.readers) > 0 {
kono
parents:
diff changeset
19 // Optimization to flatten nested multiReaders (Issue 13558).
kono
parents:
diff changeset
20 if len(mr.readers) == 1 {
kono
parents:
diff changeset
21 if r, ok := mr.readers[0].(*multiReader); ok {
kono
parents:
diff changeset
22 mr.readers = r.readers
kono
parents:
diff changeset
23 continue
kono
parents:
diff changeset
24 }
kono
parents:
diff changeset
25 }
kono
parents:
diff changeset
26 n, err = mr.readers[0].Read(p)
kono
parents:
diff changeset
27 if err == EOF {
kono
parents:
diff changeset
28 // Use eofReader instead of nil to avoid nil panic
kono
parents:
diff changeset
29 // after performing flatten (Issue 18232).
kono
parents:
diff changeset
30 mr.readers[0] = eofReader{} // permit earlier GC
kono
parents:
diff changeset
31 mr.readers = mr.readers[1:]
kono
parents:
diff changeset
32 }
kono
parents:
diff changeset
33 if n > 0 || err != EOF {
kono
parents:
diff changeset
34 if err == EOF && len(mr.readers) > 0 {
kono
parents:
diff changeset
35 // Don't return EOF yet. More readers remain.
kono
parents:
diff changeset
36 err = nil
kono
parents:
diff changeset
37 }
kono
parents:
diff changeset
38 return
kono
parents:
diff changeset
39 }
kono
parents:
diff changeset
40 }
kono
parents:
diff changeset
41 return 0, EOF
kono
parents:
diff changeset
42 }
kono
parents:
diff changeset
43
kono
parents:
diff changeset
44 // MultiReader returns a Reader that's the logical concatenation of
kono
parents:
diff changeset
45 // the provided input readers. They're read sequentially. Once all
kono
parents:
diff changeset
46 // inputs have returned EOF, Read will return EOF. If any of the readers
kono
parents:
diff changeset
47 // return a non-nil, non-EOF error, Read will return that error.
kono
parents:
diff changeset
48 func MultiReader(readers ...Reader) Reader {
kono
parents:
diff changeset
49 r := make([]Reader, len(readers))
kono
parents:
diff changeset
50 copy(r, readers)
kono
parents:
diff changeset
51 return &multiReader{r}
kono
parents:
diff changeset
52 }
kono
parents:
diff changeset
53
kono
parents:
diff changeset
54 type multiWriter struct {
kono
parents:
diff changeset
55 writers []Writer
kono
parents:
diff changeset
56 }
kono
parents:
diff changeset
57
kono
parents:
diff changeset
58 func (t *multiWriter) Write(p []byte) (n int, err error) {
kono
parents:
diff changeset
59 for _, w := range t.writers {
kono
parents:
diff changeset
60 n, err = w.Write(p)
kono
parents:
diff changeset
61 if err != nil {
kono
parents:
diff changeset
62 return
kono
parents:
diff changeset
63 }
kono
parents:
diff changeset
64 if n != len(p) {
kono
parents:
diff changeset
65 err = ErrShortWrite
kono
parents:
diff changeset
66 return
kono
parents:
diff changeset
67 }
kono
parents:
diff changeset
68 }
kono
parents:
diff changeset
69 return len(p), nil
kono
parents:
diff changeset
70 }
kono
parents:
diff changeset
71
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
72 var _ StringWriter = (*multiWriter)(nil)
111
kono
parents:
diff changeset
73
kono
parents:
diff changeset
74 func (t *multiWriter) WriteString(s string) (n int, err error) {
kono
parents:
diff changeset
75 var p []byte // lazily initialized if/when needed
kono
parents:
diff changeset
76 for _, w := range t.writers {
145
1830386684a0 gcc-9.2.0
anatofuz
parents: 131
diff changeset
77 if sw, ok := w.(StringWriter); ok {
111
kono
parents:
diff changeset
78 n, err = sw.WriteString(s)
kono
parents:
diff changeset
79 } else {
kono
parents:
diff changeset
80 if p == nil {
kono
parents:
diff changeset
81 p = []byte(s)
kono
parents:
diff changeset
82 }
kono
parents:
diff changeset
83 n, err = w.Write(p)
kono
parents:
diff changeset
84 }
kono
parents:
diff changeset
85 if err != nil {
kono
parents:
diff changeset
86 return
kono
parents:
diff changeset
87 }
kono
parents:
diff changeset
88 if n != len(s) {
kono
parents:
diff changeset
89 err = ErrShortWrite
kono
parents:
diff changeset
90 return
kono
parents:
diff changeset
91 }
kono
parents:
diff changeset
92 }
kono
parents:
diff changeset
93 return len(s), nil
kono
parents:
diff changeset
94 }
kono
parents:
diff changeset
95
kono
parents:
diff changeset
96 // MultiWriter creates a writer that duplicates its writes to all the
kono
parents:
diff changeset
97 // provided writers, similar to the Unix tee(1) command.
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
98 //
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
99 // Each write is written to each listed writer, one at a time.
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
100 // If a listed writer returns an error, that overall write operation
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
101 // stops and returns the error; it does not continue down the list.
111
kono
parents:
diff changeset
102 func MultiWriter(writers ...Writer) Writer {
131
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
103 allWriters := make([]Writer, 0, len(writers))
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
104 for _, w := range writers {
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
105 if mw, ok := w.(*multiWriter); ok {
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
106 allWriters = append(allWriters, mw.writers...)
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
107 } else {
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
108 allWriters = append(allWriters, w)
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
109 }
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
110 }
84e7813d76e9 gcc-8.2
mir3636
parents: 111
diff changeset
111 return &multiWriter{allWriters}
111
kono
parents:
diff changeset
112 }