root.go 7.1 KB

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