Browse Source

Add a mock HTTP client for Matrix clients to use. Add example test.

This will need rejigging at some point to make the test easier to set up.
kegan/tests-prep
Kegan Dougal 8 years ago
parent
commit
5c65d4cf95
  1. 7
      src/github.com/matrix-org/go-neb/clients/clients.go
  2. 6
      src/github.com/matrix-org/go-neb/goneb.go
  3. 84
      src/github.com/matrix-org/go-neb/goneb_services_test.go
  4. 10
      src/github.com/matrix-org/go-neb/matrix/matrix.go

7
src/github.com/matrix-org/go-neb/clients/clients.go

@ -7,6 +7,7 @@ import (
"github.com/matrix-org/go-neb/matrix" "github.com/matrix-org/go-neb/matrix"
"github.com/matrix-org/go-neb/plugin" "github.com/matrix-org/go-neb/plugin"
"github.com/matrix-org/go-neb/types" "github.com/matrix-org/go-neb/types"
"net/http"
"net/url" "net/url"
"strings" "strings"
"sync" "sync"
@ -40,15 +41,17 @@ func (s nextBatchStore) Load(userID string) string {
// A Clients is a collection of clients used for bot services. // A Clients is a collection of clients used for bot services.
type Clients struct { type Clients struct {
db *database.ServiceDB db *database.ServiceDB
httpClient *http.Client
dbMutex sync.Mutex dbMutex sync.Mutex
mapMutex sync.Mutex mapMutex sync.Mutex
clients map[string]clientEntry clients map[string]clientEntry
} }
// New makes a new collection of matrix clients // New makes a new collection of matrix clients
func New(db *database.ServiceDB) *Clients {
func New(db *database.ServiceDB, cli *http.Client) *Clients {
clients := &Clients{ clients := &Clients{
db: db, db: db,
httpClient: cli,
clients: make(map[string]clientEntry), // user_id => clientEntry clients: make(map[string]clientEntry), // user_id => clientEntry
} }
return clients return clients
@ -238,7 +241,7 @@ func (c *Clients) newClient(config api.ClientConfig) (*matrix.Client, error) {
return nil, err return nil, err
} }
client := matrix.NewClient(homeserverURL, config.AccessToken, config.UserID)
client := matrix.NewClient(c.httpClient, homeserverURL, config.AccessToken, config.UserID)
client.NextBatchStorer = nextBatchStore{c.db} client.NextBatchStorer = nextBatchStore{c.db}
// TODO: Check that the access token is valid for the userID by peforming // TODO: Check that the access token is valid for the userID by peforming

6
src/github.com/matrix-org/go-neb/goneb.go

@ -139,7 +139,7 @@ func loadDatabase(databaseType, databaseURL, configYAML string) (*database.Servi
return db, err return db, err
} }
func setup(e envVars, mux *http.ServeMux) {
func setup(e envVars, mux *http.ServeMux, matrixClient *http.Client) {
err := types.BaseURL(e.BaseURL) err := types.BaseURL(e.BaseURL)
if err != nil { if err != nil {
log.WithError(err).Panic("Failed to get base url") log.WithError(err).Panic("Failed to get base url")
@ -164,7 +164,7 @@ func setup(e envVars, mux *http.ServeMux) {
log.Info("Inserted ", len(cfg.Sessions), " sessions") log.Info("Inserted ", len(cfg.Sessions), " sessions")
} }
clients := clients.New(db)
clients := clients.New(db, matrixClient)
if err := clients.Start(); err != nil { if err := clients.Start(); err != nil {
log.WithError(err).Panic("Failed to start up clients") log.WithError(err).Panic("Failed to start up clients")
} }
@ -229,6 +229,6 @@ func main() {
log.Infof("Go-NEB (%+v)", e) log.Infof("Go-NEB (%+v)", e)
setup(e, http.DefaultServeMux)
setup(e, http.DefaultServeMux, http.DefaultClient)
log.Fatal(http.ListenAndServe(e.BindAddress, nil)) log.Fatal(http.ListenAndServe(e.BindAddress, nil))
} }

84
src/github.com/matrix-org/go-neb/goneb_services_test.go

@ -1,31 +1,103 @@
package main package main
import ( import (
"bytes"
"net/http" "net/http"
"net/http/httptest" "net/http/httptest"
"os" "os"
"strconv"
"testing" "testing"
) )
var mux = http.NewServeMux() var mux = http.NewServeMux()
type MockTripper struct {
handlers map[string]func(req *http.Request) (*http.Response, error)
}
func (rt MockTripper) RoundTrip(req *http.Request) (*http.Response, error) {
key := req.Method + " " + req.URL.Path
h := rt.handlers[key]
if h == nil {
panic(
"Test RoundTrip: Unhandled request: " + key + "\n" +
"Handlers: " + strconv.Itoa(len(rt.handlers)),
)
}
return h(req)
}
func (rt MockTripper) Handle(method, path string, handler func(req *http.Request) (*http.Response, error)) {
key := method + " " + path
if _, exists := rt.handlers[key]; exists {
panic("Test handler with key " + key + " already exists")
}
rt.handlers[key] = handler
}
var tripper = MockTripper{make(map[string]func(req *http.Request) (*http.Response, error))}
type nopCloser struct {
*bytes.Buffer
}
func (nopCloser) Close() error { return nil }
func newResponse(statusCode int, body string) *http.Response {
return &http.Response{
StatusCode: statusCode,
Body: nopCloser{bytes.NewBufferString(body)},
}
}
func TestMain(m *testing.M) { func TestMain(m *testing.M) {
setup(envVars{ setup(envVars{
BaseURL: "http://go.neb", BaseURL: "http://go.neb",
DatabaseType: "sqlite3", DatabaseType: "sqlite3",
DatabaseURL: ":memory:", DatabaseURL: ":memory:",
}, mux)
}, mux, &http.Client{
Transport: tripper,
})
exitCode := m.Run() exitCode := m.Run()
os.Exit(exitCode) os.Exit(exitCode)
} }
func TestNotFound(t *testing.T) {
func TestConfigureClient(t *testing.T) {
for k := range tripper.handlers {
delete(tripper.handlers, k)
}
mockWriter := httptest.NewRecorder() mockWriter := httptest.NewRecorder()
mockReq, _ := http.NewRequest("GET", "http://go.neb/foo", nil)
mux.ServeHTTP(mockWriter, mockReq)
tripper.Handle("POST", "/_matrix/client/r0/user/@link:hyrule/filter",
func(req *http.Request) (*http.Response, error) {
return newResponse(200, `{
"filter_id":"abcdef"
}`), nil
},
)
syncChan := make(chan string)
tripper.Handle("GET", "/_matrix/client/r0/sync",
func(req *http.Request) (*http.Response, error) {
syncChan <- "sync"
return newResponse(200, `{
"next_batch":"11_22_33_44",
"rooms": {}
}`), nil
},
)
expectCode := 404
mockReq, _ := http.NewRequest("POST", "http://go.neb/admin/configureClient", bytes.NewBufferString(`
{
"UserID":"@link:hyrule",
"HomeserverURL":"http://hyrule.loz",
"AccessToken":"dangeroustogoalone",
"Sync":true,
"AutoJoinRooms":true
}`))
mux.ServeHTTP(mockWriter, mockReq)
expectCode := 200
if mockWriter.Code != expectCode { if mockWriter.Code != expectCode {
t.Errorf("TestNotFound wanted HTTP status %d, got %d", expectCode, mockWriter.Code)
t.Errorf("TestConfigureClient wanted HTTP status %d, got %d", expectCode, mockWriter.Code)
} }
<-syncChan
} }

10
src/github.com/matrix-org/go-neb/matrix/matrix.go

@ -404,7 +404,11 @@ func (cli *Client) doSync(timeout int, since string) ([]byte, error) {
query["filter"] = cli.filterID query["filter"] = cli.filterID
} }
urlPath := cli.buildURLWithQuery([]string{"sync"}, query) urlPath := cli.buildURLWithQuery([]string{"sync"}, query)
res, err := http.Get(urlPath)
req, err := http.NewRequest("GET", urlPath, nil)
if err != nil {
return nil, err
}
res, err := cli.httpClient.Do(req)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -417,7 +421,7 @@ func (cli *Client) doSync(timeout int, since string) ([]byte, error) {
} }
// NewClient creates a new Matrix Client ready for syncing // NewClient creates a new Matrix Client ready for syncing
func NewClient(homeserverURL *url.URL, accessToken string, userID string) *Client {
func NewClient(httpClient *http.Client, homeserverURL *url.URL, accessToken, userID string) *Client {
cli := Client{ cli := Client{
AccessToken: accessToken, AccessToken: accessToken,
HomeserverURL: homeserverURL, HomeserverURL: homeserverURL,
@ -430,7 +434,7 @@ func NewClient(homeserverURL *url.URL, accessToken string, userID string) *Clien
// remember the token across restarts. In practice, a database backend should be used. // remember the token across restarts. In practice, a database backend should be used.
cli.NextBatchStorer = noopNextBatchStore{} cli.NextBatchStorer = noopNextBatchStore{}
cli.Rooms = make(map[string]*Room) cli.Rooms = make(map[string]*Room)
cli.httpClient = &http.Client{}
cli.httpClient = httpClient
return &cli return &cli
} }
Loading…
Cancel
Save