navidrome/model/criteria/operators.go

278 lines
6.3 KiB
Go

package criteria
import (
"errors"
"fmt"
"reflect"
"strconv"
"time"
"github.com/Masterminds/squirrel"
)
type (
All squirrel.And
And = All
)
func (all All) ToSql() (sql string, args []interface{}, err error) {
return squirrel.And(all).ToSql()
}
func (all All) MarshalJSON() ([]byte, error) {
return marshalConjunction("all", all)
}
type (
Any squirrel.Or
Or = Any
)
func (any Any) ToSql() (sql string, args []interface{}, err error) {
return squirrel.Or(any).ToSql()
}
func (any Any) MarshalJSON() ([]byte, error) {
return marshalConjunction("any", any)
}
type Is squirrel.Eq
type Eq = Is
func (is Is) ToSql() (sql string, args []interface{}, err error) {
return squirrel.Eq(mapFields(is)).ToSql()
}
func (is Is) MarshalJSON() ([]byte, error) {
return marshalExpression("is", is)
}
type IsNot squirrel.NotEq
func (in IsNot) ToSql() (sql string, args []interface{}, err error) {
return squirrel.NotEq(mapFields(in)).ToSql()
}
func (in IsNot) MarshalJSON() ([]byte, error) {
return marshalExpression("isNot", in)
}
type Gt squirrel.Gt
func (gt Gt) ToSql() (sql string, args []interface{}, err error) {
return squirrel.Gt(mapFields(gt)).ToSql()
}
func (gt Gt) MarshalJSON() ([]byte, error) {
return marshalExpression("gt", gt)
}
type Lt squirrel.Lt
func (lt Lt) ToSql() (sql string, args []interface{}, err error) {
return squirrel.Lt(mapFields(lt)).ToSql()
}
func (lt Lt) MarshalJSON() ([]byte, error) {
return marshalExpression("lt", lt)
}
type Before squirrel.Lt
func (bf Before) ToSql() (sql string, args []interface{}, err error) {
return squirrel.Lt(mapFields(bf)).ToSql()
}
func (bf Before) MarshalJSON() ([]byte, error) {
return marshalExpression("before", bf)
}
type After squirrel.Gt
func (af After) ToSql() (sql string, args []interface{}, err error) {
return squirrel.Gt(mapFields(af)).ToSql()
}
func (af After) MarshalJSON() ([]byte, error) {
return marshalExpression("after", af)
}
type Contains map[string]interface{}
func (ct Contains) ToSql() (sql string, args []interface{}, err error) {
lk := squirrel.Like{}
for f, v := range mapFields(ct) {
lk[f] = fmt.Sprintf("%%%s%%", v)
}
return lk.ToSql()
}
func (ct Contains) MarshalJSON() ([]byte, error) {
return marshalExpression("contains", ct)
}
type NotContains map[string]interface{}
func (nct NotContains) ToSql() (sql string, args []interface{}, err error) {
lk := squirrel.NotLike{}
for f, v := range mapFields(nct) {
lk[f] = fmt.Sprintf("%%%s%%", v)
}
return lk.ToSql()
}
func (nct NotContains) MarshalJSON() ([]byte, error) {
return marshalExpression("notContains", nct)
}
type StartsWith map[string]interface{}
func (sw StartsWith) ToSql() (sql string, args []interface{}, err error) {
lk := squirrel.Like{}
for f, v := range mapFields(sw) {
lk[f] = fmt.Sprintf("%s%%", v)
}
return lk.ToSql()
}
func (sw StartsWith) MarshalJSON() ([]byte, error) {
return marshalExpression("startsWith", sw)
}
type EndsWith map[string]interface{}
func (sw EndsWith) ToSql() (sql string, args []interface{}, err error) {
lk := squirrel.Like{}
for f, v := range mapFields(sw) {
lk[f] = fmt.Sprintf("%%%s", v)
}
return lk.ToSql()
}
func (sw EndsWith) MarshalJSON() ([]byte, error) {
return marshalExpression("endsWith", sw)
}
type InTheRange map[string]interface{}
func (itr InTheRange) ToSql() (sql string, args []interface{}, err error) {
var and squirrel.And
for f, v := range mapFields(itr) {
s := reflect.ValueOf(v)
if s.Kind() != reflect.Slice || s.Len() != 2 {
return "", nil, fmt.Errorf("invalid range for 'in' operator: %s", v)
}
and = append(and,
squirrel.GtOrEq{f: s.Index(0).Interface()},
squirrel.LtOrEq{f: s.Index(1).Interface()},
)
}
return and.ToSql()
}
func (itr InTheRange) MarshalJSON() ([]byte, error) {
return marshalExpression("inTheRange", itr)
}
type InTheLast map[string]interface{}
func (itl InTheLast) ToSql() (sql string, args []interface{}, err error) {
exp, err := inPeriod(itl, false)
if err != nil {
return "", nil, err
}
return exp.ToSql()
}
func (itl InTheLast) MarshalJSON() ([]byte, error) {
return marshalExpression("inTheLast", itl)
}
type NotInTheLast map[string]interface{}
func (nitl NotInTheLast) ToSql() (sql string, args []interface{}, err error) {
exp, err := inPeriod(nitl, true)
if err != nil {
return "", nil, err
}
return exp.ToSql()
}
func (nitl NotInTheLast) MarshalJSON() ([]byte, error) {
return marshalExpression("notInTheLast", nitl)
}
func inPeriod(m map[string]interface{}, negate bool) (Expression, error) {
var field string
var value interface{}
for f, v := range mapFields(m) {
field, value = f, v
break
}
str := fmt.Sprintf("%v", value)
v, err := strconv.ParseInt(str, 10, 64)
if err != nil {
return nil, err
}
firstDate := startOfPeriod(v, time.Now())
if negate {
return Or{
squirrel.Lt{field: firstDate},
squirrel.Eq{field: nil},
}, nil
}
return squirrel.Gt{field: firstDate}, nil
}
func startOfPeriod(numDays int64, from time.Time) string {
return from.Add(time.Duration(-24*numDays) * time.Hour).Format("2006-01-02")
}
type InPlaylist map[string]interface{}
func (ipl InPlaylist) ToSql() (sql string, args []interface{}, err error) {
return inList(ipl, false)
}
func (ipl InPlaylist) MarshalJSON() ([]byte, error) {
return marshalExpression("inPlaylist", ipl)
}
type NotInPlaylist map[string]interface{}
func (ipl NotInPlaylist) ToSql() (sql string, args []interface{}, err error) {
return inList(ipl, true)
}
func (ipl NotInPlaylist) MarshalJSON() ([]byte, error) {
return marshalExpression("notInPlaylist", ipl)
}
func inList(m map[string]interface{}, negate bool) (sql string, args []interface{}, err error) {
var playlistid string
var ok bool
if playlistid, ok = m["id"].(string); !ok {
return "", nil, errors.New("playlist id not given")
}
// Subquery to fetch all media files that are contained in given playlist
// Only evaluate playlist if it is public
subQuery := squirrel.Select("media_file_id").
From("playlist_tracks pl").
LeftJoin("playlist on pl.playlist_id = playlist.id").
Where(squirrel.And{
squirrel.Eq{"pl.playlist_id": playlistid},
squirrel.Eq{"playlist.public": 1}})
subQText, subQArgs, err := subQuery.PlaceholderFormat(squirrel.Question).ToSql()
if err != nil {
return "", nil, err
}
if negate {
return "media_file.id NOT IN (" + subQText + ")", subQArgs, nil
} else {
return "media_file.id IN (" + subQText + ")", subQArgs, nil
}
}