Skip to content

Commit f45ff41

Browse files
committedMay 8, 2023
init
1 parent b53505d commit f45ff41

File tree

9 files changed

+332
-0
lines changed

9 files changed

+332
-0
lines changed
 

‎.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
.idea

‎README.md

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
# HTTP请求工具
2+
3+
4+
# TODO
5+
- [x] 方便的做响应格式处理
6+
- [ ] 兼容HTTP2
7+
- [ ] 兼容JA3指纹

‎api.go

+92
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
package requests
2+
3+
import (
4+
"context"
5+
"encoding/json"
6+
"fmt"
7+
"net/http"
8+
"reflect"
9+
)
10+
11+
// ------------------------------------------------- --------------------------------------------------------------------
12+
13+
// GetYaml 响应内容是YAML格式的
14+
func GetYaml[Response any](ctx context.Context, targetUrl string, options ...*Options[any, Response]) (Response, error) {
15+
16+
if len(options) == 0 {
17+
options = append(options, NewOptions[any, Response](targetUrl, YamlResponseHandler[Response]()))
18+
}
19+
20+
options[0] = options[0].WithTargetURL(targetUrl).WithYamlResponseHandler()
21+
22+
return SendRequest[any, Response](ctx, options[0])
23+
}
24+
25+
// ------------------------------------------------- --------------------------------------------------------------------
26+
27+
// GetJson 响应内容是JSON格式的
28+
func GetJson[Response any](ctx context.Context, targetUrl string, options ...*Options[any, Response]) (Response, error) {
29+
30+
if len(options) == 0 {
31+
options = append(options, NewOptions[any, Response](targetUrl, JsonResponseHandler[Response]()))
32+
}
33+
34+
options[0] = options[0].WithTargetURL(targetUrl).
35+
WithJsonResponseHandler().
36+
AppendRequestSetting(func(httpRequest *http.Request) error {
37+
httpRequest.Header.Set("Content-Type", "application/json")
38+
return nil
39+
})
40+
41+
return SendRequest[any, Response](ctx, options[0])
42+
}
43+
44+
func PostJson[Request any, Response any](ctx context.Context, targetUrl string, request Request, options ...*Options[Request, Response]) (Response, error) {
45+
46+
if len(options) == 0 {
47+
options = append(options, NewOptions[Request, Response](targetUrl, JsonResponseHandler[Response]()))
48+
}
49+
50+
marshal, err := json.Marshal(request)
51+
if err != nil {
52+
var zero Response
53+
return zero, fmt.Errorf("PostJson json marshal request error: %s, typer = %s", err.Error(), reflect.TypeOf(request).String())
54+
}
55+
56+
options[0] = options[0].WithTargetURL(targetUrl).
57+
WithJsonResponseHandler().
58+
AppendRequestSetting(func(httpRequest *http.Request) error {
59+
httpRequest.Header.Set("Content-Type", "application/json")
60+
return nil
61+
}).
62+
WithBody(marshal)
63+
64+
return SendRequest[Request, Response](ctx, options[0])
65+
}
66+
67+
// ------------------------------------------------- --------------------------------------------------------------------
68+
69+
func GetBytes(ctx context.Context, targetUrl string, options ...*Options[any, []byte]) ([]byte, error) {
70+
71+
if len(options) == 0 {
72+
options = append(options, NewOptions[any, []byte](targetUrl, BytesResponseHandler()))
73+
}
74+
75+
options[0] = options[0].WithTargetURL(targetUrl).
76+
WithResponseHandler(BytesResponseHandler())
77+
78+
return SendRequest[any, []byte](ctx, options[0])
79+
}
80+
81+
// ------------------------------------------------- --------------------------------------------------------------------
82+
83+
func GetString(ctx context.Context, targetUrl string, options ...*Options[any, []byte]) (string, error) {
84+
responseBytes, err := GetBytes(ctx, targetUrl, options...)
85+
if err != nil {
86+
return "", err
87+
}
88+
return string(responseBytes), nil
89+
}
90+
91+
// ------------------------------------------------- --------------------------------------------------------------------
92+

‎example/request-text/main.go

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
package main
2+
3+
func main() {
4+
5+
}

