2021-02-22 19:39:22 +00:00
|
|
|
package matrix
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"regexp"
|
2021-02-25 15:35:34 +00:00
|
|
|
"strings"
|
2021-02-22 19:39:22 +00:00
|
|
|
|
2021-02-25 15:35:34 +00:00
|
|
|
"git.netflux.io/rob/esbot/internal/lang"
|
2021-02-22 19:39:22 +00:00
|
|
|
"github.com/rs/zerolog"
|
|
|
|
)
|
|
|
|
|
|
|
|
var cmdRegex = regexp.MustCompile("[[:space:]]*!(?P<Command>[a-zA-Z]+)[[:space:]]+(?P<Arg>[a-zA-Z]+)")
|
|
|
|
|
|
|
|
type EventProcessor interface {
|
|
|
|
ProcessEvents(events []*RawEvent) error
|
|
|
|
}
|
|
|
|
|
|
|
|
type processor struct {
|
2021-02-24 17:20:32 +00:00
|
|
|
baseURL string
|
2021-02-22 19:39:22 +00:00
|
|
|
accessToken string
|
|
|
|
httpclient *http.Client
|
|
|
|
processed map[string]bool
|
|
|
|
logger zerolog.Logger
|
|
|
|
}
|
|
|
|
|
2021-02-24 17:20:32 +00:00
|
|
|
func NewProcessor(baseURL, accessToken string, logger zerolog.Logger) EventProcessor {
|
2021-02-22 19:39:22 +00:00
|
|
|
return &processor{
|
2021-02-24 17:20:32 +00:00
|
|
|
baseURL: baseURL,
|
2021-02-22 19:39:22 +00:00
|
|
|
accessToken: accessToken,
|
|
|
|
httpclient: &http.Client{},
|
|
|
|
processed: make(map[string]bool),
|
|
|
|
logger: logger,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *processor) ProcessEvents(events []*RawEvent) error {
|
|
|
|
for _, e := range events {
|
|
|
|
if p.processed[e.Id] {
|
|
|
|
p.logger.Debug().Str("event_id", e.Id).Msg("already processed event")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if e.StateKey != "" {
|
|
|
|
if e.Content.Membership == "invite" {
|
|
|
|
if err := p.acceptInvite(e); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
p.processed[e.Id] = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if !cmdRegex.MatchString(e.Content.Body) {
|
|
|
|
p.processed[e.Id] = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
match := cmdRegex.FindStringSubmatch(e.Content.Body)
|
|
|
|
cmd := match[1]
|
|
|
|
arg := match[2]
|
|
|
|
|
|
|
|
switch cmd {
|
|
|
|
case "verb":
|
|
|
|
if err := p.handleVerb(e, cmd, arg); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
p.logger.Debug().Str("command", cmd).Str("arg", arg).Msg("unrecognized command")
|
|
|
|
}
|
|
|
|
|
|
|
|
p.processed[e.Id] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *processor) acceptInvite(e *RawEvent) error {
|
2021-02-24 17:20:32 +00:00
|
|
|
url := fmt.Sprintf("%s/_matrix/client/r0/rooms/%s/join", p.baseURL, e.RoomId)
|
2021-02-22 19:39:22 +00:00
|
|
|
req, _ := p.buildRequest("POST", url, bytes.NewReader([]byte("{}")))
|
|
|
|
resp, err := p.httpclient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("http error accepting invitation: %v", err)
|
|
|
|
}
|
2021-02-24 17:15:09 +00:00
|
|
|
defer resp.Body.Close()
|
2021-02-22 19:39:22 +00:00
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return p.handleErrorResponse(e, resp)
|
|
|
|
}
|
|
|
|
p.logger.Info().Str("room_id", e.RoomId).Str("sender", e.Sender).Msg("accepted invite")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-02-25 15:35:34 +00:00
|
|
|
func (p *processor) handleVerb(e *RawEvent, _, arg string) error {
|
|
|
|
verb := lang.GetVerb(strings.ToLower(arg))
|
|
|
|
var sb sendBody
|
|
|
|
if verb != nil {
|
|
|
|
sb = sendBody{
|
|
|
|
Body: verb.String(),
|
|
|
|
MsgType: "m.notice",
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
sb = sendBody{
|
|
|
|
Body: "I don't know this verb. Try another.",
|
|
|
|
MsgType: "m.notice",
|
|
|
|
}
|
2021-02-22 19:39:22 +00:00
|
|
|
}
|
|
|
|
|
2021-02-25 15:35:34 +00:00
|
|
|
encodedBody, err := json.Marshal(sb)
|
2021-02-22 19:39:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error encoding message: %v", err)
|
|
|
|
}
|
|
|
|
|
2021-02-24 17:20:32 +00:00
|
|
|
url := fmt.Sprintf("%s/_matrix/client/r0/rooms/%s/send/m.room.message", p.baseURL, e.RoomId)
|
2021-02-22 19:39:22 +00:00
|
|
|
req, _ := p.buildRequest("POST", url, bytes.NewReader(encodedBody))
|
|
|
|
resp, err := p.httpclient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("http error sending message: %v", err)
|
|
|
|
}
|
2021-02-24 17:15:09 +00:00
|
|
|
defer resp.Body.Close()
|
2021-02-22 19:39:22 +00:00
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return p.handleErrorResponse(e, resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *processor) handleErrorResponse(e *RawEvent, resp *http.Response) error {
|
|
|
|
var er errorResponse
|
|
|
|
if err := json.NewDecoder(resp.Body).Decode(&er); err != nil {
|
|
|
|
return fmt.Errorf("could not decode response: %v", err)
|
|
|
|
}
|
|
|
|
if er.Code == "M_FORBIDDEN" || er.Code == "M_UNKNOWN_TOKEN" {
|
|
|
|
p.processed[e.Id] = true
|
|
|
|
p.logger.Warn().Str("errcode", er.Code).Str("event_id", e.Id).Str("room_id", e.RoomId).Msg("forbidden, ignoring event")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
p.logger.Error().Str("errcode", er.Code).Str("error", er.Error).Msg("error sending message")
|
2021-02-24 17:20:32 +00:00
|
|
|
return fmt.Errorf("server error: %v", er.Error)
|
2021-02-22 19:39:22 +00:00
|
|
|
}
|
|
|
|
|
2021-02-24 17:20:32 +00:00
|
|
|
func (p *processor) buildRequest(method, url string, body io.Reader) (*http.Request, error) {
|
|
|
|
p.logger.Debug().Str("method", method).Str("url", url).Msg("build HTTP req")
|
|
|
|
req, err := http.NewRequest(method, url, body)
|
2021-02-22 19:39:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
req.Header.Set("Content-Type", "application/json")
|
|
|
|
req.Header.Set("Authorization", fmt.Sprintf("Bearer %v", p.accessToken))
|
|
|
|
return req, nil
|
|
|
|
}
|