2020-01-08 19:19:18 +00:00
|
|
|
package http
|
|
|
|
|
|
|
|
import (
|
2020-02-06 14:04:29 +00:00
|
|
|
"net/http"
|
2021-07-29 16:22:11 +00:00
|
|
|
"net/http/httptest"
|
2020-01-08 19:19:18 +00:00
|
|
|
"path"
|
|
|
|
"testing"
|
|
|
|
|
2021-03-30 18:10:02 +00:00
|
|
|
"github.com/influxdata/influxdb/v2/kit/platform"
|
2021-07-29 16:22:11 +00:00
|
|
|
"github.com/influxdata/influxdb/v2/kit/prom"
|
|
|
|
"github.com/influxdata/influxdb/v2/kit/prom/promtest"
|
2020-04-03 17:39:20 +00:00
|
|
|
"github.com/influxdata/influxdb/v2/pkg/testttp"
|
2021-07-29 16:22:11 +00:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
2020-01-08 19:19:18 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2021-07-29 16:22:11 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"go.uber.org/zap/zaptest"
|
2020-01-08 19:19:18 +00:00
|
|
|
)
|
|
|
|
|
2021-07-29 16:22:11 +00:00
|
|
|
func TestMetrics(t *testing.T) {
|
|
|
|
labels := []string{"handler", "method", "path", "status", "response_code", "user_agent"}
|
|
|
|
|
|
|
|
nextHandler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.URL.Path == "/" {
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if r.URL.Path == "/serverError" {
|
|
|
|
w.WriteHeader(http.StatusInternalServerError)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if r.URL.Path == "/redirect" {
|
|
|
|
w.WriteHeader(http.StatusPermanentRedirect)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(http.StatusNotFound)
|
|
|
|
})
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
reqPath string
|
|
|
|
wantCount int
|
|
|
|
labelResponse string
|
|
|
|
labelStatus string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "counter increments on OK (2XX) ",
|
|
|
|
reqPath: "/",
|
|
|
|
wantCount: 1,
|
|
|
|
labelResponse: "200",
|
|
|
|
labelStatus: "2XX",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "counter does not increment on not found (4XX)",
|
|
|
|
reqPath: "/badpath",
|
|
|
|
wantCount: 0,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "counter increments on server error (5XX)",
|
|
|
|
reqPath: "/serverError",
|
|
|
|
wantCount: 1,
|
|
|
|
labelResponse: "500",
|
|
|
|
labelStatus: "5XX",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "counter does not increment on redirect (3XX)",
|
|
|
|
reqPath: "/redirect",
|
|
|
|
wantCount: 0,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
counter := prometheus.NewCounterVec(prometheus.CounterOpts{Name: "counter"}, labels)
|
|
|
|
hist := prometheus.NewHistogramVec(prometheus.HistogramOpts{Name: "hist"}, labels)
|
|
|
|
reg := prom.NewRegistry(zaptest.NewLogger(t))
|
|
|
|
reg.MustRegister(counter, hist)
|
|
|
|
|
|
|
|
metricsMw := Metrics("testing", counter, hist)
|
|
|
|
svr := metricsMw(nextHandler)
|
|
|
|
r := httptest.NewRequest("GET", tt.reqPath, nil)
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
svr.ServeHTTP(w, r)
|
|
|
|
|
|
|
|
mfs := promtest.MustGather(t, reg)
|
|
|
|
m := promtest.FindMetric(mfs, "counter", map[string]string{
|
|
|
|
"handler": "testing",
|
|
|
|
"method": "GET",
|
|
|
|
"path": tt.reqPath,
|
|
|
|
"response_code": tt.labelResponse,
|
|
|
|
"status": tt.labelStatus,
|
|
|
|
"user_agent": "unknown",
|
|
|
|
})
|
|
|
|
|
|
|
|
if tt.wantCount == 0 {
|
|
|
|
require.Nil(t, m)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
require.Equal(t, tt.wantCount, int(m.Counter.GetValue()))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-08 19:19:18 +00:00
|
|
|
func Test_normalizePath(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
path string
|
|
|
|
expected string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "1",
|
2021-03-30 18:10:02 +00:00
|
|
|
path: path.Join("/api/v2/organizations", platform.ID(2).String()),
|
2020-01-08 19:19:18 +00:00
|
|
|
expected: "/api/v2/organizations/:id",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "2",
|
|
|
|
path: "/api/v2/organizations",
|
|
|
|
expected: "/api/v2/organizations",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "3",
|
|
|
|
path: "/",
|
|
|
|
expected: "/",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "4",
|
2021-03-30 18:10:02 +00:00
|
|
|
path: path.Join("/api/v2/organizations", platform.ID(2).String(), "users", platform.ID(3).String()),
|
2020-01-08 19:19:18 +00:00
|
|
|
expected: "/api/v2/organizations/:id/users/:id",
|
|
|
|
},
|
2021-04-08 13:57:47 +00:00
|
|
|
{
|
|
|
|
name: "5",
|
|
|
|
path: "/838442d56d.svg",
|
|
|
|
expected: "/" + fileSlug + ".svg",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "6",
|
|
|
|
path: "/838442d56d.svg/extra",
|
|
|
|
expected: "/838442d56d.svg/extra",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "7",
|
|
|
|
path: "/api/v2/restore/shards/1001",
|
|
|
|
expected: path.Join("/api/v2/restore/shards/", shardSlug),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "8",
|
|
|
|
path: "/api/v2/restore/shards/1001/extra",
|
|
|
|
expected: path.Join("/api/v2/restore/shards/", shardSlug, "extra"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "9",
|
|
|
|
path: "/api/v2/backup/shards/1005",
|
|
|
|
expected: path.Join("/api/v2/backup/shards/", shardSlug),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "10",
|
|
|
|
path: "/api/v2/backup/shards/1005/extra",
|
|
|
|
expected: path.Join("/api/v2/backup/shards/", shardSlug, "extra"),
|
|
|
|
},
|
2021-05-20 12:58:18 +00:00
|
|
|
{
|
|
|
|
name: "11",
|
|
|
|
path: "/35bb8d560d.ttf",
|
|
|
|
expected: "/" + fileSlug + ".ttf",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "12",
|
|
|
|
path: "/35bb8d560d.woff",
|
|
|
|
expected: "/" + fileSlug + ".woff",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "13",
|
|
|
|
path: "/35bb8d560d.eot",
|
|
|
|
expected: "/" + fileSlug + ".eot",
|
|
|
|
},
|
2020-01-08 19:19:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
actual := normalizePath(tt.path)
|
|
|
|
assert.Equal(t, tt.expected, actual)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2020-02-06 14:04:29 +00:00
|
|
|
|
|
|
|
func TestCors(t *testing.T) {
|
|
|
|
nextHandler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
w.Write([]byte("nextHandler"))
|
|
|
|
})
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
method string
|
|
|
|
headers []string
|
|
|
|
expectedStatus int
|
|
|
|
expectedHeaders map[string]string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "OPTIONS without Origin",
|
|
|
|
method: "OPTIONS",
|
|
|
|
expectedStatus: http.StatusMethodNotAllowed,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "OPTIONS with Origin",
|
|
|
|
method: "OPTIONS",
|
|
|
|
headers: []string{"Origin", "http://myapp.com"},
|
2020-04-07 05:27:02 +00:00
|
|
|
expectedStatus: http.StatusNoContent,
|
2020-02-06 14:04:29 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "GET with Origin",
|
|
|
|
method: "GET",
|
|
|
|
headers: []string{"Origin", "http://anotherapp.com"},
|
|
|
|
expectedStatus: http.StatusOK,
|
|
|
|
expectedHeaders: map[string]string{
|
2020-04-07 05:27:02 +00:00
|
|
|
"Access-Control-Allow-Origin": "http://anotherapp.com",
|
2020-02-06 14:04:29 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
svr := SkipOptions(SetCORS(nextHandler))
|
|
|
|
|
|
|
|
testttp.
|
|
|
|
HTTP(t, tt.method, "/", nil).
|
|
|
|
Headers("", "", tt.headers...).
|
|
|
|
Do(svr).
|
|
|
|
ExpectStatus(tt.expectedStatus).
|
|
|
|
ExpectHeaders(tt.expectedHeaders)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|