mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 20:47:35 +03:00
* feat(build): add a make target to build a msi installer locally
* Testing wrapping the executable in cmd
* build(ci): build msis in parallel
* feat(server): add LogFile config option
* Revert "Testing wrapping the executable in cmd"
This reverts commit be29592254
.
* Adding --log-file for service executable
* feat(ini): wip
* feat(ini): parse nested ini section
* fix(conf): fix fatal error messages
* Now navidrome supports INI, we can use the built-in msi ini system
and not require the VBScript to convert it into toml
* File needs to be called .ini to be parsed as an INI and correct filename
needs to be passed to the service
* fix(msi): build msi locally
* fix(msi): pipeline
* fix(msi): pipeline
* fix(msi): pipeline
* fix(msi): pipeline
* fix(msi): pipeline
* fix(msi): Makefile
* fix(msi): more clean up
* fix(log): convert LF to CRLF on Windows
* fix(msi): config filename should be case-insensitive
* fix(msi): make it a little more idiomatic
* Including the latest windows release of ffmpeg into the msi
as built by https://www.gyan.dev/ffmpeg/builds/ (linked
to on the official ffmpeg source)
* This should version independent
* Need bash expansion for the * to work
* This will run twice, once for x86 and once for x64, I'll make it cache
the executable for now as it'll be quicker
* Silencing wget
* Add ffmpeg path to the config so Navidrome knows where to find it
* refactor: download ffmpeg from our repository
* When going back from the "Are you ready to install?" it should go back to the
Settings dialogue that you just came from
* fix: comments
---------
Co-authored-by: Deluan <deluan@navidrome.org>
218 lines
5.2 KiB
Go
218 lines
5.2 KiB
Go
package cmd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/kardianos/service"
|
|
"github.com/navidrome/navidrome/conf"
|
|
"github.com/navidrome/navidrome/log"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
var (
|
|
svcStatusLabels = map[service.Status]string{
|
|
service.StatusUnknown: "Unknown",
|
|
service.StatusStopped: "Stopped",
|
|
service.StatusRunning: "Running",
|
|
}
|
|
)
|
|
|
|
func init() {
|
|
svcCmd.AddCommand(buildInstallCmd())
|
|
svcCmd.AddCommand(buildUninstallCmd())
|
|
svcCmd.AddCommand(buildStartCmd())
|
|
svcCmd.AddCommand(buildStopCmd())
|
|
svcCmd.AddCommand(buildStatusCmd())
|
|
svcCmd.AddCommand(buildExecuteCmd())
|
|
rootCmd.AddCommand(svcCmd)
|
|
}
|
|
|
|
var svcCmd = &cobra.Command{
|
|
Use: "service",
|
|
Aliases: []string{"svc"},
|
|
Short: "Manage Navidrome as a service",
|
|
Long: fmt.Sprintf("Manage Navidrome as a service, using the OS service manager (%s)", service.Platform()),
|
|
Run: runServiceCmd,
|
|
}
|
|
|
|
type svcControl struct {
|
|
ctx context.Context
|
|
cancel context.CancelFunc
|
|
done chan struct{}
|
|
}
|
|
|
|
func (p *svcControl) Start(service.Service) error {
|
|
p.done = make(chan struct{})
|
|
p.ctx, p.cancel = context.WithCancel(context.Background())
|
|
go func() {
|
|
runNavidrome(p.ctx)
|
|
close(p.done)
|
|
}()
|
|
return nil
|
|
}
|
|
|
|
func (p *svcControl) Stop(service.Service) error {
|
|
log.Info("Stopping service")
|
|
p.cancel()
|
|
select {
|
|
case <-p.done:
|
|
log.Info("Service stopped gracefully")
|
|
case <-time.After(10 * time.Second):
|
|
log.Error("Service did not stop in time. Killing it.")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
var svcInstance = sync.OnceValue(func() service.Service {
|
|
options := make(service.KeyValue)
|
|
options["Restart"] = "on-success"
|
|
options["SuccessExitStatus"] = "1 2 8 SIGKILL"
|
|
options["UserService"] = false
|
|
if conf.Server.LogFile != "" {
|
|
options["LogOutput"] = false
|
|
} else {
|
|
options["LogOutput"] = true
|
|
options["LogDirectory"] = conf.Server.DataFolder
|
|
}
|
|
svcConfig := &service.Config{
|
|
Name: "navidrome",
|
|
DisplayName: "Navidrome",
|
|
Description: "Your Personal Streaming Service",
|
|
Dependencies: []string{
|
|
"Requires=",
|
|
"After="},
|
|
WorkingDirectory: executablePath(),
|
|
Option: options,
|
|
}
|
|
arguments := []string{"service", "execute"}
|
|
if conf.Server.ConfigFile != "" {
|
|
arguments = append(arguments, "-c", conf.Server.ConfigFile)
|
|
}
|
|
svcConfig.Arguments = arguments
|
|
|
|
prg := &svcControl{}
|
|
svc, err := service.New(prg, svcConfig)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
return svc
|
|
})
|
|
|
|
func runServiceCmd(cmd *cobra.Command, _ []string) {
|
|
_ = cmd.Help()
|
|
}
|
|
|
|
func executablePath() string {
|
|
ex, err := os.Executable()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
return filepath.Dir(ex)
|
|
}
|
|
|
|
func buildInstallCmd() *cobra.Command {
|
|
runInstallCmd := func(_ *cobra.Command, _ []string) {
|
|
var err error
|
|
println("Installing service with:")
|
|
println(" working directory: " + executablePath())
|
|
println(" music folder: " + conf.Server.MusicFolder)
|
|
println(" data folder: " + conf.Server.DataFolder)
|
|
if conf.Server.LogFile != "" {
|
|
println(" log file: " + conf.Server.LogFile)
|
|
} else {
|
|
println(" logs folder: " + conf.Server.DataFolder)
|
|
}
|
|
if cfgFile != "" {
|
|
conf.Server.ConfigFile, err = filepath.Abs(cfgFile)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
println(" config file: " + conf.Server.ConfigFile)
|
|
}
|
|
err = svcInstance().Install()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
println("Service installed. Use 'navidrome svc start' to start it.")
|
|
}
|
|
|
|
return &cobra.Command{
|
|
Use: "install",
|
|
Short: "Install Navidrome service.",
|
|
Run: runInstallCmd,
|
|
}
|
|
}
|
|
|
|
func buildUninstallCmd() *cobra.Command {
|
|
return &cobra.Command{
|
|
Use: "uninstall",
|
|
Short: "Uninstall Navidrome service. Does not delete the music or data folders",
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
err := svcInstance().Uninstall()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
println("Service uninstalled. Music and data folders are still intact.")
|
|
},
|
|
}
|
|
}
|
|
|
|
func buildStartCmd() *cobra.Command {
|
|
return &cobra.Command{
|
|
Use: "start",
|
|
Short: "Start Navidrome service",
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
err := svcInstance().Start()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
println("Service started. Use 'navidrome svc status' to check its status.")
|
|
},
|
|
}
|
|
}
|
|
|
|
func buildStopCmd() *cobra.Command {
|
|
return &cobra.Command{
|
|
Use: "stop",
|
|
Short: "Stop Navidrome service",
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
err := svcInstance().Stop()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
println("Service stopped. Use 'navidrome svc status' to check its status.")
|
|
},
|
|
}
|
|
}
|
|
|
|
func buildStatusCmd() *cobra.Command {
|
|
return &cobra.Command{
|
|
Use: "status",
|
|
Short: "Show Navidrome service status",
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
status, err := svcInstance().Status()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
fmt.Printf("Navidrome is %s.\n", svcStatusLabels[status])
|
|
},
|
|
}
|
|
}
|
|
|
|
func buildExecuteCmd() *cobra.Command {
|
|
return &cobra.Command{
|
|
Use: "execute",
|
|
Short: "Run navidrome as a service in the foreground (it is very unlikely you want to run this, you are better off running just navidrome)",
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
err := svcInstance().Run()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
},
|
|
}
|
|
}
|