package utils import ( "crypto/rand" "errors" "iter" "maps" "math/big" "net/netip" "regexp" "slices" "strconv" "strings" "golang.org/x/exp/constraints" jsoniter "github.com/json-iterator/go" ) var ( Json = jsoniter.ConfigCompatibleWithStandardLibrary DNSServers = []string{"8.8.8.8:53", "8.8.4.4:53", "1.1.1.1:53", "1.0.0.1:53"} ) var ipv4Re = regexp.MustCompile(`(\d*\.).*(\.\d*)`) func ipv4Desensitize(ipv4Addr string) string { return ipv4Re.ReplaceAllString(ipv4Addr, "$1****$2") } var ipv6Re = regexp.MustCompile(`(\w*:\w*:).*(:\w*:\w*)`) func ipv6Desensitize(ipv6Addr string) string { return ipv6Re.ReplaceAllString(ipv6Addr, "$1****$2") } func IPDesensitize(ipAddr string) string { ipAddr = ipv4Desensitize(ipAddr) ipAddr = ipv6Desensitize(ipAddr) return ipAddr } func IPStringToBinary(ip string) ([]byte, error) { addr, err := netip.ParseAddr(ip) if err != nil { return nil, err } b := addr.As16() return b[:], nil } func BinaryToIPString(b []byte) string { var addr16 [16]byte copy(addr16[:], b) addr := netip.AddrFrom16(addr16) return addr.Unmap().String() } func GetIPFromHeader(headerValue string) (string, error) { a := strings.Split(headerValue, ",") h := strings.TrimSpace(a[len(a)-1]) ip, err := netip.ParseAddr(h) if err != nil { return "", err } if !ip.IsValid() { return "", errors.New("invalid ip") } return ip.String(), nil } func GenerateRandomString(n int) (string, error) { const letters = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz" lettersLength := big.NewInt(int64(len(letters))) ret := make([]byte, n) for i := 0; i < n; i++ { num, err := rand.Int(rand.Reader, lettersLength) if err != nil { return "", err } ret[i] = letters[num.Int64()] } return string(ret), nil } func Uint64SubInt64(a uint64, b int64) uint64 { if b < 0 { return a + uint64(-b) } if a < uint64(b) { return 0 } return a - uint64(b) } func IfOr[T any](a bool, x, y T) T { if a { return x } return y } func Itoa[T constraints.Integer](i T) string { switch any(i).(type) { case int, int8, int16, int32, int64: return strconv.FormatInt(int64(i), 10) case uint, uint8, uint16, uint32, uint64: return strconv.FormatUint(uint64(i), 10) default: return "" } } func MapValuesToSlice[Map ~map[K]V, K comparable, V any](m Map) []V { s := make([]V, 0, len(m)) return slices.AppendSeq(s, maps.Values(m)) } func Unique[T comparable](s []T) []T { m := make(map[T]struct{}) ret := make([]T, 0, len(s)) for _, v := range s { if _, ok := m[v]; !ok { m[v] = struct{}{} ret = append(ret, v) } } return ret } func ConvertSeq[T, U any](seq iter.Seq[T], f func(e T) U) iter.Seq[U] { return func(yield func(U) bool) { for e := range seq { if !yield(f(e)) { return } } } }