organizer/handle_file.go

137 lines
3.7 KiB
Go
Raw Normal View History

2023-01-13 19:23:36 +01:00
package main
import (
2023-01-14 12:23:00 +01:00
"errors"
2023-01-15 15:55:47 +01:00
"fmt"
2023-01-14 12:23:00 +01:00
"io"
"os"
2023-01-13 19:23:36 +01:00
"path/filepath"
2023-01-13 20:04:57 +01:00
"time"
2023-01-13 19:23:36 +01:00
"git.milar.in/nyaanime/logic"
"git.milar.in/nyaanime/model"
"git.milar.in/nyaanime/parsers"
2023-01-13 20:04:57 +01:00
"github.com/fatih/color"
2023-01-13 19:23:36 +01:00
)
2023-01-15 15:55:47 +01:00
func HandleFile(path string) <-chan byte {
out := make(chan byte, 1024)
defer close(out)
w := NewWriterFromByteChan(out)
fmt.Fprint(w, color.MagentaString("%s file found: %s\n", time.Now().Format("2006-01-02 15:04:05"), path))
2023-01-13 20:04:57 +01:00
2023-01-13 19:23:36 +01:00
for _, parser := range parsers.Parsers {
parsedFile, ok := parser.FileParser(&parser, path)
if !ok {
2023-01-15 15:55:47 +01:00
fmt.Fprint(w, color.YellowString("\tnot parsable with parser '%s'\n", parser.Identity))
2023-01-13 19:23:36 +01:00
continue
}
2023-01-15 15:55:47 +01:00
fmt.Fprint(w, color.GreenString("\tparsable with parser '%s'\n", parser.Identity))
2023-01-13 20:04:57 +01:00
2023-01-13 19:23:36 +01:00
anime, err := logic.SearchAnimeByTitle(parsedFile.OriginalAnimeTitle)
if err != nil {
2023-01-15 15:55:47 +01:00
fmt.Fprint(w, color.RedString("\tanime not found: '%s'\n", parsedFile.OriginalAnimeTitle))
2023-01-13 19:23:36 +01:00
continue
}
parsedFile.Anime = anime
2023-01-15 15:55:47 +01:00
HandleParsedFile(w, parsedFile)
break
2023-01-13 19:23:36 +01:00
}
2023-01-13 20:04:57 +01:00
2023-01-15 15:55:47 +01:00
return out
2023-01-13 19:23:36 +01:00
}
2023-01-15 15:55:47 +01:00
func HandleParsedFile(w io.Writer, parsedFile *model.ParsedFile) {
2023-01-13 19:23:36 +01:00
newFilePrio := logic.NewFilePriority(parsedFile)
oldFilePrio, animeEpNotExistLocally := logic.GetAnimeEpProps(parsedFile.AnimeEpisode())
2023-01-15 15:55:47 +01:00
// debug output
2023-01-14 12:11:10 +01:00
if animeEpNotExistLocally {
2023-01-15 15:55:47 +01:00
fmt.Fprint(w, color.YellowString("\tfile exists locally\n"))
fmt.Fprint(w, color.YellowString("\t local file: %s\n", FilePrio2Str(oldFilePrio)))
fmt.Fprint(w, color.YellowString("\t new file: %s\n", FilePrio2Str(newFilePrio)))
2023-01-14 12:11:10 +01:00
if newFilePrio.Priority > oldFilePrio.Priority {
2023-01-15 15:55:47 +01:00
fmt.Fprint(w, color.GreenString("\t overwrite local file\n"))
} else {
fmt.Fprint(w, color.YellowString("\t ignore new file\n"))
}
}
// delete files with lower priority from DownloadPath
if DeleteLowPriorityFiles && animeEpNotExistLocally && oldFilePrio.Priority >= newFilePrio.Priority {
fmt.Fprint(w, color.YellowString("\tdelete file with lower priority '%s'", parsedFile.File))
err := os.Remove(parsedFile.File)
if err != nil {
fmt.Fprint(w, color.RedString(" failed: '%s'\n", err.Error()))
2023-01-14 12:11:10 +01:00
} else {
2023-01-15 15:55:47 +01:00
fmt.Fprint(w, color.GreenString(" done\n"))
2023-01-14 12:11:10 +01:00
}
2023-01-15 15:55:47 +01:00
return
2023-01-14 12:11:10 +01:00
}
2023-01-13 19:23:36 +01:00
if !animeEpNotExistLocally || newFilePrio.Priority > oldFilePrio.Priority {
2023-01-15 13:59:48 +01:00
// TODO remove old file when overwriting existing episode (might have other file extension)
2023-01-15 15:55:47 +01:00
if err := OrganizeAnimeEpisode(w, parsedFile); err != nil {
fmt.Fprint(w, color.RedString("\terror: %s\n", err.Error()))
2023-01-14 12:11:10 +01:00
}
2023-01-13 19:23:36 +01:00
}
}
2023-01-15 15:55:47 +01:00
func OrganizeAnimeEpisode(w io.Writer, parsedFile *model.ParsedFile) error {
2023-01-13 20:04:57 +01:00
start := time.Now()
2023-01-13 19:23:36 +01:00
oldFile := filepath.Join(DownloadPath, parsedFile.File)
newFile := logic.GetAnimeEpFilepath(parsedFile.AnimeEpisode(), filepath.Ext(parsedFile.File))
2023-01-14 12:23:00 +01:00
lockFile := logic.GetAnimeEpFilepath(parsedFile.AnimeEpisode(), "lock")
2023-01-13 19:23:36 +01:00
2023-01-15 15:55:47 +01:00
fmt.Fprint(w, color.BlueString("\tmove file\n\t from: '%s'\n\t to: '%s'\n", oldFile, newFile))
2023-01-13 19:23:36 +01:00
2023-01-14 12:23:00 +01:00
if err := os.MkdirAll(filepath.Dir(newFile), os.ModePerm); err != nil {
return err
}
if err := os.Chown(filepath.Dir(newFile), Uid, Gid); err != nil {
return err
}
2023-01-15 13:59:48 +01:00
if _, err := os.Stat(newFile); err != nil && !errors.Is(err, os.ErrNotExist) {
2023-01-14 12:23:00 +01:00
return err
}
inputFile, err := os.Open(oldFile)
if err != nil {
return err
}
defer inputFile.Close()
outputFile, err := os.Create(newFile)
if err != nil {
return err
}
defer outputFile.Close()
if err := os.Chown(newFile, Uid, Gid); err != nil {
return err
}
2023-01-15 13:59:48 +01:00
written, err := io.Copy(outputFile, inputFile)
2023-01-14 12:23:00 +01:00
if err != nil {
return err
}
if err = os.Remove(oldFile); err != nil {
return err
}
2023-01-15 13:59:48 +01:00
if err = os.Remove(lockFile); err != nil && !errors.Is(err, os.ErrNotExist) {
2023-01-14 12:23:00 +01:00
return err
}
2023-01-13 19:23:36 +01:00
2023-01-15 15:55:47 +01:00
fmt.Fprint(w, color.BlueString("\t done (copied %s in %s)\n", FormatBytes(written), time.Since(start).Truncate(100*time.Millisecond)))
2023-01-13 19:23:36 +01:00
return nil
}