feat: some shit

This commit is contained in:
2023-08-08 21:50:23 +02:00
parent 6123956f08
commit c7d0e97ac0
18 changed files with 482 additions and 73 deletions

View File

@ -9,6 +9,10 @@ export default function App({ Component }: AppProps) {
<link href="/prism-material-dark.css" rel="stylesheet" />
<style>
{`
:root {
--background: rgb(43, 41, 48);
--foreground: rgb(129, 129, 129);
}
/* work-sans-regular - latin */
@font-face {
font-display: swap; /* Check https://developer.mozilla.org/en-US/docs/Web/CSS/@font-face/font-display for other options. */

View File

@ -7,23 +7,27 @@ import { extractHashTags } from "@lib/string.ts";
export const handler: Handlers = {
async GET(req, _ctx) {
const url = new URL(req.url);
const query = url.searchParams.get("q");
let query = url.searchParams.get("q");
if (!query) {
throw new BadRequestError('Query parameter "q" is required.');
}
query = decodeURIComponent(query);
const query_by = url.searchParams.get("query_by") ||
"name,description,author,tags";
let filter_by = "";
const filter_by: string[] = [];
const type = url.searchParams.get("type");
if (type) {
filter_by = `type:=${type}`;
filter_by.push(`type:=${type}`);
}
const hashTags = extractHashTags(query);
if (hashTags?.length) {
//filter_by += `tags:=${}`
filter_by.push(`tags:[${hashTags.map((t) => `\`${t}\``).join(",")}]`);
for (const tag of hashTags) {
query = query.replaceAll(`#${tag}`, "");
}
}
const typesenseClient = await getTypeSenseClient();
@ -31,12 +35,16 @@ export const handler: Handlers = {
throw new Error("Query not available");
}
console.log({ query, query_by, filter_by: filter_by.join(" && ") });
// Perform the Typesense search
const searchResults = await typesenseClient.collections("resources")
.documents().search({
q: query,
query_by,
filter_by,
facet_by: "rating,author,tags",
max_facet_values: 10,
filter_by: filter_by.join(" && "),
per_page: 50,
});

View File

@ -0,0 +1,68 @@
import { Handlers } from "$fresh/server.ts";
import { json } from "@lib/helpers.ts";
import * as tmdb from "@lib/tmdb.ts";
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
import { safeFileName } from "@lib/string.ts";
import { createDocument } from "@lib/documents.ts";
import { AccessDeniedError } from "@lib/errors.ts";
import { createSeries, getSeries, Series } from "@lib/resource/series.ts";
export const handler: Handlers = {
async GET(_, ctx) {
const series = await getSeries(ctx.params.name);
return json(series);
},
async POST(_, ctx) {
const session = ctx.state.session;
if (!session) {
throw new AccessDeniedError();
}
const tmdbId = parseInt(ctx.params.name);
const seriesDetails = await tmdb.getSeries(tmdbId);
const seriesCredits = await tmdb.getSeriesCredits(tmdbId);
const releaseDate = seriesDetails.first_air_date;
const posterPath = seriesDetails.poster_path;
const director =
seriesCredits?.crew?.filter?.((person) => person.job === "Director")[0];
let finalPath = "";
const name = seriesDetails.name || seriesDetails.original_name ||
ctx.params.name;
if (posterPath) {
const poster = await tmdb.getMoviePoster(posterPath);
const extension = fileExtension(posterPath);
finalPath = `Media/series/images/${
safeFileName(name)
}_cover.${extension}`;
await createDocument(finalPath, poster);
}
const metadata = {} as Series["meta"];
if (releaseDate) {
metadata.date = new Date(releaseDate);
}
if (finalPath) {
metadata.image = finalPath;
}
if (director) {
metadata.author = director.name;
}
const series: Series = {
id: name,
name: name,
type: "series",
description: "",
tags: [],
meta: metadata,
};
await createSeries(series);
return json(series);
},
};

View File

@ -0,0 +1,116 @@
import { HandlerContext, Handlers } from "$fresh/server.ts";
import {
createDocument,
getDocument,
transformDocument,
} from "@lib/documents.ts";
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
import * as tmdb from "@lib/tmdb.ts";
import { parse, stringify } from "https://deno.land/std@0.194.0/yaml/mod.ts";
import { formatDate, safeFileName } from "@lib/string.ts";
import { json } from "@lib/helpers.ts";
import {
AccessDeniedError,
BadRequestError,
NotFoundError,
} from "@lib/errors.ts";
import { getSeries, Series } from "@lib/resource/series.ts";
async function updateSeriesMetadata(
name: string,
metadata: Partial<Series["meta"]>,
) {
const docId = `Media/series/${name}.md`;
console.log({ docId, metadata });
let currentDoc = await getDocument(docId);
if (!currentDoc) {
throw new NotFoundError();
}
if (!currentDoc.startsWith("---\n---\n")) {
currentDoc = `---\n---\n\n${currentDoc}`;
}
const newDoc = transformDocument(currentDoc, (root) => {
const frontmatterNode = root.children.find((c) => c.type === "yaml");
const frontmatter = frontmatterNode?.value as string;
const value = parse(frontmatter) as Series["meta"];
const newValue = {
...metadata,
date: formatDate(metadata.date),
...value,
};
frontmatterNode.value = stringify(newValue);
return root;
});
console.log(newDoc);
return createDocument(docId, newDoc);
}
const POST = async (
req: Request,
ctx: HandlerContext,
): Promise<Response> => {
const session = ctx.state.session;
if (!session) {
throw new AccessDeniedError();
}
const body = await req.json();
const name = ctx.params.name;
const { tmdbId } = body;
if (!name || !tmdbId) {
throw new BadRequestError();
}
const series = await getSeries(ctx.params.name);
if (!series) {
throw new NotFoundError();
}
const seriesDetails = await tmdb.getSeries(tmdbId);
const seriesCredits = !series.meta.author &&
await tmdb.getSeriesCredits(tmdbId);
const releaseDate = seriesDetails.first_air_date;
const posterPath = seriesDetails.poster_path;
const director = seriesCredits &&
seriesCredits.crew?.filter?.((person) => person.job === "Director")[0];
let finalPath = "";
if (posterPath && !series.meta.image) {
const poster = await tmdb.getMoviePoster(posterPath);
const extension = fileExtension(posterPath);
finalPath = `Media/series/images/${safeFileName(name)}_cover.${extension}`;
await createDocument(finalPath, poster);
}
const metadata = {} as Series["meta"];
if (releaseDate) {
metadata.date = new Date(releaseDate);
}
if (finalPath) {
metadata.image = finalPath;
}
if (director && director.name) {
metadata.author = director.name;
}
await updateSeriesMetadata(name, metadata);
return json(seriesCredits);
};
export const handler: Handlers = {
POST,
};

View File

@ -6,6 +6,7 @@ import { getAllSeries, Series } from "@lib/resource/series.ts";
import { Card } from "@components/Card.tsx";
import { RedirectSearchHandler } from "@islands/Search.tsx";
import { KMenu } from "@islands/KMenu.tsx";
import { SeriesCard } from "@components/MovieCard.tsx";
export const handler: Handlers<Series[] | null> = {
async GET(_, ctx) {
@ -32,13 +33,7 @@ export default function Greet(props: PageProps<Series[] | null>) {
</header>
<Grid>
{props.data?.map((doc) => {
return (
<Card
image={doc?.meta?.image || "/placeholder.svg"}
link={`/series/${doc.id}`}
title={doc.name}
/>
);
return <SeriesCard series={doc} />;
})}
</Grid>
</MainLayout>