2021-03-13 23:06:03 +08:00
|
|
|
// Copyright 2021 The casbin Authors. All Rights Reserved.
|
2020-10-20 23:14:03 +08:00
|
|
|
//
|
|
|
|
// 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 object
|
|
|
|
|
|
|
|
import (
|
2021-02-21 23:51:40 +08:00
|
|
|
"fmt"
|
|
|
|
|
2021-07-25 09:34:25 +08:00
|
|
|
"github.com/casbin/casdoor/util"
|
2020-10-20 23:14:03 +08:00
|
|
|
"xorm.io/core"
|
|
|
|
)
|
|
|
|
|
|
|
|
type User struct {
|
|
|
|
Owner string `xorm:"varchar(100) notnull pk" json:"owner"`
|
|
|
|
Name string `xorm:"varchar(100) notnull pk" json:"name"`
|
|
|
|
CreatedTime string `xorm:"varchar(100)" json:"createdTime"`
|
2021-06-01 20:37:00 +08:00
|
|
|
UpdatedTime string `xorm:"varchar(100)" json:"updatedTime"`
|
2020-10-20 23:14:03 +08:00
|
|
|
|
2021-06-21 10:22:47 +08:00
|
|
|
Id string `xorm:"varchar(100)" json:"id"`
|
|
|
|
Type string `xorm:"varchar(100)" json:"type"`
|
|
|
|
Password string `xorm:"varchar(100)" json:"password"`
|
2021-11-04 21:08:43 +08:00
|
|
|
PasswordSalt string `xorm:"varchar(100)" json:"passwordSalt"`
|
2021-06-21 10:22:47 +08:00
|
|
|
DisplayName string `xorm:"varchar(100)" json:"displayName"`
|
|
|
|
Avatar string `xorm:"varchar(255)" json:"avatar"`
|
2021-08-21 22:52:29 +08:00
|
|
|
PermanentAvatar string `xorm:"varchar(255)" json:"permanentAvatar"`
|
2021-06-21 10:22:47 +08:00
|
|
|
Email string `xorm:"varchar(100)" json:"email"`
|
|
|
|
Phone string `xorm:"varchar(100)" json:"phone"`
|
2021-08-21 10:58:34 +08:00
|
|
|
Location string `xorm:"varchar(100)" json:"location"`
|
2021-06-21 10:22:47 +08:00
|
|
|
Address []string `json:"address"`
|
|
|
|
Affiliation string `xorm:"varchar(100)" json:"affiliation"`
|
2021-08-21 10:58:34 +08:00
|
|
|
Title string `xorm:"varchar(100)" json:"title"`
|
2021-11-19 21:37:13 +08:00
|
|
|
IdCardType string `xorm:"varchar(100)" json:"idCardType"`
|
|
|
|
IdCard string `xorm:"varchar(100)" json:"idCard"`
|
2021-08-21 10:58:34 +08:00
|
|
|
Homepage string `xorm:"varchar(100)" json:"homepage"`
|
2021-08-25 08:07:08 +08:00
|
|
|
Bio string `xorm:"varchar(100)" json:"bio"`
|
2021-06-21 10:22:47 +08:00
|
|
|
Tag string `xorm:"varchar(100)" json:"tag"`
|
2021-07-31 16:02:48 +08:00
|
|
|
Region string `xorm:"varchar(100)" json:"region"`
|
2021-06-21 10:22:47 +08:00
|
|
|
Language string `xorm:"varchar(100)" json:"language"`
|
2021-11-19 21:37:13 +08:00
|
|
|
Gender string `xorm:"varchar(100)" json:"gender"`
|
|
|
|
Birthday string `xorm:"varchar(100)" json:"birthday"`
|
|
|
|
Education string `xorm:"varchar(100)" json:"education"`
|
2021-06-21 10:22:47 +08:00
|
|
|
Score int `json:"score"`
|
2021-08-21 10:58:34 +08:00
|
|
|
Ranking int `json:"ranking"`
|
|
|
|
IsOnline bool `json:"isOnline"`
|
2021-06-21 10:22:47 +08:00
|
|
|
IsAdmin bool `json:"isAdmin"`
|
|
|
|
IsGlobalAdmin bool `json:"isGlobalAdmin"`
|
|
|
|
IsForbidden bool `json:"isForbidden"`
|
2021-11-06 15:52:03 +08:00
|
|
|
IsDeleted bool `json:"isDeleted"`
|
2021-06-21 10:22:47 +08:00
|
|
|
SignupApplication string `xorm:"varchar(100)" json:"signupApplication"`
|
|
|
|
Hash string `xorm:"varchar(100)" json:"hash"`
|
|
|
|
PreHash string `xorm:"varchar(100)" json:"preHash"`
|
2021-02-14 00:22:24 +08:00
|
|
|
|
2021-11-19 21:37:13 +08:00
|
|
|
CreatedIp string `xorm:"varchar(100)" json:"createdIp"`
|
|
|
|
LastSigninTime string `xorm:"varchar(100)" json:"lastSigninTime"`
|
|
|
|
LastSigninIp string `xorm:"varchar(100)" json:"lastSigninIp"`
|
|
|
|
|
2021-06-06 10:06:54 +08:00
|
|
|
Github string `xorm:"varchar(100)" json:"github"`
|
|
|
|
Google string `xorm:"varchar(100)" json:"google"`
|
|
|
|
QQ string `xorm:"qq varchar(100)" json:"qq"`
|
|
|
|
WeChat string `xorm:"wechat varchar(100)" json:"wechat"`
|
|
|
|
Facebook string `xorm:"facebook varchar(100)" json:"facebook"`
|
2021-06-09 11:15:49 +08:00
|
|
|
DingTalk string `xorm:"dingtalk varchar(100)" json:"dingtalk"`
|
2021-06-10 16:55:31 +08:00
|
|
|
Weibo string `xorm:"weibo varchar(100)" json:"weibo"`
|
2021-06-14 12:53:07 +08:00
|
|
|
Gitee string `xorm:"gitee varchar(100)" json:"gitee"`
|
2021-07-03 20:53:38 +08:00
|
|
|
LinkedIn string `xorm:"linkedin varchar(100)" json:"linkedin"`
|
2021-08-01 14:48:29 +08:00
|
|
|
Wecom string `xorm:"wecom varchar(100)" json:"wecom"`
|
2021-08-14 16:00:38 +08:00
|
|
|
Lark string `xorm:"lark varchar(100)" json:"lark"`
|
2021-08-19 21:03:57 +08:00
|
|
|
Gitlab string `xorm:"gitlab varchar(100)" json:"gitlab"`
|
2021-05-30 15:13:43 +08:00
|
|
|
|
2021-07-17 14:13:00 +08:00
|
|
|
Ldap string `xorm:"ldap varchar(100)" json:"ldap"`
|
2021-05-30 15:13:43 +08:00
|
|
|
Properties map[string]string `json:"properties"`
|
2020-10-20 23:14:03 +08:00
|
|
|
}
|
|
|
|
|
2021-11-06 11:32:22 +08:00
|
|
|
func GetGlobalUserCount() int {
|
|
|
|
count, err := adapter.Engine.Count(&User{})
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return int(count)
|
|
|
|
}
|
|
|
|
|
2021-02-13 00:11:12 +08:00
|
|
|
func GetGlobalUsers() []*User {
|
|
|
|
users := []*User{}
|
2021-05-02 10:30:12 +08:00
|
|
|
err := adapter.Engine.Desc("created_time").Find(&users)
|
2021-02-13 00:11:12 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return users
|
|
|
|
}
|
|
|
|
|
2021-11-06 11:32:22 +08:00
|
|
|
func GetPaginationGlobalUsers(offset, limit int) []*User {
|
|
|
|
users := []*User{}
|
|
|
|
err := adapter.Engine.Desc("created_time").Limit(limit, offset).Find(&users)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return users
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetUserCount(owner string) int {
|
|
|
|
count, err := adapter.Engine.Count(&User{Owner: owner})
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return int(count)
|
|
|
|
}
|
|
|
|
|
2021-11-19 10:43:14 +08:00
|
|
|
func GetOnlineUserCount(owner string, isOnline int) int {
|
|
|
|
count, err := adapter.Engine.Where("is_online = ?", isOnline).Count(&User{Owner: owner})
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return int(count)
|
|
|
|
}
|
|
|
|
|
2020-10-20 23:14:03 +08:00
|
|
|
func GetUsers(owner string) []*User {
|
|
|
|
users := []*User{}
|
2021-05-02 10:30:12 +08:00
|
|
|
err := adapter.Engine.Desc("created_time").Find(&users, &User{Owner: owner})
|
2020-10-20 23:14:03 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return users
|
|
|
|
}
|
|
|
|
|
2021-11-19 10:43:14 +08:00
|
|
|
func GetSortedUsers(owner string, sorter string, limit int) []*User {
|
|
|
|
users := []*User{}
|
|
|
|
err := adapter.Engine.Desc(sorter).Limit(limit, 0).Find(&users, &User{Owner: owner})
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return users
|
|
|
|
}
|
|
|
|
|
2021-11-06 11:32:22 +08:00
|
|
|
func GetPaginationUsers(owner string, offset, limit int) []*User {
|
|
|
|
users := []*User{}
|
|
|
|
err := adapter.Engine.Desc("created_time").Limit(limit, offset).Find(&users, &User{Owner: owner})
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return users
|
|
|
|
}
|
|
|
|
|
2020-10-20 23:14:03 +08:00
|
|
|
func getUser(owner string, name string) *User {
|
2021-06-21 01:01:16 +08:00
|
|
|
if owner == "" || name == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-10-20 23:14:03 +08:00
|
|
|
user := User{Owner: owner, Name: name}
|
2021-05-02 10:30:12 +08:00
|
|
|
existed, err := adapter.Engine.Get(&user)
|
2020-10-20 23:14:03 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2021-11-19 10:43:14 +08:00
|
|
|
if existed {
|
|
|
|
return &user
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetUserByEmail(owner string, email string) *User {
|
|
|
|
if owner == "" || email == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
user := User{Owner: owner, Email: email}
|
|
|
|
existed, err := adapter.Engine.Get(&user)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2020-10-20 23:14:03 +08:00
|
|
|
if existed {
|
|
|
|
return &user
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetUser(id string) *User {
|
|
|
|
owner, name := util.GetOwnerAndNameFromId(id)
|
|
|
|
return getUser(owner, name)
|
|
|
|
}
|
|
|
|
|
2021-11-20 15:46:54 +08:00
|
|
|
func GetUserNoCheck(id string) *User {
|
|
|
|
owner, name := util.GetOwnerAndNameFromIdNoCheck(id)
|
|
|
|
return getUser(owner, name)
|
|
|
|
}
|
|
|
|
|
2021-05-30 13:37:34 +08:00
|
|
|
func GetMaskedUser(user *User) *User {
|
2021-06-12 12:27:13 +08:00
|
|
|
if user == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-05-30 13:37:34 +08:00
|
|
|
if user.Password != "" {
|
|
|
|
user.Password = "***"
|
|
|
|
}
|
|
|
|
return user
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetMaskedUsers(users []*User) []*User {
|
|
|
|
for _, user := range users {
|
|
|
|
user = GetMaskedUser(user)
|
|
|
|
}
|
|
|
|
return users
|
|
|
|
}
|
|
|
|
|
2021-06-17 00:49:02 +08:00
|
|
|
func GetLastUser(owner string) *User {
|
|
|
|
user := User{Owner: owner}
|
|
|
|
existed, err := adapter.Engine.Desc("created_time", "id").Get(&user)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if existed {
|
|
|
|
return &user
|
|
|
|
}
|
2021-08-07 22:02:56 +08:00
|
|
|
|
|
|
|
return nil
|
2021-06-17 00:49:02 +08:00
|
|
|
}
|
|
|
|
|
2020-10-20 23:14:03 +08:00
|
|
|
func UpdateUser(id string, user *User) bool {
|
2021-11-20 15:46:54 +08:00
|
|
|
owner, name := util.GetOwnerAndNameFromIdNoCheck(id)
|
2021-08-21 22:52:29 +08:00
|
|
|
oldUser := getUser(owner, name)
|
|
|
|
if oldUser == nil {
|
2020-10-20 23:14:03 +08:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-05-08 18:51:37 +08:00
|
|
|
user.UpdateUserHash()
|
|
|
|
|
2021-11-20 16:21:15 +08:00
|
|
|
if user.Avatar != oldUser.Avatar && user.Avatar != "" && user.PermanentAvatar != "*" {
|
2021-08-21 23:17:33 +08:00
|
|
|
user.PermanentAvatar = getPermanentAvatarUrl(user.Owner, user.Name, user.Avatar)
|
|
|
|
}
|
|
|
|
|
2021-07-17 14:13:00 +08:00
|
|
|
affected, err := adapter.Engine.ID(core.PK{owner, name}).Cols("owner", "display_name", "avatar",
|
2021-11-06 15:52:03 +08:00
|
|
|
"location", "address", "region", "language", "affiliation", "title", "homepage", "bio", "score", "tag",
|
|
|
|
"is_admin", "is_global_admin", "is_forbidden", "is_deleted", "hash", "properties").Update(user)
|
2020-10-20 23:14:03 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2021-05-09 15:44:12 +08:00
|
|
|
return affected != 0
|
|
|
|
}
|
|
|
|
|
2021-08-21 22:52:29 +08:00
|
|
|
func UpdateUserForAllFields(id string, user *User) bool {
|
2021-06-09 21:41:27 +08:00
|
|
|
owner, name := util.GetOwnerAndNameFromId(id)
|
2021-08-21 22:52:29 +08:00
|
|
|
oldUser := getUser(owner, name)
|
|
|
|
if oldUser == nil {
|
2021-06-09 21:41:27 +08:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
user.UpdateUserHash()
|
|
|
|
|
2021-08-21 23:17:33 +08:00
|
|
|
if user.Avatar != oldUser.Avatar && user.Avatar != "" {
|
|
|
|
user.PermanentAvatar = getPermanentAvatarUrl(user.Owner, user.Name, user.Avatar)
|
|
|
|
}
|
|
|
|
|
2021-06-09 21:41:27 +08:00
|
|
|
affected, err := adapter.Engine.ID(core.PK{owner, name}).AllCols().Update(user)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return affected != 0
|
|
|
|
}
|
|
|
|
|
2021-08-21 22:52:29 +08:00
|
|
|
func UpdateUserForOriginalFields(user *User) bool {
|
|
|
|
owner, name := util.GetOwnerAndNameFromId(user.GetId())
|
|
|
|
oldUser := getUser(owner, name)
|
|
|
|
if oldUser == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-08-21 23:17:33 +08:00
|
|
|
if user.Avatar != oldUser.Avatar && user.Avatar != "" {
|
|
|
|
user.PermanentAvatar = getPermanentAvatarUrl(user.Owner, user.Name, user.Avatar)
|
|
|
|
}
|
|
|
|
|
2021-06-04 23:00:54 +08:00
|
|
|
affected, err := adapter.Engine.ID(core.PK{user.Owner, user.Name}).Cols("display_name", "password", "phone", "avatar", "affiliation", "score", "is_forbidden", "hash", "pre_hash").Update(user)
|
2021-05-09 15:44:12 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
2021-03-28 00:48:34 +08:00
|
|
|
return affected != 0
|
2020-10-20 23:14:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func AddUser(user *User) bool {
|
2021-06-17 01:12:03 +08:00
|
|
|
if user.Id == "" {
|
|
|
|
user.Id = util.GenerateId()
|
|
|
|
}
|
2021-05-16 21:22:20 +08:00
|
|
|
|
2021-05-16 22:58:30 +08:00
|
|
|
organization := GetOrganizationByUser(user)
|
2021-05-16 21:04:26 +08:00
|
|
|
user.UpdateUserPassword(organization)
|
2021-05-08 18:51:37 +08:00
|
|
|
|
2021-05-16 21:22:20 +08:00
|
|
|
user.UpdateUserHash()
|
|
|
|
user.PreHash = user.Hash
|
|
|
|
|
2021-08-21 23:17:33 +08:00
|
|
|
user.PermanentAvatar = getPermanentAvatarUrl(user.Owner, user.Name, user.Avatar)
|
|
|
|
|
2021-05-02 10:30:12 +08:00
|
|
|
affected, err := adapter.Engine.Insert(user)
|
2020-10-20 23:14:03 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return affected != 0
|
|
|
|
}
|
|
|
|
|
2021-05-02 23:06:08 +08:00
|
|
|
func AddUsers(users []*User) bool {
|
2021-05-16 21:04:26 +08:00
|
|
|
if len(users) == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-05-16 22:58:30 +08:00
|
|
|
organization := GetOrganizationByUser(users[0])
|
2021-05-08 18:51:37 +08:00
|
|
|
for _, user := range users {
|
2021-05-16 21:22:20 +08:00
|
|
|
user.UpdateUserPassword(organization)
|
|
|
|
|
2021-05-08 18:51:37 +08:00
|
|
|
user.UpdateUserHash()
|
2021-05-09 14:54:17 +08:00
|
|
|
user.PreHash = user.Hash
|
2021-08-21 23:17:33 +08:00
|
|
|
|
|
|
|
user.PermanentAvatar = getPermanentAvatarUrl(user.Owner, user.Name, user.Avatar)
|
2021-05-08 18:51:37 +08:00
|
|
|
}
|
|
|
|
|
2021-05-02 23:06:08 +08:00
|
|
|
affected, err := adapter.Engine.Insert(users)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return affected != 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func AddUsersSafe(users []*User) bool {
|
|
|
|
batchSize := 1000
|
|
|
|
|
|
|
|
if len(users) == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
affected := false
|
|
|
|
for i := 0; i < (len(users)-1)/batchSize+1; i++ {
|
|
|
|
start := i * batchSize
|
|
|
|
end := (i + 1) * batchSize
|
|
|
|
if end > len(users) {
|
|
|
|
end = len(users)
|
|
|
|
}
|
|
|
|
|
|
|
|
tmp := users[start:end]
|
2021-09-04 22:20:47 +08:00
|
|
|
// TODO: save to log instead of standard output
|
|
|
|
// fmt.Printf("Add users: [%d - %d].\n", start, end)
|
2021-05-02 23:06:08 +08:00
|
|
|
if AddUsers(tmp) {
|
|
|
|
affected = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return affected
|
|
|
|
}
|
|
|
|
|
2020-10-20 23:14:03 +08:00
|
|
|
func DeleteUser(user *User) bool {
|
2021-05-02 10:30:12 +08:00
|
|
|
affected, err := adapter.Engine.ID(core.PK{user.Owner, user.Name}).Delete(&User{})
|
2020-10-20 23:14:03 +08:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return affected != 0
|
|
|
|
}
|
2021-02-14 00:22:24 +08:00
|
|
|
|
2021-04-27 19:35:40 +08:00
|
|
|
func LinkUserAccount(user *User, field string, value string) bool {
|
|
|
|
return SetUserField(user, field, value)
|
|
|
|
}
|
|
|
|
|
2021-05-08 22:04:45 +08:00
|
|
|
func (user *User) GetId() string {
|
|
|
|
return fmt.Sprintf("%s/%s", user.Owner, user.Name)
|
|
|
|
}
|