-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy patharrayreader_js.go
113 lines (97 loc) · 2.86 KB
/
arrayreader_js.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
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
112
113
package wasmws
import (
"errors"
"io"
"sync"
"syscall/js"
)
//arrayReader is an io.ReadCloser implementation for Javascript ArrayBuffers
// See: https://developer.mozilla.org/en-US/docs/Web/API/Body/arrayBuffer
type arrayReader struct {
jsPromise js.Value
remaining []byte
read bool
err error
}
var arrayReaderPool = sync.Pool{
New: func() interface{} {
return new(arrayReader)
},
}
//newReaderArrayPromise returns a arrayReader from a JavaScript promise for
// an array buffer: See https://developer.mozilla.org/en-US/docs/Web/API/Blob/arrayBuffer
func newReaderArrayPromise(arrayPromise js.Value) *arrayReader {
ar := arrayReaderPool.Get().(*arrayReader)
ar.jsPromise = arrayPromise
return ar
}
//newReaderArrayPromise returns a arrayReader from a JavaScript array buffer:
// See: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/ArrayBuffer
func newReaderArrayBuffer(arrayBuffer js.Value) (*arrayReader, int) {
ar := arrayReaderPool.Get().(*arrayReader)
ar.remaining, ar.read = ar.fromArray(arrayBuffer), true
return ar, len(ar.remaining)
}
//Close closes the arrayReader and returns it to a pool. DO NOT USE FURTHER!
func (ar *arrayReader) Close() error {
ar.Reset()
arrayReaderPool.Put(ar)
return nil
}
//Reset makes this arrayReader ready for reuse
func (ar *arrayReader) Reset() {
const bufMax = socketStreamThresholdBytes
ar.jsPromise, ar.read, ar.err = js.Value{}, false, nil
if cap(ar.remaining) < bufMax {
ar.remaining = ar.remaining[:0]
} else {
ar.remaining = nil
}
}
//Read implements the standard io.Reader interface
func (ar *arrayReader) Read(buf []byte) (n int, err error) {
if ar.err != nil {
return 0, ar.err
}
if !ar.read {
ar.read = true
readCh, errCh := make(chan []byte, 1), make(chan error, 1)
successCallback := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
readCh <- ar.fromArray(args[0])
return nil
})
defer successCallback.Release()
failureCallback := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
errCh <- errors.New(args[0].Get("message").String()) //Send TypeError
return nil
})
defer failureCallback.Release()
//Wait for callback
ar.jsPromise.Call("then", successCallback, failureCallback)
select {
case ar.remaining = <-readCh:
case err := <-errCh:
return 0, err
}
}
if len(ar.remaining) < 1 {
return 0, io.EOF
}
n = copy(buf, ar.remaining)
ar.remaining = ar.remaining[n:]
return n, nil
}
//fromArray is a helper that that copies a JavaScript ArrayBuffer into go-space
// and uses an existing go buffer if possible.
func (ar *arrayReader) fromArray(arrayBuffer js.Value) []byte {
jsBuf := uint8Array.New(arrayBuffer)
count := jsBuf.Get("byteLength").Int()
var goBuf []byte
if count <= cap(ar.remaining) {
goBuf = ar.remaining[:count]
} else {
goBuf = make([]byte, count)
}
js.CopyBytesToGo(goBuf, jsBuf)
return goBuf
}