mirror of
https://github.com/nezhahq/nezha.git
synced 2025-01-22 12:48:14 -05:00
d21179609d
* feat(waf): return ip in string literal * rename
178 lines
3.4 KiB
Go
178 lines
3.4 KiB
Go
package utils
|
||
|
||
import (
|
||
"crypto/rand"
|
||
"errors"
|
||
"iter"
|
||
"maps"
|
||
"math/big"
|
||
"net/netip"
|
||
"os"
|
||
"regexp"
|
||
"slices"
|
||
"strconv"
|
||
"strings"
|
||
|
||
"golang.org/x/exp/constraints"
|
||
|
||
jsoniter "github.com/json-iterator/go"
|
||
)
|
||
|
||
var (
|
||
Json = jsoniter.ConfigCompatibleWithStandardLibrary
|
||
|
||
DNSServers = []string{"1.1.1.1:53", "223.5.5.5: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
|
||
}
|
||
|
||
// 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
|
||
}
|
||
|
||
func IsFileExists(path string) bool {
|
||
_, err := os.Stat(path)
|
||
return err == 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 IfOrFn[T any](a bool, x, y func() 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
|
||
}
|
||
}
|
||
}
|
||
}
|