‎go.mod

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
module github.com/crawler-go-go-go/go-requests
2+
3+
go 1.18
4+
5+
require gopkg.in/yaml.v2 v2.4.0

‎go.sum

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
2+
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
3+
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
4+
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=

‎http.go

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
package requests
2+
3+
import (
4+
"bytes"
5+
"context"
6+
"net/http"
7+
)
8+
9+
// DefaultMaxTryTimes 默认情况下的最大重试次数
10+
const DefaultMaxTryTimes = 3
11+
12+
type RequestSetting func(httpRequest *http.Request) error
13+
14+
const DefaultUserAgent = "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/112.0.0.0 Safari/537.36"
15+
16+
func DefaultUserAgentRequestSetting() RequestSetting {
17+
return func(httpRequest *http.Request) error {
18+
httpRequest.Header.Set("User-Agent", DefaultUserAgent)
19+
return nil
20+
}
21+
}
22+
23+
// ------------------------------------------------- --------------------------------------------------------------------
24+
25+
func SendRequest[Request any, Response any](ctx context.Context, options *Options[Request, Response]) (Response, error) {
26+
27+
// TODO set default params
28+
29+
var lastErr error
30+
for tryTimes := 0; tryTimes < options.MaxTryTimes; tryTimes++ {
31+
var client http.Client
32+
httpRequest, err := http.NewRequest(options.Method, options.TargetURL, bytes.NewReader(options.Body))
33+
if err != nil {
34+
lastErr = err
35+
continue
36+
}
37+
38+
httpRequest = httpRequest.WithContext(ctx)
39+
40+
for _, requestSettingFunc := range options.RequestSettingSlice {
41+
if err := requestSettingFunc(httpRequest); err != nil {
42+
lastErr = err
43+
continue
44+
}
45+
}
46+
47+
httpResponse, err := client.Do(httpRequest)
48+
if err != nil {
49+
lastErr = err
50+
continue
51+
}
52+
defer httpResponse.Body.Close()
53+
54+
response, err := options.ResponseHandler(httpResponse)
55+
if err != nil {
56+
lastErr = err
57+
continue
58+
}
59+
return response, nil
60+
}
61+
62+
var zero Response
63+
return zero, lastErr
64+
}

‎options.go

+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
package requests
2+
3+
import "net/http"
4+
5+
const DefaultMethod = http.MethodGet
6+
7+
type Options[Request any, Response any] struct {
8+
MaxTryTimes int
9+
TargetURL string
10+
Method string
11+
Body []byte
12+
RequestSettingSlice []RequestSetting
13+
ResponseHandler ResponseHandler[Response]
14+
}
15+
16+
func NewOptions[Request any, Response any](targetUrl string, responseHandler ResponseHandler[Response]) *Options[Request, Response] {
17+
return &Options[Request, Response]{
18+
MaxTryTimes: DefaultMaxTryTimes,
19+
TargetURL: targetUrl,
20+
Method: DefaultMethod,
21+
Body: []byte{},
22+
RequestSettingSlice: nil,
23+
ResponseHandler: responseHandler,
24+
}
25+
}
26+
27+
func (x *Options[Request, Response]) WithMaxTryTimes(maxTryTimes int) *Options[Request, Response] {
28+
x.MaxTryTimes = maxTryTimes
29+
return x
30+
}
31+
32+
func (x *Options[Request, Response]) WithTargetURL(targetURL string) *Options[Request, Response] {
33+
x.TargetURL = targetURL
34+
return x
35+
}
36+
37+
func (x *Options[Request, Response]) WithMethod(method string) *Options[Request, Response] {
38+
x.Method = method
39+
return x
40+
}
41+
42+
func (x *Options[Request, Response]) WithBody(body []byte) *Options[Request, Response] {
43+
if x.Method == DefaultMethod {
44+
x.Method = http.MethodPost
45+
}
46+
x.Body = body
47+
return x
48+
}
49+
50+
func (x *Options[Request, Response]) WithRequestSettingSlice(requestSettingSlice []RequestSetting) *Options[Request, Response] {
51+
x.RequestSettingSlice = requestSettingSlice
52+
return x
53+
}
54+
55+
func (x *Options[Request, Response]) AppendRequestSetting(requestSetting RequestSetting) *Options[Request, Response] {
56+
x.RequestSettingSlice = append(x.RequestSettingSlice, requestSetting)
57+
return x
58+
}
59+
60+
func (x *Options[Request, Response]) WithResponseHandler(responseHandler ResponseHandler[Response]) *Options[Request, Response] {
61+
x.ResponseHandler = responseHandler
62+
return x
63+
}
64+
65+
func (x *Options[Request, Response]) WithYamlResponseHandler() *Options[Request, Response] {
66+
x.ResponseHandler = YamlResponseHandler[Response]()
67+
return x
68+
}
69+
70+
func (x *Options[Request, Response]) WithJsonResponseHandler() *Options[Request, Response] {
71+
x.ResponseHandler = JsonResponseHandler[Response]()
72+
return x
73+
}
74+
75+

