Files
Scriberr/internal/cli/watch.go

121 lines
2.3 KiB
Go

package cli
import (
"fmt"
"log"
"os"
"path/filepath"
"strings"
"sync"
"time"
"github.com/fsnotify/fsnotify"
"github.com/spf13/cobra"
)
var watchCmd = &cobra.Command{
Use: "watch [folder]",
Short: "Watch a folder for new audio files",
Args: cobra.ExactArgs(1),
Run: runWatch,
}
func init() {
rootCmd.AddCommand(watchCmd)
}
func runWatch(cmd *cobra.Command, args []string) {
folder := args[0]
absPath, err := filepath.Abs(folder)
if err != nil {
log.Fatalf("Failed to get absolute path: %v", err)
}
if _, err := os.Stat(absPath); os.IsNotExist(err) {
log.Fatalf("Folder does not exist: %s", absPath)
}
// Save as current watch folder
if _, err := SaveConfig("", "", absPath); err != nil {
fmt.Printf("Warning: Failed to save watch folder to config: %v\n", err)
}
watchFolder(absPath)
}
func watchFolder(path string) {
watcher, err := fsnotify.NewWatcher()
if err != nil {
log.Fatal(err)
}
defer watcher.Close()
// Debounce map: filename -> timer
timers := make(map[string]*time.Timer)
var mu sync.Mutex
done := make(chan bool)
go func() {
for {
select {
case event, ok := <-watcher.Events:
if !ok {
return
}
// Only care about Write and Create events
if event.Op&fsnotify.Write == fsnotify.Write || event.Op&fsnotify.Create == fsnotify.Create {
// Check extension
ext := strings.ToLower(filepath.Ext(event.Name))
if !isAudioFile(ext) {
continue
}
mu.Lock()
if t, exists := timers[event.Name]; exists {
t.Stop()
}
// Set new timer for 2 seconds
timers[event.Name] = time.AfterFunc(2*time.Second, func() {
mu.Lock()
delete(timers, event.Name)
mu.Unlock()
log.Printf("Uploading %s...\n", event.Name)
if err := UploadFile(event.Name); err != nil {
log.Printf("Failed to upload %s: %v\n", event.Name, err)
} else {
log.Printf("Successfully uploaded %s\n", event.Name)
}
})
mu.Unlock()
}
case err, ok := <-watcher.Errors:
if !ok {
return
}
log.Println("error:", err)
}
}
}()
err = watcher.Add(path)
if err != nil {
log.Fatal(err)
}
log.Printf("Watching %s for new audio files...\n", path)
<-done
}
func isAudioFile(ext string) bool {
switch ext {
case ".mp3", ".wav", ".m4a", ".flac", ".ogg", ".aac", ".wma":
return true
default:
return false
}
}