2019-02-09 12:30:49 +00:00
|
|
|
package models
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
|
|
|
"github.com/jmoiron/sqlx"
|
2019-02-10 02:53:08 +00:00
|
|
|
"github.com/stashapp/stash/database"
|
2019-02-09 12:30:49 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type performerQueryBuilder struct {}
|
|
|
|
|
|
|
|
func NewPerformerQueryBuilder() performerQueryBuilder {
|
|
|
|
return performerQueryBuilder{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (qb *performerQueryBuilder) Create(newPerformer Performer, tx *sqlx.Tx) (*Performer, error) {
|
|
|
|
ensureTx(tx)
|
|
|
|
result, err := tx.NamedExec(
|
|
|
|
`INSERT INTO performers (image, checksum, name, url, twitter, instagram, birthdate, ethnicity, country,
|
|
|
|
eye_color, height, measurements, fake_tits, career_length, tattoos, piercings,
|
|
|
|
aliases, favorite, created_at, updated_at)
|
|
|
|
VALUES (:image, :checksum, :name, :url, :twitter, :instagram, :birthdate, :ethnicity, :country,
|
|
|
|
:eye_color, :height, :measurements, :fake_tits, :career_length, :tattoos, :piercings,
|
|
|
|
:aliases, :favorite, :created_at, :updated_at)
|
|
|
|
`,
|
|
|
|
newPerformer,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
performerID, err := result.LastInsertId()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := tx.Get(&newPerformer, `SELECT * FROM performers WHERE id = ? LIMIT 1`, performerID); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &newPerformer, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (qb *performerQueryBuilder) Update(updatedPerformer Performer, tx *sqlx.Tx) (*Performer, error) {
|
|
|
|
ensureTx(tx)
|
|
|
|
_, err := tx.NamedExec(
|
|
|
|
`UPDATE performers SET `+SqlGenKeys(updatedPerformer)+` WHERE performers.id = :id`,
|
|
|
|
updatedPerformer,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := tx.Get(&updatedPerformer, `SELECT * FROM performers WHERE id = ? LIMIT 1`, updatedPerformer.ID); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &updatedPerformer, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (qb *performerQueryBuilder) Find(id int) (*Performer, error) {
|
|
|
|
query := "SELECT * FROM performers WHERE id = ? LIMIT 1"
|
|
|
|
args := []interface{}{id}
|
|
|
|
results, err := qb.queryPerformers(query, args, nil)
|
|
|
|
if err != nil || len(results) < 1 {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &results[0], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (qb *performerQueryBuilder) FindBySceneID(sceneID int, tx *sqlx.Tx) ([]Performer, error) {
|
|
|
|
query := `
|
|
|
|
SELECT performers.* FROM performers
|
|
|
|
LEFT JOIN performers_scenes as scenes_join on scenes_join.performer_id = performers.id
|
|
|
|
LEFT JOIN scenes on scenes_join.scene_id = scenes.id
|
|
|
|
WHERE scenes.id = ?
|
|
|
|
GROUP BY performers.id
|
|
|
|
`
|
|
|
|
args := []interface{}{sceneID}
|
|
|
|
return qb.queryPerformers(query, args, tx)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (qb *performerQueryBuilder) FindByNames(names []string, tx *sqlx.Tx) ([]Performer, error) {
|
|
|
|
query := "SELECT * FROM performers WHERE name IN " + getInBinding(len(names))
|
|
|
|
var args []interface{}
|
|
|
|
for _, name := range names {
|
|
|
|
args = append(args, name)
|
|
|
|
}
|
|
|
|
return qb.queryPerformers(query, args, tx)
|
|
|
|
}
|
|
|
|
|
2019-02-11 20:36:10 +00:00
|
|
|
func (qb *performerQueryBuilder) Count() (int, error) {
|
|
|
|
return runCountQuery(buildCountQuery("SELECT performers.id FROM performers"), nil)
|
|
|
|
}
|
|
|
|
|
2019-02-09 12:30:49 +00:00
|
|
|
func (qb *performerQueryBuilder) All() ([]Performer, error) {
|
|
|
|
return qb.queryPerformers(selectAll("performers") + qb.getPerformerSort(nil), nil, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (qb *performerQueryBuilder) Query(performerFilter *PerformerFilterType, findFilter *FindFilterType) ([]Performer, int) {
|
|
|
|
if performerFilter == nil {
|
|
|
|
performerFilter = &PerformerFilterType{}
|
|
|
|
}
|
|
|
|
if findFilter == nil {
|
|
|
|
findFilter = &FindFilterType{}
|
|
|
|
}
|
|
|
|
|
|
|
|
whereClauses := []string{}
|
|
|
|
havingClauses := []string{}
|
|
|
|
args := []interface{}{}
|
|
|
|
body := selectDistinctIDs("performers")
|
|
|
|
body += `
|
|
|
|
left join performers_scenes as scenes_join on scenes_join.performer_id = performers.id
|
|
|
|
left join scenes on scenes_join.scene_id = scenes.id
|
|
|
|
`
|
|
|
|
|
|
|
|
if q := findFilter.Q; q != nil && *q != "" {
|
|
|
|
searchColumns := []string{"performers.name", "performers.checksum", "performers.birthdate", "performers.ethnicity"}
|
|
|
|
whereClauses = append(whereClauses, getSearch(searchColumns, *q))
|
|
|
|
}
|
|
|
|
|
|
|
|
if favoritesFilter := performerFilter.FilterFavorites; favoritesFilter != nil {
|
|
|
|
if *favoritesFilter == true {
|
|
|
|
whereClauses = append(whereClauses, "performers.favorite = 1")
|
|
|
|
} else {
|
|
|
|
whereClauses = append(whereClauses, "performers.favorite = 0")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sortAndPagination := qb.getPerformerSort(findFilter) + getPagination(findFilter)
|
|
|
|
idsResult, countResult := executeFindQuery("performers", body, args, sortAndPagination, whereClauses, havingClauses)
|
|
|
|
|
|
|
|
var performers []Performer
|
|
|
|
for _, id := range idsResult {
|
|
|
|
performer, _ := qb.Find(id)
|
|
|
|
performers = append(performers, *performer)
|
|
|
|
}
|
|
|
|
|
|
|
|
return performers, countResult
|
|
|
|
}
|
|
|
|
|
|
|
|
func (qb *performerQueryBuilder) getPerformerSort(findFilter *FindFilterType) string {
|
|
|
|
var sort string
|
|
|
|
var direction string
|
|
|
|
if findFilter == nil {
|
|
|
|
sort = "name"
|
|
|
|
direction = "ASC"
|
|
|
|
} else {
|
|
|
|
sort = findFilter.GetSort("name")
|
|
|
|
direction = findFilter.GetDirection()
|
|
|
|
}
|
|
|
|
return getSort(sort, direction, "performers")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (qb *performerQueryBuilder) queryPerformers(query string, args []interface{}, tx *sqlx.Tx) ([]Performer, error) {
|
|
|
|
var rows *sqlx.Rows
|
|
|
|
var err error
|
|
|
|
if tx != nil {
|
|
|
|
rows, err = tx.Queryx(query, args...)
|
|
|
|
} else {
|
|
|
|
rows, err = database.DB.Queryx(query, args...)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil && err != sql.ErrNoRows {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer rows.Close()
|
|
|
|
|
|
|
|
performers := make([]Performer, 0)
|
|
|
|
performer := Performer{}
|
|
|
|
for rows.Next() {
|
|
|
|
if err := rows.StructScan(&performer); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
performers = append(performers, performer)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := rows.Err(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return performers, nil
|
|
|
|
}
|