mirror of
https://github.com/SagerNet/sing-box.git
synced 2024-11-10 02:53:12 +00:00
103 lines
2 KiB
Go
103 lines
2 KiB
Go
package geosite
|
|
|
|
import (
|
|
"io"
|
|
"os"
|
|
|
|
E "github.com/sagernet/sing/common/exceptions"
|
|
"github.com/sagernet/sing/common/rw"
|
|
)
|
|
|
|
type Reader struct {
|
|
reader io.ReadSeeker
|
|
domainIndex map[string]int
|
|
domainLength map[string]int
|
|
}
|
|
|
|
func Open(path string) (*Reader, error) {
|
|
content, err := os.Open(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
reader := &Reader{
|
|
reader: content,
|
|
}
|
|
err = reader.readMetadata()
|
|
if err != nil {
|
|
content.Close()
|
|
return nil, err
|
|
}
|
|
return reader, nil
|
|
}
|
|
|
|
func (r *Reader) readMetadata() error {
|
|
version, err := rw.ReadByte(r.reader)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if version != 0 {
|
|
return E.New("unknown version")
|
|
}
|
|
entryLength, err := rw.ReadUVariant(r.reader)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
keys := make([]string, entryLength)
|
|
domainIndex := make(map[string]int)
|
|
domainLength := make(map[string]int)
|
|
for i := 0; i < int(entryLength); i++ {
|
|
var (
|
|
code string
|
|
codeIndex uint64
|
|
codeLength uint64
|
|
)
|
|
code, err = rw.ReadVString(r.reader)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
keys[i] = code
|
|
codeIndex, err = rw.ReadUVariant(r.reader)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
codeLength, err = rw.ReadUVariant(r.reader)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
domainIndex[code] = int(codeIndex)
|
|
domainLength[code] = int(codeLength)
|
|
}
|
|
r.domainIndex = domainIndex
|
|
r.domainLength = domainLength
|
|
return nil
|
|
}
|
|
|
|
func (r *Reader) Read(code string) ([]Item, error) {
|
|
if _, exists := r.domainIndex[code]; !exists {
|
|
return nil, E.New("code ", code, " not exists!")
|
|
}
|
|
counter := &rw.ReadCounter{Reader: r.reader}
|
|
domain := make([]Item, r.domainLength[code])
|
|
for i := range domain {
|
|
var (
|
|
item Item
|
|
err error
|
|
)
|
|
item.Type, err = rw.ReadByte(counter)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
item.Value, err = rw.ReadVString(counter)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
domain[i] = item
|
|
}
|
|
_, err := r.reader.Seek(int64(r.domainIndex[code])-counter.Count(), io.SeekCurrent)
|
|
return domain, err
|
|
}
|
|
|
|
func (r *Reader) Upstream() any {
|
|
return r.reader
|
|
}
|