2023-01-23 18:54:21 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
2023-01-28 12:56:06 +07:00
|
|
|
* Copyright (c) since 2022, v2rayA Organization <team@v2raya.org>
|
2023-01-23 18:54:21 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/binary"
|
2023-01-24 23:31:20 +07:00
|
|
|
"encoding/hex"
|
|
|
|
"fmt"
|
2023-01-23 18:54:21 +07:00
|
|
|
"net/url"
|
2023-01-28 00:50:21 +07:00
|
|
|
"reflect"
|
2023-01-24 23:31:20 +07:00
|
|
|
"strconv"
|
2023-01-23 18:54:21 +07:00
|
|
|
"strings"
|
2023-01-28 00:50:21 +07:00
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
ErrOverlayHierarchicalKey = fmt.Errorf("overlay hierarchical key")
|
2023-01-23 18:54:21 +07:00
|
|
|
)
|
|
|
|
|
|
|
|
func CloneStrings(slice []string) []string {
|
|
|
|
c := make([]string, len(slice))
|
|
|
|
copy(c, slice)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
func ARangeU32(n uint32) []uint32 {
|
|
|
|
ret := make([]uint32, n)
|
|
|
|
for i := uint32(0); i < n; i++ {
|
|
|
|
ret[i] = i
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
|
|
|
func Ipv6ByteSliceToUint32Array(_ip []byte) (ip [4]uint32) {
|
|
|
|
for j := 0; j < 16; j += 4 {
|
|
|
|
ip[j/4] = binary.LittleEndian.Uint32(_ip[j : j+4])
|
|
|
|
}
|
|
|
|
return ip
|
|
|
|
}
|
|
|
|
|
2023-01-30 16:31:42 +07:00
|
|
|
func Ipv6Uint32ArrayToByteSlice(_ip [4]uint32) (ip []byte) {
|
|
|
|
for j := 0; j < 4; j++ {
|
|
|
|
ip = binary.LittleEndian.AppendUint32(ip, _ip[j])
|
|
|
|
}
|
|
|
|
return ip
|
|
|
|
}
|
|
|
|
|
2023-01-23 18:54:21 +07:00
|
|
|
func Deduplicate(list []string) []string {
|
|
|
|
res := make([]string, 0, len(list))
|
|
|
|
m := make(map[string]struct{})
|
|
|
|
for _, v := range list {
|
|
|
|
if _, ok := m[v]; ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
m[v] = struct{}{}
|
|
|
|
res = append(res, v)
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
func Base64UrlDecode(s string) (string, error) {
|
|
|
|
s = strings.TrimSpace(s)
|
|
|
|
saver := s
|
|
|
|
if len(s)%4 > 0 {
|
|
|
|
s += strings.Repeat("=", 4-len(s)%4)
|
|
|
|
}
|
|
|
|
raw, err := base64.URLEncoding.DecodeString(s)
|
|
|
|
if err != nil {
|
|
|
|
return saver, err
|
|
|
|
}
|
|
|
|
return string(raw), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func Base64StdDecode(s string) (string, error) {
|
|
|
|
s = strings.TrimSpace(s)
|
|
|
|
saver := s
|
|
|
|
if len(s)%4 > 0 {
|
|
|
|
s += strings.Repeat("=", 4-len(s)%4)
|
|
|
|
}
|
|
|
|
raw, err := base64.StdEncoding.DecodeString(s)
|
|
|
|
if err != nil {
|
|
|
|
return saver, err
|
|
|
|
}
|
|
|
|
return string(raw), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetValue(values *url.Values, key string, value string) {
|
|
|
|
if value == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
values.Set(key, value)
|
|
|
|
}
|
2023-01-24 23:31:20 +07:00
|
|
|
|
|
|
|
func ParseMac(mac string) (addr [6]byte, err error) {
|
|
|
|
fields := strings.SplitN(mac, ":", 6)
|
|
|
|
if len(fields) != 6 {
|
|
|
|
return addr, fmt.Errorf("invalid mac: %v", mac)
|
|
|
|
}
|
|
|
|
for i, field := range fields {
|
|
|
|
v, err := hex.DecodeString(field)
|
|
|
|
if err != nil {
|
|
|
|
return addr, fmt.Errorf("parse mac %v: %w", mac, err)
|
|
|
|
}
|
|
|
|
if len(v) != 1 {
|
|
|
|
return addr, fmt.Errorf("invalid mac: %v", mac)
|
|
|
|
}
|
|
|
|
addr[i] = v[0]
|
|
|
|
}
|
|
|
|
return addr, nil
|
|
|
|
}
|
|
|
|
|
2023-01-29 06:31:52 +07:00
|
|
|
func ParsePortRange(pr string) (portRange [2]uint16, err error) {
|
2023-01-24 23:31:20 +07:00
|
|
|
fields := strings.SplitN(pr, "-", 2)
|
|
|
|
for i, field := range fields {
|
|
|
|
if field == "" {
|
|
|
|
return portRange, fmt.Errorf("bad port range: %v", pr)
|
|
|
|
}
|
|
|
|
port, err := strconv.Atoi(field)
|
|
|
|
if err != nil {
|
|
|
|
return portRange, err
|
|
|
|
}
|
|
|
|
if port < 0 || port > 0xffff {
|
|
|
|
return portRange, fmt.Errorf("port %v exceeds uint16 range", port)
|
|
|
|
}
|
2023-01-29 06:31:52 +07:00
|
|
|
portRange[i] = uint16(port)
|
2023-01-24 23:31:20 +07:00
|
|
|
}
|
|
|
|
if len(fields) == 1 {
|
|
|
|
portRange[1] = portRange[0]
|
|
|
|
}
|
|
|
|
return portRange, nil
|
|
|
|
}
|
2023-01-28 00:50:21 +07:00
|
|
|
|
|
|
|
func SetValueHierarchicalMap(m map[string]interface{}, key string, val interface{}) error {
|
|
|
|
keys := strings.Split(key, ".")
|
|
|
|
lastKey := keys[len(keys)-1]
|
|
|
|
keys = keys[:len(keys)-1]
|
|
|
|
p := &m
|
|
|
|
for _, key := range keys {
|
|
|
|
if v, ok := (*p)[key]; ok {
|
|
|
|
vv, ok := v.(map[string]interface{})
|
|
|
|
if !ok {
|
|
|
|
return ErrOverlayHierarchicalKey
|
|
|
|
}
|
|
|
|
p = &vv
|
|
|
|
} else {
|
|
|
|
(*p)[key] = make(map[string]interface{})
|
|
|
|
vv := (*p)[key].(map[string]interface{})
|
|
|
|
p = &vv
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(*p)[lastKey] = val
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetValueHierarchicalStruct(m interface{}, key string, val string) error {
|
|
|
|
ifv, err := GetValueHierarchicalStruct(m, key)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !FuzzyDecode(ifv.Addr().Interface(), val) {
|
|
|
|
return fmt.Errorf("type does not match: type \"%v\" and value \"%v\"", ifv.Kind(), val)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetValueHierarchicalStruct(m interface{}, key string) (reflect.Value, error) {
|
|
|
|
keys := strings.Split(key, ".")
|
|
|
|
ifv := reflect.Indirect(reflect.ValueOf(m))
|
|
|
|
ift := ifv.Type()
|
|
|
|
lastK := ""
|
|
|
|
for _, k := range keys {
|
|
|
|
found := false
|
|
|
|
if ift.Kind() == reflect.Struct {
|
|
|
|
for i := 0; i < ifv.NumField(); i++ {
|
|
|
|
name, ok := ift.Field(i).Tag.Lookup("mapstructure")
|
|
|
|
if ok && name == k {
|
|
|
|
found = true
|
|
|
|
ifv = ifv.Field(i)
|
|
|
|
ift = ifv.Type()
|
|
|
|
lastK = k
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
return reflect.Value{}, fmt.Errorf(`unexpected key "%v": "%v" (%v type) has no member "%v"`, key, lastK, ift.Kind().String(), k)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ifv, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func FuzzyDecode(to interface{}, val string) bool {
|
|
|
|
v := reflect.Indirect(reflect.ValueOf(to))
|
|
|
|
switch v.Kind() {
|
|
|
|
case reflect.Int:
|
|
|
|
i, err := strconv.ParseInt(val, 10, strconv.IntSize)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
case reflect.Int8:
|
|
|
|
i, err := strconv.ParseInt(val, 10, 8)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
case reflect.Int16:
|
|
|
|
i, err := strconv.ParseInt(val, 10, 16)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
case reflect.Int32:
|
|
|
|
i, err := strconv.ParseInt(val, 10, 32)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
case reflect.Int64:
|
|
|
|
switch v.Interface().(type) {
|
|
|
|
case time.Duration:
|
|
|
|
duration, err := time.ParseDuration(val)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.Set(reflect.ValueOf(duration))
|
|
|
|
default:
|
|
|
|
i, err := strconv.ParseInt(val, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
}
|
|
|
|
case reflect.Uint:
|
|
|
|
i, err := strconv.ParseUint(val, 10, strconv.IntSize)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Uint8:
|
|
|
|
i, err := strconv.ParseUint(val, 10, 8)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Uint16:
|
|
|
|
i, err := strconv.ParseUint(val, 10, 16)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Uint32:
|
|
|
|
i, err := strconv.ParseUint(val, 10, 32)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Uint64:
|
|
|
|
i, err := strconv.ParseUint(val, 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Bool:
|
|
|
|
if val == "true" || val == "1" {
|
|
|
|
v.SetBool(true)
|
|
|
|
} else if val == "false" || val == "0" {
|
|
|
|
v.SetBool(false)
|
|
|
|
} else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
case reflect.String:
|
|
|
|
v.SetString(val)
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|