提升通过域名查找服务的性能,轻松支持海量域名

This commit is contained in:
刘祥超
2021-11-15 16:56:31 +08:00
parent a95fe17ebc
commit f4edd45886
6 changed files with 310 additions and 26 deletions

View File

@@ -27,6 +27,10 @@ func MatchDomain(pattern string, domain string) (isMatched bool) {
return
}
if pattern == domain {
return true
}
if pattern == "*" {
return true
}
@@ -61,3 +65,19 @@ func MatchDomain(pattern string, domain string) (isMatched bool) {
}
return isMatched
}
// IsFuzzyDomain 判断是否为特殊域名
func IsFuzzyDomain(domain string) bool {
if len(domain) == 0 {
return true
}
if domain[0] == '.' || domain[0] == '~' {
return true
}
for _, c := range domain {
if c == '*' {
return true
}
}
return false
}

View File

@@ -81,3 +81,14 @@ func TestMatchDomain(t *testing.T) {
a.IsTrue(ok)
}
}
func TestIsSpecialDomain(t *testing.T) {
var a = assert.NewAssertion(t)
a.IsTrue(IsFuzzyDomain(""))
a.IsTrue(IsFuzzyDomain(".hello.com"))
a.IsTrue(IsFuzzyDomain("*.hello.com"))
a.IsTrue(IsFuzzyDomain("hello.*.com"))
a.IsTrue(IsFuzzyDomain("~^hello\\.com"))
a.IsFalse(IsFuzzyDomain("hello.com"))
}

View File

