mirror of
https://github.com/status-im/consul.git
synced 2025-01-11 06:16:08 +00:00
vendor: add github.com/Azure/go-autorest/autorest
This commit is contained in:
parent
9d5a016494
commit
20c149873b
191
vendor/github.com/Azure/go-autorest/LICENSE
generated
vendored
Normal file
191
vendor/github.com/Azure/go-autorest/LICENSE
generated
vendored
Normal file
@ -0,0 +1,191 @@
|
|||||||
|
|
||||||
|
Apache License
|
||||||
|
Version 2.0, January 2004
|
||||||
|
http://www.apache.org/licenses/
|
||||||
|
|
||||||
|
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||||
|
|
||||||
|
1. Definitions.
|
||||||
|
|
||||||
|
"License" shall mean the terms and conditions for use, reproduction,
|
||||||
|
and distribution as defined by Sections 1 through 9 of this document.
|
||||||
|
|
||||||
|
"Licensor" shall mean the copyright owner or entity authorized by
|
||||||
|
the copyright owner that is granting the License.
|
||||||
|
|
||||||
|
"Legal Entity" shall mean the union of the acting entity and all
|
||||||
|
other entities that control, are controlled by, or are under common
|
||||||
|
control with that entity. For the purposes of this definition,
|
||||||
|
"control" means (i) the power, direct or indirect, to cause the
|
||||||
|
direction or management of such entity, whether by contract or
|
||||||
|
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||||
|
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||||
|
|
||||||
|
"You" (or "Your") shall mean an individual or Legal Entity
|
||||||
|
exercising permissions granted by this License.
|
||||||
|
|
||||||
|
"Source" form shall mean the preferred form for making modifications,
|
||||||
|
including but not limited to software source code, documentation
|
||||||
|
source, and configuration files.
|
||||||
|
|
||||||
|
"Object" form shall mean any form resulting from mechanical
|
||||||
|
transformation or translation of a Source form, including but
|
||||||
|
not limited to compiled object code, generated documentation,
|
||||||
|
and conversions to other media types.
|
||||||
|
|
||||||
|
"Work" shall mean the work of authorship, whether in Source or
|
||||||
|
Object form, made available under the License, as indicated by a
|
||||||
|
copyright notice that is included in or attached to the work
|
||||||
|
(an example is provided in the Appendix below).
|
||||||
|
|
||||||
|
"Derivative Works" shall mean any work, whether in Source or Object
|
||||||
|
form, that is based on (or derived from) the Work and for which the
|
||||||
|
editorial revisions, annotations, elaborations, or other modifications
|
||||||
|
represent, as a whole, an original work of authorship. For the purposes
|
||||||
|
of this License, Derivative Works shall not include works that remain
|
||||||
|
separable from, or merely link (or bind by name) to the interfaces of,
|
||||||
|
the Work and Derivative Works thereof.
|
||||||
|
|
||||||
|
"Contribution" shall mean any work of authorship, including
|
||||||
|
the original version of the Work and any modifications or additions
|
||||||
|
to that Work or Derivative Works thereof, that is intentionally
|
||||||
|
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||||
|
or by an individual or Legal Entity authorized to submit on behalf of
|
||||||
|
the copyright owner. For the purposes of this definition, "submitted"
|
||||||
|
means any form of electronic, verbal, or written communication sent
|
||||||
|
to the Licensor or its representatives, including but not limited to
|
||||||
|
communication on electronic mailing lists, source code control systems,
|
||||||
|
and issue tracking systems that are managed by, or on behalf of, the
|
||||||
|
Licensor for the purpose of discussing and improving the Work, but
|
||||||
|
excluding communication that is conspicuously marked or otherwise
|
||||||
|
designated in writing by the copyright owner as "Not a Contribution."
|
||||||
|
|
||||||
|
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||||
|
on behalf of whom a Contribution has been received by Licensor and
|
||||||
|
subsequently incorporated within the Work.
|
||||||
|
|
||||||
|
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
copyright license to reproduce, prepare Derivative Works of,
|
||||||
|
publicly display, publicly perform, sublicense, and distribute the
|
||||||
|
Work and such Derivative Works in Source or Object form.
|
||||||
|
|
||||||
|
3. Grant of Patent License. Subject to the terms and conditions of
|
||||||
|
this License, each Contributor hereby grants to You a perpetual,
|
||||||
|
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||||
|
(except as stated in this section) patent license to make, have made,
|
||||||
|
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||||
|
where such license applies only to those patent claims licensable
|
||||||
|
by such Contributor that are necessarily infringed by their
|
||||||
|
Contribution(s) alone or by combination of their Contribution(s)
|
||||||
|
with the Work to which such Contribution(s) was submitted. If You
|
||||||
|
institute patent litigation against any entity (including a
|
||||||
|
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||||
|
or a Contribution incorporated within the Work constitutes direct
|
||||||
|
or contributory patent infringement, then any patent licenses
|
||||||
|
granted to You under this License for that Work shall terminate
|
||||||
|
as of the date such litigation is filed.
|
||||||
|
|
||||||
|
4. Redistribution. You may reproduce and distribute copies of the
|
||||||
|
Work or Derivative Works thereof in any medium, with or without
|
||||||
|
modifications, and in Source or Object form, provided that You
|
||||||
|
meet the following conditions:
|
||||||
|
|
||||||
|
(a) You must give any other recipients of the Work or
|
||||||
|
Derivative Works a copy of this License; and
|
||||||
|
|
||||||
|
(b) You must cause any modified files to carry prominent notices
|
||||||
|
stating that You changed the files; and
|
||||||
|
|
||||||
|
(c) You must retain, in the Source form of any Derivative Works
|
||||||
|
that You distribute, all copyright, patent, trademark, and
|
||||||
|
attribution notices from the Source form of the Work,
|
||||||
|
excluding those notices that do not pertain to any part of
|
||||||
|
the Derivative Works; and
|
||||||
|
|
||||||
|
(d) If the Work includes a "NOTICE" text file as part of its
|
||||||
|
distribution, then any Derivative Works that You distribute must
|
||||||
|
include a readable copy of the attribution notices contained
|
||||||
|
within such NOTICE file, excluding those notices that do not
|
||||||
|
pertain to any part of the Derivative Works, in at least one
|
||||||
|
of the following places: within a NOTICE text file distributed
|
||||||
|
as part of the Derivative Works; within the Source form or
|
||||||
|
documentation, if provided along with the Derivative Works; or,
|
||||||
|
within a display generated by the Derivative Works, if and
|
||||||
|
wherever such third-party notices normally appear. The contents
|
||||||
|
of the NOTICE file are for informational purposes only and
|
||||||
|
do not modify the License. You may add Your own attribution
|
||||||
|
notices within Derivative Works that You distribute, alongside
|
||||||
|
or as an addendum to the NOTICE text from the Work, provided
|
||||||
|
that such additional attribution notices cannot be construed
|
||||||
|
as modifying the License.
|
||||||
|
|
||||||
|
You may add Your own copyright statement to Your modifications and
|
||||||
|
may provide additional or different license terms and conditions
|
||||||
|
for use, reproduction, or distribution of Your modifications, or
|
||||||
|
for any such Derivative Works as a whole, provided Your use,
|
||||||
|
reproduction, and distribution of the Work otherwise complies with
|
||||||
|
the conditions stated in this License.
|
||||||
|
|
||||||
|
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||||
|
any Contribution intentionally submitted for inclusion in the Work
|
||||||
|
by You to the Licensor shall be under the terms and conditions of
|
||||||
|
this License, without any additional terms or conditions.
|
||||||
|
Notwithstanding the above, nothing herein shall supersede or modify
|
||||||
|
the terms of any separate license agreement you may have executed
|
||||||
|
with Licensor regarding such Contributions.
|
||||||
|
|
||||||
|
6. Trademarks. This License does not grant permission to use the trade
|
||||||
|
names, trademarks, service marks, or product names of the Licensor,
|
||||||
|
except as required for reasonable and customary use in describing the
|
||||||
|
origin of the Work and reproducing the content of the NOTICE file.
|
||||||
|
|
||||||
|
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||||
|
agreed to in writing, Licensor provides the Work (and each
|
||||||
|
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||||
|
implied, including, without limitation, any warranties or conditions
|
||||||
|
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||||
|
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||||
|
appropriateness of using or redistributing the Work and assume any
|
||||||
|
risks associated with Your exercise of permissions under this License.
|
||||||
|
|
||||||
|
8. Limitation of Liability. In no event and under no legal theory,
|
||||||
|
whether in tort (including negligence), contract, or otherwise,
|
||||||
|
unless required by applicable law (such as deliberate and grossly
|
||||||
|
negligent acts) or agreed to in writing, shall any Contributor be
|
||||||
|
liable to You for damages, including any direct, indirect, special,
|
||||||
|
incidental, or consequential damages of any character arising as a
|
||||||
|
result of this License or out of the use or inability to use the
|
||||||
|
Work (including but not limited to damages for loss of goodwill,
|
||||||
|
work stoppage, computer failure or malfunction, or any and all
|
||||||
|
other commercial damages or losses), even if such Contributor
|
||||||
|
has been advised of the possibility of such damages.
|
||||||
|
|
||||||
|
9. Accepting Warranty or Additional Liability. While redistributing
|
||||||
|
the Work or Derivative Works thereof, You may choose to offer,
|
||||||
|
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||||
|
or other liability obligations and/or rights consistent with this
|
||||||
|
License. However, in accepting such obligations, You may act only
|
||||||
|
on Your own behalf and on Your sole responsibility, not on behalf
|
||||||
|
of any other Contributor, and only if You agree to indemnify,
|
||||||
|
defend, and hold each Contributor harmless for any liability
|
||||||
|
incurred by, or claims asserted against, such Contributor by reason
|
||||||
|
of your accepting any such warranty or additional liability.
|
||||||
|
|
||||||
|
END OF TERMS AND CONDITIONS
|
||||||
|
|
||||||
|
Copyright 2015 Microsoft Corporation
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
115
vendor/github.com/Azure/go-autorest/autorest/autorest.go
generated
vendored
Normal file
115
vendor/github.com/Azure/go-autorest/autorest/autorest.go
generated
vendored
Normal file
@ -0,0 +1,115 @@
|
|||||||
|
/*
|
||||||
|
Package autorest implements an HTTP request pipeline suitable for use across multiple go-routines
|
||||||
|
and provides the shared routines relied on by AutoRest (see https://github.com/Azure/autorest/)
|
||||||
|
generated Go code.
|
||||||
|
|
||||||
|
The package breaks sending and responding to HTTP requests into three phases: Preparing, Sending,
|
||||||
|
and Responding. A typical pattern is:
|
||||||
|
|
||||||
|
req, err := Prepare(&http.Request{},
|
||||||
|
token.WithAuthorization())
|
||||||
|
|
||||||
|
resp, err := Send(req,
|
||||||
|
WithLogging(logger),
|
||||||
|
DoErrorIfStatusCode(http.StatusInternalServerError),
|
||||||
|
DoCloseIfError(),
|
||||||
|
DoRetryForAttempts(5, time.Second))
|
||||||
|
|
||||||
|
err = Respond(resp,
|
||||||
|
ByDiscardingBody(),
|
||||||
|
ByClosing())
|
||||||
|
|
||||||
|
Each phase relies on decorators to modify and / or manage processing. Decorators may first modify
|
||||||
|
and then pass the data along, pass the data first and then modify the result, or wrap themselves
|
||||||
|
around passing the data (such as a logger might do). Decorators run in the order provided. For
|
||||||
|
example, the following:
|
||||||
|
|
||||||
|
req, err := Prepare(&http.Request{},
|
||||||
|
WithBaseURL("https://microsoft.com/"),
|
||||||
|
WithPath("a"),
|
||||||
|
WithPath("b"),
|
||||||
|
WithPath("c"))
|
||||||
|
|
||||||
|
will set the URL to:
|
||||||
|
|
||||||
|
https://microsoft.com/a/b/c
|
||||||
|
|
||||||
|
Preparers and Responders may be shared and re-used (assuming the underlying decorators support
|
||||||
|
sharing and re-use). Performant use is obtained by creating one or more Preparers and Responders
|
||||||
|
shared among multiple go-routines, and a single Sender shared among multiple sending go-routines,
|
||||||
|
all bound together by means of input / output channels.
|
||||||
|
|
||||||
|
Decorators hold their passed state within a closure (such as the path components in the example
|
||||||
|
above). Be careful to share Preparers and Responders only in a context where such held state
|
||||||
|
applies. For example, it may not make sense to share a Preparer that applies a query string from a
|
||||||
|
fixed set of values. Similarly, sharing a Responder that reads the response body into a passed
|
||||||
|
struct (e.g., ByUnmarshallingJson) is likely incorrect.
|
||||||
|
|
||||||
|
Lastly, the Swagger specification (https://swagger.io) that drives AutoRest
|
||||||
|
(https://github.com/Azure/autorest/) precisely defines two date forms: date and date-time. The
|
||||||
|
github.com/Azure/go-autorest/autorest/date package provides time.Time derivations to ensure
|
||||||
|
correct parsing and formatting.
|
||||||
|
|
||||||
|
Errors raised by autorest objects and methods will conform to the autorest.Error interface.
|
||||||
|
|
||||||
|
See the included examples for more detail. For details on the suggested use of this package by
|
||||||
|
generated clients, see the Client described below.
|
||||||
|
*/
|
||||||
|
package autorest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// HeaderLocation specifies the HTTP Location header.
|
||||||
|
HeaderLocation = "Location"
|
||||||
|
|
||||||
|
// HeaderRetryAfter specifies the HTTP Retry-After header.
|
||||||
|
HeaderRetryAfter = "Retry-After"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ResponseHasStatusCode returns true if the status code in the HTTP Response is in the passed set
|
||||||
|
// and false otherwise.
|
||||||
|
func ResponseHasStatusCode(resp *http.Response, codes ...int) bool {
|
||||||
|
return containsInt(codes, resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetLocation retrieves the URL from the Location header of the passed response.
|
||||||
|
func GetLocation(resp *http.Response) string {
|
||||||
|
return resp.Header.Get(HeaderLocation)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetRetryAfter extracts the retry delay from the Retry-After header of the passed response. If
|
||||||
|
// the header is absent or is malformed, it will return the supplied default delay time.Duration.
|
||||||
|
func GetRetryAfter(resp *http.Response, defaultDelay time.Duration) time.Duration {
|
||||||
|
retry := resp.Header.Get(HeaderRetryAfter)
|
||||||
|
if retry == "" {
|
||||||
|
return defaultDelay
|
||||||
|
}
|
||||||
|
|
||||||
|
d, err := time.ParseDuration(retry + "s")
|
||||||
|
if err != nil {
|
||||||
|
return defaultDelay
|
||||||
|
}
|
||||||
|
|
||||||
|
return d
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewPollingRequest allocates and returns a new http.Request to poll for the passed response.
|
||||||
|
func NewPollingRequest(resp *http.Response, cancel <-chan struct{}) (*http.Request, error) {
|
||||||
|
location := GetLocation(resp)
|
||||||
|
if location == "" {
|
||||||
|
return nil, NewErrorWithResponse("autorest", "NewPollingRequest", resp, "Location header missing from response that requires polling")
|
||||||
|
}
|
||||||
|
|
||||||
|
req, err := Prepare(&http.Request{Cancel: cancel},
|
||||||
|
AsGet(),
|
||||||
|
WithBaseURL(location))
|
||||||
|
if err != nil {
|
||||||
|
return nil, NewErrorWithError(err, "autorest", "NewPollingRequest", nil, "Failure creating poll request to %s", location)
|
||||||
|
}
|
||||||
|
|
||||||
|
return req, nil
|
||||||
|
}
|
235
vendor/github.com/Azure/go-autorest/autorest/client.go
generated
vendored
Normal file
235
vendor/github.com/Azure/go-autorest/autorest/client.go
generated
vendored
Normal file
@ -0,0 +1,235 @@
|
|||||||
|
package autorest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"net/http"
|
||||||
|
"net/http/cookiejar"
|
||||||
|
"runtime"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// DefaultPollingDelay is a reasonable delay between polling requests.
|
||||||
|
DefaultPollingDelay = 60 * time.Second
|
||||||
|
|
||||||
|
// DefaultPollingDuration is a reasonable total polling duration.
|
||||||
|
DefaultPollingDuration = 15 * time.Minute
|
||||||
|
|
||||||
|
// DefaultRetryAttempts is number of attempts for retry status codes (5xx).
|
||||||
|
DefaultRetryAttempts = 3
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// defaultUserAgent builds a string containing the Go version, system archityecture and OS,
|
||||||
|
// and the go-autorest version.
|
||||||
|
defaultUserAgent = fmt.Sprintf("Go/%s (%s-%s) go-autorest/%s",
|
||||||
|
runtime.Version(),
|
||||||
|
runtime.GOARCH,
|
||||||
|
runtime.GOOS,
|
||||||
|
Version(),
|
||||||
|
)
|
||||||
|
|
||||||
|
statusCodesForRetry = []int{
|
||||||
|
http.StatusRequestTimeout, // 408
|
||||||
|
http.StatusInternalServerError, // 500
|
||||||
|
http.StatusBadGateway, // 502
|
||||||
|
http.StatusServiceUnavailable, // 503
|
||||||
|
http.StatusGatewayTimeout, // 504
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
requestFormat = `HTTP Request Begin ===================================================
|
||||||
|
%s
|
||||||
|
===================================================== HTTP Request End
|
||||||
|
`
|
||||||
|
responseFormat = `HTTP Response Begin ===================================================
|
||||||
|
%s
|
||||||
|
===================================================== HTTP Response End
|
||||||
|
`
|
||||||
|
)
|
||||||
|
|
||||||
|
// Response serves as the base for all responses from generated clients. It provides access to the
|
||||||
|
// last http.Response.
|
||||||
|
type Response struct {
|
||||||
|
*http.Response `json:"-"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// LoggingInspector implements request and response inspectors that log the full request and
|
||||||
|
// response to a supplied log.
|
||||||
|
type LoggingInspector struct {
|
||||||
|
Logger *log.Logger
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithInspection returns a PrepareDecorator that emits the http.Request to the supplied logger. The
|
||||||
|
// body is restored after being emitted.
|
||||||
|
//
|
||||||
|
// Note: Since it reads the entire Body, this decorator should not be used where body streaming is
|
||||||
|
// important. It is best used to trace JSON or similar body values.
|
||||||
|
func (li LoggingInspector) WithInspection() PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
var body, b bytes.Buffer
|
||||||
|
|
||||||
|
defer r.Body.Close()
|
||||||
|
|
||||||
|
r.Body = ioutil.NopCloser(io.TeeReader(r.Body, &body))
|
||||||
|
if err := r.Write(&b); err != nil {
|
||||||
|
return nil, fmt.Errorf("Failed to write response: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
li.Logger.Printf(requestFormat, b.String())
|
||||||
|
|
||||||
|
r.Body = ioutil.NopCloser(&body)
|
||||||
|
return p.Prepare(r)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByInspecting returns a RespondDecorator that emits the http.Response to the supplied logger. The
|
||||||
|
// body is restored after being emitted.
|
||||||
|
//
|
||||||
|
// Note: Since it reads the entire Body, this decorator should not be used where body streaming is
|
||||||
|
// important. It is best used to trace JSON or similar body values.
|
||||||
|
func (li LoggingInspector) ByInspecting() RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
var body, b bytes.Buffer
|
||||||
|
defer resp.Body.Close()
|
||||||
|
resp.Body = ioutil.NopCloser(io.TeeReader(resp.Body, &body))
|
||||||
|
if err := resp.Write(&b); err != nil {
|
||||||
|
return fmt.Errorf("Failed to write response: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
li.Logger.Printf(responseFormat, b.String())
|
||||||
|
|
||||||
|
resp.Body = ioutil.NopCloser(&body)
|
||||||
|
return r.Respond(resp)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Client is the base for autorest generated clients. It provides default, "do nothing"
|
||||||
|
// implementations of an Authorizer, RequestInspector, and ResponseInspector. It also returns the
|
||||||
|
// standard, undecorated http.Client as a default Sender.
|
||||||
|
//
|
||||||
|
// Generated clients should also use Error (see NewError and NewErrorWithError) for errors and
|
||||||
|
// return responses that compose with Response.
|
||||||
|
//
|
||||||
|
// Most customization of generated clients is best achieved by supplying a custom Authorizer, custom
|
||||||
|
// RequestInspector, and / or custom ResponseInspector. Users may log requests, implement circuit
|
||||||
|
// breakers (see https://msdn.microsoft.com/en-us/library/dn589784.aspx) or otherwise influence
|
||||||
|
// sending the request by providing a decorated Sender.
|
||||||
|
type Client struct {
|
||||||
|
Authorizer Authorizer
|
||||||
|
Sender Sender
|
||||||
|
RequestInspector PrepareDecorator
|
||||||
|
ResponseInspector RespondDecorator
|
||||||
|
|
||||||
|
// PollingDelay sets the polling frequency used in absence of a Retry-After HTTP header
|
||||||
|
PollingDelay time.Duration
|
||||||
|
|
||||||
|
// PollingDuration sets the maximum polling time after which an error is returned.
|
||||||
|
PollingDuration time.Duration
|
||||||
|
|
||||||
|
// RetryAttempts sets the default number of retry attempts for client.
|
||||||
|
RetryAttempts int
|
||||||
|
|
||||||
|
// RetryDuration sets the delay duration for retries.
|
||||||
|
RetryDuration time.Duration
|
||||||
|
|
||||||
|
// UserAgent, if not empty, will be set as the HTTP User-Agent header on all requests sent
|
||||||
|
// through the Do method.
|
||||||
|
UserAgent string
|
||||||
|
|
||||||
|
Jar http.CookieJar
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewClientWithUserAgent returns an instance of a Client with the UserAgent set to the passed
|
||||||
|
// string.
|
||||||
|
func NewClientWithUserAgent(ua string) Client {
|
||||||
|
c := Client{
|
||||||
|
PollingDelay: DefaultPollingDelay,
|
||||||
|
PollingDuration: DefaultPollingDuration,
|
||||||
|
RetryAttempts: DefaultRetryAttempts,
|
||||||
|
RetryDuration: 30 * time.Second,
|
||||||
|
UserAgent: defaultUserAgent,
|
||||||
|
}
|
||||||
|
c.AddToUserAgent(ua)
|
||||||
|
return c
|
||||||
|
}
|
||||||
|
|
||||||
|
// AddToUserAgent adds an extension to the current user agent
|
||||||
|
func (c *Client) AddToUserAgent(extension string) error {
|
||||||
|
if extension != "" {
|
||||||
|
c.UserAgent = fmt.Sprintf("%s %s", c.UserAgent, extension)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return fmt.Errorf("Extension was empty, User Agent stayed as %s", c.UserAgent)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Do implements the Sender interface by invoking the active Sender after applying authorization.
|
||||||
|
// If Sender is not set, it uses a new instance of http.Client. In both cases it will, if UserAgent
|
||||||
|
// is set, apply set the User-Agent header.
|
||||||
|
func (c Client) Do(r *http.Request) (*http.Response, error) {
|
||||||
|
if r.UserAgent() == "" {
|
||||||
|
r, _ = Prepare(r,
|
||||||
|
WithUserAgent(c.UserAgent))
|
||||||
|
}
|
||||||
|
r, err := Prepare(r,
|
||||||
|
c.WithInspection(),
|
||||||
|
c.WithAuthorization())
|
||||||
|
if err != nil {
|
||||||
|
return nil, NewErrorWithError(err, "autorest/Client", "Do", nil, "Preparing request failed")
|
||||||
|
}
|
||||||
|
resp, err := SendWithSender(c.sender(), r,
|
||||||
|
DoRetryForStatusCodes(c.RetryAttempts, c.RetryDuration, statusCodesForRetry...))
|
||||||
|
Respond(resp,
|
||||||
|
c.ByInspecting())
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// sender returns the Sender to which to send requests.
|
||||||
|
func (c Client) sender() Sender {
|
||||||
|
if c.Sender == nil {
|
||||||
|
j, _ := cookiejar.New(nil)
|
||||||
|
return &http.Client{Jar: j}
|
||||||
|
}
|
||||||
|
return c.Sender
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithAuthorization is a convenience method that returns the WithAuthorization PrepareDecorator
|
||||||
|
// from the current Authorizer. If not Authorizer is set, it uses the NullAuthorizer.
|
||||||
|
func (c Client) WithAuthorization() PrepareDecorator {
|
||||||
|
return c.authorizer().WithAuthorization()
|
||||||
|
}
|
||||||
|
|
||||||
|
// authorizer returns the Authorizer to use.
|
||||||
|
func (c Client) authorizer() Authorizer {
|
||||||
|
if c.Authorizer == nil {
|
||||||
|
return NullAuthorizer{}
|
||||||
|
}
|
||||||
|
return c.Authorizer
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithInspection is a convenience method that passes the request to the supplied RequestInspector,
|
||||||
|
// if present, or returns the WithNothing PrepareDecorator otherwise.
|
||||||
|
func (c Client) WithInspection() PrepareDecorator {
|
||||||
|
if c.RequestInspector == nil {
|
||||||
|
return WithNothing()
|
||||||
|
}
|
||||||
|
return c.RequestInspector
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByInspecting is a convenience method that passes the response to the supplied ResponseInspector,
|
||||||
|
// if present, or returns the ByIgnoring RespondDecorator otherwise.
|
||||||
|
func (c Client) ByInspecting() RespondDecorator {
|
||||||
|
if c.ResponseInspector == nil {
|
||||||
|
return ByIgnoring()
|
||||||
|
}
|
||||||
|
return c.ResponseInspector
|
||||||
|
}
|
80
vendor/github.com/Azure/go-autorest/autorest/error.go
generated
vendored
Normal file
80
vendor/github.com/Azure/go-autorest/autorest/error.go
generated
vendored
Normal file
@ -0,0 +1,80 @@
|
|||||||
|
package autorest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// UndefinedStatusCode is used when HTTP status code is not available for an error.
|
||||||
|
UndefinedStatusCode = 0
|
||||||
|
)
|
||||||
|
|
||||||
|
// DetailedError encloses a error with details of the package, method, and associated HTTP
|
||||||
|
// status code (if any).
|
||||||
|
type DetailedError struct {
|
||||||
|
Original error
|
||||||
|
|
||||||
|
// PackageType is the package type of the object emitting the error. For types, the value
|
||||||
|
// matches that produced the the '%T' format specifier of the fmt package. For other elements,
|
||||||
|
// such as functions, it is just the package name (e.g., "autorest").
|
||||||
|
PackageType string
|
||||||
|
|
||||||
|
// Method is the name of the method raising the error.
|
||||||
|
Method string
|
||||||
|
|
||||||
|
// StatusCode is the HTTP Response StatusCode (if non-zero) that led to the error.
|
||||||
|
StatusCode interface{}
|
||||||
|
|
||||||
|
// Message is the error message.
|
||||||
|
Message string
|
||||||
|
|
||||||
|
// Service Error is the response body of failed API in bytes
|
||||||
|
ServiceError []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewError creates a new Error conforming object from the passed packageType, method, and
|
||||||
|
// message. message is treated as a format string to which the optional args apply.
|
||||||
|
func NewError(packageType string, method string, message string, args ...interface{}) DetailedError {
|
||||||
|
return NewErrorWithError(nil, packageType, method, nil, message, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewErrorWithResponse creates a new Error conforming object from the passed
|
||||||
|
// packageType, method, statusCode of the given resp (UndefinedStatusCode if
|
||||||
|
// resp is nil), and message. message is treated as a format string to which the
|
||||||
|
// optional args apply.
|
||||||
|
func NewErrorWithResponse(packageType string, method string, resp *http.Response, message string, args ...interface{}) DetailedError {
|
||||||
|
return NewErrorWithError(nil, packageType, method, resp, message, args...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewErrorWithError creates a new Error conforming object from the
|
||||||
|
// passed packageType, method, statusCode of the given resp (UndefinedStatusCode
|
||||||
|
// if resp is nil), message, and original error. message is treated as a format
|
||||||
|
// string to which the optional args apply.
|
||||||
|
func NewErrorWithError(original error, packageType string, method string, resp *http.Response, message string, args ...interface{}) DetailedError {
|
||||||
|
if v, ok := original.(DetailedError); ok {
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
|
||||||
|
statusCode := UndefinedStatusCode
|
||||||
|
if resp != nil {
|
||||||
|
statusCode = resp.StatusCode
|
||||||
|
}
|
||||||
|
|
||||||
|
return DetailedError{
|
||||||
|
Original: original,
|
||||||
|
PackageType: packageType,
|
||||||
|
Method: method,
|
||||||
|
StatusCode: statusCode,
|
||||||
|
Message: fmt.Sprintf(message, args...),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error returns a formatted containing all available details (i.e., PackageType, Method,
|
||||||
|
// StatusCode, Message, and original error (if any)).
|
||||||
|
func (e DetailedError) Error() string {
|
||||||
|
if e.Original == nil {
|
||||||
|
return fmt.Sprintf("%s#%s: %s: StatusCode=%d", e.PackageType, e.Method, e.Message, e.StatusCode)
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s#%s: %s: StatusCode=%d -- Original Error: %v", e.PackageType, e.Method, e.Message, e.StatusCode, e.Original)
|
||||||
|
}
|
443
vendor/github.com/Azure/go-autorest/autorest/preparer.go
generated
vendored
Normal file
443
vendor/github.com/Azure/go-autorest/autorest/preparer.go
generated
vendored
Normal file
@ -0,0 +1,443 @@
|
|||||||
|
package autorest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"mime/multipart"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
mimeTypeJSON = "application/json"
|
||||||
|
mimeTypeFormPost = "application/x-www-form-urlencoded"
|
||||||
|
|
||||||
|
headerAuthorization = "Authorization"
|
||||||
|
headerContentType = "Content-Type"
|
||||||
|
headerUserAgent = "User-Agent"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Preparer is the interface that wraps the Prepare method.
|
||||||
|
//
|
||||||
|
// Prepare accepts and possibly modifies an http.Request (e.g., adding Headers). Implementations
|
||||||
|
// must ensure to not share or hold per-invocation state since Preparers may be shared and re-used.
|
||||||
|
type Preparer interface {
|
||||||
|
Prepare(*http.Request) (*http.Request, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PreparerFunc is a method that implements the Preparer interface.
|
||||||
|
type PreparerFunc func(*http.Request) (*http.Request, error)
|
||||||
|
|
||||||
|
// Prepare implements the Preparer interface on PreparerFunc.
|
||||||
|
func (pf PreparerFunc) Prepare(r *http.Request) (*http.Request, error) {
|
||||||
|
return pf(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
// PrepareDecorator takes and possibly decorates, by wrapping, a Preparer. Decorators may affect the
|
||||||
|
// http.Request and pass it along or, first, pass the http.Request along then affect the result.
|
||||||
|
type PrepareDecorator func(Preparer) Preparer
|
||||||
|
|
||||||
|
// CreatePreparer creates, decorates, and returns a Preparer.
|
||||||
|
// Without decorators, the returned Preparer returns the passed http.Request unmodified.
|
||||||
|
// Preparers are safe to share and re-use.
|
||||||
|
func CreatePreparer(decorators ...PrepareDecorator) Preparer {
|
||||||
|
return DecoratePreparer(
|
||||||
|
Preparer(PreparerFunc(func(r *http.Request) (*http.Request, error) { return r, nil })),
|
||||||
|
decorators...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecoratePreparer accepts a Preparer and a, possibly empty, set of PrepareDecorators, which it
|
||||||
|
// applies to the Preparer. Decorators are applied in the order received, but their affect upon the
|
||||||
|
// request depends on whether they are a pre-decorator (change the http.Request and then pass it
|
||||||
|
// along) or a post-decorator (pass the http.Request along and alter it on return).
|
||||||
|
func DecoratePreparer(p Preparer, decorators ...PrepareDecorator) Preparer {
|
||||||
|
for _, decorate := range decorators {
|
||||||
|
p = decorate(p)
|
||||||
|
}
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prepare accepts an http.Request and a, possibly empty, set of PrepareDecorators.
|
||||||
|
// It creates a Preparer from the decorators which it then applies to the passed http.Request.
|
||||||
|
func Prepare(r *http.Request, decorators ...PrepareDecorator) (*http.Request, error) {
|
||||||
|
if r == nil {
|
||||||
|
return nil, NewError("autorest", "Prepare", "Invoked without an http.Request")
|
||||||
|
}
|
||||||
|
return CreatePreparer(decorators...).Prepare(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithNothing returns a "do nothing" PrepareDecorator that makes no changes to the passed
|
||||||
|
// http.Request.
|
||||||
|
func WithNothing() PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
return p.Prepare(r)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithHeader returns a PrepareDecorator that sets the specified HTTP header of the http.Request to
|
||||||
|
// the passed value. It canonicalizes the passed header name (via http.CanonicalHeaderKey) before
|
||||||
|
// adding the header.
|
||||||
|
func WithHeader(header string, value string) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
if r.Header == nil {
|
||||||
|
r.Header = make(http.Header)
|
||||||
|
}
|
||||||
|
r.Header.Set(http.CanonicalHeaderKey(header), value)
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithBearerAuthorization returns a PrepareDecorator that adds an HTTP Authorization header whose
|
||||||
|
// value is "Bearer " followed by the supplied token.
|
||||||
|
func WithBearerAuthorization(token string) PrepareDecorator {
|
||||||
|
return WithHeader(headerAuthorization, fmt.Sprintf("Bearer %s", token))
|
||||||
|
}
|
||||||
|
|
||||||
|
// AsContentType returns a PrepareDecorator that adds an HTTP Content-Type header whose value
|
||||||
|
// is the passed contentType.
|
||||||
|
func AsContentType(contentType string) PrepareDecorator {
|
||||||
|
return WithHeader(headerContentType, contentType)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithUserAgent returns a PrepareDecorator that adds an HTTP User-Agent header whose value is the
|
||||||
|
// passed string.
|
||||||
|
func WithUserAgent(ua string) PrepareDecorator {
|
||||||
|
return WithHeader(headerUserAgent, ua)
|
||||||
|
}
|
||||||
|
|
||||||
|
// AsFormURLEncoded returns a PrepareDecorator that adds an HTTP Content-Type header whose value is
|
||||||
|
// "application/x-www-form-urlencoded".
|
||||||
|
func AsFormURLEncoded() PrepareDecorator {
|
||||||
|
return AsContentType(mimeTypeFormPost)
|
||||||
|
}
|
||||||
|
|
||||||
|
// AsJSON returns a PrepareDecorator that adds an HTTP Content-Type header whose value is
|
||||||
|
// "application/json".
|
||||||
|
func AsJSON() PrepareDecorator {
|
||||||
|
return AsContentType(mimeTypeJSON)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithMethod returns a PrepareDecorator that sets the HTTP method of the passed request. The
|
||||||
|
// decorator does not validate that the passed method string is a known HTTP method.
|
||||||
|
func WithMethod(method string) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r.Method = method
|
||||||
|
return p.Prepare(r)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// AsDelete returns a PrepareDecorator that sets the HTTP method to DELETE.
|
||||||
|
func AsDelete() PrepareDecorator { return WithMethod("DELETE") }
|
||||||
|
|
||||||
|
// AsGet returns a PrepareDecorator that sets the HTTP method to GET.
|
||||||
|
func AsGet() PrepareDecorator { return WithMethod("GET") }
|
||||||
|
|
||||||
|
// AsHead returns a PrepareDecorator that sets the HTTP method to HEAD.
|
||||||
|
func AsHead() PrepareDecorator { return WithMethod("HEAD") }
|
||||||
|
|
||||||
|
// AsOptions returns a PrepareDecorator that sets the HTTP method to OPTIONS.
|
||||||
|
func AsOptions() PrepareDecorator { return WithMethod("OPTIONS") }
|
||||||
|
|
||||||
|
// AsPatch returns a PrepareDecorator that sets the HTTP method to PATCH.
|
||||||
|
func AsPatch() PrepareDecorator { return WithMethod("PATCH") }
|
||||||
|
|
||||||
|
// AsPost returns a PrepareDecorator that sets the HTTP method to POST.
|
||||||
|
func AsPost() PrepareDecorator { return WithMethod("POST") }
|
||||||
|
|
||||||
|
// AsPut returns a PrepareDecorator that sets the HTTP method to PUT.
|
||||||
|
func AsPut() PrepareDecorator { return WithMethod("PUT") }
|
||||||
|
|
||||||
|
// WithBaseURL returns a PrepareDecorator that populates the http.Request with a url.URL constructed
|
||||||
|
// from the supplied baseUrl.
|
||||||
|
func WithBaseURL(baseURL string) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
var u *url.URL
|
||||||
|
if u, err = url.Parse(baseURL); err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
if u.Scheme == "" {
|
||||||
|
err = fmt.Errorf("autorest: No scheme detected in URL %s", baseURL)
|
||||||
|
}
|
||||||
|
if err == nil {
|
||||||
|
r.URL = u
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithCustomBaseURL returns a PrepareDecorator that replaces brace-enclosed keys within the
|
||||||
|
// request base URL (i.e., http.Request.URL) with the corresponding values from the passed map.
|
||||||
|
func WithCustomBaseURL(baseURL string, urlParameters map[string]interface{}) PrepareDecorator {
|
||||||
|
parameters := ensureValueStrings(urlParameters)
|
||||||
|
for key, value := range parameters {
|
||||||
|
baseURL = strings.Replace(baseURL, "{"+key+"}", value, -1)
|
||||||
|
}
|
||||||
|
return WithBaseURL(baseURL)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithFormData returns a PrepareDecoratore that "URL encodes" (e.g., bar=baz&foo=quux) into the
|
||||||
|
// http.Request body.
|
||||||
|
func WithFormData(v url.Values) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
s := v.Encode()
|
||||||
|
r.ContentLength = int64(len(s))
|
||||||
|
r.Body = ioutil.NopCloser(strings.NewReader(s))
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithMultiPartFormData returns a PrepareDecoratore that "URL encodes" (e.g., bar=baz&foo=quux) form parameters
|
||||||
|
// into the http.Request body.
|
||||||
|
func WithMultiPartFormData(formDataParameters map[string]interface{}) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
var body bytes.Buffer
|
||||||
|
writer := multipart.NewWriter(&body)
|
||||||
|
for key, value := range formDataParameters {
|
||||||
|
if rc, ok := value.(io.ReadCloser); ok {
|
||||||
|
var fd io.Writer
|
||||||
|
if fd, err = writer.CreateFormFile(key, key); err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
if _, err = io.Copy(fd, rc); err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err = writer.WriteField(key, ensureValueString(value)); err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err = writer.Close(); err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
if r.Header == nil {
|
||||||
|
r.Header = make(http.Header)
|
||||||
|
}
|
||||||
|
r.Header.Set(http.CanonicalHeaderKey(headerContentType), writer.FormDataContentType())
|
||||||
|
r.Body = ioutil.NopCloser(bytes.NewReader(body.Bytes()))
|
||||||
|
r.ContentLength = int64(body.Len())
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithFile returns a PrepareDecorator that sends file in request body.
|
||||||
|
func WithFile(f io.ReadCloser) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
b, err := ioutil.ReadAll(f)
|
||||||
|
if err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
r.Body = ioutil.NopCloser(bytes.NewReader(b))
|
||||||
|
r.ContentLength = int64(len(b))
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithBool returns a PrepareDecorator that encodes the passed bool into the body of the request
|
||||||
|
// and sets the Content-Length header.
|
||||||
|
func WithBool(v bool) PrepareDecorator {
|
||||||
|
return WithString(fmt.Sprintf("%v", v))
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithFloat32 returns a PrepareDecorator that encodes the passed float32 into the body of the
|
||||||
|
// request and sets the Content-Length header.
|
||||||
|
func WithFloat32(v float32) PrepareDecorator {
|
||||||
|
return WithString(fmt.Sprintf("%v", v))
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithFloat64 returns a PrepareDecorator that encodes the passed float64 into the body of the
|
||||||
|
// request and sets the Content-Length header.
|
||||||
|
func WithFloat64(v float64) PrepareDecorator {
|
||||||
|
return WithString(fmt.Sprintf("%v", v))
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithInt32 returns a PrepareDecorator that encodes the passed int32 into the body of the request
|
||||||
|
// and sets the Content-Length header.
|
||||||
|
func WithInt32(v int32) PrepareDecorator {
|
||||||
|
return WithString(fmt.Sprintf("%v", v))
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithInt64 returns a PrepareDecorator that encodes the passed int64 into the body of the request
|
||||||
|
// and sets the Content-Length header.
|
||||||
|
func WithInt64(v int64) PrepareDecorator {
|
||||||
|
return WithString(fmt.Sprintf("%v", v))
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithString returns a PrepareDecorator that encodes the passed string into the body of the request
|
||||||
|
// and sets the Content-Length header.
|
||||||
|
func WithString(v string) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
r.ContentLength = int64(len(v))
|
||||||
|
r.Body = ioutil.NopCloser(strings.NewReader(v))
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithJSON returns a PrepareDecorator that encodes the data passed as JSON into the body of the
|
||||||
|
// request and sets the Content-Length header.
|
||||||
|
func WithJSON(v interface{}) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
b, err := json.Marshal(v)
|
||||||
|
if err == nil {
|
||||||
|
r.ContentLength = int64(len(b))
|
||||||
|
r.Body = ioutil.NopCloser(bytes.NewReader(b))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithPath returns a PrepareDecorator that adds the supplied path to the request URL. If the path
|
||||||
|
// is absolute (that is, it begins with a "/"), it replaces the existing path.
|
||||||
|
func WithPath(path string) PrepareDecorator {
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
if r.URL == nil {
|
||||||
|
return r, NewError("autorest", "WithPath", "Invoked with a nil URL")
|
||||||
|
}
|
||||||
|
if r.URL, err = parseURL(r.URL, path); err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithEscapedPathParameters returns a PrepareDecorator that replaces brace-enclosed keys within the
|
||||||
|
// request path (i.e., http.Request.URL.Path) with the corresponding values from the passed map. The
|
||||||
|
// values will be escaped (aka URL encoded) before insertion into the path.
|
||||||
|
func WithEscapedPathParameters(path string, pathParameters map[string]interface{}) PrepareDecorator {
|
||||||
|
parameters := escapeValueStrings(ensureValueStrings(pathParameters))
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
if r.URL == nil {
|
||||||
|
return r, NewError("autorest", "WithEscapedPathParameters", "Invoked with a nil URL")
|
||||||
|
}
|
||||||
|
for key, value := range parameters {
|
||||||
|
path = strings.Replace(path, "{"+key+"}", value, -1)
|
||||||
|
}
|
||||||
|
if r.URL, err = parseURL(r.URL, path); err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithPathParameters returns a PrepareDecorator that replaces brace-enclosed keys within the
|
||||||
|
// request path (i.e., http.Request.URL.Path) with the corresponding values from the passed map.
|
||||||
|
func WithPathParameters(path string, pathParameters map[string]interface{}) PrepareDecorator {
|
||||||
|
parameters := ensureValueStrings(pathParameters)
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
if r.URL == nil {
|
||||||
|
return r, NewError("autorest", "WithPathParameters", "Invoked with a nil URL")
|
||||||
|
}
|
||||||
|
for key, value := range parameters {
|
||||||
|
path = strings.Replace(path, "{"+key+"}", value, -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.URL, err = parseURL(r.URL, path); err != nil {
|
||||||
|
return r, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseURL(u *url.URL, path string) (*url.URL, error) {
|
||||||
|
p := strings.TrimRight(u.String(), "/")
|
||||||
|
if !strings.HasPrefix(path, "/") {
|
||||||
|
path = "/" + path
|
||||||
|
}
|
||||||
|
return url.Parse(p + path)
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithQueryParameters returns a PrepareDecorators that encodes and applies the query parameters
|
||||||
|
// given in the supplied map (i.e., key=value).
|
||||||
|
func WithQueryParameters(queryParameters map[string]interface{}) PrepareDecorator {
|
||||||
|
parameters := ensureValueStrings(queryParameters)
|
||||||
|
return func(p Preparer) Preparer {
|
||||||
|
return PreparerFunc(func(r *http.Request) (*http.Request, error) {
|
||||||
|
r, err := p.Prepare(r)
|
||||||
|
if err == nil {
|
||||||
|
if r.URL == nil {
|
||||||
|
return r, NewError("autorest", "WithQueryParameters", "Invoked with a nil URL")
|
||||||
|
}
|
||||||
|
v := r.URL.Query()
|
||||||
|
for key, value := range parameters {
|
||||||
|
v.Add(key, value)
|
||||||
|
}
|
||||||
|
r.URL.RawQuery = createQuery(v)
|
||||||
|
}
|
||||||
|
return r, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Authorizer is the interface that provides a PrepareDecorator used to supply request
|
||||||
|
// authorization. Most often, the Authorizer decorator runs last so it has access to the full
|
||||||
|
// state of the formed HTTP request.
|
||||||
|
type Authorizer interface {
|
||||||
|
WithAuthorization() PrepareDecorator
|
||||||
|
}
|
||||||
|
|
||||||
|
// NullAuthorizer implements a default, "do nothing" Authorizer.
|
||||||
|
type NullAuthorizer struct{}
|
||||||
|
|
||||||
|
// WithAuthorization returns a PrepareDecorator that does nothing.
|
||||||
|
func (na NullAuthorizer) WithAuthorization() PrepareDecorator {
|
||||||
|
return WithNothing()
|
||||||
|
}
|
236
vendor/github.com/Azure/go-autorest/autorest/responder.go
generated
vendored
Normal file
236
vendor/github.com/Azure/go-autorest/autorest/responder.go
generated
vendored
Normal file
@ -0,0 +1,236 @@
|
|||||||
|
package autorest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"encoding/xml"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Responder is the interface that wraps the Respond method.
|
||||||
|
//
|
||||||
|
// Respond accepts and reacts to an http.Response. Implementations must ensure to not share or hold
|
||||||
|
// state since Responders may be shared and re-used.
|
||||||
|
type Responder interface {
|
||||||
|
Respond(*http.Response) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// ResponderFunc is a method that implements the Responder interface.
|
||||||
|
type ResponderFunc func(*http.Response) error
|
||||||
|
|
||||||
|
// Respond implements the Responder interface on ResponderFunc.
|
||||||
|
func (rf ResponderFunc) Respond(r *http.Response) error {
|
||||||
|
return rf(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
// RespondDecorator takes and possibly decorates, by wrapping, a Responder. Decorators may react to
|
||||||
|
// the http.Response and pass it along or, first, pass the http.Response along then react.
|
||||||
|
type RespondDecorator func(Responder) Responder
|
||||||
|
|
||||||
|
// CreateResponder creates, decorates, and returns a Responder. Without decorators, the returned
|
||||||
|
// Responder returns the passed http.Response unmodified. Responders may or may not be safe to share
|
||||||
|
// and re-used: It depends on the applied decorators. For example, a standard decorator that closes
|
||||||
|
// the response body is fine to share whereas a decorator that reads the body into a passed struct
|
||||||
|
// is not.
|
||||||
|
//
|
||||||
|
// To prevent memory leaks, ensure that at least one Responder closes the response body.
|
||||||
|
func CreateResponder(decorators ...RespondDecorator) Responder {
|
||||||
|
return DecorateResponder(
|
||||||
|
Responder(ResponderFunc(func(r *http.Response) error { return nil })),
|
||||||
|
decorators...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecorateResponder accepts a Responder and a, possibly empty, set of RespondDecorators, which it
|
||||||
|
// applies to the Responder. Decorators are applied in the order received, but their affect upon the
|
||||||
|
// request depends on whether they are a pre-decorator (react to the http.Response and then pass it
|
||||||
|
// along) or a post-decorator (pass the http.Response along and then react).
|
||||||
|
func DecorateResponder(r Responder, decorators ...RespondDecorator) Responder {
|
||||||
|
for _, decorate := range decorators {
|
||||||
|
r = decorate(r)
|
||||||
|
}
|
||||||
|
return r
|
||||||
|
}
|
||||||
|
|
||||||
|
// Respond accepts an http.Response and a, possibly empty, set of RespondDecorators.
|
||||||
|
// It creates a Responder from the decorators it then applies to the passed http.Response.
|
||||||
|
func Respond(r *http.Response, decorators ...RespondDecorator) error {
|
||||||
|
if r == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return CreateResponder(decorators...).Respond(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByIgnoring returns a RespondDecorator that ignores the passed http.Response passing it unexamined
|
||||||
|
// to the next RespondDecorator.
|
||||||
|
func ByIgnoring() RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
return r.Respond(resp)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByCopying copies the contents of the http.Response Body into the passed bytes.Buffer as
|
||||||
|
// the Body is read.
|
||||||
|
func ByCopying(b *bytes.Buffer) RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
err := r.Respond(resp)
|
||||||
|
if err == nil && resp != nil && resp.Body != nil {
|
||||||
|
resp.Body = TeeReadCloser(resp.Body, b)
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByDiscardingBody returns a RespondDecorator that first invokes the passed Responder after which
|
||||||
|
// it copies the remaining bytes (if any) in the response body to ioutil.Discard. Since the passed
|
||||||
|
// Responder is invoked prior to discarding the response body, the decorator may occur anywhere
|
||||||
|
// within the set.
|
||||||
|
func ByDiscardingBody() RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
err := r.Respond(resp)
|
||||||
|
if err == nil && resp != nil && resp.Body != nil {
|
||||||
|
if _, err := io.Copy(ioutil.Discard, resp.Body); err != nil {
|
||||||
|
return fmt.Errorf("Error discarding the response body: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByClosing returns a RespondDecorator that first invokes the passed Responder after which it
|
||||||
|
// closes the response body. Since the passed Responder is invoked prior to closing the response
|
||||||
|
// body, the decorator may occur anywhere within the set.
|
||||||
|
func ByClosing() RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
err := r.Respond(resp)
|
||||||
|
if resp != nil && resp.Body != nil {
|
||||||
|
if err := resp.Body.Close(); err != nil {
|
||||||
|
return fmt.Errorf("Error closing the response body: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByClosingIfError returns a RespondDecorator that first invokes the passed Responder after which
|
||||||
|
// it closes the response if the passed Responder returns an error and the response body exists.
|
||||||
|
func ByClosingIfError() RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
err := r.Respond(resp)
|
||||||
|
if err != nil && resp != nil && resp.Body != nil {
|
||||||
|
if err := resp.Body.Close(); err != nil {
|
||||||
|
return fmt.Errorf("Error closing the response body: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByUnmarshallingJSON returns a RespondDecorator that decodes a JSON document returned in the
|
||||||
|
// response Body into the value pointed to by v.
|
||||||
|
func ByUnmarshallingJSON(v interface{}) RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
err := r.Respond(resp)
|
||||||
|
if err == nil {
|
||||||
|
b, errInner := ioutil.ReadAll(resp.Body)
|
||||||
|
// Some responses might include a BOM, remove for successful unmarshalling
|
||||||
|
b = bytes.TrimPrefix(b, []byte("\xef\xbb\xbf"))
|
||||||
|
if errInner != nil {
|
||||||
|
err = fmt.Errorf("Error occurred reading http.Response#Body - Error = '%v'", errInner)
|
||||||
|
} else if len(strings.Trim(string(b), " ")) > 0 {
|
||||||
|
errInner = json.Unmarshal(b, v)
|
||||||
|
if errInner != nil {
|
||||||
|
err = fmt.Errorf("Error occurred unmarshalling JSON - Error = '%v' JSON = '%s'", errInner, string(b))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ByUnmarshallingXML returns a RespondDecorator that decodes a XML document returned in the
|
||||||
|
// response Body into the value pointed to by v.
|
||||||
|
func ByUnmarshallingXML(v interface{}) RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
err := r.Respond(resp)
|
||||||
|
if err == nil {
|
||||||
|
b, errInner := ioutil.ReadAll(resp.Body)
|
||||||
|
if errInner != nil {
|
||||||
|
err = fmt.Errorf("Error occurred reading http.Response#Body - Error = '%v'", errInner)
|
||||||
|
} else {
|
||||||
|
errInner = xml.Unmarshal(b, v)
|
||||||
|
if errInner != nil {
|
||||||
|
err = fmt.Errorf("Error occurred unmarshalling Xml - Error = '%v' Xml = '%s'", errInner, string(b))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithErrorUnlessStatusCode returns a RespondDecorator that emits an error unless the response
|
||||||
|
// StatusCode is among the set passed. On error, response body is fully read into a buffer and
|
||||||
|
// presented in the returned error, as well as in the response body.
|
||||||
|
func WithErrorUnlessStatusCode(codes ...int) RespondDecorator {
|
||||||
|
return func(r Responder) Responder {
|
||||||
|
return ResponderFunc(func(resp *http.Response) error {
|
||||||
|
err := r.Respond(resp)
|
||||||
|
if err == nil && !ResponseHasStatusCode(resp, codes...) {
|
||||||
|
derr := NewErrorWithResponse("autorest", "WithErrorUnlessStatusCode", resp, "%v %v failed with %s",
|
||||||
|
resp.Request.Method,
|
||||||
|
resp.Request.URL,
|
||||||
|
resp.Status)
|
||||||
|
if resp.Body != nil {
|
||||||
|
defer resp.Body.Close()
|
||||||
|
b, _ := ioutil.ReadAll(resp.Body)
|
||||||
|
derr.ServiceError = b
|
||||||
|
resp.Body = ioutil.NopCloser(bytes.NewReader(b))
|
||||||
|
}
|
||||||
|
err = derr
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithErrorUnlessOK returns a RespondDecorator that emits an error if the response StatusCode is
|
||||||
|
// anything other than HTTP 200.
|
||||||
|
func WithErrorUnlessOK() RespondDecorator {
|
||||||
|
return WithErrorUnlessStatusCode(http.StatusOK)
|
||||||
|
}
|
||||||
|
|
||||||
|
// ExtractHeader extracts all values of the specified header from the http.Response. It returns an
|
||||||
|
// empty string slice if the passed http.Response is nil or the header does not exist.
|
||||||
|
func ExtractHeader(header string, resp *http.Response) []string {
|
||||||
|
if resp != nil && resp.Header != nil {
|
||||||
|
return resp.Header[http.CanonicalHeaderKey(header)]
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ExtractHeaderValue extracts the first value of the specified header from the http.Response. It
|
||||||
|
// returns an empty string if the passed http.Response is nil or the header does not exist.
|
||||||
|
func ExtractHeaderValue(header string, resp *http.Response) string {
|
||||||
|
h := ExtractHeader(header, resp)
|
||||||
|
if len(h) > 0 {
|
||||||
|
return h[0]
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
270
vendor/github.com/Azure/go-autorest/autorest/sender.go
generated
vendored
Normal file
270
vendor/github.com/Azure/go-autorest/autorest/sender.go
generated
vendored
Normal file
@ -0,0 +1,270 @@
|
|||||||
|
package autorest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"math"
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Sender is the interface that wraps the Do method to send HTTP requests.
|
||||||
|
//
|
||||||
|
// The standard http.Client conforms to this interface.
|
||||||
|
type Sender interface {
|
||||||
|
Do(*http.Request) (*http.Response, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SenderFunc is a method that implements the Sender interface.
|
||||||
|
type SenderFunc func(*http.Request) (*http.Response, error)
|
||||||
|
|
||||||
|
// Do implements the Sender interface on SenderFunc.
|
||||||
|
func (sf SenderFunc) Do(r *http.Request) (*http.Response, error) {
|
||||||
|
return sf(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendDecorator takes and possibily decorates, by wrapping, a Sender. Decorators may affect the
|
||||||
|
// http.Request and pass it along or, first, pass the http.Request along then react to the
|
||||||
|
// http.Response result.
|
||||||
|
type SendDecorator func(Sender) Sender
|
||||||
|
|
||||||
|
// CreateSender creates, decorates, and returns, as a Sender, the default http.Client.
|
||||||
|
func CreateSender(decorators ...SendDecorator) Sender {
|
||||||
|
return DecorateSender(&http.Client{}, decorators...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// DecorateSender accepts a Sender and a, possibly empty, set of SendDecorators, which is applies to
|
||||||
|
// the Sender. Decorators are applied in the order received, but their affect upon the request
|
||||||
|
// depends on whether they are a pre-decorator (change the http.Request and then pass it along) or a
|
||||||
|
// post-decorator (pass the http.Request along and react to the results in http.Response).
|
||||||
|
func DecorateSender(s Sender, decorators ...SendDecorator) Sender {
|
||||||
|
for _, decorate := range decorators {
|
||||||
|
s = decorate(s)
|
||||||
|
}
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send sends, by means of the default http.Client, the passed http.Request, returning the
|
||||||
|
// http.Response and possible error. It also accepts a, possibly empty, set of SendDecorators which
|
||||||
|
// it will apply the http.Client before invoking the Do method.
|
||||||
|
//
|
||||||
|
// Send is a convenience method and not recommended for production. Advanced users should use
|
||||||
|
// SendWithSender, passing and sharing their own Sender (e.g., instance of http.Client).
|
||||||
|
//
|
||||||
|
// Send will not poll or retry requests.
|
||||||
|
func Send(r *http.Request, decorators ...SendDecorator) (*http.Response, error) {
|
||||||
|
return SendWithSender(&http.Client{}, r, decorators...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// SendWithSender sends the passed http.Request, through the provided Sender, returning the
|
||||||
|
// http.Response and possible error. It also accepts a, possibly empty, set of SendDecorators which
|
||||||
|
// it will apply the http.Client before invoking the Do method.
|
||||||
|
//
|
||||||
|
// SendWithSender will not poll or retry requests.
|
||||||
|
func SendWithSender(s Sender, r *http.Request, decorators ...SendDecorator) (*http.Response, error) {
|
||||||
|
return DecorateSender(s, decorators...).Do(r)
|
||||||
|
}
|
||||||
|
|
||||||
|
// AfterDelay returns a SendDecorator that delays for the passed time.Duration before
|
||||||
|
// invoking the Sender. The delay may be terminated by closing the optional channel on the
|
||||||
|
// http.Request. If canceled, no further Senders are invoked.
|
||||||
|
func AfterDelay(d time.Duration) SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (*http.Response, error) {
|
||||||
|
if !DelayForBackoff(d, 0, r.Cancel) {
|
||||||
|
return nil, fmt.Errorf("autorest: AfterDelay canceled before full delay")
|
||||||
|
}
|
||||||
|
return s.Do(r)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// AsIs returns a SendDecorator that invokes the passed Sender without modifying the http.Request.
|
||||||
|
func AsIs() SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (*http.Response, error) {
|
||||||
|
return s.Do(r)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoCloseIfError returns a SendDecorator that first invokes the passed Sender after which
|
||||||
|
// it closes the response if the passed Sender returns an error and the response body exists.
|
||||||
|
func DoCloseIfError() SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (*http.Response, error) {
|
||||||
|
resp, err := s.Do(r)
|
||||||
|
if err != nil {
|
||||||
|
Respond(resp, ByDiscardingBody(), ByClosing())
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoErrorIfStatusCode returns a SendDecorator that emits an error if the response StatusCode is
|
||||||
|
// among the set passed. Since these are artificial errors, the response body may still require
|
||||||
|
// closing.
|
||||||
|
func DoErrorIfStatusCode(codes ...int) SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (*http.Response, error) {
|
||||||
|
resp, err := s.Do(r)
|
||||||
|
if err == nil && ResponseHasStatusCode(resp, codes...) {
|
||||||
|
err = NewErrorWithResponse("autorest", "DoErrorIfStatusCode", resp, "%v %v failed with %s",
|
||||||
|
resp.Request.Method,
|
||||||
|
resp.Request.URL,
|
||||||
|
resp.Status)
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoErrorUnlessStatusCode returns a SendDecorator that emits an error unless the response
|
||||||
|
// StatusCode is among the set passed. Since these are artificial errors, the response body
|
||||||
|
// may still require closing.
|
||||||
|
func DoErrorUnlessStatusCode(codes ...int) SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (*http.Response, error) {
|
||||||
|
resp, err := s.Do(r)
|
||||||
|
if err == nil && !ResponseHasStatusCode(resp, codes...) {
|
||||||
|
err = NewErrorWithResponse("autorest", "DoErrorUnlessStatusCode", resp, "%v %v failed with %s",
|
||||||
|
resp.Request.Method,
|
||||||
|
resp.Request.URL,
|
||||||
|
resp.Status)
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoPollForStatusCodes returns a SendDecorator that polls if the http.Response contains one of the
|
||||||
|
// passed status codes. It expects the http.Response to contain a Location header providing the
|
||||||
|
// URL at which to poll (using GET) and will poll until the time passed is equal to or greater than
|
||||||
|
// the supplied duration. It will delay between requests for the duration specified in the
|
||||||
|
// RetryAfter header or, if the header is absent, the passed delay. Polling may be canceled by
|
||||||
|
// closing the optional channel on the http.Request.
|
||||||
|
func DoPollForStatusCodes(duration time.Duration, delay time.Duration, codes ...int) SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (resp *http.Response, err error) {
|
||||||
|
resp, err = s.Do(r)
|
||||||
|
|
||||||
|
if err == nil && ResponseHasStatusCode(resp, codes...) {
|
||||||
|
r, err = NewPollingRequest(resp, r.Cancel)
|
||||||
|
|
||||||
|
for err == nil && ResponseHasStatusCode(resp, codes...) {
|
||||||
|
Respond(resp,
|
||||||
|
ByDiscardingBody(),
|
||||||
|
ByClosing())
|
||||||
|
resp, err = SendWithSender(s, r,
|
||||||
|
AfterDelay(GetRetryAfter(resp, delay)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return resp, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoRetryForAttempts returns a SendDecorator that retries a failed request for up to the specified
|
||||||
|
// number of attempts, exponentially backing off between requests using the supplied backoff
|
||||||
|
// time.Duration (which may be zero). Retrying may be canceled by closing the optional channel on
|
||||||
|
// the http.Request.
|
||||||
|
func DoRetryForAttempts(attempts int, backoff time.Duration) SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (resp *http.Response, err error) {
|
||||||
|
for attempt := 0; attempt < attempts; attempt++ {
|
||||||
|
resp, err = s.Do(r)
|
||||||
|
if err == nil {
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
DelayForBackoff(backoff, attempt, r.Cancel)
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoRetryForStatusCodes returns a SendDecorator that retries for specified statusCodes for up to the specified
|
||||||
|
// number of attempts, exponentially backing off between requests using the supplied backoff
|
||||||
|
// time.Duration (which may be zero). Retrying may be canceled by closing the optional channel on
|
||||||
|
// the http.Request.
|
||||||
|
func DoRetryForStatusCodes(attempts int, backoff time.Duration, codes ...int) SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (resp *http.Response, err error) {
|
||||||
|
b := []byte{}
|
||||||
|
if r.Body != nil {
|
||||||
|
b, err = ioutil.ReadAll(r.Body)
|
||||||
|
if err != nil {
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Increment to add the first call (attempts denotes number of retries)
|
||||||
|
attempts++
|
||||||
|
for attempt := 0; attempt < attempts; attempt++ {
|
||||||
|
r.Body = ioutil.NopCloser(bytes.NewBuffer(b))
|
||||||
|
resp, err = s.Do(r)
|
||||||
|
if err != nil || !ResponseHasStatusCode(resp, codes...) {
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
DelayForBackoff(backoff, attempt, r.Cancel)
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoRetryForDuration returns a SendDecorator that retries the request until the total time is equal
|
||||||
|
// to or greater than the specified duration, exponentially backing off between requests using the
|
||||||
|
// supplied backoff time.Duration (which may be zero). Retrying may be canceled by closing the
|
||||||
|
// optional channel on the http.Request.
|
||||||
|
func DoRetryForDuration(d time.Duration, backoff time.Duration) SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (resp *http.Response, err error) {
|
||||||
|
end := time.Now().Add(d)
|
||||||
|
for attempt := 0; time.Now().Before(end); attempt++ {
|
||||||
|
resp, err = s.Do(r)
|
||||||
|
if err == nil {
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
DelayForBackoff(backoff, attempt, r.Cancel)
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithLogging returns a SendDecorator that implements simple before and after logging of the
|
||||||
|
// request.
|
||||||
|
func WithLogging(logger *log.Logger) SendDecorator {
|
||||||
|
return func(s Sender) Sender {
|
||||||
|
return SenderFunc(func(r *http.Request) (*http.Response, error) {
|
||||||
|
logger.Printf("Sending %s %s", r.Method, r.URL)
|
||||||
|
resp, err := s.Do(r)
|
||||||
|
if err != nil {
|
||||||
|
logger.Printf("%s %s received error '%v'", r.Method, r.URL, err)
|
||||||
|
} else {
|
||||||
|
logger.Printf("%s %s received %s", r.Method, r.URL, resp.Status)
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// DelayForBackoff invokes time.After for the supplied backoff duration raised to the power of
|
||||||
|
// passed attempt (i.e., an exponential backoff delay). Backoff duration is in seconds and can set
|
||||||
|
// to zero for no delay. The delay may be canceled by closing the passed channel. If terminated early,
|
||||||
|
// returns false.
|
||||||
|
// Note: Passing attempt 1 will result in doubling "backoff" duration. Treat this as a zero-based attempt
|
||||||
|
// count.
|
||||||
|
func DelayForBackoff(backoff time.Duration, attempt int, cancel <-chan struct{}) bool {
|
||||||
|
select {
|
||||||
|
case <-time.After(time.Duration(backoff.Seconds()*math.Pow(2, float64(attempt))) * time.Second):
|
||||||
|
return true
|
||||||
|
case <-cancel:
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
178
vendor/github.com/Azure/go-autorest/autorest/utility.go
generated
vendored
Normal file
178
vendor/github.com/Azure/go-autorest/autorest/utility.go
generated
vendored
Normal file
@ -0,0 +1,178 @@
|
|||||||
|
package autorest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"encoding/xml"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"net/url"
|
||||||
|
"reflect"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// EncodedAs is a series of constants specifying various data encodings
|
||||||
|
type EncodedAs string
|
||||||
|
|
||||||
|
const (
|
||||||
|
// EncodedAsJSON states that data is encoded as JSON
|
||||||
|
EncodedAsJSON EncodedAs = "JSON"
|
||||||
|
|
||||||
|
// EncodedAsXML states that data is encoded as Xml
|
||||||
|
EncodedAsXML EncodedAs = "XML"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Decoder defines the decoding method json.Decoder and xml.Decoder share
|
||||||
|
type Decoder interface {
|
||||||
|
Decode(v interface{}) error
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewDecoder creates a new decoder appropriate to the passed encoding.
|
||||||
|
// encodedAs specifies the type of encoding and r supplies the io.Reader containing the
|
||||||
|
// encoded data.
|
||||||
|
func NewDecoder(encodedAs EncodedAs, r io.Reader) Decoder {
|
||||||
|
if encodedAs == EncodedAsJSON {
|
||||||
|
return json.NewDecoder(r)
|
||||||
|
} else if encodedAs == EncodedAsXML {
|
||||||
|
return xml.NewDecoder(r)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CopyAndDecode decodes the data from the passed io.Reader while making a copy. Having a copy
|
||||||
|
// is especially useful if there is a chance the data will fail to decode.
|
||||||
|
// encodedAs specifies the expected encoding, r provides the io.Reader to the data, and v
|
||||||
|
// is the decoding destination.
|
||||||
|
func CopyAndDecode(encodedAs EncodedAs, r io.Reader, v interface{}) (bytes.Buffer, error) {
|
||||||
|
b := bytes.Buffer{}
|
||||||
|
return b, NewDecoder(encodedAs, io.TeeReader(r, &b)).Decode(v)
|
||||||
|
}
|
||||||
|
|
||||||
|
// TeeReadCloser returns a ReadCloser that writes to w what it reads from rc.
|
||||||
|
// It utilizes io.TeeReader to copy the data read and has the same behavior when reading.
|
||||||
|
// Further, when it is closed, it ensures that rc is closed as well.
|
||||||
|
func TeeReadCloser(rc io.ReadCloser, w io.Writer) io.ReadCloser {
|
||||||
|
return &teeReadCloser{rc, io.TeeReader(rc, w)}
|
||||||
|
}
|
||||||
|
|
||||||
|
type teeReadCloser struct {
|
||||||
|
rc io.ReadCloser
|
||||||
|
r io.Reader
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *teeReadCloser) Read(p []byte) (int, error) {
|
||||||
|
return t.r.Read(p)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *teeReadCloser) Close() error {
|
||||||
|
return t.rc.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func containsInt(ints []int, n int) bool {
|
||||||
|
for _, i := range ints {
|
||||||
|
if i == n {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func escapeValueStrings(m map[string]string) map[string]string {
|
||||||
|
for key, value := range m {
|
||||||
|
m[key] = url.QueryEscape(value)
|
||||||
|
}
|
||||||
|
return m
|
||||||
|
}
|
||||||
|
|
||||||
|
func ensureValueStrings(mapOfInterface map[string]interface{}) map[string]string {
|
||||||
|
mapOfStrings := make(map[string]string)
|
||||||
|
for key, value := range mapOfInterface {
|
||||||
|
mapOfStrings[key] = ensureValueString(value)
|
||||||
|
}
|
||||||
|
return mapOfStrings
|
||||||
|
}
|
||||||
|
|
||||||
|
func ensureValueString(value interface{}) string {
|
||||||
|
if value == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
switch v := value.(type) {
|
||||||
|
case string:
|
||||||
|
return v
|
||||||
|
case []byte:
|
||||||
|
return string(v)
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("%v", v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MapToValues method converts map[string]interface{} to url.Values.
|
||||||
|
func MapToValues(m map[string]interface{}) url.Values {
|
||||||
|
v := url.Values{}
|
||||||
|
for key, value := range m {
|
||||||
|
x := reflect.ValueOf(value)
|
||||||
|
if x.Kind() == reflect.Array || x.Kind() == reflect.Slice {
|
||||||
|
for i := 0; i < x.Len(); i++ {
|
||||||
|
v.Add(key, ensureValueString(x.Index(i)))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
v.Add(key, ensureValueString(value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
|
||||||
|
// String method converts interface v to string. If interface is a list, it
|
||||||
|
// joins list elements using separator.
|
||||||
|
func String(v interface{}, sep ...string) string {
|
||||||
|
if len(sep) > 0 {
|
||||||
|
return ensureValueString(strings.Join(v.([]string), sep[0]))
|
||||||
|
}
|
||||||
|
return ensureValueString(v)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Encode method encodes url path and query parameters.
|
||||||
|
func Encode(location string, v interface{}, sep ...string) string {
|
||||||
|
s := String(v, sep...)
|
||||||
|
switch strings.ToLower(location) {
|
||||||
|
case "path":
|
||||||
|
return pathEscape(s)
|
||||||
|
case "query":
|
||||||
|
return queryEscape(s)
|
||||||
|
default:
|
||||||
|
return s
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func pathEscape(s string) string {
|
||||||
|
return strings.Replace(url.QueryEscape(s), "+", "%20", -1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func queryEscape(s string) string {
|
||||||
|
return url.QueryEscape(s)
|
||||||
|
}
|
||||||
|
|
||||||
|
// This method is same as Encode() method of "net/url" go package,
|
||||||
|
// except it does not encode the query parameters because they
|
||||||
|
// already come encoded. It formats values map in query format (bar=foo&a=b).
|
||||||
|
func createQuery(v url.Values) string {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
keys := make([]string, 0, len(v))
|
||||||
|
for k := range v {
|
||||||
|
keys = append(keys, k)
|
||||||
|
}
|
||||||
|
sort.Strings(keys)
|
||||||
|
for _, k := range keys {
|
||||||
|
vs := v[k]
|
||||||
|
prefix := url.QueryEscape(k) + "="
|
||||||
|
for _, v := range vs {
|
||||||
|
if buf.Len() > 0 {
|
||||||
|
buf.WriteByte('&')
|
||||||
|
}
|
||||||
|
buf.WriteString(prefix)
|
||||||
|
buf.WriteString(v)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return buf.String()
|
||||||
|
}
|
29
vendor/github.com/Azure/go-autorest/autorest/version.go
generated
vendored
Normal file
29
vendor/github.com/Azure/go-autorest/autorest/version.go
generated
vendored
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
package autorest
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
major = 7
|
||||||
|
minor = 3
|
||||||
|
patch = 1
|
||||||
|
tag = ""
|
||||||
|
)
|
||||||
|
|
||||||
|
var versionLock sync.Once
|
||||||
|
var version string
|
||||||
|
|
||||||
|
// Version returns the semantic version (see http://semver.org).
|
||||||
|
func Version() string {
|
||||||
|
versionLock.Do(func() {
|
||||||
|
version = fmt.Sprintf("v%d.%d.%d", major, minor, patch)
|
||||||
|
|
||||||
|
if trimmed := strings.TrimPrefix(tag, "-"); trimmed != "" {
|
||||||
|
version = fmt.Sprintf("%s-%s", version, trimmed)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return version
|
||||||
|
}
|
6
vendor/vendor.json
vendored
6
vendor/vendor.json
vendored
@ -26,6 +26,12 @@
|
|||||||
"revision": "66ede4d838ea4808df6144568e8ea7660967010c",
|
"revision": "66ede4d838ea4808df6144568e8ea7660967010c",
|
||||||
"revisionTime": "2017-01-06T22:37:36Z"
|
"revisionTime": "2017-01-06T22:37:36Z"
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"checksumSHA1": "U2+FgaMOPEFg/yHLD5RbiXI1cq4=",
|
||||||
|
"path": "github.com/Azure/go-autorest/autorest",
|
||||||
|
"revision": "c2571aa487482a987546c123fef9334582786335",
|
||||||
|
"revisionTime": "2017-04-20T22:41:48Z"
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"checksumSHA1": "JhyS/zIicgtrSasHSZ6WtXGWJVk=",
|
"checksumSHA1": "JhyS/zIicgtrSasHSZ6WtXGWJVk=",
|
||||||
"path": "github.com/DataDog/datadog-go/statsd",
|
"path": "github.com/DataDog/datadog-go/statsd",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user