mirror of
https://github.com/opentofu/opentofu.git
synced 2025-01-16 11:42:58 -06:00
6fe2703665
* Remove `make updatedeps` from Travis build. We'll follow up with more specific plans around dependency updating in subsequent PRs. * Update all `make` targets to set `GO15VENDOREXPERIMENT=1` and to filter out `/vendor/` from `./...` where appropriate. * Temporarily remove `vet` from the `make test` target until we can figure out how to get it to not vet `vendor/`. (Initial experimentation failed to yield the proper incantation.) Everything is pinned to current master, with the exception of: * Azure/azure-sdk-for-go which is pinned before the breaking change today * aws/aws-sdk-go which is pinned to the most recent tag The documentation still needs to be updated, which we can do in a follow up PR. The goal here is to unblock release.
171 lines
3.2 KiB
Go
171 lines
3.2 KiB
Go
package statuscake
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/url"
|
|
"strings"
|
|
)
|
|
|
|
const apiBaseURL = "https://www.statuscake.com/API"
|
|
|
|
type responseBody struct {
|
|
io.Reader
|
|
}
|
|
|
|
func (r *responseBody) Close() error {
|
|
return nil
|
|
}
|
|
|
|
// Auth wraps the authorisation headers required for each request
|
|
type Auth struct {
|
|
Username string
|
|
Apikey string
|
|
}
|
|
|
|
func (a *Auth) validate() error {
|
|
e := make(ValidationError)
|
|
|
|
if a.Username == "" {
|
|
e["Username"] = "is required"
|
|
}
|
|
|
|
if a.Apikey == "" {
|
|
e["Apikey"] = "is required"
|
|
}
|
|
|
|
if len(e) > 0 {
|
|
return e
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type httpClient interface {
|
|
Do(*http.Request) (*http.Response, error)
|
|
}
|
|
|
|
type apiClient interface {
|
|
get(string, url.Values) (*http.Response, error)
|
|
delete(string, url.Values) (*http.Response, error)
|
|
put(string, url.Values) (*http.Response, error)
|
|
}
|
|
|
|
// Client is the http client that wraps the remote API.
|
|
type Client struct {
|
|
c httpClient
|
|
username string
|
|
apiKey string
|
|
testsClient Tests
|
|
}
|
|
|
|
// New returns a new Client
|
|
func New(auth Auth) (*Client, error) {
|
|
if err := auth.validate(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &Client{
|
|
c: &http.Client{},
|
|
username: auth.Username,
|
|
apiKey: auth.Apikey,
|
|
}, nil
|
|
}
|
|
|
|
func (c *Client) newRequest(method string, path string, v url.Values, body io.Reader) (*http.Request, error) {
|
|
url := fmt.Sprintf("%s%s", apiBaseURL, path)
|
|
if v != nil {
|
|
url = fmt.Sprintf("%s?%s", url, v.Encode())
|
|
}
|
|
|
|
r, err := http.NewRequest(method, url, body)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
r.Header.Set("Username", c.username)
|
|
r.Header.Set("API", c.apiKey)
|
|
|
|
return r, nil
|
|
}
|
|
|
|
func (c *Client) doRequest(r *http.Request) (*http.Response, error) {
|
|
resp, err := c.c.Do(r)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode < 200 || resp.StatusCode > 299 {
|
|
return nil, &httpError{
|
|
status: resp.Status,
|
|
statusCode: resp.StatusCode,
|
|
}
|
|
}
|
|
|
|
var aer autheticationErrorResponse
|
|
|
|
// We read and save the response body so that if we don't have error messages
|
|
// we can set it again for future usage
|
|
b, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
err = json.Unmarshal(b, &aer)
|
|
if err == nil && aer.ErrNo == 0 && aer.Error != "" {
|
|
return nil, &AuthenticationError{
|
|
errNo: aer.ErrNo,
|
|
message: aer.Error,
|
|
}
|
|
}
|
|
|
|
resp.Body = &responseBody{
|
|
Reader: bytes.NewReader(b),
|
|
}
|
|
|
|
return resp, nil
|
|
}
|
|
|
|
func (c *Client) get(path string, v url.Values) (*http.Response, error) {
|
|
r, err := c.newRequest("GET", path, v, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c.doRequest(r)
|
|
}
|
|
|
|
func (c *Client) put(path string, v url.Values) (*http.Response, error) {
|
|
r, err := c.newRequest("PUT", path, nil, strings.NewReader(v.Encode()))
|
|
r.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c.doRequest(r)
|
|
}
|
|
|
|
func (c *Client) delete(path string, v url.Values) (*http.Response, error) {
|
|
r, err := c.newRequest("DELETE", path, v, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return c.doRequest(r)
|
|
}
|
|
|
|
// Tests returns a client that implements the `Tests` API.
|
|
func (c *Client) Tests() Tests {
|
|
if c.testsClient == nil {
|
|
c.testsClient = newTests(c)
|
|
}
|
|
|
|
return c.testsClient
|
|
}
|