Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
b5af39c132 | ||
|
|
4aba14c4c7 | ||
|
|
a66fa5df81 | ||
|
|
f58e28412f | ||
|
|
c8376f78a2 | ||
|
|
5d6b1003e9 | ||
|
|
5ef7c5ce60 | ||
|
|
03c833acbf |
4
.gitignore
vendored
4
.gitignore
vendored
@@ -3,4 +3,6 @@ logs/
|
||||
tmp/
|
||||
data/
|
||||
downloads/
|
||||
cache/
|
||||
cache/
|
||||
session.*
|
||||
cache.db
|
||||
24
README.md
24
README.md
@@ -1,8 +1,8 @@
|
||||
# Save Any Bot
|
||||
|
||||
把 Telegram 的文件保存到各类存储端.
|
||||
把 Telegram 的文件保存到各类存储端.
|
||||
|
||||
> *就像 PikPak Bot 一样*
|
||||
> _就像 PikPak Bot 一样_
|
||||
|
||||
## 部署
|
||||
|
||||
@@ -10,11 +10,6 @@
|
||||
|
||||
在解压后目录新建 `config.toml` 文件, 参考 [config.toml.example](https://github.com/krau/SaveAny-Bot/blob/main/config.example.toml) 编辑配置文件.
|
||||
|
||||
> [!TIP]
|
||||
> 由于 Telegram 官方 Bot API 的限制, Bot 无法下载大于 20MB 的文件. 你需要部署一个本地的 Telegram Bot API 来解决这个问题, 然后将配置文件中的 telegram.api 改为你自己的 api 地址.
|
||||
>
|
||||
> 参考: [telegram-bot-api-compose](https://github.com/krau/telegram-bot-api-compose)
|
||||
|
||||
运行:
|
||||
|
||||
```bash
|
||||
@@ -24,4 +19,17 @@ chmod +x saveany-bot
|
||||
|
||||
## 使用
|
||||
|
||||
向 Bot 发送(转发)文件, 按照提示操作.
|
||||
向 Bot 发送(转发)文件, 按照提示操作.
|
||||
|
||||
## Bot API 版本 (v0.3.0 前)
|
||||
|
||||
> Bot API 版本自身不需要 API_ID 和 API_HASH, 但是部署 Telegram Bot API 服务器仍然需要.
|
||||
|
||||
由于 Telegram 官方 Bot API 的限制, Bot 无法下载大于 20MB 的文件. 你需要部署一个本地的 Telegram Bot API 来解决这个问题, 然后在配置文件改为你自己的 api 地址
|
||||
|
||||
```toml
|
||||
[telegram]
|
||||
api = "http://localhost:8081"
|
||||
```
|
||||
|
||||
参考: [telegram-bot-api-compose](https://github.com/krau/telegram-bot-api-compose)
|
||||
|
||||
115
bot/bot.go
115
bot/bot.go
@@ -3,108 +3,57 @@ package bot
|
||||
import (
|
||||
"os"
|
||||
|
||||
"github.com/duke-git/lancet/v2/slice"
|
||||
"github.com/amarnathcjd/gogram/telegram"
|
||||
"github.com/krau/SaveAny-Bot/config"
|
||||
"github.com/krau/SaveAny-Bot/logger"
|
||||
"github.com/mymmrac/telego"
|
||||
"github.com/mymmrac/telego/telegohandler"
|
||||
"github.com/mymmrac/telego/telegoutil"
|
||||
)
|
||||
|
||||
var (
|
||||
Bot *telego.Bot
|
||||
Client *telegram.Client
|
||||
)
|
||||
|
||||
func Init() {
|
||||
logger.L.Debug("Initializing bot...")
|
||||
var err error
|
||||
Bot, err = telego.NewBot(
|
||||
config.Cfg.Telegram.Token,
|
||||
telego.WithDefaultLogger(false, true),
|
||||
telego.WithAPIServer(config.Cfg.Telegram.API),
|
||||
)
|
||||
Client, err = telegram.NewClient(telegram.ClientConfig{
|
||||
AppID: config.Cfg.Telegram.AppID,
|
||||
AppHash: config.Cfg.Telegram.AppHash,
|
||||
LogLevel: telegram.LogInfo,
|
||||
})
|
||||
if err != nil {
|
||||
logger.L.Fatal("Failed to create bot: ", err)
|
||||
logger.L.Fatal("Failed to create telegram client: ", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
Bot.SetMyCommands(&telego.SetMyCommandsParams{
|
||||
Commands: []telego.BotCommand{
|
||||
{Command: "start", Description: "开始使用"},
|
||||
{Command: "help", Description: "显示帮助"},
|
||||
{Command: "silent", Description: "静默模式"},
|
||||
{Command: "storage", Description: "设置默认存储位置"},
|
||||
{Command: "save", Description: "保存文件"},
|
||||
{Command: "clean", Description: "清除文件记录"},
|
||||
},
|
||||
if err := Client.LoginBot(config.Cfg.Telegram.Token); err != nil {
|
||||
logger.L.Fatal("Failed to login bot: ", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
logger.L.Info("Bot logged in")
|
||||
_, err = Client.BotsSetBotCommands(&telegram.BotCommandScopeDefault{}, "", []*telegram.BotCommand{
|
||||
{Command: "start", Description: "开始使用"},
|
||||
{Command: "help", Description: "显示帮助"},
|
||||
{Command: "silent", Description: "静默模式"},
|
||||
{Command: "storage", Description: "设置默认存储位置"},
|
||||
{Command: "save", Description: "保存所回复文件"},
|
||||
})
|
||||
logger.L.Debug("Bot initialized")
|
||||
if err != nil {
|
||||
logger.L.Errorf("Failed to set bot commands: ", err)
|
||||
}
|
||||
logger.L.Info("Bot initialized")
|
||||
}
|
||||
|
||||
func Run() {
|
||||
if Bot == nil {
|
||||
if Client == nil {
|
||||
Init()
|
||||
}
|
||||
logger.L.Info("Start polling...")
|
||||
updates, err := Bot.UpdatesViaLongPolling(&telego.GetUpdatesParams{
|
||||
Offset: -1,
|
||||
AllowedUpdates: []string{
|
||||
telego.MessageUpdates,
|
||||
telego.CallbackQueryUpdates,
|
||||
},
|
||||
})
|
||||
if err != nil {
|
||||
logger.L.Fatal("Failed to start polling: ", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
botHandler, err := telegohandler.NewBotHandler(Bot, updates)
|
||||
if err != nil {
|
||||
logger.L.Fatal("Failed to create bot handler: ", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
defer botHandler.Stop()
|
||||
defer Bot.StopLongPolling()
|
||||
|
||||
botHandler.Use(telegohandler.PanicRecovery())
|
||||
baseGroup := botHandler.BaseGroup()
|
||||
Client.On("command:start", Start, telegram.FilterPrivate, telegram.FilterChats(config.Cfg.Telegram.Admins...))
|
||||
Client.On("command:help", Help, telegram.FilterPrivate, telegram.FilterChats(config.Cfg.Telegram.Admins...))
|
||||
Client.On("command:silent", ChangeSilentMode, telegram.FilterPrivate, telegram.FilterChats(config.Cfg.Telegram.Admins...))
|
||||
Client.On("command:storage", SetDefaultStorage, telegram.FilterPrivate, telegram.FilterChats(config.Cfg.Telegram.Admins...))
|
||||
Client.On("command:save", SaveCmd, telegram.FilterPrivate, telegram.FilterChats(config.Cfg.Telegram.Admins...))
|
||||
Client.On(telegram.OnMessage, HandleFileMessage, telegram.FilterPrivate, telegram.FilterChats(config.Cfg.Telegram.Admins...), telegram.FilterMedia)
|
||||
Client.On("callback:add", AddToQueue)
|
||||
|
||||
registerHandlers(baseGroup)
|
||||
|
||||
botHandler.Start()
|
||||
}
|
||||
|
||||
func registerHandlers(hg *telegohandler.HandlerGroup) {
|
||||
msgGroup := hg.Group(telegohandler.AnyMessage())
|
||||
msgGroup.Use(func(bot *telego.Bot, update telego.Update, next telegohandler.Handler) {
|
||||
if !slice.Contain(config.Cfg.Telegram.Admins, update.Message.From.ID) {
|
||||
bot.SendMessage(telegoutil.Message(update.Message.Chat.ChatID(), "抱歉, 该 Bot 为个人使用设计, 您可以部署自己的 SaveAnyBot 实例: https://github.com/krau/SaveAny-Bot"))
|
||||
return
|
||||
}
|
||||
next(bot, update)
|
||||
})
|
||||
|
||||
msgGroup.HandleMessageCtx(Start, telegohandler.CommandEqual("start"))
|
||||
msgGroup.HandleMessageCtx(Help, telegohandler.CommandEqual("help"))
|
||||
msgGroup.HandleMessageCtx(ChangeSilentMode, telegohandler.CommandEqual("silent"))
|
||||
msgGroup.HandleMessageCtx(SetDefaultStorage, telegohandler.CommandEqual("storage"))
|
||||
msgGroup.HandleMessageCtx(SaveFile, telegohandler.CommandEqual("save"))
|
||||
msgGroup.HandleMessageCtx(CleanReceivedFile, telegohandler.CommandEqual("clean"))
|
||||
|
||||
msgGroup.HandleMessageCtx(HandleFileMessage, func(update telego.Update) bool {
|
||||
return update.Message.Document != nil || update.Message.Video != nil || update.Message.Audio != nil
|
||||
})
|
||||
|
||||
callbackGroup := hg.Group(telegohandler.AnyCallbackQueryWithMessage())
|
||||
callbackGroup.Use(func(bot *telego.Bot, update telego.Update, next telegohandler.Handler) {
|
||||
if !slice.Contain(config.Cfg.Telegram.Admins, update.CallbackQuery.From.ID) {
|
||||
bot.AnswerCallbackQuery(telegoutil.
|
||||
CallbackQuery(update.CallbackQuery.ID).
|
||||
WithText("抱歉, 该 Bot 为个人使用设计, 您可以部署自己的 SaveAnyBot 实例: https://github.com/krau/SaveAny-Bot").
|
||||
WithShowAlert().
|
||||
WithCacheTime(60))
|
||||
return
|
||||
}
|
||||
next(bot, update)
|
||||
})
|
||||
|
||||
callbackGroup.HandleCallbackQueryCtx(AddToQueue, telegohandler.CallbackDataPrefix("add"))
|
||||
Client.Idle()
|
||||
}
|
||||
|
||||
308
bot/handlers.go
308
bot/handlers.go
@@ -3,10 +3,10 @@ package bot
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"path/filepath"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"github.com/amarnathcjd/gogram/telegram"
|
||||
"github.com/duke-git/lancet/v2/slice"
|
||||
"github.com/gookit/goutil/maputil"
|
||||
"github.com/krau/SaveAny-Bot/dao"
|
||||
@@ -15,19 +15,18 @@ import (
|
||||
"github.com/krau/SaveAny-Bot/queue"
|
||||
"github.com/krau/SaveAny-Bot/storage"
|
||||
"github.com/krau/SaveAny-Bot/types"
|
||||
"github.com/mymmrac/telego"
|
||||
"github.com/mymmrac/telego/telegoutil"
|
||||
)
|
||||
|
||||
func Start(ctx context.Context, bot *telego.Bot, message telego.Message) {
|
||||
if err := dao.CreateUser(message.From.ID); err != nil {
|
||||
func Start(message *telegram.NewMessage) error {
|
||||
if err := dao.CreateUser(message.ChatID()); err != nil {
|
||||
logger.L.Errorf("Failed to create user: %s", err)
|
||||
return
|
||||
return err
|
||||
}
|
||||
Help(ctx, bot, message)
|
||||
return Help(message)
|
||||
}
|
||||
|
||||
func Help(ctx context.Context, bot *telego.Bot, message telego.Message) {
|
||||
func Help(message *telegram.NewMessage) error {
|
||||
helpText := `
|
||||
SaveAny Bot - 转存你的 Telegram 文件
|
||||
命令:
|
||||
@@ -36,230 +35,237 @@ SaveAny Bot - 转存你的 Telegram 文件
|
||||
/silent - 静默模式
|
||||
/storage - 设置默认存储位置
|
||||
/save - 保存文件
|
||||
/clean - 清除文件记录
|
||||
|
||||
静默模式: 开启后 Bot 直接保存到收到的文件到默认位置, 不再询问
|
||||
`
|
||||
ReplyMessage(message, helpText)
|
||||
if _, err := message.Reply(helpText); err != nil {
|
||||
logger.L.Errorf("Failed to send help message: %s", err)
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func ChangeSilentMode(ctx context.Context, bot *telego.Bot, message telego.Message) {
|
||||
user, err := dao.GetUserByUserID(message.From.ID)
|
||||
func ChangeSilentMode(message *telegram.NewMessage) error {
|
||||
user, err := dao.GetUserByUserID(message.ChatID())
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
return
|
||||
return err
|
||||
}
|
||||
user.Silent = !user.Silent
|
||||
err = dao.UpdateUser(user)
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
return
|
||||
return err
|
||||
}
|
||||
ReplyMessage(message, fmt.Sprintf("已%s静默模式", map[bool]string{true: "开启", false: "关闭"}[user.Silent]))
|
||||
if _, err := message.Reply(fmt.Sprintf("已%s静默模式", map[bool]string{true: "开启", false: "关闭"}[user.Silent])); err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func SetDefaultStorage(ctx context.Context, bot *telego.Bot, message telego.Message) {
|
||||
func SetDefaultStorage(message *telegram.NewMessage) error {
|
||||
if len(storage.Storages) == 0 {
|
||||
ReplyMessage(message, "当前无可用存储端, 请检查配置.")
|
||||
return
|
||||
message.Reply("当前无可用存储端, 请检查配置.")
|
||||
return nil
|
||||
}
|
||||
_, _, args := telegoutil.ParseCommand(message.Text)
|
||||
_, _, args := telegoutil.ParseCommand(message.Text())
|
||||
availableStorages := maputil.Keys(storage.Storages)
|
||||
if len(args) == 0 {
|
||||
text := EscapeMarkdown("请提供存储位置名称, 可用项:")
|
||||
text := "请提供存储位置名称, 可用项:"
|
||||
for _, name := range availableStorages {
|
||||
text += fmt.Sprintf("\n`%s`", name)
|
||||
}
|
||||
text += fmt.Sprintf("\n`all`")
|
||||
bot.SendMessage(telegoutil.Message(message.Chat.ChatID(), text).WithParseMode(telego.ModeMarkdownV2))
|
||||
return
|
||||
message.Reply(text, telegram.SendOptions{ParseMode: telegram.MarkDown})
|
||||
return nil
|
||||
}
|
||||
storageName := args[0]
|
||||
if !slice.Contain(availableStorages, storageName) {
|
||||
ReplyMessage(message, "参数错误")
|
||||
return
|
||||
message.Reply("参数错误")
|
||||
return nil
|
||||
}
|
||||
user, err := dao.GetUserByUserID(message.From.ID)
|
||||
user, err := dao.GetUserByUserID(message.ChatID())
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
return
|
||||
return err
|
||||
}
|
||||
user.DefaultStorage = storageName
|
||||
err = dao.UpdateUser(user)
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
return
|
||||
return err
|
||||
}
|
||||
ReplyMessage(message, fmt.Sprintf("已设置默认存储位置为: %s", storageName))
|
||||
if _, err := message.Reply(fmt.Sprintf("已设置默认存储位置为: %s", storageName)); err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func SaveFile(ctx context.Context, bot *telego.Bot, message telego.Message) {
|
||||
targetMessage := message.ReplyToMessage
|
||||
if targetMessage == nil {
|
||||
ReplyMessage(message, "请回复要保存的文件")
|
||||
return
|
||||
func SaveCmd(message *telegram.NewMessage) error {
|
||||
targetMessage, err := message.GetReplyMessage()
|
||||
if err != nil {
|
||||
message.Reply("请回复要保存的文件")
|
||||
return nil
|
||||
}
|
||||
if targetMessage.Document == nil && targetMessage.Video == nil && targetMessage.Audio == nil {
|
||||
ReplyMessage(message, "回复的消息不包含文件")
|
||||
return
|
||||
}
|
||||
ctx = context.WithValue(ctx, "force", true)
|
||||
HandleFileMessage(ctx, bot, *targetMessage)
|
||||
}
|
||||
|
||||
func CleanReceivedFile(ctx context.Context, bot *telego.Bot, message telego.Message) {
|
||||
targetMessage := message.ReplyToMessage
|
||||
if targetMessage == nil {
|
||||
ReplyMessage(message, "请回复要清除记录的文件")
|
||||
return
|
||||
}
|
||||
if targetMessage.Document == nil && targetMessage.Video == nil && targetMessage.Audio == nil {
|
||||
ReplyMessage(message, "回复的消息不包含文件")
|
||||
return
|
||||
}
|
||||
var fileUniqueID string
|
||||
if targetMessage.Document != nil {
|
||||
fileUniqueID = targetMessage.Document.FileUniqueID
|
||||
} else if targetMessage.Video != nil {
|
||||
fileUniqueID = targetMessage.Video.FileUniqueID
|
||||
} else if targetMessage.Audio != nil {
|
||||
fileUniqueID = targetMessage.Audio.FileUniqueID
|
||||
if !targetMessage.IsMedia() {
|
||||
message.Reply("回复的消息不包含文件")
|
||||
return nil
|
||||
}
|
||||
|
||||
if fileUniqueID == "" {
|
||||
ReplyMessage(message, "文件信息获取失败")
|
||||
return
|
||||
}
|
||||
|
||||
if err := dao.DeleteReceivedFileByFileUniqueID(fileUniqueID); err != nil {
|
||||
logger.L.Error(err)
|
||||
ReplyMessage(message, "删除记录失败")
|
||||
return
|
||||
}
|
||||
ReplyMessage(message, "记录已删除")
|
||||
}
|
||||
|
||||
func HandleFileMessage(ctx context.Context, bot *telego.Bot, message telego.Message) {
|
||||
var fileID, fileName string
|
||||
if message.Document != nil {
|
||||
fileID = message.Document.FileID
|
||||
fileName = message.Document.FileName
|
||||
} else if message.Video != nil {
|
||||
fileID = message.Video.FileID
|
||||
fileName = message.Video.FileName
|
||||
} else if message.Audio != nil {
|
||||
fileID = message.Audio.FileID
|
||||
fileName = message.Audio.FileName
|
||||
}
|
||||
|
||||
if fileID == "" {
|
||||
logger.L.Error("File ID is empty")
|
||||
ReplyMessage(message, "文件信息获取失败")
|
||||
return
|
||||
}
|
||||
user, err := dao.GetUserByUserID(message.From.ID)
|
||||
msg, err := targetMessage.Reply("正在获取文件信息...")
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
return
|
||||
message.Reply("获取文件信息失败")
|
||||
return err
|
||||
}
|
||||
msg, err := ReplyMessage(message, "正在获取文件信息")
|
||||
|
||||
_, _, _, fileName, err := telegram.GetFileLocation(targetMessage.Media())
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
return
|
||||
}
|
||||
file, err := bot.GetFile(&telego.GetFileParams{FileID: fileID})
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
ReplyMessage(message, "获取文件信息失败")
|
||||
return
|
||||
}
|
||||
if ctx.Value("force") == nil {
|
||||
receivedFile, _ := dao.GetReceivedFileByFileID(file.FileID)
|
||||
if receivedFile != nil && receivedFile.Processing {
|
||||
bot.EditMessageText(&telego.EditMessageTextParams{
|
||||
ChatID: message.Chat.ChatID(),
|
||||
MessageID: msg.MessageID,
|
||||
Text: "该文件或许正在处理中, 使用 /save 命令回复此文件以强制加入队列\n使用 /clean 命令回复此文件以清除对应的记录",
|
||||
})
|
||||
return
|
||||
}
|
||||
targetMessage.Reply("获取文件信息失败")
|
||||
return err
|
||||
}
|
||||
if fileName == "" {
|
||||
fileName = filepath.Base(file.FilePath)
|
||||
logger.L.Error("Empty file name")
|
||||
targetMessage.Reply("文件名为空")
|
||||
return nil
|
||||
}
|
||||
|
||||
err = dao.AddReceivedFile(&model.ReceivedFile{
|
||||
FileUniqueID: file.FileUniqueID,
|
||||
FileID: file.FileID,
|
||||
if err := dao.AddReceivedFile(&model.ReceivedFile{
|
||||
Processing: false,
|
||||
FileName: fileName,
|
||||
FilePath: file.FilePath,
|
||||
FileSize: file.FileSize,
|
||||
MediaGroupID: message.MediaGroupID,
|
||||
ChatID: message.Chat.ChatID().ID,
|
||||
MessageID: message.MessageID,
|
||||
ReplyMessageID: msg.MessageID,
|
||||
})
|
||||
ChatID: targetMessage.ChatID(),
|
||||
MessageID: targetMessage.Message.ID,
|
||||
ReplyMessageID: msg.ID,
|
||||
}); err != nil {
|
||||
logger.L.Error(err)
|
||||
msg.Edit("保存文件信息失败")
|
||||
return err
|
||||
}
|
||||
|
||||
user, err := dao.GetUserByUserID(message.ChatID())
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
ReplyMessage(message, "创建任务失败")
|
||||
return
|
||||
msg.Edit("获取用户信息失败")
|
||||
return err
|
||||
}
|
||||
|
||||
if !user.Silent {
|
||||
bot.EditMessageText(&telego.EditMessageTextParams{
|
||||
ChatID: message.Chat.ChatID(),
|
||||
MessageID: msg.MessageID,
|
||||
Text: "选择要保存的位置",
|
||||
ReplyMarkup: AddTaskReplyMarkup(message.MessageID),
|
||||
msg.Edit("请选择要保存的位置:", telegram.SendOptions{
|
||||
ReplyMarkup: AddTaskReplyMarkup(targetMessage.Message.ID),
|
||||
})
|
||||
return
|
||||
return nil
|
||||
}
|
||||
|
||||
if user.DefaultStorage == "" {
|
||||
bot.EditMessageText(&telego.EditMessageTextParams{
|
||||
ChatID: message.Chat.ChatID(),
|
||||
MessageID: msg.MessageID,
|
||||
Text: "请先使用 /storage 命令设置默认存储位置, 或者关闭静默模式",
|
||||
})
|
||||
return
|
||||
msg.Edit("请先使用 /storage 命令设置默认存储位置, 或者关闭静默模式")
|
||||
return nil
|
||||
}
|
||||
|
||||
queue.AddTask(types.Task{
|
||||
Ctx: context.TODO(),
|
||||
FileID: file.FileID,
|
||||
Status: types.Pending,
|
||||
FileName: fileName,
|
||||
FilePath: file.FilePath,
|
||||
Storage: types.StorageType(user.DefaultStorage),
|
||||
ChatID: message.Chat.ChatID().ID,
|
||||
ReplyMessageID: msg.MessageID,
|
||||
ChatID: targetMessage.ChatID(),
|
||||
MessageID: targetMessage.Message.ID,
|
||||
ReplyMessageID: msg.ID,
|
||||
})
|
||||
|
||||
msg.Edit(fmt.Sprintf("已添加到队列: %s\n当前排队任务数: %d", fileName, queue.Len()))
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func AddToQueue(ctx context.Context, bot *telego.Bot, query telego.CallbackQuery) {
|
||||
args := strings.Split(query.Data, " ")
|
||||
messageID, _ := strconv.Atoi(args[1])
|
||||
receivedFile, err := dao.GetReceivedFileByChatAndMessageID(query.Message.GetChat().ID, messageID)
|
||||
func HandleFileMessage(message *telegram.NewMessage) error {
|
||||
if !message.IsMedia() {
|
||||
return nil
|
||||
}
|
||||
|
||||
user, err := dao.GetUserByUserID(message.ChatID())
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
bot.AnswerCallbackQuery(telegoutil.CallbackQuery(query.ID).WithShowAlert().WithText("获取文件信息失败").WithCacheTime(5))
|
||||
return
|
||||
return nil
|
||||
}
|
||||
|
||||
msg, err := message.Reply("正在获取文件信息...")
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
return err
|
||||
}
|
||||
|
||||
_, _, _, fileName, err := telegram.GetFileLocation(message.Media())
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
message.Reply("获取文件信息失败")
|
||||
return err
|
||||
}
|
||||
if fileName == "" {
|
||||
logger.L.Error("Empty file name")
|
||||
message.Reply("文件名为空")
|
||||
return nil
|
||||
}
|
||||
|
||||
if err := dao.AddReceivedFile(&model.ReceivedFile{
|
||||
Processing: false,
|
||||
FileName: fileName,
|
||||
ChatID: message.ChatID(),
|
||||
MessageID: message.Message.ID,
|
||||
ReplyMessageID: msg.ID,
|
||||
}); err != nil {
|
||||
logger.L.Error(err)
|
||||
msg.Edit("保存文件信息失败")
|
||||
return err
|
||||
}
|
||||
|
||||
if !user.Silent {
|
||||
msg.Edit("请选择要保存的位置:", telegram.SendOptions{
|
||||
ReplyMarkup: AddTaskReplyMarkup(message.Message.ID),
|
||||
})
|
||||
return nil
|
||||
}
|
||||
|
||||
if user.DefaultStorage == "" {
|
||||
msg.Edit("请先使用 /storage 命令设置默认存储位置, 或者关闭静默模式")
|
||||
return nil
|
||||
}
|
||||
|
||||
queue.AddTask(types.Task{
|
||||
Ctx: context.TODO(),
|
||||
Status: types.Pending,
|
||||
FileName: fileName,
|
||||
Storage: types.StorageType(user.DefaultStorage),
|
||||
ChatID: message.ChatID(),
|
||||
MessageID: message.Message.ID,
|
||||
ReplyMessageID: msg.ID,
|
||||
})
|
||||
|
||||
msg.Edit(fmt.Sprintf("已添加到队列: %s\n当前排队任务数: %d", fileName, queue.Len()))
|
||||
return nil
|
||||
}
|
||||
|
||||
func AddToQueue(query *telegram.CallbackQuery) error {
|
||||
args := strings.Split(query.DataString(), " ")
|
||||
messageID, _ := strconv.Atoi(args[1])
|
||||
logger.L.Debug(query.ChatID, messageID)
|
||||
receivedFile, err := dao.GetReceivedFileByChatAndMessageID(query.ChatID, int32(messageID))
|
||||
if err != nil {
|
||||
logger.L.Error(err)
|
||||
query.Answer("获取文件信息失败", &telegram.CallbackOptions{
|
||||
Alert: true,
|
||||
CacheTime: 5,
|
||||
})
|
||||
return err
|
||||
}
|
||||
queue.AddTask(types.Task{
|
||||
Ctx: context.TODO(),
|
||||
FileID: receivedFile.FileID,
|
||||
Status: types.Pending,
|
||||
FileName: receivedFile.FileName,
|
||||
FilePath: receivedFile.FilePath,
|
||||
Storage: types.StorageType(args[2]),
|
||||
ChatID: receivedFile.ChatID,
|
||||
MessageID: receivedFile.MessageID,
|
||||
ReplyMessageID: receivedFile.ReplyMessageID,
|
||||
})
|
||||
bot.EditMessageText(&telego.EditMessageTextParams{
|
||||
Text: fmt.Sprintf("已添加到队列, 当前排队中的任务数: %d", queue.Len()),
|
||||
MessageID: query.Message.GetMessageID(),
|
||||
ChatID: telegoutil.ID(receivedFile.ChatID),
|
||||
})
|
||||
query.Edit(fmt.Sprintf("已添加到队列: %s\n当前排队任务数: %d", receivedFile.FileName, queue.Len()))
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -1 +0,0 @@
|
||||
package bot
|
||||
52
bot/utils.go
52
bot/utils.go
@@ -4,22 +4,10 @@ import (
|
||||
"fmt"
|
||||
"regexp"
|
||||
|
||||
"github.com/amarnathcjd/gogram/telegram"
|
||||
"github.com/krau/SaveAny-Bot/storage"
|
||||
"github.com/mymmrac/telego"
|
||||
"github.com/mymmrac/telego/telegoutil"
|
||||
)
|
||||
|
||||
func FileDownloadURL(filePath string) string {
|
||||
return Bot.FileDownloadURL(filePath)
|
||||
}
|
||||
|
||||
func ReplyMessage(replyTo telego.Message, format string, args ...any) (*telego.Message, error) {
|
||||
return Bot.SendMessage(telegoutil.Messagef(replyTo.Chat.ChatID(), format, args...).
|
||||
WithReplyParameters(&telego.ReplyParameters{
|
||||
MessageID: replyTo.MessageID,
|
||||
}))
|
||||
}
|
||||
|
||||
var StorageDisplayNames = map[string]string{
|
||||
"all": "全部",
|
||||
"local": "服务器磁盘",
|
||||
@@ -27,20 +15,42 @@ var StorageDisplayNames = map[string]string{
|
||||
"webdav": "WebDAV",
|
||||
}
|
||||
|
||||
func AddTaskReplyMarkup(messageID int) *telego.InlineKeyboardMarkup {
|
||||
storageButtons := make([]telego.InlineKeyboardButton, 0)
|
||||
func AddTaskReplyMarkup(messageID int32) telegram.ReplyMarkup {
|
||||
// TODO: sort storage buttons
|
||||
storageButtons := make([]telegram.KeyboardButton, 0)
|
||||
for name := range storage.Storages {
|
||||
storageButtons = append(storageButtons, telegoutil.InlineKeyboardButton(StorageDisplayNames[string(name)]).
|
||||
WithCallbackData(fmt.Sprintf("add %d %s", messageID, name)))
|
||||
storageButtons = append(storageButtons, &telegram.KeyboardButtonCallback{
|
||||
Text: StorageDisplayNames[string(name)],
|
||||
Data: []byte(fmt.Sprintf("add %d %s", messageID, name)),
|
||||
})
|
||||
}
|
||||
|
||||
if len(storageButtons) > 1 {
|
||||
return telegoutil.InlineKeyboard(storageButtons, []telego.InlineKeyboardButton{
|
||||
telegoutil.InlineKeyboardButton("全部").WithCallbackData(fmt.Sprintf("add %d all", messageID)),
|
||||
})
|
||||
return &telegram.ReplyInlineMarkup{
|
||||
Rows: []*telegram.KeyboardButtonRow{
|
||||
{
|
||||
Buttons: storageButtons,
|
||||
},
|
||||
{
|
||||
Buttons: []telegram.KeyboardButton{
|
||||
&telegram.KeyboardButtonCallback{
|
||||
Text: "全部",
|
||||
Data: []byte(fmt.Sprintf("add %d all", messageID)),
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
if len(storageButtons) == 1 {
|
||||
return telegoutil.InlineKeyboard(storageButtons)
|
||||
return &telegram.ReplyInlineMarkup{
|
||||
Rows: []*telegram.KeyboardButtonRow{
|
||||
{
|
||||
Buttons: storageButtons,
|
||||
},
|
||||
},
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -2,6 +2,7 @@ package common
|
||||
|
||||
import (
|
||||
"path/filepath"
|
||||
"time"
|
||||
|
||||
"github.com/imroc/req/v3"
|
||||
"github.com/krau/SaveAny-Bot/config"
|
||||
@@ -10,9 +11,9 @@ import (
|
||||
var ReqClient *req.Client
|
||||
|
||||
func initClient() {
|
||||
ReqClient = req.NewClient().SetOutputDirectory(config.Cfg.Temp.BasePath)
|
||||
ReqClient = req.NewClient().SetOutputDirectory(config.Cfg.Temp.BasePath).SetTimeout(86400 * time.Second)
|
||||
}
|
||||
|
||||
func GetDownloadedFilePath(filename string) string {
|
||||
func GetCacheFilePath(filename string) string {
|
||||
return filepath.Join(config.Cfg.Temp.BasePath, filename)
|
||||
}
|
||||
|
||||
@@ -1,7 +1,8 @@
|
||||
[telegram]
|
||||
token = "" # Bot Token
|
||||
admins = [777000] # 你的 user_id
|
||||
api = "https://api.telegram.org"
|
||||
token = "" # Bot Token
|
||||
admins = [777000] # 你的 user_id
|
||||
api_id = 123456 # Telegram API ID
|
||||
api_hash = "0123456789abcdef0123456789abcdef" # Telegram API Hash
|
||||
|
||||
[log]
|
||||
level = "DEBUG" # 日志等级
|
||||
|
||||
@@ -31,9 +31,10 @@ type dbConfig struct {
|
||||
}
|
||||
|
||||
type telegramConfig struct {
|
||||
Token string `toml:"token" mapstructure:"token"`
|
||||
API string `toml:"api" mapstructure:"api"`
|
||||
Admins []int64 `toml:"admins" mapstructure:"admins"`
|
||||
Token string `toml:"token" mapstructure:"token"`
|
||||
AppID int32 `toml:"app_id" mapstructure:"app_id"`
|
||||
AppHash string `toml:"app_hash" mapstructure:"app_hash"`
|
||||
Admins []int64 `toml:"admins" mapstructure:"admins"`
|
||||
}
|
||||
|
||||
type storageConfig struct {
|
||||
|
||||
72
core/core.go
72
core/core.go
@@ -3,64 +3,38 @@ package core
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
"github.com/imroc/req/v3"
|
||||
"github.com/amarnathcjd/gogram/telegram"
|
||||
"github.com/krau/SaveAny-Bot/bot"
|
||||
"github.com/krau/SaveAny-Bot/common"
|
||||
"github.com/krau/SaveAny-Bot/config"
|
||||
"github.com/krau/SaveAny-Bot/dao"
|
||||
"github.com/krau/SaveAny-Bot/logger"
|
||||
"github.com/krau/SaveAny-Bot/queue"
|
||||
"github.com/krau/SaveAny-Bot/storage"
|
||||
"github.com/krau/SaveAny-Bot/types"
|
||||
"github.com/mymmrac/telego"
|
||||
"github.com/mymmrac/telego/telegoutil"
|
||||
)
|
||||
|
||||
func processPendingTask(task types.Task) error {
|
||||
fileRecord, err := dao.GetReceivedFileByFileID(task.FileID)
|
||||
message, err := bot.Client.GetMessageByID(task.ChatID, task.MessageID)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
fileRecord.Processing = true
|
||||
if err := dao.UpdateReceivedFile(fileRecord); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
_, err = common.ReqClient.R().SetOutputFile(task.FileName).SetDownloadCallbackWithInterval(func(info req.DownloadInfo) {
|
||||
if info.Response == nil || info.Response.Response == nil || info.Response.Response.StatusCode != 200 {
|
||||
return
|
||||
}
|
||||
process := fmt.Sprintf("%.2f%%", float64(info.DownloadedSize)/float64(info.Response.ContentLength)*100.0)
|
||||
bot.Bot.EditMessageText(&telego.EditMessageTextParams{
|
||||
ChatID: telegoutil.ID(task.ChatID),
|
||||
MessageID: task.ReplyMessageID,
|
||||
Text: "正在下载文件: " + process,
|
||||
})
|
||||
}, time.Second*time.Duration(3*func() int {
|
||||
if queue.Len() > 0 {
|
||||
return queue.Len()
|
||||
}
|
||||
return 1
|
||||
}())).Get(bot.FileDownloadURL(task.FilePath))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
bot.Bot.EditMessageText(&telego.EditMessageTextParams{
|
||||
ChatID: telegoutil.ID(task.ChatID),
|
||||
MessageID: task.ReplyMessageID,
|
||||
Text: "下载完成, 正在转存...",
|
||||
logger.L.Debugf("Start downloading file: %s", task.FileName)
|
||||
bot.Client.EditMessage(task.ChatID, task.ReplyMessageID, "正在下载文件...")
|
||||
dest, err := message.Download(&telegram.DownloadOptions{
|
||||
FileName: common.GetCacheFilePath(task.FileName),
|
||||
// ProgressCallback: func(totalBytes, downloadedBytes int64) {},
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
defer func() {
|
||||
if config.Cfg.Temp.CacheTTL > 0 {
|
||||
common.PurgeFileAfter(common.GetDownloadedFilePath(task.FileName),
|
||||
time.Duration(config.Cfg.Temp.CacheTTL)*time.Second)
|
||||
common.PurgeFileAfter(dest, time.Duration(config.Cfg.Temp.CacheTTL)*time.Second)
|
||||
} else {
|
||||
if err := common.PurgeFile(common.GetDownloadedFilePath(task.FileName)); err != nil {
|
||||
if err := common.PurgeFile(dest); err != nil {
|
||||
logger.L.Errorf("Failed to purge file: %s", err)
|
||||
}
|
||||
}
|
||||
@@ -69,9 +43,11 @@ func processPendingTask(task types.Task) error {
|
||||
task.StoragePath = task.FileName
|
||||
}
|
||||
|
||||
if err := storage.Save(task.Storage, task.Ctx, common.GetDownloadedFilePath(task.FileName), task.StoragePath); err != nil {
|
||||
bot.Client.EditMessage(task.ChatID, task.ReplyMessageID, "下载完成, 正在转存文件...")
|
||||
if err := storage.Save(task.Storage, task.Ctx, dest, task.StoragePath); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
@@ -98,24 +74,10 @@ func worker(queue *queue.TaskQueue, semaphore chan struct{}) {
|
||||
queue.AddTask(task)
|
||||
case types.Succeeded:
|
||||
logger.L.Infof("Task succeeded: %s", task.String())
|
||||
bot.Bot.EditMessageText(&telego.EditMessageTextParams{
|
||||
ChatID: telegoutil.ID(task.ChatID),
|
||||
MessageID: task.ReplyMessageID,
|
||||
Text: "文件转存完成",
|
||||
})
|
||||
if err := dao.DeleteReceivedFileByFileID(task.FileID); err != nil {
|
||||
logger.L.Errorf("Failed to delete received file: %s", err)
|
||||
}
|
||||
bot.Client.EditMessage(task.ChatID, task.ReplyMessageID, "文件保存成功")
|
||||
case types.Failed:
|
||||
logger.L.Errorf("Task failed: %s", task.String())
|
||||
bot.Bot.EditMessageText(&telego.EditMessageTextParams{
|
||||
ChatID: telegoutil.ID(task.ChatID),
|
||||
MessageID: task.ReplyMessageID,
|
||||
Text: "文件转存失败:" + "\n" + task.Error.Error(),
|
||||
})
|
||||
if err := dao.DeleteReceivedFileByFileID(task.FileID); err != nil {
|
||||
logger.L.Errorf("Failed to delete received file: %s", err)
|
||||
}
|
||||
bot.Client.EditMessage(task.ChatID, task.ReplyMessageID, "文件保存失败")
|
||||
case types.Canceled:
|
||||
logger.L.Infof("Task canceled: %s", task.String())
|
||||
default:
|
||||
|
||||
37
dao/file.go
37
dao/file.go
@@ -6,25 +6,7 @@ func AddReceivedFile(receivedFile *model.ReceivedFile) error {
|
||||
return db.Create(receivedFile).Error
|
||||
}
|
||||
|
||||
func GetReceivedFileByFileID(fileID string) (*model.ReceivedFile, error) {
|
||||
var receivedFile model.ReceivedFile
|
||||
err := db.Where("file_id = ?", fileID).First(&receivedFile).Error
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &receivedFile, nil
|
||||
}
|
||||
|
||||
func GetReceivedFileByFileUniqueID(fileUniqueID string) (*model.ReceivedFile, error) {
|
||||
var receivedFile model.ReceivedFile
|
||||
err := db.Where("file_unique_id = ?", fileUniqueID).First(&receivedFile).Error
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &receivedFile, nil
|
||||
}
|
||||
|
||||
func GetReceivedFileByChatAndMessageID(chatID int64, messageID int) (*model.ReceivedFile, error) {
|
||||
func GetReceivedFileByChatAndMessageID(chatID int64, messageID int32) (*model.ReceivedFile, error) {
|
||||
var receivedFile model.ReceivedFile
|
||||
err := db.Where("chat_id = ? AND message_id = ?", chatID, messageID).First(&receivedFile).Error
|
||||
if err != nil {
|
||||
@@ -33,23 +15,10 @@ func GetReceivedFileByChatAndMessageID(chatID int64, messageID int) (*model.Rece
|
||||
return &receivedFile, nil
|
||||
}
|
||||
|
||||
func GetReceivedFilesByMediaGroupID(mediaGroupID string) ([]model.ReceivedFile, error) {
|
||||
var receivedFiles []model.ReceivedFile
|
||||
err := db.Where("media_group_id = ?", mediaGroupID).Find(&receivedFiles).Error
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return receivedFiles, nil
|
||||
}
|
||||
|
||||
func UpdateReceivedFile(receivedFile *model.ReceivedFile) error {
|
||||
return db.Save(receivedFile).Error
|
||||
}
|
||||
|
||||
func DeleteReceivedFileByFileID(fileID string) error {
|
||||
return db.Where("file_id = ?", fileID).Delete(&model.ReceivedFile{}).Error
|
||||
}
|
||||
|
||||
func DeleteReceivedFileByFileUniqueID(fileUniqueID string) error {
|
||||
return db.Where("file_unique_id = ?", fileUniqueID).Delete(&model.ReceivedFile{}).Error
|
||||
func DeleteReceivedFile(receivedFile *model.ReceivedFile) error {
|
||||
return db.Delete(receivedFile).Error
|
||||
}
|
||||
|
||||
2
go.mod
2
go.mod
@@ -3,6 +3,7 @@ module github.com/krau/SaveAny-Bot
|
||||
go 1.23.2
|
||||
|
||||
require (
|
||||
github.com/amarnathcjd/gogram v0.0.0-20241008120348-4ce400474c46
|
||||
github.com/blang/semver v3.5.1+incompatible
|
||||
github.com/gookit/slog v0.5.6
|
||||
github.com/imroc/req/v3 v3.46.1
|
||||
@@ -30,6 +31,7 @@ require (
|
||||
github.com/jinzhu/now v1.1.5 // indirect
|
||||
github.com/mattn/go-isatty v0.0.17 // indirect
|
||||
github.com/onsi/ginkgo/v2 v2.20.2 // indirect
|
||||
github.com/pkg/errors v0.9.1 // indirect
|
||||
github.com/quic-go/qpack v0.5.1 // indirect
|
||||
github.com/quic-go/quic-go v0.47.0 // indirect
|
||||
github.com/refraction-networking/utls v1.6.7 // indirect
|
||||
|
||||
4
go.sum
4
go.sum
@@ -1,3 +1,5 @@
|
||||
github.com/amarnathcjd/gogram v0.0.0-20241008120348-4ce400474c46 h1:k3VDOsCB/k/LoIdgItpl/c+lRinqUsXSxeAE+/1Sp1E=
|
||||
github.com/amarnathcjd/gogram v0.0.0-20241008120348-4ce400474c46/go.mod h1:MPzWyqnIwVK/tji8O5pmumdoPoGHqemsIJyr7xedoE8=
|
||||
github.com/andybalholm/brotli v1.1.1 h1:PR2pgnyFznKEugtsUo0xLdDop5SKXd5Qf5ysW+7XdTA=
|
||||
github.com/andybalholm/brotli v1.1.1/go.mod h1:05ib4cKhjx3OQYUY22hTVd34Bc8upXjOLL2rKwwZBoA=
|
||||
github.com/blang/semver v3.5.1+incompatible h1:cQNTCjp13qL8KC3Nbxr/y2Bqb63oX6wdnnjpJbkM4JQ=
|
||||
@@ -111,6 +113,8 @@ github.com/onsi/gomega v1.34.1 h1:EUMJIKUjM8sKjYbtxQI9A4z2o+rruxnzNvpknOXie6k=
|
||||
github.com/onsi/gomega v1.34.1/go.mod h1:kU1QgUvBDLXBJq618Xvm2LUX6rSAfRaFRTcdOeDLwwY=
|
||||
github.com/pelletier/go-toml/v2 v2.2.2 h1:aYUidT7k73Pcl9nb2gScu7NSrKCSHIDE89b3+6Wq+LM=
|
||||
github.com/pelletier/go-toml/v2 v2.2.2/go.mod h1:1t835xjRzz80PqgE6HHgN2JOsmgYu/h4qDAS4n929Rs=
|
||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U=
|
||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
||||
|
||||
@@ -6,18 +6,11 @@ import (
|
||||
|
||||
type ReceivedFile struct {
|
||||
gorm.Model
|
||||
// FileUniqueID 和 FileID 在数据库中均不具有唯一性
|
||||
// 如需确定唯一一行, 使用 ChatID + MessageID
|
||||
FileUniqueID string `gorm:"index"`
|
||||
FileID string `gorm:"index"`
|
||||
Processing bool
|
||||
FileName string
|
||||
FilePath string
|
||||
FileSize int64
|
||||
MediaGroupID string
|
||||
ChatID int64
|
||||
MessageID int
|
||||
ReplyMessageID int
|
||||
MessageID int32
|
||||
ReplyMessageID int32
|
||||
}
|
||||
|
||||
type User struct {
|
||||
|
||||
@@ -71,7 +71,7 @@ func refreshToken(client *req.Client) {
|
||||
func (a *Alist) Init() {
|
||||
basePath = config.Cfg.Storage.Alist.BasePath
|
||||
baseUrl = config.Cfg.Storage.Alist.URL
|
||||
reqClient = req.C().SetTLSHandshakeTimeout(time.Second * 10).SetBaseURL(baseUrl)
|
||||
reqClient = req.C().SetTLSHandshakeTimeout(time.Second * 10).SetBaseURL(baseUrl).SetTimeout(time.Hour * 24)
|
||||
loginReq = &loginRequset{
|
||||
Username: config.Cfg.Storage.Alist.Username,
|
||||
Password: config.Cfg.Storage.Alist.Password,
|
||||
|
||||
@@ -6,6 +6,7 @@ import (
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/krau/SaveAny-Bot/config"
|
||||
"github.com/krau/SaveAny-Bot/logger"
|
||||
@@ -27,6 +28,7 @@ func (w *Webdav) Init() {
|
||||
logger.L.Fatalf("Failed to connect to webdav server: %v", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
Client.SetTimeout(24 * time.Hour)
|
||||
}
|
||||
|
||||
func (w *Webdav) Save(ctx context.Context, filePath, storagePath string) error {
|
||||
|
||||
@@ -24,17 +24,15 @@ var StorageTypes = []StorageType{Local, Alist, Webdav, StorageAll}
|
||||
|
||||
type Task struct {
|
||||
Ctx context.Context
|
||||
FileID string
|
||||
Error error
|
||||
Status TaskStatus
|
||||
FilePath string // telegram File object's FilePath
|
||||
FileName string
|
||||
Storage StorageType
|
||||
StoragePath string
|
||||
|
||||
// For track progress
|
||||
MessageID int32
|
||||
ChatID int64
|
||||
ReplyMessageID int
|
||||
ReplyMessageID int32
|
||||
}
|
||||
|
||||
func (t *Task) String() string {
|
||||
|
||||
Reference in New Issue
Block a user