2020-12-05 16:18:06 +00:00
|
|
|
package bdiscord
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2024-05-23 22:14:45 +00:00
|
|
|
"strings"
|
2020-12-05 16:18:06 +00:00
|
|
|
|
|
|
|
"github.com/42wim/matterbridge/bridge/config"
|
2020-12-06 17:47:10 +00:00
|
|
|
"github.com/42wim/matterbridge/bridge/helper"
|
2022-03-12 16:06:39 +00:00
|
|
|
"github.com/bwmarrin/discordgo"
|
2020-12-05 16:18:06 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// shouldMessageUseWebhooks checks if have a channel specific webhook, if we're not using auto webhooks
|
|
|
|
func (b *Bdiscord) shouldMessageUseWebhooks(msg *config.Message) bool {
|
|
|
|
if b.useAutoWebhooks {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
b.channelsMutex.RLock()
|
|
|
|
defer b.channelsMutex.RUnlock()
|
|
|
|
if ci, ok := b.channelInfoMap[msg.Channel+b.Account]; ok {
|
|
|
|
if ci.Options.WebhookURL != "" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// maybeGetLocalAvatar checks if UseLocalAvatar contains the message's
|
|
|
|
// account or protocol, and if so, returns the Discord avatar (if exists)
|
|
|
|
func (b *Bdiscord) maybeGetLocalAvatar(msg *config.Message) string {
|
|
|
|
for _, val := range b.GetStringSlice("UseLocalAvatar") {
|
|
|
|
if msg.Protocol != val && msg.Account != val {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
member, err := b.getGuildMemberByNick(msg.Username)
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return member.User.AvatarURL("")
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2024-05-23 22:14:45 +00:00
|
|
|
func (b *Bdiscord) webhookSendTextOnly(msg *config.Message, channelID string) (string, error) {
|
|
|
|
msgParts := helper.ClipOrSplitMessage(msg.Text, MessageLength, b.GetString("MessageClipped"), b.GetInt("MessageSplitMaxCount"))
|
2024-05-23 22:23:50 +00:00
|
|
|
msgIds := []string{}
|
2024-05-23 22:14:45 +00:00
|
|
|
for _, msgPart := range msgParts {
|
|
|
|
res, err := b.transmitter.Send(
|
|
|
|
channelID,
|
|
|
|
&discordgo.WebhookParams{
|
|
|
|
Content: msgPart,
|
|
|
|
Username: msg.Username,
|
|
|
|
AvatarURL: msg.Avatar,
|
|
|
|
AllowedMentions: b.getAllowedMentions(),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
} else {
|
|
|
|
msgIds = append(msgIds, res.ID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Exploit that a discord message ID is actually just a large number, so we encode a list of IDs by separating them with ";".
|
|
|
|
return strings.Join(msgIds, ";"), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Bdiscord) webhookSendFilesOnly(msg *config.Message, channelID string) error {
|
|
|
|
for _, f := range msg.Extra["file"] {
|
2024-05-23 22:23:50 +00:00
|
|
|
fi := f.(config.FileInfo) //nolint:forcetypeassert
|
2024-05-23 22:14:45 +00:00
|
|
|
file := discordgo.File{
|
|
|
|
Name: fi.Name,
|
|
|
|
ContentType: "",
|
|
|
|
Reader: bytes.NewReader(*fi.Data),
|
|
|
|
}
|
|
|
|
content := fi.Comment
|
|
|
|
|
|
|
|
// Cannot use the resulting ID for any edits anyway, so throw it away.
|
|
|
|
// This has to be re-enabled when we implement message deletion.
|
|
|
|
_, err := b.transmitter.Send(
|
|
|
|
channelID,
|
|
|
|
&discordgo.WebhookParams{
|
|
|
|
Username: msg.Username,
|
|
|
|
AvatarURL: msg.Avatar,
|
|
|
|
Files: []*discordgo.File{&file},
|
|
|
|
Content: content,
|
|
|
|
AllowedMentions: b.getAllowedMentions(),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
b.Log.Errorf("Could not send file %#v for message %#v: %s", file, msg, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-12-05 16:18:06 +00:00
|
|
|
// webhookSend send one or more message via webhook, taking care of file
|
|
|
|
// uploads (from slack, telegram or mattermost).
|
|
|
|
// Returns messageID and error.
|
2024-05-23 22:14:45 +00:00
|
|
|
func (b *Bdiscord) webhookSend(msg *config.Message, channelID string) (string, error) {
|
2020-12-05 16:18:06 +00:00
|
|
|
var (
|
2024-05-23 22:23:50 +00:00
|
|
|
res string
|
|
|
|
err error
|
2020-12-05 16:18:06 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// If avatar is unset, mutate the message to include the local avatar (but only if settings say we should do this)
|
|
|
|
if msg.Avatar == "" {
|
|
|
|
msg.Avatar = b.maybeGetLocalAvatar(msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
// WebhookParams can have either `Content` or `File`.
|
|
|
|
|
|
|
|
// We can't send empty messages.
|
|
|
|
if msg.Text != "" {
|
2024-05-23 22:14:45 +00:00
|
|
|
res, err = b.webhookSendTextOnly(msg, channelID)
|
2020-12-05 16:18:06 +00:00
|
|
|
}
|
|
|
|
|
2024-05-23 22:14:45 +00:00
|
|
|
if err == nil && msg.Extra != nil {
|
|
|
|
err = b.webhookSendFilesOnly(msg, channelID)
|
2023-03-14 22:16:22 +00:00
|
|
|
}
|
|
|
|
|
2020-12-05 16:18:06 +00:00
|
|
|
return res, err
|
|
|
|
}
|
2020-12-06 17:47:10 +00:00
|
|
|
|
|
|
|
func (b *Bdiscord) handleEventWebhook(msg *config.Message, channelID string) (string, error) {
|
|
|
|
// skip events
|
|
|
|
if msg.Event != "" && msg.Event != config.EventUserAction && msg.Event != config.EventJoinLeave && msg.Event != config.EventTopicChange {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// skip empty messages
|
|
|
|
if msg.Text == "" && (msg.Extra == nil || len(msg.Extra["file"]) == 0) {
|
|
|
|
b.Log.Debugf("Skipping empty message %#v", msg)
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// discord username must be [0..32] max
|
|
|
|
if len(msg.Username) > 32 {
|
|
|
|
msg.Username = msg.Username[0:32]
|
|
|
|
}
|
|
|
|
|
|
|
|
if msg.ID != "" {
|
2024-05-23 22:14:45 +00:00
|
|
|
// Exploit that a discord message ID is actually just a large number, and we encode a list of IDs by separating them with ";".
|
2024-05-23 22:23:50 +00:00
|
|
|
msgIds := strings.Split(msg.ID, ";")
|
2024-05-23 22:14:45 +00:00
|
|
|
msgParts := helper.ClipOrSplitMessage(b.replaceUserMentions(msg.Text), MessageLength, b.GetString("MessageClipped"), len(msgIds))
|
|
|
|
for len(msgParts) < len(msgIds) {
|
|
|
|
msgParts = append(msgParts, "((obsoleted by edit))")
|
|
|
|
}
|
2020-12-06 17:47:10 +00:00
|
|
|
b.Log.Debugf("Editing webhook message")
|
2024-05-23 22:23:50 +00:00
|
|
|
var editErr error = nil
|
2024-05-23 22:14:45 +00:00
|
|
|
for i := range msgParts {
|
|
|
|
// In case of split-messages where some parts remain the same (i.e. only a typo-fix in a huge message), this causes some noop-updates.
|
|
|
|
// TODO: Optimize away noop-updates of un-edited messages
|
2024-05-23 22:23:50 +00:00
|
|
|
editErr = b.transmitter.Edit(channelID, msgIds[i], &discordgo.WebhookParams{
|
2024-05-23 22:14:45 +00:00
|
|
|
Content: msgParts[i],
|
|
|
|
Username: msg.Username,
|
|
|
|
AllowedMentions: b.getAllowedMentions(),
|
|
|
|
})
|
2024-05-23 22:23:50 +00:00
|
|
|
if editErr != nil {
|
2024-05-23 22:14:45 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2024-05-23 22:23:50 +00:00
|
|
|
if editErr == nil {
|
2020-12-06 17:47:10 +00:00
|
|
|
return msg.ID, nil
|
|
|
|
}
|
2024-05-23 22:23:50 +00:00
|
|
|
b.Log.Errorf("Could not edit webhook message(s): %s; sending as new message(s) instead", editErr)
|
2020-12-06 17:47:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
b.Log.Debugf("Processing webhook sending for message %#v", msg)
|
2024-05-23 22:14:45 +00:00
|
|
|
msg.Text = b.replaceUserMentions(msg.Text)
|
|
|
|
msgId, err := b.webhookSend(msg, channelID)
|
2020-12-06 17:47:10 +00:00
|
|
|
if err != nil {
|
2024-05-23 22:14:45 +00:00
|
|
|
b.Log.Errorf("Could not broadcast via webhook for message %#v: %s", msgId, err)
|
2020-12-06 17:47:10 +00:00
|
|
|
return "", err
|
|
|
|
}
|
2024-05-23 22:14:45 +00:00
|
|
|
return msgId, nil
|
2020-12-06 17:47:10 +00:00
|
|
|
}
|