mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-01 19:47:37 +03:00
* support packing deb/rpm/archlinux * .-. * initial test * fix postinstall, remove execstop * bash -> sh, create toml manually if it doesn't exist (thanks debian) * don't forget that newline * postrm * comments, contrib -> packaging/linux * contrib > packaging in .goreleaser * actually add toml * openrc/sysv templates * add apk. nothing else yet * wait, we have a ntive uninstall * fix: merge errors, move packaging to release * chore: remove old goreleaser conf * ci: remove `release` dependency on `docker push` * ci: fix release version * ci: upload packages * ci: try to fix json file list * ci: replace the json file list with a txt artifact * postremove -> preremove, skip install/remove error * actually do preremove * better preremove * ci: fix * ci: fix? * ci: clean-up * ci: try to change labels and filenames * ci: fix? * ci: fix? * ci: add `make package` target * ci: make labels more readable hope it doesn't break the pipeline again * build: remove alpine and archlinux packages, for now. --------- Co-authored-by: Deluan <deluan@navidrome.org>
267 lines
6.6 KiB
Go
267 lines
6.6 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",
|
|
}
|
|
|
|
installUser string
|
|
workingDirectory string
|
|
)
|
|
|
|
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-failure"
|
|
options["SuccessExitStatus"] = "1 2 8 SIGKILL"
|
|
options["UserService"] = false
|
|
options["LogDirectory"] = conf.Server.DataFolder
|
|
options["SystemdScript"] = systemdScript
|
|
if conf.Server.LogFile != "" {
|
|
options["LogOutput"] = false
|
|
} else {
|
|
options["LogOutput"] = true
|
|
options["LogDirectory"] = conf.Server.DataFolder
|
|
}
|
|
svcConfig := &service.Config{
|
|
UserName: installUser,
|
|
Name: "navidrome",
|
|
DisplayName: "Navidrome",
|
|
Description: "Your Personal Streaming Service",
|
|
Dependencies: []string{
|
|
"After=remote-fs.target network.target",
|
|
},
|
|
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 {
|
|
if workingDirectory != "" {
|
|
return workingDirectory
|
|
}
|
|
|
|
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.")
|
|
}
|
|
|
|
cmd := &cobra.Command{
|
|
Use: "install",
|
|
Short: "Install Navidrome service.",
|
|
Run: runInstallCmd,
|
|
}
|
|
cmd.Flags().StringVarP(&installUser, "user", "u", "", "user to run service")
|
|
cmd.Flags().StringVarP(&workingDirectory, "working-directory", "w", "", "working directory of service")
|
|
|
|
return cmd
|
|
}
|
|
|
|
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)
|
|
}
|
|
},
|
|
}
|
|
}
|
|
|
|
const systemdScript = `[Unit]
|
|
Description={{.Description}}
|
|
ConditionFileIsExecutable={{.Path|cmdEscape}}
|
|
{{range $i, $dep := .Dependencies}}
|
|
{{$dep}} {{end}}
|
|
|
|
[Service]
|
|
StartLimitInterval=5
|
|
StartLimitBurst=10
|
|
ExecStart={{.Path|cmdEscape}}{{range .Arguments}} {{.|cmd}}{{end}}
|
|
{{if .WorkingDirectory}}WorkingDirectory={{.WorkingDirectory|cmdEscape}}{{end}}
|
|
{{if .UserName}}User={{.UserName}}{{end}}
|
|
{{if .Restart}}Restart={{.Restart}}{{end}}
|
|
{{if .SuccessExitStatus}}SuccessExitStatus={{.SuccessExitStatus}}{{end}}
|
|
TimeoutStopSec=20
|
|
RestartSec=120
|
|
EnvironmentFile=-/etc/sysconfig/{{.Name}}
|
|
|
|
DevicePolicy=closed
|
|
NoNewPrivileges=yes
|
|
PrivateTmp=yes
|
|
ProtectControlGroups=yes
|
|
ProtectKernelModules=yes
|
|
ProtectKernelTunables=yes
|
|
RestrictAddressFamilies=AF_UNIX AF_INET AF_INET6
|
|
RestrictNamespaces=yes
|
|
RestrictRealtime=yes
|
|
SystemCallFilter=~@clock @debug @module @mount @obsolete @reboot @setuid @swap
|
|
{{if .WorkingDirectory}}ReadWritePaths={{.WorkingDirectory|cmdEscape}}{{end}}
|
|
ProtectSystem=full
|
|
|
|
[Install]
|
|
WantedBy=multi-user.target
|
|
`
|