2015-12-30 22:01:21 +00:00
|
|
|
package acme
|
|
|
|
|
|
|
|
import (
|
2016-01-08 09:04:57 +00:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
2015-12-30 22:01:21 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"runtime"
|
|
|
|
"strings"
|
2016-02-14 06:24:19 +00:00
|
|
|
"time"
|
2015-12-30 22:01:21 +00:00
|
|
|
)
|
|
|
|
|
2016-01-27 00:57:55 +00:00
|
|
|
// UserAgent (if non-empty) will be tacked onto the User-Agent string in requests.
|
2015-12-30 22:01:21 +00:00
|
|
|
var UserAgent string
|
|
|
|
|
2016-02-14 06:24:19 +00:00
|
|
|
// defaultClient is an HTTP client with a reasonable timeout value.
|
|
|
|
var defaultClient = http.Client{Timeout: 10 * time.Second}
|
|
|
|
|
2015-12-30 22:01:21 +00:00
|
|
|
const (
|
|
|
|
// defaultGoUserAgent is the Go HTTP package user agent string. Too
|
|
|
|
// bad it isn't exported. If it changes, we should update it here, too.
|
|
|
|
defaultGoUserAgent = "Go-http-client/1.1"
|
|
|
|
|
|
|
|
// ourUserAgent is the User-Agent of this underlying library package.
|
|
|
|
ourUserAgent = "xenolf-acme"
|
|
|
|
)
|
|
|
|
|
|
|
|
// httpHead performs a HEAD request with a proper User-Agent string.
|
|
|
|
// The response body (resp.Body) is already closed when this function returns.
|
|
|
|
func httpHead(url string) (resp *http.Response, err error) {
|
|
|
|
req, err := http.NewRequest("HEAD", url, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
req.Header.Set("User-Agent", userAgent())
|
|
|
|
|
2016-02-14 06:24:19 +00:00
|
|
|
resp, err = defaultClient.Do(req)
|
2016-02-06 05:41:15 +00:00
|
|
|
if err != nil {
|
|
|
|
return resp, err
|
2015-12-30 22:01:21 +00:00
|
|
|
}
|
2016-02-06 05:41:15 +00:00
|
|
|
resp.Body.Close()
|
2015-12-30 22:01:21 +00:00
|
|
|
return resp, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// httpPost performs a POST request with a proper User-Agent string.
|
|
|
|
// Callers should close resp.Body when done reading from it.
|
|
|
|
func httpPost(url string, bodyType string, body io.Reader) (resp *http.Response, err error) {
|
|
|
|
req, err := http.NewRequest("POST", url, body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
req.Header.Set("Content-Type", bodyType)
|
|
|
|
req.Header.Set("User-Agent", userAgent())
|
|
|
|
|
2016-02-14 06:24:19 +00:00
|
|
|
return defaultClient.Do(req)
|
2015-12-30 22:01:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// httpGet performs a GET request with a proper User-Agent string.
|
|
|
|
// Callers should close resp.Body when done reading from it.
|
|
|
|
func httpGet(url string) (resp *http.Response, err error) {
|
|
|
|
req, err := http.NewRequest("GET", url, nil)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
req.Header.Set("User-Agent", userAgent())
|
|
|
|
|
2016-02-14 06:24:19 +00:00
|
|
|
return defaultClient.Do(req)
|
2015-12-30 22:01:21 +00:00
|
|
|
}
|
|
|
|
|
2016-01-08 09:04:57 +00:00
|
|
|
// getJSON performs an HTTP GET request and parses the response body
|
|
|
|
// as JSON, into the provided respBody object.
|
|
|
|
func getJSON(uri string, respBody interface{}) (http.Header, error) {
|
|
|
|
resp, err := httpGet(uri)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to get %q: %v", uri, err)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode >= http.StatusBadRequest {
|
|
|
|
return resp.Header, handleHTTPError(resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp.Header, json.NewDecoder(resp.Body).Decode(respBody)
|
|
|
|
}
|
|
|
|
|
|
|
|
// postJSON performs an HTTP POST request and parses the response body
|
|
|
|
// as JSON, into the provided respBody object.
|
|
|
|
func postJSON(j *jws, uri string, reqBody, respBody interface{}) (http.Header, error) {
|
|
|
|
jsonBytes, err := json.Marshal(reqBody)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("Failed to marshal network message...")
|
|
|
|
}
|
|
|
|
|
2016-04-14 02:33:26 +00:00
|
|
|
logf("[DEBUG] Attempting to post %s to %s", string(jsonBytes), uri)
|
|
|
|
|
2016-01-08 09:04:57 +00:00
|
|
|
resp, err := j.post(uri, jsonBytes)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("Failed to post JWS message. -> %v", err)
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode >= http.StatusBadRequest {
|
|
|
|
return resp.Header, handleHTTPError(resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
if respBody == nil {
|
|
|
|
return resp.Header, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp.Header, json.NewDecoder(resp.Body).Decode(respBody)
|
|
|
|
}
|
|
|
|
|
2015-12-30 22:01:21 +00:00
|
|
|
// userAgent builds and returns the User-Agent string to use in requests.
|
|
|
|
func userAgent() string {
|
|
|
|
ua := fmt.Sprintf("%s (%s; %s) %s %s", defaultGoUserAgent, runtime.GOOS, runtime.GOARCH, ourUserAgent, UserAgent)
|
|
|
|
return strings.TrimSpace(ua)
|
|
|
|
}
|