mirror of
https://github.com/go-vikunja/vikunja.git
synced 2026-04-25 06:35:32 +00:00
Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: kolaente <13721712+kolaente@users.noreply.github.com> Co-authored-by: kolaente <k@knt.li>
137 lines
4.9 KiB
Go
137 lines
4.9 KiB
Go
// Vikunja is a to-do list application to facilitate your life.
|
|
// Copyright 2018-present Vikunja and contributors. All rights reserved.
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
package db
|
|
|
|
import (
|
|
"strings"
|
|
|
|
"xorm.io/builder"
|
|
"xorm.io/xorm/schemas"
|
|
)
|
|
|
|
// ILIKE returns an ILIKE query on postgres and a LIKE query on all other platforms.
|
|
// Postgres' is case-sensitive by default.
|
|
// To work around this, we're using ILIKE as opposed to normal LIKE statements.
|
|
// ILIKE is preferred over LOWER(text) LIKE for performance reasons.
|
|
// See https://stackoverflow.com/q/7005302/10924593
|
|
func ILIKE(column, search string) builder.Cond {
|
|
if Type() == schemas.POSTGRES {
|
|
return builder.Expr(column+" ILIKE ?", "%"+search+"%")
|
|
}
|
|
|
|
return &builder.Like{column, "%" + search + "%"}
|
|
}
|
|
|
|
func ParadeDBAvailable() bool {
|
|
return Type() == schemas.POSTGRES && paradedbInstalled
|
|
}
|
|
|
|
// MultiFieldSearch performs an optimized search across multiple fields for ParadeDB
|
|
// using a single query rather than multiple OR conditions.
|
|
// Falls back to individual ILIKE queries for PGroonga and standard PostgreSQL.
|
|
func MultiFieldSearch(fields []string, search string) builder.Cond {
|
|
return MultiFieldSearchWithTableAlias(fields, search, "")
|
|
}
|
|
|
|
// MultiFieldSearchWithTableAlias performs an optimized search across multiple fields for ParadeDB
|
|
// with support for table aliases. When tableAlias is provided, it will be used to prefix field names
|
|
// for non-ParadeDB queries and the id field for ParadeDB queries.
|
|
func MultiFieldSearchWithTableAlias(fields []string, search, tableAlias string) builder.Cond {
|
|
if Type() == schemas.POSTGRES && paradedbInstalled {
|
|
if len(fields) == 1 {
|
|
// Single field search - use optimized match function
|
|
return builder.Expr("id @@@ paradedb.match(?, ?)", fields[0], search)
|
|
}
|
|
// Multi-field search - use disjunction_max for optimal performance
|
|
fieldMatches := make([]string, len(fields))
|
|
args := make([]interface{}, len(fields)*2)
|
|
for i, field := range fields {
|
|
fieldMatches[i] = "paradedb.match(?, ?)"
|
|
args[i*2] = field
|
|
args[i*2+1] = search
|
|
}
|
|
|
|
idField := "`id`"
|
|
if tableAlias != "" {
|
|
idField = "`" + tableAlias + "`.`id`"
|
|
}
|
|
|
|
return builder.Expr(idField+" @@@ paradedb.disjunction_max(ARRAY["+strings.Join(fieldMatches, ", ")+"])", args...)
|
|
}
|
|
|
|
// For non-PostgreSQL databases, use ILIKE on all fields
|
|
conditions := make([]builder.Cond, len(fields))
|
|
for i, field := range fields {
|
|
// Add table alias to field name if provided
|
|
fieldName := field
|
|
if tableAlias != "" {
|
|
fieldName = tableAlias + "." + field
|
|
}
|
|
conditions[i] = ILIKE(fieldName, search)
|
|
}
|
|
return builder.Or(conditions...)
|
|
}
|
|
|
|
// IsMySQLDuplicateEntryError checks if the given error is a MySQL duplicate entry error
|
|
// for the specified unique key constraint.
|
|
func IsMySQLDuplicateEntryError(err error, constraintName string) bool {
|
|
if err == nil {
|
|
return false
|
|
}
|
|
|
|
errStr := strings.ToLower(err.Error())
|
|
|
|
// Check for MySQL Error 1062 (duplicate entry) and the specific constraint
|
|
return strings.Contains(errStr, "error 1062") &&
|
|
strings.Contains(errStr, "duplicate entry") &&
|
|
strings.Contains(errStr, strings.ToLower(constraintName))
|
|
}
|
|
|
|
// IsUniqueConstraintError checks if the given error is a unique constraint violation
|
|
// for the specified constraint across all supported database types.
|
|
func IsUniqueConstraintError(err error, constraintName string) bool {
|
|
if err == nil {
|
|
return false
|
|
}
|
|
|
|
errStr := strings.ToLower(err.Error())
|
|
constraintNameLower := strings.ToLower(constraintName)
|
|
|
|
// MySQL: Error 1062 (23000): Duplicate entry ... for key ...
|
|
if strings.Contains(errStr, "error 1062") &&
|
|
strings.Contains(errStr, "duplicate entry") &&
|
|
strings.Contains(errStr, constraintNameLower) {
|
|
return true
|
|
}
|
|
|
|
// PostgreSQL: duplicate key value violates unique constraint "constraint_name"
|
|
if strings.Contains(errStr, "duplicate key value violates unique constraint") &&
|
|
strings.Contains(errStr, constraintNameLower) {
|
|
return true
|
|
}
|
|
|
|
// SQLite: UNIQUE constraint failed: table.column
|
|
if strings.Contains(errStr, "unique constraint failed") ||
|
|
(strings.Contains(errStr, "constraint failed") && strings.Contains(errStr, "unique")) {
|
|
// For SQLite, we check if it mentions the constraint or table.column pattern
|
|
return strings.Contains(errStr, constraintNameLower) ||
|
|
strings.Contains(errStr, "task_buckets")
|
|
}
|
|
|
|
return false
|
|
}
|