2016-07-06 20:24:34 +02:00
package iris
2016-07-07 01:36:38 +02:00
/ *
The most part of the context covered ,
2016-07-07 01:26:05 +02:00
the other part contains serving static methods ,
2016-07-07 01:36:38 +02:00
find remote ip , GetInt and the view engine rendering ( templates )
2016-07-07 01:26:05 +02:00
I am not waiting unexpected behaviors from the rest of the funcs ,
so that ' s all with context ' s tests .
2016-07-07 01:36:38 +02:00
CONTRIBUTE & DISCUSSION ABOUT TESTS TO : https : //github.com/iris-contrib/tests
2016-07-07 01:26:05 +02:00
* /
2016-07-07 00:25:50 +02:00
import (
"encoding/xml"
"net/url"
"strconv"
2016-07-07 01:20:04 +02:00
"strings"
2016-07-07 00:25:50 +02:00
"testing"
2016-07-07 16:02:15 +02:00
"time"
"github.com/valyala/fasthttp"
2016-07-07 00:25:50 +02:00
)
2016-07-06 20:24:34 +02:00
func TestContextReset ( t * testing . T ) {
var context Context
context . Params = PathParameters { PathParameter { Key : "testkey" , Value : "testvalue" } }
context . Reset ( nil )
if len ( context . Params ) > 0 {
t . Fatalf ( "Expecting to have %d params but got: %d" , 0 , len ( context . Params ) )
}
}
func TestContextClone ( t * testing . T ) {
var context Context
context . Params = PathParameters {
PathParameter { Key : "testkey" , Value : "testvalue" } ,
PathParameter { Key : "testkey2" , Value : "testvalue2" } ,
}
c := context . Clone ( )
if v := c . Param ( "testkey" ) ; v != context . Param ( "testkey" ) {
t . Fatalf ( "Expecting to have parameter value: %s but got: %s" , context . Param ( "testkey" ) , v )
}
if v := c . Param ( "testkey2" ) ; v != context . Param ( "testkey2" ) {
t . Fatalf ( "Expecting to have parameter value: %s but got: %s" , context . Param ( "testkey2" ) , v )
}
}
func TestContextDoNextStop ( t * testing . T ) {
var context Context
ok := false
afterStop := false
context . middleware = Middleware { HandlerFunc ( func ( * Context ) {
ok = true
} ) , HandlerFunc ( func ( * Context ) {
ok = true
} ) , HandlerFunc ( func ( * Context ) {
// this will never execute
afterStop = true
} ) }
context . Do ( )
if context . pos != 0 {
t . Fatalf ( "Expecting position 0 for context's middleware but we got: %d" , context . pos )
}
if ! ok {
t . Fatalf ( "Unexpected behavior, first context's middleware didn't executed" )
}
ok = false
context . Next ( )
if int ( context . pos ) != 1 {
t . Fatalf ( "Expecting to have position %d but we got: %d" , 1 , context . pos )
}
if ! ok {
t . Fatalf ( "Next context's middleware didn't executed" )
}
context . StopExecution ( )
if context . pos != stopExecutionPosition {
t . Fatalf ( "Context's StopExecution didn't worked, we expected to have position %d but we got %d" , stopExecutionPosition , context . pos )
}
if ! context . IsStopped ( ) {
t . Fatalf ( "Should be stopped" )
}
context . Next ( )
if afterStop {
t . Fatalf ( "We stopped the execution but the next handler was executed" )
}
}
2016-07-07 00:25:50 +02:00
func TestContextParams ( t * testing . T ) {
2016-07-06 20:24:34 +02:00
var context Context
params := PathParameters {
PathParameter { Key : "testkey" , Value : "testvalue" } ,
PathParameter { Key : "testkey2" , Value : "testvalue2" } ,
PathParameter { Key : "id" , Value : "3" } ,
PathParameter { Key : "bigint" , Value : "548921854390354" } ,
}
context . Params = params
if v := context . Param ( params [ 0 ] . Key ) ; v != params [ 0 ] . Value {
t . Fatalf ( "Expecting parameter value to be %s but we got %s" , params [ 0 ] . Value , context . Param ( "testkey" ) )
}
if v := context . Param ( params [ 1 ] . Key ) ; v != params [ 1 ] . Value {
t . Fatalf ( "Expecting parameter value to be %s but we got %s" , params [ 1 ] . Value , context . Param ( "testkey2" ) )
}
if len ( context . Params ) != len ( params ) {
t . Fatalf ( "Expecting to have %d parameters but we got %d" , len ( params ) , len ( context . Params ) )
}
if vi , err := context . ParamInt ( params [ 2 ] . Key ) ; err != nil {
t . Fatalf ( "Unexpecting error on context's ParamInt while trying to get the integer of the %s" , params [ 2 ] . Value )
} else if vi != 3 {
t . Fatalf ( "Expecting to receive %d but we got %d" , 3 , vi )
}
if vi , err := context . ParamInt64 ( params [ 3 ] . Key ) ; err != nil {
t . Fatalf ( "Unexpecting error on context's ParamInt while trying to get the integer of the %s" , params [ 2 ] . Value )
} else if vi != 548921854390354 {
t . Fatalf ( "Expecting to receive %d but we got %d" , 548921854390354 , vi )
}
2016-07-07 00:25:50 +02:00
// end-to-end test now, note that we will not test the whole mux here, this happens on http_test.go
initDefault ( )
expectedParamsStr := "param1=myparam1,param2=myparam2,param3=myparam3afterstatic,anything=/andhere/anything/you/like"
Get ( "/path/:param1/:param2/staticpath/:param3/*anything" , func ( ctx * Context ) {
paramsStr := ctx . Params . String ( )
ctx . Write ( paramsStr )
} )
Tester ( t ) . GET ( "/path/myparam1/myparam2/staticpath/myparam3afterstatic/andhere/anything/you/like" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( expectedParamsStr )
}
func TestContextURLParams ( t * testing . T ) {
initDefault ( )
passedParams := map [ string ] string { "param1" : "value1" , "param2" : "value2" }
Get ( "/" , func ( ctx * Context ) {
params := ctx . URLParams ( )
ctx . JSON ( StatusOK , params )
} )
e := Tester ( t )
e . GET ( "/" ) . WithQueryObject ( passedParams ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Equal ( passedParams )
}
// hoststring returns the full host, will return the HOST:IP
func TestContextHostString ( t * testing . T ) {
initDefault ( )
Config . Tester . ListeningAddr = "localhost:8080"
Get ( "/" , func ( ctx * Context ) {
ctx . Write ( ctx . HostString ( ) )
} )
Get ( "/wrong" , func ( ctx * Context ) {
ctx . Write ( ctx . HostString ( ) + "w" )
} )
e := Tester ( t )
e . GET ( "/" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( Config . Tester . ListeningAddr )
e . GET ( "/wrong" ) . Expect ( ) . Body ( ) . NotEqual ( Config . Tester . ListeningAddr )
}
// VirtualHostname returns the hostname only,
// if the host starts with 127.0.0.1 or localhost it gives the registered hostname part of the listening addr
func TestContextVirtualHostName ( t * testing . T ) {
initDefault ( )
vhost := "mycustomvirtualname.com"
Config . Tester . ListeningAddr = vhost + ":8080"
Get ( "/" , func ( ctx * Context ) {
ctx . Write ( ctx . VirtualHostname ( ) )
} )
Get ( "/wrong" , func ( ctx * Context ) {
ctx . Write ( ctx . VirtualHostname ( ) + "w" )
} )
e := Tester ( t )
e . GET ( "/" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( vhost )
e . GET ( "/wrong" ) . Expect ( ) . Body ( ) . NotEqual ( vhost )
}
func TestContextFormValueString ( t * testing . T ) {
initDefault ( )
var k , v string
k = "postkey"
v = "postvalue"
Post ( "/" , func ( ctx * Context ) {
ctx . Write ( k + "=" + ctx . FormValueString ( k ) )
} )
e := Tester ( t )
e . POST ( "/" ) . WithFormField ( k , v ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( k + "=" + v )
}
func TestContextSubdomain ( t * testing . T ) {
initDefault ( )
Config . Tester . ListeningAddr = "mydomain.com:9999"
//Config.Tester.ExplicitURL = true
Party ( "mysubdomain." ) . Get ( "/mypath" , func ( ctx * Context ) {
ctx . Write ( ctx . Subdomain ( ) )
} )
e := Tester ( t )
e . GET ( "/" ) . WithURL ( "http://mysubdomain.mydomain.com:9999" ) . Expect ( ) . Status ( StatusNotFound )
e . GET ( "/mypath" ) . WithURL ( "http://mysubdomain.mydomain.com:9999" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( "mysubdomain" )
//e.GET("http://mysubdomain.mydomain.com:9999").Expect().Status(StatusNotFound)
//e.GET("http://mysubdomain.mydomain.com:9999/mypath").Expect().Status(StatusOK).Body().Equal("mysubdomain")
}
type testBinderData struct {
Username string
Mail string
Data [ ] string ` form:"mydata" json:"mydata" `
}
type testBinderXMLData struct {
XMLName xml . Name ` xml:"info" `
FirstAttr string ` xml:"first,attr" `
SecondAttr string ` xml:"second,attr" `
Name string ` xml:"name",json:"name" `
Birth string ` xml:"birth",json:"birth" `
Stars int ` xml:"stars",json:"stars" `
}
func TestContextReadForm ( t * testing . T ) {
initDefault ( )
Post ( "/form" , func ( ctx * Context ) {
obj := testBinderData { }
err := ctx . ReadForm ( & obj )
if err != nil {
t . Fatalf ( "Error when parsing the FORM: %s" , err . Error ( ) )
}
ctx . JSON ( StatusOK , obj )
} )
e := Tester ( t )
passed := map [ string ] interface { } { "Username" : "myusername" , "Mail" : "mymail@iris-go.com" , "mydata" : url . Values { "[0]" : [ ] string { "mydata1" } ,
"[1]" : [ ] string { "mydata2" } } }
expectedObject := testBinderData { Username : "myusername" , Mail : "mymail@iris-go.com" , Data : [ ] string { "mydata1" , "mydata2" } }
e . POST ( "/form" ) . WithForm ( passed ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . Equal ( expectedObject )
}
func TestContextReadJSON ( t * testing . T ) {
initDefault ( )
Post ( "/json" , func ( ctx * Context ) {
obj := testBinderData { }
err := ctx . ReadJSON ( & obj )
if err != nil {
t . Fatalf ( "Error when parsing the JSON body: %s" , err . Error ( ) )
}
ctx . JSON ( StatusOK , obj )
} )
e := Tester ( t )
passed := map [ string ] interface { } { "Username" : "myusername" , "Mail" : "mymail@iris-go.com" , "mydata" : [ ] string { "mydata1" , "mydata2" } }
expectedObject := testBinderData { Username : "myusername" , Mail : "mymail@iris-go.com" , Data : [ ] string { "mydata1" , "mydata2" } }
e . POST ( "/json" ) . WithJSON ( passed ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . Equal ( expectedObject )
}
func TestContextReadXML ( t * testing . T ) {
initDefault ( )
Post ( "/xml" , func ( ctx * Context ) {
obj := testBinderXMLData { }
err := ctx . ReadXML ( & obj )
if err != nil {
t . Fatalf ( "Error when parsing the XML body: %s" , err . Error ( ) )
}
ctx . XML ( StatusOK , obj )
} )
e := Tester ( t )
expectedObj := testBinderXMLData {
XMLName : xml . Name { Local : "info" , Space : "info" } ,
FirstAttr : "this is the first attr" ,
SecondAttr : "this is the second attr" ,
Name : "Iris web framework" ,
Birth : "13 March 2016" ,
Stars : 4064 ,
}
// so far no WithXML or .XML like WithJSON and .JSON on httpexpect I added a feature request as post issue and we're waiting
expectedBody := ` < ` + expectedObj . XMLName . Local + ` first=" ` + expectedObj . FirstAttr + ` " second=" ` + expectedObj . SecondAttr + ` "><name> ` + expectedObj . Name + ` </name><birth> ` + expectedObj . Birth + ` </birth><stars> ` + strconv . Itoa ( expectedObj . Stars ) + ` </stars></info> `
e . POST ( "/xml" ) . WithText ( expectedBody ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( expectedBody )
2016-07-06 20:24:34 +02:00
}
2016-07-07 01:20:04 +02:00
// TestContextRedirectTo tests the named route redirect action
func TestContextRedirectTo ( t * testing . T ) {
initDefault ( )
h := func ( ctx * Context ) { ctx . Write ( ctx . PathString ( ) ) }
Get ( "/mypath" , h ) ( "my-path" )
Get ( "/mypostpath" , h ) ( "my-post-path" )
Get ( "mypath/with/params/:param1/:param2" , func ( ctx * Context ) {
if len ( ctx . Params ) != 2 {
t . Fatalf ( "Strange error, expecting parameters to be two but we got: %d" , len ( ctx . Params ) )
}
ctx . Write ( ctx . PathString ( ) )
} ) ( "my-path-with-params" )
Get ( "/redirect/to/:routeName/*anyparams" , func ( ctx * Context ) {
routeName := ctx . Param ( "routeName" )
var args [ ] interface { }
anyparams := ctx . Param ( "anyparams" )
if anyparams != "" && anyparams != "/" {
params := strings . Split ( anyparams [ 1 : ] , "/" ) // firstparam/secondparam
for _ , s := range params {
args = append ( args , s )
}
}
//println("Redirecting to: " + routeName + " with path: " + Path(routeName, args...))
ctx . RedirectTo ( routeName , args ... )
} )
2016-07-07 01:26:05 +02:00
2016-07-07 01:20:04 +02:00
e := Tester ( t )
2016-07-07 00:25:50 +02:00
2016-07-07 01:20:04 +02:00
e . GET ( "/redirect/to/my-path/" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( "/mypath" )
e . GET ( "/redirect/to/my-post-path/" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( "/mypostpath" )
e . GET ( "/redirect/to/my-path-with-params/firstparam/secondparam" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( "/mypath/with/params/firstparam/secondparam" )
2016-07-07 00:25:50 +02:00
}
func TestContextUserValues ( t * testing . T ) {
initDefault ( )
testCustomObjUserValue := struct { Name string } { Name : "a name" }
values := map [ string ] interface { } { "key1" : "value1" , "key2" : "value2" , "key3" : 3 , "key4" : testCustomObjUserValue , "key5" : map [ string ] string { "key" : "value" } }
Get ( "/test" , func ( ctx * Context ) {
for k , v := range values {
ctx . Set ( k , v )
}
} , func ( ctx * Context ) {
for k , v := range values {
userValue := ctx . Get ( k )
if userValue != v {
t . Fatalf ( "Expecting user value: %s to be equal with: %#v but got: %#v" , k , v , userValue )
}
if m , isMap := userValue . ( map [ string ] string ) ; isMap {
if m [ "key" ] != v . ( map [ string ] string ) [ "key" ] {
t . Fatalf ( "Expecting user value: %s to be equal with: %#v but got: %#v" , k , v . ( map [ string ] string ) [ "key" ] , m [ "key" ] )
}
} else {
if userValue != v {
t . Fatalf ( "Expecting user value: %s to be equal with: %#v but got: %#v" , k , v , userValue )
}
}
}
} )
e := Tester ( t )
e . GET ( "/test" ) . Expect ( ) . Status ( StatusOK )
}
2016-07-07 16:02:15 +02:00
func TestContextCookieSetGetRemove ( t * testing . T ) {
initDefault ( )
key := "mykey"
value := "myvalue"
Get ( "/set" , func ( ctx * Context ) {
ctx . SetCookieKV ( key , value ) // should return non empty cookies
} )
Get ( "/set_advanced" , func ( ctx * Context ) {
c := fasthttp . AcquireCookie ( )
c . SetKey ( key )
c . SetValue ( value )
c . SetHTTPOnly ( true )
c . SetExpire ( time . Now ( ) . Add ( time . Duration ( ( 60 * 60 * 24 * 7 * 4 ) ) * time . Second ) )
ctx . SetCookie ( c )
fasthttp . ReleaseCookie ( c )
} )
Get ( "/get" , func ( ctx * Context ) {
ctx . Write ( ctx . GetCookie ( key ) ) // should return my value
} )
Get ( "/remove" , func ( ctx * Context ) {
ctx . RemoveCookie ( key )
cookieFound := false
ctx . VisitAllCookies ( func ( k , v string ) {
cookieFound = true
} )
if cookieFound {
t . Fatalf ( "Cookie has been found, when it shouldn't!" )
}
ctx . Write ( ctx . GetCookie ( key ) ) // should return ""
} )
e := Tester ( t )
e . GET ( "/set" ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . NotEmpty ( )
e . GET ( "/get" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( value )
e . GET ( "/remove" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( "" )
// test again with advanced set
e . GET ( "/set_advanced" ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . NotEmpty ( )
e . GET ( "/get" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( value )
e . GET ( "/remove" ) . Expect ( ) . Status ( StatusOK ) . Body ( ) . Equal ( "" )
}
2016-07-07 00:25:50 +02:00
func TestContextFlashMessages ( t * testing . T ) {
initDefault ( )
firstKey := "name"
lastKey := "package"
2016-07-07 01:20:04 +02:00
values := PathParameters { PathParameter { Key : firstKey , Value : "kataras" } , PathParameter { Key : lastKey , Value : "iris" } }
jsonExpected := map [ string ] string { firstKey : "kataras" , lastKey : "iris" }
2016-07-07 00:25:50 +02:00
// set the flashes, the cookies are filled
Put ( "/set" , func ( ctx * Context ) {
2016-07-07 01:20:04 +02:00
for _ , v := range values {
ctx . SetFlash ( v . Key , v . Value )
2016-07-07 00:25:50 +02:00
}
} )
// get the first flash, the next should be avaiable to the next requess
Get ( "/get_first_flash" , func ( ctx * Context ) {
2016-07-07 01:20:04 +02:00
for _ , v := range values {
2016-07-08 21:33:41 +02:00
val , err := ctx . GetFlash ( v . Key )
if err == nil {
ctx . JSON ( StatusOK , map [ string ] string { v . Key : val } )
} else {
ctx . JSON ( StatusOK , nil ) // return nil
}
2016-07-07 00:25:50 +02:00
break
}
} )
// just an empty handler to test if the flashes should remeain to the next if GetFlash/GetFlashes used
Get ( "/get_no_getflash" , func ( ctx * Context ) {
} )
// get the last flash, the next should be avaiable to the next requess
Get ( "/get_last_flash" , func ( ctx * Context ) {
2016-07-07 01:20:04 +02:00
for i , v := range values {
if i == len ( values ) - 1 {
2016-07-08 22:11:39 +02:00
val , err := ctx . GetFlash ( v . Key )
if err == nil {
ctx . JSON ( StatusOK , map [ string ] string { v . Key : val } )
} else {
ctx . JSON ( StatusOK , nil ) // return nil
}
2016-07-07 00:25:50 +02:00
}
}
} )
Get ( "/get_zero_flashes" , func ( ctx * Context ) {
ctx . JSON ( StatusOK , ctx . GetFlashes ( ) ) // should return nil
} )
// we use the GetFlash to get the flash messages, the messages and the cookies should be empty after that
Get ( "/get_flash" , func ( ctx * Context ) {
kv := make ( map [ string ] string )
2016-07-07 01:20:04 +02:00
for _ , v := range values {
val , err := ctx . GetFlash ( v . Key )
2016-07-07 00:25:50 +02:00
if err == nil {
2016-07-07 01:20:04 +02:00
kv [ v . Key ] = val
2016-07-07 00:25:50 +02:00
}
}
ctx . JSON ( StatusOK , kv )
} , func ( ctx * Context ) {
// at the same request, flashes should be available
if len ( ctx . GetFlashes ( ) ) == 0 {
t . Fatalf ( "Flashes should be remeain to the whole request lifetime" )
}
} )
Get ( "/get_flashes" , func ( ctx * Context ) {
// one time one handler, using GetFlashes
kv := make ( map [ string ] string )
flashes := ctx . GetFlashes ( )
//second time on the same handler, using the GetFlash
for k := range flashes {
kv [ k ] , _ = ctx . GetFlash ( k )
}
if len ( flashes ) != len ( kv ) {
ctx . SetStatusCode ( StatusNoContent )
return
}
ctx . Next ( )
} , func ( ctx * Context ) {
// third time on a next handler
// test the if next handler has access to them(must) because flash are request lifetime now.
// print them to the client for test the response also
ctx . JSON ( StatusOK , ctx . GetFlashes ( ) )
} )
e := Tester ( t )
e . PUT ( "/set" ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . NotEmpty ( )
e . GET ( "/get_first_flash" ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . ContainsKey ( firstKey ) . NotContainsKey ( lastKey )
// just a request which does not use the flash message, so flash messages should be available on the next request
e . GET ( "/get_no_getflash" ) . Expect ( ) . Status ( StatusOK )
e . GET ( "/get_last_flash" ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . ContainsKey ( lastKey ) . NotContainsKey ( firstKey )
g := e . GET ( "/get_zero_flashes" ) . Expect ( ) . Status ( StatusOK )
g . JSON ( ) . Null ( )
g . Cookies ( ) . Empty ( )
// set the magain
e . PUT ( "/set" ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . NotEmpty ( )
// get them again using GetFlash
2016-07-07 01:20:04 +02:00
e . GET ( "/get_flash" ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . Equal ( jsonExpected )
2016-07-07 00:25:50 +02:00
// this should be empty again
g = e . GET ( "/get_zero_flashes" ) . Expect ( ) . Status ( StatusOK )
g . JSON ( ) . Null ( )
g . Cookies ( ) . Empty ( )
//set them again
e . PUT ( "/set" ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . NotEmpty ( )
// get them again using GetFlashes
2016-07-07 01:20:04 +02:00
e . GET ( "/get_flashes" ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . Equal ( jsonExpected )
2016-07-07 00:25:50 +02:00
// this should be empty again
g = e . GET ( "/get_zero_flashes" ) . Expect ( ) . Status ( StatusOK )
g . JSON ( ) . Null ( )
g . Cookies ( ) . Empty ( )
2016-07-08 21:33:41 +02:00
// test Get, and get again should return nothing
e . PUT ( "/set" ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . NotEmpty ( )
e . GET ( "/get_first_flash" ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . ContainsKey ( firstKey ) . NotContainsKey ( lastKey )
g = e . GET ( "/get_first_flash" ) . Expect ( ) . Status ( StatusOK )
g . JSON ( ) . Null ( )
g . Cookies ( ) . Empty ( )
2016-07-07 00:25:50 +02:00
}
func TestContextSessions ( t * testing . T ) {
t . Parallel ( )
values := map [ string ] interface { } {
"Name" : "iris" ,
"Months" : "4" ,
"Secret" : "dsads£2132215£%%Ssdsa" ,
}
initDefault ( )
Config . Sessions . Cookie = "mycustomsessionid"
writeValues := func ( ctx * Context ) {
sessValues := ctx . Session ( ) . GetAll ( )
ctx . JSON ( StatusOK , sessValues )
}
if testEnableSubdomain {
Party ( testSubdomain + "." ) . Get ( "/get" , func ( ctx * Context ) {
writeValues ( ctx )
} )
}
Post ( "set" , func ( ctx * Context ) {
vals := make ( map [ string ] interface { } , 0 )
if err := ctx . ReadJSON ( & vals ) ; err != nil {
t . Fatalf ( "Cannot readjson. Trace %s" , err . Error ( ) )
}
for k , v := range vals {
ctx . Session ( ) . Set ( k , v )
}
} )
Get ( "/get" , func ( ctx * Context ) {
writeValues ( ctx )
} )
Get ( "/clear" , func ( ctx * Context ) {
ctx . Session ( ) . Clear ( )
writeValues ( ctx )
} )
Get ( "/destroy" , func ( ctx * Context ) {
ctx . SessionDestroy ( )
writeValues ( ctx )
// the cookie and all values should be empty
} )
2016-07-08 22:11:39 +02:00
// request cookie should be empty
Get ( "/after_destroy" , func ( ctx * Context ) {
} )
2016-07-07 00:25:50 +02:00
e := Tester ( t )
e . POST ( "/set" ) . WithJSON ( values ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . NotEmpty ( )
e . GET ( "/get" ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . Equal ( values )
if testEnableSubdomain {
es := subdomainTester ( e )
es . Request ( "GET" , "/get" ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . Equal ( values )
}
2016-07-06 20:24:34 +02:00
2016-07-07 00:25:50 +02:00
// test destory which also clears first
d := e . GET ( "/destroy" ) . Expect ( ) . Status ( StatusOK )
:rainbow: sessions were re-written, update to 4.0.0-alpha.2, read HISTORY.md
**Sessions were re-written **
- Developers can use more than one 'session database', at the same time,
to store the sessions
- Easy to develop a custom session database (only two functions are
required (Load & Update)), [learn
more](https://github.com/iris-contrib/sessiondb/blob/master/redis/database.go)
- Session databases are located
[here](https://github.com/iris-contrib/sessiondb), contributions are
welcome
- The only frontend deleted 'thing' is the: **config.Sessions.Provider**
- No need to register a database, the sessions works out-of-the-box
- No frontend/API changes except the
`context.Session().Set/Delete/Clear`, they doesn't return errors
anymore, btw they (errors) were always nil :)
- Examples (master branch) were updated.
```sh
$ go get github.com/iris-contrib/sessiondb/$DATABASE
```
```go
db := $DATABASE.New(configurationHere{})
iris.UseSessionDB(db)
```
> Note: Book is not updated yet, examples are up-to-date as always.
2016-07-15 19:50:36 +02:00
d . JSON ( ) . Null ( )
2016-07-08 22:11:39 +02:00
// This removed: d.Cookies().Empty(). Reason:
// httpexpect counts the cookies setted or deleted at the response time, but cookie is not removed, to be really removed needs to SetExpire(now-1second) so,
// test if the cookies removed on the next request, like the browser's behavior.
e . GET ( "/after_destroy" ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . Empty ( )
2016-07-07 00:25:50 +02:00
// set and clear again
e . POST ( "/set" ) . WithJSON ( values ) . Expect ( ) . Status ( StatusOK ) . Cookies ( ) . NotEmpty ( )
e . GET ( "/clear" ) . Expect ( ) . Status ( StatusOK ) . JSON ( ) . Object ( ) . Empty ( )
2016-07-06 20:24:34 +02:00
}
2016-07-07 01:36:38 +02:00
type renderTestInformationType struct {
XMLName xml . Name ` xml:"info" `
FirstAttr string ` xml:"first,attr" `
SecondAttr string ` xml:"second,attr" `
Name string ` xml:"name",json:"name" `
Birth string ` xml:"birth",json:"birth" `
Stars int ` xml:"stars",json:"stars" `
}
func TestContextRenderRest ( t * testing . T ) {
initDefault ( )
dataContents := [ ] byte ( "Some binary data here." )
textContents := "Plain text here"
JSONPContents := map [ string ] string { "hello" : "jsonp" }
JSONPCallback := "callbackName"
JSONXMLContents := renderTestInformationType {
XMLName : xml . Name { Local : "info" , Space : "info" } , // only need to verify that later
FirstAttr : "this is the first attr" ,
SecondAttr : "this is the second attr" ,
Name : "Iris web framework" ,
Birth : "13 March 2016" ,
Stars : 4064 ,
}
markdownContents := "# Hello dynamic markdown from Iris"
Get ( "/data" , func ( ctx * Context ) {
ctx . Data ( StatusOK , dataContents )
} )
Get ( "/text" , func ( ctx * Context ) {
ctx . Text ( StatusOK , textContents )
} )
Get ( "/jsonp" , func ( ctx * Context ) {
ctx . JSONP ( StatusOK , JSONPCallback , JSONPContents )
} )
Get ( "/json" , func ( ctx * Context ) {
ctx . JSON ( StatusOK , JSONXMLContents )
} )
Get ( "/xml" , func ( ctx * Context ) {
ctx . XML ( StatusOK , JSONXMLContents )
} )
Get ( "/markdown" , func ( ctx * Context ) {
ctx . Markdown ( StatusOK , markdownContents )
} )
e := Tester ( t )
dataT := e . GET ( "/data" ) . Expect ( ) . Status ( StatusOK )
dataT . Header ( "Content-Type" ) . Equal ( "application/octet-stream" )
dataT . Body ( ) . Equal ( string ( dataContents ) )
textT := e . GET ( "/text" ) . Expect ( ) . Status ( StatusOK )
textT . Header ( "Content-Type" ) . Equal ( "text/plain; charset=UTF-8" )
textT . Body ( ) . Equal ( textContents )
JSONPT := e . GET ( "/jsonp" ) . Expect ( ) . Status ( StatusOK )
JSONPT . Header ( "Content-Type" ) . Equal ( "application/javascript; charset=UTF-8" )
JSONPT . Body ( ) . Equal ( JSONPCallback + ` ( { "hello":"jsonp"}); ` )
JSONT := e . GET ( "/json" ) . Expect ( ) . Status ( StatusOK )
JSONT . Header ( "Content-Type" ) . Equal ( "application/json; charset=UTF-8" )
JSONT . JSON ( ) . Object ( ) . Equal ( JSONXMLContents )
XMLT := e . GET ( "/xml" ) . Expect ( ) . Status ( StatusOK )
XMLT . Header ( "Content-Type" ) . Equal ( "text/xml; charset=UTF-8" )
XMLT . Body ( ) . Equal ( ` < ` + JSONXMLContents . XMLName . Local + ` first=" ` + JSONXMLContents . FirstAttr + ` " second=" ` + JSONXMLContents . SecondAttr + ` "><name> ` + JSONXMLContents . Name + ` </name><birth> ` + JSONXMLContents . Birth + ` </birth><stars> ` + strconv . Itoa ( JSONXMLContents . Stars ) + ` </stars></info> ` )
markdownT := e . GET ( "/markdown" ) . Expect ( ) . Status ( StatusOK )
markdownT . Header ( "Content-Type" ) . Equal ( "text/html; charset=UTF-8" )
markdownT . Body ( ) . Equal ( "<h1>" + markdownContents [ 2 : ] + "</h1>\n" )
}