2016-06-05 22:51:56 -05:00
|
|
|
package proxy
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"os"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
var workableServer *httptest.Server
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
|
|
|
workableServer = httptest.NewServer(http.HandlerFunc(
|
|
|
|
func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// do nothing
|
|
|
|
}))
|
|
|
|
r := m.Run()
|
|
|
|
workableServer.Close()
|
|
|
|
os.Exit(r)
|
|
|
|
}
|
|
|
|
|
|
|
|
type customPolicy struct{}
|
|
|
|
|
2016-07-31 04:04:54 -05:00
|
|
|
func (r *customPolicy) Select(pool HostPool, request *http.Request) *UpstreamHost {
|
2016-06-05 22:51:56 -05:00
|
|
|
return pool[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
func testPool() HostPool {
|
|
|
|
pool := []*UpstreamHost{
|
|
|
|
{
|
|
|
|
Name: workableServer.URL, // this should resolve (healthcheck test)
|
|
|
|
},
|
|
|
|
{
|
2016-06-29 18:52:31 -05:00
|
|
|
Name: "http://localhost:99998", // this shouldn't
|
2016-06-05 22:51:56 -05:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "http://C",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
return HostPool(pool)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRoundRobinPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
|
|
|
rrPolicy := &RoundRobin{}
|
2016-07-31 04:04:54 -05:00
|
|
|
request, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
|
|
|
|
h := rrPolicy.Select(pool, request)
|
2016-06-05 22:51:56 -05:00
|
|
|
// First selected host is 1, because counter starts at 0
|
|
|
|
// and increments before host is selected
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected first round robin host to be second host in the pool.")
|
|
|
|
}
|
2016-07-31 04:04:54 -05:00
|
|
|
h = rrPolicy.Select(pool, request)
|
2016-06-05 22:51:56 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected second round robin host to be third host in the pool.")
|
|
|
|
}
|
2016-07-31 04:04:54 -05:00
|
|
|
h = rrPolicy.Select(pool, request)
|
2016-06-05 22:51:56 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected third round robin host to be first host in the pool.")
|
|
|
|
}
|
|
|
|
// mark host as down
|
2017-02-15 10:09:42 -05:00
|
|
|
pool[1].Unhealthy = 1
|
2016-07-31 04:04:54 -05:00
|
|
|
h = rrPolicy.Select(pool, request)
|
2016-06-05 22:51:56 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected to skip down host.")
|
|
|
|
}
|
2016-06-14 16:43:06 -05:00
|
|
|
// mark host as up
|
2017-02-15 10:09:42 -05:00
|
|
|
pool[1].Unhealthy = 0
|
2016-06-14 16:43:06 -05:00
|
|
|
|
2016-07-31 04:04:54 -05:00
|
|
|
h = rrPolicy.Select(pool, request)
|
2016-06-14 16:43:06 -05:00
|
|
|
if h == pool[2] {
|
|
|
|
t.Error("Expected to balance evenly among healthy hosts")
|
|
|
|
}
|
2016-06-05 22:51:56 -05:00
|
|
|
// mark host as full
|
2016-06-14 16:43:06 -05:00
|
|
|
pool[1].Conns = 1
|
|
|
|
pool[1].MaxConns = 1
|
2016-07-31 04:04:54 -05:00
|
|
|
h = rrPolicy.Select(pool, request)
|
2016-06-14 16:43:06 -05:00
|
|
|
if h != pool[2] {
|
2016-06-05 22:51:56 -05:00
|
|
|
t.Error("Expected to skip full host.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLeastConnPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
|
|
|
lcPolicy := &LeastConn{}
|
2016-07-31 04:04:54 -05:00
|
|
|
request, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
|
2016-06-05 22:51:56 -05:00
|
|
|
pool[0].Conns = 10
|
|
|
|
pool[1].Conns = 10
|
2016-07-31 04:04:54 -05:00
|
|
|
h := lcPolicy.Select(pool, request)
|
2016-06-05 22:51:56 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected least connection host to be third host.")
|
|
|
|
}
|
|
|
|
pool[2].Conns = 100
|
2016-07-31 04:04:54 -05:00
|
|
|
h = lcPolicy.Select(pool, request)
|
2016-06-05 22:51:56 -05:00
|
|
|
if h != pool[0] && h != pool[1] {
|
|
|
|
t.Error("Expected least connection host to be first or second host.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCustomPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
|
|
|
customPolicy := &customPolicy{}
|
2016-07-31 04:04:54 -05:00
|
|
|
request, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
|
|
|
|
h := customPolicy.Select(pool, request)
|
2016-06-05 22:51:56 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected custom policy host to be the first host.")
|
|
|
|
}
|
|
|
|
}
|
2016-07-31 04:04:54 -05:00
|
|
|
|
|
|
|
func TestIPHashPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
|
|
|
ipHash := &IPHash{}
|
|
|
|
request, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
// We should be able to predict where every request is routed.
|
|
|
|
request.RemoteAddr = "172.0.0.1:80"
|
|
|
|
h := ipHash.Select(pool, request)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.2:80"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.3:80"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected ip hash policy host to be the third host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.4:80"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// we should get the same results without a port
|
|
|
|
request.RemoteAddr = "172.0.0.1"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.2"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.3"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected ip hash policy host to be the third host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.4"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// we should get a healthy host if the original host is unhealthy and a
|
|
|
|
// healthy host is available
|
|
|
|
request.RemoteAddr = "172.0.0.1"
|
2017-02-15 10:09:42 -05:00
|
|
|
pool[1].Unhealthy = 1
|
2016-07-31 04:04:54 -05:00
|
|
|
h = ipHash.Select(pool, request)
|
2016-09-27 16:38:20 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected ip hash policy host to be the third host.")
|
2016-07-31 04:04:54 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
request.RemoteAddr = "172.0.0.2"
|
|
|
|
h = ipHash.Select(pool, request)
|
2016-09-27 16:38:20 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected ip hash policy host to be the third host.")
|
2016-07-31 04:04:54 -05:00
|
|
|
}
|
2017-02-15 10:09:42 -05:00
|
|
|
pool[1].Unhealthy = 0
|
2016-07-31 04:04:54 -05:00
|
|
|
|
|
|
|
request.RemoteAddr = "172.0.0.3"
|
2017-02-15 10:09:42 -05:00
|
|
|
pool[2].Unhealthy = 1
|
2016-07-31 04:04:54 -05:00
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.4"
|
|
|
|
h = ipHash.Select(pool, request)
|
2016-09-27 16:38:20 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
2016-07-31 04:04:54 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// We should be able to resize the host pool and still be able to predict
|
|
|
|
// where a request will be routed with the same IP's used above
|
|
|
|
pool = []*UpstreamHost{
|
|
|
|
{
|
|
|
|
Name: workableServer.URL, // this should resolve (healthcheck test)
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "http://localhost:99998", // this shouldn't
|
|
|
|
},
|
|
|
|
}
|
|
|
|
pool = HostPool(pool)
|
|
|
|
request.RemoteAddr = "172.0.0.1:80"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.2:80"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.3:80"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
request.RemoteAddr = "172.0.0.4:80"
|
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should get nil when there are no healthy hosts
|
2017-02-15 10:09:42 -05:00
|
|
|
pool[0].Unhealthy = 1
|
|
|
|
pool[1].Unhealthy = 1
|
2016-07-31 04:04:54 -05:00
|
|
|
h = ipHash.Select(pool, request)
|
|
|
|
if h != nil {
|
|
|
|
t.Error("Expected ip hash policy host to be nil.")
|
|
|
|
}
|
|
|
|
}
|