mirror of
https://github.com/infiniflow/ragflow.git
synced 2026-06-29 15:31:05 +08:00
### What problem does this PR solve?
As title:
implement:
```go
chats.POST("", r.chatHandler.Create)
chats.POST("/:chat_id/sessions", r.chatSessionHandler.CreateSession)
chats.DELETE("/:chat_id/sessions", r.chatSessionHandler.DeleteSessions)
```
bug fixed:
f80d4c7843/internal/handler/chat.go (L84)
↓
```go
result, err := h.chatService.ListChats(userID, "1", keywords, page, pageSize, orderby, desc)
```
### Type of change
- [x] Bug Fix (non-breaking change which fixes an issue)
- [x] New Feature (non-breaking change which adds functionality)
- [x] Refactoring
250 lines
7.0 KiB
Go
250 lines
7.0 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 (
|
|
"fmt"
|
|
"ragflow/internal/entity"
|
|
"strings"
|
|
)
|
|
|
|
// ChatDAO chat data access object
|
|
type ChatDAO struct{}
|
|
|
|
// NewChatDAO create chat DAO
|
|
func NewChatDAO() *ChatDAO {
|
|
return &ChatDAO{}
|
|
}
|
|
|
|
// ListByTenantID list chats by tenant ID
|
|
func (dao *ChatDAO) ListByTenantID(tenantID string, status string) ([]*entity.Chat, error) {
|
|
var chats []*entity.Chat
|
|
|
|
query := DB.Model(&entity.Chat{}).
|
|
Where("tenant_id = ?", tenantID)
|
|
|
|
if status != "" {
|
|
query = query.Where("status = ?", status)
|
|
}
|
|
|
|
// Order by create_time desc
|
|
if err := query.Order("create_time DESC").Find(&chats).Error; err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return chats, nil
|
|
}
|
|
|
|
// ListByTenantIDs list chats by tenant IDs with pagination and filtering
|
|
func (dao *ChatDAO) ListByTenantIDs(tenantIDs []string, userID string, page, pageSize int, orderby string, desc bool, keywords string) ([]*entity.Chat, int64, error) {
|
|
var chats []*entity.Chat
|
|
var total int64
|
|
|
|
// Build query with join to user table for nickname and avatar
|
|
query := DB.Model(&entity.Chat{}).
|
|
Select(`
|
|
dialog.*,
|
|
user.nickname,
|
|
user.avatar as tenant_avatar
|
|
`).
|
|
Joins("LEFT JOIN user ON dialog.tenant_id = user.id")
|
|
|
|
if len(tenantIDs) > 0 {
|
|
query = query.Where("(dialog.tenant_id IN ? OR dialog.tenant_id = ?) AND dialog.status = ?", tenantIDs, userID, "1")
|
|
} else {
|
|
query = query.Where("dialog.tenant_id = ? AND dialog.status = ?", userID, "1")
|
|
}
|
|
|
|
// Apply keyword filter
|
|
if keywords != "" {
|
|
query = query.Where("LOWER(dialog.name) LIKE ?", "%"+strings.ToLower(keywords)+"%")
|
|
}
|
|
|
|
// Apply ordering
|
|
orderDirection := "ASC"
|
|
if desc {
|
|
orderDirection = "DESC"
|
|
}
|
|
query = query.Order(orderby + " " + orderDirection)
|
|
|
|
// Count total
|
|
if err := query.Count(&total).Error; err != nil {
|
|
return nil, 0, err
|
|
}
|
|
|
|
// Apply pagination
|
|
if page > 0 && pageSize > 0 {
|
|
offset := (page - 1) * pageSize
|
|
if err := query.Offset(offset).Limit(pageSize).Find(&chats).Error; err != nil {
|
|
return nil, 0, err
|
|
}
|
|
} else {
|
|
if err := query.Find(&chats).Error; err != nil {
|
|
return nil, 0, err
|
|
}
|
|
}
|
|
|
|
return chats, total, nil
|
|
}
|
|
|
|
// ListByOwnerIDs list chats by owner IDs with filtering (manual pagination)
|
|
func (dao *ChatDAO) ListByOwnerIDs(ownerIDs []string, userID string, orderby string, desc bool, keywords string) ([]*entity.Chat, int64, error) {
|
|
var chats []*entity.Chat
|
|
|
|
// Build query with join to user table
|
|
query := DB.Model(&entity.Chat{}).
|
|
Select(`
|
|
dialog.*,
|
|
user.nickname,
|
|
user.avatar as tenant_avatar
|
|
`).
|
|
Joins("LEFT JOIN user ON dialog.tenant_id = user.id").
|
|
Where("(dialog.tenant_id IN ? OR dialog.tenant_id = ?) AND dialog.status = ?", ownerIDs, userID, "1")
|
|
|
|
// Apply keyword filter
|
|
if keywords != "" {
|
|
query = query.Where("LOWER(dialog.name) LIKE ?", "%"+strings.ToLower(keywords)+"%")
|
|
}
|
|
|
|
// Filter by owner IDs (additional filter to ensure tenant_id is in ownerIDs)
|
|
query = query.Where("dialog.tenant_id IN ?", ownerIDs)
|
|
|
|
// Apply ordering
|
|
orderDirection := "ASC"
|
|
if desc {
|
|
orderDirection = "DESC"
|
|
}
|
|
query = query.Order(orderby + " " + orderDirection)
|
|
|
|
// Get all matching records
|
|
if err := query.Find(&chats).Error; err != nil {
|
|
return nil, 0, err
|
|
}
|
|
|
|
total := int64(len(chats))
|
|
|
|
return chats, total, nil
|
|
}
|
|
|
|
// GetByID gets chat by ID
|
|
func (dao *ChatDAO) GetByID(id string) (*entity.Chat, error) {
|
|
var chat entity.Chat
|
|
err := DB.Where("id = ?", id).First(&chat).Error
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &chat, nil
|
|
}
|
|
|
|
// GetByIDAndStatus gets chat by ID and status
|
|
func (dao *ChatDAO) GetByIDAndStatus(id string, status string) (*entity.Chat, error) {
|
|
var chat entity.Chat
|
|
err := DB.Where("id = ? AND status = ?", id, status).First(&chat).Error
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &chat, nil
|
|
}
|
|
|
|
// GetExistingNames gets existing dialog names for a tenant
|
|
func (dao *ChatDAO) GetExistingNames(tenantID string, status string) ([]string, error) {
|
|
var names []string
|
|
err := DB.Model(&entity.Chat{}).
|
|
Where("tenant_id = ? AND status = ?", tenantID, status).
|
|
Pluck("name", &names).Error
|
|
return names, err
|
|
}
|
|
|
|
// ExistsByNameTenantStatus checks whether a chat with the given name exists.
|
|
func (dao *ChatDAO) ExistsByNameTenantStatus(name, tenantID, status string) (bool, error) {
|
|
var count int64
|
|
err := DB.Model(&entity.Chat{}).
|
|
Where("name = ? AND tenant_id = ? AND status = ?", name, tenantID, status).
|
|
Count(&count).Error
|
|
return count > 0, err
|
|
}
|
|
|
|
// Create creates a new chat/dialog
|
|
func (dao *ChatDAO) Create(chat *entity.Chat) error {
|
|
return DB.Create(chat).Error
|
|
}
|
|
|
|
// UpdateByID updates a chat by ID
|
|
func (dao *ChatDAO) UpdateByID(id string, updates map[string]interface{}) error {
|
|
return DB.Model(&entity.Chat{}).Where("id = ?", id).Updates(updates).Error
|
|
}
|
|
|
|
// UpdateManyByID updates multiple chats by ID (batch update)
|
|
func (dao *ChatDAO) UpdateManyByID(updates []map[string]interface{}) error {
|
|
if len(updates) == 0 {
|
|
return nil
|
|
}
|
|
|
|
// Use transaction for batch update
|
|
tx := DB.Begin()
|
|
if tx.Error != nil {
|
|
return tx.Error
|
|
}
|
|
|
|
for _, update := range updates {
|
|
id, ok := update["id"].(string)
|
|
if !ok {
|
|
tx.Rollback()
|
|
return fmt.Errorf("invalid id in update")
|
|
}
|
|
|
|
// Remove id from updates map
|
|
updatesWithoutID := make(map[string]interface{})
|
|
for k, v := range update {
|
|
if k != "id" {
|
|
updatesWithoutID[k] = v
|
|
}
|
|
}
|
|
|
|
if err := tx.Model(&entity.Chat{}).Where("id = ?", id).Updates(updatesWithoutID).Error; err != nil {
|
|
tx.Rollback()
|
|
return err
|
|
}
|
|
}
|
|
|
|
return tx.Commit().Error
|
|
}
|
|
|
|
// DeleteByTenantID deletes all chats by tenant ID (hard delete)
|
|
func (dao *ChatDAO) DeleteByTenantID(tenantID string) (int64, error) {
|
|
result := DB.Unscoped().Where("tenant_id = ?", tenantID).Delete(&entity.Chat{})
|
|
return result.RowsAffected, result.Error
|
|
}
|
|
|
|
// GetAllDialogIDsByTenantID gets all dialog IDs by tenant ID
|
|
func (dao *ChatDAO) GetAllDialogIDsByTenantID(tenantID string) ([]string, error) {
|
|
var dialogIDs []string
|
|
err := DB.Model(&entity.Chat{}).
|
|
Where("tenant_id = ?", tenantID).
|
|
Pluck("id", &dialogIDs).Error
|
|
return dialogIDs, err
|
|
}
|
|
|
|
// QueryByTenantIDAndID checks if a chat exists with given tenant_id and id
|
|
// Reference: Python DialogService.query(tenant_id=tenant.tenant_id, id=chat_id, status=StatusEnum.VALID.value)
|
|
// Used for permission verification in get_chat API
|
|
func (dao *ChatDAO) QueryByTenantIDAndID(tenantID string, chatID string, status string) ([]*entity.Chat, error) {
|
|
var chats []*entity.Chat
|
|
err := DB.Where("tenant_id = ? AND id = ? AND status = ?", tenantID, chatID, status).Find(&chats).Error
|
|
return chats, err
|
|
}
|