2023-09-16 22:59:47 +00:00
|
|
|
package test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"net/url"
|
|
|
|
"os"
|
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"reflect"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
2024-05-11 19:03:25 +00:00
|
|
|
|
|
|
|
"github.com/rs/zerolog/log"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/thomiceli/opengist/internal/config"
|
|
|
|
"github.com/thomiceli/opengist/internal/db"
|
|
|
|
"github.com/thomiceli/opengist/internal/git"
|
|
|
|
"github.com/thomiceli/opengist/internal/memdb"
|
|
|
|
"github.com/thomiceli/opengist/internal/web"
|
2023-09-16 22:59:47 +00:00
|
|
|
)
|
|
|
|
|
2024-09-20 14:01:09 +00:00
|
|
|
var databaseType string
|
|
|
|
|
2023-09-16 22:59:47 +00:00
|
|
|
type testServer struct {
|
|
|
|
server *web.Server
|
|
|
|
sessionCookie string
|
|
|
|
}
|
|
|
|
|
|
|
|
func newTestServer() (*testServer, error) {
|
|
|
|
s := &testServer{
|
2024-05-27 23:30:08 +00:00
|
|
|
server: web.NewServer(true, path.Join(config.GetHomeDir(), "tmp", "sessions")),
|
2023-09-16 22:59:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
go s.start()
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *testServer) start() {
|
|
|
|
s.server.Start()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *testServer) stop() {
|
|
|
|
s.server.Stop()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *testServer) request(method, uri string, data interface{}, expectedCode int) error {
|
|
|
|
var bodyReader io.Reader
|
|
|
|
if method == http.MethodPost || method == http.MethodPut {
|
|
|
|
values := structToURLValues(data)
|
|
|
|
bodyReader = strings.NewReader(values.Encode())
|
|
|
|
}
|
|
|
|
|
|
|
|
req := httptest.NewRequest(method, "http://localhost:6157"+uri, bodyReader)
|
|
|
|
w := httptest.NewRecorder()
|
|
|
|
|
|
|
|
if method == http.MethodPost || method == http.MethodPut {
|
|
|
|
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
|
|
|
}
|
|
|
|
|
|
|
|
if s.sessionCookie != "" {
|
|
|
|
req.AddCookie(&http.Cookie{Name: "session", Value: s.sessionCookie})
|
|
|
|
}
|
|
|
|
|
|
|
|
s.server.ServeHTTP(w, req)
|
|
|
|
|
|
|
|
if w.Code != expectedCode {
|
|
|
|
return fmt.Errorf("unexpected status code %d, expected %d", w.Code, expectedCode)
|
|
|
|
}
|
|
|
|
|
|
|
|
if method == http.MethodPost {
|
|
|
|
if strings.Contains(uri, "/login") || strings.Contains(uri, "/register") {
|
|
|
|
cookie := ""
|
|
|
|
h := w.Header().Get("Set-Cookie")
|
|
|
|
parts := strings.Split(h, "; ")
|
|
|
|
for _, p := range parts {
|
|
|
|
if strings.HasPrefix(p, "session=") {
|
|
|
|
cookie = p
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if cookie == "" {
|
|
|
|
return errors.New("unable to find access session token in response headers")
|
|
|
|
}
|
|
|
|
s.sessionCookie = strings.TrimPrefix(cookie, "session=")
|
|
|
|
} else if strings.Contains(uri, "/logout") {
|
|
|
|
s.sessionCookie = ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func structToURLValues(s interface{}) url.Values {
|
|
|
|
v := url.Values{}
|
|
|
|
if s == nil {
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
|
|
|
rValue := reflect.ValueOf(s)
|
|
|
|
if rValue.Kind() != reflect.Struct {
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < rValue.NumField(); i++ {
|
|
|
|
field := rValue.Type().Field(i)
|
|
|
|
tag := field.Tag.Get("form")
|
2024-05-11 19:03:25 +00:00
|
|
|
if tag != "" || field.Anonymous {
|
2023-09-16 22:59:47 +00:00
|
|
|
if field.Type.Kind() == reflect.Int {
|
|
|
|
fieldValue := rValue.Field(i).Int()
|
|
|
|
v.Add(tag, strconv.FormatInt(fieldValue, 10))
|
|
|
|
} else if field.Type.Kind() == reflect.Slice {
|
|
|
|
fieldValue := rValue.Field(i).Interface().([]string)
|
|
|
|
for _, va := range fieldValue {
|
|
|
|
v.Add(tag, va)
|
|
|
|
}
|
2024-05-11 19:03:25 +00:00
|
|
|
} else if field.Type.Kind() == reflect.Struct {
|
|
|
|
for key, val := range structToURLValues(rValue.Field(i).Interface()) {
|
|
|
|
for _, vv := range val {
|
|
|
|
v.Add(key, vv)
|
|
|
|
}
|
|
|
|
}
|
2023-09-16 22:59:47 +00:00
|
|
|
} else {
|
|
|
|
fieldValue := rValue.Field(i).String()
|
|
|
|
v.Add(tag, fieldValue)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
|
|
|
func setup(t *testing.T) {
|
2024-09-20 14:01:09 +00:00
|
|
|
var databaseDsn string
|
|
|
|
databaseType = os.Getenv("OPENGIST_TEST_DB")
|
|
|
|
switch databaseType {
|
|
|
|
case "sqlite":
|
|
|
|
databaseDsn = "file::memory:"
|
|
|
|
case "postgres":
|
|
|
|
databaseDsn = "postgres://postgres:opengist@localhost:5432/opengist_test"
|
|
|
|
case "mysql":
|
|
|
|
databaseDsn = "mysql://root:opengist@localhost:3306/opengist_test"
|
|
|
|
}
|
|
|
|
|
2024-01-23 19:24:01 +00:00
|
|
|
_ = os.Setenv("OPENGIST_SKIP_GIT_HOOKS", "1")
|
|
|
|
|
2024-01-29 23:07:57 +00:00
|
|
|
err := config.InitConfig("", io.Discard)
|
2023-09-16 22:59:47 +00:00
|
|
|
require.NoError(t, err, "Could not init config")
|
|
|
|
|
|
|
|
err = os.MkdirAll(filepath.Join(config.GetHomeDir()), 0755)
|
|
|
|
require.NoError(t, err, "Could not create Opengist home directory")
|
|
|
|
|
|
|
|
git.ReposDirectory = path.Join("tests")
|
|
|
|
|
2024-01-04 02:38:15 +00:00
|
|
|
config.C.IndexEnabled = false
|
|
|
|
config.C.LogLevel = "debug"
|
2023-09-16 22:59:47 +00:00
|
|
|
config.InitLog()
|
|
|
|
|
|
|
|
homePath := config.GetHomeDir()
|
|
|
|
log.Info().Msg("Data directory: " + homePath)
|
|
|
|
|
2024-05-27 23:30:08 +00:00
|
|
|
err = os.MkdirAll(filepath.Join(homePath, "tmp", "sessions"), 0755)
|
2024-04-02 23:48:31 +00:00
|
|
|
require.NoError(t, err, "Could not create sessions directory")
|
|
|
|
|
2023-09-16 22:59:47 +00:00
|
|
|
err = os.MkdirAll(filepath.Join(homePath, "tmp", "repos"), 0755)
|
|
|
|
require.NoError(t, err, "Could not create tmp repos directory")
|
|
|
|
|
2024-09-20 14:01:09 +00:00
|
|
|
err = db.Setup(databaseDsn, true)
|
2023-09-16 22:59:47 +00:00
|
|
|
require.NoError(t, err, "Could not initialize database")
|
|
|
|
|
2024-09-20 14:01:09 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Fatal().Err(err).Msg("Could not initialize database")
|
|
|
|
}
|
|
|
|
|
2023-09-16 22:59:47 +00:00
|
|
|
err = memdb.Setup()
|
|
|
|
require.NoError(t, err, "Could not initialize in memory database")
|
2024-01-04 02:38:15 +00:00
|
|
|
|
|
|
|
// err = index.Open(filepath.Join(homePath, "testsindex", "opengist.index"))
|
|
|
|
// require.NoError(t, err, "Could not open index")
|
2023-09-16 22:59:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func teardown(t *testing.T, s *testServer) {
|
|
|
|
s.stop()
|
|
|
|
|
2024-09-20 14:01:09 +00:00
|
|
|
//err := db.Close()
|
|
|
|
//require.NoError(t, err, "Could not close database")
|
2023-09-16 22:59:47 +00:00
|
|
|
|
2024-09-20 14:01:09 +00:00
|
|
|
err := os.RemoveAll(path.Join(config.GetHomeDir(), "tests"))
|
2024-01-23 19:24:01 +00:00
|
|
|
require.NoError(t, err, "Could not remove repos directory")
|
|
|
|
|
|
|
|
err = os.RemoveAll(path.Join(config.GetHomeDir(), "tmp", "repos"))
|
2023-09-16 22:59:47 +00:00
|
|
|
require.NoError(t, err, "Could not remove repos directory")
|
2024-01-04 02:38:15 +00:00
|
|
|
|
2024-05-27 23:30:08 +00:00
|
|
|
err = os.RemoveAll(path.Join(config.GetHomeDir(), "tmp", "sessions"))
|
|
|
|
require.NoError(t, err, "Could not remove repos directory")
|
|
|
|
|
2024-09-20 14:01:09 +00:00
|
|
|
err = db.TruncateDatabase()
|
|
|
|
require.NoError(t, err, "Could not truncate database")
|
|
|
|
|
2024-01-04 02:38:15 +00:00
|
|
|
// err = os.RemoveAll(path.Join(config.C.OpengistHome, "testsindex"))
|
|
|
|
// require.NoError(t, err, "Could not remove repos directory")
|
|
|
|
|
|
|
|
// err = index.Close()
|
|
|
|
// require.NoError(t, err, "Could not close index")
|
2023-09-16 22:59:47 +00:00
|
|
|
}
|