2018-01-31 17:07:11 -08:00
|
|
|
package defaults
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/volatiletech/authboss"
|
|
|
|
"github.com/volatiletech/authboss/internal/mocks"
|
|
|
|
)
|
|
|
|
|
|
|
|
type testRenderer struct {
|
|
|
|
Callback func(context.Context, string, authboss.HTMLData) ([]byte, string, error)
|
|
|
|
}
|
|
|
|
|
2018-02-01 15:42:48 -08:00
|
|
|
func (t testRenderer) Load(names ...string) error {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2018-01-31 17:07:11 -08:00
|
|
|
func (t testRenderer) Render(ctx context.Context, name string, data authboss.HTMLData) ([]byte, string, error) {
|
|
|
|
return t.Callback(ctx, name, data)
|
|
|
|
}
|
|
|
|
|
2018-02-01 10:25:54 -08:00
|
|
|
func testJSONRender(ctx context.Context, name string, data authboss.HTMLData) ([]byte, string, error) {
|
|
|
|
b, err := json.Marshal(data)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return b, "application/json", nil
|
|
|
|
}
|
|
|
|
|
2018-01-31 17:07:11 -08:00
|
|
|
func TestResponder(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
renderer := testRenderer{
|
2018-02-01 10:25:54 -08:00
|
|
|
Callback: testJSONRender,
|
2018-01-31 17:07:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
responder := Responder{
|
2018-02-01 10:25:54 -08:00
|
|
|
Renderer: renderer,
|
2018-01-31 17:07:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
r := httptest.NewRequest("GET", "/", nil)
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
2018-02-01 10:25:54 -08:00
|
|
|
r = r.WithContext(context.WithValue(context.Background(), authboss.CTXKeyData, authboss.HTMLData{
|
|
|
|
"csrfname": "csrf",
|
|
|
|
"csrftoken": "12345",
|
|
|
|
}))
|
|
|
|
|
2018-01-31 17:07:11 -08:00
|
|
|
err := responder.Respond(w, r, http.StatusCreated, "some_template.tpl", authboss.HTMLData{"auth_happy": true})
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if w.Code != http.StatusCreated {
|
|
|
|
t.Error("code was wrong:", w.Code)
|
|
|
|
}
|
|
|
|
|
|
|
|
if got := w.HeaderMap.Get("Content-Type"); got != "application/json" {
|
|
|
|
t.Error("content type was wrong:", got)
|
|
|
|
}
|
|
|
|
|
|
|
|
expectData := authboss.HTMLData{
|
2018-02-01 10:25:54 -08:00
|
|
|
"csrfname": "csrf",
|
|
|
|
"csrftoken": "12345",
|
2018-01-31 17:07:11 -08:00
|
|
|
"auth_happy": true,
|
|
|
|
}
|
|
|
|
|
|
|
|
var gotData authboss.HTMLData
|
|
|
|
if err := json.Unmarshal(w.Body.Bytes(), &gotData); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(gotData, expectData) {
|
|
|
|
t.Errorf("data mismatched:\nwant: %#v\ngot: %#v", expectData, gotData)
|
|
|
|
}
|
|
|
|
}
|
2018-02-01 11:51:43 -08:00
|
|
|
|
2018-01-31 17:07:11 -08:00
|
|
|
func TestRedirector(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
renderer := testRenderer{
|
2018-02-01 10:25:54 -08:00
|
|
|
Callback: testJSONRender,
|
2018-01-31 17:07:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
redir := Redirector{
|
|
|
|
FormValueName: "redir",
|
|
|
|
Renderer: renderer,
|
|
|
|
}
|
|
|
|
|
|
|
|
r := httptest.NewRequest("POST", "/?redir=/pow", nil)
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
r.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
|
|
ro := authboss.RedirectOptions{
|
|
|
|
Success: "ok!",
|
|
|
|
Code: http.StatusTeapot,
|
|
|
|
RedirectPath: "/redirect", FollowRedirParam: false,
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := redir.Redirect(w, r, ro); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if w.Code != http.StatusTeapot {
|
|
|
|
t.Error("code is wrong:", w.Code)
|
|
|
|
}
|
|
|
|
|
|
|
|
var gotData map[string]string
|
|
|
|
if err := json.Unmarshal(w.Body.Bytes(), &gotData); err != nil {
|
2018-02-01 10:25:54 -08:00
|
|
|
t.Fatal(err)
|
2018-01-31 17:07:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if got := gotData["status"]; got != "success" {
|
|
|
|
t.Error("status was wrong:", got)
|
|
|
|
}
|
|
|
|
if got := gotData["message"]; got != "ok!" {
|
|
|
|
t.Error("message was wrong:", got)
|
|
|
|
}
|
|
|
|
if got := gotData["location"]; got != "/redirect" {
|
|
|
|
t.Error("location was wrong:", got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResponseRedirectAPIFollowRedir(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
renderer := testRenderer{
|
2018-02-01 10:25:54 -08:00
|
|
|
Callback: testJSONRender,
|
2018-01-31 17:07:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
redir := Redirector{
|
|
|
|
FormValueName: "redir",
|
|
|
|
Renderer: renderer,
|
|
|
|
}
|
|
|
|
|
|
|
|
r := httptest.NewRequest("POST", "/?redir=/pow", nil)
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
r.Header.Set("Content-Type", "application/json")
|
|
|
|
|
|
|
|
ro := authboss.RedirectOptions{
|
|
|
|
Failure: ":(",
|
|
|
|
Code: http.StatusTeapot,
|
|
|
|
RedirectPath: "/redirect", FollowRedirParam: true,
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := redir.Redirect(w, r, ro); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if w.Code != http.StatusTeapot {
|
|
|
|
t.Error("code is wrong:", w.Code)
|
|
|
|
}
|
|
|
|
|
|
|
|
var gotData map[string]string
|
|
|
|
if err := json.Unmarshal(w.Body.Bytes(), &gotData); err != nil {
|
2018-02-01 10:25:54 -08:00
|
|
|
t.Fatal(err)
|
2018-01-31 17:07:11 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if got := gotData["status"]; got != "failure" {
|
|
|
|
t.Error("status was wrong:", got)
|
|
|
|
}
|
|
|
|
if got := gotData["message"]; got != ":(" {
|
|
|
|
t.Error("message was wrong:", got)
|
|
|
|
}
|
|
|
|
if got := gotData["location"]; got != "/pow" {
|
|
|
|
t.Error("location was wrong:", got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResponseRedirectNonAPI(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
renderer := testRenderer{
|
|
|
|
Callback: func(ctx context.Context, name string, data authboss.HTMLData) ([]byte, string, error) {
|
|
|
|
return nil, "", nil
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
redir := Redirector{
|
|
|
|
FormValueName: "redir",
|
|
|
|
Renderer: renderer,
|
|
|
|
}
|
|
|
|
|
|
|
|
r := httptest.NewRequest("POST", "/?redir=/pow", nil)
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
ab := authboss.New()
|
2018-02-01 15:42:48 -08:00
|
|
|
ab.Config.Storage.SessionState = mocks.NewClientRW()
|
|
|
|
ab.Config.Storage.CookieState = mocks.NewClientRW()
|
2018-03-07 16:41:58 -08:00
|
|
|
aw := ab.NewResponse(w)
|
2018-01-31 17:07:11 -08:00
|
|
|
|
|
|
|
ro := authboss.RedirectOptions{
|
|
|
|
Success: "success", Failure: "failure",
|
|
|
|
RedirectPath: "/redirect", FollowRedirParam: false,
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := redir.Redirect(aw, r, ro); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if w.Code != http.StatusFound {
|
|
|
|
t.Error("code is wrong:", w.Code)
|
|
|
|
}
|
|
|
|
if got := w.Header().Get("Location"); got != "/redirect" {
|
|
|
|
t.Error("redirect location was wrong:", got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestResponseRedirectNonAPIFollowRedir(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
renderer := testRenderer{
|
|
|
|
Callback: func(ctx context.Context, name string, data authboss.HTMLData) ([]byte, string, error) {
|
|
|
|
return nil, "", nil
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
redir := Redirector{
|
|
|
|
FormValueName: "redir",
|
|
|
|
Renderer: renderer,
|
|
|
|
}
|
|
|
|
|
|
|
|
r := httptest.NewRequest("POST", "/?redir=/pow", nil)
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
ab := authboss.New()
|
2018-02-01 15:42:48 -08:00
|
|
|
ab.Config.Storage.SessionState = mocks.NewClientRW()
|
|
|
|
ab.Config.Storage.CookieState = mocks.NewClientRW()
|
2018-03-07 16:41:58 -08:00
|
|
|
aw := ab.NewResponse(w)
|
2018-01-31 17:07:11 -08:00
|
|
|
|
|
|
|
ro := authboss.RedirectOptions{
|
|
|
|
RedirectPath: "/redirect", FollowRedirParam: true,
|
|
|
|
}
|
|
|
|
if err := redir.Redirect(aw, r, ro); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if w.Code != http.StatusFound {
|
|
|
|
t.Error("code is wrong:", w.Code)
|
|
|
|
}
|
|
|
|
if got := w.Header().Get("Location"); got != "/pow" {
|
|
|
|
t.Error("redirect location was wrong:", got)
|
|
|
|
}
|
|
|
|
}
|