traxxx-web/src/web/scenes.js

212 lines
5.2 KiB
JavaScript

import { stringify } from '@brillout/json-serializer/stringify'; /* eslint-disable-line import/extensions */
import { fetchScenes, fetchScenesById } from '../scenes.js';
import { parseActorIdentifier } from '../query.js';
import { getIdsBySlug } from '../cache.js';
import slugify from '../../utils/slugify.js';
import promiseProps from '../../utils/promise-props.js';
export async function curateScenesQuery(query) {
const splitYears = query.years?.split(',') || [];
const splitTags = query.tags?.split(',') || [];
const splitActors = query.actors?.split(',') || [];
const splitEntities = query.e?.split(',') || [];
const mainEntity = splitEntities.find((entity) => entity.charAt(0) !== '!');
const {
tagIds,
notTagIds,
entityId,
notEntityIds,
} = await promiseProps({
tagIds: getIdsBySlug([query.tagSlug, ...splitTags.filter((tag) => tag.charAt(0) !== '!')], 'tags'),
notTagIds: getIdsBySlug([...(query.tagFilter || []), ...(splitTags.filter((tag) => tag.charAt(0) === '!').map((tag) => tag.slice(1)) || [])].map((tag) => slugify(tag)), 'tags'),
entityId: mainEntity ? getIdsBySlug([mainEntity], 'entities').then(([id]) => id) : query.entityId,
notEntityIds: getIdsBySlug(splitEntities.filter((entity) => entity.charAt(0) === '!').map((entity) => entity.slice(1)), 'entities'),
});
return {
scope: query.scope || 'latest',
query: query.q,
years: splitYears.map((year) => Number(year)).filter(Boolean) || [],
actorIds: [query.actorId, ...splitActors.filter((actor) => actor.charAt(0) !== '!').map((identifier) => parseActorIdentifier(identifier)?.id)].filter(Boolean),
notActorIds: splitActors.filter((actor) => actor.charAt(0) === '!').map((identifier) => parseActorIdentifier(identifier.slice(1))?.id).filter(Boolean),
tagIds,
notTagIds: notTagIds.filter((tagId) => !tagIds.includes(tagId)), // included tags get priority over excluded tags
entityId,
notEntityIds,
movieId: Number(query.movieId) || null,
serieId: Number(query.serieId) || null,
stashId: Number(query.stashId) || null,
isShowcased: typeof query.isShowcased === 'boolean' ? query.isShowcased : null,
};
}
export async function fetchScenesApi(req, res) {
const {
scenes,
aggYears,
aggActors,
aggTags,
aggChannels,
limit,
total,
} = await fetchScenes(await curateScenesQuery({
...req.query,
tagFilter: req.tagFilter,
}), {
page: Number(req.query.page) || 1,
limit: Number(req.query.limit) || 30,
}, req.user);
res.send(stringify({
scenes,
aggYears,
aggActors,
aggTags,
aggChannels,
limit,
total,
}));
}
export const scenesSchema = `
extend type Query {
scenes(
query: String
scope: String
entities: [String!]
actorIds: [String!]
tags: [String!]
limit: Int! = 30
page: Int! = 1
): ReleasesResult
scene(
id: Int!
): Release
scenesById(
ids: [Int!]!
): [Release]
}
type ReleasesAggregate {
actors: [Actor!]
}
type ReleasesResult {
nodes: [Release!]!
total: Int
aggregates: ReleasesAggregate
}
type Release {
id: Int!
title: String
effectiveDate: Date
date: Date
duration: Int
description: String
createdAt: Date
shootId: Int
channel: Entity
network: Entity
actors: [Actor!]!
tags: [Tag!]!
poster: Media
trailer: Media
photos: [Media!]!
covers: [Media!]!
movies: [Release!]!
}
type Tag {
id: Int!
name: String
slug: String
priority: Int
}
type Media {
id: String!
path: String
thumbnail: String
lazy: String
mime: String
hash: String
isS3: Boolean
width: Int
height: Int
size: Int
createdAt: Int
}
`;
export async function fetchScenesGraphql(query, req) {
const mainEntity = query.entities?.find((entity) => entity.charAt(0) !== '!');
const {
tagIds,
notTagIds,
entityId,
notEntityIds,
} = await promiseProps({
tagIds: getIdsBySlug(query.tags?.filter((tag) => tag.charAt(0) !== '!'), 'tags'),
notTagIds: getIdsBySlug(query.tags?.filter((tag) => tag.charAt(0) === '!').map((tag) => tag.slice(1)).map((tag) => slugify(tag)), 'tags'),
entityId: getIdsBySlug([mainEntity], 'entities').then(([id]) => id),
notEntityIds: getIdsBySlug(query.entities?.filter((entity) => entity.charAt(0) === '!').map((entity) => entity.slice(1)), 'entities'),
});
const {
scenes,
total,
/*
aggActors,
aggTags,
aggChannels,
*/
} = await fetchScenes({
query: query.query, // query query query query
tagIds,
notTagIds,
entityId,
notEntityIds,
actorIds: query.actorIds?.filter((actorId) => actorId.charAt(0) !== '!').map((actorId) => Number(actorId)),
notActorIds: query.actorIds?.filter((actorId) => actorId.charAt(0) === '!').map((actorId) => Number(actorId.slice(1))),
scope: query.query && !query.scope
? 'results'
: query.scope,
isShowcased: null,
}, {
page: query.page || 1,
limit: query.limit || 30,
aggregate: false,
}, req.user);
return {
nodes: scenes,
total,
/* restrict until deemed essential for 3rd party apps
aggregates: {
actors: aggActors,
tags: aggTags,
channels: aggChannels,
},
*/
};
}
export async function fetchScenesByIdGraphql(query, req) {
const scenes = await fetchScenesById([].concat(query.id, query.ids).filter(Boolean), {
reqUser: req.user,
includePartOf: true,
});
if (query.ids) {
return scenes;
}
return scenes[0];
}