2022-04-15 09:42:07 +02:00
|
|
|
package rnd
|
|
|
|
|
2024-01-05 16:31:07 +01:00
|
|
|
import (
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
)
|
2022-04-15 09:42:07 +02:00
|
|
|
|
2022-09-28 09:01:17 +02:00
|
|
|
// IsUnique checks if the string is a valid unique ID.
|
|
|
|
func IsUnique(s string, prefix byte) bool {
|
|
|
|
if IsUUID(s) {
|
|
|
|
// Standard UUID.
|
2022-04-15 09:42:07 +02:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2022-09-28 09:01:17 +02:00
|
|
|
return IsUID(s, prefix)
|
2022-04-15 09:42:07 +02:00
|
|
|
}
|
|
|
|
|
2022-09-28 09:01:17 +02:00
|
|
|
// IsUID checks if the string is a valid entity UID.
|
|
|
|
func IsUID(s string, prefix byte) bool {
|
2024-01-05 16:31:07 +01:00
|
|
|
// Check length.
|
2022-04-15 09:42:07 +02:00
|
|
|
if len(s) != 16 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2024-01-05 16:31:07 +01:00
|
|
|
// Check prefix.
|
|
|
|
if prefix != 0 && s[0] != prefix {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check character range.
|
2022-04-15 09:42:07 +02:00
|
|
|
if !IsAlnum(s) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2024-01-05 16:31:07 +01:00
|
|
|
// Check timestamp.
|
|
|
|
if ts := s[1:7]; ts == "000000" {
|
|
|
|
return true
|
|
|
|
} else if t, err := strconv.ParseInt(ts, 36, 64); err != nil {
|
|
|
|
return false
|
|
|
|
} else if t < 1483228800 || t > time.Now().UTC().Unix() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Valid.
|
|
|
|
return true
|
2022-04-15 09:42:07 +02:00
|
|
|
}
|
|
|
|
|
2022-09-28 09:01:17 +02:00
|
|
|
// InvalidUID checks if the UID is empty or invalid.
|
|
|
|
func InvalidUID(s string, prefix byte) bool {
|
|
|
|
return !IsUID(s, prefix)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsUUID tests if the string looks like a standard UUID.
|
|
|
|
func IsUUID(s string) bool {
|
2022-04-15 09:42:07 +02:00
|
|
|
return len(s) == 36 && IsHex(s)
|
|
|
|
}
|
|
|
|
|
|
|
|
// SanitizeUUID normalizes UUIDs found in XMP or Exif metadata.
|
|
|
|
func SanitizeUUID(s string) string {
|
|
|
|
if s == "" {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
s = strings.Replace(strings.TrimSpace(s), "\"", "", -1)
|
|
|
|
|
|
|
|
if start := strings.LastIndex(s, ":"); start != -1 {
|
|
|
|
s = s[start+1:]
|
|
|
|
}
|
|
|
|
|
2022-09-28 09:01:17 +02:00
|
|
|
if !IsUUID(s) {
|
2022-04-15 09:42:07 +02:00
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.ToLower(s)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsAlnum returns true if the string only contains alphanumeric ascii chars without whitespace.
|
|
|
|
func IsAlnum(s string) bool {
|
|
|
|
if s == "" {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, r := range s {
|
2022-09-28 09:01:17 +02:00
|
|
|
if (r < '0' || r > '9') && (r < 'A' || r > 'Z') && (r < 'a' || r > 'z') {
|
2022-04-15 09:42:07 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsHex returns true if the string only contains hex numbers, dashes and letters without whitespace.
|
|
|
|
func IsHex(s string) bool {
|
|
|
|
if s == "" {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, r := range s {
|
|
|
|
if (r < 48 || r > 57) && (r < 97 || r > 102) && (r < 65 || r > 70) && r != 45 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|