/
external.go
154 lines (131 loc) · 3.51 KB
/
external.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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
package jsonschema
import (
"context"
"io"
"net/http"
"net/url"
"os"
"time"
"github.com/go-faster/errors"
"go.uber.org/zap"
"github.com/ogen-go/ogen/internal/urlpath"
)
// ExternalResolver resolves external links.
type ExternalResolver interface {
Get(ctx context.Context, loc string) ([]byte, error)
}
var _ ExternalResolver = NoExternal{}
// NoExternal is ExternalResolver that always returns error.
type NoExternal struct{}
// Get implements ExternalResolver.
func (n NoExternal) Get(context.Context, string) ([]byte, error) {
return nil, errors.New("external references are disabled")
}
// ExternalOptions is external reference resolver options.
type ExternalOptions struct {
// HTTPClient sets http client to use. Defaults to http.DefaultClient.
HTTPClient *http.Client
// ReadFile sets function for reading files from fs. Defaults to os.ReadFile.
ReadFile func(p string) ([]byte, error)
// URLToFilePath sets function for converting url to file path. Defaults to urlpath.URLToFilePath.
URLToFilePath func(u *url.URL) (string, error)
// Logger sets logger to use. Defaults to zap.NewNop().
Logger *zap.Logger
}
func (r *ExternalOptions) setDefaults() {
if r.HTTPClient == nil {
r.HTTPClient = http.DefaultClient
}
if r.ReadFile == nil {
r.ReadFile = os.ReadFile
}
if r.URLToFilePath == nil {
r.URLToFilePath = urlpath.URLToFilePath
}
if r.Logger == nil {
r.Logger = zap.NewNop()
}
}
var _ ExternalResolver = externalResolver{}
type externalResolver struct {
client *http.Client
readFile func(p string) ([]byte, error)
urlToFilePath func(u *url.URL) (string, error)
logger *zap.Logger
}
// NewExternalResolver creates new ExternalResolver.
//
// Currently only http(s) and file schemes are supported.
func NewExternalResolver(opts ExternalOptions) ExternalResolver {
opts.setDefaults()
return externalResolver{
client: opts.HTTPClient,
readFile: opts.ReadFile,
urlToFilePath: opts.URLToFilePath,
logger: opts.Logger,
}
}
func (e externalResolver) httpGet(ctx context.Context, u *url.URL) ([]byte, error) {
req, err := http.NewRequestWithContext(ctx, http.MethodGet, u.String(), nil)
if err != nil {
return nil, errors.Wrap(err, "create request")
}
if pass, ok := u.User.Password(); ok && u.User != nil {
req.SetBasicAuth(u.User.Username(), pass)
}
start := time.Now()
resp, err := e.client.Do(req)
if err != nil {
return nil, errors.Wrap(err, "do")
}
defer func() {
if resp.Body != nil {
_ = resp.Body.Close()
}
}()
e.logger.Debug("Get",
zap.String("url", u.Redacted()),
zap.Int("status", resp.StatusCode),
zap.Duration("duration", time.Since(start)),
)
if code := resp.StatusCode; code >= 299 {
text := http.StatusText(code)
return nil, errors.Errorf("bad HTTP code %d (%s)", code, text)
}
data, err := io.ReadAll(resp.Body)
if err != nil {
return nil, errors.Wrap(err, "read data")
}
return data, nil
}
func (e externalResolver) Get(ctx context.Context, loc string) ([]byte, error) {
u, err := url.Parse(loc)
if err != nil {
return nil, err
}
var (
data []byte
scheme = u.Scheme
)
switch scheme {
case "http", "https":
data, err = e.httpGet(ctx, u)
case "file", "":
var p string
p, err = e.urlToFilePath(u)
if err != nil {
err = errors.Wrap(err, "convert url to file path")
break
}
data, err = e.readFile(p)
default:
return nil, errors.Errorf("unsupported scheme %q", scheme)
}
if err != nil {
if scheme == "" {
scheme = "file"
}
return nil, errors.Wrap(err, scheme)
}
return data, nil
}