2019-09-02 23:01:02 -05:00
|
|
|
// Copyright 2015 Matthew Holt and The Caddy Authors
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package reverseproxy
|
|
|
|
|
2019-09-09 22:44:58 -05:00
|
|
|
import (
|
2023-04-03 22:31:47 -05:00
|
|
|
"context"
|
2019-09-09 22:44:58 -05:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"testing"
|
2023-04-03 22:31:47 -05:00
|
|
|
|
2023-05-05 16:08:10 -05:00
|
|
|
"github.com/caddyserver/caddy/v2"
|
|
|
|
"github.com/caddyserver/caddy/v2/caddyconfig"
|
2023-04-03 22:31:47 -05:00
|
|
|
"github.com/caddyserver/caddy/v2/modules/caddyhttp"
|
2019-09-09 22:44:58 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
func testPool() UpstreamPool {
|
|
|
|
return UpstreamPool{
|
2022-04-27 11:39:22 -05:00
|
|
|
{Host: new(Host), Dial: "0.0.0.1"},
|
|
|
|
{Host: new(Host), Dial: "0.0.0.2"},
|
|
|
|
{Host: new(Host), Dial: "0.0.0.3"},
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRoundRobinPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
2023-05-05 16:08:10 -05:00
|
|
|
rrPolicy := RoundRobinSelection{}
|
2019-09-09 22:44:58 -05:00
|
|
|
req, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
|
2020-11-20 14:39:26 -05:00
|
|
|
h := rrPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -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.")
|
|
|
|
}
|
2020-11-20 14:39:26 -05:00
|
|
|
h = rrPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected second round robin host to be third host in the pool.")
|
|
|
|
}
|
2020-11-20 14:39:26 -05:00
|
|
|
h = rrPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected third round robin host to be first host in the pool.")
|
|
|
|
}
|
|
|
|
// mark host as down
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[1].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = rrPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected to skip down host.")
|
|
|
|
}
|
|
|
|
// mark host as up
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[1].setHealthy(true)
|
2019-09-09 22:44:58 -05:00
|
|
|
|
2020-11-20 14:39:26 -05:00
|
|
|
h = rrPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h == pool[2] {
|
|
|
|
t.Error("Expected to balance evenly among healthy hosts")
|
|
|
|
}
|
|
|
|
// mark host as full
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[1].countRequest(1)
|
2019-09-09 22:44:58 -05:00
|
|
|
pool[1].MaxRequests = 1
|
2020-11-20 14:39:26 -05:00
|
|
|
h = rrPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected to skip full host.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-20 12:42:58 -05:00
|
|
|
func TestWeightedRoundRobinPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
|
|
|
wrrPolicy := WeightedRoundRobinSelection{
|
|
|
|
Weights: []int{3, 2, 1},
|
|
|
|
totalWeight: 6,
|
|
|
|
}
|
|
|
|
req, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
|
|
|
|
h := wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected first weighted round robin host to be first host in the pool.")
|
|
|
|
}
|
|
|
|
h = wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected second weighted round robin host to be first host in the pool.")
|
|
|
|
}
|
|
|
|
// Third selected host is 1, because counter starts at 0
|
|
|
|
// and increments before host is selected
|
|
|
|
h = wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected third weighted round robin host to be second host in the pool.")
|
|
|
|
}
|
|
|
|
h = wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected fourth weighted round robin host to be second host in the pool.")
|
|
|
|
}
|
|
|
|
h = wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected fifth weighted round robin host to be third host in the pool.")
|
|
|
|
}
|
|
|
|
h = wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected sixth weighted round robin host to be first host in the pool.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// mark host as down
|
|
|
|
pool[0].setHealthy(false)
|
|
|
|
h = wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected to skip down host.")
|
|
|
|
}
|
|
|
|
// mark host as up
|
|
|
|
pool[0].setHealthy(true)
|
|
|
|
|
|
|
|
h = wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected to select first host on availablity.")
|
|
|
|
}
|
|
|
|
// mark host as full
|
|
|
|
pool[1].countRequest(1)
|
|
|
|
pool[1].MaxRequests = 1
|
|
|
|
h = wrrPolicy.Select(pool, req, nil)
|
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected to skip full host.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-09 22:44:58 -05:00
|
|
|
func TestLeastConnPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
2023-05-05 16:08:10 -05:00
|
|
|
lcPolicy := LeastConnSelection{}
|
2019-09-09 22:44:58 -05:00
|
|
|
req, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].countRequest(10)
|
|
|
|
pool[1].countRequest(10)
|
2020-11-20 14:39:26 -05:00
|
|
|
h := lcPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected least connection host to be third host.")
|
|
|
|
}
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[2].countRequest(100)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = lcPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[0] && h != pool[1] {
|
|
|
|
t.Error("Expected least connection host to be first or second host.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestIPHashPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
2023-05-05 16:08:10 -05:00
|
|
|
ipHash := IPHashSelection{}
|
2019-09-09 22:44:58 -05:00
|
|
|
req, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
|
|
|
|
// We should be able to predict where every request is routed.
|
|
|
|
req.RemoteAddr = "172.0.0.1:80"
|
2020-11-20 14:39:26 -05:00
|
|
|
h := ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
req.RemoteAddr = "172.0.0.2:80"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
req.RemoteAddr = "172.0.0.3:80"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
req.RemoteAddr = "172.0.0.4:80"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
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
|
|
|
|
req.RemoteAddr = "172.0.0.1"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
req.RemoteAddr = "172.0.0.2"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
req.RemoteAddr = "172.0.0.3"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
req.RemoteAddr = "172.0.0.4"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
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
|
2022-04-27 11:39:22 -05:00
|
|
|
req.RemoteAddr = "172.0.0.4"
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[1].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
req.RemoteAddr = "172.0.0.2"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-04-27 11:39:22 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[1].setHealthy(true)
|
2019-09-09 22:44:58 -05:00
|
|
|
|
|
|
|
req.RemoteAddr = "172.0.0.3"
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[2].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-04-27 11:39:22 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
req.RemoteAddr = "172.0.0.4"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should be able to resize the host pool and still be able to predict
|
|
|
|
// where a req will be routed with the same IP's used above
|
|
|
|
pool = UpstreamPool{
|
2022-04-27 11:39:22 -05:00
|
|
|
{Host: new(Host), Dial: "0.0.0.2"},
|
|
|
|
{Host: new(Host), Dial: "0.0.0.3"},
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
req.RemoteAddr = "172.0.0.1:80"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
2022-04-27 11:39:22 -05:00
|
|
|
req.RemoteAddr = "172.0.0.2:80"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
2022-04-27 11:39:22 -05:00
|
|
|
req.RemoteAddr = "172.0.0.3:80"
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
2022-04-27 11:39:22 -05:00
|
|
|
req.RemoteAddr = "172.0.0.4:80"
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
2019-09-09 22:44:58 -05:00
|
|
|
|
|
|
|
// We should get nil when there are no healthy hosts
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].setHealthy(false)
|
|
|
|
pool[1].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = ipHash.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != nil {
|
|
|
|
t.Error("Expected ip hash policy host to be nil.")
|
|
|
|
}
|
2021-04-29 11:52:22 -05:00
|
|
|
|
|
|
|
// Reproduce #4135
|
|
|
|
pool = UpstreamPool{
|
2022-03-06 19:43:39 -05:00
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
}
|
|
|
|
pool[0].setHealthy(false)
|
|
|
|
pool[1].setHealthy(false)
|
|
|
|
pool[2].setHealthy(false)
|
|
|
|
pool[3].setHealthy(false)
|
|
|
|
pool[4].setHealthy(false)
|
|
|
|
pool[5].setHealthy(false)
|
|
|
|
pool[6].setHealthy(false)
|
|
|
|
pool[7].setHealthy(false)
|
|
|
|
pool[8].setHealthy(true)
|
2021-04-29 11:52:22 -05:00
|
|
|
|
|
|
|
// We should get a result back when there is one healthy host left.
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h == nil {
|
|
|
|
// If it is nil, it means we missed a host even though one is available
|
|
|
|
t.Error("Expected ip hash policy host to not be nil, but it is nil.")
|
|
|
|
}
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
|
2023-04-03 22:31:47 -05:00
|
|
|
func TestClientIPHashPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
2023-05-05 16:08:10 -05:00
|
|
|
ipHash := ClientIPHashSelection{}
|
2023-04-03 22:31:47 -05:00
|
|
|
req, _ := http.NewRequest("GET", "/", nil)
|
|
|
|
req = req.WithContext(context.WithValue(req.Context(), caddyhttp.VarsCtxKey, make(map[string]any)))
|
|
|
|
|
|
|
|
// We should be able to predict where every request is routed.
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.1:80")
|
|
|
|
h := ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.2:80")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.3:80")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.4:80")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
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
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.1")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.2")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.3")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.4")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
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
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.4")
|
|
|
|
pool[1].setHealthy(false)
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.2")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
pool[1].setHealthy(true)
|
|
|
|
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.3")
|
|
|
|
pool[2].setHealthy(false)
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.4")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected ip hash policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should be able to resize the host pool and still be able to predict
|
|
|
|
// where a req will be routed with the same IP's used above
|
|
|
|
pool = UpstreamPool{
|
|
|
|
{Host: new(Host), Dial: "0.0.0.2"},
|
|
|
|
{Host: new(Host), Dial: "0.0.0.3"},
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.1:80")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.2:80")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.3:80")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
caddyhttp.SetVar(req.Context(), caddyhttp.ClientIPVarKey, "172.0.0.4:80")
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected ip hash policy host to be the first host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should get nil when there are no healthy hosts
|
|
|
|
pool[0].setHealthy(false)
|
|
|
|
pool[1].setHealthy(false)
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h != nil {
|
|
|
|
t.Error("Expected ip hash policy host to be nil.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reproduce #4135
|
|
|
|
pool = UpstreamPool{
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
}
|
|
|
|
pool[0].setHealthy(false)
|
|
|
|
pool[1].setHealthy(false)
|
|
|
|
pool[2].setHealthy(false)
|
|
|
|
pool[3].setHealthy(false)
|
|
|
|
pool[4].setHealthy(false)
|
|
|
|
pool[5].setHealthy(false)
|
|
|
|
pool[6].setHealthy(false)
|
|
|
|
pool[7].setHealthy(false)
|
|
|
|
pool[8].setHealthy(true)
|
|
|
|
|
|
|
|
// We should get a result back when there is one healthy host left.
|
|
|
|
h = ipHash.Select(pool, req, nil)
|
|
|
|
if h == nil {
|
|
|
|
// If it is nil, it means we missed a host even though one is available
|
|
|
|
t.Error("Expected ip hash policy host to not be nil, but it is nil.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-09 22:44:58 -05:00
|
|
|
func TestFirstPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
2023-05-05 16:08:10 -05:00
|
|
|
firstPolicy := FirstSelection{}
|
2019-09-09 22:44:58 -05:00
|
|
|
req := httptest.NewRequest(http.MethodGet, "/", nil)
|
|
|
|
|
2020-11-20 14:39:26 -05:00
|
|
|
h := firstPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected first policy host to be the first host.")
|
|
|
|
}
|
|
|
|
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = firstPolicy.Select(pool, req, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected first policy host to be the second host.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-03 22:31:47 -05:00
|
|
|
func TestQueryHashPolicy(t *testing.T) {
|
2023-05-05 16:08:10 -05:00
|
|
|
ctx, cancel := caddy.NewContext(caddy.Context{Context: context.Background()})
|
|
|
|
defer cancel()
|
2023-04-03 22:31:47 -05:00
|
|
|
queryPolicy := QueryHashSelection{Key: "foo"}
|
2023-05-05 16:08:10 -05:00
|
|
|
if err := queryPolicy.Provision(ctx); err != nil {
|
|
|
|
t.Errorf("Provision error: %v", err)
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
|
|
|
|
pool := testPool()
|
2023-04-03 22:31:47 -05:00
|
|
|
|
|
|
|
request := httptest.NewRequest(http.MethodGet, "/?foo=1", nil)
|
|
|
|
h := queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected query policy host to be the first host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/?foo=100000", nil)
|
|
|
|
h = queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected query policy host to be the first host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/?foo=1", nil)
|
|
|
|
pool[0].setHealthy(false)
|
|
|
|
h = queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected query policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/?foo=100000", nil)
|
|
|
|
h = queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != pool[2] {
|
|
|
|
t.Error("Expected query policy host to be the third host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should be able to resize the host pool and still be able to predict
|
|
|
|
// where a request will be routed with the same query used above
|
|
|
|
pool = UpstreamPool{
|
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/?foo=1", nil)
|
|
|
|
h = queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected query policy host to be the first host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
pool[0].setHealthy(false)
|
|
|
|
h = queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected query policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/?foo=4", nil)
|
|
|
|
h = queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected query policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
pool[0].setHealthy(false)
|
|
|
|
pool[1].setHealthy(false)
|
|
|
|
h = queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != nil {
|
|
|
|
t.Error("Expected query policy policy host to be nil.")
|
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/?foo=aa11&foo=bb22", nil)
|
|
|
|
pool = testPool()
|
|
|
|
h = queryPolicy.Select(pool, request, nil)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected query policy host to be the first host.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-09 22:44:58 -05:00
|
|
|
func TestURIHashPolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
2023-05-05 16:08:10 -05:00
|
|
|
uriPolicy := URIHashSelection{}
|
2019-09-09 22:44:58 -05:00
|
|
|
|
|
|
|
request := httptest.NewRequest(http.MethodGet, "/test", nil)
|
2020-11-20 14:39:26 -05:00
|
|
|
h := uriPolicy.Select(pool, request, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected uri policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
|
2022-04-27 11:39:22 -05:00
|
|
|
pool[2].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = uriPolicy.Select(pool, request, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[1] {
|
2022-04-27 11:39:22 -05:00
|
|
|
t.Error("Expected uri policy host to be the second host.")
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/test_2", nil)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = uriPolicy.Select(pool, request, nil)
|
2022-12-05 13:28:12 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected uri policy host to be the first host.")
|
2019-09-09 22:44:58 -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 URI's used above
|
|
|
|
pool = UpstreamPool{
|
2022-03-06 19:43:39 -05:00
|
|
|
{Host: new(Host)},
|
|
|
|
{Host: new(Host)},
|
2019-09-09 22:44:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/test", nil)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = uriPolicy.Select(pool, request, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected uri policy host to be the first host.")
|
|
|
|
}
|
|
|
|
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = uriPolicy.Select(pool, request, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected uri policy host to be the first host.")
|
|
|
|
}
|
|
|
|
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/test_2", nil)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = uriPolicy.Select(pool, request, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != pool[1] {
|
|
|
|
t.Error("Expected uri policy host to be the second host.")
|
|
|
|
}
|
|
|
|
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].setHealthy(false)
|
|
|
|
pool[1].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = uriPolicy.Select(pool, request, nil)
|
2019-09-09 22:44:58 -05:00
|
|
|
if h != nil {
|
|
|
|
t.Error("Expected uri policy policy host to be nil.")
|
|
|
|
}
|
|
|
|
}
|
2020-11-16 14:47:15 -05:00
|
|
|
|
|
|
|
func TestLeastRequests(t *testing.T) {
|
|
|
|
pool := testPool()
|
|
|
|
pool[0].Dial = "localhost:8080"
|
|
|
|
pool[1].Dial = "localhost:8081"
|
|
|
|
pool[2].Dial = "localhost:8082"
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].setHealthy(true)
|
|
|
|
pool[1].setHealthy(true)
|
|
|
|
pool[2].setHealthy(true)
|
|
|
|
pool[0].countRequest(10)
|
|
|
|
pool[1].countRequest(20)
|
|
|
|
pool[2].countRequest(30)
|
2020-11-16 14:47:15 -05:00
|
|
|
|
|
|
|
result := leastRequests(pool)
|
|
|
|
|
|
|
|
if result == nil {
|
|
|
|
t.Error("Least request should not return nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
if result != pool[0] {
|
|
|
|
t.Error("Least request should return pool[0]")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestRandomChoicePolicy(t *testing.T) {
|
|
|
|
pool := testPool()
|
|
|
|
pool[0].Dial = "localhost:8080"
|
|
|
|
pool[1].Dial = "localhost:8081"
|
|
|
|
pool[2].Dial = "localhost:8082"
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].setHealthy(false)
|
|
|
|
pool[1].setHealthy(true)
|
|
|
|
pool[2].setHealthy(true)
|
|
|
|
pool[0].countRequest(10)
|
|
|
|
pool[1].countRequest(20)
|
|
|
|
pool[2].countRequest(30)
|
2020-11-16 14:47:15 -05:00
|
|
|
|
|
|
|
request := httptest.NewRequest(http.MethodGet, "/test", nil)
|
2023-05-05 16:08:10 -05:00
|
|
|
randomChoicePolicy := RandomChoiceSelection{Choose: 2}
|
2020-11-16 14:47:15 -05:00
|
|
|
|
2020-11-20 14:39:26 -05:00
|
|
|
h := randomChoicePolicy.Select(pool, request, nil)
|
2020-11-16 14:47:15 -05:00
|
|
|
|
|
|
|
if h == nil {
|
|
|
|
t.Error("RandomChoicePolicy should not return nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
if h == pool[0] {
|
|
|
|
t.Error("RandomChoicePolicy should not choose pool[0]")
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2020-11-20 14:39:26 -05:00
|
|
|
|
|
|
|
func TestCookieHashPolicy(t *testing.T) {
|
2023-05-05 16:08:10 -05:00
|
|
|
ctx, cancel := caddy.NewContext(caddy.Context{Context: context.Background()})
|
|
|
|
defer cancel()
|
|
|
|
cookieHashPolicy := CookieHashSelection{}
|
|
|
|
if err := cookieHashPolicy.Provision(ctx); err != nil {
|
|
|
|
t.Errorf("Provision error: %v", err)
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
|
2020-11-20 14:39:26 -05:00
|
|
|
pool := testPool()
|
|
|
|
pool[0].Dial = "localhost:8080"
|
|
|
|
pool[1].Dial = "localhost:8081"
|
|
|
|
pool[2].Dial = "localhost:8082"
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].setHealthy(true)
|
|
|
|
pool[1].setHealthy(false)
|
|
|
|
pool[2].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
request := httptest.NewRequest(http.MethodGet, "/test", nil)
|
|
|
|
w := httptest.NewRecorder()
|
2023-05-05 16:08:10 -05:00
|
|
|
|
2020-11-20 14:39:26 -05:00
|
|
|
h := cookieHashPolicy.Select(pool, request, w)
|
2020-12-14 17:30:55 -05:00
|
|
|
cookieServer1 := w.Result().Cookies()[0]
|
|
|
|
if cookieServer1 == nil {
|
2021-05-05 15:52:18 -05:00
|
|
|
t.Fatal("cookieHashPolicy should set a cookie")
|
2020-11-20 14:39:26 -05:00
|
|
|
}
|
2020-12-14 17:30:55 -05:00
|
|
|
if cookieServer1.Name != "lb" {
|
2020-11-20 14:39:26 -05:00
|
|
|
t.Error("cookieHashPolicy should set a cookie with name lb")
|
|
|
|
}
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected cookieHashPolicy host to be the first only available host.")
|
|
|
|
}
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[1].setHealthy(true)
|
|
|
|
pool[2].setHealthy(true)
|
2020-11-20 14:39:26 -05:00
|
|
|
request = httptest.NewRequest(http.MethodGet, "/test", nil)
|
|
|
|
w = httptest.NewRecorder()
|
2020-12-14 17:30:55 -05:00
|
|
|
request.AddCookie(cookieServer1)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = cookieHashPolicy.Select(pool, request, w)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Error("Expected cookieHashPolicy host to stick to the first host (matching cookie).")
|
|
|
|
}
|
|
|
|
s := w.Result().Cookies()
|
|
|
|
if len(s) != 0 {
|
|
|
|
t.Error("Expected cookieHashPolicy to not set a new cookie.")
|
|
|
|
}
|
2022-03-06 19:43:39 -05:00
|
|
|
pool[0].setHealthy(false)
|
2020-11-20 14:39:26 -05:00
|
|
|
request = httptest.NewRequest(http.MethodGet, "/test", nil)
|
|
|
|
w = httptest.NewRecorder()
|
2020-12-14 17:30:55 -05:00
|
|
|
request.AddCookie(cookieServer1)
|
2020-11-20 14:39:26 -05:00
|
|
|
h = cookieHashPolicy.Select(pool, request, w)
|
|
|
|
if h == pool[0] {
|
|
|
|
t.Error("Expected cookieHashPolicy to select a new host.")
|
|
|
|
}
|
|
|
|
if w.Result().Cookies() == nil {
|
|
|
|
t.Error("Expected cookieHashPolicy to set a new cookie.")
|
|
|
|
}
|
|
|
|
}
|
2023-05-05 16:08:10 -05:00
|
|
|
|
|
|
|
func TestCookieHashPolicyWithFirstFallback(t *testing.T) {
|
|
|
|
ctx, cancel := caddy.NewContext(caddy.Context{Context: context.Background()})
|
|
|
|
defer cancel()
|
|
|
|
cookieHashPolicy := CookieHashSelection{
|
|
|
|
FallbackRaw: caddyconfig.JSONModuleObject(FirstSelection{}, "policy", "first", nil),
|
|
|
|
}
|
|
|
|
if err := cookieHashPolicy.Provision(ctx); err != nil {
|
|
|
|
t.Errorf("Provision error: %v", err)
|
|
|
|
t.FailNow()
|
|
|
|
}
|
|
|
|
|
|
|
|
pool := testPool()
|
|
|
|
pool[0].Dial = "localhost:8080"
|
|
|
|
pool[1].Dial = "localhost:8081"
|
|
|
|
pool[2].Dial = "localhost:8082"
|
|
|
|
pool[0].setHealthy(true)
|
|
|
|
pool[1].setHealthy(true)
|
|
|
|
pool[2].setHealthy(true)
|
|
|
|
request := httptest.NewRequest(http.MethodGet, "/test", nil)
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
h := cookieHashPolicy.Select(pool, request, w)
|
|
|
|
cookieServer1 := w.Result().Cookies()[0]
|
|
|
|
if cookieServer1 == nil {
|
|
|
|
t.Fatal("cookieHashPolicy should set a cookie")
|
|
|
|
}
|
|
|
|
if cookieServer1.Name != "lb" {
|
|
|
|
t.Error("cookieHashPolicy should set a cookie with name lb")
|
|
|
|
}
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Errorf("Expected cookieHashPolicy host to be the first only available host, got %s", h)
|
|
|
|
}
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/test", nil)
|
|
|
|
w = httptest.NewRecorder()
|
|
|
|
request.AddCookie(cookieServer1)
|
|
|
|
h = cookieHashPolicy.Select(pool, request, w)
|
|
|
|
if h != pool[0] {
|
|
|
|
t.Errorf("Expected cookieHashPolicy host to stick to the first host (matching cookie), got %s", h)
|
|
|
|
}
|
|
|
|
s := w.Result().Cookies()
|
|
|
|
if len(s) != 0 {
|
|
|
|
t.Error("Expected cookieHashPolicy to not set a new cookie.")
|
|
|
|
}
|
|
|
|
pool[0].setHealthy(false)
|
|
|
|
request = httptest.NewRequest(http.MethodGet, "/test", nil)
|
|
|
|
w = httptest.NewRecorder()
|
|
|
|
request.AddCookie(cookieServer1)
|
|
|
|
h = cookieHashPolicy.Select(pool, request, w)
|
|
|
|
if h != pool[1] {
|
|
|
|
t.Errorf("Expected cookieHashPolicy to select the next first available host, got %s", h)
|
|
|
|
}
|
|
|
|
if w.Result().Cookies() == nil {
|
|
|
|
t.Error("Expected cookieHashPolicy to set a new cookie.")
|
|
|
|
}
|
|
|
|
}
|