Files
ripsort/internal/sorter.go

194 lines
4.0 KiB
Go

package ripsort
import (
"fmt"
"io"
"log/slog"
"os"
"path/filepath"
"regexp"
"strings"
"git.kapelle.org/niklas/ripsort/internal/metadata"
)
func fileSupported(file string) bool {
supported := map[string]bool{
".mp3": true, ".flac": true,
".m4a": true, ".ogg": true, ".dsf": true,
}
return supported[strings.ToLower(filepath.Ext(file))]
}
func sortSong(src, dst string, updateMeta bool) error {
m, err := metadata.ReadAudioTags(src)
if err != nil {
return err
}
dstPath := pathForFile(src, *m)
finalPath := filepath.Join(dst, dstPath)
if updateMeta {
slog.Info("Copying song with updated metadata", "file", src, "dst", dstPath)
err = copyFileUpdateMetadata(src, finalPath, *m)
if err != nil {
return err
}
} else {
slog.Info("Copying song", "file", src, "dst", dstPath)
err = copyFile(src, finalPath)
if err != nil {
return err
}
}
extraFiles := checkForExtraFiles(src)
for _, extraFile := range extraFiles {
extraFilePath := pathForFile(extraFile, *m)
finalExtraFilePath := filepath.Join(dst, extraFilePath)
slog.Info("Copy extra file", "file", extraFile, "dst", extraFilePath)
err = copyFile(extraFile, finalExtraFilePath)
if err != nil {
slog.Warn("Failed to copy extra file", "file", extraFile, "err", err)
}
}
return nil
}
func sanitizeName(name *string) *string {
if name == nil || *name == "" {
return nil
}
re := regexp.MustCompile(`[<>:"/\\|?*\x00-\x1F]`)
dName := re.ReplaceAllString(*name, "_")
dName = strings.Trim(dName, " .")
if dName == "" {
return nil
}
return &dName
}
func getArtistName(m metadata.Metadata) string {
var artist *string
if len(m.Artist) > 0 {
artist = sanitizeName(&m.Artist[0])
}
if artist == nil && len(m.AlbumArtist) > 0 {
if aa := m.AlbumArtist[0]; aa != "" {
artist = sanitizeName(&aa)
}
}
if artist == nil {
return "Unknown Artist"
}
return *artist
}
func getAlbumName(m metadata.Metadata) string {
album := sanitizeName(m.Album)
if album == nil {
return "Unknown Album"
}
return *album
}
func getTitle(src string, m metadata.Metadata) string {
title := sanitizeName(m.Title)
if title == nil {
filename := strings.TrimSuffix(filepath.Base(src), filepath.Ext(src))
return *sanitizeName(&filename)
}
return *title
}
func pathForFile(src string, m metadata.Metadata) string {
ext := strings.ToLower(filepath.Ext(src))
artist := getArtistName(m)
album := getAlbumName(m)
title := getTitle(src, m)
filename := title + ext
return filepath.Join(artist, album, filename)
}
func copyFileUpdateMetadata(src, dst string, m metadata.Metadata) error {
ext := strings.ToLower(filepath.Ext(src))
if ext != ".flac" {
slog.Warn("Copying file without updating metadata. Unsupported format", "file", src)
return copyFile(src, dst)
}
dir := filepath.Dir(dst)
if err := os.MkdirAll(dir, 0o755); err != nil {
return fmt.Errorf("create destination directory: %w", err)
}
return metadata.UpdateMetadata(src, dst, m)
}
func copyFile(src, dst string) error {
in, err := os.Open(src)
if err != nil {
return fmt.Errorf("open source file: %w", err)
}
defer in.Close()
dir := filepath.Dir(dst)
if err := os.MkdirAll(dir, 0o755); err != nil {
return fmt.Errorf("create destination directory: %w", err)
}
out, err := os.Create(dst)
if err != nil {
return fmt.Errorf("create destination file: %w", err)
}
defer func() {
if cerr := out.Close(); err == nil && cerr != nil {
err = cerr
}
}()
if _, err = io.Copy(out, in); err != nil {
return fmt.Errorf("copy file contents: %w", err)
}
if err = out.Sync(); err != nil {
return fmt.Errorf("sync destination file: %w", err)
}
return nil
}
func checkForExtraFiles(originalPath string) []string {
dir := filepath.Dir(originalPath)
fileName := filepath.Base(originalPath)
fileNameNoExt := strings.TrimSuffix(fileName, filepath.Ext(fileName))
extraFiles := []string{}
lrcPath := filepath.Join(dir, fileNameNoExt+".lrc")
stat, err := os.Stat(lrcPath)
if err == nil && !stat.IsDir() {
extraFiles = append(extraFiles, lrcPath)
}
return extraFiles
}