root.go 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  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. "context"
  17. "fmt"
  18. "os"
  19. "github.com/spf13/cobra"
  20. "github.com/fatedier/frp/pkg/auth"
  21. "github.com/fatedier/frp/pkg/config"
  22. "github.com/fatedier/frp/pkg/util/log"
  23. "github.com/fatedier/frp/pkg/util/util"
  24. "github.com/fatedier/frp/pkg/util/version"
  25. "github.com/fatedier/frp/server"
  26. )
  27. const (
  28. CfgFileTypeIni = iota
  29. CfgFileTypeCmd
  30. )
  31. var (
  32. cfgFile string
  33. showVersion bool
  34. bindAddr string
  35. bindPort int
  36. kcpBindPort int
  37. proxyBindAddr string
  38. vhostHTTPPort int
  39. vhostHTTPSPort int
  40. vhostHTTPTimeout int64
  41. dashboardAddr string
  42. dashboardPort int
  43. dashboardUser string
  44. dashboardPwd string
  45. enablePrometheus bool
  46. logFile string
  47. logLevel string
  48. logMaxDays int64
  49. disableLogColor bool
  50. token string
  51. subDomainHost string
  52. allowPorts string
  53. maxPortsPerClient int64
  54. tlsOnly bool
  55. dashboardTLSMode bool
  56. dashboardTLSCertFile string
  57. dashboardTLSKeyFile string
  58. )
  59. func init() {
  60. rootCmd.PersistentFlags().StringVarP(&cfgFile, "config", "c", "", "config file of frps")
  61. rootCmd.PersistentFlags().BoolVarP(&showVersion, "version", "v", false, "version of frps")
  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(&kcpBindPort, "kcp_bind_port", "", 0, "kcp bind udp port")
  65. rootCmd.PersistentFlags().StringVarP(&proxyBindAddr, "proxy_bind_addr", "", "0.0.0.0", "proxy bind address")
  66. rootCmd.PersistentFlags().IntVarP(&vhostHTTPPort, "vhost_http_port", "", 0, "vhost http port")
  67. rootCmd.PersistentFlags().IntVarP(&vhostHTTPSPort, "vhost_https_port", "", 0, "vhost https port")
  68. rootCmd.PersistentFlags().Int64VarP(&vhostHTTPTimeout, "vhost_http_timeout", "", 60, "vhost http response header timeout")
  69. rootCmd.PersistentFlags().StringVarP(&dashboardAddr, "dashboard_addr", "", "0.0.0.0", "dashboard 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().BoolVarP(&enablePrometheus, "enable_prometheus", "", false, "enable prometheus dashboard")
  74. rootCmd.PersistentFlags().StringVarP(&logFile, "log_file", "", "console", "log file")
  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().BoolVarP(&disableLogColor, "disable_log_color", "", false, "disable log color in console")
  78. rootCmd.PersistentFlags().StringVarP(&token, "token", "t", "", "auth token")
  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().BoolVarP(&tlsOnly, "tls_only", "", false, "frps tls only")
  83. rootCmd.PersistentFlags().BoolVarP(&dashboardTLSMode, "dashboard_tls_mode", "", false, "dashboard tls mode")
  84. rootCmd.PersistentFlags().StringVarP(&dashboardTLSCertFile, "dashboard_tls_cert_file", "", "", "dashboard tls cert file")
  85. rootCmd.PersistentFlags().StringVarP(&dashboardTLSKeyFile, "dashboard_tls_key_file", "", "", "dashboard tls key file")
  86. }
  87. var rootCmd = &cobra.Command{
  88. Use: "frps",
  89. Short: "frps is the server of frp (https://github.com/fatedier/frp)",
  90. RunE: func(cmd *cobra.Command, args []string) error {
  91. if showVersion {
  92. fmt.Println(version.Full())
  93. return nil
  94. }
  95. var cfg config.ServerCommonConf
  96. var err error
  97. if cfgFile != "" {
  98. var content []byte
  99. content, err = config.GetRenderedConfFromFile(cfgFile)
  100. if err != nil {
  101. return err
  102. }
  103. cfg, err = parseServerCommonCfg(CfgFileTypeIni, content)
  104. } else {
  105. cfg, err = parseServerCommonCfg(CfgFileTypeCmd, nil)
  106. }
  107. if err != nil {
  108. return err
  109. }
  110. err = runServer(cfg)
  111. if err != nil {
  112. fmt.Println(err)
  113. os.Exit(1)
  114. }
  115. return nil
  116. },
  117. }
  118. func Execute() {
  119. if err := rootCmd.Execute(); err != nil {
  120. os.Exit(1)
  121. }
  122. }
  123. func parseServerCommonCfg(fileType int, source []byte) (cfg config.ServerCommonConf, err error) {
  124. if fileType == CfgFileTypeIni {
  125. cfg, err = config.UnmarshalServerConfFromIni(source)
  126. } else if fileType == CfgFileTypeCmd {
  127. cfg, err = parseServerCommonCfgFromCmd()
  128. }
  129. if err != nil {
  130. return
  131. }
  132. cfg.Complete()
  133. err = cfg.Validate()
  134. if err != nil {
  135. err = fmt.Errorf("parse config error: %v", err)
  136. return
  137. }
  138. return
  139. }
  140. func parseServerCommonCfgFromCmd() (cfg config.ServerCommonConf, err error) {
  141. cfg = config.GetDefaultServerConf()
  142. cfg.BindAddr = bindAddr
  143. cfg.BindPort = bindPort
  144. cfg.KCPBindPort = kcpBindPort
  145. cfg.ProxyBindAddr = proxyBindAddr
  146. cfg.VhostHTTPPort = vhostHTTPPort
  147. cfg.VhostHTTPSPort = vhostHTTPSPort
  148. cfg.VhostHTTPTimeout = vhostHTTPTimeout
  149. cfg.DashboardAddr = dashboardAddr
  150. cfg.DashboardPort = dashboardPort
  151. cfg.DashboardUser = dashboardUser
  152. cfg.DashboardPwd = dashboardPwd
  153. cfg.EnablePrometheus = enablePrometheus
  154. cfg.DashboardTLSCertFile = dashboardTLSCertFile
  155. cfg.DashboardTLSKeyFile = dashboardTLSKeyFile
  156. cfg.DashboardTLSMode = dashboardTLSMode
  157. cfg.LogFile = logFile
  158. cfg.LogLevel = logLevel
  159. cfg.LogMaxDays = logMaxDays
  160. cfg.SubDomainHost = subDomainHost
  161. cfg.TLSOnly = tlsOnly
  162. // Only token authentication is supported in cmd mode
  163. cfg.ServerConfig = auth.GetDefaultServerConf()
  164. cfg.Token = token
  165. if len(allowPorts) > 0 {
  166. // e.g. 1000-2000,2001,2002,3000-4000
  167. ports, errRet := util.ParseRangeNumbers(allowPorts)
  168. if errRet != nil {
  169. err = fmt.Errorf("parse conf error: allow_ports: %v", errRet)
  170. return
  171. }
  172. for _, port := range ports {
  173. cfg.AllowPorts[int(port)] = struct{}{}
  174. }
  175. }
  176. cfg.MaxPortsPerClient = maxPortsPerClient
  177. cfg.DisableLogColor = disableLogColor
  178. return
  179. }
  180. func runServer(cfg config.ServerCommonConf) (err error) {
  181. log.InitLog(cfg.LogWay, cfg.LogFile, cfg.LogLevel, cfg.LogMaxDays, cfg.DisableLogColor)
  182. if cfgFile != "" {
  183. log.Info("frps uses config file: %s", cfgFile)
  184. } else {
  185. log.Info("frps uses command line arguments for config")
  186. }
  187. svr, err := server.NewService(cfg)
  188. if err != nil {
  189. return err
  190. }
  191. log.Info("frps started successfully")
  192. svr.Run(context.Background())
  193. return
  194. }