mirror of
				https://github.com/bolucat/Archive.git
				synced 2025-10-31 03:46:52 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			228 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			228 lines
		
	
	
		
			5.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package v2ray
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"log"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"sort"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/Loyalsoldier/geoip/lib"
 | |
| 	router "github.com/v2fly/v2ray-core/v5/app/router/routercommon"
 | |
| 	"github.com/v2fly/v2ray-core/v5/infra/conf/rule"
 | |
| 	"google.golang.org/protobuf/proto"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	typeGeoIPdatOut = "v2rayGeoIPDat"
 | |
| 	descGeoIPdatOut = "Convert data to V2Ray GeoIP dat format"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	defaultOutputName = "geoip.dat"
 | |
| 	defaultOutputDir  = filepath.Join("./", "output", "dat")
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	lib.RegisterOutputConfigCreator(typeGeoIPdatOut, func(action lib.Action, data json.RawMessage) (lib.OutputConverter, error) {
 | |
| 		return newGeoIPDat(action, data)
 | |
| 	})
 | |
| 	lib.RegisterOutputConverter(typeGeoIPdatOut, &geoIPDatOut{
 | |
| 		Description: descGeoIPdatOut,
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func newGeoIPDat(action lib.Action, data json.RawMessage) (lib.OutputConverter, error) {
 | |
| 	var tmp struct {
 | |
| 		OutputName     string     `json:"outputName"`
 | |
| 		OutputDir      string     `json:"outputDir"`
 | |
| 		Want           []string   `json:"wantedList"`
 | |
| 		OneFilePerList bool       `json:"oneFilePerList"`
 | |
| 		OnlyIPType     lib.IPType `json:"onlyIPType"`
 | |
| 	}
 | |
| 
 | |
| 	if len(data) > 0 {
 | |
| 		if err := json.Unmarshal(data, &tmp); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if tmp.OutputName == "" {
 | |
| 		tmp.OutputName = defaultOutputName
 | |
| 	}
 | |
| 
 | |
| 	if tmp.OutputDir == "" {
 | |
| 		tmp.OutputDir = defaultOutputDir
 | |
| 	}
 | |
| 
 | |
| 	return &geoIPDatOut{
 | |
| 		Type:           typeGeoIPdatOut,
 | |
| 		Action:         action,
 | |
| 		Description:    descGeoIPdatOut,
 | |
| 		OutputName:     tmp.OutputName,
 | |
| 		OutputDir:      tmp.OutputDir,
 | |
| 		Want:           tmp.Want,
 | |
| 		OneFilePerList: tmp.OneFilePerList,
 | |
| 		OnlyIPType:     tmp.OnlyIPType,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| type geoIPDatOut struct {
 | |
| 	Type           string
 | |
| 	Action         lib.Action
 | |
| 	Description    string
 | |
| 	OutputName     string
 | |
| 	OutputDir      string
 | |
| 	Want           []string
 | |
| 	OneFilePerList bool
 | |
| 	OnlyIPType     lib.IPType
 | |
| }
 | |
| 
 | |
| func (g *geoIPDatOut) GetType() string {
 | |
| 	return g.Type
 | |
| }
 | |
| 
 | |
| func (g *geoIPDatOut) GetAction() lib.Action {
 | |
| 	return g.Action
 | |
| }
 | |
| 
 | |
| func (g *geoIPDatOut) GetDescription() string {
 | |
| 	return g.Description
 | |
| }
 | |
| 
 | |
| func (g *geoIPDatOut) Output(container lib.Container) error {
 | |
| 	// Filter want list
 | |
| 	wantList := make(map[string]bool)
 | |
| 	for _, want := range g.Want {
 | |
| 		if want = strings.ToUpper(strings.TrimSpace(want)); want != "" {
 | |
| 			wantList[want] = true
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	geoIPList := new(router.GeoIPList)
 | |
| 	geoIPList.Entry = make([]*router.GeoIP, 0, 300)
 | |
| 	updated := false
 | |
| 	switch len(wantList) {
 | |
| 	case 0:
 | |
| 		for entry := range container.Loop() {
 | |
| 			geoIP, err := g.generateGeoIP(entry)
 | |
| 			if err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			geoIPList.Entry = append(geoIPList.Entry, geoIP)
 | |
| 			updated = true
 | |
| 
 | |
| 			if g.OneFilePerList {
 | |
| 				geoIPBytes, err := proto.Marshal(geoIPList)
 | |
| 				if err != nil {
 | |
| 					return err
 | |
| 				}
 | |
| 				filename := strings.ToLower(entry.GetName()) + ".dat"
 | |
| 				if err := g.writeFile(filename, geoIPBytes); err != nil {
 | |
| 					return err
 | |
| 				}
 | |
| 				geoIPList.Entry = nil
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 	default:
 | |
| 		for name := range wantList {
 | |
| 			entry, found := container.GetEntry(name)
 | |
| 			if !found {
 | |
| 				log.Printf("❌ entry %s not found", name)
 | |
| 				continue
 | |
| 			}
 | |
| 			geoIP, err := g.generateGeoIP(entry)
 | |
| 			if err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			geoIPList.Entry = append(geoIPList.Entry, geoIP)
 | |
| 			updated = true
 | |
| 
 | |
| 			if g.OneFilePerList {
 | |
| 				geoIPBytes, err := proto.Marshal(geoIPList)
 | |
| 				if err != nil {
 | |
| 					return err
 | |
| 				}
 | |
| 				filename := strings.ToLower(entry.GetName()) + ".dat"
 | |
| 				if err := g.writeFile(filename, geoIPBytes); err != nil {
 | |
| 					return err
 | |
| 				}
 | |
| 				geoIPList.Entry = nil
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Sort to make reproducible builds
 | |
| 	g.sort(geoIPList)
 | |
| 
 | |
| 	if !g.OneFilePerList && updated {
 | |
| 		geoIPBytes, err := proto.Marshal(geoIPList)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		if err := g.writeFile(g.OutputName, geoIPBytes); err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (g *geoIPDatOut) generateGeoIP(entry *lib.Entry) (*router.GeoIP, error) {
 | |
| 	var entryCidr []string
 | |
| 	var err error
 | |
| 	switch g.OnlyIPType {
 | |
| 	case lib.IPv4:
 | |
| 		entryCidr, err = entry.MarshalText(lib.IgnoreIPv6)
 | |
| 	case lib.IPv6:
 | |
| 		entryCidr, err = entry.MarshalText(lib.IgnoreIPv4)
 | |
| 	default:
 | |
| 		entryCidr, err = entry.MarshalText()
 | |
| 	}
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	v2rayCIDR := make([]*router.CIDR, 0, 1024)
 | |
| 	for _, cidrStr := range entryCidr {
 | |
| 		cidr, err := rule.ParseIP(cidrStr)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		v2rayCIDR = append(v2rayCIDR, cidr)
 | |
| 	}
 | |
| 
 | |
| 	if len(v2rayCIDR) > 0 {
 | |
| 		return &router.GeoIP{
 | |
| 			CountryCode: entry.GetName(),
 | |
| 			Cidr:        v2rayCIDR,
 | |
| 		}, nil
 | |
| 	}
 | |
| 
 | |
| 	return nil, fmt.Errorf("entry %s has no CIDR", entry.GetName())
 | |
| }
 | |
| 
 | |
| // Sort by country code to make reproducible builds
 | |
| func (g *geoIPDatOut) sort(list *router.GeoIPList) {
 | |
| 	sort.SliceStable(list.Entry, func(i, j int) bool {
 | |
| 		return list.Entry[i].CountryCode < list.Entry[j].CountryCode
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func (g *geoIPDatOut) writeFile(filename string, geoIPBytes []byte) error {
 | |
| 	if err := os.MkdirAll(g.OutputDir, 0755); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	if err := os.WriteFile(filepath.Join(g.OutputDir, filename), geoIPBytes, 0644); err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 
 | |
| 	log.Printf("✅ [%s] %s --> %s", g.Type, filename, g.OutputDir)
 | |
| 
 | |
| 	return nil
 | |
| }
 | 
