matrix.go 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  1. package bridge
  2. import (
  3. "strings"
  4. "maunium.net/go/maulogger/v2"
  5. "maunium.net/go/mautrix"
  6. "maunium.net/go/mautrix/appservice"
  7. "maunium.net/go/mautrix/event"
  8. "maunium.net/go/mautrix/format"
  9. "maunium.net/go/mautrix/id"
  10. )
  11. type matrixHandler struct {
  12. as *appservice.AppService
  13. bridge *Bridge
  14. log maulogger.Logger
  15. cmd *commandHandler
  16. }
  17. func (b *Bridge) setupEvents() {
  18. b.eventProcessor = appservice.NewEventProcessor(b.as)
  19. b.matrixHandler = &matrixHandler{
  20. as: b.as,
  21. bridge: b,
  22. log: b.log.Sub("Matrix"),
  23. cmd: newCommandHandler(b),
  24. }
  25. b.eventProcessor.On(event.EventMessage, b.matrixHandler.handleMessage)
  26. b.eventProcessor.On(event.EventReaction, b.matrixHandler.handleReaction)
  27. b.eventProcessor.On(event.EventRedaction, b.matrixHandler.handleRedaction)
  28. b.eventProcessor.On(event.StateMember, b.matrixHandler.handleMembership)
  29. }
  30. func (mh *matrixHandler) join(evt *event.Event, intent *appservice.IntentAPI) *mautrix.RespJoinedMembers {
  31. resp, err := intent.JoinRoomByID(evt.RoomID)
  32. if err != nil {
  33. mh.log.Debugfln("Failed to join room %s as %s with invite from %s: %v", evt.RoomID, intent.UserID, evt.Sender, err)
  34. return nil
  35. }
  36. members, err := intent.JoinedMembers(resp.RoomID)
  37. if err != nil {
  38. intent.LeaveRoom(resp.RoomID)
  39. mh.log.Debugfln("Failed to get members in room %s after accepting invite from %s as %s: %v", resp.RoomID, evt.Sender, intent.UserID, err)
  40. return nil
  41. }
  42. if len(members.Joined) < 2 {
  43. intent.LeaveRoom(resp.RoomID)
  44. mh.log.Debugln("Leaving empty room", resp.RoomID, "after accepting invite from", evt.Sender, "as", intent.UserID)
  45. return nil
  46. }
  47. return members
  48. }
  49. func (mh *matrixHandler) ignoreEvent(evt *event.Event) bool {
  50. return false
  51. }
  52. func (mh *matrixHandler) handleMessage(evt *event.Event) {
  53. if mh.ignoreEvent(evt) {
  54. return
  55. }
  56. user := mh.bridge.GetUserByMXID(evt.Sender)
  57. if user == nil {
  58. mh.log.Debugln("unknown user", evt.Sender)
  59. return
  60. }
  61. content := evt.Content.AsMessage()
  62. content.RemoveReplyFallback()
  63. if content.MsgType == event.MsgText {
  64. prefix := mh.bridge.Config.Bridge.CommandPrefix
  65. hasPrefix := strings.HasPrefix(content.Body, prefix)
  66. if hasPrefix {
  67. content.Body = strings.TrimLeft(content.Body[len(prefix):], " ")
  68. }
  69. if hasPrefix || evt.RoomID == user.ManagementRoom {
  70. mh.cmd.handle(evt.RoomID, user, content.Body, content.GetReplyTo())
  71. return
  72. }
  73. }
  74. portal := mh.bridge.GetPortalByMXID(evt.RoomID)
  75. if portal != nil {
  76. portal.matrixMessages <- portalMatrixMessage{user: user, evt: evt}
  77. }
  78. }
  79. func (mh *matrixHandler) joinAndCheckMembers(evt *event.Event, intent *appservice.IntentAPI) *mautrix.RespJoinedMembers {
  80. resp, err := intent.JoinRoomByID(evt.RoomID)
  81. if err != nil {
  82. mh.log.Debugfln("Failed to join room %q as %q with invite from %q: %v", evt.RoomID, intent.UserID, evt.Sender, err)
  83. return nil
  84. }
  85. members, err := intent.JoinedMembers(resp.RoomID)
  86. if err != nil {
  87. mh.log.Debugfln("Failed to get members in room %q with invite from %q as %q: %v", resp.RoomID, evt.Sender, intent.UserID, err)
  88. return nil
  89. }
  90. if len(members.Joined) < 2 {
  91. mh.log.Debugfln("Leaving empty room %q with invite from %q as %q", resp.RoomID, evt.Sender, intent.UserID)
  92. intent.LeaveRoom(resp.RoomID)
  93. return nil
  94. }
  95. return members
  96. }
  97. func (mh *matrixHandler) sendNoticeWithmarkdown(roomID id.RoomID, message string) (*mautrix.RespSendEvent, error) {
  98. intent := mh.as.BotIntent()
  99. content := format.RenderMarkdown(message, true, false)
  100. content.MsgType = event.MsgNotice
  101. return intent.SendMessageEvent(roomID, event.EventMessage, content)
  102. }
  103. func (mh *matrixHandler) handleBotInvite(evt *event.Event) {
  104. intent := mh.as.BotIntent()
  105. user := mh.bridge.GetUserByMXID(evt.Sender)
  106. if user == nil {
  107. return
  108. }
  109. members := mh.joinAndCheckMembers(evt, intent)
  110. if members == nil {
  111. return
  112. }
  113. // If this is a DM and the user doesn't have a management room, make this
  114. // the management room.
  115. if len(members.Joined) == 2 && (user.ManagementRoom == "" || evt.Content.AsMember().IsDirect) {
  116. user.SetManagementRoom(evt.RoomID)
  117. intent.SendNotice(user.ManagementRoom, "This room has been registered as your bridge management/status room")
  118. mh.log.Debugfln("%q registered as management room with %q", evt.RoomID, evt.Sender)
  119. }
  120. // Wait to send the welcome message until we're sure we're not in an empty
  121. // room.
  122. mh.sendNoticeWithmarkdown(evt.RoomID, mh.bridge.Config.Bridge.ManagementRoomText.Welcome)
  123. if evt.RoomID == user.ManagementRoom {
  124. if user.Connected() {
  125. mh.sendNoticeWithmarkdown(evt.RoomID, mh.bridge.Config.Bridge.ManagementRoomText.Connected)
  126. } else {
  127. mh.sendNoticeWithmarkdown(evt.RoomID, mh.bridge.Config.Bridge.ManagementRoomText.NotConnected)
  128. }
  129. additionalHelp := mh.bridge.Config.Bridge.ManagementRoomText.AdditionalHelp
  130. if additionalHelp != "" {
  131. mh.sendNoticeWithmarkdown(evt.RoomID, additionalHelp)
  132. }
  133. }
  134. }
  135. func (mh *matrixHandler) handlePuppetInvite(evt *event.Event, inviter *User, puppet *Puppet) {
  136. mh.log.Warnln("handling puppet invite!")
  137. }
  138. func (mh *matrixHandler) handleMembership(evt *event.Event) {
  139. // Return early if we're supposed to ignore the event.
  140. if mh.ignoreEvent(evt) {
  141. return
  142. }
  143. // Grab the content of the event.
  144. content := evt.Content.AsMember()
  145. // Check if this is a new conversation from a matrix user to the bot
  146. if content.Membership == event.MembershipInvite && id.UserID(evt.GetStateKey()) == mh.as.BotMXID() {
  147. mh.handleBotInvite(evt)
  148. return
  149. }
  150. // Load or create a new user.
  151. user := mh.bridge.GetUserByMXID(evt.Sender)
  152. if user == nil {
  153. return
  154. }
  155. puppet := mh.bridge.GetPuppetByMXID(id.UserID(evt.GetStateKey()))
  156. // Load or create a new portal.
  157. portal := mh.bridge.GetPortalByMXID(evt.RoomID)
  158. if portal == nil {
  159. if content.Membership == event.MembershipInvite && puppet != nil {
  160. mh.handlePuppetInvite(evt, user, puppet)
  161. }
  162. return
  163. }
  164. isSelf := id.UserID(evt.GetStateKey()) == evt.Sender
  165. if content.Membership == event.MembershipLeave {
  166. if evt.Unsigned.PrevContent != nil {
  167. _ = evt.Unsigned.PrevContent.ParseRaw(evt.Type)
  168. prevContent, ok := evt.Unsigned.PrevContent.Parsed.(*event.MemberEventContent)
  169. if ok && prevContent.Membership != "join" {
  170. return
  171. }
  172. }
  173. if isSelf {
  174. portal.handleMatrixLeave(user)
  175. } else if puppet != nil {
  176. portal.handleMatrixKick(user, puppet)
  177. }
  178. } else if content.Membership == event.MembershipInvite {
  179. portal.handleMatrixInvite(user, evt)
  180. }
  181. }
  182. func (mh *matrixHandler) handleReaction(evt *event.Event) {
  183. if mh.ignoreEvent(evt) {
  184. return
  185. }
  186. portal := mh.bridge.GetPortalByMXID(evt.RoomID)
  187. if portal != nil {
  188. portal.handleMatrixReaction(evt)
  189. }
  190. }
  191. func (mh *matrixHandler) handleRedaction(evt *event.Event) {
  192. if mh.ignoreEvent(evt) {
  193. return
  194. }
  195. portal := mh.bridge.GetPortalByMXID(evt.RoomID)
  196. if portal != nil {
  197. portal.handleMatrixRedaction(evt)
  198. }
  199. }