mirror of https://github.com/stashapp/stash.git
386 lines
12 KiB
Go
386 lines
12 KiB
Go
package sqlite
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"errors"
|
|
"fmt"
|
|
|
|
"github.com/doug-martin/goqu/v9"
|
|
"github.com/jmoiron/sqlx"
|
|
"github.com/stashapp/stash/pkg/models"
|
|
"github.com/stashapp/stash/pkg/sliceutil/intslice"
|
|
)
|
|
|
|
const movieTable = "movies"
|
|
const movieIDColumn = "movie_id"
|
|
|
|
type movieQueryBuilder struct {
|
|
repository
|
|
}
|
|
|
|
var MovieReaderWriter = &movieQueryBuilder{
|
|
repository{
|
|
tableName: movieTable,
|
|
idColumn: idColumn,
|
|
},
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) Create(ctx context.Context, newObject models.Movie) (*models.Movie, error) {
|
|
var ret models.Movie
|
|
if err := qb.insertObject(ctx, newObject, &ret); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &ret, nil
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) Update(ctx context.Context, updatedObject models.MoviePartial) (*models.Movie, error) {
|
|
const partial = true
|
|
if err := qb.update(ctx, updatedObject.ID, updatedObject, partial); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return qb.Find(ctx, updatedObject.ID)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) UpdateFull(ctx context.Context, updatedObject models.Movie) (*models.Movie, error) {
|
|
const partial = false
|
|
if err := qb.update(ctx, updatedObject.ID, updatedObject, partial); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return qb.Find(ctx, updatedObject.ID)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) Destroy(ctx context.Context, id int) error {
|
|
return qb.destroyExisting(ctx, []int{id})
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) Find(ctx context.Context, id int) (*models.Movie, error) {
|
|
var ret models.Movie
|
|
if err := qb.getByID(ctx, id, &ret); err != nil {
|
|
if errors.Is(err, sql.ErrNoRows) {
|
|
return nil, nil
|
|
}
|
|
return nil, err
|
|
}
|
|
return &ret, nil
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) FindMany(ctx context.Context, ids []int) ([]*models.Movie, error) {
|
|
tableMgr := movieTableMgr
|
|
q := goqu.Select("*").From(tableMgr.table).Where(tableMgr.byIDInts(ids...))
|
|
unsorted, err := qb.getMany(ctx, q)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
ret := make([]*models.Movie, len(ids))
|
|
|
|
for _, s := range unsorted {
|
|
i := intslice.IntIndex(ids, s.ID)
|
|
ret[i] = s
|
|
}
|
|
|
|
for i := range ret {
|
|
if ret[i] == nil {
|
|
return nil, fmt.Errorf("movie with id %d not found", ids[i])
|
|
}
|
|
}
|
|
|
|
return ret, nil
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) getMany(ctx context.Context, q *goqu.SelectDataset) ([]*models.Movie, error) {
|
|
const single = false
|
|
var ret []*models.Movie
|
|
if err := queryFunc(ctx, q, single, func(r *sqlx.Rows) error {
|
|
var f models.Movie
|
|
if err := r.StructScan(&f); err != nil {
|
|
return err
|
|
}
|
|
|
|
ret = append(ret, &f)
|
|
return nil
|
|
}); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return ret, nil
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) FindByName(ctx context.Context, name string, nocase bool) (*models.Movie, error) {
|
|
query := "SELECT * FROM movies WHERE name = ?"
|
|
if nocase {
|
|
query += " COLLATE NOCASE"
|
|
}
|
|
query += " LIMIT 1"
|
|
args := []interface{}{name}
|
|
return qb.queryMovie(ctx, query, args)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) FindByNames(ctx context.Context, names []string, nocase bool) ([]*models.Movie, error) {
|
|
query := "SELECT * FROM movies WHERE name"
|
|
if nocase {
|
|
query += " COLLATE NOCASE"
|
|
}
|
|
query += " IN " + getInBinding(len(names))
|
|
var args []interface{}
|
|
for _, name := range names {
|
|
args = append(args, name)
|
|
}
|
|
return qb.queryMovies(ctx, query, args)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) Count(ctx context.Context) (int, error) {
|
|
return qb.runCountQuery(ctx, qb.buildCountQuery("SELECT movies.id FROM movies"), nil)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) All(ctx context.Context) ([]*models.Movie, error) {
|
|
return qb.queryMovies(ctx, selectAll("movies")+qb.getMovieSort(nil), nil)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) makeFilter(ctx context.Context, movieFilter *models.MovieFilterType) *filterBuilder {
|
|
query := &filterBuilder{}
|
|
|
|
query.handleCriterion(ctx, stringCriterionHandler(movieFilter.Name, "movies.name"))
|
|
query.handleCriterion(ctx, stringCriterionHandler(movieFilter.Director, "movies.director"))
|
|
query.handleCriterion(ctx, stringCriterionHandler(movieFilter.Synopsis, "movies.synopsis"))
|
|
query.handleCriterion(ctx, intCriterionHandler(movieFilter.Rating, "movies.rating", nil))
|
|
query.handleCriterion(ctx, durationCriterionHandler(movieFilter.Duration, "movies.duration", nil))
|
|
query.handleCriterion(ctx, movieIsMissingCriterionHandler(qb, movieFilter.IsMissing))
|
|
query.handleCriterion(ctx, stringCriterionHandler(movieFilter.URL, "movies.url"))
|
|
query.handleCriterion(ctx, movieStudioCriterionHandler(qb, movieFilter.Studios))
|
|
query.handleCriterion(ctx, moviePerformersCriterionHandler(qb, movieFilter.Performers))
|
|
|
|
return query
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) Query(ctx context.Context, movieFilter *models.MovieFilterType, findFilter *models.FindFilterType) ([]*models.Movie, int, error) {
|
|
if findFilter == nil {
|
|
findFilter = &models.FindFilterType{}
|
|
}
|
|
if movieFilter == nil {
|
|
movieFilter = &models.MovieFilterType{}
|
|
}
|
|
|
|
query := qb.newQuery()
|
|
distinctIDs(&query, movieTable)
|
|
|
|
if q := findFilter.Q; q != nil && *q != "" {
|
|
searchColumns := []string{"movies.name"}
|
|
query.parseQueryString(searchColumns, *q)
|
|
}
|
|
|
|
filter := qb.makeFilter(ctx, movieFilter)
|
|
|
|
query.addFilter(filter)
|
|
|
|
query.sortAndPagination = qb.getMovieSort(findFilter) + getPagination(findFilter)
|
|
idsResult, countResult, err := query.executeFind(ctx)
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
|
|
movies, err := qb.FindMany(ctx, idsResult)
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
|
|
return movies, countResult, nil
|
|
}
|
|
|
|
func movieIsMissingCriterionHandler(qb *movieQueryBuilder, isMissing *string) criterionHandlerFunc {
|
|
return func(ctx context.Context, f *filterBuilder) {
|
|
if isMissing != nil && *isMissing != "" {
|
|
switch *isMissing {
|
|
case "front_image":
|
|
f.addLeftJoin("movies_images", "", "movies_images.movie_id = movies.id")
|
|
f.addWhere("movies_images.front_image IS NULL")
|
|
case "back_image":
|
|
f.addLeftJoin("movies_images", "", "movies_images.movie_id = movies.id")
|
|
f.addWhere("movies_images.back_image IS NULL")
|
|
case "scenes":
|
|
f.addLeftJoin("movies_scenes", "", "movies_scenes.movie_id = movies.id")
|
|
f.addWhere("movies_scenes.scene_id IS NULL")
|
|
default:
|
|
f.addWhere("(movies." + *isMissing + " IS NULL OR TRIM(movies." + *isMissing + ") = '')")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func movieStudioCriterionHandler(qb *movieQueryBuilder, studios *models.HierarchicalMultiCriterionInput) criterionHandlerFunc {
|
|
h := hierarchicalMultiCriterionHandlerBuilder{
|
|
tx: qb.tx,
|
|
|
|
primaryTable: movieTable,
|
|
foreignTable: studioTable,
|
|
foreignFK: studioIDColumn,
|
|
derivedTable: "studio",
|
|
parentFK: "parent_id",
|
|
}
|
|
|
|
return h.handler(studios)
|
|
}
|
|
|
|
func moviePerformersCriterionHandler(qb *movieQueryBuilder, performers *models.MultiCriterionInput) criterionHandlerFunc {
|
|
return func(ctx context.Context, f *filterBuilder) {
|
|
if performers != nil {
|
|
if performers.Modifier == models.CriterionModifierIsNull || performers.Modifier == models.CriterionModifierNotNull {
|
|
var notClause string
|
|
if performers.Modifier == models.CriterionModifierNotNull {
|
|
notClause = "NOT"
|
|
}
|
|
|
|
f.addLeftJoin("movies_scenes", "", "movies.id = movies_scenes.movie_id")
|
|
f.addLeftJoin("performers_scenes", "", "movies_scenes.scene_id = performers_scenes.scene_id")
|
|
|
|
f.addWhere(fmt.Sprintf("performers_scenes.performer_id IS %s NULL", notClause))
|
|
return
|
|
}
|
|
|
|
if len(performers.Value) == 0 {
|
|
return
|
|
}
|
|
|
|
var args []interface{}
|
|
for _, arg := range performers.Value {
|
|
args = append(args, arg)
|
|
}
|
|
|
|
// Hack, can't apply args to join, nor inner join on a left join, so use CTE instead
|
|
f.addWith(`movies_performers AS (
|
|
SELECT movies_scenes.movie_id, performers_scenes.performer_id
|
|
FROM movies_scenes
|
|
INNER JOIN performers_scenes ON movies_scenes.scene_id = performers_scenes.scene_id
|
|
WHERE performers_scenes.performer_id IN`+getInBinding(len(performers.Value))+`
|
|
)`, args...)
|
|
f.addLeftJoin("movies_performers", "", "movies.id = movies_performers.movie_id")
|
|
|
|
switch performers.Modifier {
|
|
case models.CriterionModifierIncludes:
|
|
f.addWhere("movies_performers.performer_id IS NOT NULL")
|
|
case models.CriterionModifierIncludesAll:
|
|
f.addWhere("movies_performers.performer_id IS NOT NULL")
|
|
f.addHaving("COUNT(DISTINCT movies_performers.performer_id) = ?", len(performers.Value))
|
|
case models.CriterionModifierExcludes:
|
|
f.addWhere("movies_performers.performer_id IS NULL")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) getMovieSort(findFilter *models.FindFilterType) string {
|
|
var sort string
|
|
var direction string
|
|
if findFilter == nil {
|
|
sort = "name"
|
|
direction = "ASC"
|
|
} else {
|
|
sort = findFilter.GetSort("name")
|
|
direction = findFilter.GetDirection()
|
|
}
|
|
|
|
switch sort {
|
|
case "name": // #943 - override name sorting to use natural sort
|
|
return " ORDER BY " + getColumn("movies", sort) + " COLLATE NATURAL_CS " + direction
|
|
case "scenes_count": // generic getSort won't work for this
|
|
return getCountSort(movieTable, moviesScenesTable, movieIDColumn, direction)
|
|
default:
|
|
return getSort(sort, direction, "movies")
|
|
}
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) queryMovie(ctx context.Context, query string, args []interface{}) (*models.Movie, error) {
|
|
results, err := qb.queryMovies(ctx, query, args)
|
|
if err != nil || len(results) < 1 {
|
|
return nil, err
|
|
}
|
|
return results[0], nil
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) queryMovies(ctx context.Context, query string, args []interface{}) ([]*models.Movie, error) {
|
|
var ret models.Movies
|
|
if err := qb.query(ctx, query, args, &ret); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return []*models.Movie(ret), nil
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) UpdateImages(ctx context.Context, movieID int, frontImage []byte, backImage []byte) error {
|
|
// Delete the existing cover and then create new
|
|
if err := qb.DestroyImages(ctx, movieID); err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err := qb.tx.Exec(ctx,
|
|
`INSERT INTO movies_images (movie_id, front_image, back_image) VALUES (?, ?, ?)`,
|
|
movieID,
|
|
frontImage,
|
|
backImage,
|
|
)
|
|
|
|
return err
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) DestroyImages(ctx context.Context, movieID int) error {
|
|
// Delete the existing joins
|
|
_, err := qb.tx.Exec(ctx, "DELETE FROM movies_images WHERE movie_id = ?", movieID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) GetFrontImage(ctx context.Context, movieID int) ([]byte, error) {
|
|
query := `SELECT front_image from movies_images WHERE movie_id = ?`
|
|
return getImage(ctx, qb.tx, query, movieID)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) GetBackImage(ctx context.Context, movieID int) ([]byte, error) {
|
|
query := `SELECT back_image from movies_images WHERE movie_id = ?`
|
|
return getImage(ctx, qb.tx, query, movieID)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) FindByPerformerID(ctx context.Context, performerID int) ([]*models.Movie, error) {
|
|
query := `SELECT DISTINCT movies.*
|
|
FROM movies
|
|
INNER JOIN movies_scenes ON movies.id = movies_scenes.movie_id
|
|
INNER JOIN performers_scenes ON performers_scenes.scene_id = movies_scenes.scene_id
|
|
WHERE performers_scenes.performer_id = ?
|
|
`
|
|
args := []interface{}{performerID}
|
|
return qb.queryMovies(ctx, query, args)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) CountByPerformerID(ctx context.Context, performerID int) (int, error) {
|
|
query := `SELECT COUNT(DISTINCT movies_scenes.movie_id) AS count
|
|
FROM movies_scenes
|
|
INNER JOIN performers_scenes ON performers_scenes.scene_id = movies_scenes.scene_id
|
|
WHERE performers_scenes.performer_id = ?
|
|
`
|
|
args := []interface{}{performerID}
|
|
return qb.runCountQuery(ctx, query, args)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) FindByStudioID(ctx context.Context, studioID int) ([]*models.Movie, error) {
|
|
query := `SELECT movies.*
|
|
FROM movies
|
|
WHERE movies.studio_id = ?
|
|
`
|
|
args := []interface{}{studioID}
|
|
return qb.queryMovies(ctx, query, args)
|
|
}
|
|
|
|
func (qb *movieQueryBuilder) CountByStudioID(ctx context.Context, studioID int) (int, error) {
|
|
query := `SELECT COUNT(1) AS count
|
|
FROM movies
|
|
WHERE movies.studio_id = ?
|
|
`
|
|
args := []interface{}{studioID}
|
|
return qb.runCountQuery(ctx, query, args)
|
|
}
|