-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathsession.go
126 lines (111 loc) · 2.62 KB
/
session.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
114
115
116
117
118
119
120
121
122
123
124
125
126
package getparty
import (
"encoding/json"
"errors"
"fmt"
"log"
"os"
"strings"
"time"
"github.com/vbauerster/mpb/v8/decor"
)
// Session represents download session state
type Session struct {
restored bool
location string
URL string
OutputName string
ContentMD5 string
AcceptRanges string
ContentType string
StatusCode int
ContentLength int64
Redirected bool
Single bool
Elapsed time.Duration
HeaderMap map[string]string
Parts []*Part
}
func (s *Session) calcParts(parts uint) error {
if parts == 0 {
return ErrZeroParts
}
if !s.isResumable() {
parts = 1
}
fragment := s.ContentLength / int64(parts)
if parts != 1 && fragment < 64 {
return ErrTooFragmented
}
s.Parts = make([]*Part, parts)
s.Parts[0] = new(Part)
var stop int64
start := s.ContentLength
for i := parts - 1; i > 0; i-- {
stop = start - 1
start = stop - fragment
s.Parts[i] = &Part{
Start: start,
Stop: stop,
}
}
// if session isn't resumable stop is always negative
s.Parts[0].Stop = start - 1
s.Single = parts == 1
return nil
}
func (s *Session) loadState(name string) error {
f, err := os.Open(name)
if err != nil {
return withStack(err)
}
err = firstErr(json.NewDecoder(f).Decode(s), f.Close())
return withStack(err)
}
func (s *Session) dumpState(name string) error {
f, err := os.Create(name)
if err != nil {
return withStack(err)
}
err = firstErr(json.NewEncoder(f).Encode(s), f.Close())
return withStack(err)
}
func (s Session) isResumable() bool {
return strings.EqualFold(s.AcceptRanges, "bytes") && s.ContentLength >= 0
}
func (s Session) totalWritten() int64 {
var total int64
for _, p := range s.Parts {
total += p.Written
}
return total
}
func (s Session) summary(loggers [lEVELS]*log.Logger) {
format := fmt.Sprintf("Length: %%s [%s]", s.ContentType)
switch {
case s.isResumable():
summary := fmt.Sprintf("%d (%.1f)", s.ContentLength, decor.SizeB1024(s.ContentLength))
loggers[INFO].Printf(format, summary)
if tw := s.totalWritten(); tw != 0 {
remaining := s.ContentLength - tw
loggers[INFO].Printf("Remaining: %d (%.1f)", remaining, decor.SizeB1024(remaining))
}
case s.ContentLength < 0:
loggers[INFO].Printf(format, "unknown")
fallthrough
default:
message := "Session is not resumable"
loggers[WARN].Println(message)
loggers[DEBUG].Println(message)
}
}
func (s Session) isOutputFileExist() (bool, error) {
stat, err := os.Stat(s.OutputName)
if errors.Is(err, os.ErrNotExist) {
return false, nil
}
if err == nil && stat.IsDir() {
return true, fmt.Errorf("%q is a directory", s.OutputName)
}
return true, err
}