1
0
mirror of https://github.com/ehang-io/nps synced 2025-10-26 14:17:17 +08:00

support busyBox and sysV, for openWrt, fix #419, change update url

This commit is contained in:
ffdfgdfg 2020-02-28 00:31:17 +08:00
parent f6063e0d6c
commit 6abfa6de1a
3 changed files with 188 additions and 59 deletions

View File

@ -13,6 +13,7 @@ import (
"github.com/ccding/go-stun/stun" "github.com/ccding/go-stun/stun"
"github.com/kardianos/service" "github.com/kardianos/service"
"os" "os"
"os/exec"
"runtime" "runtime"
"strings" "strings"
"sync" "sync"
@ -37,31 +38,6 @@ var (
pprofAddr = flag.String("pprof", "", "PProf debug addr (ip:port)") pprofAddr = flag.String("pprof", "", "PProf debug addr (ip:port)")
) )
const systemdScript = `[Unit]
Description={{.Description}}
ConditionFileIsExecutable={{.Path|cmdEscape}}
{{range $i, $dep := .Dependencies}}
{{$dep}} {{end}}
[Service]
LimitNOFILE=65536
StartLimitInterval=5
StartLimitBurst=10
ExecStart={{.Path|cmdEscape}}{{range .Arguments}} {{.|cmd}}{{end}}
{{if .ChRoot}}RootDirectory={{.ChRoot|cmd}}{{end}}
{{if .WorkingDirectory}}WorkingDirectory={{.WorkingDirectory|cmdEscape}}{{end}}
{{if .UserName}}User={{.UserName}}{{end}}
{{if .ReloadSignal}}ExecReload=/bin/kill -{{.ReloadSignal}} "$MAINPID"{{end}}
{{if .PIDFile}}PIDFile={{.PIDFile|cmd}}{{end}}
{{if and .LogOutput .HasOutputFileSupport -}}
StandardOutput=file:/var/log/{{.Name}}.out
StandardError=file:/var/log/{{.Name}}.err
{{- end}}
Restart=always
RestartSec=120
[Install]
WantedBy=multi-user.target
`
func main() { func main() {
flag.Parse() flag.Parse()
logs.Reset() logs.Reset()
@ -91,7 +67,8 @@ func main() {
svcConfig.Dependencies = []string{ svcConfig.Dependencies = []string{
"Requires=network.target", "Requires=network.target",
"After=network-online.target syslog.target"} "After=network-online.target syslog.target"}
svcConfig.Option["SystemdScript"] = systemdScript svcConfig.Option["SystemdScript"] = install.SystemdScript
svcConfig.Option["SysvScript"] = install.SysvScript
} }
for _, v := range os.Args[1:] { for _, v := range os.Args[1:] {
switch v { switch v {
@ -137,17 +114,48 @@ func main() {
} }
fmt.Printf("nat type: %s \npublic address: %s\n", nat.String(), host.String()) fmt.Printf("nat type: %s \npublic address: %s\n", nat.String(), host.String())
os.Exit(0) os.Exit(0)
case "install", "start", "stop", "uninstall", "restart": case "start", "stop", "restart":
if os.Args[1] == "install" { // support busyBox and sysV, for openWrt
service.Control(s, "stop") if service.Platform() == "unix-systemv" {
service.Control(s, "uninstall") logs.Info("unix-systemv service")
install.InstallNpc() cmd := exec.Command("/etc/init.d/"+svcConfig.Name, os.Args[1])
err := cmd.Run()
if err != nil {
logs.Error(err)
}
return
} }
err := service.Control(s, os.Args[1]) err := service.Control(s, os.Args[1])
if err != nil { if err != nil {
logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error()) logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
} }
return return
case "install":
service.Control(s, "stop")
service.Control(s, "uninstall")
install.InstallNpc()
err := service.Control(s, os.Args[1])
if err != nil {
logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
}
if service.Platform() == "unix-systemv" {
logs.Info("unix-systemv service")
confPath := "/etc/init.d/" + svcConfig.Name
os.Symlink(confPath, "/etc/rc.d/S90"+svcConfig.Name)
os.Symlink(confPath, "/etc/rc.d/K02"+svcConfig.Name)
}
return
case "uninstall":
err := service.Control(s, os.Args[1])
if err != nil {
logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
}
if service.Platform() == "unix-systemv" {
logs.Info("unix-systemv service")
os.Remove("/etc/rc.d/S90" + svcConfig.Name)
os.Remove("/etc/rc.d/K02" + svcConfig.Name)
}
return
} }
} }
s.Run() s.Run()

View File

