mirror of
https://github.com/apernet/OpenGFW.git
synced 2025-04-19 19:39:13 +08:00
Merge 65b15515af090e7d45df082b90f6b8ae91d24bdc into 278d731b6f0df6665e00987fa9e8afa99244d5f6
This commit is contained in:
commit
2ed6a7f28c
@ -1,8 +1,14 @@
|
|||||||
package udp
|
package udp
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
|
"encoding/binary"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"hash/fnv"
|
||||||
|
"math/rand/v2"
|
||||||
"net"
|
"net"
|
||||||
|
"os"
|
||||||
|
|
||||||
"github.com/apernet/OpenGFW/modifier"
|
"github.com/apernet/OpenGFW/modifier"
|
||||||
|
|
||||||
@ -14,33 +20,103 @@ var _ modifier.Modifier = (*DNSModifier)(nil)
|
|||||||
|
|
||||||
var (
|
var (
|
||||||
errInvalidIP = errors.New("invalid ip")
|
errInvalidIP = errors.New("invalid ip")
|
||||||
|
errInvalidIPList = errors.New("invalid ip list")
|
||||||
|
errInvalidIpListFile = errors.New("unable to open or parse ip list file")
|
||||||
errNotValidDNSResponse = errors.New("not a valid dns response")
|
errNotValidDNSResponse = errors.New("not a valid dns response")
|
||||||
errEmptyDNSQuestion = errors.New("empty dns question")
|
errEmptyDNSQuestion = errors.New("empty dns question")
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func fmtErrInvalidIP(ip string) error {
|
||||||
|
return fmt.Errorf("invalid ip: %s", ip)
|
||||||
|
}
|
||||||
|
|
||||||
|
func fmtErrInvalidIpListFile(filePath string) error {
|
||||||
|
return fmt.Errorf("unable to open or parse ip list file: %s", filePath)
|
||||||
|
}
|
||||||
|
|
||||||
type DNSModifier struct{}
|
type DNSModifier struct{}
|
||||||
|
|
||||||
func (m *DNSModifier) Name() string {
|
func (m *DNSModifier) Name() string {
|
||||||
return "dns"
|
return "dns"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *DNSModifier) parseIpEntry(entry interface{}, i *dnsModifierInstance) error {
|
||||||
|
entryStr, ok := entry.(string)
|
||||||
|
if !ok {
|
||||||
|
return &modifier.ErrInvalidArgs{Err: errInvalidIP}
|
||||||
|
}
|
||||||
|
|
||||||
|
ip := net.ParseIP(entryStr)
|
||||||
|
if ip == nil {
|
||||||
|
return &modifier.ErrInvalidArgs{Err: fmtErrInvalidIP(entryStr)}
|
||||||
|
}
|
||||||
|
if ip4 := ip.To4(); ip4 != nil {
|
||||||
|
i.A = append(i.A, ip4)
|
||||||
|
} else {
|
||||||
|
i.AAAA = append(i.AAAA, ip)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *DNSModifier) parseIpList(list []interface{}, i *dnsModifierInstance) error {
|
||||||
|
for _, entry := range list {
|
||||||
|
if err := m.parseIpEntry(entry, i); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *DNSModifier) parseIpListFile(filePath string, i *dnsModifierInstance) error {
|
||||||
|
file, err := os.Open(filePath)
|
||||||
|
if err != nil {
|
||||||
|
return &modifier.ErrInvalidArgs{Err: fmtErrInvalidIpListFile(filePath)}
|
||||||
|
}
|
||||||
|
defer file.Close()
|
||||||
|
|
||||||
|
scanner := bufio.NewScanner(file)
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
if err := m.parseIpEntry(line, i); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return &modifier.ErrInvalidArgs{Err: fmtErrInvalidIpListFile(filePath)}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (m *DNSModifier) New(args map[string]interface{}) (modifier.Instance, error) {
|
func (m *DNSModifier) New(args map[string]interface{}) (modifier.Instance, error) {
|
||||||
i := &dnsModifierInstance{}
|
i := &dnsModifierInstance{}
|
||||||
aStr, ok := args["a"].(string)
|
i.seed = rand.Uint32()
|
||||||
if ok {
|
|
||||||
a := net.ParseIP(aStr).To4()
|
for key, value := range args {
|
||||||
if a == nil {
|
switch key {
|
||||||
return nil, &modifier.ErrInvalidArgs{Err: errInvalidIP}
|
case "a", "aaaa":
|
||||||
|
if err := m.parseIpEntry(value, i); err != nil {
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
i.A = a
|
case "list":
|
||||||
|
if list, ok := value.([]interface{}); ok {
|
||||||
|
if err := m.parseIpList(list, i); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return nil, &modifier.ErrInvalidArgs{Err: errInvalidIPList}
|
||||||
|
}
|
||||||
|
case "file":
|
||||||
|
if filePath, ok := value.(string); ok {
|
||||||
|
if err := m.parseIpListFile(filePath, i); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return nil, &modifier.ErrInvalidArgs{Err: errInvalidIpListFile}
|
||||||
}
|
}
|
||||||
aaaaStr, ok := args["aaaa"].(string)
|
|
||||||
if ok {
|
|
||||||
aaaa := net.ParseIP(aaaaStr).To16()
|
|
||||||
if aaaa == nil {
|
|
||||||
return nil, &modifier.ErrInvalidArgs{Err: errInvalidIP}
|
|
||||||
}
|
}
|
||||||
i.AAAA = aaaa
|
|
||||||
}
|
}
|
||||||
return i, nil
|
return i, nil
|
||||||
}
|
}
|
||||||
@ -48,8 +124,9 @@ func (m *DNSModifier) New(args map[string]interface{}) (modifier.Instance, error
|
|||||||
var _ modifier.UDPModifierInstance = (*dnsModifierInstance)(nil)
|
var _ modifier.UDPModifierInstance = (*dnsModifierInstance)(nil)
|
||||||
|
|
||||||
type dnsModifierInstance struct {
|
type dnsModifierInstance struct {
|
||||||
A net.IP
|
A []net.IP
|
||||||
AAAA net.IP
|
AAAA []net.IP
|
||||||
|
seed uint32
|
||||||
}
|
}
|
||||||
|
|
||||||
func (i *dnsModifierInstance) Process(data []byte) ([]byte, error) {
|
func (i *dnsModifierInstance) Process(data []byte) ([]byte, error) {
|
||||||
@ -64,26 +141,41 @@ func (i *dnsModifierInstance) Process(data []byte) ([]byte, error) {
|
|||||||
if len(dns.Questions) == 0 {
|
if len(dns.Questions) == 0 {
|
||||||
return nil, &modifier.ErrInvalidPacket{Err: errEmptyDNSQuestion}
|
return nil, &modifier.ErrInvalidPacket{Err: errEmptyDNSQuestion}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Hash the query name so that DNS response is fixed for a given query.
|
||||||
|
// Use a random seed to avoid determinism.
|
||||||
|
hashStringToIndex := func(b []byte, sliceLength int, seed uint32) int {
|
||||||
|
h := fnv.New32a()
|
||||||
|
seedBytes := make([]byte, 4)
|
||||||
|
binary.LittleEndian.PutUint32(seedBytes, seed)
|
||||||
|
h.Write(seedBytes)
|
||||||
|
h.Write(b)
|
||||||
|
hashValue := h.Sum32()
|
||||||
|
return int(hashValue % uint32(sliceLength))
|
||||||
|
}
|
||||||
|
|
||||||
// In practice, most if not all DNS clients only send one question
|
// In practice, most if not all DNS clients only send one question
|
||||||
// per packet, so we don't care about the rest for now.
|
// per packet, so we don't care about the rest for now.
|
||||||
q := dns.Questions[0]
|
q := dns.Questions[0]
|
||||||
switch q.Type {
|
switch q.Type {
|
||||||
case layers.DNSTypeA:
|
case layers.DNSTypeA:
|
||||||
if i.A != nil {
|
if i.A != nil {
|
||||||
|
idx := hashStringToIndex(q.Name, len(i.A), i.seed)
|
||||||
dns.Answers = []layers.DNSResourceRecord{{
|
dns.Answers = []layers.DNSResourceRecord{{
|
||||||
Name: q.Name,
|
Name: q.Name,
|
||||||
Type: layers.DNSTypeA,
|
Type: layers.DNSTypeA,
|
||||||
Class: layers.DNSClassIN,
|
Class: layers.DNSClassIN,
|
||||||
IP: i.A,
|
IP: i.A[idx],
|
||||||
}}
|
}}
|
||||||
}
|
}
|
||||||
case layers.DNSTypeAAAA:
|
case layers.DNSTypeAAAA:
|
||||||
if i.AAAA != nil {
|
if i.AAAA != nil {
|
||||||
|
idx := hashStringToIndex(q.Name, len(i.AAAA), i.seed)
|
||||||
dns.Answers = []layers.DNSResourceRecord{{
|
dns.Answers = []layers.DNSResourceRecord{{
|
||||||
Name: q.Name,
|
Name: q.Name,
|
||||||
Type: layers.DNSTypeAAAA,
|
Type: layers.DNSTypeAAAA,
|
||||||
Class: layers.DNSClassIN,
|
Class: layers.DNSClassIN,
|
||||||
IP: i.AAAA,
|
IP: i.AAAA[idx],
|
||||||
}}
|
}}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user