2019-05-03 00:05:14 +08:00
|
|
|
package fakeip
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2022-04-12 00:31:04 +08:00
|
|
|
"net/netip"
|
2022-08-12 13:47:51 +08:00
|
|
|
"strings"
|
2019-05-23 23:27:29 +08:00
|
|
|
"sync"
|
2019-07-26 19:09:13 +08:00
|
|
|
|
2022-04-19 17:46:13 +08:00
|
|
|
"github.com/Dreamacro/clash/common/nnip"
|
2021-10-11 20:48:58 +08:00
|
|
|
"github.com/Dreamacro/clash/component/profile/cachefile"
|
2020-05-28 12:13:05 +08:00
|
|
|
"github.com/Dreamacro/clash/component/trie"
|
2019-05-03 00:05:14 +08:00
|
|
|
)
|
|
|
|
|
2022-04-13 05:55:08 +08:00
|
|
|
const (
|
|
|
|
offsetKey = "key-offset-fake-ip"
|
|
|
|
cycleKey = "key-cycle-fake-ip"
|
|
|
|
)
|
|
|
|
|
2021-10-11 20:48:58 +08:00
|
|
|
type store interface {
|
2022-04-12 00:31:04 +08:00
|
|
|
GetByHost(host string) (netip.Addr, bool)
|
|
|
|
PutByHost(host string, ip netip.Addr)
|
|
|
|
GetByIP(ip netip.Addr) (string, bool)
|
|
|
|
PutByIP(ip netip.Addr, host string)
|
|
|
|
DelByIP(ip netip.Addr)
|
|
|
|
Exist(ip netip.Addr) bool
|
2021-10-11 20:48:58 +08:00
|
|
|
CloneTo(store)
|
2022-03-23 01:05:43 +08:00
|
|
|
FlushFakeIP() error
|
2021-10-11 20:48:58 +08:00
|
|
|
}
|
|
|
|
|
2022-09-01 11:33:47 +08:00
|
|
|
// Pool is an implementation about fake ip generator without storage
|
2019-05-03 00:05:14 +08:00
|
|
|
type Pool struct {
|
2022-04-12 00:31:04 +08:00
|
|
|
gateway netip.Addr
|
|
|
|
first netip.Addr
|
|
|
|
last netip.Addr
|
|
|
|
offset netip.Addr
|
|
|
|
cycle bool
|
|
|
|
mux sync.Mutex
|
2022-11-02 22:28:18 +08:00
|
|
|
host *trie.DomainTrie[struct{}]
|
2023-10-26 10:39:54 +08:00
|
|
|
ipnet netip.Prefix
|
2022-04-12 00:31:04 +08:00
|
|
|
store store
|
2019-07-26 19:09:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Lookup return a fake ip with host
|
2022-04-12 00:31:04 +08:00
|
|
|
func (p *Pool) Lookup(host string) netip.Addr {
|
2019-07-26 19:09:13 +08:00
|
|
|
p.mux.Lock()
|
|
|
|
defer p.mux.Unlock()
|
2022-08-12 13:47:51 +08:00
|
|
|
|
|
|
|
// RFC4343: DNS Case Insensitive, we SHOULD return result with all cases.
|
|
|
|
host = strings.ToLower(host)
|
2021-10-11 20:48:58 +08:00
|
|
|
if ip, exist := p.store.GetByHost(host); exist {
|
2019-10-11 14:01:16 +08:00
|
|
|
return ip
|
2019-07-26 19:09:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ip := p.get(host)
|
2021-10-11 20:48:58 +08:00
|
|
|
p.store.PutByHost(host, ip)
|
2019-07-26 19:09:13 +08:00
|
|
|
return ip
|
2019-05-03 00:05:14 +08:00
|
|
|
}
|
|
|
|
|
2019-07-26 19:09:13 +08:00
|
|
|
// LookBack return host with the fake ip
|
2022-04-12 00:31:04 +08:00
|
|
|
func (p *Pool) LookBack(ip netip.Addr) (string, bool) {
|
2019-05-23 23:27:29 +08:00
|
|
|
p.mux.Lock()
|
|
|
|
defer p.mux.Unlock()
|
2019-07-26 19:09:13 +08:00
|
|
|
|
2021-10-11 20:48:58 +08:00
|
|
|
return p.store.GetByIP(ip)
|
2019-07-26 19:09:13 +08:00
|
|
|
}
|
|
|
|
|
2021-10-11 20:48:58 +08:00
|
|
|
// ShouldSkipped return if domain should be skipped
|
|
|
|
func (p *Pool) ShouldSkipped(domain string) bool {
|
2019-12-28 00:10:06 +08:00
|
|
|
if p.host == nil {
|
|
|
|
return false
|
|
|
|
}
|
2019-12-28 18:44:01 +08:00
|
|
|
return p.host.Search(domain) != nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Exist returns if given ip exists in fake-ip pool
|
2022-04-12 00:31:04 +08:00
|
|
|
func (p *Pool) Exist(ip netip.Addr) bool {
|
2019-12-28 18:44:01 +08:00
|
|
|
p.mux.Lock()
|
|
|
|
defer p.mux.Unlock()
|
|
|
|
|
2021-10-11 20:48:58 +08:00
|
|
|
return p.store.Exist(ip)
|
2019-12-28 00:10:06 +08:00
|
|
|
}
|
|
|
|
|
2019-07-26 19:09:13 +08:00
|
|
|
// Gateway return gateway ip
|
2022-04-12 00:31:04 +08:00
|
|
|
func (p *Pool) Gateway() netip.Addr {
|
|
|
|
return p.gateway
|
2019-07-26 19:09:13 +08:00
|
|
|
}
|
|
|
|
|
2022-04-12 00:31:04 +08:00
|
|
|
// Broadcast return the last ip
|
|
|
|
func (p *Pool) Broadcast() netip.Addr {
|
|
|
|
return p.last
|
2022-03-15 02:43:40 +08:00
|
|
|
}
|
|
|
|
|
2020-10-17 12:52:43 +08:00
|
|
|
// IPNet return raw ipnet
|
2023-10-26 10:39:54 +08:00
|
|
|
func (p *Pool) IPNet() netip.Prefix {
|
2020-10-17 12:52:43 +08:00
|
|
|
return p.ipnet
|
|
|
|
}
|
|
|
|
|
2021-10-11 20:48:58 +08:00
|
|
|
// CloneFrom clone cache from old pool
|
|
|
|
func (p *Pool) CloneFrom(o *Pool) {
|
|
|
|
o.store.CloneTo(p.store)
|
2020-09-17 10:48:42 +08:00
|
|
|
}
|
|
|
|
|
2022-04-12 00:31:04 +08:00
|
|
|
func (p *Pool) get(host string) netip.Addr {
|
2022-04-13 05:55:08 +08:00
|
|
|
p.offset = p.offset.Next()
|
2019-05-03 00:05:14 +08:00
|
|
|
|
2022-04-13 05:55:08 +08:00
|
|
|
if !p.offset.Less(p.last) {
|
|
|
|
p.cycle = true
|
|
|
|
p.offset = p.first
|
|
|
|
}
|
2022-04-12 21:54:54 +08:00
|
|
|
|
2022-04-13 05:55:08 +08:00
|
|
|
if p.cycle || p.store.Exist(p.offset) {
|
|
|
|
p.store.DelByIP(p.offset)
|
2022-04-12 00:31:04 +08:00
|
|
|
}
|
2022-03-23 01:05:43 +08:00
|
|
|
|
2022-04-12 00:31:04 +08:00
|
|
|
p.store.PutByIP(p.offset, host)
|
|
|
|
return p.offset
|
2019-05-03 00:05:14 +08:00
|
|
|
}
|
|
|
|
|
2022-04-12 00:31:04 +08:00
|
|
|
func (p *Pool) FlushFakeIP() error {
|
2022-04-13 05:55:08 +08:00
|
|
|
err := p.store.FlushFakeIP()
|
|
|
|
if err == nil {
|
|
|
|
p.cycle = false
|
2022-04-19 17:46:13 +08:00
|
|
|
p.offset = p.first.Prev()
|
2022-04-13 05:55:08 +08:00
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Pool) StoreState() {
|
|
|
|
if s, ok := p.store.(*cachefileStore); ok {
|
|
|
|
s.PutByHost(offsetKey, p.offset)
|
|
|
|
if p.cycle {
|
|
|
|
s.PutByHost(cycleKey, p.offset)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Pool) restoreState() {
|
|
|
|
if s, ok := p.store.(*cachefileStore); ok {
|
|
|
|
if _, exist := s.GetByHost(cycleKey); exist {
|
|
|
|
p.cycle = true
|
|
|
|
}
|
|
|
|
|
|
|
|
if offset, exist := s.GetByHost(offsetKey); exist {
|
|
|
|
if p.ipnet.Contains(offset) {
|
|
|
|
p.offset = offset
|
|
|
|
} else {
|
|
|
|
_ = p.FlushFakeIP()
|
|
|
|
}
|
|
|
|
} else if s.Exist(p.first) {
|
|
|
|
_ = p.FlushFakeIP()
|
|
|
|
}
|
|
|
|
}
|
2019-05-03 00:05:14 +08:00
|
|
|
}
|
|
|
|
|
2021-10-11 20:48:58 +08:00
|
|
|
type Options struct {
|
2023-10-26 10:39:54 +08:00
|
|
|
IPNet netip.Prefix
|
2022-11-02 22:28:18 +08:00
|
|
|
Host *trie.DomainTrie[struct{}]
|
2021-10-11 20:48:58 +08:00
|
|
|
|
|
|
|
// Size sets the maximum number of entries in memory
|
|
|
|
// and does not work if Persistence is true
|
|
|
|
Size int
|
|
|
|
|
|
|
|
// Persistence will save the data to disk.
|
|
|
|
// Size will not work and record will be fully stored.
|
|
|
|
Persistence bool
|
|
|
|
}
|
|
|
|
|
2019-05-03 00:05:14 +08:00
|
|
|
// New return Pool instance
|
2021-10-11 20:48:58 +08:00
|
|
|
func New(options Options) (*Pool, error) {
|
2022-04-12 00:31:04 +08:00
|
|
|
var (
|
|
|
|
hostAddr = options.IPNet.Masked().Addr()
|
|
|
|
gateway = hostAddr.Next()
|
2022-10-09 10:48:26 +08:00
|
|
|
first = gateway.Next().Next().Next() // default start with 198.18.0.4
|
2023-10-26 10:39:54 +08:00
|
|
|
last = nnip.UnMasked(options.IPNet)
|
2022-04-12 00:31:04 +08:00
|
|
|
)
|
|
|
|
|
2022-04-19 17:46:13 +08:00
|
|
|
if !options.IPNet.IsValid() || !first.IsValid() || !first.Less(last) {
|
2019-05-03 00:05:14 +08:00
|
|
|
return nil, errors.New("ipnet don't have valid ip")
|
|
|
|
}
|
|
|
|
|
2021-10-11 20:48:58 +08:00
|
|
|
pool := &Pool{
|
2022-04-12 00:31:04 +08:00
|
|
|
gateway: gateway,
|
|
|
|
first: first,
|
|
|
|
last: last,
|
|
|
|
offset: first.Prev(),
|
|
|
|
cycle: false,
|
|
|
|
host: options.Host,
|
|
|
|
ipnet: options.IPNet,
|
2021-10-11 20:48:58 +08:00
|
|
|
}
|
|
|
|
if options.Persistence {
|
|
|
|
pool.store = &cachefileStore{
|
|
|
|
cache: cachefile.Cache(),
|
|
|
|
}
|
|
|
|
} else {
|
2022-04-05 20:23:16 +08:00
|
|
|
pool.store = newMemoryStore(options.Size)
|
2021-10-11 20:48:58 +08:00
|
|
|
}
|
|
|
|
|
2022-04-13 05:55:08 +08:00
|
|
|
pool.restoreState()
|
|
|
|
|
2021-10-11 20:48:58 +08:00
|
|
|
return pool, nil
|
2019-05-03 00:05:14 +08:00
|
|
|
}
|