mirror of
https://github.com/casdoor/casdoor.git
synced 2025-05-23 02:35:49 +08:00

* feat: support user migration from Keycloak using syncer Signed-off-by: Yixiang Zhao <seriouszyx@foxmail.com> * feat: add more Keycloak columns Signed-off-by: Yixiang Zhao <seriouszyx@foxmail.com> * fix: requested changes Signed-off-by: Yixiang Zhao <seriouszyx@foxmail.com>
238 lines
4.4 KiB
Go
238 lines
4.4 KiB
Go
// Copyright 2021 The Casdoor Authors. All Rights Reserved.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package util
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/md5"
|
|
"encoding/hex"
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
"unicode"
|
|
|
|
"github.com/google/uuid"
|
|
)
|
|
|
|
func ParseInt(s string) int {
|
|
if s == "" {
|
|
return 0
|
|
}
|
|
|
|
i, err := strconv.Atoi(s)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
return i
|
|
}
|
|
|
|
func ParseFloat(s string) float64 {
|
|
f, err := strconv.ParseFloat(s, 64)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
return f
|
|
}
|
|
|
|
func ParseBool(s string) bool {
|
|
if s == "\x01" {
|
|
return true
|
|
}
|
|
|
|
i := ParseInt(s)
|
|
return i != 0
|
|
}
|
|
|
|
func BoolToString(b bool) string {
|
|
if b {
|
|
return "1"
|
|
}
|
|
return "0"
|
|
}
|
|
|
|
//CamelToSnakeCase This function transform camelcase in snakecase LoremIpsum in lorem_ipsum
|
|
func CamelToSnakeCase(camel string) string {
|
|
var buf bytes.Buffer
|
|
for _, c := range camel {
|
|
if 'A' <= c && c <= 'Z' {
|
|
// just convert [A-Z] to _[a-z]
|
|
if buf.Len() > 0 {
|
|
buf.WriteRune('_')
|
|
}
|
|
buf.WriteRune(c - 'A' + 'a')
|
|
continue
|
|
}
|
|
buf.WriteRune(c)
|
|
}
|
|
return strings.ReplaceAll(buf.String(), " ", "")
|
|
}
|
|
|
|
func GetOwnerAndNameFromId(id string) (string, string) {
|
|
tokens := strings.Split(id, "/")
|
|
if len(tokens) != 2 {
|
|
panic(errors.New("GetOwnerAndNameFromId() error, wrong token count for ID: " + id))
|
|
}
|
|
|
|
return tokens[0], tokens[1]
|
|
}
|
|
|
|
func GetOwnerAndNameFromIdNoCheck(id string) (string, string) {
|
|
tokens := strings.SplitN(id, "/", 2)
|
|
return tokens[0], tokens[1]
|
|
}
|
|
|
|
func GenerateId() string {
|
|
return uuid.NewString()
|
|
}
|
|
|
|
func GenerateTimeId() string {
|
|
timestamp := time.Now().Unix()
|
|
tm := time.Unix(timestamp, 0)
|
|
t := tm.Format("20060102_150405")
|
|
|
|
random := uuid.NewString()[0:7]
|
|
|
|
res := fmt.Sprintf("%s_%s", t, random)
|
|
return res
|
|
}
|
|
|
|
func GenerateSimpleTimeId() string {
|
|
timestamp := time.Now().Unix()
|
|
tm := time.Unix(timestamp, 0)
|
|
t := tm.Format("20060102150405")
|
|
|
|
return t
|
|
}
|
|
|
|
func GetId(name string) string {
|
|
return fmt.Sprintf("admin/%s", name)
|
|
}
|
|
|
|
func GetMd5Hash(text string) string {
|
|
hash := md5.Sum([]byte(text))
|
|
return hex.EncodeToString(hash[:])
|
|
}
|
|
|
|
func IsStrsEmpty(strs ...string) bool {
|
|
for _, str := range strs {
|
|
if len(str) == 0 {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func GetMaxLenStr(strs ...string) string {
|
|
m := 0
|
|
i := 0
|
|
for j, str := range strs {
|
|
l := len(str)
|
|
if l > m {
|
|
m = l
|
|
i = j
|
|
}
|
|
}
|
|
return strs[i]
|
|
}
|
|
|
|
func GetMinLenStr(strs ...string) string {
|
|
m := int(^uint(0) >> 1)
|
|
i := 0
|
|
for j, str := range strs {
|
|
l := len(str)
|
|
if l < m {
|
|
m = l
|
|
i = j
|
|
}
|
|
}
|
|
return strs[i]
|
|
}
|
|
|
|
func ReadStringFromPath(path string) string {
|
|
data, err := ioutil.ReadFile(path)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
return string(data)
|
|
}
|
|
|
|
func WriteStringToPath(s string, path string) {
|
|
err := ioutil.WriteFile(path, []byte(s), 0644)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
// SnakeString transform XxYy to xx_yy
|
|
func SnakeString(s string) string {
|
|
data := make([]byte, 0, len(s)*2)
|
|
j := false
|
|
num := len(s)
|
|
for i := 0; i < num; i++ {
|
|
d := s[i]
|
|
if i > 0 && d >= 'A' && d <= 'Z' && j {
|
|
data = append(data, '_')
|
|
}
|
|
if d != '_' {
|
|
j = true
|
|
}
|
|
data = append(data, d)
|
|
}
|
|
result := strings.ToLower(string(data[:]))
|
|
return strings.ReplaceAll(result, " ", "")
|
|
}
|
|
|
|
func IsChinese(str string) bool {
|
|
var flag bool
|
|
for _, v := range str {
|
|
if unicode.Is(unicode.Han, v) {
|
|
flag = true
|
|
break
|
|
}
|
|
}
|
|
return flag
|
|
}
|
|
|
|
func GetMaskedPhone(phone string) string {
|
|
return getMaskedPhone(phone)
|
|
}
|
|
|
|
func GetMaskedEmail(email string) string {
|
|
if email == "" {
|
|
return ""
|
|
}
|
|
|
|
tokens := strings.Split(email, "@")
|
|
username := maskString(tokens[0])
|
|
domain := tokens[1]
|
|
domainTokens := strings.Split(domain, ".")
|
|
domainTokens[len(domainTokens)-2] = maskString(domainTokens[len(domainTokens)-2])
|
|
return fmt.Sprintf("%s@%s", username, strings.Join(domainTokens, "."))
|
|
}
|
|
|
|
func maskString(str string) string {
|
|
if len(str) <= 2 {
|
|
return str
|
|
} else {
|
|
return fmt.Sprintf("%c%s%c", str[0], strings.Repeat("*", len(str)-2), str[len(str)-1])
|
|
}
|
|
}
|