transmission-telegram/transmission-telegram.go

977 lines
25 KiB
Go
Raw Normal View History

2016-06-19 08:07:00 +03:00
package main
import (
2016-06-20 05:09:41 +03:00
"bytes"
2016-06-19 08:07:00 +03:00
"flag"
"fmt"
"os"
2016-06-21 18:42:30 +03:00
"regexp"
"strconv"
2016-06-19 08:07:00 +03:00
"strings"
2016-06-25 07:44:34 +03:00
"time"
2016-06-20 05:09:41 +03:00
"unicode/utf8"
2016-06-19 08:07:00 +03:00
2016-06-19 22:02:15 +03:00
"gopkg.in/telegram-bot-api.v4"
2016-06-25 07:44:34 +03:00
"github.com/dustin/go-humanize"
2016-06-19 08:07:00 +03:00
"github.com/pyed/transmission"
)
2016-06-19 22:33:57 +03:00
const VERSION = "0.0"
2016-06-19 08:07:00 +03:00
var (
2016-06-19 22:02:15 +03:00
// flags
2016-06-19 22:33:57 +03:00
BotToken string
Master string
RpcUrl string
Username string
Password string
2016-06-19 08:07:00 +03:00
2016-06-19 22:02:15 +03:00
// transmission
2016-07-01 06:14:38 +03:00
Client *transmission.TransmissionClient
2016-06-19 22:02:15 +03:00
// telegram
Bot *tgbotapi.BotAPI
Updates <-chan tgbotapi.Update
2016-06-19 08:07:00 +03:00
)
2016-06-19 22:02:15 +03:00
// init flags
2016-06-19 08:07:00 +03:00
func init() {
// define arguments and parse them.
2016-06-19 22:33:57 +03:00
flag.StringVar(&BotToken, "token", "", "Telegram bot token")
flag.StringVar(&Master, "master", "", "Your telegram handler, So the bot will only respond to you")
flag.StringVar(&RpcUrl, "url", "http://localhost:9091/transmission/rpc", "Transmission RPC URL")
flag.StringVar(&Username, "username", "", "Transmission username")
flag.StringVar(&Password, "password", "", "Transmission password")
2016-06-19 08:07:00 +03:00
// set the usage message
flag.Usage = func() {
2016-07-01 06:14:38 +03:00
fmt.Fprint(os.Stderr, "Usage: transmission-bot -token=<TOKEN> -master=<@tuser> -url=[http://] -username=[user] -password=[pass]\n\n")
2016-06-19 08:07:00 +03:00
flag.PrintDefaults()
}
flag.Parse()
// make sure that we have the two madatory arguments: telegram token & master's handler.
2016-06-19 22:33:57 +03:00
if BotToken == "" ||
Master == "" {
2016-06-19 08:07:00 +03:00
fmt.Fprintf(os.Stderr, "Error: Mandatory argument missing!\n\n")
flag.Usage()
os.Exit(1)
}
2016-06-20 05:09:41 +03:00
// make sure that the handler doesn't contain @
if strings.Contains(Master, "@") {
Master = strings.Replace(Master, "@", "", -1)
}
2016-06-19 22:02:15 +03:00
}
2016-06-19 08:07:00 +03:00
2016-06-19 22:02:15 +03:00
// init transmission
func init() {
2016-06-19 08:07:00 +03:00
// set transmission.Config, needed to establish a connection with transmission
2016-06-21 18:42:30 +03:00
// conf := transmission.Config{
// Address: RpcUrl,
// User: Username,
// Password: Password,
// }
2016-06-19 08:07:00 +03:00
// transmission.New() never returns an error, we will ignore it and test with client.Session.Update()
2016-06-21 18:42:30 +03:00
Client = transmission.New(RpcUrl, Username, Password)
// if err := Client.Session.Update(); err != nil {
// try to predict the error message, as it vague coming from pyed/transmission
// if strings.HasPrefix(err.Error(), "invalid character") { // means the user or the pass is wrong.
// fmt.Fprintf(os.Stderr, "Transmission's Username or Password is wrong.\n\n")
// } else { // any other error is probaby because of the URL
// fmt.Fprintf(os.Stderr, "Error: Couldn't connect to: %s\n", RpcUrl)
// fmt.Fprintf(os.Stderr, "Make sure to pass the right full RPC URL e.g. http://localhost:9091/transmission/rpc\n\n")
// }
// // send the vague error message too
// fmt.Fprintf(os.Stderr, "JSONError: %s\n", err)
// os.Exit(1)
// }
2016-06-19 22:02:15 +03:00
}
// init telegram
func init() {
// authorize using the token
var err error
2016-06-19 22:33:57 +03:00
Bot, err = tgbotapi.NewBotAPI(BotToken)
2016-06-19 22:02:15 +03:00
if err != nil {
fmt.Fprintf(os.Stderr, "Telegram Error: %s\n", err)
os.Exit(1)
}
2016-06-20 05:09:41 +03:00
fmt.Fprintf(os.Stdout, "Authorized on %s", Bot.Self.UserName)
2016-06-19 22:02:15 +03:00
// get a channel and sign it to 'Updates'
u := tgbotapi.NewUpdate(0)
u.Timeout = 60
Updates, err = Bot.GetUpdatesChan(u)
if err != nil {
fmt.Fprintf(os.Stderr, "Telegram Error: %s\n", err)
2016-06-19 08:07:00 +03:00
os.Exit(1)
}
}
2016-06-19 22:33:57 +03:00
func main() {
for update := range Updates {
// ignore anyone other than 'master'
2016-06-20 05:09:41 +03:00
if strings.ToLower(update.Message.From.UserName) != strings.ToLower(Master) {
2016-06-19 22:33:57 +03:00
continue
}
// ignore edited messages
if update.Message == nil {
continue
}
// tokenize the update
tokens := strings.Split(update.Message.Text, " ")
command := strings.ToLower(tokens[0])
switch command {
case "list", "/list":
// list torrents
2016-06-21 18:42:30 +03:00
// TODO take argument as tracker and list those only
2016-06-20 05:09:41 +03:00
go list(&update)
2016-06-19 22:33:57 +03:00
case "downs", "/downs":
// list downloading
2016-06-21 18:42:30 +03:00
go downs(&update)
2016-06-30 02:03:59 +03:00
case "seeding", "/seeding":
// list seeding
go seeding(&update)
case "paused", "/paused":
// list puased torrents
go paused(&update)
case "checking", "/checking":
// list verifying torrents
go checking(&update)
2016-06-19 22:33:57 +03:00
case "active", "/active":
// list active torrents
2016-06-21 18:42:30 +03:00
go active(&update)
2016-06-19 22:33:57 +03:00
case "errors", "/errors":
// list torrents with errors
2016-06-21 18:42:30 +03:00
go errors(&update)
2016-07-01 06:14:38 +03:00
case "sort", "/sort":
// sort torrents
go sort(&update, tokens[1:])
2016-06-19 22:33:57 +03:00
case "trackers", "/trackers":
// list trackers
2016-06-21 18:42:30 +03:00
go trackers(&update)
2016-06-19 22:33:57 +03:00
case "add", "/add":
// takes url to a torrent to add it
go add(&update, tokens[1:])
2016-06-19 22:33:57 +03:00
case "search", "/search":
// search for a torrent
2016-06-21 18:42:30 +03:00
go search(&update, tokens[1:])
2016-06-19 22:33:57 +03:00
case "latest", "/latest":
// get the latest torrents
2016-06-21 18:42:30 +03:00
go latest(&update, tokens[1:])
2016-06-19 22:33:57 +03:00
case "info", "/info":
// gets info on specific torrent
2016-06-25 07:44:34 +03:00
go info(&update, tokens[1:])
2016-06-19 22:33:57 +03:00
case "stop", "/stop":
// stop one torrent or more
2016-06-21 18:42:30 +03:00
go stop(&update, tokens[1:])
2016-06-19 22:33:57 +03:00
case "start", "/start":
// starts one torrent or more
2016-06-21 18:42:30 +03:00
go start(&update, tokens[1:])
2016-06-27 18:40:09 +03:00
case "check", "/check":
// verify a torrent or torrents
go check(&update, tokens[1:])
2016-06-21 18:42:30 +03:00
2016-06-19 22:33:57 +03:00
case "stats", "/stats":
// print transmission stats
2016-06-25 07:44:34 +03:00
go stats(&update)
2016-06-19 22:33:57 +03:00
case "speed", "/speed":
// print current download and upload speeds
2016-06-25 07:44:34 +03:00
go speed(&update)
2016-06-30 02:03:59 +03:00
case "count", "/count":
// sends current torrents count per status
go count(&update)
2016-06-19 22:33:57 +03:00
case "del", "/del":
// deletes a torrent but keep its data
2016-06-25 07:44:34 +03:00
go del(&update, tokens[1:])
2016-06-19 22:33:57 +03:00
case "deldata", "/deldata":
// deletes a torrents and its data
2016-06-25 07:44:34 +03:00
go deldata(&update, tokens[1:])
2016-06-19 22:33:57 +03:00
case "help", "/help":
// prints a help message
case "version", "/version":
// print transmission and transmission-telegram versions
2016-06-27 20:23:26 +03:00
go version(&update)
case "":
// might be a file received
go receiveTorrent(&update)
2016-06-19 22:33:57 +03:00
default:
// no such command, try help
2016-06-30 02:03:59 +03:00
go send("no such command, try /help", update.Message.Chat.ID)
2016-06-19 22:33:57 +03:00
}
}
}
2016-06-20 05:09:41 +03:00
// list will form and send a list of all the torrents
func list(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("list: "+err.Error(), ud.Message.Chat.ID)
return
}
buf := new(bytes.Buffer)
for i := range torrents {
buf.WriteString(fmt.Sprintf("<%d> %s\n", torrents[i].ID, torrents[i].Name))
}
if buf.Len() == 0 {
send("No torrents exist!", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
2016-06-30 02:03:59 +03:00
// downs will send the names of torrents with status 'Downloading' or in queue to
2016-06-21 18:42:30 +03:00
func downs(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("downs: "+err.Error(), ud.Message.Chat.ID)
return
}
buf := new(bytes.Buffer)
for i := range torrents {
// Downloading or in queue to download
2016-06-30 02:03:59 +03:00
if torrents[i].Status == transmission.StatusDownloading ||
torrents[i].Status == transmission.StatusDownloadPending {
buf.WriteString(fmt.Sprintf("<%d> %s\n%s (%.1f%%) ↓ %s ↑ %s R:%s\n\n",
torrents[i].ID, torrents[i].Name, torrents[i].TorrentStatus(), torrents[i].PercentDone*100,
humanize.Bytes(torrents[i].RateDownload), humanize.Bytes(torrents[i].RateUpload), torrents[i].Ratio()))
2016-06-21 18:42:30 +03:00
}
}
if buf.Len() == 0 {
send("No downloads", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
2016-06-30 02:03:59 +03:00
// seeding will send the names of the torrents with the status 'Seeding' or in the queue to
func seeding(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("seeding: "+err.Error(), ud.Message.Chat.ID)
return
}
buf := new(bytes.Buffer)
for i := range torrents {
if torrents[i].Status == transmission.StatusSeeding ||
torrents[i].Status == transmission.StatusSeedPending {
buf.WriteString(fmt.Sprintf("<%d> %s\n%s DL: %s UL: %s R:%s\n\n",
torrents[i].ID, torrents[i].Name, torrents[i].TorrentStatus(),
humanize.Bytes(torrents[i].DownloadedEver), humanize.Bytes(torrents[i].UploadedEver),
torrents[i].Ratio()))
}
}
if buf.Len() == 0 {
send("No torrents seeding", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
// paused will send the names of the torrents with status 'Paused'
func paused(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("paused: "+err.Error(), ud.Message.Chat.ID)
return
}
buf := new(bytes.Buffer)
for i := range torrents {
if torrents[i].Status == transmission.StatusStopped {
buf.WriteString(fmt.Sprintf("<%d> %s\n%s (%.1f%%) DL: %s UL: %s R:%s\n\n",
torrents[i].ID, torrents[i].Name, torrents[i].TorrentStatus(),
torrents[i].PercentDone*100, humanize.Bytes(torrents[i].DownloadedEver),
humanize.Bytes(torrents[i].UploadedEver), torrents[i].Ratio()))
}
}
if buf.Len() == 0 {
send("No paused torrents", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
// checking will send the names of torrents with the status 'verifying' or in the queue to
func checking(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("checking: "+err.Error(), ud.Message.Chat.ID)
return
}
buf := new(bytes.Buffer)
for i := range torrents {
if torrents[i].Status == transmission.StatusChecking ||
torrents[i].Status == transmission.StatusCheckPending {
buf.WriteString(fmt.Sprintf("<%d> %s\n%s (%.1f%%)\n\n",
torrents[i].ID, torrents[i].Name, torrents[i].TorrentStatus(),
torrents[i].PercentDone*100))
}
}
if buf.Len() == 0 {
send("No torrents verifying", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
2016-06-21 18:42:30 +03:00
// active will send torrents that are actively uploading
func active(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("active: "+err.Error(), ud.Message.Chat.ID)
return
}
buf := new(bytes.Buffer)
for i := range torrents {
if torrents[i].RateUpload > 0 {
2016-06-30 02:03:59 +03:00
buf.WriteString(fmt.Sprintf("<%d> %s\n%s (%.1f%%) ↓ %s ↑ %s R:%s\n\n",
torrents[i].ID, torrents[i].Name, torrents[i].TorrentStatus(),
torrents[i].PercentDone*100, humanize.Bytes(torrents[i].RateDownload),
humanize.Bytes(torrents[i].RateUpload), torrents[i].Ratio()))
2016-06-21 18:42:30 +03:00
}
}
if buf.Len() == 0 {
send("No active torrents", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
// errors will send torrents with errors
func errors(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("errors: "+err.Error(), ud.Message.Chat.ID)
return
}
buf := new(bytes.Buffer)
for i := range torrents {
if torrents[i].Error != 0 {
buf.WriteString(fmt.Sprintf("<%d> %s\n%s\n",
torrents[i].ID, torrents[i].Name, torrents[i].ErrorString))
}
}
if buf.Len() == 0 {
send("No errors", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
2016-07-01 06:14:38 +03:00
// sort changes torrents sorting
func sort(ud *tgbotapi.Update, tokens []string) {
if len(tokens) == 0 {
send(`sort takes one of (id, name, age, size, progress, download, upload, ratio)
optionally start with (rev) for reversed order
e.g. "sort rev size" to get smallest torrents first.`, ud.Message.Chat.ID)
return
}
var reversed bool
if strings.ToLower(tokens[0]) == "rev" {
reversed = true
tokens = tokens[1:]
}
switch strings.ToLower(tokens[0]) {
case "id":
if reversed {
Client.SetSort(transmission.SortRevID)
break
}
Client.SetSort(transmission.SortID)
case "name":
if reversed {
Client.SetSort(transmission.SortRevName)
break
}
Client.SetSort(transmission.SortName)
case "age":
if reversed {
Client.SetSort(transmission.SortRevAge)
break
}
Client.SetSort(transmission.SortAge)
case "size":
if reversed {
Client.SetSort(transmission.SortRevSize)
break
}
Client.SetSort(transmission.SortSize)
case "progress":
if reversed {
Client.SetSort(transmission.SortRevProgress)
break
}
Client.SetSort(transmission.SortProgress)
case "download":
if reversed {
Client.SetSort(transmission.SortRevDownloaded)
break
}
Client.SetSort(transmission.SortDownloaded)
case "upload":
if reversed {
Client.SetSort(transmission.SortRevUploaded)
break
}
Client.SetSort(transmission.SortUploaded)
case "ratio":
if reversed {
Client.SetSort(transmission.SortRevRatio)
break
}
Client.SetSort(transmission.SortRatio)
default:
send("unkown sorting method", ud.Message.Chat.ID)
return
}
if reversed {
send("sort: reversed "+tokens[0], ud.Message.Chat.ID)
return
}
send("sort: "+tokens[0], ud.Message.Chat.ID)
}
2016-06-21 18:42:30 +03:00
var trackerRegex = regexp.MustCompile(`https?://([^:/]*)`)
// trackers will send a list of trackers and how many torrents each one has
func trackers(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("trackers: "+err.Error(), ud.Message.Chat.ID)
return
}
trackers := make(map[string]int)
for i := range torrents {
for _, tracker := range torrents[i].Trackers {
sm := trackerRegex.FindSubmatch([]byte(tracker.Announce))
if len(sm) > 1 {
currentTracker := string(sm[1])
n, ok := trackers[currentTracker]
if !ok {
trackers[currentTracker] = 1
continue
}
trackers[currentTracker] = n + 1
}
}
}
buf := new(bytes.Buffer)
for k, v := range trackers {
buf.WriteString(fmt.Sprintf("%d - %s\n", v, k))
}
if buf.Len() == 0 {
send("No trackers!", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
// add takes an URL to a .torrent file to add it to transmission
func add(ud *tgbotapi.Update, tokens []string) {
if len(tokens) == 0 {
send("add: needs atleast one URL", ud.Message.Chat.ID)
return
}
// loop over the URL/s and add them
for _, url := range tokens {
cmd := transmission.NewAddCmdByURL(url)
torrent, err := Client.ExecuteAddCommand(cmd)
if err != nil {
send("add: "+err.Error(), ud.Message.Chat.ID)
continue
}
// check if torrent.Name is empty, then an error happened
if torrent.Name == "" {
send("add: error adding "+url, ud.Message.Chat.ID)
continue
}
send("Added: "+torrent.Name, ud.Message.Chat.ID)
}
}
// receiveTorrent gets an update that potentially has a .torrent file to add
func receiveTorrent(ud *tgbotapi.Update) {
if ud.Message.Document.FileID == "" {
return // has no document
}
// get the file ID and make the config
fconfig := tgbotapi.FileConfig{ud.Message.Document.FileID}
file, err := Bot.GetFile(fconfig)
if err != nil {
send("receiver: "+err.Error(), ud.Message.Chat.ID)
return
}
// add by file URL
add(ud, []string{file.Link(BotToken)})
}
2016-06-21 18:42:30 +03:00
// search takes a query and returns torrents with match
func search(ud *tgbotapi.Update, tokens []string) {
// make sure that we got a query
if len(tokens) == 0 {
send("search: needs an argument", ud.Message.Chat.ID)
return
}
query := strings.Join(tokens, " ")
// "(?i)" for case insensitivity
regx, err := regexp.Compile("(?i)" + query)
if err != nil {
send("search: "+err.Error(), ud.Message.Chat.ID)
return
}
torrents, err := Client.GetTorrents()
if err != nil {
send("search: "+err.Error(), ud.Message.Chat.ID)
return
}
buf := new(bytes.Buffer)
for i := range torrents {
if regx.MatchString(torrents[i].Name) {
buf.WriteString(fmt.Sprintf("<%d> %s\n", torrents[i].ID, torrents[i].Name))
}
}
if buf.Len() == 0 {
send("No matches!", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
// latest takes n and returns the latest n torrents
func latest(ud *tgbotapi.Update, tokens []string) {
var (
n = 5 // default to 5
err error
)
if len(tokens) > 0 {
n, err = strconv.Atoi(tokens[0])
if err != nil {
send("latest: argument must be a number", ud.Message.Chat.ID)
return
}
}
torrents, err := Client.GetTorrents()
if err != nil {
send("latest: "+err.Error(), ud.Message.Chat.ID)
return
}
// make sure that we stay in the boundaries
torrentsLen := len(torrents)
if n <= 0 || n > torrentsLen {
n = torrentsLen
}
2016-07-01 06:14:38 +03:00
// sort by age, and set reverse to true to get the latest first
torrents.SortAge(true)
2016-06-21 18:42:30 +03:00
buf := new(bytes.Buffer)
for i := range torrents[:n] {
buf.WriteString(fmt.Sprintf("<%d> %s\n", torrents[i].ID, torrents[i].Name))
}
if buf.Len() == 0 {
send("No torrents", ud.Message.Chat.ID)
return
}
send(buf.String(), ud.Message.Chat.ID)
}
2016-06-25 07:44:34 +03:00
// info takes an id of a torrent and returns some info about it
func info(ud *tgbotapi.Update, tokens []string) {
if len(tokens) == 0 {
send("info: needs a torrent ID number", ud.Message.Chat.ID)
return
}
// try to read the id
num, err := strconv.Atoi(tokens[0])
if err != nil {
send(fmt.Sprintf("info: %s is not a number", tokens[0]), ud.Message.Chat.ID)
return
}
// get the torrent
torrent, err := Client.GetTorrent(num)
if err != nil {
send(fmt.Sprintf("info: Can't find a torrent with an ID of %d", num), ud.Message.Chat.ID)
return
}
// format the info
2016-06-30 02:03:59 +03:00
info := fmt.Sprintf("<%d> %s\n%s\t%s of %s (%.1f%%)\t↓ %s ↑ %s R:%s\nUP: %s DL: %s Added: %s ETA: %s\nTracker: %s",
2016-06-25 07:44:34 +03:00
torrent.ID, torrent.Name, torrent.TorrentStatus(), humanize.Bytes(torrent.DownloadedEver), humanize.Bytes(torrent.SizeWhenDone),
2016-06-30 02:03:59 +03:00
torrent.PercentDone*100, humanize.Bytes(torrent.RateDownload), humanize.Bytes(torrent.RateUpload), torrent.Ratio(),
2016-06-25 07:44:34 +03:00
humanize.Bytes(torrent.UploadedEver), humanize.Bytes(torrent.DownloadedEver), time.Unix(torrent.AddedDate, 0).Format(time.Stamp),
2016-06-30 02:03:59 +03:00
torrent.ETA(), torrent.Trackers[0].Announce)
2016-06-25 07:44:34 +03:00
// trackers should be fixed
// send it
send(info, ud.Message.Chat.ID)
}
2016-06-27 18:40:09 +03:00
// stop takes id[s] of torrent[s] or 'all' to stop them
2016-06-21 18:42:30 +03:00
func stop(ud *tgbotapi.Update, tokens []string) {
// make sure that we got at least one argument
if len(tokens) == 0 {
send("stop: needs an argument", ud.Message.Chat.ID)
return
}
2016-06-27 18:40:09 +03:00
// if the first argument is 'all' then stop all torrents
if tokens[0] == "all" {
if err := Client.StopAll(); err != nil {
send("stop: error occurred while stopping some torrents", ud.Message.Chat.ID)
return
}
send("stopped all torrents", ud.Message.Chat.ID)
return
}
2016-06-21 18:42:30 +03:00
for _, id := range tokens {
num, err := strconv.Atoi(id)
if err != nil {
send(fmt.Sprintf("stop: %s is not a number", id), ud.Message.Chat.ID)
continue
}
status, err := Client.StopTorrent(num)
if err != nil {
send("stop: "+err.Error(), ud.Message.Chat.ID)
continue
}
torrent, err := Client.GetTorrent(num)
if err != nil {
send(fmt.Sprintf("[fail] stop: No torrent with an ID of %d", num), ud.Message.Chat.ID)
return
}
send(fmt.Sprintf("[%s] stop: %s", status, torrent.Name), ud.Message.Chat.ID)
}
}
2016-06-27 18:40:09 +03:00
// start takes id[s] of torrent[s] or 'all' to start them
2016-06-21 18:42:30 +03:00
func start(ud *tgbotapi.Update, tokens []string) {
// make sure that we got at least one argument
if len(tokens) == 0 {
send("start: needs an argument", ud.Message.Chat.ID)
return
}
2016-06-27 18:40:09 +03:00
// if the first argument is 'all' then start all torrents
if tokens[0] == "all" {
if err := Client.StartAll(); err != nil {
send("start: error occurred while starting some torrents", ud.Message.Chat.ID)
return
}
send("started all torrents", ud.Message.Chat.ID)
return
}
2016-06-21 18:42:30 +03:00
for _, id := range tokens {
num, err := strconv.Atoi(id)
if err != nil {
send(fmt.Sprintf("start: %s is not a number", id), ud.Message.Chat.ID)
continue
}
status, err := Client.StartTorrent(num)
if err != nil {
send("stop: "+err.Error(), ud.Message.Chat.ID)
continue
}
torrent, err := Client.GetTorrent(num)
if err != nil {
send(fmt.Sprintf("[fail] start: No torrent with an ID of %d", num), ud.Message.Chat.ID)
return
}
send(fmt.Sprintf("[%s] start: %s", status, torrent.Name), ud.Message.Chat.ID)
}
}
2016-06-27 18:40:09 +03:00
// check takes id[s] of torrent[s] or 'all' to verify them
func check(ud *tgbotapi.Update, tokens []string) {
// make sure that we got at least one argument
if len(tokens) == 0 {
send("check: needs an argument", ud.Message.Chat.ID)
2016-06-21 18:42:30 +03:00
return
}
2016-06-27 18:40:09 +03:00
// if the first argument is 'all' then start all torrents
if tokens[0] == "all" {
if err := Client.VerifyAll(); err != nil {
send("check: error occurred while verifying some torrents", ud.Message.Chat.ID)
return
2016-06-21 18:42:30 +03:00
}
2016-06-27 18:40:09 +03:00
send("verifying all torrents", ud.Message.Chat.ID)
return
2016-06-21 18:42:30 +03:00
}
2016-06-27 18:40:09 +03:00
for _, id := range tokens {
num, err := strconv.Atoi(id)
if err != nil {
send(fmt.Sprintf("check: %s is not a number", id), ud.Message.Chat.ID)
continue
}
status, err := Client.VerifyTorrent(num)
if err != nil {
send("stop: "+err.Error(), ud.Message.Chat.ID)
continue
}
torrent, err := Client.GetTorrent(num)
if err != nil {
send(fmt.Sprintf("[fail] check: No torrent with an ID of %d", num), ud.Message.Chat.ID)
return
}
send(fmt.Sprintf("[%s] check: %s", status, torrent.Name), ud.Message.Chat.ID)
}
2016-06-21 18:42:30 +03:00
}
2016-06-25 07:44:34 +03:00
// stats echo back transmission stats
func stats(ud *tgbotapi.Update) {
stats, err := Client.GetStats()
if err != nil {
send("stats: "+err.Error(), ud.Message.Chat.ID)
return
}
msg := fmt.Sprintf(
`
Total torrents: %d
Active: %d
Paused: %d
- Current Stats -
Downloaded: %s
Uploaded: %s
Running time: %d seconds
- Total Stats -
Sessions: %d
Downloaded: %s
Uploaded: %s
Total Running time: %d seconds
`,
stats.TorrentCount,
stats.ActiveTorrentCount,
stats.PausedTorrentCount,
humanize.Bytes(stats.CurrentStats.DownloadedBytes),
humanize.Bytes(stats.CurrentStats.UploadedBytes),
stats.CurrentStats.SecondsActive,
stats.CumulativeStats.SessionCount,
humanize.Bytes(stats.CumulativeStats.DownloadedBytes),
humanize.Bytes(stats.CumulativeStats.UploadedBytes),
stats.CumulativeStats.SecondsActive,
)
send(msg, ud.Message.Chat.ID)
}
// speed will echo back the current download and upload speeds
func speed(ud *tgbotapi.Update) {
stats, err := Client.GetStats()
if err != nil {
send("speed: "+err.Error(), ud.Message.Chat.ID)
return
}
msg := fmt.Sprintf("⬇ %s ⬆ %s", humanize.Bytes(stats.DownloadSpeed), humanize.Bytes(stats.UploadSpeed))
send(msg, ud.Message.Chat.ID)
}
2016-06-30 02:03:59 +03:00
// count returns current torrents count per status
func count(ud *tgbotapi.Update) {
torrents, err := Client.GetTorrents()
if err != nil {
send("count: "+err.Error(), ud.Message.Chat.ID)
return
}
var downloading, seeding, stopped, checking, downloadingQ, seedingQ, checkingQ int
for i := range torrents {
switch torrents[i].Status {
case transmission.StatusDownloading:
downloading++
case transmission.StatusSeeding:
seeding++
case transmission.StatusStopped:
stopped++
case transmission.StatusChecking:
checking++
case transmission.StatusDownloadPending:
downloadingQ++
case transmission.StatusSeedPending:
seedingQ++
case transmission.StatusCheckPending:
checkingQ++
}
}
msg := fmt.Sprintf("Downloading: %d\nSeeding: %d\nPaused: %d\nVerifying: %d\n\n- Waiting to -\nDownload: %d\nSeed: %d\nVerify: %d\n\nTotal: %d",
downloading, seeding, stopped, checking, downloadingQ, seedingQ, checkingQ, len(torrents))
send(msg, ud.Message.Chat.ID)
}
2016-06-25 07:44:34 +03:00
// del takes an id or more, and delete the corresponding torrent/s
func del(ud *tgbotapi.Update, tokens []string) {
// make sure that we got an argument
if len(tokens) == 0 {
send("del: needs an ID", ud.Message.Chat.ID)
return
}
// loop over tokens to read each potential id
for _, id := range tokens {
num, err := strconv.Atoi(id)
if err != nil {
send(fmt.Sprintf("del: %s is not an ID", id), ud.Message.Chat.ID)
return
}
2016-06-25 07:44:34 +03:00
name, err := Client.DeleteTorrent(num, false)
if err != nil {
send("del: "+err.Error(), ud.Message.Chat.ID)
return
}
send("Deleted: "+name, ud.Message.Chat.ID)
}
2016-06-25 07:44:34 +03:00
}
// deldata takes an id or more, and delete the corresponding torrent/s with their data
func deldata(ud *tgbotapi.Update, tokens []string) {
// make sure that we got an argument
if len(tokens) == 0 {
send("deldata: needs an ID", ud.Message.Chat.ID)
return
}
// loop over tokens to read each potential id
for _, id := range tokens {
num, err := strconv.Atoi(id)
if err != nil {
send(fmt.Sprintf("deldata: %s is not an ID", id), ud.Message.Chat.ID)
return
}
2016-06-25 07:44:34 +03:00
name, err := Client.DeleteTorrent(num, true)
if err != nil {
send("deldata: "+err.Error(), ud.Message.Chat.ID)
return
}
2016-06-25 07:44:34 +03:00
send("Deleted with data: "+name, ud.Message.Chat.ID)
}
2016-06-25 07:44:34 +03:00
}
2016-06-27 20:23:26 +03:00
// help
// version sends transmission version + transmission-telegram version
func version(ud *tgbotapi.Update) {
send(fmt.Sprintf("Transmission %s\nTransmission-telegram %s", Client.Version(), VERSION), ud.Message.Chat.ID)
}
2016-06-20 05:09:41 +03:00
// send takes a chat id and a message to send.
func send(text string, chatID int64) {
// set typing action
action := tgbotapi.NewChatAction(chatID, tgbotapi.ChatTyping)
Bot.Send(action)
// check the rune count, telegram is limited to 4096 chars per message;
// so if our message is > 4096, split it in chunks the send them.
msgRuneCount := utf8.RuneCountInString(text)
LenCheck:
if msgRuneCount > 4096 {
msg := tgbotapi.NewMessage(chatID, text[:4095])
2016-06-21 18:42:30 +03:00
msg.DisableWebPagePreview = true
2016-06-20 05:09:41 +03:00
// send current chunk
if _, err := Bot.Send(msg); err != nil {
2016-07-01 06:14:38 +03:00
fmt.Fprintf(os.Stderr, "send error: %s\n", err.Error())
2016-06-20 05:09:41 +03:00
}
// move to the next chunk
text = text[4095:]
msgRuneCount = utf8.RuneCountInString(text)
goto LenCheck
}
// if msgRuneCount < 4096, send it normally
msg := tgbotapi.NewMessage(chatID, text)
2016-06-21 18:42:30 +03:00
msg.DisableWebPagePreview = true
2016-06-20 05:09:41 +03:00
if _, err := Bot.Send(msg); err != nil {
2016-07-01 06:14:38 +03:00
fmt.Fprintf(os.Stderr, "send error: %s\n", err.Error())
2016-06-20 05:09:41 +03:00
}
}