-
Notifications
You must be signed in to change notification settings - Fork 352
/
range.go
79 lines (73 loc) · 2.07 KB
/
range.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
package httputil
import (
"fmt"
"strconv"
"strings"
)
var ErrBadRange = fmt.Errorf("unsatisfiable range")
// Range represents an RFC 2616 HTTP Range
type Range struct {
StartOffset int64
EndOffset int64
}
func (r Range) String() string {
return fmt.Sprintf("start=%d, end=%d (total=%d)", r.StartOffset, r.EndOffset, r.EndOffset-r.StartOffset+1)
}
// ParseRange parses an HTTP RFC 2616 Range header value and returns an Range object for the given object length
func ParseRange(spec string, length int64) (Range, error) {
// Amazon S3 doesn't support retrieving multiple ranges of data per GET request.
var r Range
if !strings.HasPrefix(spec, "bytes=") {
return r, ErrBadRange
}
spec = strings.TrimPrefix(spec, "bytes=")
parts := strings.Split(spec, "-")
const rangeParts = 2
if len(parts) != rangeParts {
return r, ErrBadRange
}
fromString := parts[0]
toString := parts[1]
if len(fromString) == 0 && len(toString) == 0 {
return r, ErrBadRange
}
// negative only
if len(fromString) == 0 {
endOffset, err := strconv.ParseInt(toString, 10, 64) //nolint: gomnd
if err != nil || endOffset > length {
return r, ErrBadRange
}
r.StartOffset = length - endOffset
r.EndOffset = length - 1
return r, nil
}
// positive only
if len(toString) == 0 {
beginOffset, err := strconv.ParseInt(fromString, 10, 64) //nolint: gomnd
if err != nil || beginOffset > length-1 {
return r, ErrBadRange
}
r.StartOffset = beginOffset
r.EndOffset = length - 1
return r, nil
}
// both set
beginOffset, err := strconv.ParseInt(fromString, 10, 64) //nolint: gomnd
if err != nil {
return r, ErrBadRange
}
endOffset, err := strconv.ParseInt(toString, 10, 64) //nolint: gomnd
if err != nil {
return r, ErrBadRange
}
// if endOffset exceeds length return length : this is how it works in s3 (presto for example uses range with a huge endOffset regardless to the file size)
if endOffset > length-1 {
endOffset = length - 1
}
if beginOffset > length-1 || endOffset > length-1 {
return r, ErrBadRange
}
r.StartOffset = beginOffset
r.EndOffset = endOffset
return r, nil
}