// +build linux package main import ( "fmt" "os" "path/filepath" cli "gopkg.in/urfave/cli.v2" ) func runApp(app *cli.App) { app.Commands = append(app.Commands, &cli.Command{ Name: "service", Usage: "Manages the Argo Tunnel system service", Subcommands: []*cli.Command{ &cli.Command{ Name: "install", Usage: "Install Argo Tunnel as a system service", Action: installLinuxService, }, &cli.Command{ Name: "uninstall", Usage: "Uninstall the Argo Tunnel service", Action: uninstallLinuxService, }, }, }) app.Run(os.Args) } const serviceConfigDir = "/etc/cloudflared" var systemdTemplates = []ServiceTemplate{ { Path: "/etc/systemd/system/cloudflared.service", Content: `[Unit] Description=Argo Tunnel After=network.target [Service] TimeoutStartSec=0 Type=notify ExecStart={{ .Path }} --config /etc/cloudflared/config.yml --origincert /etc/cloudflared/cert.pem --no-autoupdate Restart=on-failure RestartSec=5s [Install] WantedBy=multi-user.target `, }, { Path: "/etc/systemd/system/cloudflared-update.service", Content: `[Unit] Description=Update Argo Tunnel After=network.target [Service] ExecStart=/bin/bash -c '{{ .Path }} update; code=$?; if [ $code -eq 64 ]; then systemctl restart cloudflared; exit 0; fi; exit $code' `, }, { Path: "/etc/systemd/system/cloudflared-update.timer", Content: `[Unit] Description=Update Argo Tunnel [Timer] OnUnitActiveSec=1d [Install] WantedBy=timers.target `, }, } var sysvTemplate = ServiceTemplate{ Path: "/etc/init.d/cloudflared", FileMode: 0755, Content: `# For RedHat and cousins: # chkconfig: 2345 99 01 # description: Argo Tunnel agent # processname: {{.Path}} ### BEGIN INIT INFO # Provides: {{.Path}} # Required-Start: # Required-Stop: # Default-Start: 2 3 4 5 # Default-Stop: 0 1 6 # Short-Description: Argo Tunnel # Description: Argo Tunnel agent ### END INIT INFO name=$(basename $(readlink -f $0)) cmd="{{.Path}} --config /etc/cloudflared/config.yml --origincert /etc/cloudflared/cert.pem --pidfile /var/run/$name.pid --autoupdate-freq 24h0m0s" pid_file="/var/run/$name.pid" stdout_log="/var/log/$name.log" stderr_log="/var/log/$name.err" [ -e /etc/sysconfig/$name ] && . /etc/sysconfig/$name get_pid() { cat "$pid_file" } is_running() { [ -f "$pid_file" ] && ps $(get_pid) > /dev/null 2>&1 } case "$1" in start) if is_running; then echo "Already started" else echo "Starting $name" $cmd >> "$stdout_log" 2>> "$stderr_log" & echo $! > "$pid_file" if ! is_running; then echo "Unable to start, see $stdout_log and $stderr_log" exit 1 fi fi ;; stop) if is_running; then echo -n "Stopping $name.." kill $(get_pid) for i in {1..10} do if ! is_running; then break fi echo -n "." sleep 1 done echo if is_running; then echo "Not stopped; may still be shutting down or shutdown may have failed" exit 1 else echo "Stopped" if [ -f "$pid_file" ]; then rm "$pid_file" fi fi else echo "Not running" fi ;; restart) $0 stop if is_running; then echo "Unable to stop, will not attempt to start" exit 1 fi $0 start ;; status) if is_running; then echo "Running" else echo "Stopped" exit 1 fi ;; *) echo "Usage: $0 {start|stop|restart|status}" exit 1 ;; esac exit 0 `, } func isSystemd() bool { if _, err := os.Stat("/run/systemd/system"); err == nil { return true } return false } func installLinuxService(c *cli.Context) error { etPath, err := os.Executable() if err != nil { return fmt.Errorf("error determining executable path: %v", err) } templateArgs := ServiceTemplateArgs{Path: etPath} defaultConfigDir := filepath.Dir(c.String("config")) defaultConfigFile := filepath.Base(c.String("config")) if err = copyCredentials(serviceConfigDir, defaultConfigDir, defaultConfigFile); err != nil { logger.WithError(err).Infof("Failed to copy user configuration. Before running the service, ensure that %s contains two files, %s and %s", serviceConfigDir, credentialFile, defaultConfigFiles[0]) return err } switch { case isSystemd(): logger.Infof("Using Systemd") return installSystemd(&templateArgs) default: logger.Infof("Using Sysv") return installSysv(&templateArgs) } } func installSystemd(templateArgs *ServiceTemplateArgs) error { for _, serviceTemplate := range systemdTemplates { err := serviceTemplate.Generate(templateArgs) if err != nil { logger.WithError(err).Infof("error generating service template") return err } } if err := runCommand("systemctl", "enable", "cloudflared.service"); err != nil { logger.WithError(err).Infof("systemctl enable cloudflared.service error") return err } if err := runCommand("systemctl", "start", "cloudflared-update.timer"); err != nil { logger.WithError(err).Infof("systemctl start cloudflared-update.timer error") return err } logger.Infof("systemctl daemon-reload") return runCommand("systemctl", "daemon-reload") } func installSysv(templateArgs *ServiceTemplateArgs) error { confPath, err := sysvTemplate.ResolvePath() if err != nil { logger.WithError(err).Infof("error resolving system path") return err } if err := sysvTemplate.Generate(templateArgs); err != nil { logger.WithError(err).Infof("error generating system template") return err } for _, i := range [...]string{"2", "3", "4", "5"} { if err := os.Symlink(confPath, "/etc/rc"+i+".d/S50et"); err != nil { continue } } for _, i := range [...]string{"0", "1", "6"} { if err := os.Symlink(confPath, "/etc/rc"+i+".d/K02et"); err != nil { continue } } return nil } func uninstallLinuxService(c *cli.Context) error { switch { case isSystemd(): logger.Infof("Using Systemd") return uninstallSystemd() default: logger.Infof("Using Sysv") return uninstallSysv() } } func uninstallSystemd() error { if err := runCommand("systemctl", "disable", "cloudflared.service"); err != nil { logger.WithError(err).Infof("systemctl disable cloudflared.service error") return err } if err := runCommand("systemctl", "stop", "cloudflared-update.timer"); err != nil { logger.WithError(err).Infof("systemctl stop cloudflared-update.timer error") return err } for _, serviceTemplate := range systemdTemplates { if err := serviceTemplate.Remove(); err != nil { logger.WithError(err).Infof("error removing service template") return err } } logger.Infof("Successfully uninstall cloudflared service") return nil } func uninstallSysv() error { if err := sysvTemplate.Remove(); err != nil { logger.WithError(err).Infof("error removing service template") return err } for _, i := range [...]string{"2", "3", "4", "5"} { if err := os.Remove("/etc/rc" + i + ".d/S50et"); err != nil { continue } } for _, i := range [...]string{"0", "1", "6"} { if err := os.Remove("/etc/rc" + i + ".d/K02et"); err != nil { continue } } logger.Infof("Successfully uninstall cloudflared service") return nil }