package telebot import ( "errors" "io" "io/ioutil" "net/http" "os" "strconv" "strings" "testing" "time" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) var ( // required to test send and edit methods token = os.Getenv("TELEBOT_SECRET") chatID, _ = strconv.ParseInt(os.Getenv("CHAT_ID"), 10, 64) userID, _ = strconv.ParseInt(os.Getenv("USER_ID"), 10, 64) b, _ = newTestBot() // cached bot instance to avoid getMe method flooding to = &Chat{ID: chatID} // to chat recipient for send and edit methods user = &User{ID: userID} // to user recipient for some special cases ) func defaultSettings() Settings { return Settings{Token: token} } func newTestBot() (*Bot, error) { return NewBot(defaultSettings()) } func TestNewBot(t *testing.T) { var pref Settings _, err := NewBot(pref) assert.Error(t, err) pref.Token = "BAD TOKEN" _, err = NewBot(pref) assert.Error(t, err) pref.URL = "BAD URL" _, err = NewBot(pref) assert.Error(t, err) b, err := NewBot(Settings{Offline: true}) if err != nil { t.Fatal(err) } assert.NotNil(t, b.Me) assert.Equal(t, DefaultApiURL, b.URL) assert.Equal(t, http.DefaultClient, b.client) assert.Equal(t, 100, cap(b.Updates)) pref = defaultSettings() client := &http.Client{Timeout: time.Minute} pref.URL = "http://api.telegram.org" // not https pref.Client = client pref.Poller = &LongPoller{Timeout: time.Second} pref.Updates = 50 pref.ParseMode = ModeHTML pref.Offline = true b, err = NewBot(pref) require.NoError(t, err) assert.Equal(t, client, b.client) assert.Equal(t, pref.URL, b.URL) assert.Equal(t, pref.Poller, b.Poller) assert.Equal(t, 50, cap(b.Updates)) assert.Equal(t, ModeHTML, b.parseMode) } func TestBotHandle(t *testing.T) { if b == nil { t.Skip("Cached bot instance is bad (probably wrong or empty TELEBOT_SECRET)") } b.Handle("/start", func(c Context) error { return nil }) assert.Contains(t, b.handlers, "/start") reply := ReplyButton{Text: "reply"} b.Handle(&reply, func(c Context) error { return nil }) inline := InlineButton{Unique: "inline"} b.Handle(&inline, func(c Context) error { return nil }) btnReply := (&ReplyMarkup{}).Text("btnReply") b.Handle(&btnReply, func(c Context) error { return nil }) btnInline := (&ReplyMarkup{}).Data("", "btnInline") b.Handle(&btnInline, func(c Context) error { return nil }) assert.Contains(t, b.handlers, btnReply.CallbackUnique()) assert.Contains(t, b.handlers, btnInline.CallbackUnique()) assert.Contains(t, b.handlers, reply.CallbackUnique()) assert.Contains(t, b.handlers, inline.CallbackUnique()) } func TestBotStart(t *testing.T) { if token == "" { t.Skip("TELEBOT_SECRET is required") } pref := defaultSettings() pref.Poller = &LongPoller{} b, err := NewBot(pref) if err != nil { t.Fatal(err) } // remove webhook to be sure that bot can poll require.NoError(t, b.RemoveWebhook()) go b.Start() b.Stop() tp := newTestPoller() go func() { tp.updates <- Update{Message: &Message{Text: "/start"}} }() b, err = NewBot(pref) require.NoError(t, err) b.Poller = tp var ok bool b.Handle("/start", func(c Context) error { assert.Equal(t, c.Text(), "/start") tp.done <- struct{}{} ok = true return nil }) go b.Start() <-tp.done b.Stop() assert.True(t, ok) } func TestBotProcessUpdate(t *testing.T) { b, err := NewBot(Settings{Synchronous: true, Offline: true}) if err != nil { t.Fatal(err) } b.Handle("/start", func(c Context) error { assert.Equal(t, "/start", c.Text()) return nil }) b.Handle("hello", func(c Context) error { assert.Equal(t, "hello", c.Text()) return nil }) b.Handle(OnText, func(c Context) error { assert.Equal(t, "text", c.Text()) return nil }) b.Handle(OnPinned, func(c Context) error { assert.NotNil(t, c.Message()) return nil }) b.Handle(OnPhoto, func(c Context) error { assert.NotNil(t, c.Message().Photo) return nil }) b.Handle(OnVoice, func(c Context) error { assert.NotNil(t, c.Message().Voice) return nil }) b.Handle(OnAudio, func(c Context) error { assert.NotNil(t, c.Message().Audio) return nil }) b.Handle(OnAnimation, func(c Context) error { assert.NotNil(t, c.Message().Animation) return nil }) b.Handle(OnDocument, func(c Context) error { assert.NotNil(t, c.Message().Document) return nil }) b.Handle(OnSticker, func(c Context) error { assert.NotNil(t, c.Message().Sticker) return nil }) b.Handle(OnVideo, func(c Context) error { assert.NotNil(t, c.Message().Video) return nil }) b.Handle(OnVideoNote, func(c Context) error { assert.NotNil(t, c.Message().VideoNote) return nil }) b.Handle(OnContact, func(c Context) error { assert.NotNil(t, c.Message().Contact) return nil }) b.Handle(OnLocation, func(c Context) error { assert.NotNil(t, c.Message().Location) return nil }) b.Handle(OnVenue, func(c Context) error { assert.NotNil(t, c.Message().Venue) return nil }) b.Handle(OnInvoice, func(c Context) error { assert.NotNil(t, c.Message().Invoice) return nil }) b.Handle(OnPayment, func(c Context) error { assert.NotNil(t, c.Message().Payment) return nil }) b.Handle(OnAddedToGroup, func(c Context) error { assert.NotNil(t, c.Message().GroupCreated) return nil }) b.Handle(OnUserJoined, func(c Context) error { assert.NotNil(t, c.Message().UserJoined) return nil }) b.Handle(OnUserLeft, func(c Context) error { assert.NotNil(t, c.Message().UserLeft) return nil }) b.Handle(OnNewGroupTitle, func(c Context) error { assert.Equal(t, "title", c.Message().NewGroupTitle) return nil }) b.Handle(OnNewGroupPhoto, func(c Context) error { assert.NotNil(t, c.Message().NewGroupPhoto) return nil }) b.Handle(OnGroupPhotoDeleted, func(c Context) error { assert.True(t, c.Message().GroupPhotoDeleted) return nil }) b.Handle(OnMigration, func(c Context) error { from, to := c.Migration() assert.Equal(t, int64(1), from) assert.Equal(t, int64(2), to) return nil }) b.Handle(OnEdited, func(c Context) error { assert.Equal(t, "edited", c.Message().Text) return nil }) b.Handle(OnChannelPost, func(c Context) error { assert.Equal(t, "post", c.Message().Text) return nil }) b.Handle(OnEditedChannelPost, func(c Context) error { assert.Equal(t, "edited post", c.Message().Text) return nil }) b.Handle(OnCallback, func(c Context) error { if data := c.Callback().Data; data[0] != '\f' { assert.Equal(t, "callback", data) } return nil }) b.Handle("\funique", func(c Context) error { assert.Equal(t, "callback", c.Callback().Data) return nil }) b.Handle(OnQuery, func(c Context) error { assert.Equal(t, "query", c.Data()) return nil }) b.Handle(OnInlineResult, func(c Context) error { assert.Equal(t, "result", c.InlineResult().ResultID) return nil }) b.Handle(OnShipping, func(c Context) error { assert.Equal(t, "shipping", c.ShippingQuery().ID) return nil }) b.Handle(OnCheckout, func(c Context) error { assert.Equal(t, "checkout", c.PreCheckoutQuery().ID) return nil }) b.Handle(OnPoll, func(c Context) error { assert.Equal(t, "poll", c.Poll().ID) return nil }) b.Handle(OnPollAnswer, func(c Context) error { assert.Equal(t, "poll", c.PollAnswer().PollID) return nil }) b.ProcessUpdate(Update{Message: &Message{Text: "/start"}}) b.ProcessUpdate(Update{Message: &Message{Text: "/start@other_bot"}}) b.ProcessUpdate(Update{Message: &Message{Text: "hello"}}) b.ProcessUpdate(Update{Message: &Message{Text: "text"}}) b.ProcessUpdate(Update{Message: &Message{PinnedMessage: &Message{}}}) b.ProcessUpdate(Update{Message: &Message{Photo: &Photo{}}}) b.ProcessUpdate(Update{Message: &Message{Voice: &Voice{}}}) b.ProcessUpdate(Update{Message: &Message{Audio: &Audio{}}}) b.ProcessUpdate(Update{Message: &Message{Animation: &Animation{}}}) b.ProcessUpdate(Update{Message: &Message{Document: &Document{}}}) b.ProcessUpdate(Update{Message: &Message{Sticker: &Sticker{}}}) b.ProcessUpdate(Update{Message: &Message{Video: &Video{}}}) b.ProcessUpdate(Update{Message: &Message{VideoNote: &VideoNote{}}}) b.ProcessUpdate(Update{Message: &Message{Contact: &Contact{}}}) b.ProcessUpdate(Update{Message: &Message{Location: &Location{}}}) b.ProcessUpdate(Update{Message: &Message{Venue: &Venue{}}}) b.ProcessUpdate(Update{Message: &Message{Invoice: &Invoice{}}}) b.ProcessUpdate(Update{Message: &Message{Payment: &Payment{}}}) b.ProcessUpdate(Update{Message: &Message{Dice: &Dice{}}}) b.ProcessUpdate(Update{Message: &Message{GroupCreated: true}}) b.ProcessUpdate(Update{Message: &Message{UserJoined: &User{ID: 1}}}) b.ProcessUpdate(Update{Message: &Message{UsersJoined: []User{{ID: 1}}}}) b.ProcessUpdate(Update{Message: &Message{UserLeft: &User{}}}) b.ProcessUpdate(Update{Message: &Message{NewGroupTitle: "title"}}) b.ProcessUpdate(Update{Message: &Message{NewGroupPhoto: &Photo{}}}) b.ProcessUpdate(Update{Message: &Message{GroupPhotoDeleted: true}}) b.ProcessUpdate(Update{Message: &Message{Chat: &Chat{ID: 1}, MigrateTo: 2}}) b.ProcessUpdate(Update{EditedMessage: &Message{Text: "edited"}}) b.ProcessUpdate(Update{ChannelPost: &Message{Text: "post"}}) b.ProcessUpdate(Update{ChannelPost: &Message{PinnedMessage: &Message{}}}) b.ProcessUpdate(Update{EditedChannelPost: &Message{Text: "edited post"}}) b.ProcessUpdate(Update{Callback: &Callback{MessageID: "inline", Data: "callback"}}) b.ProcessUpdate(Update{Callback: &Callback{Data: "callback"}}) b.ProcessUpdate(Update{Callback: &Callback{Data: "\funique|callback"}}) b.ProcessUpdate(Update{Query: &Query{Text: "query"}}) b.ProcessUpdate(Update{InlineResult: &InlineResult{ResultID: "result"}}) b.ProcessUpdate(Update{ShippingQuery: &ShippingQuery{ID: "shipping"}}) b.ProcessUpdate(Update{PreCheckoutQuery: &PreCheckoutQuery{ID: "checkout"}}) b.ProcessUpdate(Update{Poll: &Poll{ID: "poll"}}) b.ProcessUpdate(Update{PollAnswer: &PollAnswer{PollID: "poll"}}) } func TestBotOnError(t *testing.T) { b, err := NewBot(Settings{Synchronous: true, Offline: true}) if err != nil { t.Fatal(err) } var ok bool b.OnError = func(err error, c Context) { assert.Equal(t, b, c.(*nativeContext).b) assert.NotNil(t, err) ok = true } b.runHandler(func(c Context) error { return errors.New("not nil") }, &nativeContext{b: b}) assert.True(t, ok) } func TestBot(t *testing.T) { if b == nil { t.Skip("Cached bot instance is bad (probably wrong or empty TELEBOT_SECRET)") } if chatID == 0 { t.Skip("CHAT_ID is required for Bot methods test") } _, err := b.Send(to, nil) assert.Equal(t, ErrUnsupportedWhat, err) _, err = b.Edit(&Message{Chat: &Chat{}}, nil) assert.Equal(t, ErrUnsupportedWhat, err) _, err = b.Send(nil, "") assert.Equal(t, ErrBadRecipient, err) _, err = b.Forward(nil, nil) assert.Equal(t, ErrBadRecipient, err) photo := &Photo{ File: FromURL("https://telegra.ph/file/65c5237b040ebf80ec278.jpg"), Caption: t.Name(), } var msg *Message t.Run("Send(what=Sendable)", func(t *testing.T) { msg, err = b.Send(to, photo) require.NoError(t, err) assert.NotNil(t, msg.Photo) assert.Equal(t, photo.Caption, msg.Caption) }) t.Run("SendAlbum()", func(t *testing.T) { _, err = b.SendAlbum(nil, nil) assert.Equal(t, ErrBadRecipient, err) _, err = b.SendAlbum(to, nil) assert.Error(t, err) msgs, err := b.SendAlbum(to, Album{photo, photo}) require.NoError(t, err) assert.Len(t, msgs, 2) assert.NotEmpty(t, msgs[0].AlbumID) }) t.Run("EditCaption()+ParseMode", func(t *testing.T) { b.parseMode = ModeHTML edited, err := b.EditCaption(msg, "new caption with html") require.NoError(t, err) assert.Equal(t, "new caption with html", edited.Caption) assert.Equal(t, EntityBold, edited.CaptionEntities[0].Type) edited, err = b.EditCaption(msg, "*new caption with markdown*", ModeMarkdown) require.NoError(t, err) assert.Equal(t, "new caption with markdown", edited.Caption) assert.Equal(t, EntityBold, edited.CaptionEntities[0].Type) b.parseMode = ModeDefault }) t.Run("Edit(what=InputMedia)", func(t *testing.T) { edited, err := b.Edit(msg, photo) require.NoError(t, err) assert.Equal(t, edited.Photo.UniqueID, photo.UniqueID) resp, err := http.Get("https://telegra.ph/file/274e5eb26f348b10bd8ee.mp4") require.NoError(t, err) defer resp.Body.Close() file, err := ioutil.TempFile("", "") require.NoError(t, err) _, err = io.Copy(file, resp.Body) require.NoError(t, err) animation := &Animation{ File: FromDisk(file.Name()), Caption: t.Name(), FileName: "animation.gif", } msg, err := b.Send(msg.Chat, animation) require.NoError(t, err) if msg.Animation != nil { assert.Equal(t, msg.Animation.FileID, animation.FileID) } else { assert.Equal(t, msg.Document.FileID, animation.FileID) } _, err = b.Edit(edited, animation) require.NoError(t, err) }) t.Run("Edit(what=Animation)", func(t *testing.T) {}) t.Run("Send(what=string)", func(t *testing.T) { msg, err = b.Send(to, t.Name()) require.NoError(t, err) assert.Equal(t, t.Name(), msg.Text) rpl, err := b.Reply(msg, t.Name()) require.NoError(t, err) assert.Equal(t, rpl.Text, msg.Text) assert.NotNil(t, rpl.ReplyTo) assert.Equal(t, rpl.ReplyTo, msg) assert.True(t, rpl.IsReply()) fwd, err := b.Forward(to, msg) require.NoError(t, err) assert.NotNil(t, msg, fwd) assert.True(t, fwd.IsForwarded()) fwd.ID += 1 // nonexistent message _, err = b.Forward(to, fwd) assert.Equal(t, ErrToForwardNotFound, err) }) t.Run("Edit(what=string)", func(t *testing.T) { msg, err = b.Edit(msg, t.Name()) require.NoError(t, err) assert.Equal(t, t.Name(), msg.Text) _, err = b.Edit(msg, msg.Text) assert.Error(t, err) // message is not modified }) t.Run("Edit(what=ReplyMarkup)", func(t *testing.T) { good := &ReplyMarkup{ InlineKeyboard: [][]InlineButton{ {{ Data: "btn", Text: "Hi Telebot!", }}, }, } bad := &ReplyMarkup{ InlineKeyboard: [][]InlineButton{ {{ Data: strings.Repeat("*", 65), Text: "Bad Button", }}, }, } edited, err := b.Edit(msg, good) require.NoError(t, err) assert.Equal(t, edited.ReplyMarkup.InlineKeyboard, good.InlineKeyboard) edited, err = b.EditReplyMarkup(edited, nil) require.NoError(t, err) assert.Nil(t, edited.ReplyMarkup.InlineKeyboard) _, err = b.Edit(edited, bad) assert.Equal(t, ErrButtonDataInvalid, err) }) t.Run("Edit(what=Location)", func(t *testing.T) { loc := &Location{Lat: 42, Lng: 69, LivePeriod: 60} edited, err := b.Send(to, loc) require.NoError(t, err) assert.NotNil(t, edited.Location) loc = &Location{Lat: loc.Lng, Lng: loc.Lat} edited, err = b.Edit(edited, *loc) require.NoError(t, err) assert.NotNil(t, edited.Location) }) // should be the last t.Run("Delete()", func(t *testing.T) { require.NoError(t, b.Delete(msg)) }) t.Run("Notify()", func(t *testing.T) { assert.Equal(t, ErrBadRecipient, b.Notify(nil, Typing)) require.NoError(t, b.Notify(to, Typing)) }) t.Run("Answer()", func(t *testing.T) { assert.Error(t, b.Answer(&Query{}, &QueryResponse{ Results: Results{&ArticleResult{}}, })) }) t.Run("Respond()", func(t *testing.T) { assert.Error(t, b.Respond(&Callback{}, &CallbackResponse{})) }) t.Run("Payments", func(t *testing.T) { assert.NotPanics(t, func() { b.Accept(&PreCheckoutQuery{}) b.Accept(&PreCheckoutQuery{}, "error") }) assert.NotPanics(t, func() { b.Ship(&ShippingQuery{}) b.Ship(&ShippingQuery{}, "error") b.Ship(&ShippingQuery{}, ShippingOption{}, ShippingOption{}) assert.Equal(t, ErrUnsupportedWhat, b.Ship(&ShippingQuery{}, 0)) }) }) t.Run("Commands", func(t *testing.T) { orig := []Command{{ Text: "test", Description: "test command", }} require.NoError(t, b.SetCommands(orig)) cmds, err := b.Commands() require.NoError(t, err) assert.Equal(t, orig, cmds) }) }