1
0
mirror of https://github.com/umputun/reproxy.git synced 2025-06-30 22:13:42 +02:00
Files
reproxy/app/proxy/proxy_test.go

304 lines
8.6 KiB
Go
Raw Normal View History

2021-04-01 02:37:28 -05:00
package proxy
import (
"context"
"fmt"
"io"
"io/ioutil"
2021-04-01 02:37:28 -05:00
"math/rand"
"net/http"
"net/http/httptest"
"os"
"path"
2021-04-01 02:37:28 -05:00
"strconv"
"testing"
"time"
R "github.com/go-pkgz/rest"
2021-04-01 02:37:28 -05:00
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
2021-04-03 14:23:23 -05:00
"github.com/umputun/reproxy/app/discovery"
"github.com/umputun/reproxy/app/discovery/provider"
2021-04-20 19:09:51 -05:00
"github.com/umputun/reproxy/app/mgmt"
2021-04-01 02:37:28 -05:00
)
func TestHttp_Do(t *testing.T) {
port := rand.Intn(10000) + 40000
2021-04-12 21:54:59 -05:00
h := Http{Timeouts: Timeouts{ResponseHeader: 200 * time.Millisecond}, Address: fmt.Sprintf("127.0.0.1:%d", port),
AccessLog: io.Discard, Signature: true, ProxyHeaders: []string{"hh1:vv1", "hh2:vv2"}, StdOutEnabled: true}
2021-04-01 02:37:28 -05:00
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)
2021-04-01 02:37:28 -05:00
w.Header().Add("h1", "v1")
2021-04-04 02:57:34 -05:00
require.Equal(t, "127.0.0.1", r.Header.Get("X-Real-IP"))
fmt.Fprintf(w, "response %s", r.URL.String())
2021-04-01 02:37:28 -05:00
}))
svc := discovery.NewService([]discovery.Provider{
&provider.Static{Rules: []string{
2021-04-05 03:37:28 -05:00
"localhost,^/api/(.*)," + ds.URL + "/123/$1,",
"127.0.0.1,^/api/(.*)," + ds.URL + "/567/$1,",
},
2021-04-12 02:29:17 -05:00
}}, time.Millisecond*10)
2021-04-01 02:37:28 -05:00
go func() {
2021-04-03 01:20:24 -05:00
_ = svc.Run(context.Background())
2021-04-01 02:37:28 -05:00
}()
2021-04-12 02:29:17 -05:00
time.Sleep(50 * time.Millisecond)
2021-04-01 02:37:28 -05:00
h.Matcher = svc
2021-04-20 19:09:51 -05:00
h.Metrics = mgmt.NewMetrics()
2021-04-01 02:37:28 -05:00
go func() {
2021-04-03 01:20:24 -05:00
_ = h.Run(ctx)
2021-04-01 02:37:28 -05:00
}()
time.Sleep(10 * time.Millisecond)
client := http.Client{}
{
2021-04-04 02:57:34 -05:00
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)
2021-04-03 01:00:09 -05:00
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, "reproxy", resp.Header.Get("App-Name"))
assert.Equal(t, "v1", resp.Header.Get("h1"))
assert.Equal(t, "vv1", resp.Header.Get("hh1"))
assert.Equal(t, "vv2", resp.Header.Get("hh2"))
}
{
resp, err := client.Get("http://localhost:" + strconv.Itoa(port) + "/api/something")
require.NoError(t, err)
2021-04-03 01:00:09 -05:00
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 /123/something", string(body))
assert.Equal(t, "reproxy", resp.Header.Get("App-Name"))
assert.Equal(t, "v1", resp.Header.Get("h1"))
}
{
resp, err := client.Get("http://127.0.0.1:" + strconv.Itoa(port) + "/bad/something")
require.NoError(t, err)
2021-04-03 01:00:09 -05:00
defer resp.Body.Close()
assert.Equal(t, http.StatusBadGateway, resp.StatusCode)
}
2021-04-01 02:37:28 -05:00
}
2021-04-03 01:20:24 -05:00
2021-04-07 21:52:14 -05:00
func TestHttp_DoWithAssets(t *testing.T) {
2021-04-05 03:37:28 -05:00
port := rand.Intn(10000) + 40000
2021-04-12 21:54:59 -05:00
h := Http{Timeouts: Timeouts{ResponseHeader: 200 * time.Millisecond}, Address: fmt.Sprintf("127.0.0.1:%d", port),
2021-04-13 14:08:15 -05:00
AccessLog: io.Discard, AssetsWebRoot: "/static", AssetsLocation: "testdata"}
2021-04-05 03:37:28 -05:00
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")
2021-04-07 21:52:14 -05:00
require.Equal(t, "127.0.0.1", r.Header.Get("X-Real-IP"))
2021-04-05 03:37:28 -05:00
fmt.Fprintf(w, "response %s", r.URL.String())
}))
svc := discovery.NewService([]discovery.Provider{
&provider.Static{Rules: []string{
2021-04-07 21:52:14 -05:00
"localhost,^/api/(.*)," + ds.URL + "/123/$1,",
"127.0.0.1,^/api/(.*)," + ds.URL + "/567/$1,",
2021-04-05 03:37:28 -05:00
},
2021-04-12 02:29:17 -05:00
}}, time.Millisecond*10)
2021-04-05 03:37:28 -05:00
go func() {
_ = svc.Run(context.Background())
}()
2021-04-12 02:29:17 -05:00
time.Sleep(50 * time.Millisecond)
2021-04-05 03:37:28 -05:00
h.Matcher = svc
2021-04-20 19:09:51 -05:00
h.Metrics = mgmt.NewMetrics()
2021-04-05 03:37:28 -05:00
go func() {
_ = h.Run(ctx)
}()
time.Sleep(10 * time.Millisecond)
client := http.Client{}
2021-04-07 21:52:14 -05:00
{
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_DoWithAssetRules(t *testing.T) {
port := rand.Intn(10000) + 40000
h := Http{Timeouts: Timeouts{ResponseHeader: 200 * time.Millisecond}, Address: fmt.Sprintf("127.0.0.1:%d", port),
AccessLog: io.Discard}
ctx, cancel := context.WithTimeout(context.Background(), 500*time.Millisecond)
defer cancel()
2021-04-07 21:52:14 -05:00
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,",
"*,/web,assets:testdata,",
},
}}, time.Millisecond*10)
go func() {
_ = svc.Run(context.Background())
}()
time.Sleep(50 * time.Millisecond)
h.Matcher = svc
2021-04-20 19:09:51 -05:00
h.Metrics = mgmt.NewMetrics()
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) + "/web/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"))
2021-04-07 21:52:14 -05:00
}
}
func TestHttp_toHttp(t *testing.T) {
tbl := []struct {
addr string
port int
res string
}{
{"localhost:1234", 80, "localhost:80"},
{"m.example.com:443", 8080, "m.example.com:8080"},
{"192.168.1.1:1443", 8080, "192.168.1.1:8080"},
}
h := Http{}
for i, tt := range tbl {
tt := tt
t.Run(strconv.Itoa(i), func(t *testing.T) {
2021-04-09 15:05:22 -05:00
assert.Equal(t, tt.res, h.toHTTP(tt.addr, tt.port))
2021-04-07 21:52:14 -05:00
})
}
2021-04-05 03:37:28 -05:00
}
func TestHttp_cachingHandler(t *testing.T) {
dir, e := ioutil.TempDir(os.TempDir(), "reproxy")
require.NoError(t, e)
e = ioutil.WriteFile(path.Join(dir, "1.html"), []byte("1.htm"), 0600)
assert.NoError(t, e)
e = ioutil.WriteFile(path.Join(dir, "2.html"), []byte("2.htm"), 0600)
assert.NoError(t, e)
defer os.RemoveAll(dir)
fh, e := R.FileServer("/static", dir)
require.NoError(t, e)
h := Http{AssetsCacheDuration: 10 * time.Second, AssetsLocation: dir, AssetsWebRoot: "/static"}
hh := R.Wrap(fh, h.cachingHandler)
ts := httptest.NewServer(hh)
defer ts.Close()
client := http.Client{Timeout: 599 * time.Second}
{
resp, err := client.Get(ts.URL + "/static/1.html")
require.NoError(t, err)
assert.Equal(t, http.StatusOK, resp.StatusCode)
t.Logf("headers: %+v", resp.Header)
assert.Equal(t, "public, max-age=10", resp.Header.Get("Cache-Control"))
assert.NotEqual(t, "", resp.Header.Get("Last-Modified"))
}
{
resp, err := client.Get(ts.URL + "/static/bad.html")
require.NoError(t, err)
assert.Equal(t, http.StatusNotFound, resp.StatusCode)
t.Logf("headers: %+v", resp.Header)
assert.Equal(t, "public, max-age=10", resp.Header.Get("Cache-Control"))
assert.Equal(t, "", resp.Header.Get("Last-Modified"))
}
{
resp, err := client.Get(ts.URL + "/%2e%2e%2f%2e%2e%2f%2e%2e%2f/etc/passwd")
require.NoError(t, err)
assert.Equal(t, http.StatusNotFound, resp.StatusCode)
t.Logf("headers: %+v", resp.Header)
assert.Equal(t, "public, max-age=10", resp.Header.Get("Cache-Control"))
assert.Equal(t, "", resp.Header.Get("Last-Modified"))
}
}