mirror of
https://github.com/netbirdio/netbird.git
synced 2026-05-30 20:49:57 +00:00
258 lines
6.2 KiB
Go
258 lines
6.2 KiB
Go
//go:build !ios && !android
|
|
|
|
package cmd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/kardianos/service"
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
"github.com/spf13/cobra"
|
|
"google.golang.org/grpc"
|
|
|
|
"github.com/netbirdio/netbird/client/proto"
|
|
"github.com/netbirdio/netbird/client/server"
|
|
"github.com/netbirdio/netbird/client/system"
|
|
"github.com/netbirdio/netbird/util"
|
|
)
|
|
|
|
func (p *program) Start(svc service.Service) error {
|
|
// Start should not block. Do the actual work async.
|
|
log.Info("starting NetBird service") //nolint
|
|
|
|
// Collect static system and platform information
|
|
system.UpdateStaticInfoAsync()
|
|
|
|
// in any case, even if configuration does not exists we run daemon to serve CLI gRPC API.
|
|
p.serv = grpc.NewServer()
|
|
|
|
daemonListener, err := listenOnAddress(daemonAddr)
|
|
if err != nil {
|
|
return fmt.Errorf("listen daemon interface: %w", err)
|
|
}
|
|
|
|
var jsonListener *socketListener
|
|
if !jsonSocketDisabled {
|
|
jsonListener, err = listenOnAddress(jsonSocket)
|
|
if err != nil {
|
|
_ = daemonListener.Close()
|
|
return fmt.Errorf("listen daemon JSON interface: %w", err)
|
|
}
|
|
} else {
|
|
removeStaleUnixSocketForAddress(jsonSocket)
|
|
}
|
|
|
|
go func() {
|
|
defer daemonListener.Close()
|
|
if jsonListener != nil {
|
|
defer jsonListener.Close()
|
|
}
|
|
|
|
if err := daemonListener.chmodUnixSocket("daemon"); err != nil {
|
|
log.Error(err)
|
|
return
|
|
}
|
|
if jsonListener != nil {
|
|
if err := jsonListener.chmodUnixSocket("daemon JSON"); err != nil {
|
|
log.Error(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
serverInstance := server.New(p.ctx, util.FindFirstLogPath(logFiles), configPath, profilesDisabled, updateSettingsDisabled, captureEnabled, networksDisabled)
|
|
if err := serverInstance.Start(); err != nil {
|
|
log.Fatalf("failed to start daemon: %v", err)
|
|
}
|
|
proto.RegisterDaemonServiceServer(p.serv, serverInstance)
|
|
|
|
p.serverInstanceMu.Lock()
|
|
p.serverInstance = serverInstance
|
|
p.serverInstanceMu.Unlock()
|
|
|
|
if jsonListener != nil {
|
|
if err := p.startJSONGateway(jsonListener, daemonAddr); err != nil {
|
|
log.Fatalf("failed to start daemon JSON server: %v", err)
|
|
}
|
|
} else {
|
|
log.Debug("daemon JSON socket disabled")
|
|
}
|
|
|
|
log.Printf("started daemon server: %v", daemonListener.address)
|
|
if err := p.serv.Serve(daemonListener.Listener); err != nil {
|
|
log.Errorf("failed to serve daemon requests: %v", err)
|
|
}
|
|
}()
|
|
return nil
|
|
}
|
|
|
|
func (p *program) Stop(srv service.Service) error {
|
|
p.serverInstanceMu.Lock()
|
|
if p.serverInstance != nil {
|
|
in := new(proto.DownRequest)
|
|
_, err := p.serverInstance.Down(p.ctx, in)
|
|
if err != nil {
|
|
log.Errorf("failed to stop daemon: %v", err)
|
|
}
|
|
}
|
|
p.serverInstanceMu.Unlock()
|
|
|
|
p.cancel()
|
|
|
|
p.jsonServMu.Lock()
|
|
jsonServ := p.jsonServ
|
|
p.jsonServMu.Unlock()
|
|
if jsonServ != nil {
|
|
shutdownCtx, shutdownCancel := context.WithTimeout(context.Background(), 2*time.Second)
|
|
if err := jsonServ.Shutdown(shutdownCtx); err != nil {
|
|
log.Errorf("failed to stop daemon JSON server gracefully: %v", err)
|
|
if err := jsonServ.Close(); err != nil {
|
|
log.Errorf("failed to close daemon JSON server: %v", err)
|
|
}
|
|
}
|
|
shutdownCancel()
|
|
}
|
|
|
|
if p.serv != nil {
|
|
p.serv.Stop()
|
|
}
|
|
|
|
time.Sleep(time.Second * 2)
|
|
log.Info("stopped NetBird service") //nolint
|
|
return nil
|
|
}
|
|
|
|
// Common setup for service control commands
|
|
func setupServiceControlCommand(cmd *cobra.Command, ctx context.Context, cancel context.CancelFunc) (service.Service, error) {
|
|
// rootCmd env vars are already applied by PersistentPreRunE.
|
|
SetFlagsFromEnvVars(serviceCmd)
|
|
|
|
cmd.SetOut(cmd.OutOrStdout())
|
|
|
|
if err := handleRebrand(cmd); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := util.InitLog(logLevel, logFiles...); err != nil {
|
|
return nil, fmt.Errorf("init log: %w", err)
|
|
}
|
|
|
|
cfg, err := newSVCConfig()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("create service config: %w", err)
|
|
}
|
|
|
|
s, err := newSVC(newProgram(ctx, cancel), cfg)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return s, nil
|
|
}
|
|
|
|
var runCmd = &cobra.Command{
|
|
Use: "run",
|
|
Short: "runs NetBird as service",
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
ctx, cancel := context.WithCancel(cmd.Context())
|
|
|
|
SetupCloseHandler(ctx, cancel)
|
|
SetupDebugHandler(ctx, nil, nil, nil, util.FindFirstLogPath(logFiles))
|
|
|
|
s, err := setupServiceControlCommand(cmd, ctx, cancel)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return s.Run()
|
|
},
|
|
}
|
|
|
|
var startCmd = &cobra.Command{
|
|
Use: "start",
|
|
Short: "starts NetBird service",
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
ctx, cancel := context.WithCancel(cmd.Context())
|
|
s, err := setupServiceControlCommand(cmd, ctx, cancel)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := s.Start(); err != nil {
|
|
return fmt.Errorf("start service: %w", err)
|
|
}
|
|
cmd.Println("NetBird service has been started")
|
|
return nil
|
|
},
|
|
}
|
|
|
|
var stopCmd = &cobra.Command{
|
|
Use: "stop",
|
|
Short: "stops NetBird service",
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
ctx, cancel := context.WithCancel(cmd.Context())
|
|
s, err := setupServiceControlCommand(cmd, ctx, cancel)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := s.Stop(); err != nil {
|
|
return fmt.Errorf("stop service: %w", err)
|
|
}
|
|
cmd.Println("NetBird service has been stopped")
|
|
return nil
|
|
},
|
|
}
|
|
|
|
var restartCmd = &cobra.Command{
|
|
Use: "restart",
|
|
Short: "restarts NetBird service",
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
ctx, cancel := context.WithCancel(cmd.Context())
|
|
s, err := setupServiceControlCommand(cmd, ctx, cancel)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := s.Restart(); err != nil {
|
|
return fmt.Errorf("restart service: %w", err)
|
|
}
|
|
cmd.Println("NetBird service has been restarted")
|
|
return nil
|
|
},
|
|
}
|
|
|
|
var svcStatusCmd = &cobra.Command{
|
|
Use: "status",
|
|
Short: "shows NetBird service status",
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
ctx, cancel := context.WithCancel(cmd.Context())
|
|
s, err := setupServiceControlCommand(cmd, ctx, cancel)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
status, err := s.Status()
|
|
if err != nil {
|
|
return fmt.Errorf("get service status: %w", err)
|
|
}
|
|
|
|
var statusText string
|
|
switch status {
|
|
case service.StatusRunning:
|
|
statusText = "Running"
|
|
case service.StatusStopped:
|
|
statusText = "Stopped"
|
|
case service.StatusUnknown:
|
|
statusText = "Unknown"
|
|
default:
|
|
statusText = fmt.Sprintf("Unknown (%d)", status)
|
|
}
|
|
|
|
cmd.Printf("NetBird service status: %s\n", statusText)
|
|
return nil
|
|
},
|
|
}
|