telebot/api.go

232 lines
5.0 KiB
Go
Raw Normal View History

package telebot
import (
2015-07-02 18:39:39 +00:00
"bytes"
"encoding/json"
2015-07-02 18:39:39 +00:00
"io"
"io/ioutil"
2020-05-31 12:00:30 +00:00
"log"
2015-07-02 18:39:39 +00:00
"mime/multipart"
"net/http"
2015-07-02 18:39:39 +00:00
"os"
2015-06-26 16:12:54 +00:00
"strconv"
"strings"
"time"
"github.com/pkg/errors"
)
// Raw lets you call any method of Bot API manually.
2020-04-06 13:04:25 +00:00
// It also handles API errors, so you only need to unwrap
// result field from json data.
func (b *Bot) Raw(method string, payload interface{}) ([]byte, error) {
2019-09-30 15:57:54 +00:00
url := b.URL + "/bot" + b.Token + "/" + method
var buf bytes.Buffer
if err := json.NewEncoder(&buf).Encode(payload); err != nil {
2020-04-06 13:04:25 +00:00
return nil, err
}
resp, err := b.client.Post(url, "application/json", &buf)
if err != nil {
2020-06-09 20:28:28 +00:00
return nil, wrapError(err)
}
resp.Close = true
defer resp.Body.Close()
2019-09-30 15:57:54 +00:00
2020-04-05 17:23:51 +00:00
data, err := ioutil.ReadAll(resp.Body)
2015-06-26 19:20:08 +00:00
if err != nil {
2020-04-06 13:04:25 +00:00
return nil, wrapError(err)
}
2020-05-31 12:00:30 +00:00
if b.verbose {
2020-06-09 19:26:28 +00:00
body, _ := json.Marshal(payload)
body = bytes.ReplaceAll(body, []byte(`\"`), []byte(`"`))
body = bytes.ReplaceAll(body, []byte(`"{`), []byte(`{`))
body = bytes.ReplaceAll(body, []byte(`}"`), []byte(`}`))
indent := func(b []byte) string {
buf.Reset()
json.Indent(&buf, b, "", "\t")
return buf.String()
}
log.Printf("[verbose] telebot: sent request\n"+
"Method: %v\nParams: %v\nResponse: %v",
method, indent(body), indent(data))
2020-05-31 12:00:30 +00:00
}
2020-04-06 13:04:25 +00:00
// returning data as well
return data, extractOk(data)
}
2020-04-06 13:04:25 +00:00
func (b *Bot) sendFiles(method string, files map[string]File, params map[string]string) ([]byte, error) {
rawFiles := make(map[string]interface{})
for name, f := range files {
switch {
case f.InCloud():
params[name] = f.FileID
case f.FileURL != "":
params[name] = f.FileURL
case f.OnDisk():
rawFiles[name] = f.FileLocal
case f.FileReader != nil:
rawFiles[name] = f.FileReader
default:
return nil, errors.Errorf("telebot: file for field %s doesn't exist", name)
}
}
2015-07-02 18:39:39 +00:00
if len(rawFiles) == 0 {
return b.Raw(method, params)
}
pipeReader, pipeWriter := io.Pipe()
writer := multipart.NewWriter(pipeWriter)
go func() {
defer pipeWriter.Close()
for field, file := range rawFiles {
if err := addFileToWriter(writer, params["file_name"], field, file); err != nil {
pipeWriter.CloseWithError(err)
return
}
}
for field, value := range params {
if err := writer.WriteField(field, value); err != nil {
pipeWriter.CloseWithError(err)
return
}
}
if err := writer.Close(); err != nil {
pipeWriter.CloseWithError(err)
return
}
}()
2015-07-02 18:39:39 +00:00
2019-09-30 15:57:54 +00:00
url := b.URL + "/bot" + b.Token + "/" + method
2015-07-02 18:39:39 +00:00
resp, err := b.client.Post(url, writer.FormDataContentType(), pipeReader)
2015-07-02 18:39:39 +00:00
if err != nil {
err = wrapError(err)
pipeReader.CloseWithError(err)
return nil, err
2015-07-02 18:39:39 +00:00
}
2020-04-06 13:04:25 +00:00
resp.Close = true
2019-12-25 23:03:10 +00:00
defer resp.Body.Close()
2015-07-02 18:39:39 +00:00
if resp.StatusCode == http.StatusInternalServerError {
2020-04-06 13:04:25 +00:00
return nil, ErrInternal
}
2020-04-05 17:23:51 +00:00
data, err := ioutil.ReadAll(resp.Body)
2015-07-02 18:39:39 +00:00
if err != nil {
2020-04-05 17:23:51 +00:00
return nil, wrapError(err)
2015-07-02 18:39:39 +00:00
}
2020-04-06 13:04:25 +00:00
return data, extractOk(data)
}
func addFileToWriter(writer *multipart.Writer, filename, field string, file interface{}) error {
var reader io.Reader
if r, ok := file.(io.Reader); ok {
reader = r
} else if path, ok := file.(string); ok {
f, err := os.Open(path)
if err != nil {
return err
}
defer f.Close()
reader = f
} else {
return errors.Errorf("telebot: file for field %v should be an io.ReadCloser or string", field)
2020-04-06 13:04:25 +00:00
}
part, err := writer.CreateFormFile(field, filename)
if err != nil {
return err
}
_, err = io.Copy(part, reader)
return err
}
func (b *Bot) sendText(to Recipient, text string, opt *SendOptions) (*Message, error) {
params := map[string]string{
"chat_id": to.Recipient(),
"text": text,
}
2020-06-07 20:12:49 +00:00
b.embedSendOptions(params, opt)
2020-04-06 13:04:25 +00:00
data, err := b.Raw("sendMessage", params)
if err != nil {
return nil, err
}
return extractMessage(data)
2015-07-02 18:39:39 +00:00
}
2018-12-15 23:54:13 +00:00
func (b *Bot) sendObject(f *File, what string, params map[string]string, files map[string]File) (*Message, error) {
sendWhat := "send" + strings.Title(what)
if what == "videoNote" {
what = "video_note"
}
2018-12-15 23:54:13 +00:00
sendFiles := map[string]File{what: *f}
for k, v := range files {
sendFiles[k] = v
}
2020-04-06 13:04:25 +00:00
data, err := b.sendFiles(sendWhat, sendFiles, params)
if err != nil {
return nil, err
2015-09-14 03:15:16 +00:00
}
2020-04-06 13:04:25 +00:00
return extractMessage(data)
}
func (b *Bot) getMe() (*User, error) {
2020-04-06 13:04:25 +00:00
data, err := b.Raw("getMe", nil)
if err != nil {
return nil, err
}
2020-04-05 18:29:45 +00:00
var resp struct {
2020-04-06 13:04:25 +00:00
Result *User
}
2020-04-06 13:04:25 +00:00
if err := json.Unmarshal(data, &resp); err != nil {
2020-04-05 18:29:45 +00:00
return nil, wrapError(err)
}
2020-04-05 18:29:45 +00:00
return resp.Result, nil
}
2020-04-26 17:43:17 +00:00
func (b *Bot) getUpdates(offset, limit int, timeout time.Duration, allowed []string) ([]Update, error) {
params := map[string]string{
"offset": strconv.Itoa(offset),
"timeout": strconv.Itoa(int(timeout / time.Second)),
}
2020-04-26 17:43:17 +00:00
if limit != 0 {
params["limit"] = strconv.Itoa(limit)
}
if len(allowed) > 0 {
data, _ := json.Marshal(allowed)
params["allowed_updates"] = string(data)
}
2020-04-06 13:04:25 +00:00
data, err := b.Raw("getUpdates", params)
2015-06-26 16:12:54 +00:00
if err != nil {
2020-04-06 13:04:25 +00:00
return nil, err
2015-06-26 16:12:54 +00:00
}
2020-04-06 13:04:25 +00:00
var resp struct {
Result []Update
}
2020-04-06 13:04:25 +00:00
if err := json.Unmarshal(data, &resp); err != nil {
return nil, wrapError(err)
}
return resp.Result, nil
2020-03-27 23:05:46 +00:00
}