2014-05-22 14:37:19 -04:00
|
|
|
package pkg
|
2014-05-21 13:13:20 -04:00
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/tls"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
|
|
|
"math"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
neturl "net/url"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
HTTP_2xx = 2
|
|
|
|
HTTP_4xx = 4
|
|
|
|
)
|
|
|
|
|
|
|
|
type HttpClient struct {
|
|
|
|
// Maximum exp backoff duration. Defaults to 5 seconds
|
|
|
|
MaxBackoff time.Duration
|
|
|
|
|
2014-05-22 14:37:19 -04:00
|
|
|
// Maximum number of connection retries. Defaults to 15
|
2014-05-21 13:13:20 -04:00
|
|
|
MaxRetries int
|
|
|
|
|
|
|
|
// HTTP client timeout, this is suggested to be low since exponential
|
|
|
|
// backoff will kick off too. Defaults to 2 seconds
|
|
|
|
Timeout time.Duration
|
|
|
|
|
2014-05-22 14:37:19 -04:00
|
|
|
// Whether or not to skip TLS verification. Defaults to false
|
2014-05-21 13:13:20 -04:00
|
|
|
SkipTLS bool
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewHttpClient() *HttpClient {
|
|
|
|
return &HttpClient{
|
|
|
|
MaxBackoff: time.Second * 5,
|
|
|
|
MaxRetries: 15,
|
|
|
|
Timeout: time.Duration(2) * time.Second,
|
|
|
|
SkipTLS: false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fetches a given URL with support for exponential backoff and maximum retries
|
|
|
|
func (h *HttpClient) Get(rawurl string) ([]byte, error) {
|
|
|
|
if rawurl == "" {
|
|
|
|
return nil, errors.New("URL is empty. Skipping.")
|
|
|
|
}
|
|
|
|
|
|
|
|
url, err := neturl.Parse(rawurl)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unfortunately, url.Parse is too generic to throw errors if a URL does not
|
|
|
|
// have a valid HTTP scheme. So, we have to do this extra validation
|
|
|
|
if !strings.HasPrefix(url.Scheme, "http") {
|
|
|
|
return nil, fmt.Errorf("URL %s does not have a valid HTTP scheme. Skipping.", rawurl)
|
|
|
|
}
|
|
|
|
|
|
|
|
dataURL := url.String()
|
|
|
|
|
|
|
|
// We need to create our own client in order to add timeout support.
|
|
|
|
// TODO(c4milo) Replace it once Go 1.3 is officially used by CoreOS
|
|
|
|
// More info: https://code.google.com/p/go/source/detail?r=ada6f2d5f99f
|
|
|
|
transport := &http.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
InsecureSkipVerify: h.SkipTLS,
|
|
|
|
},
|
|
|
|
Dial: func(network, addr string) (net.Conn, error) {
|
|
|
|
deadline := time.Now().Add(h.Timeout)
|
|
|
|
c, err := net.DialTimeout(network, addr, h.Timeout)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
c.SetDeadline(deadline)
|
|
|
|
return c, nil
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
client := &http.Client{
|
|
|
|
Transport: transport,
|
|
|
|
}
|
|
|
|
|
|
|
|
for retry := 1; retry <= h.MaxRetries; retry++ {
|
|
|
|
log.Printf("Fetching data from %s. Attempt #%d", dataURL, retry)
|
|
|
|
|
|
|
|
resp, err := client.Get(dataURL)
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
defer resp.Body.Close()
|
|
|
|
status := resp.StatusCode / 100
|
|
|
|
|
|
|
|
if status == HTTP_2xx {
|
|
|
|
return ioutil.ReadAll(resp.Body)
|
|
|
|
}
|
|
|
|
|
|
|
|
if status == HTTP_4xx {
|
|
|
|
return nil, fmt.Errorf("Not found. HTTP status code: %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Server error. HTTP status code: %d", resp.StatusCode)
|
|
|
|
} else {
|
|
|
|
log.Printf("Unable to fetch data: %s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
duration := time.Millisecond * time.Duration((math.Pow(float64(2), float64(retry)) * 100))
|
|
|
|
if duration > h.MaxBackoff {
|
|
|
|
duration = h.MaxBackoff
|
|
|
|
}
|
|
|
|
|
2014-05-22 14:53:54 -04:00
|
|
|
log.Printf("Sleeping for %v...", duration)
|
2014-05-22 14:37:19 -04:00
|
|
|
|
2014-05-21 13:13:20 -04:00
|
|
|
time.Sleep(duration)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, fmt.Errorf("Unable to fetch data. Maximum retries reached: %d", h.MaxRetries)
|
|
|
|
}
|