‎response_handler.go

+79
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
package requests
2+
3+
import (
4+
"encoding/json"
5+
"fmt"
6+
"gopkg.in/yaml.v2"
7+
"io"
8+
"net/http"
9+
"reflect"
10+
)
11+
12+
type ResponseHandler[Response any] func(httpResponse *http.Response) (Response, error)
13+
14+
func BytesResponseHandler(readResponseOnStatusCodeIn ...int) ResponseHandler[[]byte] {
15+
16+
// By default, the response body is read only when the status code is 200
17+
if len(readResponseOnStatusCodeIn) == 0 {
18+
readResponseOnStatusCodeIn = append(readResponseOnStatusCodeIn, http.StatusOK)
19+
}
20+
21+
return func(httpResponse *http.Response) ([]byte, error) {
22+
for _, status := range readResponseOnStatusCodeIn {
23+
if status == httpResponse.StatusCode {
24+
responseBodyBytes, err := io.ReadAll(httpResponse.Body)
25+
if err != nil {
26+
return nil, fmt.Errorf("response statuc code: %d, read body error: %s", httpResponse.StatusCode, err.Error())
27+
}
28+
return responseBodyBytes, nil
29+
}
30+
}
31+
return nil, fmt.Errorf("response status code: %d", httpResponse.StatusCode)
32+
}
33+
}
34+
35+
func StringResponseHandler(readResponseOnStatusCodeIn ...int) ResponseHandler[string] {
36+
return func(httpResponse *http.Response) (string, error) {
37+
responseBytes, err := BytesResponseHandler(readResponseOnStatusCodeIn...)(httpResponse)
38+
if err != nil {
39+
return "", err
40+
}
41+
return string(responseBytes), nil
42+
}
43+
}
44+
45+
func YamlResponseHandler[Response any](readResponseOnStatusCodeIn ...int) ResponseHandler[Response] {
46+
return func(httpResponse *http.Response) (Response, error) {
47+
48+
var r Response
49+
50+
responseBytes, err := BytesResponseHandler(readResponseOnStatusCodeIn...)(httpResponse)
51+
if err != nil {
52+
return r, err
53+
}
54+
55+
err = yaml.Unmarshal(responseBytes, &r)
56+
if err != nil {
57+
return r, fmt.Errorf("response body yaml unmarshal error: %s, type: %s, response body: %s", err.Error(), reflect.TypeOf(r).String(), string(responseBytes))
58+
}
59+
return r, nil
60+
}
61+
}
62+
63+
func JsonResponseHandler[Response any](readResponseOnStatusCodeIn ...int) ResponseHandler[Response] {
64+
return func(httpResponse *http.Response) (Response, error) {
65+
66+
var r Response
67+
68+
responseBytes, err := BytesResponseHandler(readResponseOnStatusCodeIn...)(httpResponse)
69+
if err != nil {
70+
return r, err
71+
}
72+
73+
err = json.Unmarshal(responseBytes, &r)
74+
if err != nil {
75+
return r, fmt.Errorf("response body json unmarshal error: %s, type: %s, response body: %s", err.Error(), reflect.TypeOf(r).String(), string(responseBytes))
76+
}
77+
return r, nil
78+
}
79+
}

0 commit comments

Comments
 (0)
Please sign in to comment.