setup functions, plugin enabler, reply markup, example commands
parent
567a37868d
commit
3940cb5953
59
bot.go
59
bot.go
|
@ -3,6 +3,7 @@ package main
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"flag"
|
"flag"
|
||||||
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
"log"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -10,8 +11,9 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type Config struct {
|
type Config struct {
|
||||||
Token string `json:"token"`
|
Token string `json:"token"`
|
||||||
Plugins map[string]string `json:"plugins"`
|
Plugins map[string]string `json:"plugins"`
|
||||||
|
EnabledPlugins map[string]bool `json:"enabled"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Plugin interface {
|
type Plugin interface {
|
||||||
|
@ -19,14 +21,16 @@ type Plugin interface {
|
||||||
GetCommands() []string
|
GetCommands() []string
|
||||||
GetHelpText() []string
|
GetHelpText() []string
|
||||||
GotCommand(string, Message, []string)
|
GotCommand(string, Message, []string)
|
||||||
|
Setup()
|
||||||
}
|
}
|
||||||
|
|
||||||
var bot *BotApi
|
var bot *BotApi
|
||||||
var plugins []Plugin
|
var plugins []Plugin
|
||||||
var config Config
|
var config Config
|
||||||
|
var configPath *string
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
configPath := flag.String("config", "config.json", "path to config.json")
|
configPath = flag.String("config", "config.json", "path to config.json")
|
||||||
|
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
|
@ -42,7 +46,34 @@ func main() {
|
||||||
debug: true,
|
debug: true,
|
||||||
})
|
})
|
||||||
|
|
||||||
plugins = []Plugin{&HelpPlugin{}, &FAPlugin{}}
|
plugins = []Plugin{&HelpPlugin{}, &FAPlugin{}, &ManagePlugin{}}
|
||||||
|
|
||||||
|
for _, plugin := range plugins {
|
||||||
|
val, ok := config.EnabledPlugins[plugin.GetName()]
|
||||||
|
|
||||||
|
if !ok {
|
||||||
|
fmt.Printf("Enable '%s'? [y/N] ", plugin.GetName())
|
||||||
|
|
||||||
|
var enabled string
|
||||||
|
fmt.Scanln(&enabled)
|
||||||
|
|
||||||
|
if strings.ToLower(enabled) == "y" {
|
||||||
|
plugin.Setup()
|
||||||
|
log.Printf("Plugin '%s' started!\n", plugin.GetName())
|
||||||
|
|
||||||
|
config.EnabledPlugins[plugin.GetName()] = true
|
||||||
|
} else {
|
||||||
|
config.EnabledPlugins[plugin.GetName()] = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if val {
|
||||||
|
plugin.Setup()
|
||||||
|
log.Printf("Plugin '%s' started!\n", plugin.GetName())
|
||||||
|
}
|
||||||
|
|
||||||
|
saveConfig()
|
||||||
|
}
|
||||||
|
|
||||||
ticker := time.NewTicker(time.Second)
|
ticker := time.NewTicker(time.Second)
|
||||||
|
|
||||||
|
@ -66,16 +97,32 @@ func main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, plugin := range plugins {
|
for _, plugin := range plugins {
|
||||||
|
val, _ := config.EnabledPlugins[plugin.GetName()]
|
||||||
|
if !val {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
parts := strings.Split(update.Message.Text, " ")
|
parts := strings.Split(update.Message.Text, " ")
|
||||||
|
command := parts[0]
|
||||||
|
|
||||||
for _, cmd := range plugin.GetCommands() {
|
for _, cmd := range plugin.GetCommands() {
|
||||||
if cmd == parts[0] {
|
if cmd == command {
|
||||||
|
if bot.config.debug {
|
||||||
|
log.Printf("'%s' matched plugin '%s'", update.Message.Text, plugin.GetName())
|
||||||
|
}
|
||||||
|
|
||||||
args := append(parts[:0], parts[1:]...)
|
args := append(parts[:0], parts[1:]...)
|
||||||
|
|
||||||
plugin.GotCommand(parts[0], update.Message, args)
|
plugin.GotCommand(command, update.Message, args)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func saveConfig() {
|
||||||
|
data, _ := json.MarshalIndent(config, "", " ")
|
||||||
|
|
||||||
|
ioutil.WriteFile(*configPath, data, 0600)
|
||||||
|
}
|
||||||
|
|
|
@ -148,6 +148,14 @@ func (bot *BotApi) sendMessage(config MessageConfig) (Message, error) {
|
||||||
if config.ReplyToMessageId != 0 {
|
if config.ReplyToMessageId != 0 {
|
||||||
v.Add("reply_to_message_id", strconv.Itoa(config.ReplyToMessageId))
|
v.Add("reply_to_message_id", strconv.Itoa(config.ReplyToMessageId))
|
||||||
}
|
}
|
||||||
|
if config.ReplyMarkup != nil {
|
||||||
|
data, err := json.Marshal(config.ReplyMarkup)
|
||||||
|
if err != nil {
|
||||||
|
return Message{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
v.Add("reply_markup", string(data))
|
||||||
|
}
|
||||||
|
|
||||||
resp, err := bot.makeRequest("sendMessage", v)
|
resp, err := bot.makeRequest("sendMessage", v)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -623,6 +631,7 @@ type MessageConfig struct {
|
||||||
Text string
|
Text string
|
||||||
DisableWebPagePreview bool
|
DisableWebPagePreview bool
|
||||||
ReplyToMessageId int
|
ReplyToMessageId int
|
||||||
|
ReplyMarkup interface{}
|
||||||
}
|
}
|
||||||
|
|
||||||
type ForwardConfig struct {
|
type ForwardConfig struct {
|
||||||
|
|
|
@ -0,0 +1,99 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"github.com/PuerkitoBio/goquery"
|
||||||
|
"github.com/ddliu/go-httpclient"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type FAPlugin struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *FAPlugin) GetName() string {
|
||||||
|
return "FA Mirrorer"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *FAPlugin) GetCommands() []string {
|
||||||
|
return []string{"/fa"}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *FAPlugin) GetHelpText() []string {
|
||||||
|
return []string{"/fa [link] - mirrors an image from FurAffinity"}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *FAPlugin) Setup() {
|
||||||
|
a, ok := config.Plugins["fa_a"]
|
||||||
|
if !ok {
|
||||||
|
fmt.Print("FurAffinity Cookie a: ")
|
||||||
|
fmt.Scanln(&a)
|
||||||
|
|
||||||
|
config.Plugins["fa_a"] = a
|
||||||
|
}
|
||||||
|
|
||||||
|
b, ok := config.Plugins["fa_b"]
|
||||||
|
if !ok {
|
||||||
|
fmt.Print("FurAffinity Cookie b: ")
|
||||||
|
fmt.Scanln(&b)
|
||||||
|
|
||||||
|
config.Plugins["fa_b"] = b
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *FAPlugin) GotCommand(command string, message Message, args []string) {
|
||||||
|
if len(args) == 0 {
|
||||||
|
bot.sendMessage(NewMessage(message.Chat.Id, "You need to include a link!"))
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
bot.sendChatAction(NewChatAction(message.Chat.Id, CHAT_UPLOAD_PHOTO))
|
||||||
|
|
||||||
|
_, err := strconv.Atoi(args[0])
|
||||||
|
if err == nil {
|
||||||
|
args[0] = "http://www.furaffinity.net/view/" + args[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := httpclient.WithCookie(&http.Cookie{
|
||||||
|
Name: "b",
|
||||||
|
Value: config.Plugins["fa_b"],
|
||||||
|
}).WithCookie(&http.Cookie{
|
||||||
|
Name: "a",
|
||||||
|
Value: config.Plugins["fa_a"],
|
||||||
|
}).Get(args[0], nil)
|
||||||
|
if err != nil {
|
||||||
|
bot.sendMessage(NewMessage(message.Chat.Id, "ERR : "+err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
doc, err := goquery.NewDocumentFromReader(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
bot.sendMessage(NewMessage(message.Chat.Id, "ERR : "+err.Error()))
|
||||||
|
}
|
||||||
|
|
||||||
|
sel := doc.Find("#submissionImg")
|
||||||
|
for i := range sel.Nodes {
|
||||||
|
single := sel.Eq(i)
|
||||||
|
|
||||||
|
val, _ := single.Attr("src")
|
||||||
|
|
||||||
|
tokens := strings.Split(val, "/")
|
||||||
|
fileName := tokens[len(tokens)-1]
|
||||||
|
|
||||||
|
output, _ := os.Create(fileName)
|
||||||
|
defer output.Close()
|
||||||
|
defer os.Remove(output.Name())
|
||||||
|
|
||||||
|
resp, _ := http.Get("http:" + val)
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
io.Copy(output, resp.Body)
|
||||||
|
|
||||||
|
bot.sendPhoto(NewPhotoUpload(message.Chat.Id, output.Name()))
|
||||||
|
}
|
||||||
|
}
|
|
@ -20,6 +20,9 @@ func (plugin *HelpPlugin) GetHelpText() []string {
|
||||||
return []string{"/help (/command) - returns help about a command"}
|
return []string{"/help (/command) - returns help about a command"}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (plugin *HelpPlugin) Setup() {
|
||||||
|
}
|
||||||
|
|
||||||
func (plugin *HelpPlugin) GotCommand(command string, message Message, args []string) {
|
func (plugin *HelpPlugin) GotCommand(command string, message Message, args []string) {
|
||||||
msg := NewMessage(message.Chat.Id, "")
|
msg := NewMessage(message.Chat.Id, "")
|
||||||
msg.ReplyToMessageId = message.MessageId
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
@ -50,7 +53,12 @@ func (plugin *HelpPlugin) GotCommand(command string, message Message, args []str
|
||||||
buffer.WriteString("\n\n")
|
buffer.WriteString("\n\n")
|
||||||
|
|
||||||
for _, plug := range plugins {
|
for _, plug := range plugins {
|
||||||
|
val, _ := config.EnabledPlugins[plugin.GetName()]
|
||||||
|
|
||||||
buffer.WriteString(plug.GetName())
|
buffer.WriteString(plug.GetName())
|
||||||
|
if !val {
|
||||||
|
buffer.WriteString(" (disabled)")
|
||||||
|
}
|
||||||
buffer.WriteString("\n")
|
buffer.WriteString("\n")
|
||||||
|
|
||||||
for _, cmd := range plug.GetHelpText() {
|
for _, cmd := range plug.GetHelpText() {
|
||||||
|
|
|
@ -0,0 +1,153 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ManagePlugin struct {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *ManagePlugin) GetName() string {
|
||||||
|
return "Plugin manager"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *ManagePlugin) GetCommands() []string {
|
||||||
|
return []string{
|
||||||
|
"/enable",
|
||||||
|
"Enable",
|
||||||
|
"/disable",
|
||||||
|
"Disable",
|
||||||
|
"/reload",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *ManagePlugin) GetHelpText() []string {
|
||||||
|
return []string{
|
||||||
|
"/enable [name] - enables a plugin",
|
||||||
|
"/disable [name] - disables a plugin",
|
||||||
|
"/reload - reloads bot configuration",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *ManagePlugin) Setup() {
|
||||||
|
}
|
||||||
|
|
||||||
|
func (plugin *ManagePlugin) GotCommand(command string, message Message, args []string) {
|
||||||
|
log.Println(command)
|
||||||
|
|
||||||
|
if command == "/enable" {
|
||||||
|
keyboard := [][]string{}
|
||||||
|
|
||||||
|
hasDisabled := false
|
||||||
|
for _, plug := range plugins {
|
||||||
|
enabled, _ := config.EnabledPlugins[plug.GetName()]
|
||||||
|
if enabled {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
hasDisabled = true
|
||||||
|
keyboard = append(keyboard, []string{"Enable " + plug.GetName()})
|
||||||
|
}
|
||||||
|
|
||||||
|
if !hasDisabled {
|
||||||
|
msg := NewMessage(message.Chat.Id, "All plugins are enabled!")
|
||||||
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
|
||||||
|
bot.sendMessage(msg)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
msg := NewMessage(message.Chat.Id, "Please specify which plugin to enable")
|
||||||
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
msg.ReplyMarkup = ReplyKeyboardMarkup{
|
||||||
|
Keyboard: keyboard,
|
||||||
|
OneTimeKeyboard: true,
|
||||||
|
Selective: true,
|
||||||
|
ResizeKeyboard: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
bot.sendMessage(msg)
|
||||||
|
} else if command == "Enable" {
|
||||||
|
pluginName := strings.SplitN(message.Text, " ", 2)
|
||||||
|
|
||||||
|
msg := NewMessage(message.Chat.Id, "")
|
||||||
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
msg.ReplyMarkup = ReplyKeyboardHide{
|
||||||
|
HideKeyboard: true,
|
||||||
|
Selective: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, ok := config.EnabledPlugins[pluginName[1]]
|
||||||
|
if !ok {
|
||||||
|
msg.Text = "Unknown plugin!"
|
||||||
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
bot.sendMessage(msg)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
config.EnabledPlugins[pluginName[1]] = true
|
||||||
|
msg.Text = fmt.Sprintf("Enabled '%s'!", pluginName[1])
|
||||||
|
bot.sendMessage(msg)
|
||||||
|
} else if command == "/disable" {
|
||||||
|
keyboard := [][]string{}
|
||||||
|
|
||||||
|
hasEnabled := false
|
||||||
|
for _, plug := range plugins {
|
||||||
|
enabled, _ := config.EnabledPlugins[plug.GetName()]
|
||||||
|
if !enabled {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
hasEnabled = true
|
||||||
|
keyboard = append(keyboard, []string{"Disable " + plug.GetName()})
|
||||||
|
}
|
||||||
|
|
||||||
|
if !hasEnabled {
|
||||||
|
msg := NewMessage(message.Chat.Id, "All plugins are disabled!")
|
||||||
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
|
||||||
|
bot.sendMessage(msg)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
msg := NewMessage(message.Chat.Id, "Please specify which plugin to disable")
|
||||||
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
msg.ReplyMarkup = ReplyKeyboardMarkup{
|
||||||
|
Keyboard: keyboard,
|
||||||
|
OneTimeKeyboard: true,
|
||||||
|
Selective: true,
|
||||||
|
ResizeKeyboard: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
bot.sendMessage(msg)
|
||||||
|
} else if command == "Disable" {
|
||||||
|
pluginName := strings.SplitN(message.Text, " ", 2)
|
||||||
|
|
||||||
|
msg := NewMessage(message.Chat.Id, "")
|
||||||
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
msg.ReplyMarkup = ReplyKeyboardHide{
|
||||||
|
HideKeyboard: true,
|
||||||
|
Selective: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
_, ok := config.EnabledPlugins[pluginName[1]]
|
||||||
|
if !ok {
|
||||||
|
msg.Text = "Unknown plugin!"
|
||||||
|
msg.ReplyToMessageId = message.MessageId
|
||||||
|
bot.sendMessage(msg)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
config.EnabledPlugins[pluginName[1]] = false
|
||||||
|
msg.Text = fmt.Sprintf("Disabled '%s'!", pluginName[1])
|
||||||
|
bot.sendMessage(msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
saveConfig()
|
||||||
|
}
|
|
@ -119,10 +119,10 @@ type UserProfilePhotos struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
type ReplyKeyboardMarkup struct {
|
type ReplyKeyboardMarkup struct {
|
||||||
Keyboard map[string]map[string]string `json:"keyboard"`
|
Keyboard [][]string `json:"keyboard"`
|
||||||
ResizeKeyboard bool `json:"resize_keyboard"`
|
ResizeKeyboard bool `json:"resize_keyboard"`
|
||||||
OneTimeKeyboard bool `json:"one_time_keyboard"`
|
OneTimeKeyboard bool `json:"one_time_keyboard"`
|
||||||
Selective bool `json:"selective"`
|
Selective bool `json:"selective"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ReplyKeyboardHide struct {
|
type ReplyKeyboardHide struct {
|
||||||
|
|
Loading…
Reference in New Issue