2018-05-14 16:26:38 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
2019-12-08 04:10:22 +00:00
|
|
|
"crypto/tls"
|
|
|
|
"net"
|
2018-05-14 16:26:38 +00:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2019-12-08 04:10:22 +00:00
|
|
|
"time"
|
2019-04-11 02:28:21 +00:00
|
|
|
|
|
|
|
"github.com/influxdata/influxdb/kit/tracing"
|
2019-12-08 04:10:22 +00:00
|
|
|
"github.com/influxdata/influxdb/pkg/httpc"
|
2018-05-14 16:26:38 +00:00
|
|
|
)
|
|
|
|
|
2019-12-08 04:10:22 +00:00
|
|
|
// NewHTTPClient creates a new httpc.Client type. This call sets all
|
|
|
|
// the options that are important to the http pkg on the httpc client.
|
|
|
|
// The default status fn and so forth will all be set for the caller.
|
2020-02-26 10:37:57 +00:00
|
|
|
// In addition, some options can be specified. Those will be added to the defaults.
|
|
|
|
func NewHTTPClient(addr, token string, insecureSkipVerify bool, opts ...httpc.ClientOptFn) (*httpc.Client, error) {
|
2019-12-08 04:10:22 +00:00
|
|
|
u, err := url.Parse(addr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2020-02-26 10:37:57 +00:00
|
|
|
defaultOpts := []httpc.ClientOptFn{
|
2019-12-08 04:10:22 +00:00
|
|
|
httpc.WithAddr(addr),
|
|
|
|
httpc.WithContentType("application/json"),
|
|
|
|
httpc.WithHTTPClient(NewClient(u.Scheme, insecureSkipVerify)),
|
|
|
|
httpc.WithInsecureSkipVerify(insecureSkipVerify),
|
|
|
|
httpc.WithStatusFn(CheckError),
|
|
|
|
}
|
|
|
|
if token != "" {
|
2020-02-26 10:37:57 +00:00
|
|
|
defaultOpts = append(defaultOpts, httpc.WithAuthToken(token))
|
2019-12-08 04:10:22 +00:00
|
|
|
}
|
2020-02-26 10:37:57 +00:00
|
|
|
opts = append(defaultOpts, opts...)
|
2019-12-08 04:10:22 +00:00
|
|
|
return httpc.New(opts...)
|
|
|
|
}
|
|
|
|
|
2018-09-17 02:39:46 +00:00
|
|
|
// Service connects to an InfluxDB via HTTP.
|
|
|
|
type Service struct {
|
|
|
|
Addr string
|
|
|
|
Token string
|
|
|
|
InsecureSkipVerify bool
|
|
|
|
|
|
|
|
*AuthorizationService
|
2020-01-21 22:22:45 +00:00
|
|
|
*BackupService
|
2019-12-07 18:54:03 +00:00
|
|
|
*BucketService
|
2020-02-24 19:41:21 +00:00
|
|
|
*TaskService
|
2019-12-07 18:54:03 +00:00
|
|
|
*DashboardService
|
2018-09-17 02:39:46 +00:00
|
|
|
*OrganizationService
|
|
|
|
*UserService
|
2019-02-14 20:32:54 +00:00
|
|
|
*VariableService
|
2019-12-07 18:54:03 +00:00
|
|
|
*WriteService
|
2020-02-25 15:22:33 +00:00
|
|
|
DocumentService
|
2020-02-26 16:34:10 +00:00
|
|
|
*CheckService
|
2020-02-26 10:37:57 +00:00
|
|
|
*NotificationEndpointService
|
|
|
|
*UserResourceMappingService
|
|
|
|
*TelegrafService
|
|
|
|
*LabelService
|
|
|
|
*SecretService
|
2018-09-17 02:39:46 +00:00
|
|
|
}
|
|
|
|
|
2020-02-26 10:37:57 +00:00
|
|
|
// NewService returns a service that is an HTTP client to a remote.
|
|
|
|
// Address and token are needed for those services that do not use httpc.Client,
|
|
|
|
// but use those for configuring.
|
|
|
|
// Usually one would do:
|
|
|
|
//
|
|
|
|
// ```
|
|
|
|
// c := NewHTTPClient(addr, token, insecureSkipVerify)
|
|
|
|
// s := NewService(c, addr token)
|
|
|
|
// ```
|
|
|
|
//
|
|
|
|
// So one should provide the same `addr` and `token` to both calls to ensure consistency
|
|
|
|
// in the behavior of the returned service.
|
|
|
|
func NewService(httpClient *httpc.Client, addr, token string) (*Service, error) {
|
2018-09-17 02:39:46 +00:00
|
|
|
return &Service{
|
2019-12-17 19:56:04 +00:00
|
|
|
Addr: addr,
|
|
|
|
Token: token,
|
|
|
|
AuthorizationService: &AuthorizationService{Client: httpClient},
|
2020-01-21 22:22:45 +00:00
|
|
|
BackupService: &BackupService{
|
|
|
|
Addr: addr,
|
|
|
|
Token: token,
|
|
|
|
},
|
|
|
|
BucketService: &BucketService{Client: httpClient},
|
2020-02-24 19:41:21 +00:00
|
|
|
TaskService: &TaskService{Client: httpClient},
|
2020-01-21 22:22:45 +00:00
|
|
|
DashboardService: &DashboardService{Client: httpClient},
|
|
|
|
OrganizationService: &OrganizationService{Client: httpClient},
|
|
|
|
UserService: &UserService{Client: httpClient},
|
|
|
|
VariableService: &VariableService{Client: httpClient},
|
2019-12-07 18:54:03 +00:00
|
|
|
WriteService: &WriteService{
|
2018-09-17 02:39:46 +00:00
|
|
|
Addr: addr,
|
|
|
|
Token: token,
|
|
|
|
},
|
2020-02-26 10:37:57 +00:00
|
|
|
DocumentService: NewDocumentService(httpClient),
|
|
|
|
CheckService: &CheckService{Client: httpClient},
|
|
|
|
NotificationEndpointService: &NotificationEndpointService{Client: httpClient},
|
|
|
|
UserResourceMappingService: &UserResourceMappingService{Client: httpClient},
|
|
|
|
TelegrafService: NewTelegrafService(httpClient),
|
|
|
|
LabelService: &LabelService{Client: httpClient},
|
|
|
|
SecretService: &SecretService{Client: httpClient},
|
2019-12-07 18:54:03 +00:00
|
|
|
}, nil
|
2018-09-17 02:39:46 +00:00
|
|
|
}
|
|
|
|
|
2019-06-26 05:48:15 +00:00
|
|
|
// NewURL concats addr and path.
|
2019-05-09 17:41:14 +00:00
|
|
|
func NewURL(addr, path string) (*url.URL, error) {
|
2018-05-14 16:26:38 +00:00
|
|
|
u, err := url.Parse(addr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
u.Path = path
|
|
|
|
return u, nil
|
|
|
|
}
|
|
|
|
|
2019-06-26 05:48:15 +00:00
|
|
|
// NewClient returns an http.Client that pools connections and injects a span.
|
2019-12-08 04:10:22 +00:00
|
|
|
func NewClient(scheme string, insecure bool) *http.Client {
|
|
|
|
return httpClient(scheme, insecure)
|
2018-05-14 16:26:38 +00:00
|
|
|
}
|
2018-08-15 20:14:51 +00:00
|
|
|
|
2019-12-08 04:10:22 +00:00
|
|
|
// SpanTransport injects the http.RoundTripper.RoundTrip() request
|
|
|
|
// with a span.
|
|
|
|
type SpanTransport struct {
|
|
|
|
base http.RoundTripper
|
2018-08-15 20:14:51 +00:00
|
|
|
}
|
|
|
|
|
2019-12-08 04:10:22 +00:00
|
|
|
// RoundTrip implements the http.RoundTripper, intercepting the base
|
|
|
|
// round trippers call and injecting a span.
|
|
|
|
func (s *SpanTransport) RoundTrip(r *http.Request) (*http.Response, error) {
|
2019-04-11 02:28:21 +00:00
|
|
|
span, _ := tracing.StartSpanFromContext(r.Context())
|
2019-06-26 05:48:15 +00:00
|
|
|
defer span.Finish()
|
2019-04-11 02:28:21 +00:00
|
|
|
tracing.InjectToHTTPRequest(span, r)
|
2019-12-08 04:10:22 +00:00
|
|
|
return s.base.RoundTrip(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
func httpClient(scheme string, insecure bool) *http.Client {
|
|
|
|
tr := &http.Transport{
|
|
|
|
Proxy: http.ProxyFromEnvironment,
|
|
|
|
DialContext: (&net.Dialer{
|
|
|
|
Timeout: 30 * time.Second,
|
|
|
|
KeepAlive: 30 * time.Second,
|
|
|
|
DualStack: true,
|
|
|
|
}).DialContext,
|
|
|
|
MaxIdleConns: 100,
|
|
|
|
IdleConnTimeout: 90 * time.Second,
|
|
|
|
TLSHandshakeTimeout: 10 * time.Second,
|
|
|
|
ExpectContinueTimeout: 1 * time.Second,
|
2019-12-07 18:54:03 +00:00
|
|
|
}
|
2019-12-08 04:10:22 +00:00
|
|
|
if scheme == "https" && insecure {
|
|
|
|
tr.TLSClientConfig = &tls.Config{InsecureSkipVerify: true}
|
2019-12-07 18:54:03 +00:00
|
|
|
}
|
2019-12-08 04:10:22 +00:00
|
|
|
return &http.Client{
|
|
|
|
Transport: &SpanTransport{
|
|
|
|
base: tr,
|
|
|
|
},
|
2019-12-07 18:54:03 +00:00
|
|
|
}
|
2019-12-07 00:45:49 +00:00
|
|
|
}
|