-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmultiwriter.go
77 lines (66 loc) · 1.32 KB
/
multiwriter.go
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
package multiwriter
import (
"io"
"sync"
)
type MultiWriter struct {
sync.RWMutex
writers []io.Writer
}
// Append writer
func (self *MultiWriter) Append(writers ...io.Writer) {
self.Lock()
defer self.Unlock()
self.writers = append(self.writers, writers...)
}
// Remove writer
func (self *MultiWriter) Remove(writers ...io.Writer) {
self.Lock()
defer self.Unlock()
for i := len(self.writers) - 1; i > 0; i-- {
for _, v := range writers {
if self.writers[i] == v {
self.writers = append(self.writers[:i], self.writers[i+1:]...)
break
}
}
}
}
// Len of writers
func (self *MultiWriter) Len() int {
self.RLock()
defer self.RUnlock()
return len(self.writers)
}
// Write implements io.Writer
func (self *MultiWriter) Write(p []byte) (n int, err error) {
self.Lock()
defer self.Unlock()
type result struct {
n int
err error
}
rs := make(chan *result)
for _, w := range self.writers {
go func(writer io.Writer) {
n, err := writer.Write(p)
rs <- &result{n, err}
}(w)
}
for range self.writers {
r := <-rs
if r.err != nil {
return r.n, r.err
}
if r.n != len(p) {
return r.n, io.ErrShortWrite
}
}
return len(p), nil
}
// New return a MultiWriter
func New(writers ...io.Writer) io.Writer {
w := make([]io.Writer, len(writers))
copy(w, writers)
return &MultiWriter{writers: w}
}