package main import ( "bufio" "fmt" "log" "net/netip" "os" "strings" "github.com/Loyalsoldier/geoip/lib" "github.com/Loyalsoldier/geoip/plugin/maxmind" "github.com/Loyalsoldier/geoip/plugin/mihomo" "github.com/Loyalsoldier/geoip/plugin/plaintext" "github.com/Loyalsoldier/geoip/plugin/singbox" "github.com/Loyalsoldier/geoip/plugin/special" "github.com/Loyalsoldier/geoip/plugin/v2ray" "github.com/spf13/cobra" ) var supportedInputFormats = map[string]bool{ strings.ToLower("clashRuleSet"): true, strings.ToLower("clashRuleSetClassical"): true, strings.ToLower("dbipCountryMMDB"): true, strings.ToLower("ipinfoCountryMMDB"): true, strings.ToLower("maxmindMMDB"): true, strings.ToLower("mihomoMRS"): true, strings.ToLower("singboxSRS"): true, strings.ToLower("surgeRuleSet"): true, strings.ToLower("text"): true, strings.ToLower("v2rayGeoIPDat"): true, } func init() { rootCmd.AddCommand(lookupCmd) lookupCmd.Flags().StringP("format", "f", "", "(Required) The input format. Available formats: text, v2rayGeoIPDat, maxmindMMDB, dbipCountryMMDB, ipinfoCountryMMDB, mihomoMRS, singboxSRS, clashRuleSet, clashRuleSetClassical, surgeRuleSet") lookupCmd.Flags().StringP("uri", "u", "", "URI of the input file, support both local file path and remote HTTP(S) URL. (Cannot be used with \"dir\" flag)") lookupCmd.Flags().StringP("dir", "d", "", "Path to the input directory. The filename without extension will be as the name of the list. (Cannot be used with \"uri\" flag)") lookupCmd.Flags().StringSliceP("searchlist", "l", []string{}, "The lists to search from, separated by comma") lookupCmd.MarkFlagRequired("format") lookupCmd.MarkFlagsOneRequired("uri", "dir") lookupCmd.MarkFlagsMutuallyExclusive("uri", "dir") lookupCmd.MarkFlagDirname("dir") } var lookupCmd = &cobra.Command{ Use: "lookup", Aliases: []string{"find"}, Short: "Lookup if specified IP or CIDR is in specified lists", Args: cobra.RangeArgs(0, 1), Run: func(cmd *cobra.Command, args []string) { // Validate format format, _ := cmd.Flags().GetString("format") format = strings.ToLower(strings.TrimSpace(format)) if _, found := supportedInputFormats[format]; !found { log.Fatal("unsupported input format") } // Set name name := "true" // Get uri uri, _ := cmd.Flags().GetString("uri") // Get dir dir, _ := cmd.Flags().GetString("dir") // Get searchlist searchList, _ := cmd.Flags().GetStringSlice("searchlist") switch len(args) > 0 { case true: // With search arg, run in once mode search := strings.ToLower(strings.TrimSpace(args[0])) if !isValidIPOrCIDR(search) { fmt.Println("false") return } instance, err := lib.NewInstance() if err != nil { log.Fatal(err) } instance.AddInput(getInputForLookup(format, name, uri, dir)) instance.AddOutput(getOutputForLookup(search, searchList...)) if err := instance.Run(); err != nil { log.Fatal(err) } case false: // No search arg, run in REPL mode instance, err := lib.NewInstance() if err != nil { log.Fatal(err) } instance.AddInput(getInputForLookup(format, name, uri, dir)) container := lib.NewContainer() if err := instance.RunInput(container); err != nil { log.Fatal(err) } fmt.Println(`Enter IP or CIDR (type "exit" to quit):`) fmt.Print(">> ") scanner := bufio.NewScanner(os.Stdin) for scanner.Scan() { search := strings.ToLower(strings.TrimSpace(scanner.Text())) if search == "" { fmt.Println() fmt.Print(">> ") continue } if search == "exit" || search == `"exit"` { break } if !isValidIPOrCIDR(search) { fmt.Println("false") fmt.Println() fmt.Print(">> ") continue } instance.ResetOutput() instance.AddOutput(getOutputForLookup(search, searchList...)) if err := instance.RunOutput(container); err != nil { log.Fatal(err) } fmt.Println() fmt.Print(">> ") } if err := scanner.Err(); err != nil { log.Fatal(err) } } }, } // Check if the input is a valid IP or CIDR func isValidIPOrCIDR(search string) bool { if search == "" { return false } var err error switch strings.Contains(search, "/") { case true: // CIDR _, err = netip.ParsePrefix(search) case false: // IP _, err = netip.ParseAddr(search) } return err == nil } func getInputForLookup(format, name, uri, dir string) lib.InputConverter { var input lib.InputConverter switch strings.ToLower(format) { case strings.ToLower(maxmind.TypeGeoLite2CountryMMDBIn): input = &maxmind.GeoLite2CountryMMDBIn{ Type: maxmind.TypeGeoLite2CountryMMDBIn, Action: lib.ActionAdd, Description: maxmind.DescGeoLite2CountryMMDBIn, URI: uri, } case strings.ToLower(maxmind.TypeDBIPCountryMMDBIn): input = &maxmind.GeoLite2CountryMMDBIn{ Type: maxmind.TypeDBIPCountryMMDBIn, Action: lib.ActionAdd, Description: maxmind.DescDBIPCountryMMDBIn, URI: uri, } case strings.ToLower(maxmind.TypeIPInfoCountryMMDBIn): input = &maxmind.GeoLite2CountryMMDBIn{ Type: maxmind.TypeIPInfoCountryMMDBIn, Action: lib.ActionAdd, Description: maxmind.DescIPInfoCountryMMDBIn, URI: uri, } case strings.ToLower(mihomo.TypeMRSIn): input = &mihomo.MRSIn{ Type: mihomo.TypeMRSIn, Action: lib.ActionAdd, Description: mihomo.DescMRSIn, Name: name, URI: uri, InputDir: dir, } case strings.ToLower(singbox.TypeSRSIn): input = &singbox.SRSIn{ Type: singbox.TypeSRSIn, Action: lib.ActionAdd, Description: singbox.DescSRSIn, Name: name, URI: uri, InputDir: dir, } case strings.ToLower(v2ray.TypeGeoIPDatIn): input = &v2ray.GeoIPDatIn{ Type: v2ray.TypeGeoIPDatIn, Action: lib.ActionAdd, Description: v2ray.DescGeoIPDatIn, URI: uri, } case strings.ToLower(plaintext.TypeTextIn): input = &plaintext.TextIn{ Type: plaintext.TypeTextIn, Action: lib.ActionAdd, Description: plaintext.DescTextIn, Name: name, URI: uri, InputDir: dir, } case strings.ToLower(plaintext.TypeClashRuleSetIPCIDRIn): input = &plaintext.TextIn{ Type: plaintext.TypeClashRuleSetIPCIDRIn, Action: lib.ActionAdd, Description: plaintext.DescClashRuleSetIPCIDRIn, Name: name, URI: uri, InputDir: dir, } case strings.ToLower(plaintext.TypeClashRuleSetClassicalIn): input = &plaintext.TextIn{ Type: plaintext.TypeClashRuleSetClassicalIn, Action: lib.ActionAdd, Description: plaintext.DescClashRuleSetClassicalIn, Name: name, URI: uri, InputDir: dir, } case strings.ToLower(plaintext.TypeSurgeRuleSetIn): input = &plaintext.TextIn{ Type: plaintext.TypeSurgeRuleSetIn, Action: lib.ActionAdd, Description: plaintext.DescSurgeRuleSetIn, Name: name, URI: uri, InputDir: dir, } default: log.Fatal("unsupported input format") } return input } func getOutputForLookup(search string, searchList ...string) lib.OutputConverter { return &special.Lookup{ Type: special.TypeLookup, Action: lib.ActionOutput, Description: special.DescLookup, Search: search, SearchList: searchList, } }