mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
* refactor(server): better sort mappings * refactor(server): simplify GetIndex * fix: recreate tables and indexes using proper collation Also add tests to ensure proper collation * chore: remove unused method * fix: sort expressions * fix: lint errors * fix: cleanup
93 lines
2.3 KiB
Go
93 lines
2.3 KiB
Go
package persistence
|
|
|
|
import (
|
|
"database/sql/driver"
|
|
"fmt"
|
|
"regexp"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/Masterminds/squirrel"
|
|
"github.com/fatih/structs"
|
|
)
|
|
|
|
type PostMapper interface {
|
|
PostMapArgs(map[string]any) error
|
|
}
|
|
|
|
func toSQLArgs(rec interface{}) (map[string]interface{}, error) {
|
|
m := structs.Map(rec)
|
|
for k, v := range m {
|
|
switch t := v.(type) {
|
|
case time.Time:
|
|
m[k] = t.Format(time.RFC3339Nano)
|
|
case *time.Time:
|
|
if t != nil {
|
|
m[k] = t.Format(time.RFC3339Nano)
|
|
}
|
|
case driver.Valuer:
|
|
var err error
|
|
m[k], err = t.Value()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
if r, ok := rec.(PostMapper); ok {
|
|
err := r.PostMapArgs(m)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
var matchFirstCap = regexp.MustCompile("(.)([A-Z][a-z]+)")
|
|
var matchAllCap = regexp.MustCompile("([a-z0-9])([A-Z])")
|
|
|
|
func toSnakeCase(str string) string {
|
|
snake := matchFirstCap.ReplaceAllString(str, "${1}_${2}")
|
|
snake = matchAllCap.ReplaceAllString(snake, "${1}_${2}")
|
|
return strings.ToLower(snake)
|
|
}
|
|
|
|
var matchUnderscore = regexp.MustCompile("_([A-Za-z])")
|
|
|
|
func toCamelCase(str string) string {
|
|
return matchUnderscore.ReplaceAllStringFunc(str, func(s string) string {
|
|
return strings.ToUpper(strings.Replace(s, "_", "", -1))
|
|
})
|
|
}
|
|
|
|
func exists(subTable string, cond squirrel.Sqlizer) existsCond {
|
|
return existsCond{subTable: subTable, cond: cond, not: false}
|
|
}
|
|
|
|
func notExists(subTable string, cond squirrel.Sqlizer) existsCond {
|
|
return existsCond{subTable: subTable, cond: cond, not: true}
|
|
}
|
|
|
|
type existsCond struct {
|
|
subTable string
|
|
cond squirrel.Sqlizer
|
|
not bool
|
|
}
|
|
|
|
func (e existsCond) ToSql() (string, []interface{}, error) {
|
|
sql, args, err := e.cond.ToSql()
|
|
sql = fmt.Sprintf("exists (select 1 from %s where %s)", e.subTable, sql)
|
|
if e.not {
|
|
sql = "not " + sql
|
|
}
|
|
return sql, args, err
|
|
}
|
|
|
|
var sortOrderRegex = regexp.MustCompile(`order_([a-z_]+)`)
|
|
|
|
// Convert the order_* columns to an expression using sort_* columns. Example:
|
|
// sort_album_name -> (coalesce(nullif(sort_album_name,”),order_album_name) collate nocase)
|
|
// It finds order column names anywhere in the substring
|
|
func mapSortOrder(order string) string {
|
|
order = strings.ToLower(order)
|
|
return sortOrderRegex.ReplaceAllString(order, "(coalesce(nullif(sort_$1,''),order_$1) collate nocase)")
|
|
}
|