File: multi.go

package info (click to toggle)
golang-github-bodgit-plumbing 1.3.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 152 kB
  • sloc: makefile: 2
file content (111 lines) | stat: -rw-r--r-- 2,455 bytes parent folder | download
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
package plumbing

import (
	"io"
)

type multiWriteCloser struct {
	writeClosers []io.WriteCloser
}

func (t *multiWriteCloser) Write(p []byte) (n int, err error) {
	for _, wc := range t.writeClosers {
		n, err = wc.Write(p)
		if err != nil {
			return
		}

		if n != len(p) {
			err = io.ErrShortWrite

			return
		}
	}

	return len(p), nil
}

func (t *multiWriteCloser) Close() (err error) {
	for _, wc := range t.writeClosers {
		err = wc.Close()
		if err != nil {
			return
		}
	}

	return
}

// MultiWriteCloser creates a writer that duplicates its writes to all the
// provided writers, similar to the Unix tee(1) command.
//
// 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.
func MultiWriteCloser(writeClosers ...io.WriteCloser) io.WriteCloser {
	allWriteClosers := make([]io.WriteCloser, 0, len(writeClosers))

	for _, wc := range writeClosers {
		if mwc, ok := wc.(*multiWriteCloser); ok {
			allWriteClosers = append(allWriteClosers, mwc.writeClosers...)
		} else {
			allWriteClosers = append(allWriteClosers, wc)
		}
	}

	return &multiWriteCloser{allWriteClosers}
}

type multiReadCloser struct {
	readClosers []io.ReadCloser
	i           int
}

func (mrc *multiReadCloser) Read(p []byte) (n int, err error) {
	for mrc.i < len(mrc.readClosers) {
		if len(mrc.readClosers) == 1 {
			if rc, ok := mrc.readClosers[0].(*multiReadCloser); ok {
				mrc.readClosers = rc.readClosers

				continue
			}
		}

		n, err = mrc.readClosers[mrc.i].Read(p)
		if err == io.EOF { //nolint:errorlint
			mrc.i++
		}

		if n > 0 || err != io.EOF { //nolint:errorlint
			if err == io.EOF && mrc.i < len(mrc.readClosers) { //nolint:errorlint
				err = nil
			}

			return
		}
	}

	return 0, io.EOF
}

func (mrc *multiReadCloser) Close() (err error) {
	for _, rc := range mrc.readClosers {
		err = rc.Close()
		if err != nil {
			return
		}
	}

	return
}

// MultiReadCloser returns an io.ReadCloser that's the logical concatenation
// of the provider input readers. They're read sequentially. Once all inputs
// have returned io.EOF, Read will return EOF. If any of the readers return
// a non-nil, non-EOF error, Read will return that error.
func MultiReadCloser(readClosers ...io.ReadCloser) io.ReadCloser {
	rc := make([]io.ReadCloser, len(readClosers))
	copy(rc, readClosers)

	return &multiReadCloser{rc, 0}
}