2017-04-25 10:24:51 +03:00
// Copyright 2017 The Gitea Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package integrations
import (
"bytes"
2017-04-30 09:30:12 +03:00
"database/sql"
2017-06-17 07:49:45 +03:00
"encoding/json"
2017-04-25 10:24:51 +03:00
"fmt"
"io"
2017-04-30 09:30:12 +03:00
"log"
2017-04-25 10:24:51 +03:00
"net/http"
2017-05-02 03:49:55 +03:00
"net/http/cookiejar"
"net/url"
2017-04-25 10:24:51 +03:00
"os"
2017-06-09 21:13:46 +03:00
"path"
2017-05-02 03:49:55 +03:00
"strings"
2017-04-25 10:24:51 +03:00
"testing"
"code.gitea.io/gitea/models"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/routers"
"code.gitea.io/gitea/routers/routes"
2017-04-28 16:23:28 +03:00
"github.com/Unknwon/com"
"github.com/stretchr/testify/assert"
2017-04-25 10:24:51 +03:00
"gopkg.in/macaron.v1"
"gopkg.in/testfixtures.v2"
)
var mac * macaron . Macaron
func TestMain ( m * testing . M ) {
2017-04-30 09:30:12 +03:00
initIntegrationTest ( )
2017-04-25 10:24:51 +03:00
mac = routes . NewMacaron ( )
routes . RegisterRoutes ( mac )
var helper testfixtures . Helper
if setting . UseMySQL {
helper = & testfixtures . MySQL { }
} else if setting . UsePostgreSQL {
helper = & testfixtures . PostgreSQL { }
} else if setting . UseSQLite3 {
helper = & testfixtures . SQLite { }
} else {
fmt . Println ( "Unsupported RDBMS for integration tests" )
os . Exit ( 1 )
}
err := models . InitFixtures (
helper ,
2017-04-28 16:23:28 +03:00
"models/fixtures/" ,
2017-04-25 10:24:51 +03:00
)
if err != nil {
fmt . Printf ( "Error initializing test database: %v\n" , err )
os . Exit ( 1 )
}
os . Exit ( m . Run ( ) )
}
2017-04-30 09:30:12 +03:00
func initIntegrationTest ( ) {
2017-06-10 05:27:13 +03:00
giteaRoot := os . Getenv ( "GITEA_ROOT" )
if giteaRoot == "" {
fmt . Println ( "Environment variable $GITEA_ROOT not set" )
2017-04-30 09:30:12 +03:00
os . Exit ( 1 )
}
2017-06-10 05:27:13 +03:00
setting . AppPath = path . Join ( giteaRoot , "gitea" )
giteaConf := os . Getenv ( "GITEA_CONF" )
if giteaConf == "" {
fmt . Println ( "Environment variable $GITEA_CONF not set" )
2017-05-02 03:49:55 +03:00
os . Exit ( 1 )
2017-06-10 05:27:13 +03:00
} else if ! path . IsAbs ( giteaConf ) {
setting . CustomConf = path . Join ( giteaRoot , giteaConf )
2017-06-09 21:13:46 +03:00
} else {
2017-06-10 05:27:13 +03:00
setting . CustomConf = giteaConf
2017-05-02 03:49:55 +03:00
}
2017-04-30 09:30:12 +03:00
setting . NewContext ( )
models . LoadConfigs ( )
switch {
case setting . UseMySQL :
db , err := sql . Open ( "mysql" , fmt . Sprintf ( "%s:%s@tcp(%s)/" ,
models . DbCfg . User , models . DbCfg . Passwd , models . DbCfg . Host ) )
defer db . Close ( )
if err != nil {
log . Fatalf ( "sql.Open: %v" , err )
}
if _ , err = db . Exec ( "CREATE DATABASE IF NOT EXISTS testgitea" ) ; err != nil {
log . Fatalf ( "db.Exec: %v" , err )
}
case setting . UsePostgreSQL :
db , err := sql . Open ( "postgres" , fmt . Sprintf ( "postgres://%s:%s@%s/?sslmode=%s" ,
models . DbCfg . User , models . DbCfg . Passwd , models . DbCfg . Host , models . DbCfg . SSLMode ) )
defer db . Close ( )
if err != nil {
log . Fatalf ( "sql.Open: %v" , err )
}
rows , err := db . Query ( fmt . Sprintf ( "SELECT 1 FROM pg_database WHERE datname = '%s'" ,
models . DbCfg . Name ) )
if err != nil {
log . Fatalf ( "db.Query: %v" , err )
}
2017-05-08 05:55:27 +03:00
defer rows . Close ( )
2017-05-09 16:42:55 +03:00
if rows . Next ( ) {
2017-05-11 18:32:43 +03:00
break
2017-04-30 09:30:12 +03:00
}
2017-05-09 16:42:55 +03:00
if _ , err = db . Exec ( "CREATE DATABASE testgitea" ) ; err != nil {
log . Fatalf ( "db.Exec: %v" , err )
}
2017-04-30 09:30:12 +03:00
}
routers . GlobalInit ( )
}
2017-06-17 18:01:03 +03:00
func prepareTestEnv ( t testing . TB ) {
2017-04-28 16:23:28 +03:00
assert . NoError ( t , models . LoadFixtures ( ) )
2017-09-12 08:51:12 +03:00
assert . NoError ( t , os . RemoveAll ( setting . RepoRootPath ) )
assert . NoError ( t , com . CopyDir ( "integrations/gitea-repositories-meta" , setting . RepoRootPath ) )
2017-04-28 16:23:28 +03:00
}
2017-05-02 03:49:55 +03:00
type TestSession struct {
jar http . CookieJar
}
func ( s * TestSession ) GetCookie ( name string ) * http . Cookie {
baseURL , err := url . Parse ( setting . AppURL )
if err != nil {
return nil
}
for _ , c := range s . jar . Cookies ( baseURL ) {
if c . Name == name {
return c
}
}
return nil
}
2017-07-07 22:36:47 +03:00
func ( s * TestSession ) MakeRequest ( t testing . TB , req * http . Request , expectedStatus int ) * TestResponse {
2017-05-02 03:49:55 +03:00
baseURL , err := url . Parse ( setting . AppURL )
assert . NoError ( t , err )
for _ , c := range s . jar . Cookies ( baseURL ) {
req . AddCookie ( c )
}
2017-07-07 22:36:47 +03:00
resp := MakeRequest ( t , req , expectedStatus )
2017-05-02 03:49:55 +03:00
ch := http . Header { }
ch . Add ( "Cookie" , strings . Join ( resp . Headers [ "Set-Cookie" ] , ";" ) )
cr := http . Request { Header : ch }
s . jar . SetCookies ( baseURL , cr . Cookies ( ) )
return resp
}
2017-06-17 07:49:45 +03:00
const userPassword = "password"
2017-07-09 05:07:29 +03:00
var loginSessionCache = make ( map [ string ] * TestSession , 10 )
2017-08-23 10:30:33 +03:00
func emptyTestSession ( t testing . TB ) * TestSession {
jar , err := cookiejar . New ( nil )
assert . NoError ( t , err )
return & TestSession { jar : jar }
}
2017-06-17 18:01:03 +03:00
func loginUser ( t testing . TB , userName string ) * TestSession {
2017-07-09 05:07:29 +03:00
if session , ok := loginSessionCache [ userName ] ; ok {
return session
}
session := loginUserWithPassword ( t , userName , userPassword )
loginSessionCache [ userName ] = session
return session
2017-06-17 07:49:45 +03:00
}
2017-06-17 18:01:03 +03:00
func loginUserWithPassword ( t testing . TB , userName , password string ) * TestSession {
2017-06-10 03:41:36 +03:00
req := NewRequest ( t , "GET" , "/user/login" )
2017-07-07 22:36:47 +03:00
resp := MakeRequest ( t , req , http . StatusOK )
2017-05-02 03:49:55 +03:00
2017-06-17 19:29:59 +03:00
doc := NewHTMLParser ( t , resp . Body )
2017-06-17 07:49:45 +03:00
req = NewRequestWithValues ( t , "POST" , "/user/login" , map [ string ] string {
"_csrf" : doc . GetCSRF ( ) ,
"user_name" : userName ,
"password" : password ,
} )
2017-07-07 22:36:47 +03:00
resp = MakeRequest ( t , req , http . StatusFound )
2017-05-02 03:49:55 +03:00
ch := http . Header { }
ch . Add ( "Cookie" , strings . Join ( resp . Headers [ "Set-Cookie" ] , ";" ) )
cr := http . Request { Header : ch }
2017-08-23 10:30:33 +03:00
session := emptyTestSession ( t )
2017-05-02 03:49:55 +03:00
baseURL , err := url . Parse ( setting . AppURL )
assert . NoError ( t , err )
2017-08-23 10:30:33 +03:00
session . jar . SetCookies ( baseURL , cr . Cookies ( ) )
2017-05-02 03:49:55 +03:00
2017-08-23 10:30:33 +03:00
return session
2017-05-02 03:49:55 +03:00
}
2017-04-25 10:24:51 +03:00
type TestResponseWriter struct {
HeaderCode int
Writer io . Writer
2017-05-02 03:49:55 +03:00
Headers http . Header
2017-04-25 10:24:51 +03:00
}
func ( w * TestResponseWriter ) Header ( ) http . Header {
2017-05-02 03:49:55 +03:00
return w . Headers
2017-04-25 10:24:51 +03:00
}
func ( w * TestResponseWriter ) Write ( b [ ] byte ) ( int , error ) {
return w . Writer . Write ( b )
}
func ( w * TestResponseWriter ) WriteHeader ( n int ) {
w . HeaderCode = n
}
type TestResponse struct {
HeaderCode int
Body [ ] byte
2017-05-02 03:49:55 +03:00
Headers http . Header
2017-04-25 10:24:51 +03:00
}
2017-06-17 18:01:03 +03:00
func NewRequest ( t testing . TB , method , urlStr string ) * http . Request {
2017-06-17 07:49:45 +03:00
return NewRequestWithBody ( t , method , urlStr , nil )
}
2017-06-25 03:15:42 +03:00
func NewRequestf ( t testing . TB , method , urlFormat string , args ... interface { } ) * http . Request {
return NewRequest ( t , method , fmt . Sprintf ( urlFormat , args ... ) )
}
2017-06-17 18:01:03 +03:00
func NewRequestWithValues ( t testing . TB , method , urlStr string , values map [ string ] string ) * http . Request {
2017-06-17 07:49:45 +03:00
urlValues := url . Values { }
for key , value := range values {
urlValues [ key ] = [ ] string { value }
}
2017-06-25 03:15:42 +03:00
req := NewRequestWithBody ( t , method , urlStr , bytes . NewBufferString ( urlValues . Encode ( ) ) )
req . Header . Add ( "Content-Type" , "application/x-www-form-urlencoded" )
return req
2017-06-17 07:49:45 +03:00
}
2017-06-17 18:01:03 +03:00
func NewRequestWithJSON ( t testing . TB , method , urlStr string , v interface { } ) * http . Request {
2017-06-17 07:49:45 +03:00
jsonBytes , err := json . Marshal ( v )
assert . NoError ( t , err )
2017-06-25 02:52:51 +03:00
req := NewRequestWithBody ( t , method , urlStr , bytes . NewBuffer ( jsonBytes ) )
req . Header . Add ( "Content-Type" , "application/json" )
return req
2017-06-10 03:41:36 +03:00
}
2017-06-17 18:01:03 +03:00
func NewRequestWithBody ( t testing . TB , method , urlStr string , body io . Reader ) * http . Request {
2017-06-17 07:49:45 +03:00
request , err := http . NewRequest ( method , urlStr , body )
2017-06-10 03:41:36 +03:00
assert . NoError ( t , err )
2017-06-17 07:49:45 +03:00
request . RequestURI = urlStr
2017-06-10 03:41:36 +03:00
return request
}
2017-07-07 22:36:47 +03:00
const NoExpectedStatus = - 1
func MakeRequest ( t testing . TB , req * http . Request , expectedStatus int ) * TestResponse {
2017-04-25 10:24:51 +03:00
buffer := bytes . NewBuffer ( nil )
respWriter := & TestResponseWriter {
2017-05-02 03:49:55 +03:00
Writer : buffer ,
Headers : make ( map [ string ] [ ] string ) ,
2017-04-25 10:24:51 +03:00
}
mac . ServeHTTP ( respWriter , req )
2017-07-07 22:36:47 +03:00
if expectedStatus != NoExpectedStatus {
2017-08-28 16:36:02 +03:00
assert . EqualValues ( t , expectedStatus , respWriter . HeaderCode ,
2017-09-14 11:16:22 +03:00
"Request URL: %s %s" , req . URL . String ( ) , buffer . String ( ) )
2017-07-07 22:36:47 +03:00
}
2017-04-25 10:24:51 +03:00
return & TestResponse {
HeaderCode : respWriter . HeaderCode ,
Body : buffer . Bytes ( ) ,
2017-05-02 03:49:55 +03:00
Headers : respWriter . Headers ,
2017-04-25 10:24:51 +03:00
}
}
2017-06-18 12:06:17 +03:00
func DecodeJSON ( t testing . TB , resp * TestResponse , v interface { } ) {
decoder := json . NewDecoder ( bytes . NewBuffer ( resp . Body ) )
assert . NoError ( t , decoder . Decode ( v ) )
}
2017-07-07 22:36:47 +03:00
func GetCSRF ( t testing . TB , session * TestSession , urlStr string ) string {
req := NewRequest ( t , "GET" , urlStr )
resp := session . MakeRequest ( t , req , http . StatusOK )
doc := NewHTMLParser ( t , resp . Body )
return doc . GetCSRF ( )
}
func RedirectURL ( t testing . TB , resp * TestResponse ) string {
urlSlice := resp . Headers [ "Location" ]
assert . NotEmpty ( t , urlSlice , "No redirect URL founds" )
return urlSlice [ 0 ]
}