npc.go 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. package main
  2. import (
  3. "ehang.io/nps/client"
  4. "ehang.io/nps/lib/common"
  5. "ehang.io/nps/lib/config"
  6. "ehang.io/nps/lib/file"
  7. "ehang.io/nps/lib/install"
  8. "ehang.io/nps/lib/version"
  9. "flag"
  10. "fmt"
  11. "github.com/astaxie/beego/logs"
  12. "github.com/ccding/go-stun/stun"
  13. "github.com/kardianos/service"
  14. "os"
  15. "runtime"
  16. "strings"
  17. "time"
  18. )
  19. var (
  20. serverAddr = flag.String("server", "", "Server addr (ip:port)")
  21. configPath = flag.String("config", "", "Configuration file path")
  22. verifyKey = flag.String("vkey", "", "Authentication key")
  23. logType = flag.String("log", "stdout", "Log output mode(stdout|file)")
  24. connType = flag.String("type", "tcp", "Connection type with the server(kcp|tcp)")
  25. proxyUrl = flag.String("proxy", "", "proxy socks5 url(eg:socks5://111:222@127.0.0.1:9007)")
  26. logLevel = flag.String("log_level", "7", "log level 0~7")
  27. registerTime = flag.Int("time", 2, "register time long /h")
  28. localPort = flag.Int("local_port", 2000, "p2p local port")
  29. password = flag.String("password", "", "p2p password flag")
  30. target = flag.String("target", "", "p2p target")
  31. localType = flag.String("local_type", "p2p", "p2p target")
  32. logPath = flag.String("log_path", "", "npc log path")
  33. debug = flag.Bool("debug", true, "npc debug")
  34. )
  35. const systemdScript = `[Unit]
  36. Description={{.Description}}
  37. ConditionFileIsExecutable={{.Path|cmdEscape}}
  38. {{range $i, $dep := .Dependencies}}
  39. {{$dep}} {{end}}
  40. [Service]
  41. LimitNOFILE=65536
  42. StartLimitInterval=5
  43. StartLimitBurst=10
  44. ExecStart={{.Path|cmdEscape}}{{range .Arguments}} {{.|cmd}}{{end}}
  45. {{if .ChRoot}}RootDirectory={{.ChRoot|cmd}}{{end}}
  46. {{if .WorkingDirectory}}WorkingDirectory={{.WorkingDirectory|cmdEscape}}{{end}}
  47. {{if .UserName}}User={{.UserName}}{{end}}
  48. {{if .ReloadSignal}}ExecReload=/bin/kill -{{.ReloadSignal}} "$MAINPID"{{end}}
  49. {{if .PIDFile}}PIDFile={{.PIDFile|cmd}}{{end}}
  50. {{if and .LogOutput .HasOutputFileSupport -}}
  51. StandardOutput=file:/var/log/{{.Name}}.out
  52. StandardError=file:/var/log/{{.Name}}.err
  53. {{- end}}
  54. Restart=always
  55. RestartSec=120
  56. [Install]
  57. WantedBy=multi-user.target
  58. `
  59. func main() {
  60. flag.Parse()
  61. logs.Reset()
  62. logs.EnableFuncCallDepth(true)
  63. logs.SetLogFuncCallDepth(3)
  64. if *logPath == "" {
  65. *logPath = common.GetNpcLogPath()
  66. }
  67. if common.IsWindows() {
  68. *logPath = strings.Replace(*logPath, "\\", "\\\\", -1)
  69. }
  70. if *debug {
  71. logs.SetLogger(logs.AdapterConsole, `{"level":`+*logLevel+`,"color":true}`)
  72. } else {
  73. logs.SetLogger(logs.AdapterFile, `{"level":`+*logLevel+`,"filename":"`+*logPath+`","daily":false,"maxlines":100000,"color":true}`)
  74. }
  75. // init service
  76. options := make(service.KeyValue)
  77. svcConfig := &service.Config{
  78. Name: "Npc",
  79. DisplayName: "nps内网穿透客户端",
  80. Description: "一款轻量级、功能强大的内网穿透代理服务器。支持tcp、udp流量转发,支持内网http代理、内网socks5代理,同时支持snappy压缩、站点保护、加密传输、多路复用、header修改等。支持web图形化管理,集成多用户模式。",
  81. Option: options,
  82. }
  83. if !common.IsWindows() {
  84. svcConfig.Dependencies = []string{
  85. "Requires=network.target",
  86. "After=network-online.target syslog.target"}
  87. svcConfig.Option["SystemdScript"] = systemdScript
  88. }
  89. for _, v := range os.Args[1:] {
  90. switch v {
  91. case "install", "start", "stop", "uninstall", "restart":
  92. continue
  93. }
  94. if !strings.Contains(v, "-service=") && !strings.Contains(v, "-debug=") {
  95. svcConfig.Arguments = append(svcConfig.Arguments, v)
  96. }
  97. }
  98. svcConfig.Arguments = append(svcConfig.Arguments, "-debug=false")
  99. prg := &npc{
  100. exit: make(chan struct{}),
  101. }
  102. s, err := service.New(prg, svcConfig)
  103. if err != nil {
  104. logs.Error(err)
  105. return
  106. }
  107. if len(os.Args) >= 2 {
  108. switch os.Args[1] {
  109. case "status":
  110. if len(os.Args) > 2 {
  111. path := strings.Replace(os.Args[2], "-config=", "", -1)
  112. client.GetTaskStatus(path)
  113. }
  114. case "register":
  115. flag.CommandLine.Parse(os.Args[2:])
  116. client.RegisterLocalIp(*serverAddr, *verifyKey, *connType, *proxyUrl, *registerTime)
  117. case "update":
  118. install.UpdateNpc()
  119. return
  120. case "nat":
  121. nat, host, err := stun.NewClient().Discover()
  122. if err != nil || host == nil {
  123. logs.Error("get nat type error", err)
  124. return
  125. }
  126. fmt.Printf("nat type: %s \npublic address: %s\n", nat.String(), host.String())
  127. os.Exit(0)
  128. case "install", "start", "stop", "uninstall", "restart":
  129. if os.Args[1] == "install" {
  130. service.Control(s, "stop")
  131. service.Control(s, "uninstall")
  132. install.InstallNpc()
  133. }
  134. err := service.Control(s, os.Args[1])
  135. if err != nil {
  136. logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
  137. }
  138. return
  139. }
  140. }
  141. s.Run()
  142. }
  143. type npc struct {
  144. exit chan struct{}
  145. }
  146. func (p *npc) Start(s service.Service) error {
  147. go p.run()
  148. return nil
  149. }
  150. func (p *npc) Stop(s service.Service) error {
  151. close(p.exit)
  152. if service.Interactive() {
  153. os.Exit(0)
  154. }
  155. return nil
  156. }
  157. func (p *npc) run() error {
  158. defer func() {
  159. if err := recover(); err != nil {
  160. const size = 64 << 10
  161. buf := make([]byte, size)
  162. buf = buf[:runtime.Stack(buf, false)]
  163. logs.Warning("npc: panic serving %v: %v\n%s", err, string(buf))
  164. }
  165. }()
  166. //p2p or secret command
  167. if *password != "" {
  168. commonConfig := new(config.CommonConfig)
  169. commonConfig.Server = *serverAddr
  170. commonConfig.VKey = *verifyKey
  171. commonConfig.Tp = *connType
  172. localServer := new(config.LocalServer)
  173. localServer.Type = *localType
  174. localServer.Password = *password
  175. localServer.Target = *target
  176. localServer.Port = *localPort
  177. commonConfig.Client = new(file.Client)
  178. commonConfig.Client.Cnf = new(file.Config)
  179. go client.StartLocalServer(localServer, commonConfig)
  180. return nil
  181. }
  182. env := common.GetEnvMap()
  183. if *serverAddr == "" {
  184. *serverAddr, _ = env["NPC_SERVER_ADDR"]
  185. }
  186. if *verifyKey == "" {
  187. *verifyKey, _ = env["NPC_SERVER_VKEY"]
  188. }
  189. logs.Info("the version of client is %s, the core version of client is %s", version.VERSION, version.GetVersion())
  190. if *verifyKey != "" && *serverAddr != "" && *configPath == "" {
  191. go func() {
  192. for {
  193. client.NewRPClient(*serverAddr, *verifyKey, *connType, *proxyUrl, nil).Start()
  194. logs.Info("It will be reconnected in five seconds")
  195. time.Sleep(time.Second * 5)
  196. }
  197. }()
  198. } else {
  199. if *configPath == "" {
  200. *configPath = "conf/npc.conf"
  201. }
  202. go client.StartFromFile(*configPath)
  203. }
  204. select {
  205. case <-p.exit:
  206. logs.Warning("stop...")
  207. }
  208. return nil
  209. }