2020-05-04 11:40:46 +02:00
|
|
|
package pilot
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2020-12-03 15:52:05 +01:00
|
|
|
"hash/fnv"
|
2020-05-04 11:40:46 +02:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2020-12-03 15:52:05 +01:00
|
|
|
"reflect"
|
2020-05-04 11:40:46 +02:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2020-12-03 15:52:05 +01:00
|
|
|
"github.com/stretchr/testify/assert"
|
2020-05-04 11:40:46 +02:00
|
|
|
"github.com/stretchr/testify/require"
|
2020-12-03 15:52:05 +01:00
|
|
|
"github.com/traefik/traefik/v2/pkg/config/dynamic"
|
2020-09-16 15:46:04 +02:00
|
|
|
"github.com/traefik/traefik/v2/pkg/metrics"
|
|
|
|
"github.com/traefik/traefik/v2/pkg/safe"
|
2020-05-04 11:40:46 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestTick(t *testing.T) {
|
|
|
|
receivedConfig := make(chan bool)
|
|
|
|
|
|
|
|
mux := http.NewServeMux()
|
|
|
|
server := httptest.NewServer(mux)
|
|
|
|
t.Cleanup(server.Close)
|
|
|
|
|
|
|
|
mux.HandleFunc("/", func(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
if req.Method != http.MethodPost {
|
|
|
|
http.Error(rw, "invalid method", http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
err := json.NewEncoder(rw).Encode(instanceInfo{ID: "123"})
|
|
|
|
if err != nil {
|
|
|
|
http.Error(rw, err.Error(), http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
mux.HandleFunc("/command", func(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
if req.Method != http.MethodPost {
|
|
|
|
http.Error(rw, "invalid method", http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
receivedConfig <- true
|
|
|
|
})
|
|
|
|
|
2020-08-10 15:26:04 +02:00
|
|
|
pilot := New("token", metrics.RegisterPilot(), safe.NewPool(context.Background()))
|
2020-12-03 15:52:05 +01:00
|
|
|
|
|
|
|
pilot.client.baseInstanceInfoURL = server.URL
|
2020-05-04 11:40:46 +02:00
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
t.Cleanup(cancel)
|
|
|
|
|
|
|
|
go pilot.Tick(ctx)
|
|
|
|
|
2020-12-03 15:52:05 +01:00
|
|
|
pilot.SetDynamicConfiguration(dynamic.Configuration{})
|
|
|
|
pilot.SetDynamicConfiguration(dynamic.Configuration{})
|
2020-05-04 11:40:46 +02:00
|
|
|
|
|
|
|
select {
|
|
|
|
case <-time.Tick(10 * time.Second):
|
|
|
|
t.Fatal("Timeout")
|
|
|
|
case <-receivedConfig:
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-03 15:52:05 +01:00
|
|
|
func TestClient_SendInstanceInfo(t *testing.T) {
|
2020-05-04 11:40:46 +02:00
|
|
|
myToken := "myToken"
|
|
|
|
|
2020-12-03 15:52:05 +01:00
|
|
|
myTokenHash, err := hashToken(myToken)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2020-05-04 11:40:46 +02:00
|
|
|
mux := http.NewServeMux()
|
|
|
|
server := httptest.NewServer(mux)
|
|
|
|
t.Cleanup(server.Close)
|
|
|
|
|
|
|
|
mux.HandleFunc("/", func(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
if req.Method != http.MethodPost {
|
|
|
|
http.Error(rw, "invalid method", http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tk := req.Header.Get(tokenHeader)
|
|
|
|
if tk != myToken {
|
|
|
|
http.Error(rw, fmt.Sprintf("invalid token: %s", tk), http.StatusUnauthorized)
|
2020-12-03 15:52:05 +01:00
|
|
|
return
|
2020-05-04 11:40:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
err := json.NewEncoder(rw).Encode(instanceInfo{ID: "123"})
|
|
|
|
if err != nil {
|
|
|
|
http.Error(rw, err.Error(), http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
mux.HandleFunc("/command", func(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
if req.Method != http.MethodPost {
|
|
|
|
http.Error(rw, "invalid method", http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tk := req.Header.Get(tokenHeader)
|
|
|
|
if tk != myToken {
|
|
|
|
http.Error(rw, fmt.Sprintf("invalid token: %s", tk), http.StatusUnauthorized)
|
2020-12-03 15:52:05 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tkh := req.Header.Get(tokenHashHeader)
|
|
|
|
if tkh != myTokenHash {
|
|
|
|
http.Error(rw, fmt.Sprintf("invalid token hash: %s", tkh), http.StatusBadRequest)
|
|
|
|
return
|
2020-05-04 11:40:46 +02:00
|
|
|
}
|
|
|
|
|
2020-12-03 15:52:05 +01:00
|
|
|
defer func() { _ = req.Body.Close() }()
|
2020-05-04 11:40:46 +02:00
|
|
|
|
|
|
|
info := &instanceInfo{}
|
|
|
|
err := json.NewDecoder(req.Body).Decode(info)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(rw, err.Error(), http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if info.ID != "123" {
|
|
|
|
http.Error(rw, fmt.Sprintf("invalid ID: %s", info.ID), http.StatusBadRequest)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
client := client{
|
2020-12-03 15:52:05 +01:00
|
|
|
baseInstanceInfoURL: server.URL,
|
|
|
|
httpClient: http.DefaultClient,
|
|
|
|
token: myToken,
|
|
|
|
tokenHash: myTokenHash,
|
|
|
|
}
|
|
|
|
|
|
|
|
err = client.SendInstanceInfo(context.Background(), []metrics.PilotMetric{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestClient_SendAnonDynConf(t *testing.T) {
|
|
|
|
myToken := "myToken"
|
|
|
|
|
|
|
|
myTokenHash, err := hashToken(myToken)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var count int
|
|
|
|
mux := http.NewServeMux()
|
|
|
|
mux.HandleFunc("/collect", func(rw http.ResponseWriter, req *http.Request) {
|
|
|
|
count++
|
|
|
|
if count == 1 {
|
|
|
|
http.Error(rw, "OOPS", http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if req.Method != http.MethodPost {
|
|
|
|
http.Error(rw, "invalid method", http.StatusMethodNotAllowed)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tkh := req.Header.Get(tokenHashHeader)
|
|
|
|
if tkh != myTokenHash {
|
|
|
|
http.Error(rw, fmt.Sprintf("invalid token hash: %s", tkh), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
defer func() { _ = req.Body.Close() }()
|
|
|
|
|
|
|
|
config := &dynamic.Configuration{}
|
|
|
|
err := json.NewDecoder(req.Body).Decode(config)
|
|
|
|
if err != nil {
|
|
|
|
http.Error(rw, err.Error(), http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
router, exists := config.HTTP.Routers["foo"]
|
|
|
|
if !exists {
|
|
|
|
http.Error(rw, "router configuration is missing", http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(router, &dynamic.Router{Service: "foo", Rule: "xxxx"}) {
|
|
|
|
http.Error(rw, fmt.Sprintf("configuration is not anonymized: %+v", router), http.StatusBadRequest)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
server := httptest.NewServer(mux)
|
|
|
|
t.Cleanup(server.Close)
|
|
|
|
|
|
|
|
client := client{
|
|
|
|
baseGatewayURL: server.URL,
|
|
|
|
httpClient: http.DefaultClient,
|
|
|
|
token: myToken,
|
|
|
|
tokenHash: myTokenHash,
|
|
|
|
}
|
|
|
|
|
|
|
|
config := dynamic.Configuration{
|
|
|
|
HTTP: &dynamic.HTTPConfiguration{
|
|
|
|
Routers: map[string]*dynamic.Router{
|
|
|
|
"foo": {
|
|
|
|
Service: "foo",
|
|
|
|
Rule: "foo.com",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
2020-05-04 11:40:46 +02:00
|
|
|
}
|
|
|
|
|
2020-12-03 15:52:05 +01:00
|
|
|
err = client.SendAnonDynConf(context.Background(), config)
|
2020-05-04 11:40:46 +02:00
|
|
|
require.NoError(t, err)
|
2020-12-03 15:52:05 +01:00
|
|
|
|
|
|
|
assert.Equal(t, 2, count)
|
|
|
|
}
|
|
|
|
|
|
|
|
func hashToken(token string) (string, error) {
|
|
|
|
tokenHash := fnv.New64a()
|
|
|
|
|
|
|
|
_, err := tokenHash.Write([]byte(token))
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Sprintf("%x", tokenHash.Sum64()), nil
|
2020-05-04 11:40:46 +02:00
|
|
|
}
|