1
0
mirror of https://github.com/pocketbase/pocketbase.git synced 2025-01-25 14:43:42 +02:00

127 lines
2.7 KiB
Go
Raw Normal View History

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{}
// 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
}
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
}
// "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 {
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-12-10 12:08:59 +02:00
if val == zeroVal {
continue
}
if _, ok := existMap[val]; ok {
continue
2022-07-07 00:19:05 +03: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.
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:
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-12-10 12:08:59 +02:00
if strings.Contains(val, "[") {
if err := json.Unmarshal([]byte(val), &result); err != nil {
// not a json array, just add the string as single array element
result = append(result, val)
}
} else {
// just add the string as single array element
result = append(result, val)
2022-07-07 00:19:05 +03:00
}
case json.Marshaler: // eg. JsonArray
raw, _ := val.MarshalJSON()
_ = json.Unmarshal(raw, &result)
2022-07-07 00:19:05 +03:00
default:
result = cast.ToStringSlice(value)
2022-07-07 00:19:05 +03:00
}
return NonzeroUniques(result)
2022-07-07 00:19:05 +03:00
}