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
|
package mc
import (
"context"
"io"
"io/ioutil"
"net/http"
)
type MeasurableConnection interface {
Start(context.Context) bool
Transferred() uint64
}
type LoadBearingDownload struct {
Path string
downloaded uint64
client *http.Client
}
func (lbc *LoadBearingDownload) Transferred() uint64 {
return lbc.downloaded
}
func (lbc *LoadBearingDownload) Start(ctx context.Context) bool {
lbc.downloaded = 0
lbc.client = &http.Client{}
get, err := lbc.client.Get(lbc.Path)
if err != nil {
return false
}
go doDownload(get, &lbc.downloaded, ctx)
return true
}
func doDownload(get *http.Response, count *uint64, ctx context.Context) {
for ctx.Err() == nil {
n, err := io.CopyN(ioutil.Discard, get.Body, 100)
if err != nil {
break
}
*count += uint64(n)
}
get.Body.Close()
}
type LoadBearingUpload struct {
Path string
uploaded uint64
client *http.Client
}
func (lbu *LoadBearingUpload) Transferred() uint64 {
return lbu.uploaded
}
type syntheticCountingReader struct {
n *uint64
ctx context.Context
}
func (s *syntheticCountingReader) Read(p []byte) (n int, err error) {
if s.ctx.Err() != nil {
return 0, io.EOF
}
err = nil
n = len(p)
*s.n += uint64(n)
return
}
func (lbu *LoadBearingUpload) Start(ctx context.Context) bool {
lbu.uploaded = 0
lbu.client = &http.Client{}
s := &syntheticCountingReader{n: &lbu.uploaded, ctx: ctx}
go func() {
resp, _ := lbu.client.Post(lbu.Path, "application/octet-stream", s)
resp.Body.Close()
}()
return true
}
|