mirror of https://github.com/v2ray/v2ray-core
validity map
parent
56976e2d1b
commit
9e84519134
|
@ -2,10 +2,11 @@ package dns
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net"
|
"net"
|
||||||
"sync"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/v2ray/v2ray-core/app"
|
"github.com/v2ray/v2ray-core/app"
|
||||||
|
"github.com/v2ray/v2ray-core/common/collect"
|
||||||
|
"github.com/v2ray/v2ray-core/common/serial"
|
||||||
)
|
)
|
||||||
|
|
||||||
type entry struct {
|
type entry struct {
|
||||||
|
@ -32,66 +33,30 @@ func (this *entry) Extend() {
|
||||||
}
|
}
|
||||||
|
|
||||||
type DnsCache struct {
|
type DnsCache struct {
|
||||||
sync.RWMutex
|
cache *collect.ValidityMap
|
||||||
cache map[string]*entry
|
|
||||||
config CacheConfig
|
config CacheConfig
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewCache(config CacheConfig) *DnsCache {
|
func NewCache(config CacheConfig) *DnsCache {
|
||||||
cache := &DnsCache{
|
cache := &DnsCache{
|
||||||
cache: make(map[string]*entry),
|
cache: collect.NewValidityMap(3600),
|
||||||
config: config,
|
config: config,
|
||||||
}
|
}
|
||||||
go cache.cleanup()
|
|
||||||
return cache
|
return cache
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *DnsCache) cleanup() {
|
|
||||||
for range time.Tick(60 * time.Second) {
|
|
||||||
entry2Remove := make([]*entry, 0, 128)
|
|
||||||
this.RLock()
|
|
||||||
for _, entry := range this.cache {
|
|
||||||
if !entry.IsValid() {
|
|
||||||
entry2Remove = append(entry2Remove, entry)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
this.RUnlock()
|
|
||||||
|
|
||||||
for _, entry := range entry2Remove {
|
|
||||||
if !entry.IsValid() {
|
|
||||||
this.Lock()
|
|
||||||
delete(this.cache, entry.domain)
|
|
||||||
this.Unlock()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (this *DnsCache) Add(context app.Context, domain string, ip net.IP) {
|
func (this *DnsCache) Add(context app.Context, domain string, ip net.IP) {
|
||||||
callerTag := context.CallerTag()
|
callerTag := context.CallerTag()
|
||||||
if !this.config.IsTrustedSource(callerTag) {
|
if !this.config.IsTrustedSource(callerTag) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
this.RLock()
|
this.cache.Set(serial.StringLiteral(domain), newEntry(domain, ip))
|
||||||
entry, found := this.cache[domain]
|
|
||||||
this.RUnlock()
|
|
||||||
if found {
|
|
||||||
entry.ip = ip
|
|
||||||
entry.Extend()
|
|
||||||
} else {
|
|
||||||
this.Lock()
|
|
||||||
this.cache[domain] = newEntry(domain, ip)
|
|
||||||
this.Unlock()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *DnsCache) Get(context app.Context, domain string) net.IP {
|
func (this *DnsCache) Get(context app.Context, domain string) net.IP {
|
||||||
this.RLock()
|
if value := this.cache.Get(serial.StringLiteral(domain)); value != nil {
|
||||||
entry, found := this.cache[domain]
|
return value.(*entry).ip
|
||||||
this.RUnlock()
|
|
||||||
if found {
|
|
||||||
return entry.ip
|
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,71 @@
|
||||||
|
package collect
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/v2ray/v2ray-core/common/serial"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Validity interface {
|
||||||
|
IsValid() bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type entry struct {
|
||||||
|
key string
|
||||||
|
value Validity
|
||||||
|
}
|
||||||
|
|
||||||
|
type ValidityMap struct {
|
||||||
|
sync.RWMutex
|
||||||
|
cache map[string]Validity
|
||||||
|
cleanupIntervalSec int
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewValidityMap(cleanupIntervalSec int) *ValidityMap {
|
||||||
|
instance := &ValidityMap{
|
||||||
|
cache: make(map[string]Validity),
|
||||||
|
cleanupIntervalSec: cleanupIntervalSec,
|
||||||
|
}
|
||||||
|
go instance.cleanup()
|
||||||
|
return instance
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *ValidityMap) cleanup() {
|
||||||
|
for range time.Tick(time.Duration(this.cleanupIntervalSec) * time.Second) {
|
||||||
|
entry2Remove := make([]entry, 0, 128)
|
||||||
|
this.RLock()
|
||||||
|
for key, value := range this.cache {
|
||||||
|
if !value.IsValid() {
|
||||||
|
entry2Remove = append(entry2Remove, entry{
|
||||||
|
key: key,
|
||||||
|
value: value,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.RUnlock()
|
||||||
|
|
||||||
|
for _, entry := range entry2Remove {
|
||||||
|
if !entry.value.IsValid() {
|
||||||
|
this.Lock()
|
||||||
|
delete(this.cache, entry.key)
|
||||||
|
this.Unlock()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *ValidityMap) Set(key serial.String, value Validity) {
|
||||||
|
this.Lock()
|
||||||
|
this.cache[key.String()] = value
|
||||||
|
this.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *ValidityMap) Get(key serial.String) Validity {
|
||||||
|
this.RLock()
|
||||||
|
defer this.RUnlock()
|
||||||
|
if value, found := this.cache[key.String()]; found {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
Loading…
Reference in New Issue