mirror of
https://github.com/memohai/Memoh.git
synced 2026-04-27 07:16:19 +09:00
53f080503c
attachment fallback query
766 lines
22 KiB
Go
766 lines
22 KiB
Go
package telegram
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
"unicode/utf8"
|
|
|
|
tgbotapi "github.com/go-telegram-bot-api/telegram-bot-api/v5"
|
|
"github.com/memohai/memoh/internal/channel"
|
|
"github.com/memohai/memoh/internal/media"
|
|
)
|
|
|
|
func TestResolveTelegramSender(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
externalID, displayName, attrs := resolveTelegramSender(nil)
|
|
if externalID != "" || displayName != "" || len(attrs) != 0 {
|
|
t.Fatalf("expected empty sender")
|
|
}
|
|
msg := &tgbotapi.Message{
|
|
From: &tgbotapi.User{ID: 123, UserName: "alice"},
|
|
}
|
|
externalID, displayName, attrs = resolveTelegramSender(msg)
|
|
if externalID != "123" || displayName != "alice" {
|
|
t.Fatalf("unexpected sender: %s %s", externalID, displayName)
|
|
}
|
|
if attrs["user_id"] != "123" || attrs["username"] != "alice" {
|
|
t.Fatalf("unexpected attrs: %#v", attrs)
|
|
}
|
|
}
|
|
|
|
func TestIsTelegramBotMentioned(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
t.Run("text mention", func(t *testing.T) {
|
|
t.Parallel()
|
|
msg := &tgbotapi.Message{
|
|
Text: "hello @MemohBot",
|
|
}
|
|
if !isTelegramBotMentioned(msg, "memohbot") {
|
|
t.Fatalf("expected bot mention from text")
|
|
}
|
|
})
|
|
|
|
t.Run("entity text mention", func(t *testing.T) {
|
|
t.Parallel()
|
|
msg := &tgbotapi.Message{
|
|
Entities: []tgbotapi.MessageEntity{
|
|
{
|
|
Type: "text_mention",
|
|
User: &tgbotapi.User{IsBot: true},
|
|
},
|
|
},
|
|
}
|
|
if !isTelegramBotMentioned(msg, "") {
|
|
t.Fatalf("expected bot mention from text_mention entity")
|
|
}
|
|
})
|
|
|
|
t.Run("not mentioned", func(t *testing.T) {
|
|
t.Parallel()
|
|
msg := &tgbotapi.Message{
|
|
Text: "hello everyone",
|
|
}
|
|
if isTelegramBotMentioned(msg, "memohbot") {
|
|
t.Fatalf("expected no mention")
|
|
}
|
|
})
|
|
}
|
|
|
|
func TestTelegramDescriptorIncludesStreaming(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
caps := adapter.Descriptor().Capabilities
|
|
if !caps.Streaming {
|
|
t.Fatal("expected streaming capability")
|
|
}
|
|
if !caps.Media {
|
|
t.Fatal("expected media capability")
|
|
}
|
|
}
|
|
|
|
func TestBuildTelegramAttachmentIncludesPlatformReference(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
att := adapter.buildTelegramAttachment(nil, channel.AttachmentFile, "file_1", "doc.txt", "text/plain", 10)
|
|
if att.PlatformKey != "file_1" {
|
|
t.Fatalf("unexpected platform key: %s", att.PlatformKey)
|
|
}
|
|
if att.SourcePlatform != Type.String() {
|
|
t.Fatalf("unexpected source platform: %s", att.SourcePlatform)
|
|
}
|
|
}
|
|
|
|
func TestBuildTelegramAttachmentInfersTypeFromMime(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
att := adapter.buildTelegramAttachment(nil, channel.AttachmentFile, "file_2", "photo.jpg", "IMAGE/JPEG; charset=utf-8", 10)
|
|
if att.Type != channel.AttachmentImage {
|
|
t.Fatalf("expected image type, got: %s", att.Type)
|
|
}
|
|
if att.Mime != "image/jpeg" {
|
|
t.Fatalf("expected normalized mime image/jpeg, got: %s", att.Mime)
|
|
}
|
|
}
|
|
|
|
func TestTelegramResolveAttachmentRequiresReference(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
_, err := adapter.ResolveAttachment(context.Background(), channel.ChannelConfig{}, channel.Attachment{})
|
|
if err == nil {
|
|
t.Fatal("expected error when attachment has no platform_key/url")
|
|
}
|
|
if !strings.Contains(err.Error(), "platform_key") {
|
|
t.Fatalf("expected platform_key error, got: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestParseReplyToMessageID(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
if got := parseReplyToMessageID(nil); got != 0 {
|
|
t.Fatalf("nil reply should return 0: %d", got)
|
|
}
|
|
if got := parseReplyToMessageID(&channel.ReplyRef{}); got != 0 {
|
|
t.Fatalf("empty MessageID should return 0: %d", got)
|
|
}
|
|
if got := parseReplyToMessageID(&channel.ReplyRef{MessageID: " 123 "}); got != 123 {
|
|
t.Fatalf("expected 123: %d", got)
|
|
}
|
|
if got := parseReplyToMessageID(&channel.ReplyRef{MessageID: "abc"}); got != 0 {
|
|
t.Fatalf("invalid number should return 0: %d", got)
|
|
}
|
|
}
|
|
|
|
func TestResolveTelegramParseMode(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
if got := resolveTelegramParseMode(channel.MessageFormatMarkdown); got != tgbotapi.ModeMarkdown {
|
|
t.Fatalf("markdown should return ModeMarkdown: %s", got)
|
|
}
|
|
if got := resolveTelegramParseMode(channel.MessageFormatPlain); got != "" {
|
|
t.Fatalf("plain should return empty: %s", got)
|
|
}
|
|
if got := resolveTelegramParseMode(channel.MessageFormatRich); got != "" {
|
|
t.Fatalf("rich should return empty: %s", got)
|
|
}
|
|
}
|
|
|
|
func TestBuildTelegramReplyRef(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
if buildTelegramReplyRef(nil, "123") != nil {
|
|
t.Fatal("nil msg should return nil")
|
|
}
|
|
msg := &tgbotapi.Message{}
|
|
if buildTelegramReplyRef(msg, "123") != nil {
|
|
t.Fatal("msg without ReplyToMessage should return nil")
|
|
}
|
|
msg.ReplyToMessage = &tgbotapi.Message{MessageID: 42}
|
|
ref := buildTelegramReplyRef(msg, " -100 ")
|
|
if ref == nil {
|
|
t.Fatal("expected non-nil ref")
|
|
}
|
|
if ref.MessageID != "42" || ref.Target != "-100" {
|
|
t.Fatalf("unexpected ref: %+v", ref)
|
|
}
|
|
}
|
|
|
|
func TestPickTelegramPhoto(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
if got := pickTelegramPhoto(nil); got.FileID != "" {
|
|
t.Fatalf("nil should return zero: %+v", got)
|
|
}
|
|
if got := pickTelegramPhoto([]tgbotapi.PhotoSize{}); got.FileID != "" {
|
|
t.Fatalf("empty slice should return zero: %+v", got)
|
|
}
|
|
one := tgbotapi.PhotoSize{FileID: "a", FileSize: 100, Width: 10, Height: 10}
|
|
if got := pickTelegramPhoto([]tgbotapi.PhotoSize{one}); got.FileID != "a" {
|
|
t.Fatalf("single photo should return it: %+v", got)
|
|
}
|
|
photos := []tgbotapi.PhotoSize{
|
|
{FileID: "small", FileSize: 100, Width: 100, Height: 100},
|
|
{FileID: "large", FileSize: 500, Width: 200, Height: 200},
|
|
}
|
|
if got := pickTelegramPhoto(photos); got.FileID != "large" {
|
|
t.Fatalf("should pick largest by size: %+v", got)
|
|
}
|
|
}
|
|
|
|
func TestBuildTelegramMediaGroupInboundMessageAggregatesAttachments(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
bot := &tgbotapi.BotAPI{
|
|
Token: "test",
|
|
Self: tgbotapi.User{ID: 1001, UserName: "memohbot"},
|
|
}
|
|
cfg := channel.ChannelConfig{BotID: "bot-1"}
|
|
first := &tgbotapi.Message{
|
|
MessageID: 101,
|
|
MediaGroupID: "group-1",
|
|
Date: 1710000000,
|
|
Chat: &tgbotapi.Chat{ID: -10001, Type: "group", Title: "G1"},
|
|
From: &tgbotapi.User{ID: 10, UserName: "alice"},
|
|
Photo: []tgbotapi.PhotoSize{
|
|
{FileID: "photo-1", Width: 320, Height: 240, FileSize: 10},
|
|
},
|
|
}
|
|
second := &tgbotapi.Message{
|
|
MessageID: 102,
|
|
MediaGroupID: "group-1",
|
|
Date: 1710000001,
|
|
Chat: &tgbotapi.Chat{ID: -10001, Type: "group", Title: "G1"},
|
|
From: &tgbotapi.User{ID: 10, UserName: "alice"},
|
|
Caption: "album caption",
|
|
Photo: []tgbotapi.PhotoSize{
|
|
{FileID: "photo-2", Width: 640, Height: 480, FileSize: 20},
|
|
},
|
|
}
|
|
|
|
inbound, ok := adapter.buildTelegramMediaGroupInboundMessage(bot, cfg, []*tgbotapi.Message{first, second})
|
|
if !ok {
|
|
t.Fatal("expected grouped inbound message")
|
|
}
|
|
if inbound.Message.Text != "album caption" {
|
|
t.Fatalf("unexpected grouped text: %q", inbound.Message.Text)
|
|
}
|
|
if len(inbound.Message.Attachments) != 2 {
|
|
t.Fatalf("expected 2 attachments, got %d", len(inbound.Message.Attachments))
|
|
}
|
|
if inbound.Message.Attachments[0].PlatformKey != "photo-1" || inbound.Message.Attachments[1].PlatformKey != "photo-2" {
|
|
t.Fatalf("unexpected attachment order: %#v", inbound.Message.Attachments)
|
|
}
|
|
if inbound.Message.ID != "102" {
|
|
t.Fatalf("expected anchor message id 102, got %s", inbound.Message.ID)
|
|
}
|
|
if inbound.ReplyTarget != "-10001" {
|
|
t.Fatalf("unexpected reply target: %q", inbound.ReplyTarget)
|
|
}
|
|
if got := inbound.Metadata["media_group_id"]; got != "group-1" {
|
|
t.Fatalf("unexpected media_group_id metadata: %#v", got)
|
|
}
|
|
if got := inbound.Metadata["media_group_size"]; got != 2 {
|
|
t.Fatalf("unexpected media_group_size metadata: %#v", got)
|
|
}
|
|
}
|
|
|
|
func TestIsTelegramMediaGroupForChat(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
if isTelegramMediaGroupForChat("12:group-a", 12) == false {
|
|
t.Fatal("expected same chat key to match")
|
|
}
|
|
if isTelegramMediaGroupForChat("123:group-a", 12) {
|
|
t.Fatal("expected different chat key to not match")
|
|
}
|
|
if isTelegramMediaGroupForChat("", 12) {
|
|
t.Fatal("expected empty key to not match")
|
|
}
|
|
if isTelegramMediaGroupForChat("12:group-a", 0) {
|
|
t.Fatal("expected zero chat id to not match")
|
|
}
|
|
}
|
|
|
|
func TestTelegramAdapter_Type(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
if adapter.Type() != Type {
|
|
t.Fatalf("Type should return telegram: %s", adapter.Type())
|
|
}
|
|
}
|
|
|
|
func TestTelegramAdapter_OpenStreamEmptyTarget(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
ctx := context.Background()
|
|
cfg := channel.ChannelConfig{}
|
|
_, err := adapter.OpenStream(ctx, cfg, "", channel.StreamOptions{})
|
|
if err == nil {
|
|
t.Fatal("empty target should return error")
|
|
}
|
|
if !strings.Contains(err.Error(), "target") {
|
|
t.Fatalf("expected target in error: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestResolveTelegramSender_SenderChat(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
msg := &tgbotapi.Message{
|
|
SenderChat: &tgbotapi.Chat{ID: 456, UserName: "group", Title: "My Group"},
|
|
}
|
|
externalID, displayName, attrs := resolveTelegramSender(msg)
|
|
if externalID != "456" {
|
|
t.Fatalf("unexpected externalID: %s", externalID)
|
|
}
|
|
if displayName != "My Group" {
|
|
t.Fatalf("unexpected displayName: %s", displayName)
|
|
}
|
|
if attrs["sender_chat_id"] != "456" || attrs["sender_chat_username"] != "group" {
|
|
t.Fatalf("unexpected attrs: %#v", attrs)
|
|
}
|
|
}
|
|
|
|
func TestBuildTelegramAudio(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cfg, err := buildTelegramAudio("@channel", tgbotapi.FileID("f1"))
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if cfg.ChannelUsername != "@channel" {
|
|
t.Fatalf("unexpected channel: %s", cfg.ChannelUsername)
|
|
}
|
|
_, err = buildTelegramAudio("invalid", tgbotapi.FileID("f1"))
|
|
if err == nil {
|
|
t.Fatal("invalid target should return error")
|
|
}
|
|
if !strings.Contains(err.Error(), "chat_id") {
|
|
t.Fatalf("expected chat_id in error: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestBuildTelegramVoice(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cfg, err := buildTelegramVoice("@ch", tgbotapi.FileID("f1"))
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if cfg.ChannelUsername != "@ch" {
|
|
t.Fatalf("unexpected channel: %s", cfg.ChannelUsername)
|
|
}
|
|
_, err = buildTelegramVoice("x", tgbotapi.FileID("f1"))
|
|
if err == nil {
|
|
t.Fatal("invalid target should return error")
|
|
}
|
|
}
|
|
|
|
func TestBuildTelegramVideo(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cfg, err := buildTelegramVideo("@ch", tgbotapi.FileID("f1"))
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if cfg.ChannelUsername != "@ch" {
|
|
t.Fatalf("unexpected channel: %s", cfg.ChannelUsername)
|
|
}
|
|
_, err = buildTelegramVideo("bad", tgbotapi.FileID("f1"))
|
|
if err == nil {
|
|
t.Fatal("invalid target should return error")
|
|
}
|
|
}
|
|
|
|
func TestBuildTelegramAnimation(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
cfg, err := buildTelegramAnimation("@ch", tgbotapi.FileID("f1"))
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if cfg.ChannelUsername != "@ch" {
|
|
t.Fatalf("unexpected channel: %s", cfg.ChannelUsername)
|
|
}
|
|
_, err = buildTelegramAnimation("x", tgbotapi.FileID("f1"))
|
|
if err == nil {
|
|
t.Fatal("invalid target should return error")
|
|
}
|
|
}
|
|
|
|
func TestTelegramAdapter_NormalizeAndResolve(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
norm, err := adapter.NormalizeConfig(map[string]any{"botToken": "t1"})
|
|
if err != nil {
|
|
t.Fatalf("NormalizeConfig: %v", err)
|
|
}
|
|
if norm["botToken"] != "t1" {
|
|
t.Fatalf("unexpected normalized: %#v", norm)
|
|
}
|
|
userNorm, err := adapter.NormalizeUserConfig(map[string]any{"username": "u1"})
|
|
if err != nil {
|
|
t.Fatalf("NormalizeUserConfig: %v", err)
|
|
}
|
|
if userNorm["username"] != "u1" {
|
|
t.Fatalf("unexpected user config: %#v", userNorm)
|
|
}
|
|
if got := adapter.NormalizeTarget("https://t.me/x"); got != "@x" {
|
|
t.Fatalf("NormalizeTarget: %s", got)
|
|
}
|
|
target, err := adapter.ResolveTarget(map[string]any{"chat_id": "123"})
|
|
if err != nil {
|
|
t.Fatalf("ResolveTarget: %v", err)
|
|
}
|
|
if target != "123" {
|
|
t.Fatalf("ResolveTarget: %s", target)
|
|
}
|
|
}
|
|
|
|
func TestIsTelegramMessageNotModified(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
// Exact production error from Telegram API (editMessageText when content unchanged).
|
|
const productionMessageNotModified = "Bad Request: message is not modified: specified new message content and reply markup are exactly the same as a current content and reply markup of the message"
|
|
|
|
tests := []struct {
|
|
name string
|
|
err error
|
|
want bool
|
|
}{
|
|
{"nil", nil, false},
|
|
{"plain error", fmt.Errorf("network error"), false},
|
|
{"other api error", tgbotapi.Error{Code: 400, Message: "Bad Request: chat not found"}, false},
|
|
{"message is not modified", tgbotapi.Error{Code: 400, Message: productionMessageNotModified}, true},
|
|
{"production exact", tgbotapi.Error{Code: 400, Message: productionMessageNotModified}, true},
|
|
{"same text but code 500", tgbotapi.Error{Code: 500, Message: "message is not modified"}, false},
|
|
{"wrapped same", fmt.Errorf("wrapped: %w", tgbotapi.Error{Code: 400, Message: "Bad Request: message is not modified"}), true},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
got := isTelegramMessageNotModified(tt.err)
|
|
if got != tt.want {
|
|
t.Fatalf("isTelegramMessageNotModified() = %v, want %v", got, tt.want)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIsTelegramTooManyRequests(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tests := []struct {
|
|
name string
|
|
err error
|
|
want bool
|
|
}{
|
|
{"nil", nil, false},
|
|
{"429", tgbotapi.Error{Code: 429, Message: "Too Many Requests"}, true},
|
|
{"400", tgbotapi.Error{Code: 400, Message: "Bad Request"}, false},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
got := isTelegramTooManyRequests(tt.err)
|
|
if got != tt.want {
|
|
t.Fatalf("isTelegramTooManyRequests() = %v, want %v", got, tt.want)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestGetTelegramRetryAfter(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tests := []struct {
|
|
name string
|
|
err error
|
|
want time.Duration
|
|
}{
|
|
{"nil", nil, 0},
|
|
{"no retry_after", tgbotapi.Error{Code: 429, Message: "Too Many Requests"}, 0},
|
|
{"retry_after 2", tgbotapi.Error{Code: 429, Message: "Too Many Requests", ResponseParameters: tgbotapi.ResponseParameters{RetryAfter: 2}}, 2 * time.Second},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
got := getTelegramRetryAfter(tt.err)
|
|
if got != tt.want {
|
|
t.Fatalf("getTelegramRetryAfter() = %v, want %v", got, tt.want)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestTruncateTelegramText(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
short := "hello"
|
|
if got := truncateTelegramText(short); got != short {
|
|
t.Fatalf("short text should not be truncated: %q", got)
|
|
}
|
|
|
|
// Exactly at limit.
|
|
exact := strings.Repeat("a", telegramMaxMessageLength)
|
|
if got := truncateTelegramText(exact); got != exact {
|
|
t.Fatalf("exact-limit text should not be truncated, len=%d", len(got))
|
|
}
|
|
|
|
// Over limit with ASCII.
|
|
over := strings.Repeat("a", telegramMaxMessageLength+100)
|
|
got := truncateTelegramText(over)
|
|
if len(got) > telegramMaxMessageLength {
|
|
t.Fatalf("truncated text should be <= %d bytes: got %d", telegramMaxMessageLength, len(got))
|
|
}
|
|
if !strings.HasSuffix(got, "...") {
|
|
t.Fatalf("truncated text should end with '...': %q", got[len(got)-10:])
|
|
}
|
|
|
|
// Over limit with multi-byte characters (Chinese: 3 bytes each).
|
|
multi := strings.Repeat("\u4f60", telegramMaxMessageLength)
|
|
got = truncateTelegramText(multi)
|
|
if len(got) > telegramMaxMessageLength {
|
|
t.Fatalf("truncated multi-byte text should be <= %d bytes: got %d", telegramMaxMessageLength, len(got))
|
|
}
|
|
if !strings.HasSuffix(got, "...") {
|
|
t.Fatal("truncated multi-byte text should end with '...'")
|
|
}
|
|
// Verify no broken runes.
|
|
trimmed := strings.TrimSuffix(got, "...")
|
|
for i := 0; i < len(trimmed); {
|
|
r, size := utf8.DecodeRuneInString(trimmed[i:])
|
|
if r == utf8.RuneError && size == 1 {
|
|
t.Fatalf("truncated text contains invalid UTF-8 at byte %d", i)
|
|
}
|
|
i += size
|
|
}
|
|
}
|
|
|
|
func TestSanitizeTelegramText(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
valid := "hello world"
|
|
if got := sanitizeTelegramText(valid); got != valid {
|
|
t.Fatalf("valid text should not change: %q", got)
|
|
}
|
|
|
|
// Invalid UTF-8 byte sequence.
|
|
invalid := "hello\xff\xfeworld"
|
|
got := sanitizeTelegramText(invalid)
|
|
if !utf8.ValidString(got) {
|
|
t.Fatalf("sanitized text should be valid UTF-8: %q", got)
|
|
}
|
|
if got != "helloworld" {
|
|
t.Fatalf("expected invalid bytes stripped: %q", got)
|
|
}
|
|
}
|
|
|
|
func TestEditTelegramMessageText_429ReturnsError(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
var sendCalls int
|
|
origSend := sendEditForTest
|
|
sendEditForTest = func(_ *tgbotapi.BotAPI, _ tgbotapi.EditMessageTextConfig) error {
|
|
sendCalls++
|
|
return tgbotapi.Error{
|
|
Code: 429,
|
|
Message: "Too Many Requests",
|
|
ResponseParameters: tgbotapi.ResponseParameters{RetryAfter: 1},
|
|
}
|
|
}
|
|
defer func() { sendEditForTest = origSend }()
|
|
|
|
bot := &tgbotapi.BotAPI{Token: "test"}
|
|
err := editTelegramMessageText(bot, 1, 1, "hi", "")
|
|
if err == nil {
|
|
t.Fatal("editTelegramMessageText on 429 should return error for caller to handle")
|
|
}
|
|
if !isTelegramTooManyRequests(err) {
|
|
t.Fatalf("expected 429 error: %v", err)
|
|
}
|
|
if sendCalls != 1 {
|
|
t.Fatalf("send should be called once (no internal retry): got %d", sendCalls)
|
|
}
|
|
}
|
|
|
|
func TestTelegramAdapter_ImplementsProcessingStatusNotifier(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
var _ channel.ProcessingStatusNotifier = adapter
|
|
}
|
|
|
|
func TestProcessingStarted_EmptyParams(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
ctx := context.Background()
|
|
cfg := channel.ChannelConfig{}
|
|
msg := channel.InboundMessage{}
|
|
|
|
handle, err := adapter.ProcessingStarted(ctx, cfg, msg, channel.ProcessingStatusInfo{})
|
|
if err != nil {
|
|
t.Fatalf("empty params should not error: %v", err)
|
|
}
|
|
if handle.Token != "" {
|
|
t.Fatalf("empty params should return empty handle: %q", handle.Token)
|
|
}
|
|
}
|
|
|
|
func TestProcessingCompleted_EmptyHandle(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
ctx := context.Background()
|
|
|
|
err := adapter.ProcessingCompleted(ctx, channel.ChannelConfig{}, channel.InboundMessage{}, channel.ProcessingStatusInfo{}, channel.ProcessingStatusHandle{})
|
|
if err != nil {
|
|
t.Fatalf("empty handle should be no-op: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestProcessingFailed_DelegatesToCompleted(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
adapter := NewTelegramAdapter(nil)
|
|
ctx := context.Background()
|
|
|
|
err := adapter.ProcessingFailed(ctx, channel.ChannelConfig{}, channel.InboundMessage{}, channel.ProcessingStatusInfo{}, channel.ProcessingStatusHandle{}, fmt.Errorf("test"))
|
|
if err != nil {
|
|
t.Fatalf("empty handle should be no-op: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestResolveTelegramFile_PlatformKey(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
att := channel.Attachment{Type: channel.AttachmentImage, PlatformKey: "file_id_123"}
|
|
file, err := resolveTelegramFile("", "file_id_123", "", "", att, "", "", nil)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if _, ok := file.(tgbotapi.FileID); !ok {
|
|
t.Fatalf("expected FileID, got %T", file)
|
|
}
|
|
}
|
|
|
|
func TestResolveTelegramFile_PublicURL(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
att := channel.Attachment{Type: channel.AttachmentImage}
|
|
file, err := resolveTelegramFile("https://example.com/img.png", "", "", "", att, "", "", nil)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if _, ok := file.(tgbotapi.FileURL); !ok {
|
|
t.Fatalf("expected FileURL, got %T", file)
|
|
}
|
|
}
|
|
|
|
func TestResolveTelegramFile_DataURL(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
dataURL := "data:image/png;base64,iVBORw0KGgo="
|
|
att := channel.Attachment{Type: channel.AttachmentImage, Mime: "image/png", Name: "test.png"}
|
|
file, err := resolveTelegramFile("", "", dataURL, "", att, "", "", nil)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
fb, ok := file.(tgbotapi.FileBytes)
|
|
if !ok {
|
|
t.Fatalf("expected FileBytes, got %T", file)
|
|
}
|
|
if fb.Name != "test.png" {
|
|
t.Fatalf("expected name test.png, got %q", fb.Name)
|
|
}
|
|
if len(fb.Bytes) == 0 {
|
|
t.Fatal("expected non-empty bytes")
|
|
}
|
|
}
|
|
|
|
func TestResolveTelegramFile_NoReference(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
att := channel.Attachment{Type: channel.AttachmentImage}
|
|
_, err := resolveTelegramFile("", "", "", "", att, "", "", nil)
|
|
if err == nil {
|
|
t.Fatal("expected error when no reference available")
|
|
}
|
|
}
|
|
|
|
func TestResolveTelegramFile_ContainerPathFallsToBase64(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
dataURL := "data:image/jpeg;base64,/9j/4AAQ"
|
|
att := channel.Attachment{Type: channel.AttachmentImage, Mime: "image/jpeg"}
|
|
file, err := resolveTelegramFile("/data/media/image/a.jpg", "", dataURL, "", att, "", "", nil)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if _, ok := file.(tgbotapi.FileBytes); !ok {
|
|
t.Fatalf("expected FileBytes for container path + base64, got %T", file)
|
|
}
|
|
}
|
|
|
|
type mockAssetOpener struct {
|
|
data []byte
|
|
mime string
|
|
}
|
|
|
|
func (m *mockAssetOpener) Open(_ context.Context, _, _ string) (io.ReadCloser, media.Asset, error) {
|
|
return io.NopCloser(strings.NewReader(string(m.data))), media.Asset{Mime: m.mime}, nil
|
|
}
|
|
|
|
func TestResolveTelegramFile_ContentHash(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
opener := &mockAssetOpener{data: []byte("fake-png-bytes"), mime: "image/png"}
|
|
att := channel.Attachment{Type: channel.AttachmentImage, ContentHash: "asset-123", Name: "output.png"}
|
|
file, err := resolveTelegramFile("", "", "", "", att, "asset-123", "bot-1", opener)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
fb, ok := file.(tgbotapi.FileBytes)
|
|
if !ok {
|
|
t.Fatalf("expected FileBytes from asset reader, got %T", file)
|
|
}
|
|
if fb.Name != "output.png" {
|
|
t.Fatalf("expected name output.png, got %q", fb.Name)
|
|
}
|
|
if string(fb.Bytes) != "fake-png-bytes" {
|
|
t.Fatalf("expected fake-png-bytes, got %q", string(fb.Bytes))
|
|
}
|
|
}
|
|
|
|
func TestResolveTelegramFile_ContentHashPriorityOverURL(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
opener := &mockAssetOpener{data: []byte("from-storage"), mime: "image/jpeg"}
|
|
att := channel.Attachment{Type: channel.AttachmentImage, ContentHash: "a1"}
|
|
file, err := resolveTelegramFile("https://example.com/fallback.jpg", "", "", "", att, "a1", "bot-1", opener)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
fb, ok := file.(tgbotapi.FileBytes)
|
|
if !ok {
|
|
t.Fatalf("expected FileBytes (asset priority over URL), got %T", file)
|
|
}
|
|
if string(fb.Bytes) != "from-storage" {
|
|
t.Fatalf("expected from-storage, got %q", string(fb.Bytes))
|
|
}
|
|
}
|
|
|
|
func TestFileNameFromMime(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
tests := []struct {
|
|
mime string
|
|
fallbackType string
|
|
want string
|
|
}{
|
|
{"image/png", "image", "image.png"},
|
|
{"image/jpeg", "image", "image.jpg"},
|
|
{"image/gif", "image", "image.gif"},
|
|
{"video/mp4", "video", "video.mp4"},
|
|
{"", "image", "image.png"},
|
|
{"application/octet-stream", "", "file.bin"},
|
|
}
|
|
for _, tt := range tests {
|
|
if got := fileNameFromMime(tt.mime, tt.fallbackType); got != tt.want {
|
|
t.Errorf("fileNameFromMime(%q, %q) = %q, want %q", tt.mime, tt.fallbackType, got, tt.want)
|
|
}
|
|
}
|
|
}
|