You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
231 lines
5.0 KiB
231 lines
5.0 KiB
package telebot |
|
|
|
import ( |
|
"bytes" |
|
"encoding/json" |
|
"fmt" |
|
"io" |
|
"io/ioutil" |
|
"log" |
|
"mime/multipart" |
|
"net/http" |
|
"os" |
|
"strconv" |
|
"strings" |
|
"time" |
|
) |
|
|
|
// Raw lets you call any method of Bot API manually. |
|
// 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) { |
|
url := b.URL + "/bot" + b.Token + "/" + method |
|
|
|
var buf bytes.Buffer |
|
if err := json.NewEncoder(&buf).Encode(payload); err != nil { |
|
return nil, err |
|
} |
|
|
|
resp, err := b.client.Post(url, "application/json", &buf) |
|
if err != nil { |
|
return nil, wrapError(err) |
|
} |
|
resp.Close = true |
|
defer resp.Body.Close() |
|
|
|
data, err := ioutil.ReadAll(resp.Body) |
|
if err != nil { |
|
return nil, wrapError(err) |
|
} |
|
|
|
if b.verbose { |
|
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)) |
|
} |
|
|
|
// returning data as well |
|
return data, extractOk(data) |
|
} |
|
|
|
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, fmt.Errorf("telebot: file for field %s doesn't exist", name) |
|
} |
|
} |
|
|
|
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, files[field].fileName, 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 |
|
} |
|
}() |
|
|
|
url := b.URL + "/bot" + b.Token + "/" + method |
|
|
|
resp, err := b.client.Post(url, writer.FormDataContentType(), pipeReader) |
|
if err != nil { |
|
err = wrapError(err) |
|
pipeReader.CloseWithError(err) |
|
return nil, err |
|
} |
|
resp.Close = true |
|
defer resp.Body.Close() |
|
|
|
if resp.StatusCode == http.StatusInternalServerError { |
|
return nil, ErrInternal |
|
} |
|
|
|
data, err := ioutil.ReadAll(resp.Body) |
|
if err != nil { |
|
return nil, wrapError(err) |
|
} |
|
|
|
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 fmt.Errorf("telebot: file for field %v should be io.ReadCloser or string", field) |
|
} |
|
|
|
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, |
|
} |
|
b.embedSendOptions(params, opt) |
|
|
|
data, err := b.Raw("sendMessage", params) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return extractMessage(data) |
|
} |
|
|
|
func (b *Bot) sendMedia(media Media, params map[string]string, files map[string]File) (*Message, error) { |
|
kind := media.MediaType() |
|
what := "send" + strings.Title(kind) |
|
|
|
if kind == "videoNote" { |
|
kind = "video_note" |
|
} |
|
|
|
sendFiles := map[string]File{kind: *media.MediaFile()} |
|
for k, v := range files { |
|
sendFiles[k] = v |
|
} |
|
|
|
data, err := b.sendFiles(what, sendFiles, params) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
return extractMessage(data) |
|
} |
|
|
|
func (b *Bot) getMe() (*User, error) { |
|
data, err := b.Raw("getMe", nil) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
var resp struct { |
|
Result *User |
|
} |
|
if err := json.Unmarshal(data, &resp); err != nil { |
|
return nil, wrapError(err) |
|
} |
|
return resp.Result, nil |
|
} |
|
|
|
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)), |
|
} |
|
|
|
if limit != 0 { |
|
params["limit"] = strconv.Itoa(limit) |
|
} |
|
if len(allowed) > 0 { |
|
data, _ := json.Marshal(allowed) |
|
params["allowed_updates"] = string(data) |
|
} |
|
|
|
data, err := b.Raw("getUpdates", params) |
|
if err != nil { |
|
return nil, err |
|
} |
|
|
|
var resp struct { |
|
Result []Update |
|
} |
|
if err := json.Unmarshal(data, &resp); err != nil { |
|
return nil, wrapError(err) |
|
} |
|
return resp.Result, nil |
|
}
|
|
|