mirror of
https://github.com/oauth2-proxy/oauth2-proxy.git
synced 2025-01-24 05:26:55 +02:00
174 lines
4.5 KiB
Go
174 lines
4.5 KiB
Go
package requests
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/http"
|
|
|
|
"github.com/bitly/go-simplejson"
|
|
)
|
|
|
|
// Builder allows users to construct a request and then either get the requests
|
|
// response via Do(), parse the response into a simplejson.Json via JSON(),
|
|
// or to parse the json response into an object via UnmarshalInto().
|
|
type Builder interface {
|
|
WithContext(context.Context) Builder
|
|
WithBody(io.Reader) Builder
|
|
WithMethod(string) Builder
|
|
WithHeaders(http.Header) Builder
|
|
SetHeader(key, value string) Builder
|
|
Do() (*http.Response, error)
|
|
UnmarshalInto(interface{}) error
|
|
UnmarshalJSON() (*simplejson.Json, error)
|
|
}
|
|
|
|
type builder struct {
|
|
context context.Context
|
|
method string
|
|
endpoint string
|
|
body io.Reader
|
|
header http.Header
|
|
response *http.Response
|
|
}
|
|
|
|
// New provides a new Builder for the given endpoint.
|
|
func New(endpoint string) Builder {
|
|
return &builder{
|
|
endpoint: endpoint,
|
|
method: "GET",
|
|
}
|
|
}
|
|
|
|
// WithContext adds a context to the request.
|
|
// If no context is provided, context.Background() is used instead.
|
|
func (r *builder) WithContext(ctx context.Context) Builder {
|
|
r.context = ctx
|
|
return r
|
|
}
|
|
|
|
// WithBody adds a body to the request.
|
|
func (r *builder) WithBody(body io.Reader) Builder {
|
|
r.body = body
|
|
return r
|
|
}
|
|
|
|
// WithMethod sets the request method. Defaults to "GET".
|
|
func (r *builder) WithMethod(method string) Builder {
|
|
r.method = method
|
|
return r
|
|
}
|
|
|
|
// WithHeaders replaces the request header map with the given header map.
|
|
func (r *builder) WithHeaders(header http.Header) Builder {
|
|
r.header = header
|
|
return r
|
|
}
|
|
|
|
// SetHeader sets a single header to the given value.
|
|
// May be used to add multiple headers.
|
|
func (r *builder) SetHeader(key, value string) Builder {
|
|
if r.header == nil {
|
|
r.header = make(http.Header)
|
|
}
|
|
r.header.Set(key, value)
|
|
return r
|
|
}
|
|
|
|
// Do performs the request and returns the response in its raw form.
|
|
// If the request has already been performed, returns the previous result.
|
|
// This will not allow you to repeat a request.
|
|
func (r *builder) Do() (*http.Response, error) {
|
|
if r.response != nil {
|
|
// Request has already been done
|
|
return r.response, nil
|
|
}
|
|
|
|
// Must provide a non-nil context to NewRequestWithContext
|
|
if r.context == nil {
|
|
r.context = context.Background()
|
|
}
|
|
|
|
req, err := http.NewRequestWithContext(r.context, r.method, r.endpoint, r.body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error creating request: %v", err)
|
|
}
|
|
req.Header = r.header
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error performing request: %v", err)
|
|
}
|
|
|
|
r.response = resp
|
|
return resp, nil
|
|
}
|
|
|
|
// UnmarshalInto performs the request and attempts to unmarshal the response into the
|
|
// the given interface. The response body is assumed to be JSON.
|
|
// The response must have a 200 status otherwise an error will be returned.
|
|
func (r *builder) UnmarshalInto(into interface{}) error {
|
|
resp, err := r.Do()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return UnmarshalInto(resp, into)
|
|
}
|
|
|
|
// UnmarshalJSON performs the request and attempts to unmarshal the response into a
|
|
// simplejson.Json. The response body is assume to be JSON.
|
|
// The response must have a 200 status otherwise an error will be returned.
|
|
func (r *builder) UnmarshalJSON() (*simplejson.Json, error) {
|
|
resp, err := r.Do()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
body, err := getResponseBody(resp)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
data, err := simplejson.NewJson(body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error reading json: %v", err)
|
|
}
|
|
return data, nil
|
|
}
|
|
|
|
// UnmarshalInto attempts to unmarshal the response into the the given interface.
|
|
// The response body is assumed to be JSON.
|
|
// The response must have a 200 status otherwise an error will be returned.
|
|
func UnmarshalInto(resp *http.Response, into interface{}) error {
|
|
body, err := getResponseBody(resp)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := json.Unmarshal(body, into); err != nil {
|
|
return fmt.Errorf("error unmarshalling body: %v", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// getResponseBody extracts the response body, but will only return the body
|
|
// if the response was successful.
|
|
func getResponseBody(resp *http.Response) ([]byte, error) {
|
|
defer resp.Body.Close()
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error reading response body: %v", err)
|
|
}
|
|
|
|
// Only unmarshal body if the response was successful
|
|
if resp.StatusCode != http.StatusOK {
|
|
return nil, fmt.Errorf("unexpected status \"%d\": %s", resp.StatusCode, body)
|
|
}
|
|
|
|
return body, nil
|
|
}
|