2022-07-07 00:19:05 +03:00
|
|
|
package list
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/spf13/cast"
|
|
|
|
)
|
|
|
|
|
|
|
|
var cachedPatterns = map[string]*regexp.Regexp{}
|
|
|
|
|
2022-07-10 14:13:44 +08:00
|
|
|
// ExistInSlice checks whether a comparable element exists in a slice of the same type.
|
2022-07-07 00:19:05 +03:00
|
|
|
func ExistInSlice[T comparable](item T, list []T) bool {
|
|
|
|
if len(list) == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, v := range list {
|
|
|
|
if v == item {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExistInSliceWithRegex checks whether a string exists in a slice
|
|
|
|
// either by direct match, or by a regular expression (eg. `^\w+$`).
|
|
|
|
//
|
|
|
|
// _Note: Only list items starting with '^' and ending with '$' are treated as regular expressions!_
|
|
|
|
func ExistInSliceWithRegex(str string, list []string) bool {
|
|
|
|
for _, field := range list {
|
|
|
|
isRegex := strings.HasPrefix(field, "^") && strings.HasSuffix(field, "$")
|
|
|
|
|
|
|
|
if !isRegex {
|
|
|
|
// check for direct match
|
|
|
|
if str == field {
|
|
|
|
return true
|
|
|
|
}
|
2022-07-13 00:52:09 +08:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for regex match
|
|
|
|
pattern, ok := cachedPatterns[field]
|
|
|
|
if !ok {
|
|
|
|
var err error
|
|
|
|
pattern, err = regexp.Compile(field)
|
|
|
|
if err != nil {
|
|
|
|
continue
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
2022-07-13 00:52:09 +08:00
|
|
|
// "cache" the pattern to avoid compiling it every time
|
|
|
|
cachedPatterns[field] = pattern
|
|
|
|
}
|
|
|
|
|
|
|
|
if pattern != nil && pattern.MatchString(str) {
|
|
|
|
return true
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// ToInterfaceSlice converts a generic slice to slice of interfaces.
|
|
|
|
func ToInterfaceSlice[T any](list []T) []any {
|
|
|
|
result := make([]any, len(list))
|
|
|
|
|
|
|
|
for i := range list {
|
|
|
|
result[i] = list[i]
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
// NonzeroUniques returns only the nonzero unique values from a slice.
|
|
|
|
func NonzeroUniques[T comparable](list []T) []T {
|
2022-07-13 00:52:09 +08:00
|
|
|
result := make([]T, 0, len(list))
|
|
|
|
existMap := make(map[T]struct{}, len(list))
|
2022-07-07 00:19:05 +03:00
|
|
|
|
|
|
|
var zeroVal T
|
|
|
|
|
|
|
|
for _, val := range list {
|
2022-07-13 00:52:09 +08:00
|
|
|
if _, ok := existMap[val]; ok || val == zeroVal {
|
|
|
|
continue
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
2022-07-13 00:52:09 +08:00
|
|
|
existMap[val] = struct{}{}
|
|
|
|
result = append(result, val)
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
// ToUniqueStringSlice casts `value` to a slice of non-zero unique strings.
|
2022-07-13 00:52:09 +08:00
|
|
|
func ToUniqueStringSlice(value any) (result []string) {
|
2022-07-07 00:19:05 +03:00
|
|
|
switch val := value.(type) {
|
|
|
|
case nil:
|
|
|
|
// nothing to cast
|
|
|
|
case []string:
|
2022-07-13 00:52:09 +08:00
|
|
|
result = val
|
2022-07-07 00:19:05 +03:00
|
|
|
case string:
|
|
|
|
if val == "" {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if it is a json encoded array of strings
|
2022-07-13 00:52:09 +08:00
|
|
|
if err := json.Unmarshal([]byte(val), &result); err != nil {
|
2022-07-07 00:19:05 +03:00
|
|
|
// not a json array, just add the string as single array element
|
2022-07-13 00:52:09 +08:00
|
|
|
result = append(result, val)
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
case json.Marshaler: // eg. JsonArray
|
|
|
|
raw, _ := val.MarshalJSON()
|
2022-07-13 00:52:09 +08:00
|
|
|
_ = json.Unmarshal(raw, &result)
|
2022-07-07 00:19:05 +03:00
|
|
|
default:
|
2022-07-13 00:52:09 +08:00
|
|
|
result = cast.ToStringSlice(value)
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|
|
|
|
|
2022-07-13 00:52:09 +08:00
|
|
|
return NonzeroUniques(result)
|
2022-07-07 00:19:05 +03:00
|
|
|
}
|