root.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. // Copyright 2018 fatedier, fatedier@gmail.com
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. package main
  15. import (
  16. "fmt"
  17. "io/ioutil"
  18. "os"
  19. "github.com/spf13/cobra"
  20. "github.com/fatedier/frp/g"
  21. "github.com/fatedier/frp/models/config"
  22. "github.com/fatedier/frp/server"
  23. "github.com/fatedier/frp/utils/log"
  24. "github.com/fatedier/frp/utils/util"
  25. "github.com/fatedier/frp/utils/version"
  26. )
  27. const (
  28. CfgFileTypeIni = iota
  29. CfgFileTypeCmd
  30. )
  31. var (
  32. cfgFile string
  33. showVersion bool
  34. bindAddr string
  35. bindPort int
  36. bindUdpPort int
  37. kcpBindPort int
  38. proxyBindAddr string
  39. vhostHttpPort int
  40. vhostHttpsPort int
  41. dashboardAddr string
  42. dashboardPort int
  43. dashboardUser string
  44. dashboardPwd string
  45. assetsDir string
  46. logFile string
  47. logWay string
  48. logLevel string
  49. logMaxDays int64
  50. token string
  51. authTimeout int64
  52. subDomainHost string
  53. tcpMux bool
  54. allowPorts string
  55. maxPoolCount int64
  56. maxPortsPerClient int64
  57. vhostHttpTimeout int64
  58. )
  59. func init() {
  60. rootCmd.PersistentFlags().StringVarP(&cfgFile, "", "c", "", "config file of frps")
  61. rootCmd.PersistentFlags().BoolVarP(&showVersion, "version", "v", false, "version of frpc")
  62. rootCmd.PersistentFlags().StringVarP(&bindAddr, "bind_addr", "", "0.0.0.0", "bind address")
  63. rootCmd.PersistentFlags().IntVarP(&bindPort, "bind_port", "p", 7000, "bind port")
  64. rootCmd.PersistentFlags().IntVarP(&bindUdpPort, "bind_udp_port", "", 0, "bind udp port")
  65. rootCmd.PersistentFlags().IntVarP(&kcpBindPort, "kcp_bind_port", "", 0, "kcp bind udp port")
  66. rootCmd.PersistentFlags().StringVarP(&proxyBindAddr, "proxy_bind_addr", "", "0.0.0.0", "proxy bind address")
  67. rootCmd.PersistentFlags().IntVarP(&vhostHttpPort, "vhost_http_port", "", 0, "vhost http port")
  68. rootCmd.PersistentFlags().IntVarP(&vhostHttpsPort, "vhost_https_port", "", 0, "vhost https port")
  69. rootCmd.PersistentFlags().StringVarP(&dashboardAddr, "dashboard_addr", "", "0.0.0.0", "dasboard address")
  70. rootCmd.PersistentFlags().IntVarP(&dashboardPort, "dashboard_port", "", 0, "dashboard port")
  71. rootCmd.PersistentFlags().StringVarP(&dashboardUser, "dashboard_user", "", "admin", "dashboard user")
  72. rootCmd.PersistentFlags().StringVarP(&dashboardPwd, "dashboard_pwd", "", "admin", "dashboard password")
  73. rootCmd.PersistentFlags().StringVarP(&logFile, "log_file", "", "console", "log file")
  74. rootCmd.PersistentFlags().StringVarP(&logWay, "log_way", "", "console", "log way")
  75. rootCmd.PersistentFlags().StringVarP(&logLevel, "log_level", "", "info", "log level")
  76. rootCmd.PersistentFlags().Int64VarP(&logMaxDays, "log_max_days", "", 3, "log_max_days")
  77. rootCmd.PersistentFlags().StringVarP(&token, "token", "t", "", "auth token")
  78. rootCmd.PersistentFlags().Int64VarP(&authTimeout, "auth_timeout", "", 900, "auth timeout")
  79. rootCmd.PersistentFlags().StringVarP(&subDomainHost, "subdomain_host", "", "", "subdomain host")
  80. rootCmd.PersistentFlags().StringVarP(&allowPorts, "allow_ports", "", "", "allow ports")
  81. rootCmd.PersistentFlags().Int64VarP(&maxPortsPerClient, "max_ports_per_client", "", 0, "max ports per client")
  82. rootCmd.PersistentFlags().Int64VarP(&vhostHttpTimeout, "vhost_http_timeout", "", 30, "vhost http timeout")
  83. }
  84. var rootCmd = &cobra.Command{
  85. Use: "frps",
  86. Short: "frps is the server of frp (https://github.com/fatedier/frp)",
  87. RunE: func(cmd *cobra.Command, args []string) error {
  88. if showVersion {
  89. fmt.Println(version.Full())
  90. return nil
  91. }
  92. var err error
  93. if cfgFile != "" {
  94. err = parseServerCommonCfg(CfgFileTypeIni, cfgFile)
  95. } else {
  96. err = parseServerCommonCfg(CfgFileTypeCmd, "")
  97. }
  98. if err != nil {
  99. return err
  100. }
  101. err = runServer()
  102. if err != nil {
  103. fmt.Println(err)
  104. os.Exit(1)
  105. }
  106. return nil
  107. },
  108. }
  109. func Execute() {
  110. if err := rootCmd.Execute(); err != nil {
  111. os.Exit(1)
  112. }
  113. }
  114. func parseServerCommonCfg(fileType int, filePath string) (err error) {
  115. if fileType == CfgFileTypeIni {
  116. err = parseServerCommonCfgFromIni(filePath)
  117. } else if fileType == CfgFileTypeCmd {
  118. err = parseServerCommonCfgFromCmd()
  119. }
  120. if err != nil {
  121. return
  122. }
  123. g.GlbServerCfg.CfgFile = filePath
  124. err = g.GlbServerCfg.ServerCommonConf.Check()
  125. if err != nil {
  126. return
  127. }
  128. config.InitServerCfg(&g.GlbServerCfg.ServerCommonConf)
  129. return
  130. }
  131. func parseServerCommonCfgFromIni(filePath string) (err error) {
  132. b, err := ioutil.ReadFile(filePath)
  133. if err != nil {
  134. return err
  135. }
  136. content := string(b)
  137. cfg, err := config.UnmarshalServerConfFromIni(&g.GlbServerCfg.ServerCommonConf, content)
  138. if err != nil {
  139. return err
  140. }
  141. g.GlbServerCfg.ServerCommonConf = *cfg
  142. return
  143. }
  144. func parseServerCommonCfgFromCmd() (err error) {
  145. g.GlbServerCfg.BindAddr = bindAddr
  146. g.GlbServerCfg.BindPort = bindPort
  147. g.GlbServerCfg.BindUdpPort = bindUdpPort
  148. g.GlbServerCfg.KcpBindPort = kcpBindPort
  149. g.GlbServerCfg.ProxyBindAddr = proxyBindAddr
  150. g.GlbServerCfg.VhostHttpPort = vhostHttpPort
  151. g.GlbServerCfg.VhostHttpsPort = vhostHttpsPort
  152. g.GlbServerCfg.DashboardAddr = dashboardAddr
  153. g.GlbServerCfg.DashboardPort = dashboardPort
  154. g.GlbServerCfg.DashboardUser = dashboardUser
  155. g.GlbServerCfg.DashboardPwd = dashboardPwd
  156. g.GlbServerCfg.LogFile = logFile
  157. g.GlbServerCfg.LogWay = logWay
  158. g.GlbServerCfg.LogLevel = logLevel
  159. g.GlbServerCfg.LogMaxDays = logMaxDays
  160. g.GlbServerCfg.Token = token
  161. g.GlbServerCfg.AuthTimeout = authTimeout
  162. g.GlbServerCfg.SubDomainHost = subDomainHost
  163. if len(allowPorts) > 0 {
  164. // e.g. 1000-2000,2001,2002,3000-4000
  165. ports, errRet := util.ParseRangeNumbers(allowPorts)
  166. if errRet != nil {
  167. err = fmt.Errorf("Parse conf error: allow_ports: %v", errRet)
  168. return
  169. }
  170. for _, port := range ports {
  171. g.GlbServerCfg.AllowPorts[int(port)] = struct{}{}
  172. }
  173. }
  174. g.GlbServerCfg.MaxPortsPerClient = maxPortsPerClient
  175. g.GlbServerCfg.VhostHttpTimeout = vhostHttpTimeout
  176. return
  177. }
  178. func runServer() (err error) {
  179. log.InitLog(g.GlbServerCfg.LogWay, g.GlbServerCfg.LogFile, g.GlbServerCfg.LogLevel,
  180. g.GlbServerCfg.LogMaxDays)
  181. svr, err := server.NewService()
  182. if err != nil {
  183. return err
  184. }
  185. log.Info("Start frps success")
  186. server.ServerService = svr
  187. svr.Run()
  188. return
  189. }