stash/pkg/sqlite/tag.go

323 lines
8.3 KiB
Go

package sqlite
import (
"database/sql"
"errors"
"fmt"
"github.com/stashapp/stash/pkg/models"
)
const tagTable = "tags"
const tagIDColumn = "tag_id"
type tagQueryBuilder struct {
repository
}
func NewTagReaderWriter(tx dbi) *tagQueryBuilder {
return &tagQueryBuilder{
repository{
tx: tx,
tableName: tagTable,
idColumn: idColumn,
},
}
}
func (qb *tagQueryBuilder) Create(newObject models.Tag) (*models.Tag, error) {
var ret models.Tag
if err := qb.insertObject(newObject, &ret); err != nil {
return nil, err
}
return &ret, nil
}
func (qb *tagQueryBuilder) Update(updatedObject models.Tag) (*models.Tag, error) {
const partial = false
if err := qb.update(updatedObject.ID, updatedObject, partial); err != nil {
return nil, err
}
return qb.Find(updatedObject.ID)
}
func (qb *tagQueryBuilder) Destroy(id int) error {
// TODO - add delete cascade to foreign key
// delete tag from scenes and markers first
_, err := qb.tx.Exec("DELETE FROM scenes_tags WHERE tag_id = ?", id)
if err != nil {
return err
}
// TODO - add delete cascade to foreign key
_, err = qb.tx.Exec("DELETE FROM scene_markers_tags WHERE tag_id = ?", id)
if err != nil {
return err
}
// cannot unset primary_tag_id in scene_markers because it is not nullable
countQuery := "SELECT COUNT(*) as count FROM scene_markers where primary_tag_id = ?"
args := []interface{}{id}
primaryMarkers, err := qb.runCountQuery(countQuery, args)
if err != nil {
return err
}
if primaryMarkers > 0 {
return errors.New("Cannot delete tag used as a primary tag in scene markers")
}
return qb.destroyExisting([]int{id})
}
func (qb *tagQueryBuilder) Find(id int) (*models.Tag, error) {
var ret models.Tag
if err := qb.get(id, &ret); err != nil {
if err == sql.ErrNoRows {
return nil, nil
}
return nil, err
}
return &ret, nil
}
func (qb *tagQueryBuilder) FindMany(ids []int) ([]*models.Tag, error) {
var tags []*models.Tag
for _, id := range ids {
tag, err := qb.Find(id)
if err != nil {
return nil, err
}
if tag == nil {
return nil, fmt.Errorf("tag with id %d not found", id)
}
tags = append(tags, tag)
}
return tags, nil
}
func (qb *tagQueryBuilder) FindBySceneID(sceneID int) ([]*models.Tag, error) {
query := `
SELECT tags.* FROM tags
LEFT JOIN scenes_tags as scenes_join on scenes_join.tag_id = tags.id
WHERE scenes_join.scene_id = ?
GROUP BY tags.id
`
query += qb.getTagSort(nil)
args := []interface{}{sceneID}
return qb.queryTags(query, args)
}
func (qb *tagQueryBuilder) FindByImageID(imageID int) ([]*models.Tag, error) {
query := `
SELECT tags.* FROM tags
LEFT JOIN images_tags as images_join on images_join.tag_id = tags.id
WHERE images_join.image_id = ?
GROUP BY tags.id
`
query += qb.getTagSort(nil)
args := []interface{}{imageID}
return qb.queryTags(query, args)
}
func (qb *tagQueryBuilder) FindByGalleryID(galleryID int) ([]*models.Tag, error) {
query := `
SELECT tags.* FROM tags
LEFT JOIN galleries_tags as galleries_join on galleries_join.tag_id = tags.id
WHERE galleries_join.gallery_id = ?
GROUP BY tags.id
`
query += qb.getTagSort(nil)
args := []interface{}{galleryID}
return qb.queryTags(query, args)
}
func (qb *tagQueryBuilder) FindBySceneMarkerID(sceneMarkerID int) ([]*models.Tag, error) {
query := `
SELECT tags.* FROM tags
LEFT JOIN scene_markers_tags as scene_markers_join on scene_markers_join.tag_id = tags.id
WHERE scene_markers_join.scene_marker_id = ?
GROUP BY tags.id
`
query += qb.getTagSort(nil)
args := []interface{}{sceneMarkerID}
return qb.queryTags(query, args)
}
func (qb *tagQueryBuilder) FindByName(name string, nocase bool) (*models.Tag, error) {
query := "SELECT * FROM tags WHERE name = ?"
if nocase {
query += " COLLATE NOCASE"
}
query += " LIMIT 1"
args := []interface{}{name}
return qb.queryTag(query, args)
}
func (qb *tagQueryBuilder) FindByNames(names []string, nocase bool) ([]*models.Tag, error) {
query := "SELECT * FROM tags 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.queryTags(query, args)
}
func (qb *tagQueryBuilder) Count() (int, error) {
return qb.runCountQuery(qb.buildCountQuery("SELECT tags.id FROM tags"), nil)
}
func (qb *tagQueryBuilder) All() ([]*models.Tag, error) {
return qb.queryTags(selectAll("tags")+qb.getTagSort(nil), nil)
}
func (qb *tagQueryBuilder) AllSlim() ([]*models.Tag, error) {
return qb.queryTags("SELECT tags.id, tags.name FROM tags "+qb.getTagSort(nil), nil)
}
func (qb *tagQueryBuilder) Query(tagFilter *models.TagFilterType, findFilter *models.FindFilterType) ([]*models.Tag, int, error) {
if tagFilter == nil {
tagFilter = &models.TagFilterType{}
}
if findFilter == nil {
findFilter = &models.FindFilterType{}
}
query := qb.newQuery()
query.body = selectDistinctIDs(tagTable)
/*
query.body += `
left join tags_image on tags_image.tag_id = tags.id
left join scenes_tags on scenes_tags.tag_id = tags.id
left join scene_markers_tags on scene_markers_tags.tag_id = tags.id
left join scene_markers on scene_markers.primary_tag_id = tags.id OR scene_markers.id = scene_markers_tags.scene_marker_id
left join scenes on scenes_tags.scene_id = scenes.id`
*/
// the presence of joining on scene_markers.primary_tag_id and scene_markers_tags.tag_id
// appears to confuse sqlite and causes serious performance issues.
// Disabling querying/sorting on marker count for now.
query.body += `
left join tags_image on tags_image.tag_id = tags.id
left join scenes_tags on scenes_tags.tag_id = tags.id
left join scenes on scenes_tags.scene_id = scenes.id`
if q := findFilter.Q; q != nil && *q != "" {
searchColumns := []string{"tags.name"}
clause, thisArgs := getSearchBinding(searchColumns, *q, false)
query.addWhere(clause)
query.addArg(thisArgs...)
}
if isMissingFilter := tagFilter.IsMissing; isMissingFilter != nil && *isMissingFilter != "" {
switch *isMissingFilter {
case "image":
query.addWhere("tags_image.tag_id IS NULL")
default:
query.addWhere("tags." + *isMissingFilter + " IS NULL")
}
}
if sceneCount := tagFilter.SceneCount; sceneCount != nil {
clause, count := getIntCriterionWhereClause("count(distinct scenes_tags.scene_id)", *sceneCount)
query.addHaving(clause)
if count == 1 {
query.addArg(sceneCount.Value)
}
}
// if markerCount := tagFilter.MarkerCount; markerCount != nil {
// clause, count := getIntCriterionWhereClause("count(distinct scene_markers.id)", *markerCount)
// query.addHaving(clause)
// if count == 1 {
// query.addArg(markerCount.Value)
// }
// }
query.sortAndPagination = qb.getTagSort(findFilter) + getPagination(findFilter)
idsResult, countResult, err := query.executeFind()
if err != nil {
return nil, 0, err
}
var tags []*models.Tag
for _, id := range idsResult {
tag, err := qb.Find(id)
if err != nil {
return nil, 0, err
}
tags = append(tags, tag)
}
return tags, countResult, nil
}
func (qb *tagQueryBuilder) getTagSort(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()
}
return getSort(sort, direction, "tags")
}
func (qb *tagQueryBuilder) queryTag(query string, args []interface{}) (*models.Tag, error) {
results, err := qb.queryTags(query, args)
if err != nil || len(results) < 1 {
return nil, err
}
return results[0], nil
}
func (qb *tagQueryBuilder) queryTags(query string, args []interface{}) ([]*models.Tag, error) {
var ret models.Tags
if err := qb.query(query, args, &ret); err != nil {
return nil, err
}
return []*models.Tag(ret), nil
}
func (qb *tagQueryBuilder) imageRepository() *imageRepository {
return &imageRepository{
repository: repository{
tx: qb.tx,
tableName: "tags_image",
idColumn: tagIDColumn,
},
imageColumn: "image",
}
}
func (qb *tagQueryBuilder) GetImage(tagID int) ([]byte, error) {
return qb.imageRepository().get(tagID)
}
func (qb *tagQueryBuilder) HasImage(tagID int) (bool, error) {
return qb.imageRepository().exists(tagID)
}
func (qb *tagQueryBuilder) UpdateImage(tagID int, image []byte) error {
return qb.imageRepository().replace(tagID, image)
}
func (qb *tagQueryBuilder) DestroyImage(tagID int) error {
return qb.imageRepository().destroy([]int{tagID})
}