/* * TamTam Bot API */ package tamtam import ( "bytes" "context" "encoding/json" "fmt" "io" "io/ioutil" "log" "mime/multipart" "net/http" "net/url" "os" "strconv" "time" ) type Api struct { key string version string url *url.URL timeout int pause int logging bool } // New TamTam Api object func New(key string) *Api { u, _ := url.Parse("https://botapi.tamtam.chat/") return &Api{ key: key, url: u, version: "1.0.3", timeout: 30, pause: 1, logging: false, } } func (a *Api) EnableLogging() { a.logging = true } // region Misc methods func (a *Api) GetMe() (*UserWithPhoto, error) { result := new(UserWithPhoto) values := url.Values{} body, err := a.request(http.MethodGet, "me", values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) GetUploadURL(uploadType UploadType) (*UploadEndpoint, error) { result := new(UploadEndpoint) values := url.Values{} values.Set("type", string(uploadType)) body, err := a.request(http.MethodPost, "uploads", values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) UploadMedia(uploadType UploadType, filename string) (*UploadedInfo, error) { bodyBuf := &bytes.Buffer{} bodyWriter := multipart.NewWriter(bodyBuf) // this step is very important fileWriter, err := bodyWriter.CreateFormFile("data", filename) if err != nil { return nil, err } // open file handle fh, err := os.Open(filename) if err != nil { return nil, err } defer fh.Close() //iocopy _, err = io.Copy(fileWriter, fh) if err != nil { return nil, err } if err := bodyWriter.Close(); err != nil { return nil, err } target, err := a.GetUploadURL(uploadType) if err != nil { return nil, err } contentType := bodyWriter.FormDataContentType() resp, err := http.Post(target.Url, contentType, bodyBuf) if err != nil { return nil, err } defer resp.Body.Close() result := new(UploadedInfo) return result, json.NewDecoder(resp.Body).Decode(result) } func (a *Api) GetUpdatesLoop(ctx context.Context, updates chan interface{}) error { for { select { case <-ctx.Done(): return nil case <-time.After(time.Duration(a.pause) * time.Second): var marker int64 for { upds, err := a.getUpdates(50, a.timeout, marker, []string{}) if err != nil { return err } if len(upds.Updates) == 0 { break } for _, u := range upds.Updates { updates <- a.bytesToProperUpdate(u) } marker = upds.Marker } } } } func (a *Api) GetHandler(updates chan interface{}) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { defer r.Body.Close() b, _ := ioutil.ReadAll(r.Body) updates <- a.bytesToProperUpdate(b) } } // endregion // region Chat methods func (a *Api) GetChats(count, marker int64) (*ChatList, error) { result := new(ChatList) values := url.Values{} if count > 0 { values.Set("count", strconv.Itoa(int(count))) } if marker > 0 { values.Set("marker", strconv.Itoa(int(marker))) } body, err := a.request(http.MethodGet, "chats", values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) GetChat(chatID int64) (*Chat, error) { result := new(Chat) values := url.Values{} body, err := a.request(http.MethodGet, fmt.Sprintf("chats/%d", chatID), values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) GetChatMembership(chatID int64) (*ChatMember, error) { result := new(ChatMember) values := url.Values{} body, err := a.request(http.MethodGet, fmt.Sprintf("chats/%d/members/me", chatID), values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) GetChatMembers(chatID, count, marker int64) (*ChatMembersList, error) { result := new(ChatMembersList) values := url.Values{} if count > 0 { values.Set("count", strconv.Itoa(int(count))) } if marker > 0 { values.Set("marker", strconv.Itoa(int(marker))) } body, err := a.request(http.MethodGet, fmt.Sprintf("chats/%d/members", chatID), values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) LeaveChat(chatID int64) (*SimpleQueryResult, error) { result := new(SimpleQueryResult) values := url.Values{} body, err := a.request(http.MethodDelete, fmt.Sprintf("chats/%d/members/me", chatID), values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) EditChat(chatID int64, update *ChatPatch) (*Chat, error) { result := new(Chat) values := url.Values{} body, err := a.request(http.MethodPatch, fmt.Sprintf("chats/%d", chatID), values, update) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) AddMember(chatID int64, users UserIdsList) (*SimpleQueryResult, error) { result := new(SimpleQueryResult) values := url.Values{} body, err := a.request(http.MethodPost, fmt.Sprintf("chats/%d/members", chatID), values, users) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) RemoveMember(chatID int64, userID int64) (*SimpleQueryResult, error) { result := new(SimpleQueryResult) values := url.Values{} values.Set("user_id", strconv.Itoa(int(userID))) body, err := a.request(http.MethodDelete, fmt.Sprintf("chats/%d/members", chatID), values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) SendAction(chatID int64, action SenderAction) (*SimpleQueryResult, error) { result := new(SimpleQueryResult) values := url.Values{} body, err := a.request(http.MethodPost, fmt.Sprintf("chats/%d/actions", chatID), values, ActionRequestBody{Action: action}) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } // endregion // region Message methods func (a *Api) GetMessages(chatID int64, messageIDs []string, from int64, to int64, count int64) (*MessageList, error) { result := new(MessageList) values := url.Values{} if chatID != 0 { values.Set("chat_id", strconv.Itoa(int(chatID))) } if len(messageIDs) > 0 { for _, mid := range messageIDs { values.Add("message_ids", mid) } } if from != 0 { values.Set("from", strconv.Itoa(int(from))) } if to != 0 { values.Set("to", strconv.Itoa(int(to))) } if count > 0 { values.Set("count", strconv.Itoa(int(count))) } body, err := a.request(http.MethodGet, "messages", values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) SendMessage(chatID int64, userID int64, message *NewMessageBody) (*Message, error) { result := new(Message) values := url.Values{} if chatID != 0 { values.Set("chat_id", strconv.Itoa(int(chatID))) } if userID != 0 { values.Set("user_id", strconv.Itoa(int(userID))) } body, err := a.request(http.MethodPost, "messages", values, message) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) EditMessage(messageID int64, message *NewMessageBody) (*SimpleQueryResult, error) { result := new(SimpleQueryResult) values := url.Values{} values.Set("message_id", strconv.Itoa(int(messageID))) body, err := a.request(http.MethodPut, "messages", values, message) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) DeleteMessage(messageID int64) (*SimpleQueryResult, error) { result := new(SimpleQueryResult) values := url.Values{} values.Set("message_id", strconv.Itoa(int(messageID))) body, err := a.request(http.MethodDelete, "messages", values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) AnswerOnCallback(callbackID int64, callback *CallbackAnswer) (*SimpleQueryResult, error) { result := new(SimpleQueryResult) values := url.Values{} values.Set("callback_id", strconv.Itoa(int(callbackID))) body, err := a.request(http.MethodPost, "answers", values, callback) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } // endregion // region Subscriptions func (a *Api) GetSubscriptions() (*GetSubscriptionsResult, error) { result := new(GetSubscriptionsResult) values := url.Values{} body, err := a.request(http.MethodGet, "subscriptions", values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) Subscribe(subscribeURL string, updateTypes []string) (*SimpleQueryResult, error) { subscription := &SubscriptionRequestBody{ Url: subscribeURL, UpdateTypes: updateTypes, Version: a.version, } result := new(SimpleQueryResult) values := url.Values{} body, err := a.request(http.MethodPost, "subscriptions", values, subscription) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } func (a *Api) Unsubscribe(subscriptionURL string) (*SimpleQueryResult, error) { result := new(SimpleQueryResult) values := url.Values{} values.Set("url", subscriptionURL) body, err := a.request(http.MethodDelete, "subscriptions", values, nil) if err != nil { return result, err } defer body.Close() return result, json.NewDecoder(body).Decode(result) } // endregion // region Internal func (a *Api) bytesToProperUpdate(b []byte) interface{} { u := new(Update) _ = json.Unmarshal(b, u) switch u.UpdateType { case UpdateTypeMessageCallback: upd := UpdateMessageCallback{} _ = json.Unmarshal(b, &upd) return upd case UpdateTypeMessageCreated: upd := UpdateMessageCreated{} _ = json.Unmarshal(b, &upd) for _, att := range upd.Message.Body.RawAttachments { upd.Message.Body.Attachments = append(upd.Message.Body.Attachments, a.bytesToProperAttachment(att)) } return upd case UpdateTypeMessageRemoved: upd := UpdateMessageRemoved{} _ = json.Unmarshal(b, &upd) return upd case UpdateTypeMessageEdited: upd := UpdateMessageEdited{} _ = json.Unmarshal(b, &upd) for _, att := range upd.Message.Body.RawAttachments { upd.Message.Body.Attachments = append(upd.Message.Body.Attachments, a.bytesToProperAttachment(att)) } return upd case UpdateTypeMessageRestored: upd := UpdateMessageRestored{} _ = json.Unmarshal(b, &upd) return upd case UpdateTypeBotAdded: upd := UpdateBotAdded{} _ = json.Unmarshal(b, &upd) return upd case UpdateTypeBotRemoved: upd := UpdateBotRemoved{} _ = json.Unmarshal(b, &upd) return upd case UpdateTypeUserAdded: upd := UpdateUserAdded{} _ = json.Unmarshal(b, &upd) return upd case UpdateTypeUserRemoved: upd := UpdateUserRemoved{} _ = json.Unmarshal(b, &upd) return upd case UpdateTypeBotStarted: upd := UpdateBotStarted{} _ = json.Unmarshal(b, &upd) return upd case UpdateTypeChatTitleChanged: upd := UpdateChatTitleChanged{} _ = json.Unmarshal(b, &upd) return upd } return nil } func (a *Api) bytesToProperAttachment(b []byte) interface{} { attachment := new(Attachment) _ = json.Unmarshal(b, attachment) switch attachment.Type { case AttachmentAudio: res := &AudioAttachment{} _ = json.Unmarshal(b, &res) return res case AttachmentContact: res := &ContactAttachment{} _ = json.Unmarshal(b, &res) return res case AttachmentFile: res := &FileAttachment{} _ = json.Unmarshal(b, &res) return res case AttachmentImage: res := &PhotoAttachment{} _ = json.Unmarshal(b, &res) return res case AttachmentKeyboard: res := &InlineKeyboardAttachment{} _ = json.Unmarshal(b, &res) return res case AttachmentLocation: res := &LocationAttachment{} _ = json.Unmarshal(b, &res) return res case AttachmentShare: res := &ShareAttachment{} _ = json.Unmarshal(b, &res) return res case AttachmentSticker: res := &StickerAttachment{} _ = json.Unmarshal(b, &res) return res case AttachmentVideo: res := &VideoAttachment{} _ = json.Unmarshal(b, &res) return res } return attachment } func (a *Api) getUpdates(limit int, timeout int, marker int64, types []string) (*UpdateList, error) { result := new(UpdateList) values := url.Values{} if limit > 0 { values.Set("limit", strconv.Itoa(limit)) } if timeout > 0 { values.Set("timeout", strconv.Itoa(timeout)) } if marker > 0 { values.Set("marker", strconv.Itoa(int(marker))) } if len(types) > 0 { for _, t := range types { values.Add("types", t) } } body, err := a.request(http.MethodGet, "updates", values, nil) if err != nil { return result, err } defer body.Close() jb, _ := ioutil.ReadAll(body) if a.logging { log.Printf("Received: %s", string(jb)) } return result, json.Unmarshal(jb, result) } func (a *Api) request(method, path string, query url.Values, body interface{}) (io.ReadCloser, error) { j, err := json.Marshal(body) if err != nil { return nil, err } return a.requestReader(method, path, query, bytes.NewReader(j)) } func (a *Api) requestReader(method, path string, query url.Values, body io.Reader) (io.ReadCloser, error) { c := http.DefaultClient u := *a.url u.Path = path query.Set("access_token", a.key) query.Set("v", a.version) u.RawQuery = query.Encode() if a.logging { log.Printf("Sent: [%s %s] Query: %#v", method, path, query) } req, err := http.NewRequest(method, u.String(), body) if err != nil { return nil, err } resp, err := c.Do(req) if resp.StatusCode != http.StatusOK { errObj := new(Error) err = json.NewDecoder(resp.Body).Decode(errObj) if err != nil { return nil, err } return nil, fmt.Errorf("code=%s message=%s error=%s", errObj.Code, errObj.Message, errObj.Error) } return resp.Body, err } // endregion