Files
ragflow/internal/dao/user.go
chanx a3e6c2e84a Fix: Enhanced user management functionality and cascading data deletion. (#13594)
### What problem does this PR solve?
Fix: Enhanced user management functionality and cascading data deletion.

Added tenant and related data initialization functionality during user
creation, including tenants, user-tenant relationships, LLM
configuration, and root folder.
Added cascading deletion logic for user deletion, ensuring that all
associated data is cleaned up simultaneously when a user is deleted.
Implemented a Werkzeug-compatible password hash algorithm (scrypt) and
verification functionality.
Added multiple DAO methods to support batch data operations and
cascading deletion.
Improved user login processing and added token signing functionality.
### Type of change

- [x] Bug Fix (non-breaking change which fixes an issue)
2026-03-13 16:53:54 +08:00

130 lines
3.5 KiB
Go

//
// Copyright 2026 The InfiniFlow 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 dao
import (
"ragflow/internal/model"
)
// UserDAO user data access object
type UserDAO struct{}
// NewUserDAO create user DAO
func NewUserDAO() *UserDAO {
return &UserDAO{}
}
// Create create user
func (dao *UserDAO) Create(user *model.User) error {
return DB.Create(user).Error
}
// GetByID get user by ID
func (dao *UserDAO) GetByID(id uint) (*model.User, error) {
var user model.User
err := DB.First(&user, id).Error
if err != nil {
return nil, err
}
return &user, nil
}
// GetByUsername get user by username
func (dao *UserDAO) GetByUsername(username string) (*model.User, error) {
var user model.User
err := DB.Where("username = ?", username).First(&user).Error
if err != nil {
return nil, err
}
return &user, nil
}
// GetByEmail get user by email
func (dao *UserDAO) GetByEmail(email string) (*model.User, error) {
var user model.User
query := DB.Where("email = ?", email)
err := query.First(&user).Error
if err != nil {
return nil, err
}
return &user, nil
}
// GetByAccessToken get user by access token
func (dao *UserDAO) GetByAccessToken(token string) (*model.User, error) {
var user model.User
err := DB.Where("access_token = ?", token).First(&user).Error
if err != nil {
return nil, err
}
return &user, nil
}
// Update update user
func (dao *UserDAO) Update(user *model.User) error {
return DB.Save(user).Error
}
// UpdateAccessToken update user's access token
func (dao *UserDAO) UpdateAccessToken(user *model.User, token string) error {
return DB.Model(user).Update("access_token", token).Error
}
// List list users (only active users with status != "0")
func (dao *UserDAO) List(offset, limit int) ([]*model.User, int64, error) {
var users []*model.User
var total int64
// Only count users with status != "0" (not deleted)
if err := DB.Model(&model.User{}).Count(&total).Error; err != nil {
return nil, 0, err
}
query := DB.Model(&model.User{})
if offset > 0 {
query = query.Offset(offset)
}
if limit > 0 {
query = query.Limit(limit)
}
err := query.Find(&users).Error
return users, total, err
}
// Delete delete user
func (dao *UserDAO) Delete(id uint) error {
return DB.Delete(&model.User{}, id).Error
}
// DeleteByID delete user by string ID (soft delete - set status to 0)
func (dao *UserDAO) DeleteByID(id string) error {
return DB.Model(&model.User{}).Where("id = ?", id).Update("status", "0").Error
}
// HardDelete hard delete user by string ID
func (dao *UserDAO) HardDelete(id string) error {
return DB.Unscoped().Where("id = ?", id).Delete(&model.User{}).Error
}
// ListByEmail list users by email (only active users with status != "0")
// Returns all users matching the given email address
func (dao *UserDAO) ListByEmail(email string) ([]*model.User, error) {
var users []*model.User
err := DB.Where("email = ? AND (status != ? OR status IS NULL)", email, "0").Find(&users).Error
return users, err
}