Replace the WebSocket + Cloudflare Durable Object architecture with a single POST /sync endpoint. The CLI now operates autonomously with local state (tasks.json) and syncs bidirectionally via adaptive-interval HTTP polling (3s watching, 60s idle). - Remove transport_ws, transport_hybrid, transport_http (~2,600 lines) - Add SyncClient with adaptive interval loop - Add LocalState for CLI-side task persistence - Add TaskStateFromUpdate() helper (DRY) - Extract finalize() to deduplicate processTask/processTaskRetry - Consolidate shortID() into agent.ShortID (was in 3 packages) - Wire GetActiveCount so `unarr status` shows active tasks - Remove poll_interval, heartbeat_interval, ws_url from config - Simplify ProgressReporter (sync replaces direct HTTP reporting)
40 lines
915 B
Go
40 lines
915 B
Go
//go:build !windows
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"log"
|
|
"os"
|
|
"os/signal"
|
|
"syscall"
|
|
|
|
"github.com/torrentclaw/unarr/internal/agent"
|
|
"github.com/torrentclaw/unarr/internal/config"
|
|
)
|
|
|
|
// ReloadableConfig holds a reference to the daemon for hot-reload.
|
|
type ReloadableConfig struct {
|
|
Daemon *agent.Daemon
|
|
}
|
|
|
|
// startReloadWatcher listens for SIGUSR1 and reloads config.
|
|
// With the sync-based architecture, intervals are fixed (3s watching, 60s idle).
|
|
// Hot-reload now mainly serves as a signal to re-read config for future settings.
|
|
func startReloadWatcher(rc *ReloadableConfig) {
|
|
sigCh := make(chan os.Signal, 1)
|
|
signal.Notify(sigCh, syscall.SIGUSR1)
|
|
|
|
go func() {
|
|
for range sigCh {
|
|
log.Println("Received SIGUSR1, reloading config...")
|
|
|
|
_, err := config.Load("")
|
|
if err != nil {
|
|
log.Printf("Config reload failed: %v", err)
|
|
continue
|
|
}
|
|
|
|
log.Println("Config reloaded successfully")
|
|
}
|
|
}()
|
|
}
|