-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathadapters.go
126 lines (106 loc) · 3.03 KB
/
adapters.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 requests
import (
"bytes"
"context"
"fmt"
"github.com/pkg/errors"
"github.com/cvelab/requests/ext"
"github.com/cvelab/requests/internal/decoder"
"github.com/cvelab/requests/internal/processbar"
"github.com/cvelab/requests/internal/tracer"
"github.com/cvelab/requests/types"
"io"
"log"
"net/http"
"net/http/httptrace"
nUrl "net/url"
"strconv"
"strings"
"time"
)
// 后续转为接口,需要引入其他adapter
type adapter struct {
context context.Context
tracerEnable bool
}
func (a *adapter) send(client *http.Client, prep *prepareRequest, hooks *types.HooksDict) *Response {
trace := prepareTracer(a.tracerEnable, fmt.Sprintf("%s >>>>> %s", prep.method, prep.url))
req := &Request{Request: &http.Request{Proto: prep.proto}}
a.context = httptrace.WithClientTrace(a.context, trace.ClientTrace)
req.Request = req.WithContext(a.context)
req.Method = prep.method
url, _ := nUrl.Parse(prep.url)
req.URL = url
if prep.headers != nil {
req.Header = *prep.headers
if host := req.Header.Get("Host"); host != "" {
req.Host = host
}
if lens := req.Header.Get("Content-Length"); lens != "" {
length, err := strconv.ParseInt(lens, 10, 64)
if err == nil {
req.ContentLength = length
}
}
if strings.ToLower(req.Header.Get("Transfer-Encoding")) == "chunked" {
req.ContentLength = 0
}
}
if prep.body != nil {
req.Body = *prep.body
}
requestHandle := ext.DisPatchHook("request", *hooks, *req).(Request)
clientHandle := ext.DisPatchHook("client", *hooks, *client).(http.Client)
if a.tracerEnable {
trace.StartTime = time.Now()
}
resp, err := clientHandle.Do(requestHandle.Request)
if a.tracerEnable {
trace.EndTime = time.Now()
log.Println(trace.ToString())
}
if resp == nil || err != nil {
log.Printf("%+v\n", errors.WithStack(err))
return nil
}
response := a.buildResponse(req.Request, resp)
if response == nil {
return nil
}
responseHandle := ext.DisPatchHook("response", *hooks, *response).(Response)
return &responseHandle
}
func (a *adapter) buildResponse(req *http.Request, resp *http.Response) *Response {
// 允许接入多个writer
buf := &bytes.Buffer{}
opts := a.context.Value("processOptions").([]processbar.Option)
pb := processbar.NewProcessBar(resp.ContentLength, opts...)
_, err := io.Copy(io.MultiWriter(buf, pb), resp.Body)
if err != nil {
log.Printf("%+v\n", errors.WithStack(err))
return nil
}
defer resp.Body.Close()
raw := buf.Bytes()
//raw, err := ioutil.ReadAll(resp.Body)
// 判断raw长度是否需要解码
encoding := resp.Header.Get("Content-Encoding")
if err = decoder.DecompressRaw(&raw, encoding); err != nil {
log.Printf("%+v\n", errors.WithStack(err))
return nil
}
r := &Response{
Ok: resp.StatusCode == 200,
Response: resp,
Raw: &raw,
Html: string(raw),
cookies: append(resp.Cookies(), req.Cookies()...),
}
return r
}
func prepareTracer(enable bool, prefix string) *tracer.Tracer {
newTracer := tracer.Tracer{Enable: enable}
newTracer.Output.Prefix = prefix
trace := newTracer.Init()
return trace
}