mirror of
https://github.com/rocky-linux/peridot.git
synced 2024-11-27 23:46:25 +00:00
173 lines
5.0 KiB
Go
173 lines
5.0 KiB
Go
|
// Copyright 2016 The Go Authors. All rights reserved.
|
||
|
// Use of this source code is governed by a BSD-style
|
||
|
// license that can be found in the LICENSE file.
|
||
|
|
||
|
package gensupport
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"encoding/json"
|
||
|
"errors"
|
||
|
"net/http"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
// Hook is the type of a function that is called once before each HTTP request
|
||
|
// that is sent by a generated API. It returns a function that is called after
|
||
|
// the request returns.
|
||
|
// Hooks are not called if the context is nil.
|
||
|
type Hook func(ctx context.Context, req *http.Request) func(resp *http.Response)
|
||
|
|
||
|
var hooks []Hook
|
||
|
|
||
|
// RegisterHook registers a Hook to be called before each HTTP request by a
|
||
|
// generated API. Hooks are called in the order they are registered. Each
|
||
|
// hook can return a function; if it is non-nil, it is called after the HTTP
|
||
|
// request returns. These functions are called in the reverse order.
|
||
|
// RegisterHook should not be called concurrently with itself or SendRequest.
|
||
|
func RegisterHook(h Hook) {
|
||
|
hooks = append(hooks, h)
|
||
|
}
|
||
|
|
||
|
// SendRequest sends a single HTTP request using the given client.
|
||
|
// If ctx is non-nil, it calls all hooks, then sends the request with
|
||
|
// req.WithContext, then calls any functions returned by the hooks in
|
||
|
// reverse order.
|
||
|
func SendRequest(ctx context.Context, client *http.Client, req *http.Request) (*http.Response, error) {
|
||
|
// Disallow Accept-Encoding because it interferes with the automatic gzip handling
|
||
|
// done by the default http.Transport. See https://github.com/google/google-api-go-client/issues/219.
|
||
|
if _, ok := req.Header["Accept-Encoding"]; ok {
|
||
|
return nil, errors.New("google api: custom Accept-Encoding headers not allowed")
|
||
|
}
|
||
|
if ctx == nil {
|
||
|
return client.Do(req)
|
||
|
}
|
||
|
// Call hooks in order of registration, store returned funcs.
|
||
|
post := make([]func(resp *http.Response), len(hooks))
|
||
|
for i, h := range hooks {
|
||
|
fn := h(ctx, req)
|
||
|
post[i] = fn
|
||
|
}
|
||
|
|
||
|
// Send request.
|
||
|
resp, err := send(ctx, client, req)
|
||
|
|
||
|
// Call returned funcs in reverse order.
|
||
|
for i := len(post) - 1; i >= 0; i-- {
|
||
|
if fn := post[i]; fn != nil {
|
||
|
fn(resp)
|
||
|
}
|
||
|
}
|
||
|
return resp, err
|
||
|
}
|
||
|
|
||
|
func send(ctx context.Context, client *http.Client, req *http.Request) (*http.Response, error) {
|
||
|
if client == nil {
|
||
|
client = http.DefaultClient
|
||
|
}
|
||
|
resp, err := client.Do(req.WithContext(ctx))
|
||
|
// If we got an error, and the context has been canceled,
|
||
|
// the context's error is probably more useful.
|
||
|
if err != nil {
|
||
|
select {
|
||
|
case <-ctx.Done():
|
||
|
err = ctx.Err()
|
||
|
default:
|
||
|
}
|
||
|
}
|
||
|
return resp, err
|
||
|
}
|
||
|
|
||
|
// SendRequestWithRetry sends a single HTTP request using the given client,
|
||
|
// with retries if a retryable error is returned.
|
||
|
// If ctx is non-nil, it calls all hooks, then sends the request with
|
||
|
// req.WithContext, then calls any functions returned by the hooks in
|
||
|
// reverse order.
|
||
|
func SendRequestWithRetry(ctx context.Context, client *http.Client, req *http.Request) (*http.Response, error) {
|
||
|
// Disallow Accept-Encoding because it interferes with the automatic gzip handling
|
||
|
// done by the default http.Transport. See https://github.com/google/google-api-go-client/issues/219.
|
||
|
if _, ok := req.Header["Accept-Encoding"]; ok {
|
||
|
return nil, errors.New("google api: custom Accept-Encoding headers not allowed")
|
||
|
}
|
||
|
if ctx == nil {
|
||
|
return client.Do(req)
|
||
|
}
|
||
|
// Call hooks in order of registration, store returned funcs.
|
||
|
post := make([]func(resp *http.Response), len(hooks))
|
||
|
for i, h := range hooks {
|
||
|
fn := h(ctx, req)
|
||
|
post[i] = fn
|
||
|
}
|
||
|
|
||
|
// Send request with retry.
|
||
|
resp, err := sendAndRetry(ctx, client, req)
|
||
|
|
||
|
// Call returned funcs in reverse order.
|
||
|
for i := len(post) - 1; i >= 0; i-- {
|
||
|
if fn := post[i]; fn != nil {
|
||
|
fn(resp)
|
||
|
}
|
||
|
}
|
||
|
return resp, err
|
||
|
}
|
||
|
|
||
|
func sendAndRetry(ctx context.Context, client *http.Client, req *http.Request) (*http.Response, error) {
|
||
|
if client == nil {
|
||
|
client = http.DefaultClient
|
||
|
}
|
||
|
|
||
|
var resp *http.Response
|
||
|
var err error
|
||
|
|
||
|
// Loop to retry the request, up to the context deadline.
|
||
|
var pause time.Duration
|
||
|
bo := backoff()
|
||
|
|
||
|
for {
|
||
|
select {
|
||
|
case <-ctx.Done():
|
||
|
// If we got an error, and the context has been canceled,
|
||
|
// the context's error is probably more useful.
|
||
|
if err == nil {
|
||
|
err = ctx.Err()
|
||
|
}
|
||
|
return resp, err
|
||
|
case <-time.After(pause):
|
||
|
}
|
||
|
|
||
|
resp, err = client.Do(req.WithContext(ctx))
|
||
|
|
||
|
var status int
|
||
|
if resp != nil {
|
||
|
status = resp.StatusCode
|
||
|
}
|
||
|
|
||
|
// Check if we can retry the request. A retry can only be done if the error
|
||
|
// is retryable and the request body can be re-created using GetBody (this
|
||
|
// will not be possible if the body was unbuffered).
|
||
|
if req.GetBody == nil || !shouldRetry(status, err) {
|
||
|
break
|
||
|
}
|
||
|
var errBody error
|
||
|
req.Body, errBody = req.GetBody()
|
||
|
if errBody != nil {
|
||
|
break
|
||
|
}
|
||
|
|
||
|
pause = bo.Pause()
|
||
|
if resp != nil && resp.Body != nil {
|
||
|
resp.Body.Close()
|
||
|
}
|
||
|
}
|
||
|
return resp, err
|
||
|
}
|
||
|
|
||
|
// DecodeResponse decodes the body of res into target. If there is no body,
|
||
|
// target is unchanged.
|
||
|
func DecodeResponse(target interface{}, res *http.Response) error {
|
||
|
if res.StatusCode == http.StatusNoContent {
|
||
|
return nil
|
||
|
}
|
||
|
return json.NewDecoder(res.Body).Decode(target)
|
||
|
}
|