2017-11-04 14:55:25 +01:00
|
|
|
package helper
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2018-02-03 01:11:11 +01:00
|
|
|
"fmt"
|
2019-02-27 00:41:50 +01:00
|
|
|
"image/png"
|
2017-11-04 14:55:25 +01:00
|
|
|
"io"
|
|
|
|
"net/http"
|
2018-06-09 14:35:02 +02:00
|
|
|
"regexp"
|
2018-04-19 13:04:12 +02:00
|
|
|
"strings"
|
2017-11-04 14:55:25 +01:00
|
|
|
"time"
|
2018-07-22 00:27:49 +02:00
|
|
|
"unicode/utf8"
|
2018-06-09 12:47:40 +02:00
|
|
|
|
2019-02-27 00:41:50 +01:00
|
|
|
"golang.org/x/image/webp"
|
|
|
|
|
2018-06-09 12:47:40 +02:00
|
|
|
"github.com/42wim/matterbridge/bridge/config"
|
2019-11-17 21:18:01 +01:00
|
|
|
"github.com/gomarkdown/markdown"
|
2020-02-02 18:35:43 +01:00
|
|
|
"github.com/gomarkdown/markdown/html"
|
2019-11-17 21:18:01 +01:00
|
|
|
"github.com/gomarkdown/markdown/parser"
|
2018-12-26 15:16:09 +01:00
|
|
|
"github.com/sirupsen/logrus"
|
2017-11-04 14:55:25 +01:00
|
|
|
)
|
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
// DownloadFile downloads the given non-authenticated URL.
|
2017-11-04 14:55:25 +01:00
|
|
|
func DownloadFile(url string) (*[]byte, error) {
|
2018-02-24 23:35:53 +01:00
|
|
|
return DownloadFileAuth(url, "")
|
|
|
|
}
|
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
// DownloadFileAuth downloads the given URL using the specified authentication token.
|
2018-02-24 23:35:53 +01:00
|
|
|
func DownloadFileAuth(url string, auth string) (*[]byte, error) {
|
2017-11-04 14:55:25 +01:00
|
|
|
var buf bytes.Buffer
|
|
|
|
client := &http.Client{
|
|
|
|
Timeout: time.Second * 5,
|
|
|
|
}
|
|
|
|
req, err := http.NewRequest("GET", url, nil)
|
2018-02-24 23:35:53 +01:00
|
|
|
if auth != "" {
|
|
|
|
req.Header.Add("Authorization", auth)
|
|
|
|
}
|
2017-11-04 14:55:25 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-02-07 00:04:02 +01:00
|
|
|
defer resp.Body.Close()
|
2017-11-04 14:55:25 +01:00
|
|
|
io.Copy(&buf, resp.Body)
|
|
|
|
data := buf.Bytes()
|
|
|
|
return &data, nil
|
|
|
|
}
|
2017-11-24 23:27:13 +01:00
|
|
|
|
2021-02-15 22:34:14 +01:00
|
|
|
// DownloadFileAuthRocket downloads the given URL using the specified Rocket user ID and authentication token.
|
|
|
|
func DownloadFileAuthRocket(url, token, userID string) (*[]byte, error) {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
client := &http.Client{
|
|
|
|
Timeout: time.Second * 5,
|
|
|
|
}
|
|
|
|
req, err := http.NewRequest("GET", url, nil)
|
|
|
|
|
|
|
|
req.Header.Add("X-Auth-Token", token)
|
|
|
|
req.Header.Add("X-User-Id", userID)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
resp, err := client.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
_, err = io.Copy(&buf, resp.Body)
|
|
|
|
data := buf.Bytes()
|
|
|
|
return &data, err
|
|
|
|
}
|
|
|
|
|
2018-11-14 22:43:52 +01:00
|
|
|
// GetSubLines splits messages in newline-delimited lines. If maxLineLength is
|
2019-02-23 22:51:27 +01:00
|
|
|
// specified as non-zero GetSubLines will also clip long lines to the maximum
|
|
|
|
// length and insert a warning marker that the line was clipped.
|
2018-11-14 22:43:52 +01:00
|
|
|
//
|
|
|
|
// TODO: The current implementation has the inconvenient that it disregards
|
|
|
|
// word boundaries when splitting but this is hard to solve without potentially
|
|
|
|
// breaking formatting and other stylistic effects.
|
2021-05-27 21:45:23 +02:00
|
|
|
func GetSubLines(message string, maxLineLength int, clippingMessage string) []string {
|
|
|
|
if clippingMessage == "" {
|
|
|
|
clippingMessage = " <clipped message>"
|
|
|
|
}
|
2018-11-14 22:43:52 +01:00
|
|
|
|
|
|
|
var lines []string
|
|
|
|
for _, line := range strings.Split(strings.TrimSpace(message), "\n") {
|
2022-11-26 23:53:48 +01:00
|
|
|
if line == "" {
|
|
|
|
// Prevent sending empty messages, so we'll skip this line
|
|
|
|
// if it has no content.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-11-14 22:43:52 +01:00
|
|
|
if maxLineLength == 0 || len([]byte(line)) <= maxLineLength {
|
|
|
|
lines = append(lines, line)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// !!! WARNING !!!
|
|
|
|
// Before touching the splitting logic below please ensure that you PROPERLY
|
|
|
|
// understand how strings, runes and range loops over strings work in Go.
|
|
|
|
// A good place to start is to read https://blog.golang.org/strings. :-)
|
|
|
|
var splitStart int
|
|
|
|
var startOfPreviousRune int
|
|
|
|
for i := range line {
|
|
|
|
if i-splitStart > maxLineLength-len([]byte(clippingMessage)) {
|
|
|
|
lines = append(lines, line[splitStart:startOfPreviousRune]+clippingMessage)
|
|
|
|
splitStart = startOfPreviousRune
|
|
|
|
}
|
|
|
|
startOfPreviousRune = i
|
2017-11-24 23:27:13 +01:00
|
|
|
}
|
2018-11-14 22:43:52 +01:00
|
|
|
// This last append is safe to do without looking at the remaining byte-length
|
|
|
|
// as we assume that the byte-length of the last rune will never exceed that of
|
|
|
|
// the byte-length of the clipping message.
|
|
|
|
lines = append(lines, line[splitStart:])
|
2017-11-24 23:27:13 +01:00
|
|
|
}
|
2018-11-14 22:43:52 +01:00
|
|
|
return lines
|
2017-11-24 23:27:13 +01:00
|
|
|
}
|
2018-02-03 01:11:11 +01:00
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
// HandleExtra manages the supplementary details stored inside a message's 'Extra' field map.
|
2018-02-03 01:11:11 +01:00
|
|
|
func HandleExtra(msg *config.Message, general *config.Protocol) []config.Message {
|
|
|
|
extra := msg.Extra
|
|
|
|
rmsg := []config.Message{}
|
2018-11-15 20:43:43 +01:00
|
|
|
for _, f := range extra[config.EventFileFailureSize] {
|
2018-11-07 21:35:59 +01:00
|
|
|
fi := f.(config.FileInfo)
|
|
|
|
text := fmt.Sprintf("file %s too big to download (%#v > allowed size: %#v)", fi.Name, fi.Size, general.MediaDownloadSize)
|
2019-02-23 22:51:27 +01:00
|
|
|
rmsg = append(rmsg, config.Message{
|
|
|
|
Text: text,
|
|
|
|
Username: "<system> ",
|
|
|
|
Channel: msg.Channel,
|
|
|
|
Account: msg.Account,
|
|
|
|
})
|
2018-02-03 01:11:11 +01:00
|
|
|
}
|
|
|
|
return rmsg
|
|
|
|
}
|
2018-02-20 00:54:35 +01:00
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
// GetAvatar constructs a URL for a given user-avatar if it is available in the cache.
|
2018-02-20 00:54:35 +01:00
|
|
|
func GetAvatar(av map[string]string, userid string, general *config.Protocol) string {
|
|
|
|
if sha, ok := av[userid]; ok {
|
2018-02-20 17:15:54 +01:00
|
|
|
return general.MediaServerDownload + "/" + sha + "/" + userid + ".png"
|
2018-02-20 00:54:35 +01:00
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
2018-02-23 23:07:23 +01:00
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
// HandleDownloadSize checks a specified filename against the configured download blacklist
|
|
|
|
// and checks a specified file-size against the configure limit.
|
|
|
|
func HandleDownloadSize(logger *logrus.Entry, msg *config.Message, name string, size int64, general *config.Protocol) error {
|
2018-06-09 14:35:02 +02:00
|
|
|
// check blacklist here
|
|
|
|
for _, entry := range general.MediaDownloadBlackList {
|
|
|
|
if entry != "" {
|
|
|
|
re, err := regexp.Compile(entry)
|
|
|
|
if err != nil {
|
2019-02-23 22:51:27 +01:00
|
|
|
logger.Errorf("incorrect regexp %s for %s", entry, msg.Account)
|
2018-06-09 14:35:02 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
if re.MatchString(name) {
|
|
|
|
return fmt.Errorf("Matching blacklist %s. Not downloading %s", entry, name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-02-23 22:51:27 +01:00
|
|
|
logger.Debugf("Trying to download %#v with size %#v", name, size)
|
2018-02-23 23:07:23 +01:00
|
|
|
if int(size) > general.MediaDownloadSize {
|
2018-11-15 20:43:43 +01:00
|
|
|
msg.Event = config.EventFileFailureSize
|
2019-02-23 22:51:27 +01:00
|
|
|
msg.Extra[msg.Event] = append(msg.Extra[msg.Event], config.FileInfo{
|
|
|
|
Name: name,
|
|
|
|
Comment: msg.Text,
|
|
|
|
Size: size,
|
|
|
|
})
|
2018-02-23 23:07:23 +01:00
|
|
|
return fmt.Errorf("File %#v to large to download (%#v). MediaDownloadSize is %#v", name, size, general.MediaDownloadSize)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
// HandleDownloadData adds the data for a remote file into a Matterbridge gateway message.
|
|
|
|
func HandleDownloadData(logger *logrus.Entry, msg *config.Message, name, comment, url string, data *[]byte, general *config.Protocol) {
|
2022-02-05 14:45:54 +01:00
|
|
|
HandleDownloadData2(logger, msg, name, "", comment, url, data, general)
|
|
|
|
}
|
|
|
|
|
|
|
|
// HandleDownloadData adds the data for a remote file into a Matterbridge gateway message.
|
|
|
|
func HandleDownloadData2(logger *logrus.Entry, msg *config.Message, name, id, comment, url string, data *[]byte, general *config.Protocol) {
|
2018-02-23 23:07:23 +01:00
|
|
|
var avatar bool
|
2019-02-23 22:51:27 +01:00
|
|
|
logger.Debugf("Download OK %#v %#v", name, len(*data))
|
2018-11-15 20:43:43 +01:00
|
|
|
if msg.Event == config.EventAvatarDownload {
|
2018-02-23 23:07:23 +01:00
|
|
|
avatar = true
|
|
|
|
}
|
2019-02-23 22:51:27 +01:00
|
|
|
msg.Extra["file"] = append(msg.Extra["file"], config.FileInfo{
|
2022-02-05 14:45:54 +01:00
|
|
|
Name: name,
|
|
|
|
Data: data,
|
|
|
|
URL: url,
|
|
|
|
Comment: comment,
|
|
|
|
Avatar: avatar,
|
|
|
|
NativeID: id,
|
2019-02-23 22:51:27 +01:00
|
|
|
})
|
2018-02-23 23:07:23 +01:00
|
|
|
}
|
2018-04-19 12:53:49 +02:00
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
var emptyLineMatcher = regexp.MustCompile("\n+")
|
|
|
|
|
|
|
|
// RemoveEmptyNewLines collapses consecutive newline characters into a single one and
|
|
|
|
// trims any preceding or trailing newline characters as well.
|
2018-04-19 12:53:49 +02:00
|
|
|
func RemoveEmptyNewLines(msg string) string {
|
2019-02-23 22:51:27 +01:00
|
|
|
return emptyLineMatcher.ReplaceAllString(strings.Trim(msg, "\n"), "\n")
|
2018-04-19 12:53:49 +02:00
|
|
|
}
|
2018-07-22 00:27:49 +02:00
|
|
|
|
2019-02-23 22:51:27 +01:00
|
|
|
// ClipMessage trims a message to the specified length if it exceeds it and adds a warning
|
|
|
|
// to the message in case it does so.
|
2021-05-27 21:45:23 +02:00
|
|
|
func ClipMessage(text string, length int, clippingMessage string) string {
|
|
|
|
if clippingMessage == "" {
|
|
|
|
clippingMessage = " <clipped message>"
|
|
|
|
}
|
|
|
|
|
2018-07-22 00:27:49 +02:00
|
|
|
if len(text) > length {
|
2019-02-23 22:51:27 +01:00
|
|
|
text = text[:length-len(clippingMessage)]
|
2024-05-24 00:02:09 +02:00
|
|
|
for len(text) > 0 {
|
|
|
|
if r, _ := utf8.DecodeLastRuneInString(text); r == utf8.RuneError {
|
|
|
|
text = text[:len(text)-1]
|
|
|
|
// Note: DecodeLastRuneInString only returns the constant value "1" in
|
|
|
|
// case of an error. We do not yet know whether the last rune is now
|
|
|
|
// actually valid. Example: "€" is 0xE2 0x82 0xAC. If we happen to split
|
|
|
|
// the string just before 0xAC, and go back only one byte, that would
|
|
|
|
// leave us with a string that ends in the byte 0xE2, which is not a valid
|
|
|
|
// rune, so we need to try again.
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
2018-07-22 00:27:49 +02:00
|
|
|
}
|
2019-02-23 22:51:27 +01:00
|
|
|
text += clippingMessage
|
2018-07-22 00:27:49 +02:00
|
|
|
}
|
|
|
|
return text
|
|
|
|
}
|
2019-01-06 22:25:19 +01:00
|
|
|
|
2024-05-24 00:14:45 +02:00
|
|
|
func ClipOrSplitMessage(text string, length int, clippingMessage string, splitMax int) []string {
|
|
|
|
var msgParts []string
|
2024-05-24 00:23:50 +02:00
|
|
|
remainingText := text
|
2024-05-24 00:14:45 +02:00
|
|
|
// Invariant of this splitting loop: No text is lost (msgParts+remainingText is the original text),
|
|
|
|
// and all parts is guaranteed to satisfy the length requirement.
|
2024-05-24 00:23:50 +02:00
|
|
|
for len(msgParts) < splitMax-1 && len(remainingText) > length {
|
2024-05-24 00:14:45 +02:00
|
|
|
// Decision: The text needs to be split (again).
|
|
|
|
var chunk string
|
2024-05-24 00:23:50 +02:00
|
|
|
wasted := 0
|
2024-05-24 00:14:45 +02:00
|
|
|
// The longest UTF-8 encoding of a valid rune is 4 bytes (0xF4 0x8F 0xBF 0xBF, encoding U+10FFFF),
|
|
|
|
// so we should never need to waste 4 or more bytes at a time.
|
|
|
|
for wasted < 4 && wasted < length {
|
2024-05-24 00:23:50 +02:00
|
|
|
chunk = remainingText[:length-wasted]
|
2024-05-24 00:14:45 +02:00
|
|
|
if r, _ := utf8.DecodeLastRuneInString(chunk); r == utf8.RuneError {
|
|
|
|
wasted += 1
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Note: At this point, "chunk" might still be invalid, if "text" is very broken.
|
|
|
|
msgParts = append(msgParts, chunk)
|
|
|
|
remainingText = remainingText[len(chunk):]
|
|
|
|
}
|
|
|
|
msgParts = append(msgParts, ClipMessage(remainingText, length, clippingMessage))
|
|
|
|
return msgParts
|
|
|
|
}
|
|
|
|
|
2019-11-17 21:18:01 +01:00
|
|
|
// ParseMarkdown takes in an input string as markdown and parses it to html
|
2019-01-06 22:25:19 +01:00
|
|
|
func ParseMarkdown(input string) string {
|
2020-12-06 19:38:32 +01:00
|
|
|
extensions := parser.HardLineBreak | parser.NoIntraEmphasis | parser.FencedCode
|
2019-11-17 21:18:01 +01:00
|
|
|
markdownParser := parser.NewWithExtensions(extensions)
|
2020-02-02 18:35:43 +01:00
|
|
|
renderer := html.NewRenderer(html.RendererOptions{
|
|
|
|
Flags: 0,
|
|
|
|
})
|
|
|
|
parsedMarkdown := markdown.ToHTML([]byte(input), markdownParser, renderer)
|
2019-11-17 21:18:01 +01:00
|
|
|
res := string(parsedMarkdown)
|
2019-03-03 00:29:29 +01:00
|
|
|
res = strings.TrimPrefix(res, "<p>")
|
|
|
|
res = strings.TrimSuffix(res, "</p>\n")
|
|
|
|
return res
|
2019-01-06 22:25:19 +01:00
|
|
|
}
|
2019-02-27 00:41:50 +01:00
|
|
|
|
Support Telegram animated stickers (tgs) format (#1173)
This is half a fix for #874
This patch introduces a new config flag:
- MediaConvertTgs
These need to be treated independently from the existing
MediaConvertWebPToPNG flag because Tgs→WebP results in an
*animated* WebP, and the WebP→PNG converter can't handle
animated WebP files yet.
Furthermore, some platforms (like discord) don't even support
animated WebP files, so the user may want to fall back to
static PNGs (not APNGs).
The final reason why this is only half a fix is that this
introduces an external dependency, namely lottie, to be
installed like this:
$ pip3 install lottie cairosvg
This patch works by writing the tgs to a temporary file in /tmp,
calling lottie to convert it (this conversion may take several seconds!),
and then deleting the temporary file.
The temporary file is absolutely necessary, as lottie refuses to
work on non-seekable files.
If anyone comes up with a reasonable use case where /tmp is
unavailable, I can add yet another config option for that, if desired.
Telegram will bail out if the option is configured but lottie isn't found.
2020-08-23 22:34:28 +02:00
|
|
|
// ConvertWebPToPNG converts input data (which should be WebP format) to PNG format
|
2019-02-27 00:41:50 +01:00
|
|
|
func ConvertWebPToPNG(data *[]byte) error {
|
|
|
|
r := bytes.NewReader(*data)
|
|
|
|
m, err := webp.Decode(r)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
var output []byte
|
|
|
|
w := bytes.NewBuffer(output)
|
|
|
|
if err := png.Encode(w, m); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
*data = w.Bytes()
|
|
|
|
return nil
|
|
|
|
}
|