2021-01-18 20:59:04 -05:00
|
|
|
|
package utils
|
|
|
|
|
|
|
|
|
|
import (
|
2022-12-16 10:34:14 -05:00
|
|
|
|
"crypto/rand"
|
2024-11-22 21:21:01 -05:00
|
|
|
|
"errors"
|
2025-01-04 07:49:14 -05:00
|
|
|
|
"iter"
|
2024-12-21 11:05:41 -05:00
|
|
|
|
"maps"
|
2022-12-16 10:34:14 -05:00
|
|
|
|
"math/big"
|
2024-11-22 21:21:01 -05:00
|
|
|
|
"net/netip"
|
2021-01-28 10:19:59 -05:00
|
|
|
|
"os"
|
2021-05-27 08:48:12 -04:00
|
|
|
|
"regexp"
|
2024-12-21 11:05:41 -05:00
|
|
|
|
"slices"
|
2024-10-21 00:11:02 -04:00
|
|
|
|
"strconv"
|
2022-05-17 08:16:46 -04:00
|
|
|
|
"strings"
|
2022-03-18 11:13:22 -04:00
|
|
|
|
|
2024-10-21 00:11:02 -04:00
|
|
|
|
"golang.org/x/exp/constraints"
|
|
|
|
|
|
2022-03-18 11:13:22 -04:00
|
|
|
|
jsoniter "github.com/json-iterator/go"
|
2021-01-18 20:59:04 -05:00
|
|
|
|
)
|
|
|
|
|
|
2024-10-17 09:03:03 -04:00
|
|
|
|
var (
|
|
|
|
|
Json = jsoniter.ConfigCompatibleWithStandardLibrary
|
|
|
|
|
|
2024-10-18 10:06:01 -04:00
|
|
|
|
DNSServers = []string{"1.1.1.1:53", "223.5.5.5:53"}
|
2024-10-17 09:03:03 -04:00
|
|
|
|
)
|
2022-03-18 11:13:22 -04:00
|
|
|
|
|
2021-05-27 08:48:12 -04:00
|
|
|
|
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
|
|
|
|
|
}
|
2022-05-17 08:16:46 -04:00
|
|
|
|
|
2024-11-22 21:21:01 -05:00
|
|
|
|
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])
|
2024-11-25 08:20:08 -05:00
|
|
|
|
ip, err := netip.ParseAddr(h)
|
2024-11-22 21:21:01 -05:00
|
|
|
|
if err != nil {
|
|
|
|
|
return "", err
|
|
|
|
|
}
|
|
|
|
|
if !ip.IsValid() {
|
|
|
|
|
return "", errors.New("invalid ip")
|
|
|
|
|
}
|
2024-11-25 08:20:08 -05:00
|
|
|
|
return ip.String(), nil
|
2024-11-22 21:21:01 -05:00
|
|
|
|
}
|
|
|
|
|
|
2022-05-17 08:16:46 -04:00
|
|
|
|
// SplitIPAddr 传入/分割的v4v6混合地址,返回v4和v6地址与有效地址
|
|
|
|
|
func SplitIPAddr(v4v6Bundle string) (string, string, string) {
|
|
|
|
|
ipList := strings.Split(v4v6Bundle, "/")
|
|
|
|
|
ipv4 := ""
|
|
|
|
|
ipv6 := ""
|
|
|
|
|
validIP := ""
|
|
|
|
|
if len(ipList) > 1 {
|
|
|
|
|
// 双栈
|
|
|
|
|
ipv4 = ipList[0]
|
|
|
|
|
ipv6 = ipList[1]
|
|
|
|
|
validIP = ipv4
|
|
|
|
|
} else if len(ipList) == 1 {
|
|
|
|
|
// 仅ipv4|ipv6
|
|
|
|
|
if strings.Contains(ipList[0], ":") {
|
|
|
|
|
ipv6 = ipList[0]
|
|
|
|
|
validIP = ipv6
|
|
|
|
|
} else {
|
|
|
|
|
ipv4 = ipList[0]
|
|
|
|
|
validIP = ipv4
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return ipv4, ipv6, validIP
|
|
|
|
|
}
|
2022-06-02 21:45:11 -04:00
|
|
|
|
|
2022-06-24 08:43:38 -04:00
|
|
|
|
func IsFileExists(path string) bool {
|
|
|
|
|
_, err := os.Stat(path)
|
|
|
|
|
return err == nil
|
2022-06-02 21:45:11 -04:00
|
|
|
|
}
|
2022-12-16 10:34:14 -05:00
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
2024-08-10 22:35:19 -04:00
|
|
|
|
|
|
|
|
|
func Uint64SubInt64(a uint64, b int64) uint64 {
|
|
|
|
|
if b < 0 {
|
|
|
|
|
return a + uint64(-b)
|
|
|
|
|
}
|
2024-08-11 22:06:55 -04:00
|
|
|
|
if a < uint64(b) {
|
|
|
|
|
return 0
|
|
|
|
|
}
|
2024-08-10 22:35:19 -04:00
|
|
|
|
return a - uint64(b)
|
|
|
|
|
}
|
2024-10-20 11:23:04 -04:00
|
|
|
|
|
|
|
|
|
func IfOr[T any](a bool, x, y T) T {
|
|
|
|
|
if a {
|
|
|
|
|
return x
|
|
|
|
|
}
|
|
|
|
|
return y
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func IfOrFn[T any](a bool, x, y func() T) T {
|
|
|
|
|
if a {
|
|
|
|
|
return x()
|
|
|
|
|
}
|
|
|
|
|
return y()
|
|
|
|
|
}
|
2024-10-21 00:11:02 -04:00
|
|
|
|
|
|
|
|
|
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 ""
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-12-21 11:05:41 -05:00
|
|
|
|
|
|
|
|
|
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))
|
|
|
|
|
}
|
2024-12-22 04:23:55 -05:00
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
2025-01-04 07:49:14 -05:00
|
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|