summaryrefslogtreecommitdiff
path: root/mc/mc.go
blob: 0b1d69bce2f23dcc94ddc5d60e9c88ffe730b2e4 (plain)
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
}