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];
}