@@ -1,19 +1,71 @@
package serverconfigs
import "strings"
import (
"github.com/TeaOSLab/EdgeCommon/pkg/configutils"
"strings"
"sync"
)
type ServerAddressGroup struct {
fullAddr string
Servers []*ServerConfig
servers []*ServerConfig
// 域名和服务映射
strictDomainMap map[string]map[string]*ServerConfig // domain[:2] => {domain => *ServerConfig}
fuzzyDomainMap map[string]*ServerConfig // special domain => *ServerConfig
cacheLocker sync.RWMutex
cacheDomainMap map[string]map[string]*ServerConfig // domain[:2] => {domain => *ServerConfig}
countCacheDomains int
// 支持CNAME的服务
cnameDomainMap map[string]map[string]*ServerConfig // domain[:2] => {domain => *ServerConfig}
// 第一个TLS Server
firstTLSServer *ServerConfig
}
func NewServerAddressGroup(fullAddr string) *ServerAddressGroup {
return &ServerAddressGroup{fullAddr: fullAddr}
return &ServerAddressGroup{
fullAddr: fullAddr,
strictDomainMap: map[string]map[string]*ServerConfig{},
fuzzyDomainMap: map[string]*ServerConfig{},
cacheDomainMap: map[string]map[string]*ServerConfig{},
cnameDomainMap: map[string]map[string]*ServerConfig{},
}
}
// Add 添加服务
func (this *ServerAddressGroup) Add(server *ServerConfig) {
this.Servers = append(this.Servers, server)
for _, serverName := range server.AllStrictNames() {
var prefix = this.domainPrefix(serverName)
domainsMap, ok := this.strictDomainMap[prefix]
if ok {
domainsMap[serverName] = server
} else {
this.strictDomainMap[prefix] = map[string]*ServerConfig{serverName: server}
}
// CNAME
if server.SupportCNAME {
cnameDomainsMap, ok := this.cnameDomainMap[prefix]
if ok {
cnameDomainsMap[serverName] = server
} else {
this.cnameDomainMap[prefix] = map[string]*ServerConfig{serverName: server}
}
}
}
for _, serverName := range server.AllFuzzyNames() {
this.fuzzyDomainMap[serverName] = server
}
this.servers = append(this.servers, server)
// 第一个TLS Server
if this.firstTLSServer == nil && server.SSLPolicy() != nil && server.SSLPolicy().IsOn {
this.firstTLSServer = server
}
}
// FullAddr 获取完整的地址
@@ -40,6 +92,11 @@ func (this *ServerAddressGroup) Addr() string {
return strings.TrimPrefix(this.fullAddr, protocol.String()+"://")
}
// Servers 读取所有服务
func (this *ServerAddressGroup) Servers() []*ServerConfig {
return this.servers
}
// IsHTTP 判断当前分组是否为HTTP
func (this *ServerAddressGroup) IsHTTP() bool {
p := this.Protocol()
@@ -78,8 +135,82 @@ func (this *ServerAddressGroup) IsUDP() bool {
// FirstServer 获取第一个Server
func (this *ServerAddressGroup) FirstServer() *ServerConfig {
if len(this.Servers) > 0 {
return this.Servers[0]
if len(this.servers) > 0 {
return this.servers[0]
}
return nil
}
// FirstTLSServer 获取第一个TLS Server
func (this *ServerAddressGroup) FirstTLSServer() *ServerConfig {
return this.firstTLSServer
}
// MatchServerName 使用域名查找服务
func (this *ServerAddressGroup) MatchServerName(serverName string) *ServerConfig {
var prefix = this.domainPrefix(serverName)
// 试图从缓存中读取
this.cacheLocker.RLock()
if len(this.cacheDomainMap) > 0 {
domainMap, ok := this.cacheDomainMap[prefix]
if ok {
server, ok := domainMap[serverName]
if ok {
return server
}
}
}
this.cacheLocker.RUnlock()
domainMap, ok := this.strictDomainMap[prefix]
if ok {
server, ok := domainMap[serverName]
if ok {
return server
}
}
for pattern, server := range this.fuzzyDomainMap {
if configutils.MatchDomain(pattern, serverName) {
// 加入到缓存
this.cacheLocker.Lock()
// 限制缓存的最大尺寸,防止内存耗尽
if this.countCacheDomains < 1_000_000 {
domainMap, ok := this.cacheDomainMap[prefix]
if ok {
domainMap[serverName] = server
} else {
this.cacheDomainMap[prefix] = map[string]*ServerConfig{serverName: server}
}
this.countCacheDomains++
}
this.cacheLocker.Unlock()
return server
}
}
return nil
}
// MatchServerCNAME 使用CNAME查找服务
func (this *ServerAddressGroup) MatchServerCNAME(serverName string) *ServerConfig {
var prefix = this.domainPrefix(serverName)
domainMap, ok := this.cnameDomainMap[prefix]
if ok {
server, ok := domainMap[serverName]
if ok {
return server
}
}
return nil
}
func (this *ServerAddressGroup) domainPrefix(domain string) string {
if len(domain) < 2 {
return domain
}
return domain[:2]
}

View File

@@ -2,7 +2,9 @@ package serverconfigs
import (
"github.com/iwind/TeaGo/assert"
"github.com/iwind/TeaGo/types"
"testing"
"time"
)
func TestServerAddressGroup_Protocol(t *testing.T) {
@@ -32,3 +34,79 @@ func TestServerAddressGroup_Protocol(t *testing.T) {
a.IsTrue(group.Addr() == "/tmp/my.sock")
}
}
func TestServerAddressGroup_MatchServerName(t *testing.T) {
var group = NewServerAddressGroup("")
for i := 0; i < 1_000_000; i++ {
group.Add(&ServerConfig{
ServerNames: []*ServerNameConfig{
{
Name: "hello" + types.String(i) + ".com",
SubNames: []string{},
},
},
})
}
group.Add(&ServerConfig{
ServerNames: []*ServerNameConfig{
{
Name: "hello.com",
SubNames: []string{},
},
},
})
group.Add(&ServerConfig{
ServerNames: []*ServerNameConfig{
{
Name: "*.hello.com",
SubNames: []string{},
},
},
})
var before = time.Now()
defer func() {
t.Log(time.Since(before).Seconds()*1000, "ms")
}()
t.Log(group.MatchServerName("hello99999.com").AllStrictNames())
t.Log(group.MatchServerName("hello.com").AllStrictNames())
t.Log(group.MatchServerName("world.hello.com").AllFuzzyNames())
for i := 0; i < 100_000; i++ {
_ = group.MatchServerName("world.hello.com")
}
}
func TestServerAddressGroup_MatchServerCNAME(t *testing.T) {
var group = NewServerAddressGroup("")
group.Add(&ServerConfig{
ServerNames: []*ServerNameConfig{
{
Name: "hello.com",
SubNames: []string{},
},
},
SupportCNAME: true,
})
group.Add(&ServerConfig{
ServerNames: []*ServerNameConfig{
{
Name: "*.hello.com",
SubNames: []string{},
},
},
})
var before = time.Now()
defer func() {
t.Log(time.Since(before).Seconds()*1000, "ms")
}()
server := group.MatchServerCNAME("hello.com")
if server != nil {
t.Log(server.AllStrictNames())
} else {
t.Log(server)
}
t.Log(group.MatchServerCNAME("world.hello.com"))
}

View File

@@ -315,35 +315,60 @@ func (this *ServerConfig) IsUDPFamily() bool {
return this.UDP != nil
}
// MatchName 判断是否和域名匹配
func (this *ServerConfig) MatchName(name string) bool {
if len(name) == 0 {
return false
}
if len(this.AliasServerNames) > 0 && configutils.MatchDomains(this.AliasServerNames, name) {
return true
}
for _, serverName := range this.ServerNames {
if serverName.Match(name) {
return true
// AllStrictNames 所有严格域名
func (this *ServerConfig) AllStrictNames() []string {
var result = []string{}
for _, name := range this.AliasServerNames {
if len(name) > 0 {
if !configutils.IsFuzzyDomain(name) {
result = append(result, name)
}
}
}
return false
for _, serverName := range this.ServerNames {
var name = serverName.Name
if len(name) > 0 {
if !configutils.IsFuzzyDomain(name) {
result = append(result, name)
}
}
for _, name := range serverName.SubNames {
if len(name) > 0 {
if !configutils.IsFuzzyDomain(name) {
result = append(result, name)
}
}
}
}
return result
}
// MatchNameStrictly 判断是否严格匹配
func (this *ServerConfig) MatchNameStrictly(name string) bool {
for _, serverName := range this.AliasServerNames {
if serverName == name {
return true
// AllFuzzyNames 所有模糊域名
func (this *ServerConfig) AllFuzzyNames() []string {
var result = []string{}
for _, name := range this.AliasServerNames {
if len(name) > 0 {
if configutils.IsFuzzyDomain(name) {
result = append(result, name)
}
}
}
for _, serverName := range this.ServerNames {
if serverName.Name == name {
return true
var name = serverName.Name
if len(name) > 0 {
if configutils.IsFuzzyDomain(name) {
result = append(result, name)
}
}
for _, name := range serverName.SubNames {
if len(name) > 0 {
if configutils.IsFuzzyDomain(name) {
result = append(result, name)
}
}
}
}
return false
return result
}
// SSLPolicy SSL信息

View File

@@ -72,3 +72,22 @@ func TestServerConfig_Protocols(t *testing.T) {
t.Log(server.FullAddresses())
}
}
func TestServerConfig_AllStrictNames(t *testing.T) {
var config = &ServerConfig{
AliasServerNames: []string{"hello.com", ".hello.com"},
ServerNames: []*ServerNameConfig{
{
Name: "hello2.com",
},
{
SubNames: []string{"hello3.com", "hello4.com", "*.hello5.com"},
},
{
Name: "~hello.com",
},
},
}
t.Log(config.AllStrictNames())
t.Log(config.AllFuzzyNames())
}