Files
EdgeAPI/internal/utils/lookup.go

199 lines
4.4 KiB
Go
Raw Normal View History

2022-09-10 16:13:21 +08:00
package utils
import (
2023-03-21 11:38:20 +08:00
"errors"
teaconst "github.com/TeaOSLab/EdgeAPI/internal/const"
2023-06-05 12:36:29 +08:00
"github.com/TeaOSLab/EdgeAPI/internal/utils/taskutils"
2022-09-10 16:13:21 +08:00
"github.com/TeaOSLab/EdgeCommon/pkg/configutils"
2023-06-05 12:36:29 +08:00
"github.com/TeaOSLab/EdgeCommon/pkg/dnsconfigs"
"github.com/fsnotify/fsnotify"
"github.com/iwind/TeaGo/lists"
2023-03-21 11:38:20 +08:00
"github.com/iwind/TeaGo/logs"
2022-09-10 16:13:21 +08:00
"github.com/miekg/dns"
2023-06-05 12:36:29 +08:00
"sync"
2022-09-10 16:13:21 +08:00
)
2023-03-21 11:38:20 +08:00
var sharedDNSClient *dns.Client
var sharedDNSConfig *dns.ClientConfig
2023-06-05 12:36:29 +08:00
var sharedDNSLocker = &sync.RWMutex{}
2023-03-21 11:38:20 +08:00
func init() {
if !teaconst.IsMain {
return
}
2023-06-05 12:36:29 +08:00
var resolvConfFile = "/etc/resolv.conf"
config, err := dns.ClientConfigFromFile(resolvConfFile)
2023-03-21 11:38:20 +08:00
if err != nil {
logs.Println("ERROR: configure dns client failed: " + err.Error())
return
}
sharedDNSConfig = config
sharedDNSClient = &dns.Client{}
2023-06-05 12:36:29 +08:00
// 监视文件变化,以便及时更新配置
go func() {
watcher, watcherErr := fsnotify.NewWatcher()
if watcherErr == nil {
err = watcher.Add(resolvConfFile)
for range watcher.Events {
newConfig, err := dns.ClientConfigFromFile(resolvConfFile)
if err == nil && newConfig != nil {
sharedDNSLocker.Lock()
sharedDNSConfig = newConfig
sharedDNSLocker.Unlock()
}
}
}
}()
2023-03-21 11:38:20 +08:00
}
2022-09-10 16:13:21 +08:00
// LookupCNAME 查询CNAME记录
// TODO 可以设置使用的DNS主机地址
func LookupCNAME(host string) (string, error) {
2023-03-21 11:38:20 +08:00
if sharedDNSClient == nil {
return "", errors.New("could not find dns client")
2022-09-10 16:13:21 +08:00
}
var m = new(dns.Msg)
m.SetQuestion(host+".", dns.TypeCNAME)
m.RecursionDesired = true
var lastErr error
2023-06-05 12:36:29 +08:00
var serverAddrs = composeDNSResolverAddrs(nil)
for _, serverAddr := range serverAddrs {
r, _, err := sharedDNSClient.Exchange(m, serverAddr)
2022-09-10 16:13:21 +08:00
if err != nil {
lastErr = err
continue
}
if len(r.Answer) == 0 {
continue
}
return r.Answer[0].(*dns.CNAME).Target, nil
}
return "", lastErr
}
// LookupNS 查询NS记录
2023-06-05 12:36:29 +08:00
func LookupNS(host string, extraResolvers []*dnsconfigs.DNSResolver) ([]string, error) {
2022-09-10 16:13:21 +08:00
var m = new(dns.Msg)
m.SetQuestion(host+".", dns.TypeNS)
m.RecursionDesired = true
var result = []string{}
var lastErr error
var hasValidServer = false
2023-06-05 12:36:29 +08:00
var serverAddrs = composeDNSResolverAddrs(extraResolvers)
if len(serverAddrs) == 0 {
return nil, nil
}
taskErr := taskutils.RunConcurrent(serverAddrs, taskutils.DefaultConcurrent, func(task any, locker *sync.RWMutex) {
var serverAddr = task.(string)
r, _, err := sharedDNSClient.Exchange(m, serverAddr)
2022-09-10 16:13:21 +08:00
if err != nil {
lastErr = err
2023-06-05 12:36:29 +08:00
return
2022-09-10 16:13:21 +08:00
}
hasValidServer = true
if len(r.Answer) == 0 {
2023-06-05 12:36:29 +08:00
return
2022-09-10 16:13:21 +08:00
}
for _, answer := range r.Answer {
2023-06-05 12:36:29 +08:00
var value = answer.(*dns.NS).Ns
locker.Lock()
if len(value) > 0 && !lists.ContainsString(result, value) {
result = append(result, value)
}
locker.Unlock()
2022-09-10 16:13:21 +08:00
}
2023-06-05 12:36:29 +08:00
})
if taskErr != nil {
return result, taskErr
2022-09-10 16:13:21 +08:00
}
if hasValidServer {
return result, nil
}
return nil, lastErr
}
// LookupTXT 获取CNAME
2023-06-05 12:36:29 +08:00
func LookupTXT(host string, extraResolvers []*dnsconfigs.DNSResolver) ([]string, error) {
2022-09-10 16:13:21 +08:00
var m = new(dns.Msg)
2023-03-21 11:38:20 +08:00
m.SetQuestion(host+".", dns.TypeTXT)
2022-09-10 16:13:21 +08:00
m.RecursionDesired = true
var lastErr error
var result = []string{}
var hasValidServer = false
2023-06-05 12:36:29 +08:00
var serverAddrs = composeDNSResolverAddrs(extraResolvers)
if len(serverAddrs) == 0 {
return nil, nil
}
taskErr := taskutils.RunConcurrent(serverAddrs, taskutils.DefaultConcurrent, func(task any, locker *sync.RWMutex) {
var serverAddr = task.(string)
r, _, err := sharedDNSClient.Exchange(m, serverAddr)
2022-09-10 16:13:21 +08:00
if err != nil {
lastErr = err
2023-06-05 12:36:29 +08:00
return
2022-09-10 16:13:21 +08:00
}
hasValidServer = true
if len(r.Answer) == 0 {
2023-06-05 12:36:29 +08:00
return
2022-09-10 16:13:21 +08:00
}
for _, answer := range r.Answer {
2023-06-05 12:36:29 +08:00
for _, txt := range answer.(*dns.TXT).Txt {
locker.Lock()
if len(txt) > 0 && !lists.ContainsString(result, txt) {
result = append(result, txt)
}
locker.Unlock()
}
2022-09-10 16:13:21 +08:00
}
2023-06-05 12:36:29 +08:00
})
if taskErr != nil {
return result, taskErr
2022-09-10 16:13:21 +08:00
}
if hasValidServer {
return result, nil
}
return nil, lastErr
}
2023-06-05 12:36:29 +08:00
// 组合DNS解析服务器地址
func composeDNSResolverAddrs(extraResolvers []*dnsconfigs.DNSResolver) []string {
sharedDNSLocker.RLock()
defer sharedDNSLocker.RUnlock()
// 这里不处理重复,方便我们可以多次重试
var servers = sharedDNSConfig.Servers
var port = sharedDNSConfig.Port
var serverAddrs = []string{}
for _, serverAddr := range servers {
serverAddrs = append(serverAddrs, configutils.QuoteIP(serverAddr)+":"+port)
}
for _, resolver := range extraResolvers {
serverAddrs = append(serverAddrs, resolver.Addr())
}
return serverAddrs
}