123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 |
- // Copyright 2023 The frp Authors
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- package legacy
- import (
- "strings"
- "gopkg.in/ini.v1"
- legacyauth "github.com/fatedier/frp/pkg/auth/legacy"
- )
- type HTTPPluginOptions struct {
- Name string `ini:"name"`
- Addr string `ini:"addr"`
- Path string `ini:"path"`
- Ops []string `ini:"ops"`
- TLSVerify bool `ini:"tlsVerify"`
- }
- // ServerCommonConf contains information for a server service. It is
- // recommended to use GetDefaultServerConf instead of creating this object
- // directly, so that all unspecified fields have reasonable default values.
- type ServerCommonConf struct {
- legacyauth.ServerConfig `ini:",extends"`
- // BindAddr specifies the address that the server binds to. By default,
- // this value is "0.0.0.0".
- BindAddr string `ini:"bind_addr" json:"bind_addr"`
- // BindPort specifies the port that the server listens on. By default, this
- // value is 7000.
- BindPort int `ini:"bind_port" json:"bind_port"`
- // KCPBindPort specifies the KCP port that the server listens on. If this
- // value is 0, the server will not listen for KCP connections. By default,
- // this value is 0.
- KCPBindPort int `ini:"kcp_bind_port" json:"kcp_bind_port"`
- // QUICBindPort specifies the QUIC port that the server listens on.
- // Set this value to 0 will disable this feature.
- // By default, the value is 0.
- QUICBindPort int `ini:"quic_bind_port" json:"quic_bind_port"`
- // QUIC protocol options
- QUICKeepalivePeriod int `ini:"quic_keepalive_period" json:"quic_keepalive_period"`
- QUICMaxIdleTimeout int `ini:"quic_max_idle_timeout" json:"quic_max_idle_timeout"`
- QUICMaxIncomingStreams int `ini:"quic_max_incoming_streams" json:"quic_max_incoming_streams"`
- // ProxyBindAddr specifies the address that the proxy binds to. This value
- // may be the same as BindAddr.
- ProxyBindAddr string `ini:"proxy_bind_addr" json:"proxy_bind_addr"`
- // VhostHTTPPort specifies the port that the server listens for HTTP Vhost
- // requests. If this value is 0, the server will not listen for HTTP
- // requests. By default, this value is 0.
- VhostHTTPPort int `ini:"vhost_http_port" json:"vhost_http_port"`
- // VhostHTTPSPort specifies the port that the server listens for HTTPS
- // Vhost requests. If this value is 0, the server will not listen for HTTPS
- // requests. By default, this value is 0.
- VhostHTTPSPort int `ini:"vhost_https_port" json:"vhost_https_port"`
- // TCPMuxHTTPConnectPort specifies the port that the server listens for TCP
- // HTTP CONNECT requests. If the value is 0, the server will not multiplex TCP
- // requests on one single port. If it's not - it will listen on this value for
- // HTTP CONNECT requests. By default, this value is 0.
- TCPMuxHTTPConnectPort int `ini:"tcpmux_httpconnect_port" json:"tcpmux_httpconnect_port"`
- // If TCPMuxPassthrough is true, frps won't do any update on traffic.
- TCPMuxPassthrough bool `ini:"tcpmux_passthrough" json:"tcpmux_passthrough"`
- // VhostHTTPTimeout specifies the response header timeout for the Vhost
- // HTTP server, in seconds. By default, this value is 60.
- VhostHTTPTimeout int64 `ini:"vhost_http_timeout" json:"vhost_http_timeout"`
- // DashboardAddr specifies the address that the dashboard binds to. By
- // default, this value is "0.0.0.0".
- DashboardAddr string `ini:"dashboard_addr" json:"dashboard_addr"`
- // DashboardPort specifies the port that the dashboard listens on. If this
- // value is 0, the dashboard will not be started. By default, this value is
- // 0.
- DashboardPort int `ini:"dashboard_port" json:"dashboard_port"`
- // DashboardTLSCertFile specifies the path of the cert file that the server will
- // load. If "dashboard_tls_cert_file", "dashboard_tls_key_file" are valid, the server will use this
- // supplied tls configuration.
- DashboardTLSCertFile string `ini:"dashboard_tls_cert_file" json:"dashboard_tls_cert_file"`
- // DashboardTLSKeyFile specifies the path of the secret key that the server will
- // load. If "dashboard_tls_cert_file", "dashboard_tls_key_file" are valid, the server will use this
- // supplied tls configuration.
- DashboardTLSKeyFile string `ini:"dashboard_tls_key_file" json:"dashboard_tls_key_file"`
- // DashboardTLSMode specifies the mode of the dashboard between HTTP or HTTPS modes. By
- // default, this value is false, which is HTTP mode.
- DashboardTLSMode bool `ini:"dashboard_tls_mode" json:"dashboard_tls_mode"`
- // DashboardUser specifies the username that the dashboard will use for
- // login.
- DashboardUser string `ini:"dashboard_user" json:"dashboard_user"`
- // DashboardPwd specifies the password that the dashboard will use for
- // login.
- DashboardPwd string `ini:"dashboard_pwd" json:"dashboard_pwd"`
- // EnablePrometheus will export prometheus metrics on {dashboard_addr}:{dashboard_port}
- // in /metrics api.
- EnablePrometheus bool `ini:"enable_prometheus" json:"enable_prometheus"`
- // AssetsDir specifies the local directory that the dashboard will load
- // resources from. If this value is "", assets will be loaded from the
- // bundled executable using statik. By default, this value is "".
- AssetsDir string `ini:"assets_dir" json:"assets_dir"`
- // LogFile specifies a file where logs will be written to. This value will
- // only be used if LogWay is set appropriately. By default, this value is
- // "console".
- LogFile string `ini:"log_file" json:"log_file"`
- // LogWay specifies the way logging is managed. Valid values are "console"
- // or "file". If "console" is used, logs will be printed to stdout. If
- // "file" is used, logs will be printed to LogFile. By default, this value
- // is "console".
- LogWay string `ini:"log_way" json:"log_way"`
- // LogLevel specifies the minimum log level. Valid values are "trace",
- // "debug", "info", "warn", and "error". By default, this value is "info".
- LogLevel string `ini:"log_level" json:"log_level"`
- // LogMaxDays specifies the maximum number of days to store log information
- // before deletion. This is only used if LogWay == "file". By default, this
- // value is 0.
- LogMaxDays int64 `ini:"log_max_days" json:"log_max_days"`
- // DisableLogColor disables log colors when LogWay == "console" when set to
- // true. By default, this value is false.
- DisableLogColor bool `ini:"disable_log_color" json:"disable_log_color"`
- // DetailedErrorsToClient defines whether to send the specific error (with
- // debug info) to frpc. By default, this value is true.
- DetailedErrorsToClient bool `ini:"detailed_errors_to_client" json:"detailed_errors_to_client"`
- // SubDomainHost specifies the domain that will be attached to sub-domains
- // requested by the client when using Vhost proxying. For example, if this
- // value is set to "frps.com" and the client requested the subdomain
- // "test", the resulting URL would be "test.frps.com". By default, this
- // value is "".
- SubDomainHost string `ini:"subdomain_host" json:"subdomain_host"`
- // TCPMux toggles TCP stream multiplexing. This allows multiple requests
- // from a client to share a single TCP connection. By default, this value
- // is true.
- TCPMux bool `ini:"tcp_mux" json:"tcp_mux"`
- // TCPMuxKeepaliveInterval specifies the keep alive interval for TCP stream multiplier.
- // If TCPMux is true, heartbeat of application layer is unnecessary because it can only rely on heartbeat in TCPMux.
- TCPMuxKeepaliveInterval int64 `ini:"tcp_mux_keepalive_interval" json:"tcp_mux_keepalive_interval"`
- // TCPKeepAlive specifies the interval between keep-alive probes for an active network connection between frpc and frps.
- // If negative, keep-alive probes are disabled.
- TCPKeepAlive int64 `ini:"tcp_keepalive" json:"tcp_keepalive"`
- // Custom404Page specifies a path to a custom 404 page to display. If this
- // value is "", a default page will be displayed. By default, this value is
- // "".
- Custom404Page string `ini:"custom_404_page" json:"custom_404_page"`
- // AllowPorts specifies a set of ports that clients are able to proxy to.
- // If the length of this value is 0, all ports are allowed. By default,
- // this value is an empty set.
- AllowPorts map[int]struct{} `ini:"-" json:"-"`
- // Original string.
- AllowPortsStr string `ini:"-" json:"-"`
- // MaxPoolCount specifies the maximum pool size for each proxy. By default,
- // this value is 5.
- MaxPoolCount int64 `ini:"max_pool_count" json:"max_pool_count"`
- // MaxPortsPerClient specifies the maximum number of ports a single client
- // may proxy to. If this value is 0, no limit will be applied. By default,
- // this value is 0.
- MaxPortsPerClient int64 `ini:"max_ports_per_client" json:"max_ports_per_client"`
- // TLSOnly specifies whether to only accept TLS-encrypted connections.
- // By default, the value is false.
- TLSOnly bool `ini:"tls_only" json:"tls_only"`
- // TLSCertFile specifies the path of the cert file that the server will
- // load. If "tls_cert_file", "tls_key_file" are valid, the server will use this
- // supplied tls configuration. Otherwise, the server will use the tls
- // configuration generated by itself.
- TLSCertFile string `ini:"tls_cert_file" json:"tls_cert_file"`
- // TLSKeyFile specifies the path of the secret key that the server will
- // load. If "tls_cert_file", "tls_key_file" are valid, the server will use this
- // supplied tls configuration. Otherwise, the server will use the tls
- // configuration generated by itself.
- TLSKeyFile string `ini:"tls_key_file" json:"tls_key_file"`
- // TLSTrustedCaFile specifies the paths of the client cert files that the
- // server will load. It only works when "tls_only" is true. If
- // "tls_trusted_ca_file" is valid, the server will verify each client's
- // certificate.
- TLSTrustedCaFile string `ini:"tls_trusted_ca_file" json:"tls_trusted_ca_file"`
- // HeartBeatTimeout specifies the maximum time to wait for a heartbeat
- // before terminating the connection. It is not recommended to change this
- // value. By default, this value is 90. Set negative value to disable it.
- HeartbeatTimeout int64 `ini:"heartbeat_timeout" json:"heartbeat_timeout"`
- // UserConnTimeout specifies the maximum time to wait for a work
- // connection. By default, this value is 10.
- UserConnTimeout int64 `ini:"user_conn_timeout" json:"user_conn_timeout"`
- // HTTPPlugins specify the server plugins support HTTP protocol.
- HTTPPlugins map[string]HTTPPluginOptions `ini:"-" json:"http_plugins"`
- // UDPPacketSize specifies the UDP packet size
- // By default, this value is 1500
- UDPPacketSize int64 `ini:"udp_packet_size" json:"udp_packet_size"`
- // Enable golang pprof handlers in dashboard listener.
- // Dashboard port must be set first.
- PprofEnable bool `ini:"pprof_enable" json:"pprof_enable"`
- // NatHoleAnalysisDataReserveHours specifies the hours to reserve nat hole analysis data.
- NatHoleAnalysisDataReserveHours int64 `ini:"nat_hole_analysis_data_reserve_hours" json:"nat_hole_analysis_data_reserve_hours"`
- }
- // GetDefaultServerConf returns a server configuration with reasonable defaults.
- // Note: Some default values here will be set to empty and will be converted to them
- // new configuration through the 'Complete' function to set them as the default
- // values of the new configuration.
- func GetDefaultServerConf() ServerCommonConf {
- return ServerCommonConf{
- ServerConfig: legacyauth.GetDefaultServerConf(),
- DashboardAddr: "0.0.0.0",
- LogFile: "console",
- LogWay: "console",
- DetailedErrorsToClient: true,
- TCPMux: true,
- AllowPorts: make(map[int]struct{}),
- HTTPPlugins: make(map[string]HTTPPluginOptions),
- }
- }
- func UnmarshalServerConfFromIni(source any) (ServerCommonConf, error) {
- f, err := ini.LoadSources(ini.LoadOptions{
- Insensitive: false,
- InsensitiveSections: false,
- InsensitiveKeys: false,
- IgnoreInlineComment: true,
- AllowBooleanKeys: true,
- }, source)
- if err != nil {
- return ServerCommonConf{}, err
- }
- s, err := f.GetSection("common")
- if err != nil {
- return ServerCommonConf{}, err
- }
- common := GetDefaultServerConf()
- err = s.MapTo(&common)
- if err != nil {
- return ServerCommonConf{}, err
- }
- // allow_ports
- allowPortStr := s.Key("allow_ports").String()
- if allowPortStr != "" {
- common.AllowPortsStr = allowPortStr
- }
- // plugin.xxx
- pluginOpts := make(map[string]HTTPPluginOptions)
- for _, section := range f.Sections() {
- name := section.Name()
- if !strings.HasPrefix(name, "plugin.") {
- continue
- }
- opt, err := loadHTTPPluginOpt(section)
- if err != nil {
- return ServerCommonConf{}, err
- }
- pluginOpts[opt.Name] = *opt
- }
- common.HTTPPlugins = pluginOpts
- return common, nil
- }
- func loadHTTPPluginOpt(section *ini.Section) (*HTTPPluginOptions, error) {
- name := strings.TrimSpace(strings.TrimPrefix(section.Name(), "plugin."))
- opt := &HTTPPluginOptions{}
- err := section.MapTo(opt)
- if err != nil {
- return nil, err
- }
- opt.Name = name
- return opt, nil
- }
|