移除dhowden/tag,使用ffprobe替代
All checks were successful
Go CI / test-and-build (push) Successful in 1m1s
Web CI / lint-test-build (push) Successful in 22s

This commit is contained in:
2026-04-06 16:10:34 +08:00
parent fcb5732b40
commit 744fa578f1
4 changed files with 51 additions and 30 deletions

1
go.mod
View File

@@ -3,7 +3,6 @@ module butterfliu
go 1.23.5 go 1.23.5
require ( require (
github.com/dhowden/tag v0.0.0-20240417053706-3d75831295e8
github.com/go-chi/chi/v5 v5.2.3 github.com/go-chi/chi/v5 v5.2.3
github.com/mattn/go-sqlite3 v1.14.32 github.com/mattn/go-sqlite3 v1.14.32
) )

2
go.sum
View File

@@ -1,5 +1,3 @@
github.com/dhowden/tag v0.0.0-20240417053706-3d75831295e8 h1:OtSeLS5y0Uy01jaKK4mA/WVIYtpzVm63vLVAPzJXigg=
github.com/dhowden/tag v0.0.0-20240417053706-3d75831295e8/go.mod h1:apkPC/CR3s48O2D7Y++n1XWEpgPNNCjXYga3PPbJe2E=
github.com/go-chi/chi/v5 v5.2.3 h1:WQIt9uxdsAbgIYgid+BpYc+liqQZGMHRaUwp0JUcvdE= github.com/go-chi/chi/v5 v5.2.3 h1:WQIt9uxdsAbgIYgid+BpYc+liqQZGMHRaUwp0JUcvdE=
github.com/go-chi/chi/v5 v5.2.3/go.mod h1:L2yAIGWB3H+phAw1NxKwWM+7eUH/lU8pOMm5hHcoops= github.com/go-chi/chi/v5 v5.2.3/go.mod h1:L2yAIGWB3H+phAw1NxKwWM+7eUH/lU8pOMm5hHcoops=
github.com/mattn/go-sqlite3 v1.14.32 h1:JD12Ag3oLy1zQA+BNn74xRgaBbdhbNIDYvQUEuuErjs= github.com/mattn/go-sqlite3 v1.14.32 h1:JD12Ag3oLy1zQA+BNn74xRgaBbdhbNIDYvQUEuuErjs=

View File

@@ -1,12 +1,14 @@
package scanner package scanner
import ( import (
"encoding/json"
"fmt" "fmt"
"os" "os"
"os/exec"
"path/filepath" "path/filepath"
"slices"
"strconv"
"strings" "strings"
"github.com/dhowden/tag"
) )
type ScannedSong struct { type ScannedSong struct {
@@ -17,13 +19,28 @@ type ScannedSong struct {
Path string Path string
} }
// ffprobeFormat holds the JSON output from ffprobe format section
type ffprobeFormat struct {
Tags map[string]string `json:"tags"`
Duration string `json:"duration"`
}
// ffprobeOutput holds the full ffprobe JSON output
type ffprobeOutput struct {
Format ffprobeFormat `json:"format"`
}
func ScanDirectory(dirPath string) ([]ScannedSong, error) { func ScanDirectory(dirPath string) ([]ScannedSong, error) {
songs := []ScannedSong{} songs := []ScannedSong{}
if _, err := os.Stat(dirPath); os.IsNotExist(err) { if _, err := exec.LookPath("ffprobe"); err != nil {
return songs, err return songs, fmt.Errorf("ffprobe not found in PATH: %v", err)
}
err := filepath.Walk(dirPath, func(path string, info os.FileInfo, err error) error {
if err != nil {
return err
} }
filepath.Walk(dirPath, func(path string, info os.FileInfo, err error) error {
if info.IsDir() { if info.IsDir() {
return nil return nil
@@ -34,48 +51,61 @@ func ScanDirectory(dirPath string) ([]ScannedSong, error) {
} }
if song, err := processAudioFile(path); err != nil { if song, err := processAudioFile(path); err != nil {
return err // Log error but continue scanning other files
return nil
} else { } else {
songs = append(songs, song) songs = append(songs, song)
} }
return nil return nil
}) })
return songs, nil return songs, err
} }
func processAudioFile(filePath string) (ScannedSong, error) { func processAudioFile(filePath string) (ScannedSong, error) {
file, err := os.Open(filePath) cmd := exec.Command("ffprobe", "-v", "quiet", "-print_format", "json", "-show_format", filePath)
output, err := cmd.Output()
if err != nil { if err != nil {
return ScannedSong{}, fmt.Errorf("failed to open file: %v", err) return ScannedSong{}, fmt.Errorf("ffprobe failed for %s: %v", filePath, err)
}
defer file.Close()
metadata, err := tag.ReadFrom(file)
if err != nil {
return ScannedSong{}, fmt.Errorf("failed to read metadata: %v", err)
} }
title := metadata.Title() var probeOutput ffprobeOutput
if err := json.Unmarshal(output, &probeOutput); err != nil {
return ScannedSong{}, fmt.Errorf("failed to parse ffprobe output for %s: %v", filePath, err)
}
tags := probeOutput.Format.Tags
if tags == nil {
tags = make(map[string]string)
}
title := tags["title"]
if title == "" { if title == "" {
title = filepath.Base(filePath) title = filepath.Base(filePath)
} }
artist := metadata.Artist() artist := tags["artist"]
if artist == "" { if artist == "" {
artist = "Unknown Artist" artist = "Unknown Artist"
} }
album := metadata.Album() album := tags["album"]
if album == "" { if album == "" {
album = "Unknown Album" album = "Unknown Album"
} }
duration := 0
if probeOutput.Format.Duration != "" {
if d, err := strconv.ParseFloat(probeOutput.Format.Duration, 64); err == nil {
duration = int(d)
}
}
return ScannedSong{ return ScannedSong{
Title: title, Title: title,
Artist: artist, Artist: artist,
Album: album, Album: album,
Duration: 100, Duration: duration,
Path: filePath, Path: filePath,
}, nil }, nil
} }
@@ -83,11 +113,5 @@ func processAudioFile(filePath string) (ScannedSong, error) {
func isAudioFile(path string) bool { func isAudioFile(path string) bool {
ext := strings.ToLower(filepath.Ext(path)) ext := strings.ToLower(filepath.Ext(path))
audioExtensions := []string{".mp3", ".flac", ".m4a", ".wav", ".ogg"} audioExtensions := []string{".mp3", ".flac", ".m4a", ".wav", ".ogg"}
return slices.Contains(audioExtensions, ext)
for _, audioExt := range audioExtensions {
if ext == audioExt {
return true
}
}
return false
} }

View File

@@ -64,7 +64,7 @@
<div class="tech-grid"> <div class="tech-grid">
<div class="tech-item"> <div class="tech-item">
<h3>后端</h3> <h3>后端</h3>
<p>Go 1.23 · chi 路由 · SQLite · dhowden/tag</p> <p>Go 1.23 · chi 路由 · SQLite · ffprobe</p>
</div> </div>
<div class="tech-item"> <div class="tech-item">
<h3>前端</h3> <h3>前端</h3>