Files
reproxy/app/discovery/provider/docker_test.go
Umputun efded2cf5e flip order in docker proxy+static labels #55
we want proxy rule to have priority over assets rule
2021-04-23 04:05:14 -05:00

238 lines
7.0 KiB
Go

package provider
import (
"context"
"fmt"
"io/ioutil"
"log"
"net/http"
"net/http/httptest"
"strings"
"testing"
"time"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/umputun/reproxy/app/discovery"
)
func TestDocker_List(t *testing.T) {
dclient := &DockerClientMock{
ListContainersFunc: func() ([]containerInfo, error) {
return []containerInfo{
{
Name: "c0", State: "running", IP: "127.0.0.2", Ports: []int{12348},
Labels: map[string]string{"reproxy.route": "^/a/(.*)", "reproxy.dest": "/a/$1",
"reproxy.server": "example.com", "reproxy.ping": "/ping"},
},
{
Name: "c1", State: "running", IP: "127.0.0.2", Ports: []int{12345},
Labels: map[string]string{"reproxy.route": "^/api/123/(.*)", "reproxy.dest": "/blah/$1",
"reproxy.server": "example.com", "reproxy.ping": "/ping"},
},
{
Name: "c2", State: "running", IP: "127.0.0.3", Ports: []int{12346},
Labels: map[string]string{"reproxy.enabled": "y"},
},
{
Name: "c3", State: "stopped",
},
{
Name: "c4", State: "running", IP: "127.0.0.2", Ports: []int{12345},
},
{
Name: "c5", State: "running", IP: "127.0.0.122", Ports: []int{2345},
Labels: map[string]string{"reproxy.enabled": "false"},
},
}, nil
},
}
d := Docker{DockerClient: dclient}
res, err := d.List()
require.NoError(t, err)
require.Equal(t, 3, len(res))
assert.Equal(t, "^/api/123/(.*)", res[0].SrcMatch.String())
assert.Equal(t, "http://127.0.0.2:12345/blah/$1", res[0].Dst)
assert.Equal(t, "example.com", res[0].Server)
assert.Equal(t, "http://127.0.0.2:12345/ping", res[0].PingURL)
assert.Equal(t, "^/a/(.*)", res[1].SrcMatch.String())
assert.Equal(t, "http://127.0.0.2:12348/a/$1", res[1].Dst)
assert.Equal(t, "http://127.0.0.2:12348/ping", res[1].PingURL)
assert.Equal(t, "example.com", res[1].Server)
assert.Equal(t, "^/(.*)", res[2].SrcMatch.String())
assert.Equal(t, "http://127.0.0.3:12346/$1", res[2].Dst)
assert.Equal(t, "http://127.0.0.3:12346/ping", res[2].PingURL)
assert.Equal(t, "*", res[2].Server)
}
func TestDocker_ListWithAutoAPI(t *testing.T) {
dclient := &DockerClientMock{
ListContainersFunc: func() ([]containerInfo, error) {
return []containerInfo{
{
Name: "c1", State: "running", IP: "127.0.0.2", Ports: []int{1345, 12345},
Labels: map[string]string{"reproxy.route": "^/api/123/(.*)", "reproxy.dest": "/blah/$1",
"reproxy.port": "12345", "reproxy.server": "example.com, example2.com", "reproxy.ping": "/ping"},
},
{
Name: "c2", State: "running", IP: "127.0.0.3", Ports: []int{12346},
},
{
Name: "c3", State: "stopped",
},
{
Name: "c4", State: "running", IP: "127.0.0.122", Ports: []int{2345},
Labels: map[string]string{"reproxy.enabled": "false"},
},
}, nil
},
}
d := Docker{DockerClient: dclient, AutoAPI: true}
res, err := d.List()
require.NoError(t, err)
require.Equal(t, 3, len(res))
assert.Equal(t, "^/api/123/(.*)", res[0].SrcMatch.String())
assert.Equal(t, "http://127.0.0.2:12345/blah/$1", res[0].Dst)
assert.Equal(t, "example.com", res[0].Server)
assert.Equal(t, "http://127.0.0.2:12345/ping", res[0].PingURL)
assert.Equal(t, "^/api/123/(.*)", res[1].SrcMatch.String())
assert.Equal(t, "http://127.0.0.2:12345/blah/$1", res[1].Dst)
assert.Equal(t, "example2.com", res[1].Server)
assert.Equal(t, "http://127.0.0.2:12345/ping", res[1].PingURL)
assert.Equal(t, "^/api/c2/(.*)", res[2].SrcMatch.String())
assert.Equal(t, "http://127.0.0.3:12346/$1", res[2].Dst)
assert.Equal(t, "http://127.0.0.3:12346/ping", res[2].PingURL)
assert.Equal(t, "*", res[2].Server)
}
func TestDocker_ListPriority(t *testing.T) {
dclient := &DockerClientMock{
ListContainersFunc: func() ([]containerInfo, error) {
return []containerInfo{
{
Name: "c0", State: "running", IP: "127.0.0.2", Ports: []int{12348},
Labels: map[string]string{"reproxy.route": "^/whoami/(.*)", "reproxy.dest": "/$1",
"reproxy.server": "example.com", "reproxy.ping": "/ping", "reproxy.assets": "/:/assets/static1"},
},
}, nil
},
}
d := Docker{DockerClient: dclient}
res, err := d.List()
require.NoError(t, err)
require.Equal(t, 2, len(res))
assert.Equal(t, discovery.MTProxy, res[0].MatchType)
assert.Equal(t, discovery.MTStatic, res[1].MatchType)
}
func TestDocker_refresh(t *testing.T) {
containers := make(chan []containerInfo)
d := Docker{
DockerClient: &DockerClientMock{
ListContainersFunc: func() ([]containerInfo, error) {
return <-containers, nil
},
},
RefreshInterval: time.Nanosecond,
}
events := make(chan discovery.ProviderID)
ctx, cancel := context.WithTimeout(context.Background(), time.Second)
defer cancel()
stub := func(id string) containerInfo {
return containerInfo{ID: id, Name: id, State: "running", IP: "127.0.0." + id, Ports: []int{12345}}
}
recv := func() {
select {
case <-events:
return
case <-time.After(time.Second):
t.Fatal("No refresh notification was received after 1s")
}
}
go func() {
if err := d.events(ctx, events); err != context.Canceled {
log.Fatal(err)
}
}()
// Start some
containers <- []containerInfo{stub("1"), stub("2")}
recv()
// Nothing changed
containers <- []containerInfo{stub("1"), stub("2")}
time.Sleep(time.Millisecond)
assert.Empty(t, events, "unexpected refresh notification")
// Stopped
containers <- []containerInfo{stub("1")}
recv()
// One changed
containers <- []containerInfo{
{ID: "1", Name: "1", State: "running", IP: "127.42.42.42", Ports: []int{12345}},
}
recv()
time.Sleep(time.Millisecond)
assert.Empty(t, events, "unexpect refresh notification from events channel")
}
func TestDockerClient(t *testing.T) {
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
require.Equal(t, `/v1.22/containers/json`, r.URL.Path)
// obtained using curl --unix-socket /var/run/docker.sock http://localhost/v1.41/containers/json
resp, err := ioutil.ReadFile("testdata/containers.json")
require.NoError(t, err)
w.Write(resp)
}))
defer srv.Close()
addr := fmt.Sprintf("tcp://%s", strings.TrimPrefix(srv.URL, "http://"))
client := NewDockerClient(addr, "bridge")
c, err := client.ListContainers()
require.NoError(t, err, "unexpected error while listing containers")
assert.Len(t, c, 2)
assert.NotEmpty(t, c[0].ID)
assert.Equal(t, "nginx", c[0].Name)
assert.Equal(t, "running", c[0].State)
assert.Equal(t, "172.17.0.3", c[0].IP)
assert.Equal(t, "y", c[0].Labels["reproxy.enabled"])
assert.Equal(t, []int{80}, c[0].Ports)
assert.Equal(t, time.Unix(1618417435, 0), c[0].TS)
assert.Empty(t, c[1].IP)
}
func TestDockerClient_error(t *testing.T) {
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
http.Error(w, `{"message": "bruh"}`, http.StatusInternalServerError)
}))
defer srv.Close()
addr := fmt.Sprintf("tcp://%s", strings.TrimPrefix(srv.URL, "http://"))
client := NewDockerClient(addr, "bridge")
_, err := client.ListContainers()
require.EqualError(t, err, "unexpected error from docker daemon: bruh")
}