2023-01-23 18:54:21 +07:00
|
|
|
/*
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
2023-03-14 14:01:55 +07:00
|
|
|
* Copyright (c) 2022-2023, daeuniverse Organization <dae@v2raya.org>
|
2023-01-23 18:54:21 +07:00
|
|
|
*/
|
|
|
|
|
|
|
|
package common
|
|
|
|
|
|
|
|
import (
|
2023-02-15 00:53:53 +07:00
|
|
|
"crypto/aes"
|
|
|
|
"crypto/cipher"
|
2023-01-23 18:54:21 +07:00
|
|
|
"encoding/base64"
|
|
|
|
"encoding/binary"
|
2023-01-24 23:31:20 +07:00
|
|
|
"encoding/hex"
|
|
|
|
"fmt"
|
2023-06-04 10:38:05 +07:00
|
|
|
"github.com/mzz2017/softwind/netproxy"
|
2023-02-13 17:26:31 +07:00
|
|
|
"net/netip"
|
2023-01-23 18:54:21 +07:00
|
|
|
"net/url"
|
2023-02-09 19:17:45 +07:00
|
|
|
"path/filepath"
|
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"
|
2023-02-21 15:10:44 +07:00
|
|
|
"unsafe"
|
2023-04-23 12:27:29 +07:00
|
|
|
|
|
|
|
internal "github.com/daeuniverse/dae/pkg/ebpf_internal"
|
|
|
|
"github.com/vishvananda/netlink"
|
|
|
|
"golang.org/x/net/dns/dnsmessage"
|
|
|
|
"golang.org/x/sys/unix"
|
2023-01-28 00:50:21 +07:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
ErrOverlayHierarchicalKey = fmt.Errorf("overlay hierarchical key")
|
2023-01-23 18:54:21 +07:00
|
|
|
)
|
|
|
|
|
2023-02-08 19:15:24 +07:00
|
|
|
type UrlOrEmpty struct {
|
|
|
|
Url *url.URL
|
|
|
|
Empty bool
|
|
|
|
}
|
|
|
|
|
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 {
|
2023-02-25 01:38:21 +07:00
|
|
|
ip[j/4] = internal.NativeEndian.Uint32(_ip[j : j+4])
|
2023-01-23 18:54:21 +07:00
|
|
|
}
|
|
|
|
return ip
|
|
|
|
}
|
|
|
|
|
2023-02-13 02:41:59 +07:00
|
|
|
func Ipv6ByteSliceToUint8Array(_ip []byte) (ip [16]uint8) {
|
|
|
|
copy(ip[:], _ip)
|
|
|
|
return ip
|
|
|
|
}
|
|
|
|
|
2023-01-30 16:31:42 +07:00
|
|
|
func Ipv6Uint32ArrayToByteSlice(_ip [4]uint32) (ip []byte) {
|
2023-02-05 09:29:00 +07:00
|
|
|
ip = make([]byte, 16)
|
2023-01-30 16:31:42 +07:00
|
|
|
for j := 0; j < 4; j++ {
|
2023-02-25 01:38:21 +07:00
|
|
|
internal.NativeEndian.PutUint32(ip[j*4:], _ip[j])
|
2023-01-30 16:31:42 +07:00
|
|
|
}
|
|
|
|
return ip
|
|
|
|
}
|
|
|
|
|
2023-01-23 18:54:21 +07:00
|
|
|
func Deduplicate(list []string) []string {
|
2023-03-24 12:30:02 +07:00
|
|
|
if list == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2023-01-23 18:54:21 +07:00
|
|
|
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:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseInt(val, 0, strconv.IntSize)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
case reflect.Int8:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseInt(val, 0, 8)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
case reflect.Int16:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseInt(val, 0, 16)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
case reflect.Int32:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseInt(val, 0, 32)
|
2023-01-28 00:50:21 +07:00
|
|
|
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:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseInt(val, 0, 64)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetInt(i)
|
|
|
|
}
|
|
|
|
case reflect.Uint:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseUint(val, 0, strconv.IntSize)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Uint8:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseUint(val, 0, 8)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Uint16:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseUint(val, 0, 16)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Uint32:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseUint(val, 0, 32)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Uint64:
|
2023-06-04 10:38:05 +07:00
|
|
|
i, err := strconv.ParseUint(val, 0, 64)
|
2023-01-28 00:50:21 +07:00
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
v.SetUint(i)
|
|
|
|
case reflect.Bool:
|
2023-02-11 12:34:12 +07:00
|
|
|
switch strings.ToLower(val) {
|
2023-02-20 17:06:54 +07:00
|
|
|
case "true", "t", "1", "y", "yes", "on":
|
2023-01-28 00:50:21 +07:00
|
|
|
v.SetBool(true)
|
2023-02-20 17:06:54 +07:00
|
|
|
case "false", "f", "0", "n", "no", "off":
|
2023-01-28 00:50:21 +07:00
|
|
|
v.SetBool(false)
|
2023-02-11 12:34:12 +07:00
|
|
|
default:
|
2023-01-28 00:50:21 +07:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
case reflect.String:
|
|
|
|
v.SetString(val)
|
2023-02-08 15:07:23 +07:00
|
|
|
case reflect.Struct:
|
|
|
|
switch v.Interface().(type) {
|
2023-02-08 19:15:24 +07:00
|
|
|
case UrlOrEmpty:
|
2023-02-08 15:07:23 +07:00
|
|
|
if val == "" {
|
2023-02-08 19:15:24 +07:00
|
|
|
v.Set(reflect.ValueOf(UrlOrEmpty{
|
2023-02-08 15:07:23 +07:00
|
|
|
Url: nil,
|
|
|
|
Empty: true,
|
|
|
|
}))
|
|
|
|
} else {
|
|
|
|
u, err := url.Parse(val)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
2023-02-08 19:15:24 +07:00
|
|
|
v.Set(reflect.ValueOf(UrlOrEmpty{
|
2023-02-08 15:07:23 +07:00
|
|
|
Url: u,
|
|
|
|
Empty: false,
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
2023-04-29 00:08:46 +07:00
|
|
|
case reflect.Slice:
|
|
|
|
switch v.Interface().(type) {
|
|
|
|
case []string:
|
|
|
|
v.Set(reflect.ValueOf(strings.Split(val, ",")))
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
2023-01-28 00:50:21 +07:00
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
2023-02-09 19:17:45 +07:00
|
|
|
|
2023-02-09 22:17:49 +07:00
|
|
|
func EnsureFileInSubDir(filePath string, dir string) (err error) {
|
2023-02-09 19:17:45 +07:00
|
|
|
fileDir := filepath.Dir(filePath)
|
|
|
|
if len(dir) == 0 {
|
|
|
|
return fmt.Errorf("bad dir: %v", dir)
|
|
|
|
}
|
|
|
|
rel, err := filepath.Rel(dir, fileDir)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if strings.HasPrefix(rel, "..") {
|
|
|
|
return fmt.Errorf("file is out of scope: %v", rel)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2023-02-09 22:17:49 +07:00
|
|
|
|
|
|
|
func MapKeys(m interface{}) (keys []string, err error) {
|
|
|
|
v := reflect.ValueOf(m)
|
|
|
|
if v.Kind() != reflect.Map {
|
|
|
|
return nil, fmt.Errorf("MapKeys requires map[string]*")
|
|
|
|
}
|
|
|
|
if v.Type().Key().Kind() != reflect.String {
|
|
|
|
return nil, fmt.Errorf("MapKeys requires map[string]*")
|
|
|
|
}
|
|
|
|
_keys := v.MapKeys()
|
|
|
|
keys = make([]string, 0, len(_keys))
|
|
|
|
for _, k := range _keys {
|
|
|
|
keys = append(keys, k.String())
|
|
|
|
}
|
|
|
|
return keys, nil
|
|
|
|
}
|
2023-02-12 10:33:12 +07:00
|
|
|
|
|
|
|
func GetTagFromLinkLikePlaintext(link string) (tag string, afterTag string) {
|
|
|
|
iColon := strings.Index(link, ":")
|
|
|
|
if iColon == -1 {
|
|
|
|
return "", link
|
|
|
|
}
|
|
|
|
// If first colon is like "://" in "scheme://linkbody", no tag is present.
|
|
|
|
if strings.HasPrefix(link[iColon:], "://") {
|
|
|
|
return "", link
|
|
|
|
}
|
|
|
|
// Else tag is the part before colon.
|
|
|
|
return link[:iColon], link[iColon+1:]
|
|
|
|
}
|
2023-02-12 16:17:51 +07:00
|
|
|
|
|
|
|
func BoolToString(b bool) string {
|
|
|
|
if b {
|
|
|
|
return "1"
|
|
|
|
} else {
|
|
|
|
return "0"
|
|
|
|
}
|
|
|
|
}
|
2023-02-13 17:26:31 +07:00
|
|
|
|
2023-02-25 01:38:21 +07:00
|
|
|
func ConvergeAddr(addr netip.Addr) netip.Addr {
|
2023-02-13 17:26:31 +07:00
|
|
|
if addr.Is4In6() {
|
|
|
|
addr = netip.AddrFrom4(addr.As4())
|
|
|
|
}
|
|
|
|
return addr
|
|
|
|
}
|
2023-02-15 00:53:53 +07:00
|
|
|
|
2023-02-25 01:38:21 +07:00
|
|
|
func ConvergeAddrPort(addrPort netip.AddrPort) netip.AddrPort {
|
|
|
|
if addrPort.Addr().Is4In6() {
|
|
|
|
return netip.AddrPortFrom(netip.AddrFrom4(addrPort.Addr().As4()), addrPort.Port())
|
|
|
|
}
|
|
|
|
return addrPort
|
|
|
|
}
|
|
|
|
|
2023-02-15 00:53:53 +07:00
|
|
|
func NewGcm(key []byte) (cipher.AEAD, error) {
|
|
|
|
block, err := aes.NewCipher(key)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return cipher.NewGCM(block)
|
|
|
|
}
|
|
|
|
|
|
|
|
func AddrToDnsType(addr netip.Addr) dnsmessage.Type {
|
|
|
|
if addr.Is4() {
|
|
|
|
return dnsmessage.TypeA
|
|
|
|
} else {
|
|
|
|
return dnsmessage.TypeAAAA
|
|
|
|
}
|
|
|
|
}
|
2023-02-21 15:10:44 +07:00
|
|
|
|
|
|
|
// Htons converts the unsigned short integer hostshort from host byte order to network byte order.
|
|
|
|
func Htons(i uint16) uint16 {
|
|
|
|
b := make([]byte, 2)
|
|
|
|
binary.BigEndian.PutUint16(b, i)
|
|
|
|
return *(*uint16)(unsafe.Pointer(&b[0]))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ntohs converts the unsigned short integer hostshort from host byte order to network byte order.
|
|
|
|
func Ntohs(i uint16) uint16 {
|
|
|
|
bytes := *(*[2]byte)(unsafe.Pointer(&i))
|
|
|
|
return binary.BigEndian.Uint16(bytes[:])
|
|
|
|
}
|
2023-03-24 12:30:02 +07:00
|
|
|
|
|
|
|
func GetDefaultIfnames() (defaultIfs []string, err error) {
|
|
|
|
linkList, err := netlink.LinkList()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
nextLink:
|
|
|
|
for _, link := range linkList {
|
|
|
|
if link.Attrs().Flags&unix.RTF_UP != unix.RTF_UP {
|
|
|
|
// Interface is down.
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, family := range []int{unix.AF_INET, unix.AF_INET6} {
|
|
|
|
rs, err := netlink.RouteList(link, family)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, route := range rs {
|
|
|
|
if route.Dst != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// Have no dst, it is a default route.
|
|
|
|
defaultIfs = append(defaultIfs, link.Attrs().Name)
|
|
|
|
continue nextLink
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Deduplicate(defaultIfs), nil
|
|
|
|
}
|
2023-05-13 14:38:28 +07:00
|
|
|
|
2023-06-04 10:38:05 +07:00
|
|
|
func MagicNetwork(network string, mark uint32) string {
|
|
|
|
if mark == 0 {
|
|
|
|
return network
|
|
|
|
} else {
|
|
|
|
return netproxy.MagicNetwork{
|
|
|
|
Network: network,
|
|
|
|
Mark: mark,
|
|
|
|
}.Encode()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-13 14:38:28 +07:00
|
|
|
func IsValidHttpMethod(method string) bool {
|
|
|
|
switch method {
|
|
|
|
case "GET", "POST", "PUT", "PATCH", "DELETE", "COPY", "HEAD", "OPTIONS", "LINK", "UNLINK", "PURGE", "LOCK", "UNLOCK", "PROPFIND", "CONNECT", "TRACE":
|
|
|
|
return true
|
|
|
|
default:
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
2023-05-30 21:10:32 +07:00
|
|
|
|
|
|
|
func StringSet(list []string) map[string]struct{} {
|
|
|
|
m := make(map[string]struct{})
|
|
|
|
for _, s := range list {
|
|
|
|
m[s] = struct{}{}
|
|
|
|
}
|
|
|
|
return m
|
|
|
|
}
|