| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715 | package api_testimport (	"bytes"	"encoding/json"	"fmt"	"net"	"net/http"	"net/http/httptest"	"os"	"path/filepath"	"runtime"	"strconv"	"testing"	"time"	"github.com/go-chi/render"	_ "github.com/go-sql-driver/mysql"	_ "github.com/lib/pq"	_ "github.com/mattn/go-sqlite3"	"github.com/rs/zerolog"	"github.com/drakkan/sftpgo/api"	"github.com/drakkan/sftpgo/config"	"github.com/drakkan/sftpgo/dataprovider"	"github.com/drakkan/sftpgo/logger"	"github.com/drakkan/sftpgo/sftpd")const (	defaultUsername       = "test_user"	defaultPassword       = "test_password"	testPubKey            = "ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABgQC03jj0D+djk7pxIf/0OhrxrchJTRZklofJ1NoIu4752Sq02mdXmarMVsqJ1cAjV5LBVy3D1F5U6XW4rppkXeVtd04Pxb09ehtH0pRRPaoHHlALiJt8CoMpbKYMA8b3KXPPriGxgGomvtU2T2RMURSwOZbMtpsugfjYSWenyYX+VORYhylWnSXL961LTyC21ehd6d6QnW9G7E5hYMITMY9TuQZz3bROYzXiTsgN0+g6Hn7exFQp50p45StUMfV/SftCMdCxlxuyGny2CrN/vfjO7xxOo2uv7q1qm10Q46KPWJQv+pgZ/OfL+EDjy07n5QVSKHlbx+2nT4Q0EgOSQaCTYwn3YjtABfIxWwgAFdyj6YlPulCL22qU4MYhDcA6PSBwDdf8hvxBfvsiHdM+JcSHvv8/VeJhk6CmnZxGY0fxBupov27z3yEO8nAg8k+6PaUiW1MSUfuGMF/ktB8LOstXsEPXSszuyXiOv4DaryOXUiSn7bmRqKcEFlJusO6aZP0= nicola@p1"	logSender             = "APITesting"	userPath              = "/api/v1/user"	activeConnectionsPath = "/api/v1/sftp_connection"	quotaScanPath         = "/api/v1/quota_scan")var (	defaultPerms = []string{dataprovider.PermAny}	homeBasePath string	testServer   *httptest.Server)func TestMain(m *testing.M) {	if runtime.GOOS == "windows" {		homeBasePath = "C:\\"	} else {		homeBasePath = "/tmp"	}	configDir := ".."	logfilePath := filepath.Join(configDir, "sftpgo_api_test.log")	confName := "sftpgo.conf"	logger.InitLogger(logfilePath, 5, 1, 28, false, zerolog.DebugLevel)	configFilePath := filepath.Join(configDir, confName)	config.LoadConfig(configFilePath)	providerConf := config.GetProviderConf()	err := dataprovider.Initialize(providerConf, configDir)	if err != nil {		logger.Warn(logSender, "error initializing data provider: %v", err)		os.Exit(1)	}	dataProvider := dataprovider.GetProvider()	httpdConf := config.GetHTTPDConfig()	router := api.GetHTTPRouter()	httpdConf.BindPort = 8081	api.SetBaseURL("http://127.0.0.1:8081")	sftpd.SetDataProvider(dataProvider)	api.SetDataProvider(dataProvider)	go func() {		logger.Debug(logSender, "initializing HTTP server with config %+v", httpdConf)		s := &http.Server{			Addr:           fmt.Sprintf("%s:%d", httpdConf.BindAddress, httpdConf.BindPort),			Handler:        router,			ReadTimeout:    300 * time.Second,			WriteTimeout:   300 * time.Second,			MaxHeaderBytes: 1 << 20, // 1MB		}		if err := s.ListenAndServe(); err != nil {			logger.Error(logSender, "could not start HTTP server: %v", err)		}	}()	testServer = httptest.NewServer(api.GetHTTPRouter())	defer testServer.Close()	waitTCPListening(fmt.Sprintf("%s:%d", httpdConf.BindAddress, httpdConf.BindPort))	exitCode := m.Run()	os.Remove(logfilePath)	os.Exit(exitCode)}func TestBasicUserHandling(t *testing.T) {	user, err := api.AddUser(getTestUser(), http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	user.MaxSessions = 10	user.QuotaSize = 4096	user.QuotaFiles = 2	user.UploadBandwidth = 128	user.DownloadBandwidth = 64	user, err = api.UpdateUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to update user: %v", err)	}	users, err := api.GetUsers(0, 0, defaultUsername, http.StatusOK)	if err != nil {		t.Errorf("unable to get users: %v", err)	}	if len(users) != 1 {		t.Errorf("number of users mismatch, expected: 1, actual: %v", len(users))	}	err = api.RemoveUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to remove: %v", err)	}}func TestAddUserNoCredentials(t *testing.T) {	u := getTestUser()	u.Password = ""	u.PublicKey = []string{}	_, err := api.AddUser(u, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error adding user with no credentials: %v", err)	}}func TestAddUserNoUsername(t *testing.T) {	u := getTestUser()	u.Username = ""	_, err := api.AddUser(u, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error adding user with no home dir: %v", err)	}}func TestAddUserNoHomeDir(t *testing.T) {	u := getTestUser()	u.HomeDir = ""	_, err := api.AddUser(u, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error adding user with no home dir: %v", err)	}}func TestAddUserInvalidHomeDir(t *testing.T) {	u := getTestUser()	u.HomeDir = "relative_path"	_, err := api.AddUser(u, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error adding user with invalid home dir: %v", err)	}}func TestAddUserNoPerms(t *testing.T) {	u := getTestUser()	u.Permissions = []string{}	_, err := api.AddUser(u, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error adding user with no perms: %v", err)	}}func TestAddUserInvalidPerms(t *testing.T) {	u := getTestUser()	u.Permissions = []string{"invalidPerm"}	_, err := api.AddUser(u, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error adding user with no perms: %v", err)	}}func TestUserPublicKey(t *testing.T) {	u := getTestUser()	invalidPubKey := "invalid"	validPubKey := "ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABgQC03jj0D+djk7pxIf/0OhrxrchJTRZklofJ1NoIu4752Sq02mdXmarMVsqJ1cAjV5LBVy3D1F5U6XW4rppkXeVtd04Pxb09ehtH0pRRPaoHHlALiJt8CoMpbKYMA8b3KXPPriGxgGomvtU2T2RMURSwOZbMtpsugfjYSWenyYX+VORYhylWnSXL961LTyC21ehd6d6QnW9G7E5hYMITMY9TuQZz3bROYzXiTsgN0+g6Hn7exFQp50p45StUMfV/SftCMdCxlxuyGny2CrN/vfjO7xxOo2uv7q1qm10Q46KPWJQv+pgZ/OfL+EDjy07n5QVSKHlbx+2nT4Q0EgOSQaCTYwn3YjtABfIxWwgAFdyj6YlPulCL22qU4MYhDcA6PSBwDdf8hvxBfvsiHdM+JcSHvv8/VeJhk6CmnZxGY0fxBupov27z3yEO8nAg8k+6PaUiW1MSUfuGMF/ktB8LOstXsEPXSszuyXiOv4DaryOXUiSn7bmRqKcEFlJusO6aZP0= nicola@p1"	u.PublicKey = []string{invalidPubKey}	_, err := api.AddUser(u, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error adding user with invalid pub key: %v", err)	}	u.PublicKey = []string{validPubKey}	user, err := api.AddUser(u, http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	user.PublicKey = []string{validPubKey, invalidPubKey}	_, err = api.UpdateUser(user, http.StatusBadRequest)	if err != nil {		t.Errorf("update user with invalid public key must fail: %v", err)	}	user.PublicKey = []string{validPubKey, validPubKey, validPubKey}	_, err = api.UpdateUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to update user: %v", err)	}	err = api.RemoveUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to remove: %v", err)	}}func TestUpdateUser(t *testing.T) {	user, err := api.AddUser(getTestUser(), http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	user.HomeDir = filepath.Join(homeBasePath, "testmod")	user.UID = 33	user.GID = 101	user.MaxSessions = 10	user.QuotaSize = 4096	user.QuotaFiles = 2	user.Permissions = []string{dataprovider.PermCreateDirs, dataprovider.PermDelete, dataprovider.PermDownload}	user.UploadBandwidth = 1024	user.DownloadBandwidth = 512	user, err = api.UpdateUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to update user: %v", err)	}	err = api.RemoveUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to remove: %v", err)	}}func TestUpdateUserNoCredentials(t *testing.T) {	user, err := api.AddUser(getTestUser(), http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	user.Password = ""	user.PublicKey = []string{}	// password and public key will be omitted from json serialization if empty and so they will remain unchanged	// and no validation error will be raised	_, err = api.UpdateUser(user, http.StatusOK)	if err != nil {		t.Errorf("unexpected error updating user with no credentials: %v", err)	}	err = api.RemoveUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to remove: %v", err)	}}func TestUpdateUserEmptyHomeDir(t *testing.T) {	user, err := api.AddUser(getTestUser(), http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	user.HomeDir = ""	_, err = api.UpdateUser(user, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error updating user with empty home dir: %v", err)	}	err = api.RemoveUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to remove: %v", err)	}}func TestUpdateUserInvalidHomeDir(t *testing.T) {	user, err := api.AddUser(getTestUser(), http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	user.HomeDir = "relative_path"	_, err = api.UpdateUser(user, http.StatusBadRequest)	if err != nil {		t.Errorf("unexpected error updating user with empty home dir: %v", err)	}	err = api.RemoveUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to remove: %v", err)	}}func TestUpdateNonExistentUser(t *testing.T) {	_, err := api.UpdateUser(getTestUser(), http.StatusNotFound)	if err != nil {		t.Errorf("unable to update user: %v", err)	}}func TestGetNonExistentUser(t *testing.T) {	_, err := api.GetUserByID(0, http.StatusNotFound)	if err != nil {		t.Errorf("unable to get user: %v", err)	}}func TestDeleteNonExistentUser(t *testing.T) {	err := api.RemoveUser(getTestUser(), http.StatusNotFound)	if err != nil {		t.Errorf("unable to remove user: %v", err)	}}func TestAddDuplicateUser(t *testing.T) {	user, err := api.AddUser(getTestUser(), http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	_, err = api.AddUser(getTestUser(), http.StatusInternalServerError)	if err != nil {		t.Errorf("unable to add second user: %v", err)	}	err = api.RemoveUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to remove user: %v", err)	}}func TestGetUsers(t *testing.T) {	user1, err := api.AddUser(getTestUser(), http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	u := getTestUser()	u.Username = defaultUsername + "1"	user2, err := api.AddUser(u, http.StatusOK)	if err != nil {		t.Errorf("unable to add second user: %v", err)	}	users, err := api.GetUsers(0, 0, "", http.StatusOK)	if err != nil {		t.Errorf("unable to get users: %v", err)	}	if len(users) < 2 {		t.Errorf("at least 2 users are expected")	}	users, err = api.GetUsers(1, 0, "", http.StatusOK)	if err != nil {		t.Errorf("unable to get users: %v", err)	}	if len(users) != 1 {		t.Errorf("1 user is expected")	}	users, err = api.GetUsers(1, 1, "", http.StatusOK)	if err != nil {		t.Errorf("unable to get users: %v", err)	}	if len(users) != 1 {		t.Errorf("1 user is expected")	}	err = api.RemoveUser(user1, http.StatusOK)	if err != nil {		t.Errorf("unable to remove user: %v", err)	}	err = api.RemoveUser(user2, http.StatusOK)	if err != nil {		t.Errorf("unable to remove user: %v", err)	}}func TestGetQuotaScans(t *testing.T) {	_, err := api.GetQuotaScans(http.StatusOK)	if err != nil {		t.Errorf("unable to get quota scans: %v", err)	}}func TestStartQuotaScan(t *testing.T) {	user, err := api.AddUser(getTestUser(), http.StatusOK)	if err != nil {		t.Errorf("unable to add user: %v", err)	}	err = api.StartQuotaScan(user, http.StatusCreated)	if err != nil {		t.Errorf("unable to start quota scan: %v", err)	}	err = api.RemoveUser(user, http.StatusOK)	if err != nil {		t.Errorf("unable to remove user: %v", err)	}}func TestGetSFTPConnections(t *testing.T) {	_, err := api.GetSFTPConnections(http.StatusOK)	if err != nil {		t.Errorf("unable to get sftp connections: %v", err)	}}func TestCloseActiveSFTPConnection(t *testing.T) {	err := api.CloseSFTPConnection("non_existent_id", http.StatusNotFound)	if err != nil {		t.Errorf("unexpected error closing non existent sftp connection: %v", err)	}}// test using mock http serverfunc TestBasicUserHandlingMock(t *testing.T) {	user := getTestUser()	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	err := render.DecodeJSON(rr.Body, &user)	if err != nil {		t.Errorf("Error get user: %v", err)	}	req, _ = http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr = executeRequest(req)	checkResponseCode(t, http.StatusInternalServerError, rr.Code)	user.MaxSessions = 10	user.UploadBandwidth = 128	userAsJSON = getUserAsJSON(t, user)	req, _ = http.NewRequest(http.MethodPut, userPath+"/"+strconv.FormatInt(user.ID, 10), bytes.NewBuffer(userAsJSON))	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	req, _ = http.NewRequest(http.MethodGet, userPath+"/"+strconv.FormatInt(user.ID, 10), nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	var updatedUser dataprovider.User	err = render.DecodeJSON(rr.Body, &updatedUser)	if err != nil {		t.Errorf("Error decoding updated user: %v", err)	}	if user.MaxSessions != updatedUser.MaxSessions || user.UploadBandwidth != updatedUser.UploadBandwidth {		t.Errorf("Error modifying user actual: %v, %v", updatedUser.MaxSessions, updatedUser.UploadBandwidth)	}	req, _ = http.NewRequest(http.MethodDelete, userPath+"/"+strconv.FormatInt(user.ID, 10), nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)}func TestGetUserByIdInvalidParamsMock(t *testing.T) {	req, _ := http.NewRequest(http.MethodGet, userPath+"/0", nil)	rr := executeRequest(req)	checkResponseCode(t, http.StatusNotFound, rr.Code)	req, _ = http.NewRequest(http.MethodGet, userPath+"/a", nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)}func TestAddUserNoUsernameMock(t *testing.T) {	user := getTestUser()	user.Username = ""	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)}func TestAddUserInvalidHomeDirMock(t *testing.T) {	user := getTestUser()	user.HomeDir = "relative_path"	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)}func TestAddUserInvalidPermsMock(t *testing.T) {	user := getTestUser()	user.Permissions = []string{}	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)}func TestAddUserInvalidJsonMock(t *testing.T) {	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer([]byte("invalid json")))	rr := executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)}func TestUpdateUserInvalidJsonMock(t *testing.T) {	user := getTestUser()	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	err := render.DecodeJSON(rr.Body, &user)	if err != nil {		t.Errorf("Error get user: %v", err)	}	req, _ = http.NewRequest(http.MethodPut, userPath+"/"+strconv.FormatInt(user.ID, 10), bytes.NewBuffer([]byte("Invalid json")))	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)	req, _ = http.NewRequest(http.MethodDelete, userPath+"/"+strconv.FormatInt(user.ID, 10), nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)}func TestUpdateUserInvalidParamsMock(t *testing.T) {	user := getTestUser()	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	err := render.DecodeJSON(rr.Body, &user)	if err != nil {		t.Errorf("Error get user: %v", err)	}	user.HomeDir = ""	userAsJSON = getUserAsJSON(t, user)	req, _ = http.NewRequest(http.MethodPut, userPath+"/"+strconv.FormatInt(user.ID, 10), bytes.NewBuffer(userAsJSON))	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)	userID := user.ID	user.ID = 0	userAsJSON = getUserAsJSON(t, user)	req, _ = http.NewRequest(http.MethodPut, userPath+"/"+strconv.FormatInt(userID, 10), bytes.NewBuffer(userAsJSON))	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)	user.ID = userID	req, _ = http.NewRequest(http.MethodPut, userPath+"/0", bytes.NewBuffer(userAsJSON))	rr = executeRequest(req)	checkResponseCode(t, http.StatusNotFound, rr.Code)	req, _ = http.NewRequest(http.MethodPut, userPath+"/a", bytes.NewBuffer(userAsJSON))	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)	req, _ = http.NewRequest(http.MethodDelete, userPath+"/"+strconv.FormatInt(user.ID, 10), nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)}func TestGetUsersMock(t *testing.T) {	user := getTestUser()	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	err := render.DecodeJSON(rr.Body, &user)	if err != nil {		t.Errorf("Error get user: %v", err)	}	req, _ = http.NewRequest(http.MethodGet, userPath+"?limit=510&offset=0&order=ASC&username="+defaultUsername, nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	var users []dataprovider.User	err = render.DecodeJSON(rr.Body, &users)	if err != nil {		t.Errorf("Error decoding users: %v", err)	}	if len(users) != 1 {		t.Errorf("1 user is expected")	}	req, _ = http.NewRequest(http.MethodGet, userPath+"?limit=a&offset=0&order=ASC", nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)	req, _ = http.NewRequest(http.MethodGet, userPath+"?limit=1&offset=a&order=ASC", nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)	req, _ = http.NewRequest(http.MethodGet, userPath+"?limit=1&offset=0&order=ASCa", nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)	req, _ = http.NewRequest(http.MethodDelete, userPath+"/"+strconv.FormatInt(user.ID, 10), nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)}func TestDeleteUserInvalidParamsMock(t *testing.T) {	req, _ := http.NewRequest(http.MethodDelete, userPath+"/0", nil)	rr := executeRequest(req)	checkResponseCode(t, http.StatusNotFound, rr.Code)	req, _ = http.NewRequest(http.MethodDelete, userPath+"/a", nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)}func TestGetQuotaScansMock(t *testing.T) {	req, err := http.NewRequest("GET", quotaScanPath, nil)	if err != nil {		t.Errorf("error get quota scan: %v", err)	}	rr := executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)}func TestStartQuotaScanMock(t *testing.T) {	user := getTestUser()	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, userPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	err := render.DecodeJSON(rr.Body, &user)	if err != nil {		t.Errorf("Error get user: %v", err)	}	_, err = os.Stat(user.HomeDir)	if err == nil {		os.Remove(user.HomeDir)	}	userAsJSON = getUserAsJSON(t, user)	req, _ = http.NewRequest(http.MethodPost, quotaScanPath, bytes.NewBuffer(userAsJSON))	rr = executeRequest(req)	checkResponseCode(t, http.StatusCreated, rr.Code)	req, _ = http.NewRequest(http.MethodGet, quotaScanPath, nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)	var scans []sftpd.ActiveQuotaScan	err = render.DecodeJSON(rr.Body, &scans)	if err != nil {		t.Errorf("Error get active scans: %v", err)	}	for len(scans) > 0 {		req, _ = http.NewRequest(http.MethodGet, quotaScanPath, nil)		rr = executeRequest(req)		checkResponseCode(t, http.StatusOK, rr.Code)		err = render.DecodeJSON(rr.Body, &scans)		if err != nil {			t.Errorf("Error get active scans: %v", err)			break		}	}	_, err = os.Stat(user.HomeDir)	if err != nil && os.IsNotExist(err) {		os.MkdirAll(user.HomeDir, 0777)	}	req, _ = http.NewRequest(http.MethodPost, quotaScanPath, bytes.NewBuffer(userAsJSON))	rr = executeRequest(req)	checkResponseCode(t, http.StatusCreated, rr.Code)	req, _ = http.NewRequest(http.MethodDelete, userPath+"/"+strconv.FormatInt(user.ID, 10), nil)	rr = executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)}func TestStartQuotaScanBadUserMock(t *testing.T) {	user := getTestUser()	userAsJSON := getUserAsJSON(t, user)	req, _ := http.NewRequest(http.MethodPost, quotaScanPath, bytes.NewBuffer(userAsJSON))	rr := executeRequest(req)	checkResponseCode(t, http.StatusNotFound, rr.Code)}func TestStartQuotaScanNonExistentUserMock(t *testing.T) {	req, _ := http.NewRequest(http.MethodPost, quotaScanPath, bytes.NewBuffer([]byte("invalid json")))	rr := executeRequest(req)	checkResponseCode(t, http.StatusBadRequest, rr.Code)}func TestGetSFTPConnectionsMock(t *testing.T) {	req, _ := http.NewRequest(http.MethodGet, activeConnectionsPath, nil)	rr := executeRequest(req)	checkResponseCode(t, http.StatusOK, rr.Code)}func TestDeleteActiveConnectionMock(t *testing.T) {	req, _ := http.NewRequest(http.MethodDelete, activeConnectionsPath+"/connectionID", nil)	rr := executeRequest(req)	checkResponseCode(t, http.StatusNotFound, rr.Code)}func TestNotFoundMock(t *testing.T) {	req, _ := http.NewRequest(http.MethodGet, "/non/existing/path", nil)	rr := executeRequest(req)	checkResponseCode(t, http.StatusNotFound, rr.Code)}func TestMethodNotAllowedMock(t *testing.T) {	req, _ := http.NewRequest(http.MethodPost, activeConnectionsPath, nil)	rr := executeRequest(req)	checkResponseCode(t, http.StatusMethodNotAllowed, rr.Code)}func waitTCPListening(address string) {	for {		conn, err := net.Dial("tcp", address)		if err != nil {			logger.WarnToConsole("tcp server %v not listening: %v\n", address, err)			time.Sleep(100 * time.Millisecond)			continue		}		logger.InfoToConsole("tcp server %v now listening\n", address)		defer conn.Close()		break	}}func getTestUser() dataprovider.User {	return dataprovider.User{		Username:    defaultUsername,		Password:    defaultPassword,		HomeDir:     filepath.Join(homeBasePath, defaultUsername),		Permissions: defaultPerms,	}}func getUserAsJSON(t *testing.T, user dataprovider.User) []byte {	json, err := json.Marshal(user)	if err != nil {		t.Errorf("error get user as json: %v", err)		return []byte("{}")	}	return json}func executeRequest(req *http.Request) *httptest.ResponseRecorder {	rr := httptest.NewRecorder()	testServer.Config.Handler.ServeHTTP(rr, req)	return rr}func checkResponseCode(t *testing.T, expected, actual int) {	if expected != actual {		t.Errorf("Expected response code %d. Got %d", expected, actual)	}}
 |