npc.go 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  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. "os/exec"
  16. "runtime"
  17. "strings"
  18. "sync"
  19. "time"
  20. )
  21. var (
  22. serverAddr = flag.String("server", "", "Server addr (ip:port)")
  23. configPath = flag.String("config", "", "Configuration file path")
  24. verifyKey = flag.String("vkey", "", "Authentication key")
  25. logType = flag.String("log", "stdout", "Log output mode(stdout|file)")
  26. connType = flag.String("type", "tcp", "Connection type with the server(kcp|tcp)")
  27. proxyUrl = flag.String("proxy", "", "proxy socks5 url(eg:socks5://111:222@127.0.0.1:9007)")
  28. logLevel = flag.String("log_level", "7", "log level 0~7")
  29. registerTime = flag.Int("time", 2, "register time long /h")
  30. localPort = flag.Int("local_port", 2000, "p2p local port")
  31. password = flag.String("password", "", "p2p password flag")
  32. target = flag.String("target", "", "p2p target")
  33. localType = flag.String("local_type", "p2p", "p2p target")
  34. logPath = flag.String("log_path", "", "npc log path")
  35. debug = flag.Bool("debug", true, "npc debug")
  36. pprofAddr = flag.String("pprof", "", "PProf debug addr (ip:port)")
  37. stunAddr = flag.String("stun_addr", "stun.stunprotocol.org:3478", "stun server address (eg:stun.stunprotocol.org:3478)")
  38. ver = flag.Bool("version", false, "show current version")
  39. )
  40. func main() {
  41. flag.Parse()
  42. logs.Reset()
  43. logs.EnableFuncCallDepth(true)
  44. logs.SetLogFuncCallDepth(3)
  45. if *ver {
  46. common.PrintVersion()
  47. return
  48. }
  49. if *logPath == "" {
  50. *logPath = common.GetNpcLogPath()
  51. }
  52. if common.IsWindows() {
  53. *logPath = strings.Replace(*logPath, "\\", "\\\\", -1)
  54. }
  55. if *debug {
  56. logs.SetLogger(logs.AdapterConsole, `{"level":`+*logLevel+`,"color":true}`)
  57. } else {
  58. logs.SetLogger(logs.AdapterFile, `{"level":`+*logLevel+`,"filename":"`+*logPath+`","daily":false,"maxlines":100000,"color":true}`)
  59. }
  60. // init service
  61. options := make(service.KeyValue)
  62. svcConfig := &service.Config{
  63. Name: "Npc",
  64. DisplayName: "nps内网穿透客户端",
  65. Description: "一款轻量级、功能强大的内网穿透代理服务器。支持tcp、udp流量转发,支持内网http代理、内网socks5代理,同时支持snappy压缩、站点保护、加密传输、多路复用、header修改等。支持web图形化管理,集成多用户模式。",
  66. Option: options,
  67. }
  68. if !common.IsWindows() {
  69. svcConfig.Dependencies = []string{
  70. "Requires=network.target",
  71. "After=network-online.target syslog.target"}
  72. svcConfig.Option["SystemdScript"] = install.SystemdScript
  73. svcConfig.Option["SysvScript"] = install.SysvScript
  74. }
  75. for _, v := range os.Args[1:] {
  76. switch v {
  77. case "install", "start", "stop", "uninstall", "restart":
  78. continue
  79. }
  80. if !strings.Contains(v, "-service=") && !strings.Contains(v, "-debug=") {
  81. svcConfig.Arguments = append(svcConfig.Arguments, v)
  82. }
  83. }
  84. svcConfig.Arguments = append(svcConfig.Arguments, "-debug=false")
  85. prg := &npc{
  86. exit: make(chan struct{}),
  87. }
  88. s, err := service.New(prg, svcConfig)
  89. if err != nil {
  90. logs.Error(err, "service function disabled")
  91. run()
  92. // run without service
  93. wg := sync.WaitGroup{}
  94. wg.Add(1)
  95. wg.Wait()
  96. return
  97. }
  98. if len(os.Args) >= 2 {
  99. switch os.Args[1] {
  100. case "status":
  101. if len(os.Args) > 2 {
  102. path := strings.Replace(os.Args[2], "-config=", "", -1)
  103. client.GetTaskStatus(path)
  104. }
  105. case "register":
  106. flag.CommandLine.Parse(os.Args[2:])
  107. client.RegisterLocalIp(*serverAddr, *verifyKey, *connType, *proxyUrl, *registerTime)
  108. case "update":
  109. install.UpdateNpc()
  110. return
  111. case "nat":
  112. c := stun.NewClient()
  113. c.SetServerAddr(*stunAddr)
  114. nat, host, err := c.Discover()
  115. if err != nil || host == nil {
  116. logs.Error("get nat type error", err)
  117. return
  118. }
  119. fmt.Printf("nat type: %s \npublic address: %s\n", nat.String(), host.String())
  120. os.Exit(0)
  121. case "start", "stop", "restart":
  122. // support busyBox and sysV, for openWrt
  123. if service.Platform() == "unix-systemv" {
  124. logs.Info("unix-systemv service")
  125. cmd := exec.Command("/etc/init.d/"+svcConfig.Name, os.Args[1])
  126. err := cmd.Run()
  127. if err != nil {
  128. logs.Error(err)
  129. }
  130. return
  131. }
  132. err := service.Control(s, os.Args[1])
  133. if err != nil {
  134. logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
  135. }
  136. return
  137. case "install":
  138. service.Control(s, "stop")
  139. service.Control(s, "uninstall")
  140. install.InstallNpc()
  141. err := service.Control(s, os.Args[1])
  142. if err != nil {
  143. logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
  144. }
  145. if service.Platform() == "unix-systemv" {
  146. logs.Info("unix-systemv service")
  147. confPath := "/etc/init.d/" + svcConfig.Name
  148. os.Symlink(confPath, "/etc/rc.d/S90"+svcConfig.Name)
  149. os.Symlink(confPath, "/etc/rc.d/K02"+svcConfig.Name)
  150. }
  151. return
  152. case "uninstall":
  153. err := service.Control(s, os.Args[1])
  154. if err != nil {
  155. logs.Error("Valid actions: %q\n%s", service.ControlAction, err.Error())
  156. }
  157. if service.Platform() == "unix-systemv" {
  158. logs.Info("unix-systemv service")
  159. os.Remove("/etc/rc.d/S90" + svcConfig.Name)
  160. os.Remove("/etc/rc.d/K02" + svcConfig.Name)
  161. }
  162. return
  163. }
  164. }
  165. s.Run()
  166. }
  167. type npc struct {
  168. exit chan struct{}
  169. }
  170. func (p *npc) Start(s service.Service) error {
  171. go p.run()
  172. return nil
  173. }
  174. func (p *npc) Stop(s service.Service) error {
  175. close(p.exit)
  176. if service.Interactive() {
  177. os.Exit(0)
  178. }
  179. return nil
  180. }
  181. func (p *npc) run() error {
  182. defer func() {
  183. if err := recover(); err != nil {
  184. const size = 64 << 10
  185. buf := make([]byte, size)
  186. buf = buf[:runtime.Stack(buf, false)]
  187. logs.Warning("npc: panic serving %v: %v\n%s", err, string(buf))
  188. }
  189. }()
  190. run()
  191. select {
  192. case <-p.exit:
  193. logs.Warning("stop...")
  194. }
  195. return nil
  196. }
  197. func run() {
  198. common.InitPProfFromArg(*pprofAddr)
  199. //p2p or secret command
  200. if *password != "" {
  201. commonConfig := new(config.CommonConfig)
  202. commonConfig.Server = *serverAddr
  203. commonConfig.VKey = *verifyKey
  204. commonConfig.Tp = *connType
  205. localServer := new(config.LocalServer)
  206. localServer.Type = *localType
  207. localServer.Password = *password
  208. localServer.Target = *target
  209. localServer.Port = *localPort
  210. commonConfig.Client = new(file.Client)
  211. commonConfig.Client.Cnf = new(file.Config)
  212. go client.StartLocalServer(localServer, commonConfig)
  213. return
  214. }
  215. env := common.GetEnvMap()
  216. if *serverAddr == "" {
  217. *serverAddr, _ = env["NPC_SERVER_ADDR"]
  218. }
  219. if *verifyKey == "" {
  220. *verifyKey, _ = env["NPC_SERVER_VKEY"]
  221. }
  222. logs.Info("the version of client is %s, the core version of client is %s", version.VERSION, version.GetVersion())
  223. if *verifyKey != "" && *serverAddr != "" && *configPath == "" {
  224. go func() {
  225. for {
  226. client.NewRPClient(*serverAddr, *verifyKey, *connType, *proxyUrl, nil).Start()
  227. logs.Info("It will be reconnected in five seconds")
  228. time.Sleep(time.Second * 5)
  229. }
  230. }()
  231. } else {
  232. if *configPath == "" {
  233. *configPath = "conf/npc.conf"
  234. }
  235. go client.StartFromFile(*configPath)
  236. }
  237. }