2020-04-02 20:14:43 +02:00
|
|
|
// Copyright 2012 The Gorilla Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
2020-04-02 20:02:33 +02:00
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2020-04-02 20:14:43 +02:00
|
|
|
package mongostore_test
|
2020-04-02 20:02:33 +02:00
|
|
|
|
|
|
|
import (
|
2020-04-02 20:14:43 +02:00
|
|
|
"bytes"
|
2020-04-02 20:02:33 +02:00
|
|
|
"context"
|
|
|
|
"encoding/gob"
|
2020-04-02 20:14:43 +02:00
|
|
|
"github.com/dalu/mongostore"
|
2020-04-02 20:02:33 +02:00
|
|
|
"github.com/gorilla/sessions"
|
2020-04-02 20:03:54 +02:00
|
|
|
"go.mongodb.org/mongo-driver/mongo"
|
|
|
|
"go.mongodb.org/mongo-driver/mongo/options"
|
|
|
|
"go.mongodb.org/mongo-driver/mongo/readpref"
|
2020-04-02 20:14:43 +02:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"testing"
|
|
|
|
"time"
|
2020-04-02 20:02:33 +02:00
|
|
|
)
|
|
|
|
|
2020-04-02 20:14:43 +02:00
|
|
|
// NewRecorder returns an initialized ResponseRecorder.
|
|
|
|
func NewRecorder() *httptest.ResponseRecorder {
|
|
|
|
return &httptest.ResponseRecorder{
|
|
|
|
HeaderMap: make(http.Header),
|
|
|
|
Body: new(bytes.Buffer),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2020-04-02 20:02:33 +02:00
|
|
|
type FlashMessage struct {
|
|
|
|
Type int
|
|
|
|
Message string
|
|
|
|
}
|
|
|
|
|
2020-04-02 20:14:43 +02:00
|
|
|
func TestFlashes(t *testing.T) {
|
2020-04-02 20:02:33 +02:00
|
|
|
var req *http.Request
|
|
|
|
var rsp *httptest.ResponseRecorder
|
|
|
|
var hdr http.Header
|
|
|
|
var err error
|
|
|
|
var ok bool
|
|
|
|
var cookies []string
|
|
|
|
var session *sessions.Session
|
|
|
|
var flashes []interface{}
|
|
|
|
|
|
|
|
ctx, _ := context.WithTimeout(context.Background(), 3*time.Second)
|
|
|
|
client, err := mongo.Connect(ctx, options.Client().ApplyURI("mongodb://localhost:27017"))
|
|
|
|
err = client.Ping(ctx, readpref.Primary())
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2020-04-02 20:14:43 +02:00
|
|
|
store := mongostore.NewMongoStore(client, "test", "sessions", 3*time.Second)
|
2020-04-02 20:02:33 +02:00
|
|
|
store.Options.Path = "/"
|
|
|
|
store.Options.MaxAge = 86400 * 30 * 365
|
|
|
|
defer store.Close()
|
|
|
|
|
2020-04-02 20:14:43 +02:00
|
|
|
// Round 1 ----------------------------------------------------------------
|
|
|
|
|
2020-04-02 20:02:33 +02:00
|
|
|
req, _ = http.NewRequest("GET", "http://localhost:8080/", nil)
|
2020-04-02 20:14:43 +02:00
|
|
|
rsp = NewRecorder()
|
2020-04-02 20:02:33 +02:00
|
|
|
// Get a session.
|
|
|
|
if session, err = store.Get(req, "session-key"); err != nil {
|
|
|
|
t.Fatalf("Error getting session: %v", err)
|
|
|
|
}
|
|
|
|
// Get a flash.
|
|
|
|
flashes = session.Flashes()
|
|
|
|
if len(flashes) != 0 {
|
|
|
|
t.Errorf("Expected empty flashes; Got %v", flashes)
|
|
|
|
}
|
|
|
|
// Add some flashes.
|
|
|
|
session.AddFlash("foo")
|
|
|
|
session.AddFlash("bar")
|
|
|
|
// Custom key.
|
|
|
|
session.AddFlash("baz", "custom_key")
|
|
|
|
// Save.
|
|
|
|
if err = sessions.Save(req, rsp); err != nil {
|
|
|
|
t.Fatalf("Error saving session: %v", err)
|
|
|
|
}
|
|
|
|
hdr = rsp.Header()
|
|
|
|
cookies, ok = hdr["Set-Cookie"]
|
|
|
|
if !ok || len(cookies) != 1 {
|
2020-04-02 20:14:43 +02:00
|
|
|
t.Fatal("No cookies. Header:", hdr)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err = store.Get(req, "session:key"); err.Error() != "sessions: invalid character in cookie name: session:key" {
|
|
|
|
t.Fatalf("Expected error due to invalid cookie name")
|
2020-04-02 20:02:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Round 2 ----------------------------------------------------------------
|
|
|
|
|
|
|
|
req, _ = http.NewRequest("GET", "http://localhost:8080/", nil)
|
|
|
|
req.Header.Add("Cookie", cookies[0])
|
2020-04-02 20:14:43 +02:00
|
|
|
rsp = NewRecorder()
|
2020-04-02 20:02:33 +02:00
|
|
|
// Get a session.
|
|
|
|
if session, err = store.Get(req, "session-key"); err != nil {
|
|
|
|
t.Fatalf("Error getting session: %v", err)
|
|
|
|
}
|
|
|
|
// Check all saved values.
|
|
|
|
flashes = session.Flashes()
|
|
|
|
if len(flashes) != 2 {
|
|
|
|
t.Fatalf("Expected flashes; Got %v", flashes)
|
|
|
|
}
|
|
|
|
if flashes[0] != "foo" || flashes[1] != "bar" {
|
|
|
|
t.Errorf("Expected foo,bar; Got %v", flashes)
|
|
|
|
}
|
|
|
|
flashes = session.Flashes()
|
|
|
|
if len(flashes) != 0 {
|
|
|
|
t.Errorf("Expected dumped flashes; Got %v", flashes)
|
|
|
|
}
|
|
|
|
// Custom key.
|
|
|
|
flashes = session.Flashes("custom_key")
|
|
|
|
if len(flashes) != 1 {
|
|
|
|
t.Errorf("Expected flashes; Got %v", flashes)
|
|
|
|
} else if flashes[0] != "baz" {
|
|
|
|
t.Errorf("Expected baz; Got %v", flashes)
|
|
|
|
}
|
|
|
|
flashes = session.Flashes("custom_key")
|
|
|
|
if len(flashes) != 0 {
|
|
|
|
t.Errorf("Expected dumped flashes; Got %v", flashes)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Round 3 ----------------------------------------------------------------
|
|
|
|
// Custom type
|
|
|
|
|
|
|
|
req, _ = http.NewRequest("GET", "http://localhost:8080/", nil)
|
2020-04-02 20:14:43 +02:00
|
|
|
rsp = NewRecorder()
|
2020-04-02 20:02:33 +02:00
|
|
|
// Get a session.
|
|
|
|
if session, err = store.Get(req, "session-key"); err != nil {
|
|
|
|
t.Fatalf("Error getting session: %v", err)
|
|
|
|
}
|
|
|
|
// Get a flash.
|
|
|
|
flashes = session.Flashes()
|
|
|
|
if len(flashes) != 0 {
|
|
|
|
t.Errorf("Expected empty flashes; Got %v", flashes)
|
|
|
|
}
|
|
|
|
// Add some flashes.
|
|
|
|
session.AddFlash(&FlashMessage{42, "foo"})
|
|
|
|
// Save.
|
|
|
|
if err = sessions.Save(req, rsp); err != nil {
|
|
|
|
t.Fatalf("Error saving session: %v", err)
|
|
|
|
}
|
|
|
|
hdr = rsp.Header()
|
|
|
|
cookies, ok = hdr["Set-Cookie"]
|
|
|
|
if !ok || len(cookies) != 1 {
|
2020-04-02 20:14:43 +02:00
|
|
|
t.Fatal("No cookies. Header:", hdr)
|
2020-04-02 20:02:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Round 4 ----------------------------------------------------------------
|
|
|
|
// Custom type
|
|
|
|
|
|
|
|
req, _ = http.NewRequest("GET", "http://localhost:8080/", nil)
|
|
|
|
req.Header.Add("Cookie", cookies[0])
|
2020-04-02 20:14:43 +02:00
|
|
|
rsp = NewRecorder()
|
2020-04-02 20:02:33 +02:00
|
|
|
// Get a session.
|
|
|
|
if session, err = store.Get(req, "session-key"); err != nil {
|
|
|
|
t.Fatalf("Error getting session: %v", err)
|
|
|
|
}
|
|
|
|
// Check all saved values.
|
|
|
|
flashes = session.Flashes()
|
|
|
|
if len(flashes) != 1 {
|
|
|
|
t.Fatalf("Expected flashes; Got %v", flashes)
|
|
|
|
}
|
|
|
|
custom := flashes[0].(FlashMessage)
|
|
|
|
if custom.Type != 42 || custom.Message != "foo" {
|
|
|
|
t.Errorf("Expected %#v, got %#v", FlashMessage{42, "foo"}, custom)
|
|
|
|
}
|
|
|
|
|
2020-04-02 20:14:43 +02:00
|
|
|
// Round 5 ----------------------------------------------------------------
|
|
|
|
// Check if a request shallow copy resets the request context data store.
|
|
|
|
|
|
|
|
req, _ = http.NewRequest("GET", "http://localhost:8080/", nil)
|
|
|
|
|
|
|
|
// Get a session.
|
|
|
|
if session, err = store.Get(req, "session-key"); err != nil {
|
|
|
|
t.Fatalf("Error getting session: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put a test value into the session data store.
|
|
|
|
session.Values["test"] = "test-value"
|
|
|
|
|
|
|
|
// Create a shallow copy of the request.
|
|
|
|
req = req.WithContext(req.Context())
|
|
|
|
|
|
|
|
// Get the session again.
|
|
|
|
if session, err = store.Get(req, "session-key"); err != nil {
|
|
|
|
t.Fatalf("Error getting session: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the previous inserted value still exists.
|
|
|
|
if session.Values["test"] == nil {
|
|
|
|
t.Fatalf("Session test value is lost in the request context!")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if the previous inserted value has the same value.
|
|
|
|
if session.Values["test"] != "test-value" {
|
|
|
|
t.Fatalf("Session test value is changed in the request context!")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCookieStoreMapPanic(t *testing.T) {
|
|
|
|
defer func() {
|
|
|
|
err := recover()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
ctx, _ := context.WithTimeout(context.Background(), 3*time.Second)
|
|
|
|
client, err := mongo.Connect(ctx, options.Client().ApplyURI("mongodb://localhost:27017"))
|
|
|
|
err = client.Ping(ctx, readpref.Primary())
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
store := mongostore.NewMongoStore(client, "test", "sessions", 3*time.Second, []byte("aaa0defe5d2839cbc46fc4f080cd7adc"))
|
|
|
|
store.Options.Path = "/"
|
|
|
|
store.Options.MaxAge = 86400 * 30 * 365
|
|
|
|
defer store.Close()
|
|
|
|
|
|
|
|
req, err := http.NewRequest("GET", "http://www.example.com", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("failed to create request", err)
|
|
|
|
}
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
session := sessions.NewSession(store, "hello")
|
|
|
|
|
|
|
|
session.Values["data"] = "hello-world"
|
|
|
|
|
|
|
|
err = session.Save(req, w)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("failed to save session", err)
|
2020-04-02 20:02:33 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
gob.Register(FlashMessage{})
|
|
|
|
}
|