bridge.go 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. // mautrix-whatsapp - A Matrix-WhatsApp puppeting bridge.
  2. // Copyright (C) 2021 Tulir Asokan
  3. //
  4. // This program is free software: you can redistribute it and/or modify
  5. // it under the terms of the GNU Affero General Public License as published by
  6. // the Free Software Foundation, either version 3 of the License, or
  7. // (at your option) any later version.
  8. //
  9. // This program is distributed in the hope that it will be useful,
  10. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. // GNU Affero General Public License for more details.
  13. //
  14. // You should have received a copy of the GNU Affero General Public License
  15. // along with this program. If not, see <https://www.gnu.org/licenses/>.
  16. package config
  17. import (
  18. "strconv"
  19. "strings"
  20. "text/template"
  21. "go.mau.fi/whatsmeow/types"
  22. "maunium.net/go/mautrix/event"
  23. "maunium.net/go/mautrix/id"
  24. )
  25. type BridgeConfig struct {
  26. UsernameTemplate string `yaml:"username_template"`
  27. DisplaynameTemplate string `yaml:"displayname_template"`
  28. DeliveryReceipts bool `yaml:"delivery_receipts"`
  29. MaxConnectionAttempts int `yaml:"max_connection_attempts"`
  30. ConnectionRetryDelay int `yaml:"connection_retry_delay"`
  31. ReportConnectionRetry bool `yaml:"report_connection_retry"`
  32. PortalMessageBuffer int `yaml:"portal_message_buffer"`
  33. CallNotices struct {
  34. Start bool `yaml:"start"`
  35. End bool `yaml:"end"`
  36. } `yaml:"call_notices"`
  37. HistorySync struct {
  38. CreatePortals bool `yaml:"create_portals"`
  39. Backfill bool `yaml:"backfill"`
  40. DoublePuppetBackfill bool `yaml:"double_puppet_backfill"`
  41. }
  42. UserAvatarSync bool `yaml:"user_avatar_sync"`
  43. BridgeMatrixLeave bool `yaml:"bridge_matrix_leave"`
  44. SyncWithCustomPuppets bool `yaml:"sync_with_custom_puppets"`
  45. SyncDirectChatList bool `yaml:"sync_direct_chat_list"`
  46. DefaultBridgeReceipts bool `yaml:"default_bridge_receipts"`
  47. DefaultBridgePresence bool `yaml:"default_bridge_presence"`
  48. LoginSharedSecret string `yaml:"login_shared_secret"`
  49. PrivateChatPortalMeta bool `yaml:"private_chat_portal_meta"`
  50. BridgeNotices bool `yaml:"bridge_notices"`
  51. ResendBridgeInfo bool `yaml:"resend_bridge_info"`
  52. MuteBridging bool `yaml:"mute_bridging"`
  53. ArchiveTag string `yaml:"archive_tag"`
  54. PinnedTag string `yaml:"pinned_tag"`
  55. TagOnlyOnCreate bool `yaml:"tag_only_on_create"`
  56. MarkReadOnlyOnCreate bool `yaml:"mark_read_only_on_create"`
  57. EnableStatusBroadcast bool `yaml:"enable_status_broadcast"`
  58. WhatsappThumbnail bool `yaml:"whatsapp_thumbnail"`
  59. AllowUserInvite bool `yaml:"allow_user_invite"`
  60. CommandPrefix string `yaml:"command_prefix"`
  61. ManagementRoomText struct {
  62. Welcome string `yaml:"welcome"`
  63. WelcomeConnected string `yaml:"welcome_connected"`
  64. WelcomeUnconnected string `yaml:"welcome_unconnected"`
  65. AdditionalHelp string `yaml:"additional_help"`
  66. } `yaml:"management_room_text"`
  67. Encryption struct {
  68. Allow bool `yaml:"allow"`
  69. Default bool `yaml:"default"`
  70. KeySharing struct {
  71. Allow bool `yaml:"allow"`
  72. RequireCrossSigning bool `yaml:"require_cross_signing"`
  73. RequireVerification bool `yaml:"require_verification"`
  74. } `yaml:"key_sharing"`
  75. } `yaml:"encryption"`
  76. Permissions PermissionConfig `yaml:"permissions"`
  77. Relay RelaybotConfig `yaml:"relay"`
  78. usernameTemplate *template.Template `yaml:"-"`
  79. displaynameTemplate *template.Template `yaml:"-"`
  80. }
  81. func (bc *BridgeConfig) setDefaults() {
  82. bc.MaxConnectionAttempts = 3
  83. bc.ConnectionRetryDelay = -1
  84. bc.ReportConnectionRetry = true
  85. bc.PortalMessageBuffer = 128
  86. bc.CallNotices.Start = true
  87. bc.CallNotices.End = true
  88. bc.HistorySync.CreatePortals = true
  89. bc.UserAvatarSync = true
  90. bc.BridgeMatrixLeave = true
  91. bc.SyncWithCustomPuppets = true
  92. bc.DefaultBridgePresence = true
  93. bc.DefaultBridgeReceipts = true
  94. bc.BridgeNotices = true
  95. bc.EnableStatusBroadcast = true
  96. bc.ManagementRoomText.Welcome = "Hello, I'm a WhatsApp bridge bot."
  97. bc.ManagementRoomText.WelcomeConnected = "Use `help` for help."
  98. bc.ManagementRoomText.WelcomeUnconnected = "Use `help` for help or `login` to log in."
  99. }
  100. type umBridgeConfig BridgeConfig
  101. func (bc *BridgeConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
  102. err := unmarshal((*umBridgeConfig)(bc))
  103. if err != nil {
  104. return err
  105. }
  106. bc.usernameTemplate, err = template.New("username").Parse(bc.UsernameTemplate)
  107. if err != nil {
  108. return err
  109. }
  110. bc.displaynameTemplate, err = template.New("displayname").Parse(bc.DisplaynameTemplate)
  111. if err != nil {
  112. return err
  113. }
  114. return nil
  115. }
  116. type UsernameTemplateArgs struct {
  117. UserID id.UserID
  118. }
  119. type legacyContactInfo struct {
  120. types.ContactInfo
  121. Phone string
  122. Notify string
  123. VName string
  124. Name string
  125. Short string
  126. JID string
  127. }
  128. func (bc BridgeConfig) FormatDisplayname(jid types.JID, contact types.ContactInfo) (string, int8) {
  129. var buf strings.Builder
  130. _ = bc.displaynameTemplate.Execute(&buf, legacyContactInfo{
  131. ContactInfo: contact,
  132. Notify: contact.PushName,
  133. VName: contact.BusinessName,
  134. Name: contact.FullName,
  135. Short: contact.FirstName,
  136. Phone: "+" + jid.User,
  137. JID: "+" + jid.User,
  138. })
  139. var quality int8
  140. switch {
  141. case len(contact.PushName) > 0 || len(contact.BusinessName) > 0:
  142. quality = 3
  143. case len(contact.FullName) > 0 || len(contact.FirstName) > 0:
  144. quality = 2
  145. default:
  146. quality = 1
  147. }
  148. return buf.String(), quality
  149. }
  150. func (bc BridgeConfig) FormatUsername(username string) string {
  151. var buf strings.Builder
  152. _ = bc.usernameTemplate.Execute(&buf, username)
  153. return buf.String()
  154. }
  155. type PermissionConfig map[string]PermissionLevel
  156. type PermissionLevel int
  157. const (
  158. PermissionLevelDefault PermissionLevel = 0
  159. PermissionLevelRelaybot PermissionLevel = 5
  160. PermissionLevelUser PermissionLevel = 10
  161. PermissionLevelAdmin PermissionLevel = 100
  162. )
  163. func (pc *PermissionConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
  164. rawPC := make(map[string]string)
  165. err := unmarshal(&rawPC)
  166. if err != nil {
  167. return err
  168. }
  169. if *pc == nil {
  170. *pc = make(map[string]PermissionLevel)
  171. }
  172. for key, value := range rawPC {
  173. switch strings.ToLower(value) {
  174. case "relaybot":
  175. (*pc)[key] = PermissionLevelRelaybot
  176. case "user":
  177. (*pc)[key] = PermissionLevelUser
  178. case "admin":
  179. (*pc)[key] = PermissionLevelAdmin
  180. default:
  181. val, err := strconv.Atoi(value)
  182. if err != nil {
  183. (*pc)[key] = PermissionLevelDefault
  184. } else {
  185. (*pc)[key] = PermissionLevel(val)
  186. }
  187. }
  188. }
  189. return nil
  190. }
  191. func (pc *PermissionConfig) MarshalYAML() (interface{}, error) {
  192. if *pc == nil {
  193. return nil, nil
  194. }
  195. rawPC := make(map[string]string)
  196. for key, value := range *pc {
  197. switch value {
  198. case PermissionLevelRelaybot:
  199. rawPC[key] = "relaybot"
  200. case PermissionLevelUser:
  201. rawPC[key] = "user"
  202. case PermissionLevelAdmin:
  203. rawPC[key] = "admin"
  204. default:
  205. rawPC[key] = strconv.Itoa(int(value))
  206. }
  207. }
  208. return rawPC, nil
  209. }
  210. func (pc PermissionConfig) IsRelaybotWhitelisted(userID id.UserID) bool {
  211. return pc.GetPermissionLevel(userID) >= PermissionLevelRelaybot
  212. }
  213. func (pc PermissionConfig) IsWhitelisted(userID id.UserID) bool {
  214. return pc.GetPermissionLevel(userID) >= PermissionLevelUser
  215. }
  216. func (pc PermissionConfig) IsAdmin(userID id.UserID) bool {
  217. return pc.GetPermissionLevel(userID) >= PermissionLevelAdmin
  218. }
  219. func (pc PermissionConfig) GetPermissionLevel(userID id.UserID) PermissionLevel {
  220. permissions, ok := pc[string(userID)]
  221. if ok {
  222. return permissions
  223. }
  224. _, homeserver, _ := userID.Parse()
  225. permissions, ok = pc[homeserver]
  226. if len(homeserver) > 0 && ok {
  227. return permissions
  228. }
  229. permissions, ok = pc["*"]
  230. if ok {
  231. return permissions
  232. }
  233. return PermissionLevelDefault
  234. }
  235. type RelaybotConfig struct {
  236. Enabled bool `yaml:"enabled"`
  237. AdminOnly bool `yaml:"admin_only"`
  238. MessageFormats map[event.MessageType]string `yaml:"message_formats"`
  239. messageTemplates *template.Template `yaml:"-"`
  240. }
  241. type umRelaybotConfig RelaybotConfig
  242. func (rc *RelaybotConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
  243. err := unmarshal((*umRelaybotConfig)(rc))
  244. if err != nil {
  245. return err
  246. }
  247. rc.messageTemplates = template.New("messageTemplates")
  248. for key, format := range rc.MessageFormats {
  249. _, err := rc.messageTemplates.New(string(key)).Parse(format)
  250. if err != nil {
  251. return err
  252. }
  253. }
  254. return nil
  255. }
  256. type Sender struct {
  257. UserID id.UserID
  258. *event.MemberEventContent
  259. }
  260. type formatData struct {
  261. Sender Sender
  262. Message string
  263. Content *event.MessageEventContent
  264. }
  265. func (rc *RelaybotConfig) FormatMessage(content *event.MessageEventContent, sender id.UserID, member *event.MemberEventContent) (string, error) {
  266. var output strings.Builder
  267. err := rc.messageTemplates.ExecuteTemplate(&output, string(content.MsgType), formatData{
  268. Sender: Sender{
  269. UserID: sender,
  270. MemberEventContent: member,
  271. },
  272. Content: content,
  273. Message: content.FormattedBody,
  274. })
  275. return output.String(), err
  276. }