2018-04-17 15:42:43 +03:00
|
|
|
// gomuks - A terminal Matrix client written in Go.
|
|
|
|
// Copyright (C) 2018 Tulir Asokan
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2018-05-02 23:45:17 +03:00
|
|
|
package matrix
|
|
|
|
|
|
|
|
import (
|
2018-05-22 17:24:47 +03:00
|
|
|
"encoding/json"
|
2018-05-10 20:03:19 +03:00
|
|
|
"fmt"
|
2018-05-22 17:24:47 +03:00
|
|
|
"github.com/stretchr/testify/assert"
|
2018-05-10 20:03:19 +03:00
|
|
|
"io/ioutil"
|
2018-11-14 00:00:35 +02:00
|
|
|
"maunium.net/go/mautrix"
|
2018-05-22 17:24:47 +03:00
|
|
|
"maunium.net/go/gomuks/config"
|
|
|
|
"net/http"
|
2018-05-10 20:31:11 +03:00
|
|
|
"os"
|
2018-05-22 17:24:47 +03:00
|
|
|
"strings"
|
|
|
|
"testing"
|
2018-05-02 23:45:17 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestContainer_InitClient_Empty(t *testing.T) {
|
2018-05-10 20:31:11 +03:00
|
|
|
defer os.RemoveAll("/tmp/gomuks-mxtest-0")
|
2018-05-02 23:45:17 +03:00
|
|
|
cfg := config.NewConfig("/tmp/gomuks-mxtest-0", "/tmp/gomuks-mxtest-0")
|
|
|
|
cfg.HS = "https://matrix.org"
|
|
|
|
c := Container{config: cfg}
|
|
|
|
assert.Nil(t, c.InitClient())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestContainer_GetCachePath(t *testing.T) {
|
2018-05-10 20:31:11 +03:00
|
|
|
defer os.RemoveAll("/tmp/gomuks-mxtest-1")
|
2018-05-02 23:45:17 +03:00
|
|
|
cfg := config.NewConfig("/tmp/gomuks-mxtest-1", "/tmp/gomuks-mxtest-1")
|
|
|
|
c := Container{config: cfg}
|
|
|
|
assert.Equal(t, "/tmp/gomuks-mxtest-1/media/maunium.net/foobar", c.GetCachePath("maunium.net", "foobar"))
|
|
|
|
}
|
2018-05-10 20:03:19 +03:00
|
|
|
|
|
|
|
func TestContainer_SendMarkdownMessage_NoMarkdown(t *testing.T) {
|
|
|
|
c := Container{client: mockClient(func(req *http.Request) (*http.Response, error) {
|
|
|
|
if req.Method != http.MethodPut || !strings.HasPrefix(req.URL.Path, "/_matrix/client/r0/rooms/!foo:example.com/send/m.room.message/") {
|
|
|
|
return nil, fmt.Errorf("unexpected query: %s %s", req.Method, req.URL.Path)
|
|
|
|
}
|
|
|
|
|
|
|
|
body := parseBody(req)
|
|
|
|
assert.Equal(t, "m.text", body["msgtype"])
|
|
|
|
assert.Equal(t, "test message", body["body"])
|
|
|
|
return mockResponse(http.StatusOK, `{"event_id": "!foobar1:example.com"}`), nil
|
|
|
|
})}
|
|
|
|
|
|
|
|
evtID, err := c.SendMarkdownMessage("!foo:example.com", "m.text", "test message")
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, "!foobar1:example.com", evtID)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestContainer_SendMarkdownMessage_WithMarkdown(t *testing.T) {
|
|
|
|
c := Container{client: mockClient(func(req *http.Request) (*http.Response, error) {
|
|
|
|
if req.Method != http.MethodPut || !strings.HasPrefix(req.URL.Path, "/_matrix/client/r0/rooms/!foo:example.com/send/m.room.message/") {
|
|
|
|
return nil, fmt.Errorf("unexpected query: %s %s", req.Method, req.URL.Path)
|
|
|
|
}
|
|
|
|
|
|
|
|
body := parseBody(req)
|
|
|
|
assert.Equal(t, "m.text", body["msgtype"])
|
|
|
|
assert.Equal(t, "**formatted** <u>test</u> _message_", body["body"])
|
|
|
|
assert.Equal(t, "<strong>formatted</strong> <u>test</u> <em>message</em>", body["formatted_body"])
|
|
|
|
return mockResponse(http.StatusOK, `{"event_id": "!foobar2:example.com"}`), nil
|
|
|
|
})}
|
|
|
|
|
|
|
|
evtID, err := c.SendMarkdownMessage("!foo:example.com", "m.text", "**formatted** <u>test</u> _message_")
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, "!foobar2:example.com", evtID)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestContainer_SendTyping(t *testing.T) {
|
2018-11-14 00:00:35 +02:00
|
|
|
var calls []mautrix.ReqTyping
|
2018-05-10 20:03:19 +03:00
|
|
|
c := Container{client: mockClient(func(req *http.Request) (*http.Response, error) {
|
2018-05-10 20:31:11 +03:00
|
|
|
if req.Method != http.MethodPut || req.URL.Path != "/_matrix/client/r0/rooms/!foo:example.com/typing/@user:example.com" {
|
2018-05-10 20:03:19 +03:00
|
|
|
return nil, fmt.Errorf("unexpected query: %s %s", req.Method, req.URL.Path)
|
|
|
|
}
|
|
|
|
|
|
|
|
rawBody, err := ioutil.ReadAll(req.Body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-11-14 00:00:35 +02:00
|
|
|
call := mautrix.ReqTyping{}
|
2018-05-10 20:03:19 +03:00
|
|
|
err = json.Unmarshal(rawBody, &call)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
calls = append(calls, call)
|
|
|
|
|
|
|
|
return mockResponse(http.StatusOK, `{}`), nil
|
|
|
|
})}
|
|
|
|
|
|
|
|
c.SendTyping("!foo:example.com", true)
|
|
|
|
c.SendTyping("!foo:example.com", true)
|
|
|
|
c.SendTyping("!foo:example.com", true)
|
|
|
|
c.SendTyping("!foo:example.com", false)
|
|
|
|
c.SendTyping("!foo:example.com", true)
|
|
|
|
c.SendTyping("!foo:example.com", false)
|
|
|
|
assert.Len(t, calls, 4)
|
|
|
|
assert.True(t, calls[0].Typing)
|
|
|
|
assert.False(t, calls[1].Typing)
|
|
|
|
assert.True(t, calls[2].Typing)
|
|
|
|
assert.False(t, calls[3].Typing)
|
|
|
|
}
|
|
|
|
|
2018-05-10 20:31:11 +03:00
|
|
|
func TestContainer_JoinRoom(t *testing.T) {
|
|
|
|
defer os.RemoveAll("/tmp/gomuks-mxtest-2")
|
|
|
|
cfg := config.NewConfig("/tmp/gomuks-mxtest-2", "/tmp/gomuks-mxtest-2")
|
|
|
|
c := Container{client: mockClient(func(req *http.Request) (*http.Response, error) {
|
|
|
|
if req.Method == http.MethodPost && req.URL.Path == "/_matrix/client/r0/join/!foo:example.com" {
|
|
|
|
return mockResponse(http.StatusOK, `{"room_id": "!foo:example.com"}`), nil
|
|
|
|
} else if req.Method == http.MethodPost && req.URL.Path == "/_matrix/client/r0/rooms/!foo:example.com/leave" {
|
|
|
|
return mockResponse(http.StatusOK, `{}`), nil
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("unexpected query: %s %s", req.Method, req.URL.Path)
|
|
|
|
}), config: cfg}
|
|
|
|
|
|
|
|
room, err := c.JoinRoom("!foo:example.com", "")
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, "!foo:example.com", room.ID)
|
|
|
|
assert.False(t, room.HasLeft)
|
|
|
|
|
|
|
|
err = c.LeaveRoom("!foo:example.com")
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.True(t, room.HasLeft)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestContainer_Download(t *testing.T) {
|
|
|
|
defer os.RemoveAll("/tmp/gomuks-mxtest-3")
|
|
|
|
cfg := config.NewConfig("/tmp/gomuks-mxtest-3", "/tmp/gomuks-mxtest-3")
|
2018-05-22 16:54:35 +03:00
|
|
|
cfg.LoadAll()
|
2018-05-10 20:31:11 +03:00
|
|
|
callCounter := 0
|
|
|
|
c := Container{client: mockClient(func(req *http.Request) (*http.Response, error) {
|
|
|
|
if req.Method != http.MethodGet || req.URL.Path != "/_matrix/media/v1/download/example.com/foobar" {
|
|
|
|
return nil, fmt.Errorf("unexpected query: %s %s", req.Method, req.URL.Path)
|
|
|
|
}
|
|
|
|
callCounter++
|
|
|
|
return mockResponse(http.StatusOK, `example file`), nil
|
|
|
|
}), config: cfg}
|
|
|
|
|
|
|
|
// Check that download works
|
|
|
|
data, hs, id, err := c.Download("mxc://example.com/foobar")
|
|
|
|
assert.Equal(t, "example.com", hs)
|
|
|
|
assert.Equal(t, "foobar", id)
|
|
|
|
assert.Equal(t, 1, callCounter)
|
|
|
|
assert.Equal(t, []byte("example file"), data)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
|
|
|
|
// Check that cache works
|
|
|
|
data, _, _, err = c.Download("mxc://example.com/foobar")
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, []byte("example file"), data)
|
|
|
|
assert.Equal(t, 1, callCounter)
|
|
|
|
}
|
|
|
|
|
2018-05-11 08:12:11 +03:00
|
|
|
func TestContainer_Download_InvalidURL(t *testing.T) {
|
|
|
|
c := Container{}
|
|
|
|
data, hs, id, err := c.Download("mxc://invalid mxc")
|
|
|
|
assert.NotNil(t, err)
|
|
|
|
assert.Empty(t, id)
|
|
|
|
assert.Empty(t, hs)
|
|
|
|
assert.Empty(t, data)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestContainer_GetHistory(t *testing.T) {
|
|
|
|
c := Container{client: mockClient(func(req *http.Request) (*http.Response, error) {
|
|
|
|
if req.Method != http.MethodGet || req.URL.Path != "/_matrix/client/r0/rooms/!foo:maunium.net/messages" {
|
|
|
|
return nil, fmt.Errorf("unexpected query: %s %s", req.Method, req.URL.Path)
|
|
|
|
}
|
|
|
|
return mockResponse(http.StatusOK, `{"start": "123", "end": "456", "chunk": [{"event_id": "it works"}]}`), nil
|
|
|
|
})}
|
|
|
|
|
|
|
|
history, prevBatch, err := c.GetHistory("!foo:maunium.net", "123", 5)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
assert.Equal(t, "it works", history[0].ID)
|
|
|
|
assert.Equal(t, "456", prevBatch)
|
|
|
|
}
|
|
|
|
|
2018-11-14 00:00:35 +02:00
|
|
|
func mockClient(fn func(*http.Request) (*http.Response, error)) *mautrix.Client {
|
|
|
|
client, _ := mautrix.NewClient("https://example.com", "@user:example.com", "foobar")
|
2018-05-10 20:03:19 +03:00
|
|
|
client.Client = &http.Client{Transport: MockRoundTripper{RT: fn}}
|
|
|
|
return client
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseBody(req *http.Request) map[string]interface{} {
|
|
|
|
rawBody, err := ioutil.ReadAll(req.Body)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
data := make(map[string]interface{})
|
|
|
|
|
|
|
|
err = json.Unmarshal(rawBody, &data)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return data
|
|
|
|
}
|
|
|
|
|
|
|
|
func mockResponse(status int, body string) *http.Response {
|
|
|
|
return &http.Response{
|
|
|
|
StatusCode: status,
|
|
|
|
Body: ioutil.NopCloser(strings.NewReader(body)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type MockRoundTripper struct {
|
|
|
|
RT func(*http.Request) (*http.Response, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t MockRoundTripper) RoundTrip(req *http.Request) (*http.Response, error) {
|
|
|
|
return t.RT(req)
|
|
|
|
}
|