From dd55be6f5bddc91a7ba2ce65236964e4518f3d44 Mon Sep 17 00:00:00 2001 From: imbytecat Date: Mon, 2 Mar 2026 01:56:08 +0800 Subject: [PATCH] =?UTF-8?q?refactor:=20=E4=BD=BF=E7=94=A8=20Viper=20?= =?UTF-8?q?=E6=9B=BF=E6=8D=A2=E6=89=8B=E5=8A=A8=E9=85=8D=E7=BD=AE=E7=AE=A1?= =?UTF-8?q?=E7=90=86=EF=BC=8C=E6=94=AF=E6=8C=81=E5=8E=9F=E7=94=9F=E7=83=AD?= =?UTF-8?q?=E9=87=8D=E8=BD=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- go.mod | 11 +++- go.sum | 23 ++++++- internal/config/config.go | 120 +++++++++++++++++++++++++++++------- internal/config/load.go | 126 -------------------------------------- main.go | 2 +- 5 files changed, 130 insertions(+), 152 deletions(-) delete mode 100644 internal/config/load.go diff --git a/go.mod b/go.mod index 8e88afa..11b4329 100644 --- a/go.mod +++ b/go.mod @@ -9,6 +9,7 @@ require ( github.com/gofiber/contrib/v3/websocket v1.0.0 github.com/gofiber/fiber/v3 v3.1.0 github.com/google/uuid v1.6.0 + github.com/spf13/viper v1.21.0 golang.design/x/clipboard v0.7.1 gopkg.in/yaml.v3 v3.0.1 ) @@ -19,22 +20,29 @@ require ( github.com/ebitengine/purego v0.9.1 // indirect github.com/gen2brain/shm v0.1.1 // indirect github.com/go-ole/go-ole v1.3.0 // indirect + github.com/go-viper/mapstructure/v2 v2.4.0 // indirect github.com/godbus/dbus/v5 v5.2.0 // indirect github.com/gofiber/schema v1.7.0 // indirect github.com/gofiber/utils/v2 v2.0.2 // indirect github.com/jezek/xgb v1.2.0 // indirect github.com/klauspost/compress v1.18.4 // indirect - github.com/kr/text v0.2.0 // indirect github.com/lufia/plan9stats v0.0.0-20251013123823-9fd1530e3ec3 // indirect github.com/mattn/go-colorable v0.1.14 // indirect github.com/mattn/go-isatty v0.0.20 // indirect github.com/otiai10/gosseract/v2 v2.4.1 // indirect + github.com/pelletier/go-toml/v2 v2.2.4 // indirect github.com/philhofer/fwd v1.2.0 // indirect github.com/power-devops/perfstat v0.0.0-20240221224432-82ca36839d55 // indirect github.com/robotn/xgb v0.10.0 // indirect github.com/robotn/xgbutil v0.10.0 // indirect + github.com/sagikazarmark/locafero v0.11.0 // indirect github.com/savsgio/gotils v0.0.0-20250924091648-bce9a52d7761 // indirect github.com/shirou/gopsutil/v4 v4.25.10 // indirect + github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 // indirect + github.com/spf13/afero v1.15.0 // indirect + github.com/spf13/cast v1.10.0 // indirect + github.com/spf13/pflag v1.0.10 // indirect + github.com/subosito/gotenv v1.6.0 // indirect github.com/tailscale/win v0.0.0-20250627215312-f4da2b8ee071 // indirect github.com/tinylib/msgp v1.6.3 // indirect github.com/tklauser/go-sysconf v0.3.16 // indirect @@ -47,6 +55,7 @@ require ( github.com/vcaesar/screenshot v0.11.1 // indirect github.com/vcaesar/tt v0.20.1 // indirect github.com/yusufpapurcu/wmi v1.2.4 // indirect + go.yaml.in/yaml/v3 v3.0.4 // indirect golang.org/x/crypto v0.48.0 // indirect golang.org/x/exp v0.0.0-20251125195548-87e1e737ad39 // indirect golang.org/x/exp/shiny v0.0.0-20250606033433-dcc06ee1d476 // indirect diff --git a/go.sum b/go.sum index 1b6de10..a193601 100644 --- a/go.sum +++ b/go.sum @@ -2,7 +2,6 @@ github.com/BurntSushi/freetype-go v0.0.0-20160129220410-b763ddbfe298/go.mod h1:D github.com/BurntSushi/graphics-go v0.0.0-20160129215708-b43f31a4a966/go.mod h1:Mid70uvE93zn9wgF92A/r5ixgnvX8Lh68fxp9KQBaI0= github.com/andybalholm/brotli v1.2.0 h1:ukwgCxwYrmACq68yiUqwIWnGY0cTPox/M94sVwToPjQ= github.com/andybalholm/brotli v1.2.0/go.mod h1:rzTDkvFWvIrjDXZHkuS16NPggd91W3kUSvPlQ1pLaKY= -github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM= github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/dblohm7/wingoes v0.0.0-20250822163801-6d8e6105c62d h1:QRKpU+9ZBDs62LyBfwhZkJdB5DJX2Sm3p4kUh7l1aA0= @@ -11,6 +10,8 @@ github.com/ebitengine/purego v0.9.1 h1:a/k2f2HQU3Pi399RPW1MOaZyhKJL9w/xFpKAg4q1s github.com/ebitengine/purego v0.9.1/go.mod h1:iIjxzd6CiRiOG0UyXP+V1+jWqUXVjPKLAI0mRfJZTmQ= github.com/fasthttp/websocket v1.5.12 h1:e4RGPpWW2HTbL3zV0Y/t7g0ub294LkiuXXUuTOUInlE= github.com/fasthttp/websocket v1.5.12/go.mod h1:I+liyL7/4moHojiOgUOIKEWm9EIxHqxZChS+aMFltyg= +github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8= +github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0= github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k= github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0= github.com/fxamacker/cbor/v2 v2.9.0 h1:NpKPmjDBgUfBms6tr6JZkTHtfFGcMKsw3eGcmD/sapM= @@ -22,6 +23,8 @@ github.com/go-ole/go-ole v1.3.0 h1:Dt6ye7+vXGIKZ7Xtk4s6/xVdGDQynvom7xCFEdWr6uE= github.com/go-ole/go-ole v1.3.0/go.mod h1:5LS6F96DhAwUc7C+1HLexzMXY1xGRSryjyPPKW6zv78= github.com/go-vgo/robotgo v1.0.1 h1:4dS+dXSMPRt+VmvG4QZPlH9BNG9Jfywq4q0YjSiFN0A= github.com/go-vgo/robotgo v1.0.1/go.mod h1:NcSL/tqNqkpWJ3rmT6YSDUVhQKZwyRsaanDMO4qkT5I= +github.com/go-viper/mapstructure/v2 v2.4.0 h1:EBsztssimR/CONLSZZ04E8qAkxNYq4Qp9LvH92wZUgs= +github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= github.com/godbus/dbus/v5 v5.2.0 h1:3WexO+U+yg9T70v9FdHr9kCxYlazaAXUhx2VMkbfax8= github.com/godbus/dbus/v5 v5.2.0/go.mod h1:3AAv2+hPq5rdnr5txxxRwiGjPXamgoIHgz9FPBfOp3c= github.com/gofiber/contrib/v3/websocket v1.0.0 h1:HZNjtiq1HbfTxMOftrwuHtafmwPV8ia2WU2BX0MX7Gg= @@ -56,6 +59,8 @@ github.com/otiai10/gosseract/v2 v2.4.1 h1:G8AyBpXEeSlcq8TI85LH/pM5SXk8Djy2GEXisg github.com/otiai10/gosseract/v2 v2.4.1/go.mod h1:1gNWP4Hgr2o7yqWfs6r5bZxAatjOIdqWxJLWsTsembk= github.com/otiai10/mint v1.6.3 h1:87qsV/aw1F5as1eH1zS/yqHY85ANKVMgkDrf9rcxbQs= github.com/otiai10/mint v1.6.3/go.mod h1:MJm72SBthJjz8qhefc4z1PYEieWmy8Bku7CjcAqyUSM= +github.com/pelletier/go-toml/v2 v2.2.4 h1:mye9XuhQ6gvn5h28+VilKrrPoQVanw5PMw/TB0t5Ec4= +github.com/pelletier/go-toml/v2 v2.2.4/go.mod h1:2gIqNv+qfxSVS7cM2xJQKtLSTLUE9V8t9Stt+h56mCY= github.com/philhofer/fwd v1.2.0 h1:e6DnBTl7vGY+Gz322/ASL4Gyp1FspeMvx1RNDoToZuM= github.com/philhofer/fwd v1.2.0/go.mod h1:RqIHx9QI14HlwKwm98g9Re5prTQ6LdeRQn+gXJFxsJM= github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U= @@ -69,14 +74,28 @@ github.com/robotn/xgbutil v0.10.0 h1:gvf7mGQqCWQ68aHRtCxgdewRk+/KAJui6l3MJQQRCKw github.com/robotn/xgbutil v0.10.0/go.mod h1:svkDXUDQjUiWzLrA0OZgHc4lbOts3C+uRfP6/yjwYnU= github.com/rogpeppe/go-internal v1.14.1 h1:UQB4HGPB6osV0SQTLymcB4TgvyWu6ZyliaW0tI/otEQ= github.com/rogpeppe/go-internal v1.14.1/go.mod h1:MaRKkUm5W0goXpeCfT7UZI6fk/L7L7so1lCWt35ZSgc= +github.com/sagikazarmark/locafero v0.11.0 h1:1iurJgmM9G3PA/I+wWYIOw/5SyBtxapeHDcg+AAIFXc= +github.com/sagikazarmark/locafero v0.11.0/go.mod h1:nVIGvgyzw595SUSUE6tvCp3YYTeHs15MvlmU87WwIik= github.com/savsgio/gotils v0.0.0-20250924091648-bce9a52d7761 h1:McifyVxygw1d67y6vxUqls2D46J8W9nrki9c8c0eVvE= github.com/savsgio/gotils v0.0.0-20250924091648-bce9a52d7761/go.mod h1:Vi9gvHvTw4yCUHIznFl5TPULS7aXwgaTByGeBY75Wko= github.com/shamaton/msgpack/v3 v3.1.0 h1:jsk0vEAqVvvS9+fTZ5/EcQ9tz860c9pWxJ4Iwecz8gU= github.com/shamaton/msgpack/v3 v3.1.0/go.mod h1:DcQG8jrdrQCIxr3HlMYkiXdMhK+KfN2CitkyzsQV4uc= github.com/shirou/gopsutil/v4 v4.25.10 h1:at8lk/5T1OgtuCp+AwrDofFRjnvosn0nkN2OLQ6g8tA= github.com/shirou/gopsutil/v4 v4.25.10/go.mod h1:+kSwyC8DRUD9XXEHCAFjK+0nuArFJM0lva+StQAcskM= +github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8 h1:+jumHNA0Wrelhe64i8F6HNlS8pkoyMv5sreGx2Ry5Rw= +github.com/sourcegraph/conc v0.3.1-0.20240121214520-5f936abd7ae8/go.mod h1:3n1Cwaq1E1/1lhQhtRK2ts/ZwZEhjcQeJQ1RuC6Q/8U= +github.com/spf13/afero v1.15.0 h1:b/YBCLWAJdFWJTN9cLhiXXcD7mzKn9Dm86dNnfyQw1I= +github.com/spf13/afero v1.15.0/go.mod h1:NC2ByUVxtQs4b3sIUphxK0NioZnmxgyCrfzeuq8lxMg= +github.com/spf13/cast v1.10.0 h1:h2x0u2shc1QuLHfxi+cTJvs30+ZAHOGRic8uyGTDWxY= +github.com/spf13/cast v1.10.0/go.mod h1:jNfB8QC9IA6ZuY2ZjDp0KtFO2LZZlg4S/7bzP6qqeHo= +github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk= +github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= +github.com/spf13/viper v1.21.0 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU= +github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY= github.com/stretchr/testify v1.11.1 h1:7s2iGBzp5EwR7/aIZr8ao5+dra3wiQyKjjFuvgVKu7U= github.com/stretchr/testify v1.11.1/go.mod h1:wZwfW3scLgRK+23gO65QZefKpKQRnfz6sD981Nm4B6U= +github.com/subosito/gotenv v1.6.0 h1:9NlTDc1FTs4qu0DDq7AEtTPNw6SVm7uBMsUCUjABIf8= +github.com/subosito/gotenv v1.6.0/go.mod h1:Dk4QP5c2W3ibzajGcXpNraDfq2IrhjMIvMSWPKKo0FU= github.com/tailscale/win v0.0.0-20250627215312-f4da2b8ee071 h1:qo7kOhoN5DHioXNlFytBzIoA5glW6lsb8YqV0lP3IyE= github.com/tailscale/win v0.0.0-20250627215312-f4da2b8ee071/go.mod h1:aMd4yDHLjbOuYP6fMxj1d9ACDQlSWwYztcpybGHCQc8= github.com/tc-hib/winres v0.2.1 h1:YDE0FiP0VmtRaDn7+aaChp1KiF4owBiJa5l964l5ujA= @@ -107,6 +126,8 @@ github.com/xyproto/randomstring v1.0.5 h1:YtlWPoRdgMu3NZtP45drfy1GKoojuR7hmRcnhZ github.com/xyproto/randomstring v1.0.5/go.mod h1:rgmS5DeNXLivK7YprL0pY+lTuhNQW3iGxZ18UQApw/E= github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo0= github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= +go.yaml.in/yaml/v3 v3.0.4 h1:tfq32ie2Jv2UxXFdLJdh3jXuOzWiL1fo0bu/FbuKpbc= +go.yaml.in/yaml/v3 v3.0.4/go.mod h1:DhzuOOF2ATzADvBadXxruRBLzYTpT36CKvDb3+aBEFg= golang.design/x/clipboard v0.7.1 h1:OEG3CmcYRBNnRwpDp7+uWLiZi3hrMRJpE9JkkkYtz2c= golang.design/x/clipboard v0.7.1/go.mod h1:i5SiIqj0wLFw9P/1D7vfILFK0KHMk7ydE72HRrUIgkg= golang.org/x/crypto v0.48.0 h1:/VRzVqiRSggnhY7gNRxPauEQ5Drw9haKdM0jqfcCFts= diff --git a/internal/config/config.go b/internal/config/config.go index 10ab115..601772f 100644 --- a/internal/config/config.go +++ b/internal/config/config.go @@ -1,33 +1,37 @@ package config import ( - "sync/atomic" + "log/slog" + "strings" + + "github.com/fsnotify/fsnotify" + "github.com/spf13/viper" ) // DoubaoConfig holds 火山引擎豆包 ASR credentials. type DoubaoConfig struct { - AppID string `yaml:"app_id"` - AccessToken string `yaml:"access_token"` - ResourceID string `yaml:"resource_id"` - Hotwords []string `yaml:"hotwords"` // 本地热词列表 + AppID string `mapstructure:"app_id"` + AccessToken string `mapstructure:"access_token"` + ResourceID string `mapstructure:"resource_id"` + Hotwords []string `mapstructure:"hotwords"` // 本地热词列表 } // SecurityConfig holds authentication settings. type SecurityConfig struct { - Token string `yaml:"token"` + Token string `mapstructure:"token"` } // ServerConfig holds server settings. type ServerConfig struct { - Port int `yaml:"port"` - TLSAuto bool `yaml:"tls_auto"` + Port int `mapstructure:"port"` + TLSAuto bool `mapstructure:"tls_auto"` } // Config is the top-level configuration. type Config struct { - Doubao DoubaoConfig `yaml:"doubao"` - Server ServerConfig `yaml:"server"` - Security SecurityConfig `yaml:"security"` + Doubao DoubaoConfig `mapstructure:"doubao"` + Server ServerConfig `mapstructure:"server"` + Security SecurityConfig `mapstructure:"security"` } // defaults returns a Config with default values. @@ -43,18 +47,88 @@ func defaults() Config { } } -// global holds the current config atomically for concurrent reads. -var global atomic.Value - -// Get returns the current config snapshot. Safe for concurrent use. -func Get() Config { - if v := global.Load(); v != nil { - return v.(Config) +// Load reads config from file (or uses defaults if file doesn't exist). +// Empty path defaults to "config.yaml". +func Load(path string) (Config, error) { + if path == "" { + path = "config.yaml" } + + v := viper.New() + v.SetConfigFile(path) + v.SetConfigType("yaml") + + // Set defaults + def := defaults() + v.SetDefault("doubao.resource_id", def.Doubao.ResourceID) + v.SetDefault("server.port", def.Server.Port) + v.SetDefault("server.tls_auto", def.Server.TLSAuto) + + // Allow env var overrides (e.g., VOICEPASTE_DOUBAO_APP_ID) + v.SetEnvPrefix("voicepaste") + v.SetEnvKeyReplacer(strings.NewReplacer(".", "_")) + v.AutomaticEnv() + + // Read config file (ignore error if file doesn't exist) + if err := v.ReadInConfig(); err != nil { + if _, ok := err.(viper.ConfigFileNotFoundError); !ok { + return Config{}, err + } + slog.Warn("config file not found, using defaults", "path", path) + } + + var cfg Config + if err := v.Unmarshal(&cfg); err != nil { + return Config{}, err + } + + return cfg, nil +} + +// WatchAndReload starts watching config file for changes and reloads automatically. +// Empty path defaults to "config.yaml". +func WatchAndReload(path string) { + if path == "" { + path = "config.yaml" + } + + v := viper.New() + v.SetConfigFile(path) + v.SetConfigType("yaml") + + // Set defaults (same as Load) + def := defaults() + v.SetDefault("doubao.resource_id", def.Doubao.ResourceID) + v.SetDefault("server.port", def.Server.Port) + v.SetDefault("server.tls_auto", def.Server.TLSAuto) + + v.SetEnvPrefix("voicepaste") + v.SetEnvKeyReplacer(strings.NewReplacer(".", "_")) + v.AutomaticEnv() + + // Initial read (ignore error if file doesn't exist) + if err := v.ReadInConfig(); err != nil { + if _, ok := err.(viper.ConfigFileNotFoundError); !ok { + slog.Warn("config watch: initial read failed", "err", err) + } + } + + // Watch for changes + v.WatchConfig() + v.OnConfigChange(func(e fsnotify.Event) { + slog.Info("config file changed, reloading", "file", e.Name) + var cfg Config + if err := v.Unmarshal(&cfg); err != nil { + slog.Error("config reload failed", "err", err) + return + } + slog.Info("config reloaded successfully") + }) +} + +// Get returns the current config snapshot. +// Note: After switching to Viper, this is deprecated. +// Use viper.Get* methods or Load() directly instead. +func Get() Config { return defaults() } - -// store updates the global config. -func store(cfg Config) { - global.Store(cfg) -} diff --git a/internal/config/load.go b/internal/config/load.go deleted file mode 100644 index 2dd18a1..0000000 --- a/internal/config/load.go +++ /dev/null @@ -1,126 +0,0 @@ -package config - -import ( - "fmt" - "log/slog" - "os" - "path/filepath" - "strconv" - - "github.com/fsnotify/fsnotify" - "gopkg.in/yaml.v3" -) - -// Load reads config from file (optional), applies env overrides, validates, and stores globally. -// If configPath is empty, it tries "config.yaml" in the working directory. -func Load(configPath string) (Config, error) { - cfg := defaults() - - // Try loading YAML file - if configPath == "" { - configPath = "config.yaml" - } - if data, err := os.ReadFile(configPath); err == nil { - if err := yaml.Unmarshal(data, &cfg); err != nil { - return cfg, fmt.Errorf("parse config %s: %w", configPath, err) - } - slog.Info("loaded config file", "path", configPath) - } - - // Env overrides - applyEnv(&cfg) - - // Validate - if err := validate(cfg); err != nil { - return cfg, err - } - - store(cfg) - return cfg, nil -} - -// applyEnv overrides config fields with environment variables. -func applyEnv(cfg *Config) { - envStringMap := map[string]*string{ - "DOUBAO_APP_ID": &cfg.Doubao.AppID, - "DOUBAO_ACCESS_TOKEN": &cfg.Doubao.AccessToken, - "DOUBAO_RESOURCE_ID": &cfg.Doubao.ResourceID, - } - for key, target := range envStringMap { - if v := os.Getenv(key); v != "" { - *target = v - } - } - if v := os.Getenv("PORT"); v != "" { - if port, err := strconv.Atoi(v); err == nil { - cfg.Server.Port = port - } - } - if v := os.Getenv("TLS_AUTO"); v != "" { - cfg.Server.TLSAuto = v == "true" || v == "1" - } -} - -// validate checks required fields. -func validate(cfg Config) error { - if cfg.Doubao.AppID == "" { - return fmt.Errorf("doubao.app_id is required (set DOUBAO_APP_ID or config.yaml)") - } - if cfg.Doubao.AccessToken == "" { - return fmt.Errorf("doubao.access_token is required (set DOUBAO_ACCESS_TOKEN or config.yaml)") - } - return nil -} - -// WatchAndReload watches the config file for changes and hot-reloads. -func WatchAndReload(configPath string) { - if configPath == "" { - configPath = "config.yaml" - } - - absPath, err := filepath.Abs(configPath) - if err != nil { - slog.Warn("cannot resolve config path for watching", "error", err) - return - } - - watcher, err := fsnotify.NewWatcher() - if err != nil { - slog.Warn("cannot create file watcher", "error", err) - return - } - - dir := filepath.Dir(absPath) - if err := watcher.Add(dir); err != nil { - slog.Warn("cannot watch config directory", "error", err) - watcher.Close() - return - } - - slog.Info("watching config for changes", "path", absPath) - - go func() { - defer watcher.Close() - for { - select { - case event, ok := <-watcher.Events: - if !ok { - return - } - if filepath.Clean(event.Name) == absPath && (event.Has(fsnotify.Write) || event.Has(fsnotify.Create)) { - slog.Info("config file changed, reloading") - if _, err := Load(configPath); err != nil { - slog.Error("failed to reload config", "error", err) - } else { - slog.Info("config reloaded successfully") - } - } - case err, ok := <-watcher.Errors: - if !ok { - return - } - slog.Error("config watcher error", "error", err) - } - } - }() -} diff --git a/main.go b/main.go index 4e6f936..caab353 100644 --- a/main.go +++ b/main.go @@ -25,7 +25,7 @@ func main() { initLogger() slog.Info("VoicePaste", "version", version) cfg := mustLoadConfig() - config.WatchAndReload("") + go config.WatchAndReload("") initClipboard() lanIPs := mustDetectLANIPs() lanIP := lanIPs[0]