548 lines
14 KiB
Go
548 lines
14 KiB
Go
// 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/>.
|
|
|
|
package matrix
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"net/url"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
|
|
"gopkg.in/russross/blackfriday.v2"
|
|
"maunium.net/go/gomatrix"
|
|
"maunium.net/go/gomuks/config"
|
|
"maunium.net/go/gomuks/debug"
|
|
"maunium.net/go/gomuks/interface"
|
|
"maunium.net/go/gomuks/lib/bfhtml"
|
|
"maunium.net/go/gomuks/matrix/pushrules"
|
|
"maunium.net/go/gomuks/matrix/rooms"
|
|
)
|
|
|
|
// Container is a wrapper for a gomatrix Client and some other stuff.
|
|
//
|
|
// It is used for all Matrix calls from the UI and Matrix event handlers.
|
|
type Container struct {
|
|
client *gomatrix.Client
|
|
syncer *GomuksSyncer
|
|
gmx ifc.Gomuks
|
|
ui ifc.GomuksUI
|
|
config *config.Config
|
|
running bool
|
|
stop chan bool
|
|
|
|
typing int64
|
|
}
|
|
|
|
// NewContainer creates a new Container for the given Gomuks instance.
|
|
func NewContainer(gmx ifc.Gomuks) *Container {
|
|
c := &Container{
|
|
config: gmx.Config(),
|
|
ui: gmx.UI(),
|
|
gmx: gmx,
|
|
}
|
|
|
|
return c
|
|
}
|
|
|
|
// InitClient initializes the gomatrix client and connects to the homeserver specified in the config.
|
|
func (c *Container) InitClient() error {
|
|
if len(c.config.HS) == 0 {
|
|
return fmt.Errorf("no homeserver in config")
|
|
}
|
|
|
|
if c.client != nil {
|
|
c.Stop()
|
|
c.client = nil
|
|
}
|
|
|
|
var mxid, accessToken string
|
|
if c.config.Session != nil {
|
|
accessToken = c.config.Session.AccessToken
|
|
mxid = c.config.UserID
|
|
}
|
|
|
|
var err error
|
|
c.client, err = gomatrix.NewClient(c.config.HS, mxid, accessToken)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c.stop = make(chan bool, 1)
|
|
|
|
if c.config.Session != nil && len(accessToken) > 0 {
|
|
go c.Start()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Initialized returns whether or not the gomatrix client is initialized (see InitClient())
|
|
func (c *Container) Initialized() bool {
|
|
return c.client != nil
|
|
}
|
|
|
|
// Login sends a password login request with the given username and password.
|
|
func (c *Container) Login(user, password string) error {
|
|
resp, err := c.client.Login(&gomatrix.ReqLogin{
|
|
Type: "m.login.password",
|
|
User: user,
|
|
Password: password,
|
|
InitialDeviceDisplayName: "gomuks",
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
c.client.SetCredentials(resp.UserID, resp.AccessToken)
|
|
c.config.UserID = resp.UserID
|
|
c.config.Save()
|
|
|
|
c.config.Session = c.config.NewSession(resp.UserID)
|
|
c.config.Session.AccessToken = resp.AccessToken
|
|
c.config.Session.Save()
|
|
|
|
go c.Start()
|
|
|
|
return nil
|
|
}
|
|
|
|
// Stop stops the Matrix syncer.
|
|
func (c *Container) Stop() {
|
|
if c.running {
|
|
debug.Print("Stopping Matrix container...")
|
|
c.stop <- true
|
|
c.client.StopSync()
|
|
}
|
|
}
|
|
|
|
// Client returns the underlying gomatrix client object.
|
|
func (c *Container) Client() *gomatrix.Client {
|
|
return c.client
|
|
}
|
|
|
|
// UpdatePushRules fetches the push notification rules from the server and stores them in the current Session object.
|
|
func (c *Container) UpdatePushRules() {
|
|
debug.Print("Updating push rules...")
|
|
resp, err := pushrules.GetPushRules(c.client)
|
|
if err != nil {
|
|
debug.Print("Failed to fetch push rules:", err)
|
|
}
|
|
c.config.Session.PushRules = resp
|
|
}
|
|
|
|
// PushRules returns the push notification rules. If no push rules are cached, UpdatePushRules() will be called first.
|
|
func (c *Container) PushRules() *pushrules.PushRuleset {
|
|
if c.config.Session.PushRules == nil {
|
|
c.UpdatePushRules()
|
|
}
|
|
return c.config.Session.PushRules
|
|
}
|
|
|
|
// OnLogout stops the syncer and moves the UI back to the login view.
|
|
func (c *Container) OnLogout() {
|
|
c.Stop()
|
|
c.ui.SetView(ifc.ViewLogin)
|
|
}
|
|
|
|
// OnLogin initializes the syncer and updates the room list.
|
|
func (c *Container) OnLogin() {
|
|
c.client.Store = c.config.Session
|
|
|
|
debug.Print("Initializing syncer")
|
|
c.syncer = NewGomuksSyncer(c.config.Session)
|
|
c.syncer.OnEventType("m.room.message", c.HandleMessage)
|
|
c.syncer.OnEventType("m.room.member", c.HandleMembership)
|
|
c.syncer.OnEventType("m.typing", c.HandleTyping)
|
|
c.syncer.OnEventType("m.push_rules", c.HandlePushRules)
|
|
c.syncer.OnEventType("m.tag", c.HandleTag)
|
|
c.syncer.InitDoneCallback = func() {
|
|
c.config.Session.InitialSyncDone = true
|
|
c.ui.Render()
|
|
}
|
|
c.client.Syncer = c.syncer
|
|
|
|
debug.Print("Setting existing rooms")
|
|
c.ui.MainView().SetRooms(c.config.Session.Rooms)
|
|
|
|
debug.Print("OnLogin() done.")
|
|
}
|
|
|
|
// Start moves the UI to the main view, calls OnLogin() and runs the syncer forever until stopped with Stop()
|
|
func (c *Container) Start() {
|
|
defer debug.Recover()
|
|
|
|
c.ui.SetView(ifc.ViewMain)
|
|
c.OnLogin()
|
|
|
|
if c.client == nil {
|
|
return
|
|
}
|
|
|
|
debug.Print("Starting sync...")
|
|
c.running = true
|
|
for {
|
|
select {
|
|
case <-c.stop:
|
|
debug.Print("Stopping sync...")
|
|
c.running = false
|
|
return
|
|
default:
|
|
if err := c.client.Sync(); err != nil {
|
|
debug.Print("Sync() errored", err)
|
|
} else {
|
|
debug.Print("Sync() returned without error")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// HandleMessage is the event handler for the m.room.message timeline event.
|
|
func (c *Container) HandleMessage(evt *gomatrix.Event) {
|
|
mainView := c.ui.MainView()
|
|
|
|
roomView := mainView.GetRoom(evt.RoomID)
|
|
if roomView == nil {
|
|
debug.Printf("Failed to handle event %v: No room view found.", evt)
|
|
return
|
|
}
|
|
|
|
message := mainView.ParseEvent(roomView, evt)
|
|
if message != nil {
|
|
roomView.AddMessage(message, ifc.AppendMessage)
|
|
if c.syncer.FirstSyncDone {
|
|
pushRules := c.PushRules().GetActions(roomView.MxRoom(), evt).Should()
|
|
mainView.NotifyMessage(roomView.MxRoom(), message, pushRules)
|
|
c.ui.Render()
|
|
}
|
|
} else {
|
|
debug.Printf("Parsing event %v failed (ParseEvent() returned nil).", evt)
|
|
}
|
|
}
|
|
|
|
// HandlePushRules is the event handler for the m.push_rules account data event.
|
|
func (c *Container) HandlePushRules(evt *gomatrix.Event) {
|
|
debug.Print("Received updated push rules")
|
|
var err error
|
|
c.config.Session.PushRules, err = pushrules.EventToPushRules(evt)
|
|
if err != nil {
|
|
debug.Print("Failed to convert event to push rules:", err)
|
|
}
|
|
}
|
|
|
|
// HandleTag is the event handler for the m.tag account data event.
|
|
func (c *Container) HandleTag(evt *gomatrix.Event) {
|
|
room := c.config.Session.GetRoom(evt.RoomID)
|
|
|
|
tags, _ := evt.Content["tags"].(map[string]interface{})
|
|
newTags := make([]rooms.RoomTag, len(tags))
|
|
index := 0
|
|
for tag, infoifc := range tags {
|
|
info, _ := infoifc.(map[string]interface{})
|
|
order, _ := info["order"].(float64)
|
|
newTags[index] = rooms.RoomTag{
|
|
Tag: tag,
|
|
Order: order,
|
|
}
|
|
index++
|
|
}
|
|
|
|
mainView := c.ui.MainView()
|
|
mainView.UpdateTags(room, newTags)
|
|
}
|
|
|
|
func (c *Container) processOwnMembershipChange(evt *gomatrix.Event) {
|
|
membership, _ := evt.Content["membership"].(string)
|
|
prevMembership := "leave"
|
|
if evt.Unsigned.PrevContent != nil {
|
|
prevMembership, _ = evt.Unsigned.PrevContent["membership"].(string)
|
|
}
|
|
debug.Printf("Processing own membership change: %s->%s in %s", prevMembership, membership, evt.RoomID)
|
|
if membership == prevMembership {
|
|
return
|
|
}
|
|
switch membership {
|
|
case "join":
|
|
c.ui.MainView().AddRoom(evt.RoomID)
|
|
case "leave":
|
|
c.ui.MainView().RemoveRoom(evt.RoomID)
|
|
case "invite":
|
|
// TODO handle
|
|
debug.Printf("%s invited the user to %s", evt.Sender, evt.RoomID)
|
|
}
|
|
}
|
|
|
|
// HandleMembership is the event handler for the m.room.membership state event.
|
|
func (c *Container) HandleMembership(evt *gomatrix.Event) {
|
|
if evt.StateKey != nil && *evt.StateKey == c.config.Session.UserID {
|
|
c.processOwnMembershipChange(evt)
|
|
}
|
|
|
|
if !c.config.Session.InitialSyncDone /*&& evt.Timestamp < time.Now().Add(-1*time.Hour).Unix()*/ {
|
|
// We don't care about other users' membership events in the initial sync.
|
|
return
|
|
}
|
|
|
|
mainView := c.ui.MainView()
|
|
roomView := mainView.GetRoom(evt.RoomID)
|
|
if roomView == nil {
|
|
return
|
|
}
|
|
|
|
message := mainView.ParseEvent(roomView, evt)
|
|
if message != nil {
|
|
// TODO this shouldn't be necessary
|
|
//roomView.MxRoom().UpdateState(evt)
|
|
// TODO This should probably also be in a different place
|
|
//roomView.UpdateUserList()
|
|
|
|
roomView.AddMessage(message, ifc.AppendMessage)
|
|
// We don't want notifications at startup.
|
|
if c.syncer.FirstSyncDone {
|
|
pushRules := c.PushRules().GetActions(roomView.MxRoom(), evt).Should()
|
|
mainView.NotifyMessage(roomView.MxRoom(), message, pushRules)
|
|
c.ui.Render()
|
|
}
|
|
}
|
|
}
|
|
|
|
// HandleTyping is the event handler for the m.typing event.
|
|
func (c *Container) HandleTyping(evt *gomatrix.Event) {
|
|
users := evt.Content["user_ids"].([]interface{})
|
|
|
|
strUsers := make([]string, len(users))
|
|
for i, user := range users {
|
|
strUsers[i] = user.(string)
|
|
}
|
|
c.ui.MainView().SetTyping(evt.RoomID, strUsers)
|
|
}
|
|
|
|
// SendMessage sends a message with the given text to the given room.
|
|
func (c *Container) SendMessage(roomID, msgtype, text string) (string, error) {
|
|
defer debug.Recover()
|
|
c.SendTyping(roomID, false)
|
|
resp, err := c.client.SendMessageEvent(roomID, "m.room.message",
|
|
gomatrix.TextMessage{MsgType: msgtype, Body: text})
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return resp.EventID, nil
|
|
}
|
|
|
|
func (c *Container) RenderMarkdown(text string) string {
|
|
parser := blackfriday.New(
|
|
blackfriday.WithExtensions(blackfriday.NoIntraEmphasis |
|
|
blackfriday.Tables |
|
|
blackfriday.FencedCode |
|
|
blackfriday.Strikethrough |
|
|
blackfriday.SpaceHeadings |
|
|
blackfriday.DefinitionLists))
|
|
ast := parser.Parse([]byte(text))
|
|
|
|
renderer := bfhtml.HTMLRenderer{
|
|
HTMLRenderer: blackfriday.NewHTMLRenderer(blackfriday.HTMLRendererParameters{
|
|
Flags: blackfriday.UseXHTML,
|
|
}),
|
|
}
|
|
|
|
var buf strings.Builder
|
|
renderer.RenderHeader(&buf, ast)
|
|
ast.Walk(func(node *blackfriday.Node, entering bool) blackfriday.WalkStatus {
|
|
return renderer.RenderNode(&buf, node, entering)
|
|
})
|
|
renderer.RenderFooter(&buf, ast)
|
|
return buf.String()
|
|
}
|
|
|
|
var mentionRegex = regexp.MustCompile("\\[(.+?)]\\(https://matrix.to/#/@.+?:.+?\\)")
|
|
var roomRegex = regexp.MustCompile("\\[.+?]\\(https://matrix.to/#/(#.+?:[^/]+?)\\)")
|
|
|
|
func (c *Container) SendMarkdownMessage(roomID, msgtype, text string) (string, error) {
|
|
defer debug.Recover()
|
|
|
|
html := c.RenderMarkdown(text)
|
|
if html == text {
|
|
return c.SendMessage(roomID, msgtype, text)
|
|
}
|
|
|
|
// Remove markdown link stuff from plaintext mentions and room links
|
|
text = mentionRegex.ReplaceAllString(text, "$1")
|
|
text = roomRegex.ReplaceAllString(text, "$1")
|
|
|
|
c.SendTyping(roomID, false)
|
|
resp, err := c.client.SendMessageEvent(roomID, "m.room.message",
|
|
map[string]interface{}{
|
|
"msgtype": msgtype,
|
|
"body": text,
|
|
"format": "org.matrix.custom.html",
|
|
"formatted_body": html,
|
|
})
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return resp.EventID, nil
|
|
}
|
|
|
|
// SendTyping sets whether or not the user is typing in the given room.
|
|
func (c *Container) SendTyping(roomID string, typing bool) {
|
|
defer debug.Recover()
|
|
ts := time.Now().Unix()
|
|
if c.typing > ts && typing {
|
|
return
|
|
}
|
|
|
|
if typing {
|
|
c.client.UserTyping(roomID, true, 20000)
|
|
c.typing = ts + 15
|
|
} else {
|
|
c.client.UserTyping(roomID, false, 0)
|
|
c.typing = 0
|
|
}
|
|
}
|
|
|
|
// JoinRoom makes the current user try to join the given room.
|
|
func (c *Container) JoinRoom(roomID string) error {
|
|
if len(roomID) == 0 {
|
|
return fmt.Errorf("invalid room ID")
|
|
}
|
|
|
|
server := ""
|
|
if roomID[0] == '!' {
|
|
server = roomID[strings.Index(roomID, ":")+1:]
|
|
}
|
|
|
|
_, err := c.client.JoinRoom(roomID, server, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// LeaveRoom makes the current user leave the given room.
|
|
func (c *Container) LeaveRoom(roomID string) error {
|
|
if len(roomID) == 0 {
|
|
return fmt.Errorf("invalid room ID")
|
|
}
|
|
|
|
_, err := c.client.LeaveRoom(roomID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// getState requests the state of the given room.
|
|
func (c *Container) getState(roomID string) []*gomatrix.Event {
|
|
content := make([]*gomatrix.Event, 0)
|
|
err := c.client.StateEvent(roomID, "", "", &content)
|
|
if err != nil {
|
|
debug.Print("Error getting state of", roomID, err)
|
|
return nil
|
|
}
|
|
return content
|
|
}
|
|
|
|
// GetHistory fetches room history.
|
|
func (c *Container) GetHistory(roomID, prevBatch string, limit int) ([]gomatrix.Event, string, error) {
|
|
resp, err := c.client.Messages(roomID, prevBatch, "", 'b', limit)
|
|
if err != nil {
|
|
return nil, "", err
|
|
}
|
|
return resp.Chunk, resp.End, nil
|
|
}
|
|
|
|
// GetRoom gets the room instance stored in the session.
|
|
//
|
|
// If the room doesn't have any state events stored, its state will be updated.
|
|
func (c *Container) GetRoom(roomID string) *rooms.Room {
|
|
room := c.config.Session.GetRoom(roomID)
|
|
if room != nil && len(room.State) == 0 {
|
|
/*events := c.getState(room.ID)
|
|
if events != nil {
|
|
for _, event := range events {
|
|
room.UpdateState(event)
|
|
}
|
|
}*/
|
|
}
|
|
return room
|
|
}
|
|
|
|
var mxcRegex = regexp.MustCompile("mxc://(.+)/(.+)")
|
|
|
|
func (c *Container) Download(mxcURL string) (data []byte, hs, id string, err error) {
|
|
parts := mxcRegex.FindStringSubmatch(mxcURL)
|
|
if parts == nil || len(parts) != 3 {
|
|
err = fmt.Errorf("invalid matrix content URL")
|
|
return
|
|
}
|
|
|
|
hs = parts[1]
|
|
id = parts[2]
|
|
|
|
cacheFile := c.GetCachePath(hs, id)
|
|
if _, err = os.Stat(cacheFile); err != nil {
|
|
data, err = ioutil.ReadFile(cacheFile)
|
|
if err == nil {
|
|
return
|
|
}
|
|
}
|
|
|
|
data, err = c.download(hs, id, cacheFile)
|
|
return
|
|
}
|
|
|
|
func (c *Container) download(hs, id, cacheFile string) (data []byte, err error) {
|
|
dlURL, _ := url.Parse(c.client.HomeserverURL.String())
|
|
dlURL.Path = path.Join(dlURL.Path, "/_matrix/media/v1/download", hs, id)
|
|
|
|
var resp *http.Response
|
|
resp, err = c.client.Client.Get(dlURL.String())
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
var buf bytes.Buffer
|
|
_, err = io.Copy(&buf, resp.Body)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
data = buf.Bytes()
|
|
|
|
err = ioutil.WriteFile(cacheFile, data, 0600)
|
|
return
|
|
}
|
|
|
|
func (c *Container) GetCachePath(homeserver, fileID string) string {
|
|
dir := filepath.Join(c.config.MediaDir, homeserver)
|
|
|
|
err := os.MkdirAll(dir, 0700)
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
|
|
return filepath.Join(dir, fileID)
|
|
}
|