259 lines
6.5 KiB
JavaScript
259 lines
6.5 KiB
JavaScript
import Router from 'express-promise-router';
|
|
import { stringify } from '@brillout/json-serializer/stringify'; /* eslint-disable-line import/extensions */
|
|
|
|
import {
|
|
fetchScenes,
|
|
fetchScenesById,
|
|
fetchSceneRevisions,
|
|
createSceneRevision,
|
|
reviewSceneRevision,
|
|
} from '../scenes.js';
|
|
|
|
import { parseActorIdentifier } from '../query.js';
|
|
import { getIdsBySlug } from '../cache.js';
|
|
import slugify from '../../utils/slugify.js';
|
|
import { HttpError } from '../errors.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,
|
|
};
|
|
}
|
|
|
|
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: String
|
|
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,
|
|
},
|
|
*/
|
|
};
|
|
}
|
|
|
|
async function fetchSceneApi(req, res) {
|
|
const [scene] = await fetchScenesById([Number(req.params.sceneId)], { reqUser: req.user });
|
|
|
|
if (!scene) {
|
|
throw new HttpError(`No scene with ID ${req.params.sceneId} found`, 404);
|
|
}
|
|
|
|
res.send(scene);
|
|
}
|
|
|
|
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];
|
|
}
|
|
|
|
async function fetchSceneRevisionsApi(req, res) {
|
|
const revisions = await fetchSceneRevisions(Number(req.params.revisionId) || null, req.query, req.user);
|
|
|
|
res.send(revisions);
|
|
}
|
|
|
|
async function createSceneRevisionApi(req, res) {
|
|
await createSceneRevision(Number(req.body.sceneId), req.body, req.user);
|
|
|
|
res.status(204).send();
|
|
}
|
|
|
|
async function reviewSceneRevisionApi(req, res) {
|
|
await reviewSceneRevision(Number(req.params.revisionId), req.body.isApproved, req.body, req.user);
|
|
|
|
res.status(204).send();
|
|
}
|
|
|
|
export const scenesRouter = Router();
|
|
|
|
scenesRouter.get('/api/scenes', fetchScenesApi);
|
|
scenesRouter.get('/api/scenes/:sceneId', fetchSceneApi);
|
|
|
|
scenesRouter.get('/api/revisions/scenes', fetchSceneRevisionsApi);
|
|
scenesRouter.get('/api/revisions/scenes/:revisionId', fetchSceneRevisionsApi);
|
|
scenesRouter.post('/api/revisions/scenes', createSceneRevisionApi);
|
|
scenesRouter.post('/api/revisions/scenes/:revisionId/reviews', reviewSceneRevisionApi);
|