2016-02-08 16:52:13 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
2016-02-20 20:40:24 +00:00
|
|
|
"encoding/base64"
|
2016-02-08 17:05:46 +00:00
|
|
|
"encoding/json"
|
2016-02-09 08:59:25 +00:00
|
|
|
"errors"
|
2016-02-20 20:40:24 +00:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
2016-02-08 16:52:13 +00:00
|
|
|
"log"
|
2016-02-20 20:40:24 +00:00
|
|
|
"net"
|
2016-02-08 16:52:13 +00:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2016-02-09 10:33:38 +00:00
|
|
|
"strconv"
|
2016-02-09 08:59:25 +00:00
|
|
|
|
2016-02-09 14:55:37 +00:00
|
|
|
"github.com/hamcha/clessy/tg"
|
2016-02-08 16:52:13 +00:00
|
|
|
)
|
|
|
|
|
2016-02-19 11:20:13 +00:00
|
|
|
// APIEndpoint is Telegram's current Bot API base url endpoint
|
2016-02-08 16:52:13 +00:00
|
|
|
const APIEndpoint = "https://api.telegram.org/"
|
|
|
|
|
2016-02-19 11:20:13 +00:00
|
|
|
// Telegram is the API client for the Telegram Bot API
|
2016-02-08 16:52:13 +00:00
|
|
|
type Telegram struct {
|
|
|
|
Token string
|
|
|
|
}
|
|
|
|
|
2016-02-19 11:20:13 +00:00
|
|
|
// mkAPI creates a Telegram instance from a Bot API token
|
2016-02-08 16:52:13 +00:00
|
|
|
func mkAPI(token string) *Telegram {
|
|
|
|
tg := new(Telegram)
|
|
|
|
tg.Token = token
|
|
|
|
return tg
|
|
|
|
}
|
|
|
|
|
2016-02-19 11:20:13 +00:00
|
|
|
// SetWebhook sets the webhook address so that Telegram knows where to send updates
|
2016-02-09 10:33:38 +00:00
|
|
|
func (t Telegram) SetWebhook(webhook string) {
|
2016-02-08 16:52:13 +00:00
|
|
|
resp, err := http.PostForm(t.apiURL("setWebhook"), url.Values{"url": {webhook}})
|
2016-02-09 10:33:38 +00:00
|
|
|
if !checkerr("SetWebhook", err) {
|
2016-02-08 16:52:13 +00:00
|
|
|
defer resp.Body.Close()
|
2016-02-08 17:05:46 +00:00
|
|
|
var result tg.APIResponse
|
|
|
|
err = json.NewDecoder(resp.Body).Decode(&result)
|
2016-02-08 16:52:13 +00:00
|
|
|
if err != nil {
|
2016-02-09 10:33:38 +00:00
|
|
|
log.Println("[SetWebhook] Could not read reply: " + err.Error())
|
2016-02-08 17:05:46 +00:00
|
|
|
return
|
2016-02-08 16:52:13 +00:00
|
|
|
}
|
2016-02-09 08:59:25 +00:00
|
|
|
if result.Ok {
|
|
|
|
log.Println("Webhook successfully set!")
|
|
|
|
} else {
|
2016-02-09 10:33:38 +00:00
|
|
|
log.Printf("[SetWebhook] Error setting webhook (errcode %d): %s\n", *(result.ErrCode), *(result.Description))
|
2016-02-09 08:59:25 +00:00
|
|
|
panic(errors.New("Cannot set webhook"))
|
|
|
|
}
|
2016-02-08 16:52:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-19 11:20:13 +00:00
|
|
|
// SendTextMessage sends an HTML-styled text message to a specified chat
|
2016-02-09 10:33:38 +00:00
|
|
|
func (t Telegram) SendTextMessage(data tg.ClientTextMessageData) {
|
|
|
|
postdata := url.Values{
|
|
|
|
"chat_id": {strconv.Itoa(data.ChatID)},
|
|
|
|
"text": {data.Text},
|
|
|
|
"parse_mode": {"HTML"},
|
|
|
|
}
|
|
|
|
if data.ReplyID != nil {
|
|
|
|
postdata["reply_to_message_id"] = []string{strconv.Itoa(*(data.ReplyID))}
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := http.PostForm(t.apiURL("sendMessage"), postdata)
|
|
|
|
checkerr("SendTextMessage", err)
|
|
|
|
}
|
|
|
|
|
2016-02-20 20:40:24 +00:00
|
|
|
// GetFile sends a "getFile" API call to Telegram's servers and fetches the file
|
|
|
|
// specified afterward. The file will be then send back to the client that requested it
|
|
|
|
// with the specified callback id.
|
|
|
|
func (t Telegram) GetFile(data tg.FileRequestData, client net.Conn, callback int) {
|
|
|
|
fail := func(msg string) {
|
|
|
|
errmsg, _ := json.Marshal(tg.BrokerUpdate{
|
|
|
|
Type: tg.BError,
|
|
|
|
Error: &msg,
|
|
|
|
Callback: &callback,
|
|
|
|
})
|
|
|
|
fmt.Fprintln(client, string(errmsg))
|
|
|
|
}
|
|
|
|
|
|
|
|
postdata := url.Values{
|
|
|
|
"file_id": {data.FileID},
|
|
|
|
}
|
|
|
|
resp, err := http.PostForm(t.apiURL("getFile"), postdata)
|
|
|
|
if checkerr("GetFile/post", err) {
|
|
|
|
fail("Server didn't like my request")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var filespecs tg.APIFile
|
|
|
|
err = json.NewDecoder(resp.Body).Decode(&filespecs)
|
|
|
|
if checkerr("GetFile/json.Decode", err) {
|
|
|
|
fail("Server sent garbage (or error)")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2016-02-20 20:46:55 +00:00
|
|
|
fmt.Println(filespecs)
|
|
|
|
|
|
|
|
path := "https://api.telegram.org/file/bot" + t.Token + "/" + *(filespecs.Path)
|
2016-02-20 20:40:24 +00:00
|
|
|
fileresp, err := http.Get(path)
|
|
|
|
if checkerr("GetFile/get", err) {
|
|
|
|
fail("Could not retrieve file from Telegram's servers")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer fileresp.Body.Close()
|
|
|
|
|
|
|
|
rawdata, err := ioutil.ReadAll(fileresp.Body)
|
|
|
|
if checkerr("GetFile/ioutil.ReadAll", err) {
|
|
|
|
fail("Could not read file data")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
rawlen := len(rawdata)
|
|
|
|
if rawlen != *filespecs.Size {
|
|
|
|
// ???
|
|
|
|
log.Printf("[GetFile] WARN ?? Downloaded file does not match provided filesize: %d != %d\n", rawlen, *filespecs.Size)
|
|
|
|
}
|
|
|
|
b64data := base64.StdEncoding.EncodeToString(rawdata)
|
|
|
|
|
|
|
|
clientmsg, err := json.Marshal(tg.BrokerUpdate{
|
|
|
|
Type: tg.BFile,
|
|
|
|
Bytes: &b64data,
|
|
|
|
Callback: &callback,
|
|
|
|
})
|
|
|
|
if checkerr("GetFile/json.Marshal", err) {
|
|
|
|
fail("Could not serialize reply JSON")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Fprintln(client, string(clientmsg))
|
|
|
|
}
|
|
|
|
|
2016-02-08 16:57:27 +00:00
|
|
|
func (t Telegram) apiURL(method string) string {
|
2016-02-08 16:52:13 +00:00
|
|
|
return APIEndpoint + "bot" + t.Token + "/" + method
|
|
|
|
}
|
|
|
|
|
2016-02-08 16:57:27 +00:00
|
|
|
func checkerr(method string, err error) bool {
|
2016-02-08 16:52:13 +00:00
|
|
|
if err != nil {
|
2016-02-20 20:40:24 +00:00
|
|
|
log.Printf("[%s] Error: %s\n", method, err.Error())
|
2016-02-08 16:52:13 +00:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|