|
@@ -18,14 +18,17 @@ import (
|
|
|
"fmt"
|
|
|
"strconv"
|
|
|
"strings"
|
|
|
+ "sync"
|
|
|
|
|
|
ini "github.com/vaughan0/go-ini"
|
|
|
|
|
|
+ "frp/utils/log"
|
|
|
"frp/utils/vhost"
|
|
|
)
|
|
|
|
|
|
// common config
|
|
|
var (
|
|
|
+ ConfigFile string = "./frps.ini"
|
|
|
BindAddr string = "0.0.0.0"
|
|
|
BindPort int64 = 7000
|
|
|
VhostHttpPort int64 = 0 // if VhostHttpPort equals 0, don't listen a public port for http
|
|
@@ -37,20 +40,39 @@ var (
|
|
|
HeartBeatTimeout int64 = 90
|
|
|
UserConnTimeout int64 = 10
|
|
|
|
|
|
- VhostMuxer *vhost.HttpMuxer
|
|
|
+ VhostMuxer *vhost.HttpMuxer
|
|
|
+ ProxyServers map[string]*ProxyServer = make(map[string]*ProxyServer) // all proxy servers info and resources
|
|
|
+ ProxyServersMutex sync.RWMutex
|
|
|
)
|
|
|
|
|
|
-var ProxyServers map[string]*ProxyServer = make(map[string]*ProxyServer)
|
|
|
-
|
|
|
func LoadConf(confFile string) (err error) {
|
|
|
+ err = loadCommonConf(confFile)
|
|
|
+ if err != nil {
|
|
|
+ return err
|
|
|
+ }
|
|
|
+
|
|
|
+ // load all proxy server's configure and initialize
|
|
|
+ // and set ProxyServers map
|
|
|
+ newProxyServers, err := loadProxyConf(confFile)
|
|
|
+ if err != nil {
|
|
|
+ return err
|
|
|
+ }
|
|
|
+ for _, proxyServer := range newProxyServers {
|
|
|
+ proxyServer.Init()
|
|
|
+ }
|
|
|
+ ProxyServersMutex.Lock()
|
|
|
+ ProxyServers = newProxyServers
|
|
|
+ ProxyServersMutex.Unlock()
|
|
|
+ return nil
|
|
|
+}
|
|
|
+
|
|
|
+func loadCommonConf(confFile string) error {
|
|
|
var tmpStr string
|
|
|
var ok bool
|
|
|
-
|
|
|
conf, err := ini.LoadFile(confFile)
|
|
|
if err != nil {
|
|
|
return err
|
|
|
}
|
|
|
-
|
|
|
// common
|
|
|
tmpStr, ok = conf.Get("common", "bind_addr")
|
|
|
if ok {
|
|
@@ -95,18 +117,26 @@ func LoadConf(confFile string) (err error) {
|
|
|
if ok {
|
|
|
LogMaxDays, _ = strconv.ParseInt(tmpStr, 10, 64)
|
|
|
}
|
|
|
+ return nil
|
|
|
+}
|
|
|
|
|
|
+func loadProxyConf(confFile string) (proxyServers map[string]*ProxyServer, err error) {
|
|
|
+ var ok bool
|
|
|
+ proxyServers = make(map[string]*ProxyServer)
|
|
|
+ conf, err := ini.LoadFile(confFile)
|
|
|
+ if err != nil {
|
|
|
+ return proxyServers, err
|
|
|
+ }
|
|
|
// servers
|
|
|
for name, section := range conf {
|
|
|
if name != "common" {
|
|
|
- proxyServer := &ProxyServer{}
|
|
|
- proxyServer.CustomDomains = make([]string, 0)
|
|
|
+ proxyServer := NewProxyServer()
|
|
|
proxyServer.Name = name
|
|
|
|
|
|
proxyServer.Type, ok = section["type"]
|
|
|
if ok {
|
|
|
if proxyServer.Type != "tcp" && proxyServer.Type != "http" {
|
|
|
- return fmt.Errorf("Parse ini file error: proxy [%s] type error", proxyServer.Name)
|
|
|
+ return proxyServers, fmt.Errorf("Parse conf error: proxy [%s] type error", proxyServer.Name)
|
|
|
}
|
|
|
} else {
|
|
|
proxyServer.Type = "tcp"
|
|
@@ -114,7 +144,7 @@ func LoadConf(confFile string) (err error) {
|
|
|
|
|
|
proxyServer.AuthToken, ok = section["auth_token"]
|
|
|
if !ok {
|
|
|
- return fmt.Errorf("Parse ini file error: proxy [%s] no auth_token found", proxyServer.Name)
|
|
|
+ return proxyServers, fmt.Errorf("Parse conf error: proxy [%s] no auth_token found", proxyServer.Name)
|
|
|
}
|
|
|
|
|
|
// for tcp
|
|
@@ -128,10 +158,10 @@ func LoadConf(confFile string) (err error) {
|
|
|
if ok {
|
|
|
proxyServer.ListenPort, err = strconv.ParseInt(portStr, 10, 64)
|
|
|
if err != nil {
|
|
|
- return fmt.Errorf("Parse ini file error: proxy [%s] listen_port error", proxyServer.Name)
|
|
|
+ return proxyServers, fmt.Errorf("Parse conf error: proxy [%s] listen_port error", proxyServer.Name)
|
|
|
}
|
|
|
} else {
|
|
|
- return fmt.Errorf("Parse ini file error: proxy [%s] listen_port not found", proxyServer.Name)
|
|
|
+ return proxyServers, fmt.Errorf("Parse conf error: proxy [%s] listen_port not found", proxyServer.Name)
|
|
|
}
|
|
|
} else if proxyServer.Type == "http" {
|
|
|
// for http
|
|
@@ -142,20 +172,53 @@ func LoadConf(confFile string) (err error) {
|
|
|
suffix = fmt.Sprintf(":%d", VhostHttpPort)
|
|
|
}
|
|
|
proxyServer.CustomDomains = strings.Split(domainStr, ",")
|
|
|
+ if len(proxyServer.CustomDomains) == 0 {
|
|
|
+ return proxyServers, fmt.Errorf("Parse conf error: proxy [%s] custom_domains must be set when type equals http", proxyServer.Name)
|
|
|
+ }
|
|
|
for i, domain := range proxyServer.CustomDomains {
|
|
|
proxyServer.CustomDomains[i] = strings.ToLower(strings.TrimSpace(domain)) + suffix
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
+ proxyServers[proxyServer.Name] = proxyServer
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return proxyServers, nil
|
|
|
+}
|
|
|
|
|
|
+// the function can only reload proxy configures
|
|
|
+// common section won't be changed
|
|
|
+func ReloadConf(confFile string) (err error) {
|
|
|
+ loadProxyServers, err := loadProxyConf(confFile)
|
|
|
+ if err != nil {
|
|
|
+ return err
|
|
|
+ }
|
|
|
+
|
|
|
+ ProxyServersMutex.Lock()
|
|
|
+ for name, proxyServer := range loadProxyServers {
|
|
|
+ oldProxyServer, ok := ProxyServers[name]
|
|
|
+ if ok {
|
|
|
+ if !oldProxyServer.Compare(proxyServer) {
|
|
|
+ oldProxyServer.Close()
|
|
|
+ proxyServer.Init()
|
|
|
+ ProxyServers[name] = proxyServer
|
|
|
+ log.Info("ProxyName [%s] configure change, restart", name)
|
|
|
+ }
|
|
|
+ } else {
|
|
|
proxyServer.Init()
|
|
|
- ProxyServers[proxyServer.Name] = proxyServer
|
|
|
+ ProxyServers[name] = proxyServer
|
|
|
+ log.Info("ProxyName [%s] is new, init it", name)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- if len(ProxyServers) == 0 {
|
|
|
- return fmt.Errorf("Parse ini file error: no proxy config found")
|
|
|
+ for name, oldProxyServer := range ProxyServers {
|
|
|
+ _, ok := loadProxyServers[name]
|
|
|
+ if !ok {
|
|
|
+ oldProxyServer.Close()
|
|
|
+ delete(ProxyServers, name)
|
|
|
+ log.Info("ProxyName [%s] deleted, close it", name)
|
|
|
+ }
|
|
|
}
|
|
|
-
|
|
|
+ ProxyServersMutex.Unlock()
|
|
|
return nil
|
|
|
}
|