@ -12,6 +12,7 @@ import (
"flag" "flag"
"log" "log"
"os" "os"
"os/exec"
"path/filepath" "path/filepath"
"runtime" "runtime"
"strings" "strings"
@ -29,31 +30,6 @@ var (
level string level string
) )
const systemdScript = `[Unit]
Description={{.Description}}
ConditionFileIsExecutable={{.Path|cmdEscape}}
{{range $i, $dep := .Dependencies}}
{{$dep}} {{end}}
[Service]
LimitNOFILE=65536
StartLimitInterval=5
StartLimitBurst=10
ExecStart={{.Path|cmdEscape}}{{range .Arguments}} {{.|cmd}}{{end}}
{{if .ChRoot}}RootDirectory={{.ChRoot|cmd}}{{end}}
{{if .WorkingDirectory}}WorkingDirectory={{.WorkingDirectory|cmdEscape}}{{end}}
{{if .UserName}}User={{.UserName}}{{end}}
{{if .ReloadSignal}}ExecReload=/bin/kill -{{.ReloadSignal}} "$MAINPID"{{end}}
{{if .PIDFile}}PIDFile={{.PIDFile|cmd}}{{end}}
{{if and .LogOutput .HasOutputFileSupport -}}
StandardOutput=file:/var/log/{{.Name}}.out
StandardError=file:/var/log/{{.Name}}.err
{{- end}}
Restart=always
RestartSec=120
[Install]
WantedBy=multi-user.target
`
func main() { func main() {
flag.Parse() flag.Parse()
// init log // init log
@ -92,7 +68,8 @@ func main() {
svcConfig.Dependencies = []string{ svcConfig.Dependencies = []string{
"Requires=network.target", "Requires=network.target",
"After=network-online.target syslog.target"} "After=network-online.target syslog.target"}
svcConfig.Option["SystemdScript"] = systemdScript svcConfig.Option["SystemdScript"] = install.SystemdScript
svcConfig.Option["SysvScript"] = install.SysvScript
} }
prg := &nps{} prg := &nps{}
prg.exit = make(chan struct{}) prg.exit = make(chan struct{})
@ -127,13 +104,39 @@ func main() {
if err != nil { if err != nil {
logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error()) logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
} }
if service.Platform() == "unix-systemv" {
logs.Info("unix-systemv service")
confPath := "/etc/init.d/" + svcConfig.Name
os.Symlink(confPath, "/etc/rc.d/S90"+svcConfig.Name)
os.Symlink(confPath, "/etc/rc.d/K02"+svcConfig.Name)
}
return return
case "start", "restart", "stop", "uninstall": case "start", "restart", "stop":
if service.Platform() == "unix-systemv" {
logs.Info("unix-systemv service")
cmd := exec.Command("/etc/init.d/"+svcConfig.Name, os.Args[1])
err := cmd.Run()
if err != nil {
logs.Error(err)
}
return
}
err := service.Control(s, os.Args[1]) err := service.Control(s, os.Args[1])
if err != nil { if err != nil {
logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error()) logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
} }
return return
case "uninstall":
err := service.Control(s, os.Args[1])
if err != nil {
logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
}
if service.Platform() == "unix-systemv" {
logs.Info("unix-systemv service")
os.Remove("/etc/rc.d/S90" + svcConfig.Name)
os.Remove("/etc/rc.d/K02" + svcConfig.Name)
}
return
case "update": case "update":
install.UpdateNps() install.UpdateNps()
return return

View File

@ -16,6 +16,124 @@ import (
"strings" "strings"
) )
// Keep it in sync with the template from service_sysv_linux.go file
// Use "ps | grep -v grep | grep $(get_pid)" because "ps PID" may not work on OpenWrt
const SysvScript = `#!/bin/sh
# For RedHat and cousins:
# chkconfig: - 99 01
# description: {{.Description}}
# processname: {{.Path}}
### BEGIN INIT INFO
# Provides: {{.Path}}
# Required-Start:
# Required-Stop:
# Default-Start: 2 3 4 5
# Default-Stop: 0 1 6
# Short-Description: {{.DisplayName}}
# Description: {{.Description}}
### END INIT INFO
cmd="{{.Path}}{{range .Arguments}} {{.|cmd}}{{end}}"
name=$(basename $(readlink -f $0))
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 | grep -v grep | grep $(get_pid) > /dev/null 2>&1
}
case "$1" in
start)
if is_running; then
echo "Already started"
else
echo "Starting $name"
{{if .WorkingDirectory}}cd '{{.WorkingDirectory}}'{{end}}
$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 $(seq 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
`
const SystemdScript = `[Unit]
Description={{.Description}}
ConditionFileIsExecutable={{.Path|cmdEscape}}
{{range $i, $dep := .Dependencies}}
{{$dep}} {{end}}
[Service]
LimitNOFILE=65536
StartLimitInterval=5
StartLimitBurst=10
ExecStart={{.Path|cmdEscape}}{{range .Arguments}} {{.|cmd}}{{end}}
{{if .ChRoot}}RootDirectory={{.ChRoot|cmd}}{{end}}
{{if .WorkingDirectory}}WorkingDirectory={{.WorkingDirectory|cmdEscape}}{{end}}
{{if .UserName}}User={{.UserName}}{{end}}
{{if .ReloadSignal}}ExecReload=/bin/kill -{{.ReloadSignal}} "$MAINPID"{{end}}
{{if .PIDFile}}PIDFile={{.PIDFile|cmd}}{{end}}
{{if and .LogOutput .HasOutputFileSupport -}}
StandardOutput=file:/var/log/{{.Name}}.out
StandardError=file:/var/log/{{.Name}}.err
{{- end}}
Restart=always
RestartSec=120
[Install]
WantedBy=multi-user.target
`
func UpdateNps() { func UpdateNps() {
destPath := downloadLatest("server") destPath := downloadLatest("server")
//复制文件到对应目录 //复制文件到对应目录
@ -36,7 +154,7 @@ type release struct {
func downloadLatest(bin string) string { func downloadLatest(bin string) string {
// get version // get version
data, err := http.Get("https://api.github.com/repos/cnlh/nps/releases/latest") data, err := http.Get("https://api.github.com/repos/ehang-io/nps/releases/latest")
if err != nil { if err != nil {
log.Fatal(err.Error()) log.Fatal(err.Error())
} }