mirror of
https://github.com/idanoo/autobrr
synced 2025-07-23 00:39:13 +00:00
278 lines
6.3 KiB
Go
278 lines
6.3 KiB
Go
package action
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/autobrr/autobrr/internal/domain"
|
|
"github.com/autobrr/autobrr/internal/download_client"
|
|
"github.com/autobrr/autobrr/pkg/qbittorrent"
|
|
|
|
"github.com/rs/zerolog/log"
|
|
)
|
|
|
|
const REANNOUNCE_MAX_ATTEMPTS = 30
|
|
const REANNOUNCE_INTERVAL = 7000
|
|
|
|
type Service interface {
|
|
RunActions(torrentFile string, hash string, filter domain.Filter) error
|
|
Store(action domain.Action) (*domain.Action, error)
|
|
Fetch() ([]domain.Action, error)
|
|
Delete(actionID int) error
|
|
ToggleEnabled(actionID int) error
|
|
}
|
|
|
|
type service struct {
|
|
repo domain.ActionRepo
|
|
clientSvc download_client.Service
|
|
}
|
|
|
|
func NewService(repo domain.ActionRepo, clientSvc download_client.Service) Service {
|
|
return &service{repo: repo, clientSvc: clientSvc}
|
|
}
|
|
|
|
func (s *service) RunActions(torrentFile string, hash string, filter domain.Filter) error {
|
|
for _, action := range filter.Actions {
|
|
if !action.Enabled {
|
|
// only run active actions
|
|
continue
|
|
}
|
|
|
|
log.Debug().Msgf("process action: %v", action.Name)
|
|
|
|
switch action.Type {
|
|
case domain.ActionTypeTest:
|
|
go s.test(torrentFile)
|
|
|
|
case domain.ActionTypeWatchFolder:
|
|
go s.watchFolder(action.WatchFolder, torrentFile)
|
|
|
|
case domain.ActionTypeQbittorrent:
|
|
go func() {
|
|
err := s.qbittorrent(action, hash, torrentFile)
|
|
if err != nil {
|
|
log.Error().Err(err).Msg("error sending torrent to client")
|
|
}
|
|
}()
|
|
|
|
// deluge
|
|
// pvr *arr
|
|
// exec
|
|
default:
|
|
panic("implement me")
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *service) Store(action domain.Action) (*domain.Action, error) {
|
|
// validate data
|
|
|
|
a, err := s.repo.Store(action)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return a, nil
|
|
}
|
|
|
|
func (s *service) Delete(actionID int) error {
|
|
if err := s.repo.Delete(actionID); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *service) Fetch() ([]domain.Action, error) {
|
|
actions, err := s.repo.List()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return actions, nil
|
|
}
|
|
|
|
func (s *service) ToggleEnabled(actionID int) error {
|
|
if err := s.repo.ToggleEnabled(actionID); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (s *service) test(torrentFile string) {
|
|
log.Info().Msgf("action TEST: %v", torrentFile)
|
|
}
|
|
|
|
func (s *service) watchFolder(dir string, torrentFile string) {
|
|
log.Debug().Msgf("action WATCH_FOLDER: %v file: %v", dir, torrentFile)
|
|
|
|
// Open original file
|
|
original, err := os.Open(torrentFile)
|
|
if err != nil {
|
|
log.Fatal().Err(err)
|
|
}
|
|
defer original.Close()
|
|
|
|
tmpFileName := strings.Split(torrentFile, "/")
|
|
fullFileName := fmt.Sprintf("%v/%v", dir, tmpFileName[1])
|
|
|
|
// Create new file
|
|
newFile, err := os.Create(fullFileName)
|
|
if err != nil {
|
|
log.Fatal().Err(err)
|
|
}
|
|
defer newFile.Close()
|
|
|
|
// Copy file
|
|
_, err = io.Copy(newFile, original)
|
|
if err != nil {
|
|
log.Fatal().Err(err)
|
|
}
|
|
|
|
log.Info().Msgf("action WATCH_FOLDER: wrote file: %v", fullFileName)
|
|
}
|
|
|
|
func (s *service) qbittorrent(action domain.Action, hash string, torrentFile string) error {
|
|
log.Debug().Msgf("action QBITTORRENT: %v", torrentFile)
|
|
|
|
// get client for action
|
|
client, err := s.clientSvc.FindByID(action.ClientID)
|
|
if err != nil {
|
|
log.Error().Err(err).Msgf("error finding client: %v", action.ClientID)
|
|
return err
|
|
}
|
|
|
|
if client == nil {
|
|
return err
|
|
}
|
|
|
|
qbtSettings := qbittorrent.Settings{
|
|
Hostname: client.Host,
|
|
Port: uint(client.Port),
|
|
Username: client.Username,
|
|
Password: client.Password,
|
|
SSL: client.SSL,
|
|
}
|
|
|
|
qbt := qbittorrent.NewClient(qbtSettings)
|
|
// save cookies?
|
|
err = qbt.Login()
|
|
if err != nil {
|
|
log.Error().Err(err).Msgf("error logging into client: %v", action.ClientID)
|
|
return err
|
|
}
|
|
|
|
// TODO check for active downloads and other rules
|
|
|
|
options := map[string]string{}
|
|
|
|
if action.Paused {
|
|
options["paused"] = "true"
|
|
}
|
|
if action.SavePath != "" {
|
|
options["savepath"] = action.SavePath
|
|
options["autoTMM"] = "false"
|
|
}
|
|
if action.Category != "" {
|
|
options["category"] = action.Category
|
|
}
|
|
if action.Tags != "" {
|
|
options["tags"] = action.Tags
|
|
}
|
|
if action.LimitUploadSpeed > 0 {
|
|
options["upLimit"] = strconv.FormatInt(action.LimitUploadSpeed, 10)
|
|
}
|
|
if action.LimitDownloadSpeed > 0 {
|
|
options["dlLimit"] = strconv.FormatInt(action.LimitDownloadSpeed, 10)
|
|
}
|
|
|
|
err = qbt.AddTorrentFromFile(torrentFile, options)
|
|
if err != nil {
|
|
log.Error().Err(err).Msgf("error sending to client: %v", action.ClientID)
|
|
return err
|
|
}
|
|
|
|
if !action.Paused && hash != "" {
|
|
err = checkTrackerStatus(*qbt, hash)
|
|
if err != nil {
|
|
log.Error().Err(err).Msgf("could not get tracker status for torrent: %v", hash)
|
|
return err
|
|
}
|
|
}
|
|
|
|
log.Debug().Msgf("torrent %v successfully added to: %v", hash, client.Name)
|
|
|
|
return nil
|
|
}
|
|
|
|
func checkTrackerStatus(qb qbittorrent.Client, hash string) error {
|
|
announceOK := false
|
|
attempts := 0
|
|
|
|
for attempts < REANNOUNCE_MAX_ATTEMPTS {
|
|
log.Debug().Msgf("RE-ANNOUNCE %v attempt: %v", hash, attempts)
|
|
|
|
// initial sleep to give tracker a head start
|
|
time.Sleep(REANNOUNCE_INTERVAL * time.Millisecond)
|
|
|
|
trackers, err := qb.GetTorrentTrackers(hash)
|
|
if err != nil {
|
|
log.Error().Err(err).Msgf("could not get trackers of torrent: %v", hash)
|
|
return err
|
|
}
|
|
|
|
// check if status not working or something else
|
|
_, working := findTrackerStatus(trackers, qbittorrent.TrackerStatusOK)
|
|
|
|
if !working {
|
|
err = qb.ReAnnounceTorrents([]string{hash})
|
|
if err != nil {
|
|
log.Error().Err(err).Msgf("could not get re-announce torrent: %v", hash)
|
|
return err
|
|
}
|
|
|
|
attempts++
|
|
continue
|
|
} else {
|
|
log.Debug().Msgf("RE-ANNOUNCE %v OK", hash)
|
|
|
|
announceOK = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if !announceOK {
|
|
log.Debug().Msgf("RE-ANNOUNCE %v took too long, deleting torrent", hash)
|
|
|
|
err := qb.DeleteTorrents([]string{hash}, false)
|
|
if err != nil {
|
|
log.Error().Err(err).Msgf("could not delete torrent: %v", hash)
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Check if status not working or something else
|
|
// https://github.com/qbittorrent/qBittorrent/wiki/WebUI-API-(qBittorrent-4.1)#get-torrent-trackers
|
|
// 0 Tracker is disabled (used for DHT, PeX, and LSD)
|
|
// 1 Tracker has not been contacted yet
|
|
// 2 Tracker has been contacted and is working
|
|
// 3 Tracker is updating
|
|
// 4 Tracker has been contacted, but it is not working (or doesn't send proper replies)
|
|
func findTrackerStatus(slice []qbittorrent.TorrentTracker, status qbittorrent.TrackerStatus) (int, bool) {
|
|
for i, item := range slice {
|
|
if item.Status == status {
|
|
return i, true
|
|
}
|
|
}
|
|
return -1, false
|
|
}
|