mirror of
https://github.com/TecharoHQ/anubis.git
synced 2026-04-23 08:36:41 +00:00
681c2cc2ed
* feat(internal): add basic auth HTTP middleware Signed-off-by: Xe Iaso <me@xeiaso.net> * feat(config): add HTTP basic auth for metrics Signed-off-by: Xe Iaso <me@xeiaso.net> * feat(metrics): wire up basic auth Signed-off-by: Xe Iaso <me@xeiaso.net> * doc: document HTTP basic auth for metrics server Signed-off-by: Xe Iaso <me@xeiaso.net> * chore: spelling Signed-off-by: Xe Iaso <me@xeiaso.net> * docs(admin/policies): give people a python command Signed-off-by: Xe Iaso <me@xeiaso.net> --------- Signed-off-by: Xe Iaso <me@xeiaso.net>
138 lines
3.0 KiB
Go
138 lines
3.0 KiB
Go
package internal
|
|
|
|
import (
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"strings"
|
|
"testing"
|
|
)
|
|
|
|
func okHandler() http.Handler {
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusOK)
|
|
_, _ = w.Write([]byte("ok"))
|
|
})
|
|
}
|
|
|
|
func TestBasicAuth(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const (
|
|
realm = "test-realm"
|
|
username = "admin"
|
|
password = "hunter2"
|
|
)
|
|
|
|
for _, tt := range []struct {
|
|
name string
|
|
setAuth bool
|
|
user string
|
|
pass string
|
|
wantStatus int
|
|
wantBody string
|
|
wantChall bool
|
|
}{
|
|
{
|
|
name: "valid credentials",
|
|
setAuth: true,
|
|
user: username,
|
|
pass: password,
|
|
wantStatus: http.StatusOK,
|
|
wantBody: "ok",
|
|
},
|
|
{
|
|
name: "missing credentials",
|
|
setAuth: false,
|
|
wantStatus: http.StatusUnauthorized,
|
|
wantChall: true,
|
|
},
|
|
{
|
|
name: "wrong username",
|
|
setAuth: true,
|
|
user: "nobody",
|
|
pass: password,
|
|
wantStatus: http.StatusUnauthorized,
|
|
wantChall: true,
|
|
},
|
|
{
|
|
name: "wrong password",
|
|
setAuth: true,
|
|
user: username,
|
|
pass: "wrong",
|
|
wantStatus: http.StatusUnauthorized,
|
|
wantChall: true,
|
|
},
|
|
{
|
|
name: "empty supplied credentials",
|
|
setAuth: true,
|
|
user: "",
|
|
pass: "",
|
|
wantStatus: http.StatusUnauthorized,
|
|
wantChall: true,
|
|
},
|
|
} {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
h := BasicAuth(realm, username, password, okHandler())
|
|
|
|
req := httptest.NewRequest(http.MethodGet, "/", nil)
|
|
if tt.setAuth {
|
|
req.SetBasicAuth(tt.user, tt.pass)
|
|
}
|
|
rec := httptest.NewRecorder()
|
|
h.ServeHTTP(rec, req)
|
|
|
|
if rec.Code != tt.wantStatus {
|
|
t.Errorf("status = %d, want %d", rec.Code, tt.wantStatus)
|
|
}
|
|
|
|
if tt.wantBody != "" && rec.Body.String() != tt.wantBody {
|
|
t.Errorf("body = %q, want %q", rec.Body.String(), tt.wantBody)
|
|
}
|
|
|
|
chall := rec.Header().Get("WWW-Authenticate")
|
|
if tt.wantChall {
|
|
if chall == "" {
|
|
t.Error("WWW-Authenticate header missing on 401")
|
|
}
|
|
if !strings.Contains(chall, realm) {
|
|
t.Errorf("WWW-Authenticate = %q, want realm %q", chall, realm)
|
|
}
|
|
} else if chall != "" {
|
|
t.Errorf("unexpected WWW-Authenticate header: %q", chall)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestBasicAuthPassthrough(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, tt := range []struct {
|
|
name string
|
|
username string
|
|
password string
|
|
}{
|
|
{name: "empty username", username: "", password: "hunter2"},
|
|
{name: "empty password", username: "admin", password: ""},
|
|
{name: "both empty", username: "", password: ""},
|
|
} {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
h := BasicAuth("realm", tt.username, tt.password, okHandler())
|
|
|
|
req := httptest.NewRequest(http.MethodGet, "/", nil)
|
|
rec := httptest.NewRecorder()
|
|
h.ServeHTTP(rec, req)
|
|
|
|
if rec.Code != http.StatusOK {
|
|
t.Errorf("status = %d, want %d (passthrough expected)", rec.Code, http.StatusOK)
|
|
}
|
|
if rec.Body.String() != "ok" {
|
|
t.Errorf("body = %q, want %q", rec.Body.String(), "ok")
|
|
}
|
|
})
|
|
}
|
|
} |