2010-12-03 05:34:57 +01:00
|
|
|
// Copyright 2010 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package io
|
|
|
|
|
2017-01-14 01:05:42 +01:00
|
|
|
type eofReader struct{}
|
|
|
|
|
|
|
|
func (eofReader) Read([]byte) (int, error) {
|
|
|
|
return 0, EOF
|
|
|
|
}
|
|
|
|
|
2010-12-03 05:34:57 +01:00
|
|
|
type multiReader struct {
|
|
|
|
readers []Reader
|
|
|
|
}
|
|
|
|
|
2011-12-03 03:17:34 +01:00
|
|
|
func (mr *multiReader) Read(p []byte) (n int, err error) {
|
2010-12-03 05:34:57 +01:00
|
|
|
for len(mr.readers) > 0 {
|
2017-01-14 01:05:42 +01:00
|
|
|
// Optimization to flatten nested multiReaders (Issue 13558).
|
2016-07-22 20:15:38 +02:00
|
|
|
if len(mr.readers) == 1 {
|
|
|
|
if r, ok := mr.readers[0].(*multiReader); ok {
|
|
|
|
mr.readers = r.readers
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
2010-12-03 05:34:57 +01:00
|
|
|
n, err = mr.readers[0].Read(p)
|
2016-09-10 15:14:00 +02:00
|
|
|
if err == EOF {
|
2017-01-14 01:05:42 +01:00
|
|
|
// Use eofReader instead of nil to avoid nil panic
|
|
|
|
// after performing flatten (Issue 18232).
|
|
|
|
mr.readers[0] = eofReader{} // permit earlier GC
|
2016-09-10 15:14:00 +02:00
|
|
|
mr.readers = mr.readers[1:]
|
|
|
|
}
|
2011-12-03 03:17:34 +01:00
|
|
|
if n > 0 || err != EOF {
|
2016-09-10 15:14:00 +02:00
|
|
|
if err == EOF && len(mr.readers) > 0 {
|
|
|
|
// Don't return EOF yet. More readers remain.
|
2010-12-03 05:34:57 +01:00
|
|
|
err = nil
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2011-12-03 03:17:34 +01:00
|
|
|
return 0, EOF
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// MultiReader returns a Reader that's the logical concatenation of
|
2016-07-22 20:15:38 +02:00
|
|
|
// the provided input readers. They're read sequentially. Once all
|
2014-06-07 00:37:27 +02:00
|
|
|
// inputs have returned EOF, Read will return EOF. If any of the readers
|
|
|
|
// return a non-nil, non-EOF error, Read will return that error.
|
2010-12-03 05:34:57 +01:00
|
|
|
func MultiReader(readers ...Reader) Reader {
|
2014-07-19 10:53:52 +02:00
|
|
|
r := make([]Reader, len(readers))
|
|
|
|
copy(r, readers)
|
|
|
|
return &multiReader{r}
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type multiWriter struct {
|
|
|
|
writers []Writer
|
|
|
|
}
|
|
|
|
|
2011-12-03 03:17:34 +01:00
|
|
|
func (t *multiWriter) Write(p []byte) (n int, err error) {
|
2010-12-03 05:34:57 +01:00
|
|
|
for _, w := range t.writers {
|
|
|
|
n, err = w.Write(p)
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if n != len(p) {
|
|
|
|
err = ErrShortWrite
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return len(p), nil
|
|
|
|
}
|
|
|
|
|
2019-01-18 20:04:36 +01:00
|
|
|
var _ StringWriter = (*multiWriter)(nil)
|
2016-02-03 22:58:02 +01:00
|
|
|
|
|
|
|
func (t *multiWriter) WriteString(s string) (n int, err error) {
|
|
|
|
var p []byte // lazily initialized if/when needed
|
|
|
|
for _, w := range t.writers {
|
2019-01-18 20:04:36 +01:00
|
|
|
if sw, ok := w.(StringWriter); ok {
|
2016-02-03 22:58:02 +01:00
|
|
|
n, err = sw.WriteString(s)
|
|
|
|
} else {
|
|
|
|
if p == nil {
|
|
|
|
p = []byte(s)
|
|
|
|
}
|
|
|
|
n, err = w.Write(p)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if n != len(s) {
|
|
|
|
err = ErrShortWrite
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return len(s), nil
|
|
|
|
}
|
|
|
|
|
2010-12-03 05:34:57 +01:00
|
|
|
// MultiWriter creates a writer that duplicates its writes to all the
|
|
|
|
// provided writers, similar to the Unix tee(1) command.
|
2018-01-09 02:23:08 +01:00
|
|
|
//
|
|
|
|
// Each write is written to each listed writer, one at a time.
|
|
|
|
// If a listed writer returns an error, that overall write operation
|
|
|
|
// stops and returns the error; it does not continue down the list.
|
2010-12-03 05:34:57 +01:00
|
|
|
func MultiWriter(writers ...Writer) Writer {
|
2018-01-09 02:23:08 +01:00
|
|
|
allWriters := make([]Writer, 0, len(writers))
|
|
|
|
for _, w := range writers {
|
|
|
|
if mw, ok := w.(*multiWriter); ok {
|
|
|
|
allWriters = append(allWriters, mw.writers...)
|
|
|
|
} else {
|
|
|
|
allWriters = append(allWriters, w)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return &multiWriter{allWriters}
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|