etcd/proxy/director.go
Brian Waldon a155f0bda6 proxy: rewrite stdlib ReverseProxy
The ReverseProxy code from the standard library doesn't actually
give us the control that we want. Pull it down and rip out what
we don't need, adding tests in the process.

All available endpoints are attempted when proxying a request. If a
proxied request fails, the upstream will be considered unavailable
for 5s and no more requests will be proxied to it. After the 5s is
up, the endpoint will be put back to rotation.
2014-09-12 22:56:23 -07:00

107 lines
1.9 KiB
Go

package proxy
import (
"errors"
"fmt"
"log"
"net/url"
"sync"
"time"
)
const (
// amount of time an endpoint will be held in a failed
// state before being reconsidered for proxied requests
endpointFailureWait = 5 * time.Second
)
func newDirector(urls []string) (*director, error) {
if len(urls) == 0 {
return nil, errors.New("one or more endpoints required")
}
endpoints := make([]*endpoint, len(urls))
for i, v := range urls {
u, err := url.Parse(v)
if err != nil {
return nil, fmt.Errorf("invalid endpoint %q: %v", v, err)
}
if u.Scheme == "" {
return nil, fmt.Errorf("invalid endpoint %q: scheme required", v)
}
if u.Host == "" {
return nil, fmt.Errorf("invalid endpoint %q: host empty", v)
}
endpoints[i] = newEndpoint(*u)
}
d := director{ep: endpoints}
return &d, nil
}
type director struct {
ep []*endpoint
}
func (d *director) endpoints() []*endpoint {
filtered := make([]*endpoint, 0)
for _, ep := range d.ep {
if ep.Available {
filtered = append(filtered, ep)
}
}
return filtered
}
func newEndpoint(u url.URL) *endpoint {
ep := endpoint{
URL: u,
Available: true,
failFunc: timedUnavailabilityFunc(endpointFailureWait),
}
return &ep
}
type endpoint struct {
sync.Mutex
URL url.URL
Available bool
failFunc func(ep *endpoint)
}
func (ep *endpoint) Failed() {
ep.Lock()
if !ep.Available {
ep.Unlock()
return
}
ep.Available = false
ep.Unlock()
log.Printf("proxy: marked endpoint %s unavailable", ep.URL.String())
if ep.failFunc == nil {
log.Printf("proxy: no failFunc defined, endpoint %s will be unavailable forever.", ep.URL.String())
return
}
ep.failFunc(ep)
}
func timedUnavailabilityFunc(wait time.Duration) func(*endpoint) {
return func(ep *endpoint) {
time.AfterFunc(wait, func() {
ep.Available = true
log.Printf("proxy: marked endpoint %s available", ep.URL.String())
})
}
}