forked from ipfs/go-ipfs-cmds
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathresponseemitter.go
83 lines (67 loc) · 1.91 KB
/
responseemitter.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
78
79
80
81
82
83
package cmds
import (
"errors"
"fmt"
"io"
)
var (
ErrClosedEmitter = errors.New("cmds: emit on closed emitter")
ErrClosingClosedEmitter = errors.New("cmds: closing closed emitter")
)
// Single can be used to signal to any ResponseEmitter that only one value will be emitted.
// This is important e.g. for the http.ResponseEmitter so it can set the HTTP headers appropriately.
type Single struct {
Value interface{}
}
func (s Single) String() string {
return fmt.Sprintf("Single{%v}", s.Value)
}
func (s Single) GoString() string {
return fmt.Sprintf("Single{%#v}", s.Value)
}
// EmitOnce is a helper that emits a value wrapped in Single, to signal that this will be the only value sent.
func EmitOnce(re ResponseEmitter, v interface{}) error {
return re.Emit(Single{v})
}
// ResponseEmitter encodes and sends the command code's output to the client.
// It is all a command can write to.
type ResponseEmitter interface {
// Close closes the underlying transport.
Close() error
// CloseWithError closes the underlying transport and makes subsequent read
// calls return the passed error.
CloseWithError(error) error
// SetLength sets the length of the output
// err is an interface{} so we don't have to manually convert to error.
SetLength(length uint64)
// Emit sends a value.
// If value is io.Reader we just copy that to the connection
// other values are marshalled.
Emit(value interface{}) error
}
// Copy sends all values received on res to re. If res is closed, it closes re.
func Copy(re ResponseEmitter, res Response) error {
re.SetLength(res.Length())
for {
v, err := res.Next()
if err != nil {
if err == io.EOF {
return re.Close()
}
return re.CloseWithError(err)
}
err = re.Emit(v)
if err != nil {
return err
}
}
}
func EmitChan(re ResponseEmitter, ch <-chan interface{}) error {
for v := range ch {
err := re.Emit(v)
if err != nil {
return err
}
}
return nil
}