mirror of
https://github.com/umputun/reproxy.git
synced 2024-11-24 08:12:31 +02:00
fix comments, more tests
This commit is contained in:
parent
ae538c0574
commit
e7d08008bf
@ -53,7 +53,7 @@ func NewService(providers []Provider) *Service {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Run runs blocking loop getting events from all providers
|
// Run runs blocking loop getting events from all providers
|
||||||
// and updating mappers on each event
|
// and updating all mappers on each event
|
||||||
func (s *Service) Run(ctx context.Context) error {
|
func (s *Service) Run(ctx context.Context) error {
|
||||||
|
|
||||||
var evChs []<-chan struct{}
|
var evChs []<-chan struct{}
|
||||||
@ -79,7 +79,7 @@ func (s *Service) Run(ctx context.Context) error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Match url to all providers mappers
|
// Match url to all mappers
|
||||||
func (s *Service) Match(srv, src string) (string, bool) {
|
func (s *Service) Match(srv, src string) (string, bool) {
|
||||||
|
|
||||||
s.lock.RLock()
|
s.lock.RLock()
|
||||||
|
@ -16,11 +16,12 @@ import (
|
|||||||
|
|
||||||
//go:generate moq -out docker_client_mock.go -skip-ensure -fmt goimports . DockerClient
|
//go:generate moq -out docker_client_mock.go -skip-ensure -fmt goimports . DockerClient
|
||||||
|
|
||||||
// Docker provide watch compatible changes from containers
|
// Docker provider watches compatible for stop/start changes from containers and maps by
|
||||||
// and maps by default from ^/api/%s/(.*) to http://%s:%d/$1, i.e. http://example.com/api/my_container/something
|
// default from ^/api/%s/(.*) to http://%s:%d/$1, i.e. http://example.com/api/my_container/something
|
||||||
// will be mapped to http://172.17.42.1:8080/something. Ip will be the internal ip of the container and port - exposed the one
|
// will be mapped to http://172.17.42.1:8080/something. Ip will be the internal ip of the container and port exposed
|
||||||
|
// in the Dockerfile.
|
||||||
// Alternatively labels can alter this. reproxy.route sets source route, and reproxy.dest sets the destination.
|
// Alternatively labels can alter this. reproxy.route sets source route, and reproxy.dest sets the destination.
|
||||||
// Optional reproxy.server enforces match by server name (hostname).
|
// Optional reproxy.server enforces match by server name (hostname) and reproxy.ping sets the health check url
|
||||||
type Docker struct {
|
type Docker struct {
|
||||||
DockerClient DockerClient
|
DockerClient DockerClient
|
||||||
Excludes []string
|
Excludes []string
|
||||||
|
@ -14,8 +14,7 @@ import (
|
|||||||
"github.com/umputun/reproxy/app/discovery"
|
"github.com/umputun/reproxy/app/discovery"
|
||||||
)
|
)
|
||||||
|
|
||||||
// File implements file-based provider
|
// File implements file-based provider, defined with yaml file
|
||||||
// Each line contains src:dst pairs, i.e. ^/api/svc1/(.*) http://127.0.0:8080/blah/$1
|
|
||||||
type File struct {
|
type File struct {
|
||||||
FileName string
|
FileName string
|
||||||
CheckInterval time.Duration
|
CheckInterval time.Duration
|
||||||
|
@ -12,7 +12,7 @@ import (
|
|||||||
|
|
||||||
// Static provider, rules are server,from,to
|
// Static provider, rules are server,from,to
|
||||||
type Static struct {
|
type Static struct {
|
||||||
Rules []string // each rule is 5 elements comma separated. server,source url,destination,ping
|
Rules []string // each rule is 4 elements comma separated - server,source_url,destination,ping
|
||||||
}
|
}
|
||||||
|
|
||||||
// Events returns channel updating once
|
// Events returns channel updating once
|
||||||
|
72
app/proxy/health_test.go
Normal file
72
app/proxy/health_test.go
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
package proxy
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"math/rand"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"strconv"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
"github.com/umputun/reproxy/app/discovery"
|
||||||
|
"github.com/umputun/reproxy/app/discovery/provider"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestHttp_healthHandler(t *testing.T) {
|
||||||
|
port := rand.Intn(10000) + 40000
|
||||||
|
h := Http{TimeOut: 200 * time.Millisecond, Address: fmt.Sprintf("127.0.0.1:%d", port)}
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), 500*time.Millisecond)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
ds := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
t.Logf("req: %v", r)
|
||||||
|
w.Header().Add("h1", "v1")
|
||||||
|
fmt.Fprintf(w, "response %s", r.URL.String())
|
||||||
|
}))
|
||||||
|
|
||||||
|
ps := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
t.Logf("req: %v", r)
|
||||||
|
if r.URL.Path == "/123/ping" {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.WriteHeader(http.StatusBadRequest)
|
||||||
|
}))
|
||||||
|
|
||||||
|
svc := discovery.NewService([]discovery.Provider{
|
||||||
|
&provider.Static{Rules: []string{
|
||||||
|
"localhost,^/api/(.*)," + ds.URL + "/123/$1," + ps.URL + "/123/ping",
|
||||||
|
"127.0.0.1,^/api/(.*)," + ds.URL + "/567/$1," + ps.URL + "/567/ping",
|
||||||
|
},
|
||||||
|
}})
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
_ = svc.Run(context.Background())
|
||||||
|
}()
|
||||||
|
|
||||||
|
h.Matcher = svc
|
||||||
|
go func() {
|
||||||
|
_ = h.Run(ctx)
|
||||||
|
}()
|
||||||
|
time.Sleep(10 * time.Millisecond)
|
||||||
|
|
||||||
|
client := http.Client{}
|
||||||
|
req, err := http.NewRequest("GET", "http://127.0.0.1:"+strconv.Itoa(port)+"/health", nil)
|
||||||
|
require.NoError(t, err)
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer resp.Body.Close()
|
||||||
|
assert.Equal(t, http.StatusExpectationFailed, resp.StatusCode)
|
||||||
|
|
||||||
|
res := map[string]interface{}{}
|
||||||
|
err = json.NewDecoder(resp.Body).Decode(&res)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, "failed", res["status"])
|
||||||
|
assert.Equal(t, 1., res["passed"])
|
||||||
|
assert.Equal(t, 1., res["failed"])
|
||||||
|
assert.Contains(t, res["errors"], "400 Bad Request")
|
||||||
|
}
|
@ -2,7 +2,6 @@ package proxy
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
@ -90,6 +89,72 @@ func TestHttp_Do(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestHttp_DoWithAssets(t *testing.T) {
|
||||||
|
port := rand.Intn(10000) + 40000
|
||||||
|
h := Http{TimeOut: 200 * time.Millisecond, Address: fmt.Sprintf("127.0.0.1:%d", port),
|
||||||
|
AccessLog: io.Discard, AssetsWebRoot: "/static", AssetsLocation: "testdata", DisableSignature: true}
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), 500*time.Millisecond)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
ds := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
t.Logf("req: %v", r)
|
||||||
|
w.Header().Add("h1", "v1")
|
||||||
|
require.Equal(t, "127.0.0.1", r.Header.Get("X-Real-IP"))
|
||||||
|
fmt.Fprintf(w, "response %s", r.URL.String())
|
||||||
|
}))
|
||||||
|
|
||||||
|
svc := discovery.NewService([]discovery.Provider{
|
||||||
|
&provider.Static{Rules: []string{
|
||||||
|
"localhost,^/api/(.*)," + ds.URL + "/123/$1,",
|
||||||
|
"127.0.0.1,^/api/(.*)," + ds.URL + "/567/$1,",
|
||||||
|
},
|
||||||
|
}})
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
_ = svc.Run(context.Background())
|
||||||
|
}()
|
||||||
|
|
||||||
|
h.Matcher = svc
|
||||||
|
go func() {
|
||||||
|
_ = h.Run(ctx)
|
||||||
|
}()
|
||||||
|
time.Sleep(10 * time.Millisecond)
|
||||||
|
|
||||||
|
client := http.Client{}
|
||||||
|
|
||||||
|
{
|
||||||
|
req, err := http.NewRequest("GET", "http://127.0.0.1:"+strconv.Itoa(port)+"/api/something", nil)
|
||||||
|
require.NoError(t, err)
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer resp.Body.Close()
|
||||||
|
assert.Equal(t, http.StatusOK, resp.StatusCode)
|
||||||
|
t.Logf("%+v", resp.Header)
|
||||||
|
|
||||||
|
body, err := io.ReadAll(resp.Body)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, "response /567/something", string(body))
|
||||||
|
assert.Equal(t, "", resp.Header.Get("App-Name"))
|
||||||
|
assert.Equal(t, "v1", resp.Header.Get("h1"))
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
resp, err := client.Get("http://localhost:" + strconv.Itoa(port) + "/static/1.html")
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer resp.Body.Close()
|
||||||
|
assert.Equal(t, http.StatusOK, resp.StatusCode)
|
||||||
|
t.Logf("%+v", resp.Header)
|
||||||
|
|
||||||
|
body, err := io.ReadAll(resp.Body)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, "test html", string(body))
|
||||||
|
assert.Equal(t, "", resp.Header.Get("App-Name"))
|
||||||
|
assert.Equal(t, "", resp.Header.Get("h1"))
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
func TestHttp_toHttp(t *testing.T) {
|
func TestHttp_toHttp(t *testing.T) {
|
||||||
|
|
||||||
tbl := []struct {
|
tbl := []struct {
|
||||||
@ -111,57 +176,3 @@ func TestHttp_toHttp(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestHttp_healthHandler(t *testing.T) {
|
|
||||||
port := rand.Intn(10000) + 40000
|
|
||||||
h := Http{TimeOut: 200 * time.Millisecond, Address: fmt.Sprintf("127.0.0.1:%d", port)}
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 500*time.Millisecond)
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
ds := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
t.Logf("req: %v", r)
|
|
||||||
w.Header().Add("h1", "v1")
|
|
||||||
fmt.Fprintf(w, "response %s", r.URL.String())
|
|
||||||
}))
|
|
||||||
|
|
||||||
ps := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
t.Logf("req: %v", r)
|
|
||||||
if r.URL.Path == "/123/ping" {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
w.WriteHeader(http.StatusBadRequest)
|
|
||||||
}))
|
|
||||||
|
|
||||||
svc := discovery.NewService([]discovery.Provider{
|
|
||||||
&provider.Static{Rules: []string{
|
|
||||||
"localhost,^/api/(.*)," + ds.URL + "/123/$1," + ps.URL + "/123/ping",
|
|
||||||
"127.0.0.1,^/api/(.*)," + ds.URL + "/567/$1," + ps.URL + "/567/ping",
|
|
||||||
},
|
|
||||||
}})
|
|
||||||
|
|
||||||
go func() {
|
|
||||||
_ = svc.Run(context.Background())
|
|
||||||
}()
|
|
||||||
|
|
||||||
h.Matcher = svc
|
|
||||||
go func() {
|
|
||||||
_ = h.Run(ctx)
|
|
||||||
}()
|
|
||||||
time.Sleep(10 * time.Millisecond)
|
|
||||||
|
|
||||||
client := http.Client{}
|
|
||||||
req, err := http.NewRequest("GET", "http://127.0.0.1:"+strconv.Itoa(port)+"/health", nil)
|
|
||||||
require.NoError(t, err)
|
|
||||||
resp, err := client.Do(req)
|
|
||||||
require.NoError(t, err)
|
|
||||||
defer resp.Body.Close()
|
|
||||||
assert.Equal(t, http.StatusExpectationFailed, resp.StatusCode)
|
|
||||||
|
|
||||||
res := map[string]interface{}{}
|
|
||||||
err = json.NewDecoder(resp.Body).Decode(&res)
|
|
||||||
require.NoError(t, err)
|
|
||||||
assert.Equal(t, "failed", res["status"])
|
|
||||||
assert.Equal(t, 1., res["passed"])
|
|
||||||
assert.Equal(t, 1., res["failed"])
|
|
||||||
assert.Contains(t, res["errors"], "400 Bad Request")
|
|
||||||
}
|
|
||||||
|
1
app/proxy/testdata/1.html
vendored
Normal file
1
app/proxy/testdata/1.html
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
test html
|
@ -8,7 +8,7 @@ Type=simple
|
|||||||
Restart=always
|
Restart=always
|
||||||
RestartSec=1
|
RestartSec=1
|
||||||
User=root
|
User=root
|
||||||
ExecStart=/usr/bin/reproxy -f /etc/reproxy.yml -u
|
ExecStart=/usr/bin/reproxy --file.enabled --file.name=/etc/reproxy.yml
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
WantedBy=multi-user.target
|
WantedBy=multi-user.target
|
Loading…
Reference in New Issue
Block a user