2023-04-17 21:34:59 +08:00
|
|
|
package testserver
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
2023-09-17 22:55:13 +08:00
|
|
|
// sqlite driver.
|
|
|
|
_ "modernc.org/sqlite"
|
|
|
|
|
2023-07-31 20:55:40 +08:00
|
|
|
"github.com/usememos/memos/api/auth"
|
2023-04-17 21:34:59 +08:00
|
|
|
"github.com/usememos/memos/server"
|
|
|
|
"github.com/usememos/memos/server/profile"
|
2023-06-17 21:25:46 +08:00
|
|
|
"github.com/usememos/memos/store"
|
2023-10-05 23:11:29 +08:00
|
|
|
"github.com/usememos/memos/store/db"
|
2023-04-17 21:34:59 +08:00
|
|
|
"github.com/usememos/memos/test"
|
|
|
|
)
|
|
|
|
|
|
|
|
type TestingServer struct {
|
|
|
|
server *server.Server
|
|
|
|
client *http.Client
|
|
|
|
profile *profile.Profile
|
|
|
|
cookie string
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewTestingServer(ctx context.Context, t *testing.T) (*TestingServer, error) {
|
|
|
|
profile := test.GetTestingProfile(t)
|
2023-10-05 23:11:29 +08:00
|
|
|
dbDriver, err := db.NewDBDriver(profile)
|
2023-09-27 11:56:20 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to create db driver")
|
2023-04-17 21:34:59 +08:00
|
|
|
}
|
2023-10-05 23:11:29 +08:00
|
|
|
if err := dbDriver.Migrate(ctx); err != nil {
|
2023-08-26 07:33:45 +08:00
|
|
|
return nil, errors.Wrap(err, "failed to migrate db")
|
|
|
|
}
|
2023-04-17 21:34:59 +08:00
|
|
|
|
2023-10-05 23:11:29 +08:00
|
|
|
store := store.New(dbDriver, profile)
|
2023-06-17 21:25:46 +08:00
|
|
|
server, err := server.NewServer(ctx, profile, store)
|
2023-04-17 21:34:59 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to create server")
|
|
|
|
}
|
|
|
|
|
|
|
|
s := &TestingServer{
|
|
|
|
server: server,
|
|
|
|
client: &http.Client{},
|
|
|
|
profile: profile,
|
|
|
|
cookie: "",
|
|
|
|
}
|
2023-06-17 21:25:46 +08:00
|
|
|
errChan := make(chan error, 1)
|
2023-04-17 21:34:59 +08:00
|
|
|
|
|
|
|
go func() {
|
|
|
|
if err := s.server.Start(ctx); err != nil {
|
|
|
|
if err != http.ErrServerClosed {
|
|
|
|
errChan <- errors.Wrap(err, "failed to run main server")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
if err := s.waitForServerStart(errChan); err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to start server")
|
|
|
|
}
|
|
|
|
|
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *TestingServer) Shutdown(ctx context.Context) {
|
|
|
|
s.server.Shutdown(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *TestingServer) waitForServerStart(errChan <-chan error) error {
|
|
|
|
ticker := time.NewTicker(100 * time.Millisecond)
|
|
|
|
defer ticker.Stop()
|
|
|
|
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ticker.C:
|
|
|
|
if s == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
e := s.server.GetEcho()
|
|
|
|
if e == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
addr := e.ListenerAddr()
|
|
|
|
if addr != nil && strings.Contains(addr.String(), ":") {
|
|
|
|
return nil // was started
|
|
|
|
}
|
|
|
|
case err := <-errChan:
|
|
|
|
if err == http.ErrServerClosed {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *TestingServer) request(method, uri string, body io.Reader, params, header map[string]string) (io.ReadCloser, error) {
|
|
|
|
fullURL := fmt.Sprintf("http://localhost:%d%s", s.profile.Port, uri)
|
|
|
|
req, err := http.NewRequest(method, fullURL, body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "fail to create a new %s request(%q)", method, fullURL)
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range header {
|
|
|
|
req.Header.Set(k, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
q := url.Values{}
|
|
|
|
for k, v := range params {
|
|
|
|
q.Add(k, v)
|
|
|
|
}
|
|
|
|
if len(q) > 0 {
|
|
|
|
req.URL.RawQuery = q.Encode()
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := s.client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrapf(err, "fail to send a %s request(%q)", method, fullURL)
|
|
|
|
}
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
body, err := io.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to read http response body")
|
|
|
|
}
|
|
|
|
return nil, errors.Errorf("http response error code %v body %q", resp.StatusCode, string(body))
|
|
|
|
}
|
|
|
|
|
|
|
|
if method == "POST" {
|
2023-06-17 21:25:46 +08:00
|
|
|
if strings.Contains(uri, "/api/v1/auth/login") || strings.Contains(uri, "/api/v1/auth/signup") {
|
2023-04-17 21:34:59 +08:00
|
|
|
cookie := ""
|
|
|
|
h := resp.Header.Get("Set-Cookie")
|
|
|
|
parts := strings.Split(h, "; ")
|
|
|
|
for _, p := range parts {
|
2023-07-09 21:13:26 +08:00
|
|
|
if strings.HasPrefix(p, fmt.Sprintf("%s=", auth.AccessTokenCookieName)) {
|
2023-04-17 21:34:59 +08:00
|
|
|
cookie = p
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if cookie == "" {
|
2023-09-29 13:04:54 +08:00
|
|
|
return nil, errors.New("unable to find access token in the login response headers")
|
2023-04-17 21:34:59 +08:00
|
|
|
}
|
|
|
|
s.cookie = cookie
|
2023-08-10 09:01:38 +08:00
|
|
|
} else if strings.Contains(uri, "/api/v1/auth/signout") {
|
2023-04-17 21:34:59 +08:00
|
|
|
s.cookie = ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return resp.Body, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// get sends a GET client request.
|
|
|
|
func (s *TestingServer) get(url string, params map[string]string) (io.ReadCloser, error) {
|
|
|
|
return s.request("GET", url, nil, params, map[string]string{
|
|
|
|
"Cookie": s.cookie,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// post sends a POST client request.
|
|
|
|
func (s *TestingServer) post(url string, body io.Reader, params map[string]string) (io.ReadCloser, error) {
|
|
|
|
return s.request("POST", url, body, params, map[string]string{
|
|
|
|
"Cookie": s.cookie,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// patch sends a PATCH client request.
|
|
|
|
func (s *TestingServer) patch(url string, body io.Reader, params map[string]string) (io.ReadCloser, error) {
|
|
|
|
return s.request("PATCH", url, body, params, map[string]string{
|
|
|
|
"Cookie": s.cookie,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// delete sends a DELETE client request.
|
|
|
|
func (s *TestingServer) delete(url string, params map[string]string) (io.ReadCloser, error) {
|
|
|
|
return s.request("DELETE", url, nil, params, map[string]string{
|
|
|
|
"Cookie": s.cookie,
|
|
|
|
})
|
|
|
|
}
|