Compare commits

...

32 Commits

Author SHA1 Message Date
Max Richter
47d32d68c7 fix: lazily import sharp to fix commonjs error 2026-01-10 20:13:15 +01:00
Max Richter
c74a19b527 fix(ci): make dockerfile work 2026-01-10 19:49:07 +01:00
Max Richter
8d712322c0 fix: make it work with new vite 2026-01-10 19:28:09 +01:00
Max Richter
694feb083d fix: remove all linter errors 2026-01-10 15:06:43 +01:00
Max Richter
e55f787a29 feat: trying to add hashes to scripts 2026-01-10 13:03:29 +01:00
e65938ecc2 fix: round instead of floor in IngredientList 2026-01-08 16:43:55 +01:00
Max Richter
7dda2dd60d feat: add kmenu icon for mobile 2025-11-12 16:12:57 +01:00
Max Richter
7ad08daf80 fix: make recipe crawling work 2025-11-12 15:41:30 +01:00
Max Richter
92126882b6 feat: correctly size search result items 2025-11-12 13:35:39 +01:00
Max Richter
655fc648e6 feat: fallback to unsplash cover when article contains no image 2025-11-09 23:52:53 +01:00
Max Richter
6c6b69a46a fix: make search work 2025-11-07 18:58:23 +01:00
Max Richter
97c5b7f93c fix: better position covers 2025-11-07 18:26:31 +01:00
Max Richter
bf3483019c fix: make unseen search work 2025-11-07 18:18:39 +01:00
Max Richter
5502c17c28 fix: correctly display noise gradient 2025-11-07 17:54:56 +01:00
Max Richter
581f1c1926 fix: make search usable again 2025-11-05 00:42:53 +01:00
Max Richter
7664abe089 fix: actually write image 2025-11-04 16:04:51 +01:00
Max Richter
bed7d1a11b feat: better log article creations 2025-11-04 13:57:32 +01:00
Max Richter
56a104c8b9 feat: use better names for md files 2025-11-04 13:26:49 +01:00
Max Richter
3103ed19fb fix: correctly fetch marka pi in background 2025-11-04 12:58:26 +01:00
Max Richter
fea9b69d4d feat: cache marka api responses 2025-11-04 12:09:17 +01:00
Max Richter
bb4e895770 fix: display correct name for series 2025-11-03 01:06:27 +01:00
Max Richter
28e9de4dc8 fix: make sure to only decode thumbhash once 2025-11-03 00:57:44 +01:00
Max Richter
ebb897dca4 fix: correctly embed styles.css 2025-11-03 00:46:49 +01:00
Max Richter
696082250d fix: soo many lint errors 2025-11-03 00:03:27 +01:00
Max Richter
c13420c3ab chore: deno fmt 2025-11-02 21:58:02 +01:00
Max Richter
21124dfe00 fix: remove unused imports 2025-11-02 21:56:26 +01:00
Max Richter
928782c453 fix: accessing nonexistant variable 2025-11-02 21:14:41 +01:00
Max Richter
098da12ac4 fix: make recipe ingredinets interactive 2025-11-02 20:01:01 +01:00
Max Richter
d4a7763b15 fix: accessing nonexistant variable 2025-11-02 19:19:57 +01:00
Max Richter
21841b4dc4 fix: use correct url 2025-11-02 19:03:34 +01:00
Max Richter
e6b90cb785 feat: refactor whole bunch of stuff 2025-11-02 19:03:11 +01:00
Max Richter
81ebc8f5e0 fix: display correct name on movie page 2025-10-31 20:15:00 +01:00
178 changed files with 6343 additions and 3138 deletions

2
.gitignore vendored
View File

@@ -1,4 +1,3 @@
# dotenv environment variable files
.env .env
.env.development.local .env.development.local
.env.test.local .env.test.local
@@ -10,3 +9,4 @@ data-dev/
_fresh/ _fresh/
node_modules/ node_modules/
mise.toml mise.toml

View File

@@ -1,10 +1,10 @@
FROM denoland/deno:2.5.4 AS build FROM denoland/deno:2.6.4 AS build
RUN apt-get update && apt-get install -y --no-install-recommends \ RUN apt-get update && apt-get install -y --no-install-recommends \
curl ffmpeg && \ curl ffmpeg && \
deno run -A npm:playwright install --with-deps firefox &&\ deno run -A npm:playwright install --with-deps firefox &&\
apt-get clean && \ apt-get clean && \
rm -rf /var/lib/apt/lists/* rm -rf /var/lib/apt/lists/*
WORKDIR /app WORKDIR /app
@@ -15,11 +15,11 @@ COPY . .
ENV DATA_DIR=/app/data ENV DATA_DIR=/app/data
RUN mkdir -p $DATA_DIR && \ RUN mkdir -p $DATA_DIR && \
deno install --allow-import --allow-ffi --allow-scripts=npm:sharp@0.33.5-rc.1 -e main.ts &&\ deno install --allow-import --allow-ffi -e main.ts &&\
sed -i -e 's/"deno"/"no-deno"/' node_modules/@libsql/client/package.json &&\ sed -i -e 's/"deno"/"no-deno"/' node_modules/@libsql/client/package.json &&\
deno task build deno install npm:@libsql/linux-x64-gnu &&\
deno task build
EXPOSE 8000 EXPOSE 8000
CMD ["run", "-A", "main.ts"] CMD ["task", "start"]

View File

@@ -5,18 +5,13 @@ Started" guide here: https://fresh.deno.dev/docs/getting-started
### Usage ### Usage
Make sure to install Deno: https://deno.land/manual/getting_started/installation Make sure to install Deno:
https://docs.deno.com/runtime/getting_started/installation
Then start the project: Then start the project in development mode:
``` ```
deno task start deno task dev
``` ```
This will watch the project directory and restart as necessary. This will watch the project directory and restart as necessary.
## FIX
```
sed -i -e 's/"deno"/"no-deno"/' node_modules/@libsql/client/package.json
```

1
assets/styles.css Normal file
View File

@@ -0,0 +1 @@
@import "tailwindcss";

2
client.ts Normal file
View File

@@ -0,0 +1,2 @@
// Import CSS files here for hot module reloading to work.
import "./assets/styles.css";

View File

@@ -1,12 +1,10 @@
import { JSX } from "preact"; import { ButtonHTMLAttributes } from "preact";
import { IS_BROWSER } from "$fresh/runtime.ts";
export function Button(props: JSX.HTMLAttributes<HTMLButtonElement>) { export function Button(props: ButtonHTMLAttributes<HTMLButtonElement>) {
return ( return (
<button <button
{...props} {...props}
disabled={!IS_BROWSER || props.disabled} class={`cursor-pointer px-2 py-1 ${props.class ? props.class : ""}`}
class={`px-2 py-1 ${props.class ? props.class : " "}`}
/> />
); );
} }

View File

@@ -1,9 +1,8 @@
import { isYoutubeLink } from "@lib/string.ts"; import { isYoutubeLink } from "@lib/string.ts";
import { IconBrandYoutube } from "@components/icons.tsx"; import { IconBrandYoutube } from "@components/icons.tsx";
import { GenericResource } from "@lib/types.ts";
import { SmallRating } from "@components/Rating.tsx"; import { SmallRating } from "@components/Rating.tsx";
import { Link } from "@islands/Link.tsx";
import { parseRating } from "@lib/helpers.ts"; import { parseRating } from "@lib/helpers.ts";
import { GenericResource, getNameOfResource } from "@lib/marka/schema.ts";
export function Card( export function Card(
{ {
@@ -11,20 +10,20 @@ export function Card(
rating, rating,
title, title,
image, image,
thumbnail, thumbhash,
backgroundColor, backgroundColor,
backgroundSize = 100, backgroundSize = 100,
}: { }: {
backgroundSize?: number; backgroundSize?: number;
backgroundColor?: string; backgroundColor?: string;
thumbnail?: string; thumbhash?: string;
link?: string; link?: string;
title?: string; title?: string;
image?: string; image?: string;
rating?: number; rating?: number;
}, },
) { ) {
const backgroundStyle: preact.JSX.CSSProperties = { const backgroundStyle: preact.CSSProperties = {
backgroundSize: "cover", backgroundSize: "cover",
backgroundColor: backgroundColor, backgroundColor: backgroundColor,
}; };
@@ -36,25 +35,23 @@ export function Card(
} }
return ( return (
<Link <a
href={link} href={link}
style={backgroundStyle} style={backgroundStyle}
data-thumb={thumbnail} data-thumb={thumbhash}
class="text-white rounded-3xl shadow-md relative class="text-white rounded-3xl shadow-md relative
lg:w-56 lg:h-56 lg:w-56 lg:h-56
sm:w-48 sm:h-48 sm:w-48 sm:h-48
w-[37vw] h-[37vw]" w-[37vw] h-[37vw]"
> >
{true && ( <span class="absolute top-0 left-0 overflow-hidden rounded-3xl w-full h-full">
<span class="absolute top-0 left-0 overflow-hidden rounded-3xl w-full h-full"> <img
<img class="w-full h-full object-cover"
class="w-full h-full object-cover" data-thumb-img
data-thumb-img loading="lazy"
loading="lazy" src={image || "/placeholder.svg"}
src={image || "/placeholder.svg"} />
/> </span>
</span>
)}
<div <div
class="p-4 flex flex-col justify-between relative z-10" class="p-4 flex flex-col justify-between relative z-10"
style={{ style={{
@@ -90,26 +87,29 @@ export function Card(
)} )}
</div> </div>
<div class="absolute inset-x-0 bottom-0 h-3/4" /> <div class="absolute inset-x-0 bottom-0 h-3/4" />
</Link> </a>
); );
} }
export function ResourceCard( export function ResourceCard(
{ res, sublink = "movies" }: { sublink?: string; res: GenericResource }, { res, sublink = "movies" }: { sublink?: string; res: GenericResource },
) { ) {
const img = res?.content?.image || res?.content?.cover; const img = res?.image?.url;
const imageUrl = img const imageUrl = img
? `/api/images?image=${img}&width=200&height=200` ? `/api/images?image=${img}&width=200&height=200`
: "/placeholder.svg"; : "/placeholder.svg";
const rating = res.content.reviewRating?.ratingValue
? parseRating(res.content.reviewRating.ratingValue)
: undefined;
return ( return (
<Card <Card
title={res.content?.name || res.content?.itemReviewed?.name || res.content?.headline || title={getNameOfResource(res)}
res?.name} backgroundColor={res.image?.average}
backgroundColor={res.meta?.average} thumbhash={res.image?.thumbhash}
rating={parseRating(res.content?.reviewRating?.ratingValue)} rating={rating}
thumbnail={res.cover}
image={imageUrl} image={imageUrl}
link={`/${sublink}/${res.name.replace(/\.md$/g, "")}`} link={`/${sublink}/${res.name.replace(/\.md$/g, "")}`}
/> />

View File

@@ -1,67 +1,5 @@
import { Signal, useSignal } from "@preact/signals"; import { Signal, useSignal } from "@preact/signals";
import { useId, useState } from "preact/hooks"; import { useId } from "preact/hooks";
interface CheckboxProps {
label: string;
isChecked?: boolean;
onChange: (isChecked: boolean) => void;
}
const Checkbox2: preact.FunctionalComponent<CheckboxProps> = (
{ label, isChecked = false, onChange },
) => {
const [checked, setChecked] = useState(isChecked);
const toggleCheckbox = () => {
const newChecked = !checked;
setChecked(newChecked);
onChange(newChecked);
};
return (
<div
class="flex items-center rounded-xl p-1 pl-4"
style={{ background: "var(--background)", color: "var(--foreground)" }}
>
<span>
{label}
</span>
<label
class="relative flex cursor-pointer items-center rounded-full p-3"
for="checkbox"
data-ripple-dark="true"
>
<input
type="checkbox"
class="before:content[''] peer relative h-5 w-5 cursor-pointer appearance-none rounded-md border border-blue-gray-200 transition-all before:absolute before:top-2/4 before:left-2/4 before:block before:h-12 before:w-12 before:-translate-y-2/4 before:-translate-x-2/4 before:rounded-full before:bg-blue-gray-500 before:opacity-0 before:transition-opacity hover:before:opacity-10"
id="checkbox"
checked
/>
<div
class={`pointer-events-none absolute top-2/4 left-2/4 -translate-y-2/4 -translate-x-2/4 text-white opacity-${
checked ? 100 : 0
} transition-opacity`}
>
<svg
xmlns="http://www.w3.org/2000/svg"
class="h-3.5 w-3.5"
viewBox="0 0 20 20"
fill="white"
stroke="currentColor"
stroke-width="1"
>
<path
fill-rule="evenodd"
d="M16.707 5.293a1 1 0 010 1.414l-8 8a1 1 0 01-1.414 0l-4-4a1 1 0 011.414-1.414L8 12.586l7.293-7.293a1 1 0 011.414 0z"
clip-rule="evenodd"
>
</path>
</svg>
</div>
</label>
</div>
);
};
const Checkbox = ( const Checkbox = (
{ label, checked = useSignal(false) }: { { label, checked = useSignal(false) }: {

View File

@@ -1,4 +1,4 @@
import { asset } from "$fresh/runtime.ts"; import { asset } from "fresh/runtime";
export const Emoji = (props: { class?: string; name: string }) => { export const Emoji = (props: { class?: string; name: string }) => {
return props.name return props.name
@@ -10,5 +10,5 @@ export const Emoji = (props: { class?: string; name: string }) => {
/> />
) )
: <span>{props.name}</span> : <span>{props.name}</span>
: <></>; : null;
}; };

View File

@@ -1,5 +1,4 @@
import { asset } from "$fresh/runtime.ts"; import { asset } from "fresh/runtime";
import * as CSS from "https://esm.sh/csstype@3.1.2";
interface ResponsiveAttributes { interface ResponsiveAttributes {
srcset: string; srcset: string;
@@ -34,11 +33,11 @@ const Image = (
class: string; class: string;
src: string; src: string;
alt?: string; alt?: string;
thumbnail?: string; thumbhash?: string;
fill?: boolean; fill?: boolean;
width?: number | string; width?: number | string;
height?: number | string; height?: number | string;
style?: CSS.HtmlAttributes; style?: preact.CSSProperties;
}, },
) => { ) => {
const responsiveAttributes = generateResponsiveAttributes( const responsiveAttributes = generateResponsiveAttributes(
@@ -47,6 +46,11 @@ const Image = (
"/api/images", "/api/images",
); );
const hasDimensions = typeof props.width === "number" &&
typeof props.height === "number";
const sizes = hasDimensions ? "" : responsiveAttributes.sizes;
const srcset = hasDimensions ? "" : responsiveAttributes.srcset;
return ( return (
<span <span
style={{ style={{
@@ -55,16 +59,15 @@ const Image = (
height: props.fill ? "100%" : "", height: props.fill ? "100%" : "",
zIndex: props.fill ? -1 : "", zIndex: props.fill ? -1 : "",
}} }}
data-thumb={props.thumbnail} data-thumb={props.thumbhash}
> >
<img <img
data-thumb={props.thumbnail}
data-thumb-img data-thumb-img
loading="lazy" loading="lazy"
alt={props.alt} alt={props.alt}
style={props.style} style={props.style}
srcset={responsiveAttributes.srcset} sizes={sizes}
sizes={responsiveAttributes.sizes} srcset={srcset}
src={`/api/images?image=${asset(props.src)}${ src={`/api/images?image=${asset(props.src)}${
props.width ? `&width=${props.width}` : "" props.width ? `&width=${props.width}` : ""
}${props.height ? `&height=${props.height}` : ""}`} }${props.height ? `&height=${props.height}` : ""}`}

View File

@@ -1,5 +1,6 @@
import { GenericResource } from "@lib/types.ts"; import { Head } from "fresh/runtime";
import { Head } from "$fresh/runtime.ts"; import { GenericResource, getNameOfResource } from "@lib/marka/schema.ts";
import { formatDate } from "@lib/string.ts";
function generateJsonLd(resource: GenericResource): string { function generateJsonLd(resource: GenericResource): string {
const imageUrl = resource.content?.image const imageUrl = resource.content?.image
@@ -8,34 +9,34 @@ function generateJsonLd(resource: GenericResource): string {
const baseSchema: Record<string, unknown> = { const baseSchema: Record<string, unknown> = {
"@context": "https://schema.org", "@context": "https://schema.org",
"@type": resource.content?._type, // Converts type to PascalCase "@type": resource.content?._type,
name: resource.name, name: resource.name,
description: resource.content || resource.meta?.average || "", description: resource.content || "",
keywords: resource.tags?.join(", ") || "", keywords: resource.content.keywords?.join(", ") || "",
image: imageUrl, image: imageUrl,
}; };
if (resource.meta?.author) { if (resource.content?.author) {
baseSchema.author = { baseSchema.author = {
"@type": "Person", "@type": "Person",
name: resource.meta.author, name: resource.content.author,
}; };
} }
if (resource.meta?.date) { if (resource.content?.datePublished) {
try { try {
baseSchema.datePublished = new Date(resource.meta.date).toISOString(); baseSchema.datePublished = formatDate(
resource.content.datePublished,
);
} catch (_) { } catch (_) {
// Ignore invalid date // Ignore invalid date
} }
} }
if (resource.meta?.rating) { if (resource.content?.reviewRating) {
baseSchema.aggregateRating = { baseSchema.reviewRating = {
"@type": "AggregateRating", "@type": "Rating",
ratingValue: resource.meta.rating, ...resource.content.reviewRating,
ratingCount: 1,
bestRating: 5, // Assuming a scale of 1 to 10
}; };
} }
@@ -51,7 +52,7 @@ export function MetaTags({ resource }: { resource: GenericResource }) {
return ( return (
<> <>
<Head> <Head>
<meta property="og:title" content={resource.content?.name} /> <meta property="og:title" content={getNameOfResource(resource)} />
<meta property="og:type" content={resource.content?._type} /> <meta property="og:type" content={resource.content?._type} />
<meta <meta
property="og:image" property="og:image"
@@ -59,6 +60,7 @@ export function MetaTags({ resource }: { resource: GenericResource }) {
/> />
<script <script
type="application/ld+json" type="application/ld+json"
// deno-lint-ignore react-no-danger
dangerouslySetInnerHTML={{ __html: jsonLd }} dangerouslySetInnerHTML={{ __html: jsonLd }}
/> />
</Head> </Head>

View File

@@ -6,22 +6,22 @@ import { IconArrowNarrowLeft } from "@components/icons.tsx";
import { IconEdit } from "@components/icons.tsx"; import { IconEdit } from "@components/icons.tsx";
import { useContext } from "preact/hooks"; import { useContext } from "preact/hooks";
const HeroContext = createContext<{ image?: string; thumbnail?: string }>({ const HeroContext = createContext<{ image?: string; thumbhash?: string }>({
image: undefined, image: undefined,
thumbnail: undefined, thumbhash: undefined,
}); });
function Wrapper( function Wrapper(
{ children, image, thumbnail }: { { children, image, thumbhash }: {
children: ComponentChildren; children: ComponentChildren;
image?: string; image?: string;
thumbnail?: string; thumbhash?: string;
}, },
) { ) {
return ( return (
<div <div
class={`flex justify-between flex-col relative w-full ${ class={`flex justify-between flex-col relative w-full ${
image ? "min-h-[400px]" : "min-h-[200px]" image ? "min-h-100" : "min-h-50"
} rounded-3xl overflow-hidden`} } rounded-3xl overflow-hidden`}
> >
<HeroContext.Provider value={{ image }}> <HeroContext.Provider value={{ image }}>
@@ -30,7 +30,7 @@ function Wrapper(
<Image <Image
fill fill
src={image} src={image}
thumbnail={thumbnail} thumbhash={thumbhash}
alt="Recipe Banner" alt="Recipe Banner"
// style={{ objectPosition: "0% 25%" }} // style={{ objectPosition: "0% 25%" }}
class="absolute object-cover w-full h-full -z-10" class="absolute object-cover w-full h-full -z-10"
@@ -62,7 +62,7 @@ function Title(
> >
{children} {children}
{link && {link &&
<IconExternalLink />} <IconExternalLink class="h-6 w-6" />}
</h2> </h2>
</OuterTag> </OuterTag>
); );
@@ -104,23 +104,24 @@ function Header({ children }: { children: ComponentChildren }) {
function Subline( function Subline(
{ entries, children }: { { entries, children }: {
children?: ComponentChildren; children?: ComponentChildren;
entries: (string | { href: string; title: string })[]; entries: (string | undefined | { href: string; title: string })[];
}, },
) { ) {
const ctx = useContext(HeroContext); const ctx = useContext(HeroContext);
return ( return (
<div <div
class={`relative flex items-center z-10 flex gap-5 font-sm text-light mt-3`} class={`relative items-center z-10 flex gap-5 font-sm text-light mt-3`}
style={{ color: ctx.image ? "#1F1F1F" : "white" }} style={{ color: ctx.image ? "#1F1F1F" : "white" }}
> >
{children} {children}
{entries.filter((s) => {entries.filter((s) =>
s && (typeof s === "string" ? s?.length > 1 : true) s && (typeof s === "string" ? s?.length > 1 : true)
).map((s) => { ).map((s) => {
if (!s) return;
if (typeof s === "string") { if (typeof s === "string") {
return <span>{s}</span>; return <span key={s}>{s}</span>;
} else { } else {
return <a href={s.href}>{s.title}</a>; return <a key={s.href} href={s.href}>{s.title}</a>;
} }
})} })}
</div> </div>

View File

@@ -1,5 +1,5 @@
import { IconStar, IconStarFilled } from "@components/icons.tsx"; import { IconStar, IconStarFilled } from "@components/icons.tsx";
import { useSignal } from "@preact/signals"; import { Signal, useSignal } from "@preact/signals";
import { useState } from "preact/hooks"; import { useState } from "preact/hooks";
export const SmallRating = ( export const SmallRating = (
@@ -24,27 +24,30 @@ export const SmallRating = (
}; };
export const Rating = ( export const Rating = (
props: { max?: number; rating: number }, { max, rating = useSignal(0) }: {
max?: number;
rating: Signal<number | undefined>;
},
) => { ) => {
const [rating, setRating] = useState(props.rating);
const [hover, setHover] = useState(0); const [hover, setHover] = useState(0);
const max = useSignal(props.max || 5);
const ratingValue = rating.value || 0;
return ( return (
<div <div
class="flex items-center gap-2 px-5 rounded-2xl bg-gray-200 z-10" class="flex items-center gap-2 px-5 rounded-2xl bg-gray-200 z-10 h-full"
style={{ color: "var(--foreground)", background: "var(--background)" }} style={{ color: "var(--foreground)", background: "var(--background)" }}
> >
{Array.from({ length: max.value }).map((_, i) => { {Array.from({ length: max || 5 }).map((_, i) => {
return ( return (
<span <span
class={`cursor-pointer opacity-${ class={`cursor-pointer opacity-${
(i + 1) <= rating ? 100 : (i + 1) <= hover ? 20 : 100 (i + 1) <= ratingValue ? 100 : (i + 1) <= hover ? 20 : 100
}`} }`}
onMouseOver={() => setHover(i + 1)} onMouseOver={() => setHover(i + 1)}
onClick={() => setRating(i + 1)} onClick={() => (rating.value = i + 1)}
> >
{(i + 1) <= rating || (i + 1) <= hover {(i + 1) <= ratingValue || (i + 1) <= hover
? <IconStarFilled class="w-4 h-4" /> ? <IconStarFilled class="w-4 h-4" />
: <IconStar class="w-4 h-4" />} : <IconStar class="w-4 h-4" />}
</span> </span>

View File

@@ -10,9 +10,9 @@ export const Star = (
> >
{Array.from({ length: max }).map((_, i) => { {Array.from({ length: max }).map((_, i) => {
if ((i + 1) <= rating) { if ((i + 1) <= rating) {
return <IconStarFilled class="w-4 h-4" />; return <IconStarFilled key={i} class="w-4 h-4" />;
} }
return <IconStar class="w-4 h-4" />; return <IconStar key={i} class="w-4 h-4" />;
})} })}
</div> </div>
); );

View File

@@ -1,19 +1,23 @@
export { default as IconStar } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/star.tsx"; export {
export { default as IconStarFilled } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/star-filled.tsx"; TbAlertCircle as IconAlertCircle,
export { default as IconExternalLink } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/external-link.tsx"; TbArrowLeft as IconArrowLeft,
export { default as IconArrowNarrowLeft } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/arrow-narrow-left.tsx"; TbArrowNarrowLeft as IconArrowNarrowLeft,
export { default as IconEdit } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/edit.tsx"; TbBrandYoutube as IconBrandYoutube,
export { default as IconArrowLeft } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/arrow-left.tsx"; TbCircleMinus as IconCircleMinus,
export { default as IconError404 } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/error-404.tsx"; TbCirclePlus as IconCirclePlus,
export { default as IconSquareRoundedPlus } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/square-rounded-plus.tsx"; TbEdit as IconEdit,
export { default as IconReportSearch } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/report-search.tsx"; TbError404 as IconError404,
export { default as IconRefresh } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/refresh.tsx"; TbExternalLink as IconExternalLink,
export { default as IconCirclePlus } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/circle-plus.tsx"; TbGhost as IconGhost,
export { default as IconCircleMinus } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/circle-minus.tsx"; TbLoader2 as IconLoader2,
export { default as IconLoader2 } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/loader-2.tsx"; TbLogin as IconLogin,
export { default as IconLogin } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/login.tsx"; TbLogout as IconLogout,
export { default as IconLogout } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/logout.tsx"; TbMenu2 as IconMenu2,
export { default as IconSearch } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/search.tsx"; TbRefresh as IconRefresh,
export { default as IconGhost } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/ghost.tsx"; TbReportSearch as IconReportSearch,
export { default as IconBrandYoutube } from "https://deno.land/x/tabler_icons_tsx@0.0.3/tsx/brand-youtube.tsx"; TbSearch as IconSearch,
export { default as IconWand } from "https://deno.land/x/tabler_icons_tsx@0.0.5/tsx/wand.tsx"; TbSquareRoundedPlus as IconSquareRoundedPlus,
TbStar as IconStar,
TbStarFilled as IconStarFilled,
TbWand as IconWand,
} from "@preact-icons/tb";

View File

@@ -1,6 +1,6 @@
import { ComponentChildren } from "preact"; import { ComponentChildren } from "preact";
import Search from "@islands/Search.tsx"; import Search from "@islands/Search.tsx";
import { GenericResource } from "@lib/types.ts"; import { GenericResource } from "@lib/marka/schema.ts";
export type Props = { export type Props = {
children: ComponentChildren; children: ComponentChildren;
@@ -12,17 +12,25 @@ export type Props = {
searchResults?: GenericResource[]; searchResults?: GenericResource[];
}; };
function getQFromUrl(u: string | URL): string | null {
try {
const _u = typeof u === "string" ? new URL(u) : u;
return _u?.searchParams.get("q");
} catch (_e) {
return null;
}
}
export const MainLayout = ( export const MainLayout = (
{ children, url, context, searchResults }: Props, { children, url, context, searchResults }: Props,
) => { ) => {
const _url = typeof url === "string" ? new URL(url) : url; const q = getQFromUrl(url);
const hasSearch = _url?.search?.includes("q=");
if (hasSearch) { if (typeof q === "string") {
return ( return (
<Search <Search
q={_url.searchParams.get("q")}
{...context} {...context}
q={q}
results={searchResults} results={searchResults}
/> />
); );

View File

@@ -4,10 +4,10 @@ services:
context: . context: .
dockerfile: Dockerfile dockerfile: Dockerfile
volumes: volumes:
- .:/app # Mount the local directory to /app in the container - .:/app
working_dir: /app # Set the working directory inside the container to /app working_dir: /app
command: run --env-file -A --watch=static/,routes/ dev.ts # Custom start command command: deno task dev --host 0.0.0.0
ports: ports:
- "8000:8000" # Expose the container port - "8000:8000"
environment: environment:
- DATA_DIR=/app/data # Set the environment variable inside the container - DATA_DIR=/app/data

117
deno.json
View File

@@ -1,52 +1,99 @@
{ {
"lock": false, "nodeModulesDir": "manual",
"nodeModulesDir": "auto",
"unstable": ["cron"],
"tasks": { "tasks": {
"check": "deno fmt --check && deno lint && deno check **/*.ts && deno check **/*.tsx", "check": "deno fmt --check . && deno lint . && deno check",
"dev": "deno run --env-file -A --watch=static/,routes/ dev.ts", "dev": "vite",
"start": "deno run --env-file -A main.ts",
"db": "deno run --env-file -A npm:drizzle-kit", "db": "deno run --env-file -A npm:drizzle-kit",
"build": "deno run -A dev.ts build", "build": "vite build",
"preview": "deno run -A main.ts", "start": "deno serve -A _fresh/server.js",
"update": "deno run -A -r https://fresh.deno.dev/update ." "update": "deno run -A -r jsr:@fresh/update ."
}, },
"lint": { "rules": { "tags": ["fresh", "recommended"] } }, "lint": {
"rules": {
"tags": [
"fresh",
"recommended"
]
}
},
"exclude": [
"**/_fresh/*"
],
"imports": { "imports": {
"$fresh/": "https://deno.land/x/fresh@1.7.3/", "@cmd-johnson/oauth2-client": "jsr:@cmd-johnson/oauth2-client@^2.0.0",
"@components": "./components", "@components": "./components",
"@components/": "./components/", "@components/": "./components/",
"@denosaurs/emoji": "jsr:@denosaurs/emoji@^0.3.1", "@deno/gfm": "jsr:@deno/gfm@^0.11.0",
"@islands": "./islands", "@islands": "./islands",
"@islands/": "./islands/", "@islands/": "./islands/",
"@lib": "./lib", "@lib": "./lib",
"@lib/": "./lib/", "@lib/": "./lib/",
"@libsql/client": "npm:@libsql/client@^0.14.0", "@/": "./",
"@preact/signals": "https://esm.sh/*@preact/signals@1.2.2", "@libsql/client": "npm:@libsql/client@^0.17.0",
"@preact/signals-core": "https://esm.sh/*@preact/signals-core@1.5.1", "@libsql/linux-x64-gnu": "npm:@libsql/linux-x64-gnu@^0.5.22",
"@std/http": "jsr:@std/http@^1.0.12", "@openai/openai": "jsr:@openai/openai@^6.16.0",
"@std/yaml": "jsr:@std/yaml@^1.0.5", "@preact-icons/tb": "jsr:@preact-icons/tb@^1.0.14",
"@std/http": "jsr:@std/http@^1.0.23",
"@std/media-types": "jsr:@std/media-types@^1.1.0",
"@zaubrik/djwt": "jsr:@zaubrik/djwt@^3.0.2",
"defuddle": "npm:defuddle@^0.6.6", "defuddle": "npm:defuddle@^0.6.6",
"drizzle-kit": "npm:drizzle-kit@^0.30.1", "drizzle-kit": "npm:drizzle-kit@^0.31.8",
"drizzle-orm": "npm:drizzle-orm@^0.38.3", "drizzle-orm": "npm:drizzle-orm@^0.45.1",
"fresh": "jsr:@fresh/core@^2.2.0",
"fuzzysort": "npm:fuzzysort@^3.1.0", "fuzzysort": "npm:fuzzysort@^3.1.0",
"jsdom": "npm:jsdom@^24.1.3", "gfm": "jsr:@deno/gfm@0.11.0",
"playwright": "npm:playwright@^1.49.1", "jsdom": "npm:jsdom@^27.4.0",
"moviedb-promise": "npm:moviedb-promise@^4.0.7",
"parse-ingredient": "npm:parse-ingredient@^1.3.1",
"playwright-extra": "npm:playwright-extra@^4.3.6", "playwright-extra": "npm:playwright-extra@^4.3.6",
"preact": "https://esm.sh/preact@10.22.0", "preact": "npm:preact@^10.27.2",
"preact-render-to-string": "https://esm.sh/*preact-render-to-string@6.2.2", "@preact/signals": "npm:@preact/signals@^2.5.0",
"preact/": "https://esm.sh/preact@10.22.0/", "@fresh/plugin-vite": "jsr:@fresh/plugin-vite@^1.0.8",
"gfm": "jsr:@deno/gfm",
"puppeteer-extra-plugin-stealth": "npm:puppeteer-extra-plugin-stealth@^2.11.2", "puppeteer-extra-plugin-stealth": "npm:puppeteer-extra-plugin-stealth@^2.11.2",
"tailwindcss": "npm:tailwindcss@^3.4.17", "sharp": "npm:sharp@^0.34.5",
"tailwindcss/": "npm:/tailwindcss@^3.4.17/", "thumbhash": "npm:thumbhash@^0.1.1",
"tailwindcss/plugin": "npm:/tailwindcss@^3.4.17/plugin.js", "turndown": "npm:turndown@^7.2.2",
"camelcase-css": "npm:camelcase-css", "vite": "npm:vite@^7.1.3",
"tsx": "npm:tsx@^4.19.2", "tailwindcss": "npm:tailwindcss@^4.1.10",
"yaml": "https://deno.land/std@0.197.0/yaml/mod.ts", "@tailwindcss/vite": "npm:@tailwindcss/vite@^4.1.12",
"zod": "npm:zod@^3.24.1", "zod": "npm:zod@^4.3.5"
"fs": "https://deno.land/std/fs/mod.ts"
}, },
"compilerOptions": { "jsx": "react-jsx", "jsxImportSource": "preact" }, "compilerOptions": {
"exclude": ["**/_fresh/*"] "lib": [
"dom",
"dom.asynciterable",
"dom.iterable",
"deno.ns"
],
"jsx": "precompile",
"jsxImportSource": "preact",
"jsxPrecompileSkipElements": [
"a",
"img",
"source",
"body",
"html",
"head",
"title",
"meta",
"script",
"link",
"style",
"base",
"noscript",
"template"
],
"types": [
"vite/client"
]
},
"allowScripts": {
"allow": [
"npm:sharp@0.34.5",
"npm:esbuild@0.27.2",
"npm:esbuild@0.18.20",
"npm:esbuild@0.25.12",
"npm:esbuild@0.25.7"
]
}
} }

2912
deno.lock generated Normal file

File diff suppressed because it is too large Load Diff

8
dev.ts
View File

@@ -1,8 +0,0 @@
#!/usr/bin/env -S deno run -A --watch=static/,routes/
import dev from "$fresh/dev.ts";
import config from "./fresh.config.ts";
await dev(import.meta.url, "./main.ts", config);
Deno.exit(0);

View File

@@ -1,20 +1,22 @@
CREATE TABLE `performance` ( CREATE TABLE `performance` (
`path` text NOT NULL, `path` text NOT NULL,
`search` text, `search` text,
`time` integer NOT NULL, `time` integer NOT NULL,
`created_at` integer DEFAULT (current_timestamp) `created_at` integer DEFAULT (CURRENT_TIMESTAMP)
); );
--> statement-breakpoint --> statement-breakpoint
CREATE TABLE `session` ( CREATE TABLE `session` (
`id` text PRIMARY KEY NOT NULL, `id` text PRIMARY KEY NOT NULL,
`created_at` integer DEFAULT (current_timestamp), `created_at` integer DEFAULT (CURRENT_TIMESTAMP),
`expires_at` integer NOT NULL, `expires_at` integer NOT NULL,
`user_id` text NOT NULL `user_id` text NOT NULL
); );
--> statement-breakpoint --> statement-breakpoint
CREATE TABLE `user` ( CREATE TABLE `user` (
`id` text PRIMARY KEY NOT NULL, `id` text PRIMARY KEY NOT NULL,
`created_at` integer DEFAULT (current_timestamp) NOT NULL, `created_at` integer DEFAULT (CURRENT_TIMESTAMP) NOT NULL,
`email` text NOT NULL, `email` text NOT NULL,
`name` text NOT NULL `name` text NOT NULL
); );

View File

@@ -1 +1,4 @@
ALTER TABLE `performance` ALTER COLUMN "created_at" TO "created_at" integer DEFAULT (STRFTIME('%s', 'now') * 1000); ALTER TABLE
`performance`
ALTER COLUMN
"created_at" TO "created_at" integer DEFAULT (STRFTIME('%s', 'now') * 1000);

View File

@@ -1,7 +1,7 @@
CREATE TABLE `image` ( CREATE TABLE `image` (
`created_at` integer DEFAULT (current_timestamp), `created_at` integer DEFAULT (CURRENT_TIMESTAMP),
`url` text NOT NULL, `url` text NOT NULL,
`average` text NOT NULL, `average` text NOT NULL,
`blurhash` text NOT NULL, `blurhash` text NOT NULL,
`mime` text NOT NULL `mime` text NOT NULL
); );

View File

@@ -1,7 +1,7 @@
CREATE TABLE `document` ( CREATE TABLE `document` (
`name` text NOT NULL, `name` text NOT NULL,
`last_modified` integer NOT NULL, `last_modified` integer NOT NULL,
`contentType` text NOT NULL, `contentType` text NOT NULL,
`size` integer NOT NULL, `size` integer NOT NULL,
`perm` text NOT NULL `perm` text NOT NULL
); );

View File

@@ -1,13 +1,38 @@
PRAGMA foreign_keys=OFF;--> statement-breakpoint PRAGMA foreign_keys = OFF;
CREATE TABLE `__new_document` (
`name` text PRIMARY KEY NOT NULL,
`last_modified` integer NOT NULL,
`contentType` text NOT NULL,
`size` integer NOT NULL,
`perm` text NOT NULL
);
--> statement-breakpoint --> statement-breakpoint
INSERT INTO `__new_document`("name", "last_modified", "contentType", "size", "perm") SELECT "name", "last_modified", "contentType", "size", "perm" FROM `document`;--> statement-breakpoint CREATE TABLE `__new_document` (
DROP TABLE `document`;--> statement-breakpoint `name` text PRIMARY KEY NOT NULL,
ALTER TABLE `__new_document` RENAME TO `document`;--> statement-breakpoint `last_modified` integer NOT NULL,
PRAGMA foreign_keys=ON; `contentType` text NOT NULL,
`size` integer NOT NULL,
`perm` text NOT NULL
);
--> statement-breakpoint
INSERT INTO
`__new_document`(
"name",
"last_modified",
"contentType",
"size",
"perm"
)
SELECT
"name",
"last_modified",
"contentType",
"size",
"perm"
FROM
`document`;
--> statement-breakpoint
DROP TABLE `document`;
--> statement-breakpoint
ALTER TABLE
`__new_document` RENAME TO `document`;
--> statement-breakpoint
PRAGMA foreign_keys = ON;

View File

@@ -1 +1,2 @@
ALTER TABLE `document` RENAME COLUMN "contentType" TO "content_type"; ALTER TABLE
`document` RENAME COLUMN "contentType" TO "content_type";

View File

@@ -1 +1,4 @@
ALTER TABLE `document` ADD `content` text; ALTER TABLE
`document`
ADD
`content` text;

View File

@@ -1 +1,4 @@
ALTER TABLE `document` ALTER COLUMN "content" TO "content" text NOT NULL; ALTER TABLE
`document`
ALTER COLUMN
"content" TO "content" text NOT NULL;

View File

@@ -1 +1,4 @@
ALTER TABLE `document` ALTER COLUMN "content" TO "content" text; ALTER TABLE
`document`
ALTER COLUMN
"content" TO "content" text;

View File

@@ -1,12 +1,17 @@
CREATE TABLE `cache` ( CREATE TABLE `cache` (
`scope` text NOT NULL, `scope` text NOT NULL,
`key` text PRIMARY KEY NOT NULL, `key` text PRIMARY KEY NOT NULL,
`json` text, `json` text,
`binary` blob, `binary` blob,
`created_at` integer DEFAULT (current_timestamp), `created_at` integer DEFAULT (CURRENT_TIMESTAMP),
`expires_at` integer `expires_at` integer
); );
--> statement-breakpoint --> statement-breakpoint
CREATE INDEX `key_idx` ON `cache` (`key`);--> statement-breakpoint CREATE INDEX `key_idx` ON `cache` (`key`);
CREATE INDEX `scope_idx` ON `cache` (`scope`);--> statement-breakpoint
CREATE INDEX `name_idx` ON `document` (`name`); --> statement-breakpoint
CREATE INDEX `scope_idx` ON `cache` (`scope`);
--> statement-breakpoint
CREATE INDEX `name_idx` ON `document` (`name`);

View File

@@ -0,0 +1,2 @@
ALTER TABLE
`image` RENAME COLUMN "blurhash" TO "thumbhash";

View File

@@ -0,0 +1,22 @@
DROP INDEX "key_idx";
--> statement-breakpoint
DROP INDEX "scope_idx";
--> statement-breakpoint
DROP INDEX "name_idx";
--> statement-breakpoint
ALTER TABLE
`image`
ALTER COLUMN
"created_at" TO "created_at" integer DEFAULT (unixepoch());
--> statement-breakpoint
CREATE INDEX `key_idx` ON `cache` (`key`);
--> statement-breakpoint
CREATE INDEX `scope_idx` ON `cache` (`scope`);
--> statement-breakpoint
CREATE INDEX `name_idx` ON `document` (`name`);

View File

@@ -132,4 +132,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -132,4 +132,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -178,4 +178,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -223,4 +223,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -223,4 +223,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -225,4 +225,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -230,4 +230,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -230,4 +230,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -230,4 +230,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -306,4 +306,4 @@
"internal": { "internal": {
"indexes": {} "indexes": {}
} }
} }

View File

@@ -0,0 +1,311 @@
{
"version": "6",
"dialect": "sqlite",
"id": "685b57ca-45e0-4373-baee-fc3abb4f2d74",
"prevId": "5694a345-e55c-4aa3-9f29-1045b28f5203",
"tables": {
"cache": {
"name": "cache",
"columns": {
"scope": {
"name": "scope",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"key": {
"name": "key",
"type": "text",
"primaryKey": true,
"notNull": true,
"autoincrement": false
},
"json": {
"name": "json",
"type": "text",
"primaryKey": false,
"notNull": false,
"autoincrement": false
},
"binary": {
"name": "binary",
"type": "blob",
"primaryKey": false,
"notNull": false,
"autoincrement": false
},
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false,
"default": "(current_timestamp)"
},
"expires_at": {
"name": "expires_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false
}
},
"indexes": {
"key_idx": {
"name": "key_idx",
"columns": [
"key"
],
"isUnique": false
},
"scope_idx": {
"name": "scope_idx",
"columns": [
"scope"
],
"isUnique": false
}
},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"document": {
"name": "document",
"columns": {
"name": {
"name": "name",
"type": "text",
"primaryKey": true,
"notNull": true,
"autoincrement": false
},
"last_modified": {
"name": "last_modified",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"content": {
"name": "content",
"type": "text",
"primaryKey": false,
"notNull": false,
"autoincrement": false
},
"content_type": {
"name": "content_type",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"size": {
"name": "size",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"perm": {
"name": "perm",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
}
},
"indexes": {
"name_idx": {
"name": "name_idx",
"columns": [
"name"
],
"isUnique": false
}
},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"image": {
"name": "image",
"columns": {
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false,
"default": "(current_timestamp)"
},
"url": {
"name": "url",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"average": {
"name": "average",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"thumbhash": {
"name": "thumbhash",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"mime": {
"name": "mime",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"performance": {
"name": "performance",
"columns": {
"path": {
"name": "path",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"search": {
"name": "search",
"type": "text",
"primaryKey": false,
"notNull": false,
"autoincrement": false
},
"time": {
"name": "time",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false,
"default": "(STRFTIME('%s', 'now') * 1000)"
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"session": {
"name": "session",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true,
"autoincrement": false
},
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false,
"default": "(current_timestamp)"
},
"expires_at": {
"name": "expires_at",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"user_id": {
"name": "user_id",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"user": {
"name": "user",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true,
"autoincrement": false
},
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false,
"default": "(current_timestamp)"
},
"email": {
"name": "email",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"name": {
"name": "name",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
}
},
"views": {},
"enums": {},
"_meta": {
"schemas": {},
"tables": {},
"columns": {
"\"image\".\"blurhash\"": "\"image\".\"thumbhash\""
}
},
"internal": {
"indexes": {}
}
}

View File

@@ -0,0 +1,309 @@
{
"version": "6",
"dialect": "sqlite",
"id": "11bbbc9d-3c0c-4fb9-893f-c87d5b8660d0",
"prevId": "685b57ca-45e0-4373-baee-fc3abb4f2d74",
"tables": {
"cache": {
"name": "cache",
"columns": {
"scope": {
"name": "scope",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"key": {
"name": "key",
"type": "text",
"primaryKey": true,
"notNull": true,
"autoincrement": false
},
"json": {
"name": "json",
"type": "text",
"primaryKey": false,
"notNull": false,
"autoincrement": false
},
"binary": {
"name": "binary",
"type": "blob",
"primaryKey": false,
"notNull": false,
"autoincrement": false
},
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false,
"default": "(current_timestamp)"
},
"expires_at": {
"name": "expires_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false
}
},
"indexes": {
"key_idx": {
"name": "key_idx",
"columns": [
"key"
],
"isUnique": false
},
"scope_idx": {
"name": "scope_idx",
"columns": [
"scope"
],
"isUnique": false
}
},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"document": {
"name": "document",
"columns": {
"name": {
"name": "name",
"type": "text",
"primaryKey": true,
"notNull": true,
"autoincrement": false
},
"last_modified": {
"name": "last_modified",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"content": {
"name": "content",
"type": "text",
"primaryKey": false,
"notNull": false,
"autoincrement": false
},
"content_type": {
"name": "content_type",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"size": {
"name": "size",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"perm": {
"name": "perm",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
}
},
"indexes": {
"name_idx": {
"name": "name_idx",
"columns": [
"name"
],
"isUnique": false
}
},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"image": {
"name": "image",
"columns": {
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false,
"default": "(unixepoch())"
},
"url": {
"name": "url",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"average": {
"name": "average",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"thumbhash": {
"name": "thumbhash",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"mime": {
"name": "mime",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"performance": {
"name": "performance",
"columns": {
"path": {
"name": "path",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"search": {
"name": "search",
"type": "text",
"primaryKey": false,
"notNull": false,
"autoincrement": false
},
"time": {
"name": "time",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false,
"default": "(STRFTIME('%s', 'now') * 1000)"
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"session": {
"name": "session",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true,
"autoincrement": false
},
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": false,
"autoincrement": false,
"default": "(current_timestamp)"
},
"expires_at": {
"name": "expires_at",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"user_id": {
"name": "user_id",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
},
"user": {
"name": "user",
"columns": {
"id": {
"name": "id",
"type": "text",
"primaryKey": true,
"notNull": true,
"autoincrement": false
},
"created_at": {
"name": "created_at",
"type": "integer",
"primaryKey": false,
"notNull": true,
"autoincrement": false,
"default": "(current_timestamp)"
},
"email": {
"name": "email",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
},
"name": {
"name": "name",
"type": "text",
"primaryKey": false,
"notNull": true,
"autoincrement": false
}
},
"indexes": {},
"foreignKeys": {},
"compositePrimaryKeys": {},
"uniqueConstraints": {},
"checkConstraints": {}
}
},
"views": {},
"enums": {},
"_meta": {
"schemas": {},
"tables": {},
"columns": {}
},
"internal": {
"indexes": {}
}
}

View File

@@ -71,6 +71,20 @@
"when": 1736172911816, "when": 1736172911816,
"tag": "0009_free_robin_chapel", "tag": "0009_free_robin_chapel",
"breakpoints": true "breakpoints": true
},
{
"idx": 10,
"version": "6",
"when": 1762099260474,
"tag": "0010_youthful_tyrannus",
"breakpoints": true
},
{
"idx": 11,
"version": "6",
"when": 1768058169808,
"tag": "0011_reflective_frank_castle",
"breakpoints": true
} }
] ]
} }

View File

@@ -1,5 +0,0 @@
import { defineConfig } from "$fresh/server.ts";
import tailwind from "$fresh/plugins/tailwind.ts";
export default defineConfig({
plugins: [tailwind()],
});

View File

@@ -1,138 +0,0 @@
// DO NOT EDIT. This file is generated by Fresh.
// This file SHOULD be checked into source version control.
// This file is automatically updated during development when running `dev.ts`.
import * as $_404 from "./routes/_404.tsx";
import * as $_app from "./routes/_app.tsx";
import * as $_layout from "./routes/_layout.tsx";
import * as $_middleware from "./routes/_middleware.ts";
import * as $admin_cache_index from "./routes/admin/cache/index.tsx";
import * as $admin_log_index from "./routes/admin/log/index.tsx";
import * as $admin_performance_index from "./routes/admin/performance/index.tsx";
import * as $api_articles_name_ from "./routes/api/articles/[name].ts";
import * as $api_articles_create_index from "./routes/api/articles/create/index.ts";
import * as $api_articles_index from "./routes/api/articles/index.ts";
import * as $api_auth_callback from "./routes/api/auth/callback.ts";
import * as $api_auth_login from "./routes/api/auth/login.ts";
import * as $api_auth_logout from "./routes/api/auth/logout.ts";
import * as $api_cache from "./routes/api/cache.ts";
import * as $api_images_index from "./routes/api/images/index.ts";
import * as $api_index from "./routes/api/index.ts";
import * as $api_logs from "./routes/api/logs.ts";
import * as $api_movies_name_ from "./routes/api/movies/[name].ts";
import * as $api_movies_enhance_name_ from "./routes/api/movies/enhance/[name].ts";
import * as $api_movies_index from "./routes/api/movies/index.ts";
import * as $api_query_index from "./routes/api/query/index.ts";
import * as $api_recipes_name_ from "./routes/api/recipes/[name].ts";
import * as $api_recipes_create_index from "./routes/api/recipes/create/index.ts";
import * as $api_recipes_create_parseJsonLd from "./routes/api/recipes/create/parseJsonLd.ts";
import * as $api_recipes_index from "./routes/api/recipes/index.ts";
import * as $api_recommendation_all from "./routes/api/recommendation/all.ts";
import * as $api_recommendation_data from "./routes/api/recommendation/data.ts";
import * as $api_recommendation_index from "./routes/api/recommendation/index.ts";
import * as $api_recommendation_movie_id_ from "./routes/api/recommendation/movie/[id].ts";
import * as $api_series_name_ from "./routes/api/series/[name].ts";
import * as $api_series_enhance_name_ from "./routes/api/series/enhance/[name].ts";
import * as $api_series_index from "./routes/api/series/index.ts";
import * as $api_tmdb_id_ from "./routes/api/tmdb/[id].ts";
import * as $api_tmdb_credits_id_ from "./routes/api/tmdb/credits/[id].ts";
import * as $api_tmdb_query from "./routes/api/tmdb/query.ts";
import * as $articles_name_ from "./routes/articles/[name].tsx";
import * as $articles_index from "./routes/articles/index.tsx";
import * as $index from "./routes/index.tsx";
import * as $movies_name_ from "./routes/movies/[name].tsx";
import * as $movies_index from "./routes/movies/index.tsx";
import * as $recipes_name_ from "./routes/recipes/[name].tsx";
import * as $recipes_index from "./routes/recipes/index.tsx";
import * as $series_name_ from "./routes/series/[name].tsx";
import * as $series_index from "./routes/series/index.tsx";
import * as $Counter from "./islands/Counter.tsx";
import * as $IngredientsList from "./islands/IngredientsList.tsx";
import * as $KMenu from "./islands/KMenu.tsx";
import * as $KMenu_commands from "./islands/KMenu/commands.ts";
import * as $KMenu_commands_add_movie_infos from "./islands/KMenu/commands/add_movie_infos.ts";
import * as $KMenu_commands_add_series_infos from "./islands/KMenu/commands/add_series_infos.ts";
import * as $KMenu_commands_create_article from "./islands/KMenu/commands/create_article.ts";
import * as $KMenu_commands_create_movie from "./islands/KMenu/commands/create_movie.ts";
import * as $KMenu_commands_create_recipe from "./islands/KMenu/commands/create_recipe.ts";
import * as $KMenu_commands_create_recommendations from "./islands/KMenu/commands/create_recommendations.ts";
import * as $KMenu_commands_create_series from "./islands/KMenu/commands/create_series.ts";
import * as $KMenu_types from "./islands/KMenu/types.ts";
import * as $Link from "./islands/Link.tsx";
import * as $Recommendations from "./islands/Recommendations.tsx";
import * as $Search from "./islands/Search.tsx";
import type { Manifest } from "$fresh/server.ts";
const manifest = {
routes: {
"./routes/_404.tsx": $_404,
"./routes/_app.tsx": $_app,
"./routes/_layout.tsx": $_layout,
"./routes/_middleware.ts": $_middleware,
"./routes/admin/cache/index.tsx": $admin_cache_index,
"./routes/admin/log/index.tsx": $admin_log_index,
"./routes/admin/performance/index.tsx": $admin_performance_index,
"./routes/api/articles/[name].ts": $api_articles_name_,
"./routes/api/articles/create/index.ts": $api_articles_create_index,
"./routes/api/articles/index.ts": $api_articles_index,
"./routes/api/auth/callback.ts": $api_auth_callback,
"./routes/api/auth/login.ts": $api_auth_login,
"./routes/api/auth/logout.ts": $api_auth_logout,
"./routes/api/cache.ts": $api_cache,
"./routes/api/images/index.ts": $api_images_index,
"./routes/api/index.ts": $api_index,
"./routes/api/logs.ts": $api_logs,
"./routes/api/movies/[name].ts": $api_movies_name_,
"./routes/api/movies/enhance/[name].ts": $api_movies_enhance_name_,
"./routes/api/movies/index.ts": $api_movies_index,
"./routes/api/query/index.ts": $api_query_index,
"./routes/api/recipes/[name].ts": $api_recipes_name_,
"./routes/api/recipes/create/index.ts": $api_recipes_create_index,
"./routes/api/recipes/create/parseJsonLd.ts":
$api_recipes_create_parseJsonLd,
"./routes/api/recipes/index.ts": $api_recipes_index,
"./routes/api/recommendation/all.ts": $api_recommendation_all,
"./routes/api/recommendation/data.ts": $api_recommendation_data,
"./routes/api/recommendation/index.ts": $api_recommendation_index,
"./routes/api/recommendation/movie/[id].ts": $api_recommendation_movie_id_,
"./routes/api/series/[name].ts": $api_series_name_,
"./routes/api/series/enhance/[name].ts": $api_series_enhance_name_,
"./routes/api/series/index.ts": $api_series_index,
"./routes/api/tmdb/[id].ts": $api_tmdb_id_,
"./routes/api/tmdb/credits/[id].ts": $api_tmdb_credits_id_,
"./routes/api/tmdb/query.ts": $api_tmdb_query,
"./routes/articles/[name].tsx": $articles_name_,
"./routes/articles/index.tsx": $articles_index,
"./routes/index.tsx": $index,
"./routes/movies/[name].tsx": $movies_name_,
"./routes/movies/index.tsx": $movies_index,
"./routes/recipes/[name].tsx": $recipes_name_,
"./routes/recipes/index.tsx": $recipes_index,
"./routes/series/[name].tsx": $series_name_,
"./routes/series/index.tsx": $series_index,
},
islands: {
"./islands/Counter.tsx": $Counter,
"./islands/IngredientsList.tsx": $IngredientsList,
"./islands/KMenu.tsx": $KMenu,
"./islands/KMenu/commands.ts": $KMenu_commands,
"./islands/KMenu/commands/add_movie_infos.ts":
$KMenu_commands_add_movie_infos,
"./islands/KMenu/commands/add_series_infos.ts":
$KMenu_commands_add_series_infos,
"./islands/KMenu/commands/create_article.ts":
$KMenu_commands_create_article,
"./islands/KMenu/commands/create_movie.ts": $KMenu_commands_create_movie,
"./islands/KMenu/commands/create_recipe.ts": $KMenu_commands_create_recipe,
"./islands/KMenu/commands/create_recommendations.ts":
$KMenu_commands_create_recommendations,
"./islands/KMenu/commands/create_series.ts": $KMenu_commands_create_series,
"./islands/KMenu/types.ts": $KMenu_types,
"./islands/Link.tsx": $Link,
"./islands/Recommendations.tsx": $Recommendations,
"./islands/Search.tsx": $Search,
},
baseUrl: import.meta.url,
} satisfies Manifest;
export default manifest;

View File

@@ -1,6 +1,6 @@
import type { Signal } from "@preact/signals"; import type { Signal } from "@preact/signals";
import { Button } from "@components/Button.tsx"; import { Button } from "@components/Button.tsx";
import { IconCircleMinus, IconCirclePlus } from "@components/icons.tsx"; import { TbCircleMinus, TbCirclePlus } from "@preact-icons/tb";
interface CounterProps { interface CounterProps {
count: Signal<number>; count: Signal<number>;
@@ -10,21 +10,21 @@ export default function Counter(props: CounterProps) {
props.count.value = Math.max(1, props.count.value); props.count.value = Math.max(1, props.count.value);
return ( return (
<div class="flex items-center px-1 py-2 rounded-xl"> <div class="flex items-center px-1 py-2 rounded-xl">
<Button <Button onClick={() => props.count.value -= 1}>
class="" <TbCircleMinus class="h-6 w-6" />
onClick={() => props.count.value -= 1}
>
<IconCircleMinus />
</Button> </Button>
<input <input
class="text-3xl bg-transparent inline text-center -mx-4" class="text-3xl bg-transparent inline text-center -mx-4"
type="number" type="number"
size={props.count.toString().length} size={props.count.toString().length}
value={props.count} value={props.count}
onInput={(ev) => props.count.value = ev.target?.value} onInput={(ev) => {
const target = ev.target as HTMLInputElement;
props.count.value = Math.max(1, Number(target.value));
}}
/> />
<Button onClick={() => props.count.value += 1}> <Button onClick={() => props.count.value += 1}>
<IconCirclePlus /> <TbCirclePlus class="h-6 w-6" />
</Button> </Button>
</div> </div>
); );

View File

@@ -1,24 +1,19 @@
import { Signal } from "@preact/signals"; import { Signal } from "@preact/signals";
import type { Ingredient, IngredientGroup } from "@lib/recipeSchema.ts"; import type { Ingredient, IngredientGroup } from "@lib/recipeSchema.ts";
import { FunctionalComponent } from "preact";
import { unitsOfMeasure } from "@lib/parseIngredient.ts"; import { unitsOfMeasure } from "@lib/parseIngredient.ts";
import { renderMarkdown } from "@lib/documents.ts";
function formatAmount(num: number) { function formatAmount(num: number) {
if (num === 0) return ""; if (num === 0) return "";
return (Math.floor(num * 4) / 4).toString(); return (Math.round(num * 4) / 4).toString();
} }
function formatUnit(unit: string, amount: number) { function formatUnit(unit: string, amount: number) {
const unitKey = unit.toLowerCase() as keyof typeof unitsOfMeasure; if (!unit) return "";
const unitKey = unit.toLowerCase() as (keyof typeof unitsOfMeasure);
if (unitKey in unitsOfMeasure) { if (unitKey in unitsOfMeasure) {
if (amount > 1 && unitsOfMeasure[unitKey].plural !== undefined) { if (amount > 1 && unitsOfMeasure[unitKey].plural !== undefined) {
return unitsOfMeasure[unitKey].plural; return unitsOfMeasure[unitKey].plural;
} }
if (unitKey !== "cup") {
return unitsOfMeasure[unitKey].short;
}
return unitKey.toString(); return unitKey.toString();
} else { } else {
return unit; return unit;
@@ -43,39 +38,46 @@ const Ingredient = (
return ( return (
<tr key={key}> <tr key={key}>
<td class="pr-4 py-2"> <td class="pr-4 py-1">
{formatAmount(finalAmount || 0)} {formatAmount(finalAmount || 0)}
<span class="ml-0.5 opacity-50"> <span class="ml-0.5 opacity-50">
{formatUnit(unit, finalAmount || 0)} {formatUnit(unit, finalAmount || 0)}
</span> </span>
</td> </td>
<td class="px-4 py-2">{name}</td> <td class="px-4 py-1">{name}</td>
</tr> </tr>
); );
}; };
export const IngredientsList: FunctionalComponent< export const IngredientsList = (
{ { ingredients, amount, portion }: {
ingredients: (Ingredient | IngredientGroup)[]; ingredients: (Ingredient | IngredientGroup)[];
amount: Signal<number>; amount: Signal<number>;
portion?: number; portion?: number;
} },
> = (
{ ingredients, amount, portion },
) => { ) => {
return ( return (
<table class="w-full border-collapse table-auto"> <table class="w-full border-collapse table-auto">
<tbody> <tbody>
{ingredients.filter((s) => !!s?.length).map((item) => { {ingredients.map((item) => {
if ("items" in item) {
return item.items.map((ing, i) => {
return (
<Ingredient
key={i}
ingredient={ing}
amount={amount}
portion={portion}
/>
);
});
} else {
return ( return (
<div dangerouslySetInnerHTML={{ __html: renderMarkdown(item) }}> <Ingredient ingredient={item} amount={amount} portion={portion} />
</div>
); );
// return ( }
// <Ingredient ingredient={item} amount={amount} portion={portion} /> })}
// ); </tbody>
})} </table>
</tbody> );
</table> };
);
};

View File

@@ -4,7 +4,7 @@ import { useEventListener } from "@lib/hooks/useEventListener.ts";
import { menus } from "@islands/KMenu/commands.ts"; import { menus } from "@islands/KMenu/commands.ts";
import { MenuEntry } from "@islands/KMenu/types.ts"; import { MenuEntry } from "@islands/KMenu/types.ts";
import * as icons from "@components/icons.tsx"; import * as icons from "@components/icons.tsx";
import { IS_BROWSER } from "$fresh/runtime.ts"; import { isKMenuOpen } from "@lib/kmenu.ts";
const KMenuEntry = ( const KMenuEntry = (
{ entry, activeIndex, index }: { { entry, activeIndex, index }: {
entry: MenuEntry; entry: MenuEntry;
@@ -21,7 +21,7 @@ const KMenuEntry = (
: "text-gray-400" : "text-gray-400"
}`} }`}
> >
{entry?.icon && icons[entry.icon]({ class: "w-4 h-4 mr-1" })} {entry?.icon && icons[entry.icon]({ class: "min-w-4 h-4 mr-1" })}
{entry.title} {entry.title}
</div> </div>
); );
@@ -42,7 +42,7 @@ export const KMenu = (
const input = useRef<HTMLInputElement>(null); const input = useRef<HTMLInputElement>(null);
const commandInput = useSignal(""); const commandInput = useSignal("");
const visible = useSignal(false); const visible = isKMenuOpen;
if (visible.value === false) { if (visible.value === false) {
setTimeout(() => { setTimeout(() => {
activeMenuType.value = "main"; activeMenuType.value = "main";
@@ -103,8 +103,9 @@ export const KMenu = (
} }
useEventListener("keydown", (ev: KeyboardEvent) => { useEventListener("keydown", (ev: KeyboardEvent) => {
const target = ev.target as HTMLElement;
if (ev.key === "k") { if (ev.key === "k") {
if (ev?.target?.nodeName == "INPUT") { if (target.nodeName == "INPUT") {
return; return;
} }
@@ -153,27 +154,29 @@ export const KMenu = (
} else { } else {
input.current?.blur(); input.current?.blur();
} }
}, IS_BROWSER ? document?.body : undefined); }, typeof document !== "undefined" ? document?.body : undefined);
return ( return (
<div <div
class={`${visible.value ? "opacity-100" : "opacity-0"} pointer-events-${ class={`${visible.value ? "opacity-100" : "opacity-0"} ${
visible.value ? "auto" : "none" visible.value ? "pointer-events-auto" : "pointer-events-none"
} transition grid place-items-center w-full h-full fixed top-0 left-0 z-50`} } transition grid place-items-center w-full h-full fixed top-0 left-0 z-50`}
style={{ background: "#141217ee" }} style={{ background: "#141217ee" }}
> >
<div <div
class={`relative w-1/2 max-h-64 max-w-[400px] rounded-2xl shadow-2xl nnoisy-gradient overflow-hidden after:opacity-10 bg-gray-700`} class={`relative w-1/2 max-h-64 max-w-100 rounded-2xl shadow-2xl nnoisy-gradient overflow-hidden after:opacity-10 bg-gray-700`}
style={{ background: "#2B2930", color: "#818181" }} style={{ background: "#2B2930", color: "#818181" }}
> >
<div <div
class={`grid h-12 text-gray-400 ${ class={`grid min-h-12 text-gray-400 ${
activeState.value !== "loading" && "border-b" (activeState.value === "normal" || activeState.value === "input") &&
"border-b"
} border-gray-500 `} } border-gray-500 `}
style={{ style={{
gridTemplateColumns: activeState.value !== "loading" gridTemplateColumns:
? "auto 1fr" (activeState.value === "normal" || activeState.value === "input")
: "1fr", ? "auto 1fr"
: "1fr",
}} }}
> >
{(activeState.value === "normal" || activeState.value === "input") && {(activeState.value === "normal" || activeState.value === "input") &&
@@ -197,12 +200,18 @@ export const KMenu = (
)} )}
{activeState.value === "loading" && ( {activeState.value === "loading" && (
<div class="py-3 px-4 flex items-center gap-2"> <div class="py-3 px-4 flex items-center gap-2">
<icons.IconLoader2 class="animate-spin w-4 h-4" /> <icons.IconLoader2 class="animate-spin min-w-4 h-4" />
{loadingText.value || "Loading..."} {loadingText.value || "Loading..."}
</div> </div>
)} )}
{activeState.value === "error" && (
<div class="py-3 px-4 flex items-center gap-2 text-red-400">
<icons.IconAlertCircle class="min-w-4 h-4" />
{loadingText.value || "An error occurred"}
</div>
)}
</div> </div>
{activeState.value === "normal" && {(activeState.value === "normal" || activeState.value === "input") &&
( (
<div <div
class="" class=""

View File

@@ -5,8 +5,8 @@ import { createNewArticle } from "@islands/KMenu/commands/create_article.ts";
import { getCookie } from "@lib/string.ts"; import { getCookie } from "@lib/string.ts";
import { addSeriesInfo } from "@islands/KMenu/commands/add_series_infos.ts"; import { addSeriesInfo } from "@islands/KMenu/commands/add_series_infos.ts";
import { createNewSeries } from "@islands/KMenu/commands/create_series.ts"; import { createNewSeries } from "@islands/KMenu/commands/create_series.ts";
import { updateAllRecommendations } from "@islands/KMenu/commands/create_recommendations.ts";
import { createNewRecipe } from "@islands/KMenu/commands/create_recipe.ts"; import { createNewRecipe } from "@islands/KMenu/commands/create_recipe.ts";
import { enhanceArticleInfo } from "@islands/KMenu/commands/enhance_article_infos.ts";
export const menus: Record<string, Menu> = { export const menus: Record<string, Menu> = {
main: { main: {
@@ -77,6 +77,7 @@ export const menus: Record<string, Menu> = {
createNewSeries, createNewSeries,
createNewRecipe, createNewRecipe,
addMovieInfos, addMovieInfos,
enhanceArticleInfo,
// updateAllRecommendations, // updateAllRecommendations,
], ],
}, },

View File

@@ -1,46 +1,68 @@
import { Movie } from "@lib/resource/movies.ts";
import { TMDBMovie } from "@lib/types.ts"; import { TMDBMovie } from "@lib/types.ts";
import { getCookie } from "@lib/string.ts"; import { getCookie } from "@lib/string.ts";
import { MenuEntry } from "../types.ts";
import { ReviewResource } from "@lib/marka/schema.ts";
export const addMovieInfos: MenuEntry = { export const addMovieInfos: MenuEntry = {
title: "Add Movie infos", title: "Add Movie infos",
meta: "", meta: "",
icon: "IconReportSearch", icon: "IconReportSearch",
cb: async (state, context) => { cb: async (state, context) => {
state.activeState.value = "loading"; try {
const movie = context as Movie; state.activeState.value = "loading";
const movie = context as ReviewResource;
const query = movie.name; const query = movie.name;
const response = await fetch( const response = await fetch(
`/api/tmdb/query?q=${encodeURIComponent(query)}`, `/api/tmdb/query?q=${encodeURIComponent(query)}`,
); );
const json = await response.json() as TMDBMovie[]; if (!response.ok) {
throw new Error(await response.text());
}
const menuID = `result/${movie.name}`; const json = await response.json() as TMDBMovie[];
state.menus[menuID] = { const menuID = `result/${movie.name}`;
title: "Select",
entries: json.map((m) => ({
title: `${m.title} released ${m.release_date}`,
cb: async () => {
state.activeState.value = "loading";
await fetch(`/api/movies/enhance/${movie.name}/`, {
method: "POST",
body: JSON.stringify({ tmdbId: m.id }),
});
state.visible.value = false;
state.activeState.value = "normal";
window.location.reload();
},
})),
};
state.activeMenu.value = menuID; state.menus[menuID] = {
state.commandInput.value = ""; title: "Select",
state.activeState.value = "normal"; entries: json.map((m) => ({
title: `${m.title} released ${m.release_date}`,
cb: async () => {
try {
state.activeState.value = "loading";
await fetch(`/api/movies/enhance/${movie.name}/`, {
method: "POST",
body: JSON.stringify({ tmdbId: m.id }),
});
state.visible.value = false;
state.activeState.value = "normal";
globalThis.location.reload();
} catch (e) {
state.activeState.value = "error";
if (e instanceof Error) {
if ("message" in e) {
state.loadingText.value = e.message;
}
}
}
},
})),
};
state.activeMenu.value = menuID;
state.commandInput.value = "";
state.activeState.value = "normal";
} catch (e) {
state.activeState.value = "error";
if (e instanceof Error) {
if ("message" in e) {
state.loadingText.value = e.message;
}
}
}
}, },
visible: () => { visible: () => {
const loc = globalThis["location"]; const loc = globalThis["location"];

View File

@@ -1,46 +1,68 @@
import { MenuEntry } from "@islands/KMenu/types.ts"; import { MenuEntry } from "@islands/KMenu/types.ts";
import { TMDBSeries } from "@lib/types.ts"; import { TMDBSeries } from "@lib/types.ts";
import { getCookie } from "@lib/string.ts"; import { getCookie } from "@lib/string.ts";
import { Series } from "@lib/resource/series.ts"; import { ReviewResource } from "@lib/marka/schema.ts";
export const addSeriesInfo: MenuEntry = { export const addSeriesInfo: MenuEntry = {
title: "Add Series infos", title: "Add Series infos",
meta: "", meta: "",
icon: "IconReportSearch", icon: "IconReportSearch",
cb: async (state, context) => { cb: async (state, context) => {
state.activeState.value = "loading"; try {
const series = context as Series; state.activeState.value = "loading";
const series = context as ReviewResource;
const query = series.name; const query = series.name;
const response = await fetch( const response = await fetch(
`/api/tmdb/query?q=${encodeURIComponent(query)}&type=serie`, `/api/tmdb/query?q=${encodeURIComponent(query)}&type=serie`,
); );
const json = await response.json() as TMDBSeries[]; if (!response.ok) {
throw new Error(await response.text());
}
const menuID = `result/${series.name}`; const json = await response.json() as TMDBSeries[];
state.menus[menuID] = { const menuID = `result/${series.name}`;
title: "Select",
entries: json.map((m) => ({
title: `${m.name || m.original_name} released ${m.first_air_date}`,
cb: async () => {
state.activeState.value = "loading";
await fetch(`/api/series/enhance/${series.name}/`, {
method: "POST",
body: JSON.stringify({ tmdbId: m.id }),
});
state.visible.value = false;
state.activeState.value = "normal";
//window.location.reload();
},
})),
};
state.commandInput.value = ""; state.menus[menuID] = {
state.activeMenu.value = menuID; title: "Select",
state.activeState.value = "normal"; entries: json.map((m) => ({
title: `${m.name || m.original_name} released ${m.first_air_date}`,
cb: async () => {
try {
state.activeState.value = "loading";
await fetch(`/api/series/enhance/${series.name}/`, {
method: "POST",
body: JSON.stringify({ tmdbId: m.id }),
});
state.visible.value = false;
state.activeState.value = "normal";
//window.location.reload();
} catch (e) {
state.activeState.value = "error";
if (e instanceof Error) {
if ("message" in e) {
state.loadingText.value = e.message;
}
}
}
},
})),
};
state.commandInput.value = "";
state.activeMenu.value = menuID;
state.activeState.value = "normal";
} catch (e) {
state.activeState.value = "error";
if (e instanceof Error) {
if ("message" in e) {
state.loadingText.value = e.message;
}
}
}
}, },
visible: () => { visible: () => {
const loc = globalThis["location"]; const loc = globalThis["location"];

View File

@@ -22,14 +22,16 @@ export const createNewArticle: MenuEntry = {
state.activeState.value = "loading"; state.activeState.value = "loading";
fetchStream("/api/articles/create?url=" + value, (chunk) => { fetchStream("/api/articles/create?url=" + value, (chunk) => {
if (chunk.startsWith("id:")) { if (chunk.type === "error") {
state.activeState.value = "error";
state.loadingText.value = chunk.message;
} else if (chunk.type === "finished") {
state.loadingText.value = "Finished"; state.loadingText.value = "Finished";
setTimeout(() => { setTimeout(() => {
window.location.href = "/articles/" + globalThis.location.href = "/articles/" + chunk.url;
chunk.replace("id:", "").trim();
}, 500); }, 500);
} else { } else {
state.loadingText.value = chunk; state.loadingText.value = chunk.message;
} }
}); });
} }

View File

@@ -1,8 +1,8 @@
import { MenuEntry } from "@islands/KMenu/types.ts"; import { MenuEntry } from "@islands/KMenu/types.ts";
import { TMDBMovie } from "@lib/types.ts"; import { TMDBMovie } from "@lib/types.ts";
import { debounce } from "@lib/helpers.ts"; import { debounce } from "@lib/helpers.ts";
import { Movie } from "@lib/resource/movies.ts";
import { getCookie } from "@lib/string.ts"; import { getCookie } from "@lib/string.ts";
import { ReviewResource } from "@lib/marka/schema.ts";
export const createNewMovie: MenuEntry = { export const createNewMovie: MenuEntry = {
title: "Create new movie", title: "Create new movie",
@@ -31,35 +31,60 @@ export const createNewMovie: MenuEntry = {
let currentQuery: string; let currentQuery: string;
const search = debounce(async function search(query: string) { const search = debounce(async function search(query: string) {
currentQuery = query; try {
if (query.length < 2) { currentQuery = query;
return; if (query.length < 2) {
return;
}
const response = await fetch("/api/tmdb/query?q=" + query);
if (!response.ok) {
throw new Error(await response.text());
}
const movies = await response.json() as TMDBMovie[];
if (query !== currentQuery) return;
state.menus["input_link"] = {
title: "Search",
entries: movies.map((r) => {
return {
title: `${r.title} - ${r.release_date}`,
cb: async () => {
try {
state.activeState.value = "loading";
const response = await fetch("/api/movies/" + r.id, {
method: "POST",
});
if (!response.ok) {
throw new Error(await response.text());
}
const movie = await response.json() as ReviewResource;
unsub();
globalThis.location.href = "/movies/" + movie.name;
} catch (e) {
state.activeState.value = "error";
if (e instanceof Error) {
if ("message" in e) {
state.loadingText.value = e.message;
}
}
}
},
};
}),
};
state.activeMenu.value = "input_link";
} catch (e) {
state.activeState.value = "error";
if (e instanceof Error) {
if ("message" in e) {
state.loadingText.value = e.message;
}
}
} }
const response = await fetch("/api/tmdb/query?q=" + query);
const movies = await response.json() as TMDBMovie[];
if (query !== currentQuery) return;
state.menus["input_link"] = {
title: "Search",
entries: movies.map((r) => {
return {
title: `${r.title} - ${r.release_date}`,
cb: async () => {
state.activeState.value = "loading";
const response = await fetch("/api/movies/" + r.id, {
method: "POST",
});
const movie = await response.json() as Movie;
unsub();
window.location.href = "/movies/" + movie.name;
},
};
}),
};
state.activeMenu.value = "input_link";
}, 500); }, 500);
const unsub = state.commandInput.subscribe((value) => { const unsub = state.commandInput.subscribe((value) => {

View File

@@ -21,15 +21,17 @@ export const createNewRecipe: MenuEntry = {
state.activeState.value = "loading"; state.activeState.value = "loading";
fetchStream("/api/recipes/create?url=" + value, (chunk) => { fetchStream("/api/recipes/create?url=" + value, (msg) => {
if (chunk.startsWith("id:")) { if (msg.type === "error") {
state.activeState.value = "error";
state.loadingText.value = msg.message;
} else if (msg.type === "finished") {
state.loadingText.value = "Finished"; state.loadingText.value = "Finished";
setTimeout(() => { setTimeout(() => {
globalThis.location.href = "/recipes/" + globalThis.location.href = "/recipes/" + msg.url;
chunk.replace("id:", "").trim();
}, 500); }, 500);
} else { } else {
state.loadingText.value = chunk; state.loadingText.value = msg.message;
} }
}); });
} }

View File

@@ -10,12 +10,15 @@ export const updateAllRecommendations: MenuEntry = {
state.activeState.value = "loading"; state.activeState.value = "loading";
fetchStream("/api/recommendation/all", (chunk) => { fetchStream("/api/recommendation/all", (chunk) => {
if (chunk.toLowerCase().includes("finish")) { if (chunk.type === "error") {
state.activeState.value = "error";
state.loadingText.value = chunk.message;
} else if (chunk.type === "finished") {
setTimeout(() => { setTimeout(() => {
window.location.reload(); globalThis.location.reload();
}, 500); }, 500);
} else { } else {
state.loadingText.value = chunk; state.loadingText.value = chunk.message;
} }
}); });
}, },

View File

@@ -1,8 +1,8 @@
import { MenuEntry } from "@islands/KMenu/types.ts"; import { MenuEntry } from "@islands/KMenu/types.ts";
import { TMDBSeries } from "@lib/types.ts"; import { TMDBSeries } from "@lib/types.ts";
import { debounce } from "@lib/helpers.ts"; import { debounce } from "@lib/helpers.ts";
import { Series } from "@lib/resource/series.ts";
import { getCookie } from "@lib/string.ts"; import { getCookie } from "@lib/string.ts";
import { ReviewResource } from "@lib/marka/schema.ts";
export const createNewSeries: MenuEntry = { export const createNewSeries: MenuEntry = {
title: "Create new series", title: "Create new series",
@@ -31,38 +31,63 @@ export const createNewSeries: MenuEntry = {
let currentQuery: string; let currentQuery: string;
const search = debounce(async function search(query: string) { const search = debounce(async function search(query: string) {
currentQuery = query; try {
if (query.length < 2) { currentQuery = query;
return; if (query.length < 2) {
return;
}
const response = await fetch(
"/api/tmdb/query?q=" + query + "&type=series",
);
if (!response.ok) {
throw new Error(await response.text());
}
const series = await response.json() as TMDBSeries[];
if (query !== currentQuery) return;
state.menus["input_link"] = {
title: "Search",
entries: series.map((r) => {
return {
title: `${r.name} - ${r.first_air_date}`,
cb: async () => {
try {
state.activeState.value = "loading";
const response = await fetch("/api/series/" + r.id, {
method: "POST",
});
if (!response.ok) {
throw new Error(await response.text());
}
const series = await response.json() as ReviewResource;
unsub();
globalThis.location.href = "/series/" + series.name;
} catch (e) {
state.activeState.value = "error";
if (e instanceof Error) {
if ("message" in e) {
state.loadingText.value = e.message;
}
}
}
},
};
}),
};
state.commandInput.value = "";
state.activeMenu.value = "input_link";
} catch (e) {
state.activeState.value = "error";
if (e instanceof Error) {
if ("message" in e) {
state.loadingText.value = e.message;
}
}
} }
const response = await fetch(
"/api/tmdb/query?q=" + query + "&type=series",
);
const series = await response.json() as TMDBSeries[];
if (query !== currentQuery) return;
state.menus["input_link"] = {
title: "Search",
entries: series.map((r) => {
return {
title: `${r.name} - ${r.first_air_date}`,
cb: async () => {
state.activeState.value = "loading";
const response = await fetch("/api/series/" + r.id, {
method: "POST",
});
const series = await response.json() as Series;
unsub();
window.location.href = "/series/" + series.name;
},
};
}),
};
state.commandInput.value = "";
state.activeMenu.value = "input_link";
}, 500); }, 500);
const unsub = state.commandInput.subscribe((value) => { const unsub = state.commandInput.subscribe((value) => {

View File

@@ -0,0 +1,41 @@
import { getCookie } from "@lib/string.ts";
import { MenuEntry } from "../types.ts";
import { ArticleResource } from "@lib/marka/schema.ts";
import { fetchStream } from "@lib/helpers.ts";
export const enhanceArticleInfo: MenuEntry = {
title: "Enhance Article Info",
meta: "Update metadata and content from source url",
icon: "IconReportSearch",
cb: (state, context) => {
state.activeState.value = "loading";
const article = context as ArticleResource;
fetchStream(
`/api/articles/enhance/${article.name}/`,
(chunk) => {
if (chunk.type === "error") {
state.activeState.value = "error";
state.loadingText.value = chunk.message;
} else if (chunk.type == "finished") {
state.loadingText.value = "Finished";
setTimeout(() => {
state.visible.value = false;
state.activeState.value = "normal";
globalThis.location.reload();
}, 500);
} else {
state.loadingText.value = chunk.message;
}
},
{ method: "POST" },
);
},
visible: () => {
const loc = globalThis["location"];
if (!getCookie("session_cookie")) return false;
return (loc?.pathname?.includes("article") &&
!loc.pathname.endsWith("articles"));
},
};

View File

@@ -6,7 +6,7 @@ type IconKey = keyof typeof icons;
export type MenuState = { export type MenuState = {
activeMenu: Signal<string>; activeMenu: Signal<string>;
activeState: Signal<"input" | "error" | "normal" | "loading">; activeState: Signal<"input" | "error" | "normal" | "loading">;
loadingText:Signal<string>; loadingText: Signal<string>;
commandInput: Signal<string>; commandInput: Signal<string>;
visible: Signal<boolean>; visible: Signal<boolean>;
menus: Record<string, Menu>; menus: Record<string, Menu>;

14
islands/KMenuButton.tsx Normal file
View File

@@ -0,0 +1,14 @@
import { Button } from "@components/Button.tsx";
import { IconMenu2 } from "@components/icons.tsx";
import { isKMenuOpen } from "@lib/kmenu.ts";
export default function KMenuButton() {
return (
<Button
class="fixed bottom-4 right-4 md:hidden bg-gray-800 text-white p-3 rounded-full shadow-lg z-50"
onClick={() => (isKMenuOpen.value = true)}
>
<IconMenu2 class="w-6 h-6" />
</Button>
);
}

View File

@@ -1,18 +1,20 @@
import { useEffect } from "preact/hooks"; import { useEffect } from "preact/hooks";
declare global { declare global {
// deno-lint-ignore no-var
var loadingTimeout: ReturnType<typeof setTimeout> | undefined; var loadingTimeout: ReturnType<typeof setTimeout> | undefined;
} }
export function Link( export function Link(
{ href, children, class: _class, style }: { props: {
href?: string; href?: string;
class?: string; class?: string;
style?: preact.JSX.CSSProperties; style?: preact.CSSProperties;
children: preact.ComponentChildren; children: preact.ComponentChildren;
"data-thumb"?: string;
}, },
) { ) {
const { href, children, class: _class, style } = props;
const thumbhash = props["data-thumb"];
function handleClick() { function handleClick() {
if (globalThis.loadingTimeout) { if (globalThis.loadingTimeout) {
return; return;
@@ -41,6 +43,7 @@ export function Link(
href={href} href={href}
style={style} style={style}
onClick={handleClick} onClick={handleClick}
data-thumb={thumbhash}
class={_class} class={_class}
> >
{children} {children}

View File

@@ -1,5 +1,6 @@
import { useCallback, useState } from "preact/hooks"; import { useCallback, useState } from "preact/hooks";
import { IconWand } from "@components/icons.tsx"; import { IconWand } from "@components/icons.tsx";
import { RecommendationResource } from "@lib/recommendation.ts";
type RecommendationState = "disabled" | "loading"; type RecommendationState = "disabled" | "loading";
@@ -11,7 +12,7 @@ export function Recommendations(
}, },
) { ) {
const [state, setState] = useState<RecommendationState>("disabled"); const [state, setState] = useState<RecommendationState>("disabled");
const [results, setResults] = useState(); const [results, setResults] = useState<RecommendationResource[]>();
const startFetch = useCallback( const startFetch = useCallback(
async () => { async () => {
@@ -44,9 +45,9 @@ export function Recommendations(
<div class="flex gap-5 items-center mb-4"> <div class="flex gap-5 items-center mb-4">
<img <img
class="w-12 h-12 rounded-full object-cover" class="w-12 h-12 rounded-full object-cover"
src={`https://image.tmdb.org/t/p/original${res.poster_path}`} src={`https://image.tmdb.org/t/p/original${res.id}`}
/> />
<p>{res.title}</p> <p>{res.id}</p>
</div> </div>
); );
})} })}
@@ -66,6 +67,7 @@ export function Recommendations(
{!results && state === "disabled" && {!results && state === "disabled" &&
( (
<button <button
type="submit"
onClick={startFetch} onClick={startFetch}
class="ml-8 mt-2 font-thin flex items-center gap-2 text-white my-2" class="ml-8 mt-2 font-thin flex items-center gap-2 text-white my-2"
> >

View File

@@ -2,15 +2,15 @@ import { useEffect, useRef } from "preact/hooks";
import useDebouncedCallback from "@lib/hooks/useDebouncedCallback.ts"; import useDebouncedCallback from "@lib/hooks/useDebouncedCallback.ts";
import { IconLoader2, IconSearch } from "@components/icons.tsx"; import { IconLoader2, IconSearch } from "@components/icons.tsx";
import { useEventListener } from "@lib/hooks/useEventListener.ts"; import { useEventListener } from "@lib/hooks/useEventListener.ts";
import { GenericResource } from "@lib/types.ts";
import { resources } from "@lib/resources.ts"; import { resources } from "@lib/resources.ts";
import { getCookie } from "@lib/string.ts"; import { getCookie } from "@lib/string.ts";
import { IS_BROWSER } from "$fresh/runtime.ts"; import { IS_BROWSER } from "fresh/runtime";
import Checkbox from "@components/Checkbox.tsx"; import Checkbox from "@components/Checkbox.tsx";
import { Rating } from "@components/Rating.tsx"; import { Rating } from "@components/Rating.tsx";
import { useSignal } from "@preact/signals"; import { useSignal } from "@preact/signals";
import Image from "@components/Image.tsx"; import Image from "@components/Image.tsx";
import { Emoji } from "@components/Emoji.tsx"; import { Emoji } from "@components/Emoji.tsx";
import { GenericResource, getNameOfResource } from "@lib/marka/schema.ts";
export async function fetchQueryResource(url: URL, type = "") { export async function fetchQueryResource(url: URL, type = "") {
const query = url.searchParams.get("q"); const query = url.searchParams.get("q");
@@ -23,7 +23,7 @@ export async function fetchQueryResource(url: URL, type = "") {
url.searchParams.set("status", "not-seen"); url.searchParams.set("status", "not-seen");
} }
if (type) { if (type) {
url.searchParams.set("types", type); url.searchParams.set("type", type);
} }
const response = await fetch(url); const response = await fetch(url);
const jsonData = await response.json(); const jsonData = await response.json();
@@ -33,28 +33,30 @@ export async function fetchQueryResource(url: URL, type = "") {
} }
} }
export const RedirectSearchHandler = () => { export function RedirectSearchHandler() {
if (getCookie("session_cookie")) { useEventListener("keydown", (e: KeyboardEvent) => {
useEventListener("keydown", (e: KeyboardEvent) => { if (getCookie("session_cookie")) {
if (e?.target?.nodeName == "INPUT") return; const target = e.target as HTMLInputElement;
if (target.nodeName == "INPUT") return;
if ( if (
e.key === "?" && e.key === "?" &&
globalThis.location.search === "" globalThis.location.search === ""
) { ) {
globalThis.location.href += "?q="; globalThis.location.href += "?q=";
} }
}, IS_BROWSER ? document?.body : undefined); }
} }, IS_BROWSER ? document?.body : undefined);
// deno-lint-ignore jsx-no-useless-fragment
return <></>; return <></>;
}; }
const SearchResultImage = ({ src }: { src: string }) => { const SearchResultImage = ({ src }: { src: string }) => {
return ( return (
<Image <Image
class="object-cover w-12 h-12 rounded-full" class="object-cover w-12 h-12 rounded-full"
width="50" width={100}
height="50" height={100}
src={src} src={src}
alt="preview image" alt="preview image"
/> />
@@ -67,8 +69,9 @@ export const SearchResultItem = (
showEmoji?: boolean; showEmoji?: boolean;
}, },
) => { ) => {
const resourceType = resources[item.type]; const resourceType =
const href = resourceType ? `${resourceType.link}/${item.id}` : ""; resources[item.content._type.toLowerCase() as keyof typeof resources];
const href = item?.path.replace("/resources", "").replace(/\.md$/, "");
return ( return (
<a <a
href={href} href={href}
@@ -77,8 +80,8 @@ export const SearchResultItem = (
{showEmoji && resourceType {showEmoji && resourceType
? <Emoji class="w-7 h-7" name={resourceType.emoji} /> ? <Emoji class="w-7 h-7" name={resourceType.emoji} />
: ""} : ""}
{item.meta?.image && <SearchResultImage src={item.meta?.image} />} {item.image && <SearchResultImage src={item.image?.url} />}
{item?.name} {getNameOfResource(item)}
</a> </a>
); );
}; };
@@ -111,6 +114,7 @@ const Search = (
const searchQuery = useSignal(q); const searchQuery = useSignal(q);
const data = useSignal<GenericResource[] | undefined>(results); const data = useSignal<GenericResource[] | undefined>(results);
const isLoading = useSignal(false); const isLoading = useSignal(false);
const rating = useSignal<number | undefined>(undefined);
const showSeenStatus = useSignal(false); const showSeenStatus = useSignal(false);
const inputRef = useRef<HTMLInputElement>(null); const inputRef = useRef<HTMLInputElement>(null);
@@ -119,8 +123,10 @@ const Search = (
if (u.searchParams.get("q") !== searchQuery.value) { if (u.searchParams.get("q") !== searchQuery.value) {
u.searchParams.set("q", searchQuery.value); u.searchParams.set("q", searchQuery.value);
} }
if (showSeenStatus.value) { if (showSeenStatus.value === true) {
u.searchParams.set("rating", "0"); u.searchParams.set("rating", "0");
} else if (rating.value) {
u.searchParams.set("rating", rating.value.toString());
} else { } else {
u.searchParams.delete("rating"); u.searchParams.delete("rating");
} }
@@ -159,7 +165,7 @@ const Search = (
useEffect(() => { useEffect(() => {
debouncedFetchData(); // Call the debounced fetch function with the updated search query debouncedFetchData(); // Call the debounced fetch function with the updated search query
}, [searchQuery.value, showSeenStatus.value]); }, [searchQuery.value, showSeenStatus.value, rating.value]);
useEffect(() => { useEffect(() => {
debouncedFetchData(); debouncedFetchData();
@@ -184,8 +190,12 @@ const Search = (
onInput={handleInputChange} onInput={handleInputChange}
/> />
</div> </div>
<Checkbox label="seen" checked={showSeenStatus} /> <Checkbox label="unrated" checked={showSeenStatus} />
<Rating rating={4} /> <div
class={showSeenStatus.value ? "opacity-10" : ""}
>
<Rating rating={rating} />
</div>
</header> </header>
{data.value?.length && !isLoading.value {data.value?.length && !isLoading.value
? <SearchResultList showEmoji={!type} result={data.value} /> ? <SearchResultList showEmoji={!type} result={data.value} />

View File

@@ -1,4 +1,4 @@
import { OAuth2Client } from "https://deno.land/x/oauth2_client@v1.0.2/mod.ts"; import { OAuth2Client } from "@cmd-johnson/oauth2-client";
import { import {
GITEA_CLIENT_ID, GITEA_CLIENT_ID,
GITEA_CLIENT_SECRET, GITEA_CLIENT_SECRET,

View File

@@ -12,7 +12,9 @@ export const userTable = sqliteTable("user", {
id: text() id: text()
.primaryKey(), .primaryKey(),
createdAt: integer("created_at", { mode: "timestamp" }) createdAt: integer("created_at", { mode: "timestamp" })
.default(sql`(current_timestamp)`) .default(sql`
(CURRENT_TIMESTAMP)
`)
.notNull(), .notNull(),
email: text() email: text()
.notNull(), .notNull(),
@@ -24,7 +26,9 @@ export const sessionTable = sqliteTable("session", {
id: text("id") id: text("id")
.primaryKey(), .primaryKey(),
createdAt: integer("created_at", { mode: "timestamp_ms" }).default( createdAt: integer("created_at", { mode: "timestamp_ms" }).default(
sql`(current_timestamp)`, sql`
(CURRENT_TIMESTAMP)
`,
), ),
expiresAt: integer("expires_at", { mode: "timestamp" }) expiresAt: integer("expires_at", { mode: "timestamp" })
.notNull(), .notNull(),
@@ -38,16 +42,20 @@ export const performanceTable = sqliteTable("performance", {
time: int().notNull(), time: int().notNull(),
createdAt: integer("created_at", { createdAt: integer("created_at", {
mode: "timestamp_ms", mode: "timestamp_ms",
}).default(sql`(STRFTIME('%s', 'now') * 1000)`), }).default(sql`
(STRFTIME('%s', 'now') * 1000)
`),
}); });
export const imageTable = sqliteTable("image", { export const imageTable = sqliteTable("image", {
createdAt: integer("created_at", { mode: "timestamp" }).default( createdAt: integer("created_at", { mode: "timestamp" }).default(
sql`(current_timestamp)`, sql`
(unixepoch())
`,
), ),
url: text().notNull(), url: text().notNull(),
average: text().notNull(), average: text().notNull(),
blurhash: text().notNull(), thumbhash: text().notNull(),
mime: text().notNull(), mime: text().notNull(),
}); });
@@ -70,7 +78,9 @@ export const cacheTable = sqliteTable("cache", {
json: text({ mode: "json" }), json: text({ mode: "json" }),
binary: blob(), binary: blob(),
createdAt: integer("created_at", { mode: "timestamp" }).default( createdAt: integer("created_at", { mode: "timestamp" }).default(
sql`(current_timestamp)`, sql`
(CURRENT_TIMESTAMP)
`,
), ),
expiresAt: integer("expires_at", { mode: "timestamp" }), expiresAt: integer("expires_at", { mode: "timestamp" }),
}, (table) => { }, (table) => {

View File

@@ -4,7 +4,6 @@ import path from "node:path";
const DB_FILE = "file:" + path.resolve(DATA_DIR, "db.sqlite"); const DB_FILE = "file:" + path.resolve(DATA_DIR, "db.sqlite");
// You can specify any property from the libsql connection options
export const db = drizzle({ export const db = drizzle({
connection: { connection: {
url: DB_FILE, url: DB_FILE,

View File

@@ -7,6 +7,7 @@ export const PROXY_PASSWORD = Deno.env.get("PROXY_PASSWORD");
export const TMDB_API_KEY = Deno.env.get("TMDB_API_KEY"); export const TMDB_API_KEY = Deno.env.get("TMDB_API_KEY");
export const OPENAI_API_KEY = Deno.env.get("OPENAI_API_KEY"); export const OPENAI_API_KEY = Deno.env.get("OPENAI_API_KEY");
export const YOUTUBE_API_KEY = Deno.env.get("YOUTUBE_API_KEY"); export const YOUTUBE_API_KEY = Deno.env.get("YOUTUBE_API_KEY");
export const UNSPLASH_API_KEY = Deno.env.get("UNSPLASH_API_KEY");
export const TELEGRAM_API_KEY = Deno.env.get("TELEGRAM_API_KEY")!; export const TELEGRAM_API_KEY = Deno.env.get("TELEGRAM_API_KEY")!;
export const GITEA_SERVER = Deno.env.get("GITEA_SERVER"); export const GITEA_SERVER = Deno.env.get("GITEA_SERVER");
@@ -18,6 +19,7 @@ const duration = Deno.env.get("SESSION_DURATION");
export const SESSION_DURATION = duration ? +duration : (60 * 60 * 24); export const SESSION_DURATION = duration ? +duration : (60 * 60 * 24);
export const MARKA_API_KEY = Deno.env.get("MARKA_API_KEY"); export const MARKA_API_KEY = Deno.env.get("MARKA_API_KEY");
export const MARKA_API_URL = Deno.env.get("MARKA_API_URL");
export const JWT_SECRET = Deno.env.get("JWT_SECRET"); export const JWT_SECRET = Deno.env.get("JWT_SECRET");

View File

@@ -1,8 +1,9 @@
import { FreshContext } from "$fresh/server.ts"; import { Context } from "fresh";
import { State } from "../utils.ts";
class DomainError extends Error { class DomainError extends Error {
status = 500; status = 500;
render?: (ctx: FreshContext) => void; render?: (ctx: Context<State>) => void;
constructor(public statusText = "Internal Server Error") { constructor(public statusText = "Internal Server Error") {
super(); super();
} }

View File

@@ -31,19 +31,54 @@ export const fixRenderedMarkdown = (content: string) => {
}); });
}; };
export async function fetchStream(url: string, cb: (chunk: string) => void) { type StreamMessage = {
const response = await fetch(url); type: "info";
const reader = response?.body?.getReader(); message: string;
if (reader) { } | {
while (true) { type: "error";
const { done, value } = await reader.read(); message: string;
if (done) return; } | {
const data = new TextDecoder().decode(value); type: "warning";
data message: string;
.split("$") } | {
.filter((d) => d && d.length) type: "finished";
.map((d) => cb(Array.isArray(d) ? d[0] : d)); url: string;
} };
export async function fetchStream(
url: string,
cb: (chunk: StreamMessage) => void,
init?: RequestInit,
) {
const res = await fetch(url, init);
if (!res.body) return;
let buffer = "";
const reader = res.body
.pipeThrough(new TextDecoderStream())
.pipeThrough(
new TransformStream<string, string>({
transform(chunk, controller) {
buffer += chunk;
let idx;
while ((idx = buffer.indexOf("\n")) >= 0) {
const line = buffer.slice(0, idx).trim();
buffer = buffer.slice(idx + 1);
if (line) controller.enqueue(line);
}
},
flush(controller) {
const line = buffer.trim();
if (line) controller.enqueue(line);
},
}),
)
.getReader();
while (true) {
const { done, value } = await reader.read();
if (done) break;
cb(JSON.parse(value));
} }
} }
@@ -58,32 +93,53 @@ export function hashString(message: string) {
} }
export const createStreamResponse = () => { export const createStreamResponse = () => {
let controller: ReadableStreamController<ArrayBufferView>; const encoder = new TextEncoder();
const body = new ReadableStream({ let controller: ReadableStreamDefaultController<Uint8Array>;
start(cont) {
controller = cont; const body = new ReadableStream<Uint8Array>({
start(c) {
controller = c;
}, },
}); });
const response = new Response(body, { const response = new Response(body, {
headers: { headers: {
"content-type": "text/plain", // newline-delimited JSON
"content-type": "application/x-ndjson; charset=utf-8",
// prevent intermediaries from buffering/transforming
"cache-control": "no-cache, no-transform",
"x-content-type-options": "nosniff", "x-content-type-options": "nosniff",
// nginx hint to disable proxy buffering
"x-accel-buffering": "no",
// if you control compression, keep it off for streams
// "content-encoding": "identity",
}, },
}); });
function cancel() { const send = (obj: unknown) => {
controller.close(); controller.enqueue(encoder.encode(JSON.stringify(obj) + "\n")); // ← delimiter
};
const cancel = () => controller.close();
function info(message: string) {
return send({ type: "info", message });
} }
function enqueue(chunk: string) { function error(message: string) {
controller?.enqueue(new TextEncoder().encode("$" + chunk)); return send({ type: "error", message });
}
function warning(message: string) {
return send({ type: "warning", message });
} }
return { return {
response, response,
cancel, cancel,
enqueue, send,
info,
error,
warning,
}; };
}; };
@@ -102,16 +158,14 @@ export function debounce<T extends (...args: Parameters<T>) => void>(
} }
export function parseRating(rating: string | number) { export function parseRating(rating: string | number) {
if (typeof rating === "string") { if (typeof rating == "number") return rating;
try { try {
return parseInt(rating); const res = parseInt(rating);
} catch (_e) { if (!Number.isNaN(res)) return res;
// This is okay } catch (_e) {
} // This is okay
return [...rating.matchAll(/⭐/g)].length;
} }
return rating; return rating.length / 2;
} }
export async function convertOggToMp3( export async function convertOggToMp3(

View File

@@ -1,286 +1,93 @@
import { useEffect, useMemo, useRef } from "preact/hooks"; import { useEffect, useMemo, useRef } from "preact/hooks";
export interface CallOptions { type Debounced<T extends (...args: unknown[]) => unknown> =
/** & ((
* Controls if the function should be invoked on the leading edge of the timeout. ...args: Parameters<T>
*/ ) => void)
leading?: boolean; & {
/** cancel: () => void;
* Controls if the function should be invoked on the trailing edge of the timeout. flush: () => void;
*/ pending: () => boolean;
trailing?: boolean; };
}
export interface Options extends CallOptions {
/**
* The maximum time the given function is allowed to be delayed before it's invoked.
*/
maxWait?: number;
}
export interface ControlFunctions {
/**
* Cancel pending function invocations
*/
cancel: () => void;
/**
* Immediately invoke pending function invocations
*/
flush: () => void;
/**
* Returns `true` if there are any pending function invocations
*/
isPending: () => boolean;
}
/**
* Subsequent calls to the debounced function `debounced.callback` return the result of the last func invocation.
* Note, that if there are no previous invocations it's mean you will get undefined. You should check it in your code properly.
*/
export interface DebouncedState<T extends (...args: any) => ReturnType<T>>
extends ControlFunctions {
(...args: Parameters<T>): ReturnType<T> | undefined;
}
/**
* Creates a debounced function that delays invoking `func` until after `wait`
* milliseconds have elapsed since the last time the debounced function was
* invoked, or until the next browser frame is drawn.
*
* The debounced function comes with a `cancel` method to cancel delayed `func`
* invocations and a `flush` method to immediately invoke them.
*
* Provide `options` to indicate whether `func` should be invoked on the leading
* and/or trailing edge of the `wait` timeout. The `func` is invoked with the
* last arguments provided to the debounced function.
*
* Subsequent calls to the debounced function return the result of the last
* `func` invocation.
*
* **Note:** If `leading` and `trailing` options are `true`, `func` is
* invoked on the trailing edge of the timeout only if the debounced function
* is invoked more than once during the `wait` timeout.
*
* If `wait` is `0` and `leading` is `false`, `func` invocation is deferred
* until the next tick, similar to `setTimeout` with a timeout of `0`.
*
* If `wait` is omitted in an environment with `requestAnimationFrame`, `func`
* invocation will be deferred until the next frame is drawn (typically about
* 16ms).
*
* See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)
* for details over the differences between `debounce` and `throttle`.
*
* @category Function
* @param {Function} func The function to debounce.
* @param {number} [wait=0]
* The number of milliseconds to delay; if omitted, `requestAnimationFrame` is
* used (if available, otherwise it will be setTimeout(...,0)).
* @param {Object} [options={}] The options object.
* Controls if `func` should be invoked on the leading edge of the timeout.
* @param {boolean} [options.leading=false]
* The maximum time `func` is allowed to be delayed before it's invoked.
* @param {number} [options.maxWait]
* Controls if `func` should be invoked the trailing edge of the timeout.
* @param {boolean} [options.trailing=true]
* @returns {Function} Returns the new debounced function.
* @example
*
* // Avoid costly calculations while the window size is in flux.
* const resizeHandler = useDebouncedCallback(calculateLayout, 150);
* window.addEventListener('resize', resizeHandler)
*
* // Invoke `sendMail` when clicked, debouncing subsequent calls.
* const clickHandler = useDebouncedCallback(sendMail, 300, {
* leading: true,
* trailing: false,
* })
* <button onClick={clickHandler}>click me</button>
*
* // Ensure `batchLog` is invoked once after 1 second of debounced calls.
* const debounced = useDebouncedCallback(batchLog, 250, { 'maxWait': 1000 })
* const source = new EventSource('/stream')
* source.addEventListener('message', debounced)
*
* // Cancel the trailing debounced invocation.
* window.addEventListener('popstate', debounced.cancel)
*
* // Check for pending invocations.
* const status = debounced.pending() ? "Pending..." : "Ready"
*/
export default function useDebouncedCallback< export default function useDebouncedCallback<
T extends (...args: any) => ReturnType<T>, T extends (...args: unknown[]) => unknown,
>( >(
func: T, callback: T,
wait?: number, delay: number,
options?: Options, options?: {
): DebouncedState<T> { /** Call on the leading edge. Default: false */
const lastCallTime = useRef(null); leading?: boolean;
const lastInvokeTime = useRef(0); /** Call on the trailing edge. Default: true */
const timerId = useRef(null); trailing?: boolean;
const lastArgs = useRef<unknown[]>([]); },
const lastThis = useRef<unknown>(); ): Debounced<T> {
const result = useRef<ReturnType<T>>(); const callbackRef = useRef(callback);
const funcRef = useRef(func); const timerRef = useRef<number | null>(null);
const mounted = useRef(true); const argsRef = useRef<Parameters<T> | null>(null);
// Always use the latest callback without re-creating the debounced fn
useEffect(() => { useEffect(() => {
funcRef.current = func; callbackRef.current = callback;
}, [func]); }, [callback]);
// Bypass `requestAnimationFrame` by explicitly setting `wait=0`. const leading = !!options?.leading;
const useRAF = !wait && wait !== 0 && typeof window !== "undefined"; const trailing = options?.trailing !== false; // default true
if (typeof func !== "function") { const debounced = useMemo<Debounced<T>>(() => {
throw new TypeError("Expected a function"); const clear = () => {
} if (timerRef.current != null) {
clearTimeout(timerRef.current);
wait = +wait || 0; timerRef.current = null;
options = options || {};
const leading = !!options.leading;
const trailing = "trailing" in options ? !!options.trailing : true; // `true` by default
const maxing = "maxWait" in options;
const maxWait = maxing ? Math.max(+options.maxWait || 0, wait) : null;
useEffect(() => {
mounted.current = true;
return () => {
mounted.current = false;
};
}, []);
// You may have a question, why we have so many code under the useMemo definition.
//
// This was made as we want to escape from useCallback hell and
// not to initialize a number of functions each time useDebouncedCallback is called.
//
// It means that we have less garbage for our GC calls which improves performance.
// Also, it makes this library smaller.
//
// And the last reason, that the code without lots of useCallback with deps is easier to read.
// You have only one place for that.
const debounced = useMemo(() => {
const invokeFunc = (time: number) => {
const args = lastArgs.current;
const thisArg = lastThis.current;
lastArgs.current = lastThis.current = null;
lastInvokeTime.current = time;
return (result.current = funcRef.current.apply(thisArg, args));
};
const startTimer = (pendingFunc: () => void, wait: number) => {
if (useRAF) cancelAnimationFrame(timerId.current);
timerId.current = useRAF
? requestAnimationFrame(pendingFunc)
: setTimeout(pendingFunc, wait);
};
const shouldInvoke = (time: number) => {
if (!mounted.current) return false;
const timeSinceLastCall = time - lastCallTime.current;
const timeSinceLastInvoke = time - lastInvokeTime.current;
// Either this is the first call, activity has stopped and we're at the
// trailing edge, the system time has gone backwards and we're treating
// it as the trailing edge, or we've hit the `maxWait` limit.
return (
!lastCallTime.current ||
timeSinceLastCall >= wait ||
timeSinceLastCall < 0 ||
(maxing && timeSinceLastInvoke >= maxWait)
);
};
const trailingEdge = (time: number) => {
timerId.current = null;
// Only invoke if we have `lastArgs` which means `func` has been
// debounced at least once.
if (trailing && lastArgs.current) {
return invokeFunc(time);
} }
lastArgs.current = lastThis.current = null;
return result.current;
}; };
const timerExpired = () => { const invoke = () => {
const time = Date.now(); const a = argsRef.current;
if (shouldInvoke(time)) { argsRef.current = null;
return trailingEdge(time); if (a) {
callbackRef.current(...a);
} }
// https://github.com/xnimorz/use-debounce/issues/97 };
if (!mounted.current) {
return; const fn = ((...args: Parameters<T>) => {
const shouldCallLeading = leading && timerRef.current == null;
argsRef.current = args;
if (timerRef.current != null) clearTimeout(timerRef.current);
timerRef.current = globalThis.setTimeout(() => {
timerRef.current = null;
if (trailing) invoke();
}, delay);
if (shouldCallLeading) {
// Leading edge call happens immediately
invoke();
} }
// Remaining wait calculation }) as Debounced<T>;
const timeSinceLastCall = time - lastCallTime.current;
const timeSinceLastInvoke = time - lastInvokeTime.current;
const timeWaiting = wait - timeSinceLastCall;
const remainingWait = maxing
? Math.min(timeWaiting, maxWait - timeSinceLastInvoke)
: timeWaiting;
// Restart the timer fn.cancel = () => {
startTimer(timerExpired, remainingWait); argsRef.current = null;
clear();
}; };
const func: DebouncedState<T> = (...args: Parameters<T>): ReturnType<T> => { fn.flush = () => {
const time = Date.now(); if (timerRef.current != null) {
const isInvoking = shouldInvoke(time); clear();
invoke();
lastArgs.current = args;
lastThis.current = this;
lastCallTime.current = time;
if (isInvoking) {
if (!timerId.current && mounted.current) {
// Reset any `maxWait` timer.
lastInvokeTime.current = lastCallTime.current;
// Start the timer for the trailing edge.
startTimer(timerExpired, wait);
// Invoke the leading edge.
return leading ? invokeFunc(lastCallTime.current) : result.current;
}
if (maxing) {
// Handle invocations in a tight loop.
startTimer(timerExpired, wait);
return invokeFunc(lastCallTime.current);
}
} }
if (!timerId.current) {
startTimer(timerExpired, wait);
}
return result.current;
}; };
func.cancel = () => { fn.pending = () => timerRef.current != null;
if (timerId.current) {
useRAF
? cancelAnimationFrame(timerId.current)
: clearTimeout(timerId.current);
}
lastInvokeTime.current = 0;
lastArgs.current =
lastCallTime.current =
lastThis.current =
timerId.current =
null;
};
func.isPending = () => { return fn;
return !!timerId.current; // Recreate only if timing/edge behavior changes
}; }, [delay, leading, trailing]);
func.flush = () => { // Cancel on unmount
return !timerId.current ? result.current : trailingEdge(Date.now()); useEffect(() => () => debounced.cancel(), [debounced]);
};
return func;
}, [leading, maxing, wait, maxWait, trailing, useRAF]);
return debounced; return debounced;
} }

View File

@@ -6,7 +6,7 @@ export function useEventListener<T extends Event>(
element: typeof globalThis | HTMLElement = globalThis, element: typeof globalThis | HTMLElement = globalThis,
) { ) {
// Create a ref that stores handler // Create a ref that stores handler
const savedHandler = useRef<(event: Event) => void>(); const savedHandler = useRef<(event: T) => void>();
// Update ref.current value if handler changes. // Update ref.current value if handler changes.
// This allows our effect below to always get latest handler ... // This allows our effect below to always get latest handler ...
@@ -27,11 +27,11 @@ export function useEventListener<T extends Event>(
const eventListener = (event: T) => savedHandler?.current?.(event); const eventListener = (event: T) => savedHandler?.current?.(event);
// Add event listener // Add event listener
element.addEventListener(eventName, eventListener); element.addEventListener(eventName, (ev) => eventListener(ev as T));
// Remove event listener on cleanup // Remove event listener on cleanup
return () => { return () => {
element.removeEventListener(eventName, eventListener); element.removeEventListener(eventName, (ev) => eventListener(ev as T));
}; };
}, },
[eventName, element], // Re-run if eventName or element changes [eventName, element], // Re-run if eventName or element changes

View File

@@ -6,7 +6,7 @@ type ThrottleOptions = {
}; };
const useThrottledCallback = ( const useThrottledCallback = (
callback: (...args: any[]) => void, callback: (...args: unknown[]) => void,
delay: number, delay: number,
options: ThrottleOptions = {}, options: ThrottleOptions = {},
) => { ) => {
@@ -24,7 +24,7 @@ const useThrottledCallback = (
}; };
}, [timer]); }, [timer]);
const throttledCallback = (...args: any[]) => { const throttledCallback = (...args: unknown[]) => {
const now = Date.now(); const now = Date.now();
if (leading && !isLeading) { if (leading && !isLeading) {
@@ -52,4 +52,3 @@ const useThrottledCallback = (
}; };
export default useThrottledCallback; export default useThrottledCallback;

View File

@@ -1,14 +1,17 @@
import { useEffect, useRef } from "preact/hooks"; import { useEffect, useRef } from "preact/hooks";
export function useTraceUpdate(props) { export function useTraceUpdate(props: Record<string, unknown>) {
const prev = useRef(props); const prev = useRef(props);
useEffect(() => { useEffect(() => {
const changedProps = Object.entries(props).reduce((ps, [k, v]) => { const changedProps = Object.entries(props).reduce(
if (prev.current[k] !== v) { (ps: Record<string, unknown>, [k, v]) => {
ps[k] = [prev.current[k], v]; if (prev.current[k] !== v) {
} ps[k] = [prev.current[k], v];
return ps; }
}, {}); return ps;
},
{},
);
if (Object.keys(changedProps).length > 0) { if (Object.keys(changedProps).length > 0) {
console.log("Changed props:", changedProps); console.log("Changed props:", changedProps);
} }

View File

@@ -1,19 +1,18 @@
import { rgbToHex } from "@lib/string.ts"; import { rgbToHex } from "@lib/string.ts";
import { createLogger } from "@lib/log/index.ts"; import { createLogger } from "@lib/log/index.ts";
import { generateThumbhash } from "@lib/thumbhash.ts"; import { generateThumbhash } from "@lib/thumbhash.ts";
import { parseMediaType } from "https://deno.land/std@0.224.0/media_types/parse_media_type.ts"; import { parseMediaType } from "@std/media-types";
import path from "node:path"; import path from "node:path";
import { ensureDir } from "fs"; import { mkdir } from "node:fs/promises";
import { DATA_DIR } from "@lib/env.ts"; import { DATA_DIR } from "@lib/env.ts";
import { db } from "@lib/db/sqlite.ts"; import { db } from "@lib/db/sqlite.ts";
import { imageTable } from "@lib/db/schema.ts"; import { imageTable } from "@lib/db/schema.ts";
import { eq } from "drizzle-orm"; import { eq } from "drizzle-orm";
import sharp from "npm:sharp@next";
const log = createLogger("cache/image"); const log = createLogger("cache/image");
const imageDir = path.join(DATA_DIR, "images"); const imageDir = path.join(DATA_DIR, "images");
await ensureDir(imageDir); await mkdir(imageDir, { recursive: true });
async function getRemoteImage(imageUrl: string) { async function getRemoteImage(imageUrl: string) {
try { try {
@@ -100,7 +99,7 @@ async function getLocalImagePath(
hostname, hostname,
pathname.split("/").filter((s) => s.length).join("-"), pathname.split("/").filter((s) => s.length).join("-"),
); );
await ensureDir(imagePath); await mkdir(imagePath, { recursive: true });
if (width || height) { if (width || height) {
imagePath = path.join(imagePath, `${width ?? "-"}x${height ?? "-"}`); imagePath = path.join(imagePath, `${width ?? "-"}x${height ?? "-"}`);
@@ -134,7 +133,7 @@ async function getLocalImage(
*/ */
async function storeLocalImage( async function storeLocalImage(
url: string, url: string,
content: ArrayBuffer, content: Uint8Array<ArrayBuffer> | ArrayBuffer,
{ width, height }: { width?: number; height?: number } = {}, { width, height }: { width?: number; height?: number } = {},
) { ) {
const isValid = await verifyImage(new Uint8Array(content)); const isValid = await verifyImage(new Uint8Array(content));
@@ -158,6 +157,8 @@ async function resizeImage(
mediaType: string; mediaType: string;
}, },
) { ) {
const sharp = (await import("sharp")).default;
try { try {
log.debug("Resizing image", { params }); log.debug("Resizing image", { params });
@@ -211,6 +212,8 @@ async function resizeImage(
async function createThumbhash( async function createThumbhash(
image: Uint8Array, image: Uint8Array,
): Promise<{ hash: string; average: string }> { ): Promise<{ hash: string; average: string }> {
const sharp = (await import("sharp")).default;
try { try {
const resizedImage = await sharp(image) const resizedImage = await sharp(image)
.resize(100, 100, { fit: "cover" }) // Keep aspect ratio within bounds .resize(100, 100, { fit: "cover" }) // Keep aspect ratio within bounds
@@ -219,7 +222,11 @@ async function createThumbhash(
.raw() .raw()
.toBuffer(); .toBuffer();
const [hash, average] = generateThumbhash(resizedImage, 100, 100); const [hash, average] = generateThumbhash(
new Uint8Array(resizedImage),
100,
100,
);
return { return {
hash: btoa(String.fromCharCode(...hash)), hash: btoa(String.fromCharCode(...hash)),
@@ -234,6 +241,8 @@ async function createThumbhash(
* Verifies that an image buffer contains valid image data * Verifies that an image buffer contains valid image data
*/ */
async function verifyImage(imageBuffer: Uint8Array): Promise<boolean> { async function verifyImage(imageBuffer: Uint8Array): Promise<boolean> {
const sharp = (await import("sharp")).default;
try { try {
const metadata = await sharp(imageBuffer).metadata(); const metadata = await sharp(imageBuffer).metadata();
return !!(metadata.width && metadata.height && metadata.format); return !!(metadata.width && metadata.height && metadata.format);
@@ -249,7 +258,7 @@ async function verifyImage(imageBuffer: Uint8Array): Promise<boolean> {
export async function getImageContent( export async function getImageContent(
url: string, url: string,
{ width, height }: { width?: number; height?: number } = {}, { width, height }: { width?: number; height?: number } = {},
): Promise<{ content: ArrayBuffer; mimeType: string }> { ): Promise<{ content: Uint8Array<ArrayBuffer>; mimeType: string }> {
log.debug("Getting image content", { url, width, height }); log.debug("Getting image content", { url, width, height });
// Check if we have the image metadata in database // Check if we have the image metadata in database
@@ -267,8 +276,8 @@ export async function getImageContent(
// Fetch and cache original if needed // Fetch and cache original if needed
if (!originalImage) { if (!originalImage) {
const fetchedImage = await getRemoteImage(url); const fetchedImage = await getRemoteImage(url);
await storeLocalImage(url, fetchedImage.buffer);
originalImage = new Uint8Array(fetchedImage.buffer); originalImage = new Uint8Array(fetchedImage.buffer);
await storeLocalImage(url, originalImage);
} }
// Resize image // Resize image
@@ -313,7 +322,7 @@ export async function getImage(url: string) {
// Store in database // Store in database
const [newImage] = await db.insert(imageTable).values({ const [newImage] = await db.insert(imageTable).values({
url: url, url: url,
blurhash: thumbhash.hash, thumbhash: thumbhash.hash,
average: thumbhash.average, average: thumbhash.average,
mime: imageContent.mediaType, mime: imageContent.mediaType,
}).returning(); }).returning();

3
lib/kmenu.ts Normal file
View File

@@ -0,0 +1,3 @@
import { signal } from "@preact/signals";
export const isKMenuOpen = signal(false);

View File

@@ -1,12 +1,12 @@
import * as env from "@lib/env.ts"; import * as env from "@lib/env.ts";
import { ensureDir } from "fs";
import { join } from "node:path"; import { join } from "node:path";
import { getLogLevel, LOG_LEVEL } from "@lib/log/types.ts"; import { getLogLevel, LOG_LEVEL } from "@lib/log/types.ts";
import { mkdir } from "node:fs/promises";
export const LOG_DIR = join(env.DATA_DIR, "logs"); export const LOG_DIR = join(env.DATA_DIR, "logs");
// Ensure the log directory exists // Ensure the log directory exists
await ensureDir(LOG_DIR); await mkdir(LOG_DIR, { recursive: true });
export let logLevel = getLogLevel(env.LOG_LEVEL); export let logLevel = getLogLevel(env.LOG_LEVEL);
export function setLogLevel(level: LOG_LEVEL) { export function setLogLevel(level: LOG_LEVEL) {

View File

@@ -36,12 +36,12 @@ export async function getLogs() {
.map((line) => { .map((line) => {
const [date, ...rest] = line.split(" | "); const [date, ...rest] = line.split(" | ");
const parsed = JSON.parse(rest.join(" | ")) as Log; const parsed = JSON.parse(rest.join(" | ")) as Log;
const dateObj = new Date(date);
return { return {
...parsed, ...parsed,
date: new Date(date), date: isNaN(dateObj.getTime()) ? new Date() : dateObj,
} as Log; } as Log;
}); });
console.log(logs);
// Return the logs sorted by date // Return the logs sorted by date
return logs.sort((a, b) => a.date.getTime() - b.date.getTime()); return logs.sort((a, b) => a.date.getTime() - b.date.getTime());

View File

@@ -38,13 +38,13 @@ export function createLogger(scope: string, _options?: LoggerOptions): Logger {
export function loggerFromStream(stream: StreamResponse) { export function loggerFromStream(stream: StreamResponse) {
return { return {
debug: (...data: unknown[]) => debug: (...data: unknown[]) =>
stream.enqueue(`${data.length > 1 ? data.join(" ") : data[0]}`), stream.info(`${data.length > 1 ? data.join(" ") : data[0]}`),
info: (...data: unknown[]) => info: (...data: unknown[]) =>
stream.enqueue(`${data.length > 1 ? data.join(" ") : data[0]}`), stream.info(`${data.length > 1 ? data.join(" ") : data[0]}`),
error: (...data: unknown[]) => error: (...data: unknown[]) =>
stream.enqueue(`[ERROR]: ${data.length > 1 ? data.join(" ") : data[0]}`), stream.error(`[ERROR]: ${data.length > 1 ? data.join(" ") : data[0]}`),
warn: (...data: unknown[]) => warn: (...data: unknown[]) =>
stream.enqueue(`[WARN]: ${data.length > 1 ? data.join(" ") : data[0]}`), stream.warning(`[WARN]: ${data.length > 1 ? data.join(" ") : data[0]}`),
}; };
} }

View File

@@ -1,35 +0,0 @@
import { MARKA_API_KEY } from "./env.ts";
const url = `https://marka.max-richter.dev/resources`;
//const url = "http://localhost:8080/resources";
export async function fetchResource(resource: string) {
try {
const response = await fetch(
`${url}/${resource}`,
);
return response.json();
} catch (_e) {
return [];
}
}
export async function createResource(
path: string,
content: string | object | ArrayBuffer,
) {
const isJson = typeof content === "object" &&
!(content instanceof ArrayBuffer);
const fetchUrl = `${url}/${path}`;
const response = await fetch(fetchUrl, {
method: "POST",
headers: {
"Content-Type": isJson ? "application/json" : "",
"Authentication": MARKA_API_KEY,
},
body: isJson ? JSON.stringify(content) : content,
});
if (!response.ok) {
throw new Error(`Failed to create resource: ${response.status}`);
}
return response.json();
}

117
lib/marka/index.ts Normal file
View File

@@ -0,0 +1,117 @@
import { createCache } from "../cache.ts";
import { MARKA_API_KEY, MARKA_API_URL } from "../env.ts";
import { getImage } from "../image.ts";
import { GenericResource } from "./schema.ts";
async function addImageToResource<T extends GenericResource>(
resource: GenericResource,
): Promise<T> {
const imageUrl = resource?.content?.image;
if (imageUrl) {
try {
const absoluteImageUrl = (imageUrl.startsWith("https://") ||
imageUrl.startsWith("http://"))
? imageUrl
: `${MARKA_API_URL}/${imageUrl}`;
const image = await getImage(absoluteImageUrl);
return { ...resource, image } as T;
} catch (e) {
console.log(`Failed to fetch image: ${imageUrl}`, e);
}
}
return resource as T;
}
type Resource = GenericResource & {
content: GenericResource["content"] & Array<GenericResource>;
};
const fetchCache = createCache<Resource>("marka");
const cacheLock = new Map<string, Promise<Resource>>();
async function fetchAndStoreUrl(url: string) {
const response = await fetch(url);
if (!response.ok) {
throw new Error(`Failed to fetch resource: ${response.status}`);
}
const res = await response.json();
fetchCache.set(url, res);
return res;
}
async function cachedFetch(
url: string,
): Promise<Resource | undefined> {
if (fetchCache.has(url)) {
fetchAndStoreUrl(url); // Fetch the url in the background
return fetchCache.get(url);
}
if (cacheLock.has(url)) return cacheLock.get(url);
const response = fetchAndStoreUrl(url);
cacheLock.set(url, response);
const res = await response;
cacheLock.delete(url);
return res;
}
export async function fetchResource<T extends GenericResource>(
resource: string,
): Promise<T | undefined> {
try {
const d = `${MARKA_API_URL}/resources/${resource}`;
const res = await cachedFetch(d);
if (!res) return;
return addImageToResource<T>(res);
} catch (_e) {
return;
}
}
export async function listResources<T extends GenericResource>(
resource: string,
): Promise<T[]> {
try {
const d = `${MARKA_API_URL}/resources/${resource}`;
const list = await cachedFetch(d);
if (!list) return [];
return Promise.all(
list?.content
.filter((a) => a?.content?._type)
.map((res) => addImageToResource(res) as Promise<T>),
);
} catch (_e) {
console.log(`Failed to fetch resource: ${resource}`, _e);
return [];
}
}
export async function createResource(
path: string,
content: string | object | ArrayBuffer,
) {
const isJson = typeof content === "object" &&
!(content instanceof ArrayBuffer);
const fetchUrl = `${MARKA_API_URL}/resources/${path}`;
const headers = new Headers();
headers.append("Content-Type", isJson ? "application/json" : "");
if (MARKA_API_KEY) {
headers.append("Authentication", MARKA_API_KEY);
}
const response = await fetch(fetchUrl, {
method: "POST",
headers,
body: isJson ? JSON.stringify(content) : content,
});
if (!response.ok) {
const text = await response.text();
throw new Error(
`failed to create resource (resources/${path}): ${
text || response.status
}`,
);
}
return response.json();
}

140
lib/marka/schema.ts Normal file
View File

@@ -0,0 +1,140 @@
import { z } from "zod";
import { imageTable } from "../db/schema.ts";
export const PersonSchema = z.object({
_type: z.literal("Person"),
name: z.string().optional(),
});
export const ReviewRatingSchema = z.object({
bestRating: z.number().optional(),
worstRating: z.number().optional(),
// Accept number or string (e.g., "⭐️⭐️⭐️⭐️⭐️")
ratingValue: z.union([z.number(), z.string()]).optional(),
});
const WithAuthor = z.object({ author: PersonSchema.optional() });
const WithKeywords = z.object({ keywords: z.array(z.string()).optional() });
const WithImage = z.object({ image: z.string().optional() });
const WithDatePublished = z.object({ datePublished: z.string().optional() });
const BaseContent = WithAuthor.merge(WithKeywords)
.merge(WithImage)
.merge(WithDatePublished);
export const BaseFileSchema = z.object({
type: z.literal("file"),
name: z.string(),
path: z.string(),
modTime: z.string(), // ISO timestamp string
mime: z.string(),
size: z.number().int().nonnegative(),
});
const makeContentSchema = <
TName extends "Article" | "Review" | "Recipe",
TShape extends z.ZodRawShape,
>(
name: TName,
shape: TShape,
) =>
z
.object({
_type: z.literal(name),
keywords: z.array(z.string()).optional(),
})
.merge(BaseContent)
.extend(shape);
export const ArticleContentSchema = makeContentSchema("Article", {
headline: z.string().optional(),
articleBody: z.string().optional(),
url: z.string().optional(),
reviewRating: ReviewRatingSchema.optional(),
});
export const ReviewContentSchema = makeContentSchema("Review", {
tmdbId: z.number().optional(),
link: z.string().optional(),
reviewRating: ReviewRatingSchema.optional(),
reviewBody: z.string().optional(),
itemReviewed: z
.object({
name: z.string().optional(),
})
.optional(),
});
export const RecipeContentSchema = makeContentSchema("Recipe", {
description: z.string().optional(),
name: z.string().optional(),
reviewRating: ReviewRatingSchema.optional(),
recipeIngredient: z.array(z.string()).optional(),
recipeInstructions: z.array(z.string()).optional(),
totalTime: z.string().optional(),
recipeYield: z.number().optional(),
url: z.string().optional(),
});
export const articleMetadataSchema = z.object({
headline: z.union([z.null(), z.string()]).describe("Headline of the article"),
author: z.union([z.null(), z.string()]).describe("Author of the article"),
datePublished: z.union([z.null(), z.string()]).describe(
"Date the article was published",
),
keywords: z.union([z.null(), z.array(z.string())]).describe(
"Keywords for the article",
),
});
export const ArticleSchema = BaseFileSchema.extend({
content: ArticleContentSchema,
});
export const ReviewSchema = BaseFileSchema.extend({
content: ReviewContentSchema,
});
export const RecipeSchema = BaseFileSchema.extend({
content: RecipeContentSchema,
});
export const GenericResourceSchema = z.union([
ArticleSchema,
ReviewSchema,
RecipeSchema,
]);
export type Person = z.infer<typeof PersonSchema>;
export type ReviewRating = z.infer<typeof ReviewRatingSchema>;
export type BaseFile = z.infer<typeof BaseFileSchema>;
export type ArticleResource = z.infer<typeof ArticleSchema> & {
image?: typeof imageTable.$inferSelect;
};
export type ReviewResource = z.infer<typeof ReviewSchema> & {
image?: typeof imageTable.$inferSelect;
};
export type RecipeResource = z.infer<typeof RecipeSchema> & {
image?: typeof imageTable.$inferSelect;
};
export type GenericResource = z.infer<typeof GenericResourceSchema> & {
image?: typeof imageTable.$inferSelect;
};
export function getNameOfResource(res: GenericResource): string {
if (res.content?._type === "Article" && res.content.headline) {
return res.content.headline;
}
if (res.content?._type === "Review" && res.content.itemReviewed?.name) {
return res.content.itemReviewed.name;
}
if (res.content?._type === "Recipe" && res.content.name) {
return res.content.name;
}
return "Unnamed Resource";
}

View File

@@ -1,7 +1,7 @@
import { render } from "gfm"; import { render } from "gfm";
import "https://esm.sh/prismjs@1.29.0/components/prism-typescript?no-check"; import "prismjs/components/prism-typescript.js";
import "https://esm.sh/prismjs@1.29.0/components/prism-bash?no-check"; import "prismjs/components/prism-bash.js";
import "https://esm.sh/prismjs@1.29.0/components/prism-rust?no-check"; import "prismjs/components/prism-rust.js";
export type Document = { export type Document = {
name: string; name: string;
@@ -44,11 +44,3 @@ export function renderMarkdown(doc: string) {
allowMath: true, allowMath: true,
}); });
} }
export function createDocument(
path: string,
entry: string,
mimetype = "image/jpeg",
) {
console.log("creating", { path, entry, mimetype });
}

View File

@@ -1,14 +1,14 @@
import OpenAI from "https://deno.land/x/openai@v4.69.0/mod.ts"; import OpenAI, { toFile } from "@openai/openai";
import { zodResponseFormat } from "https://deno.land/x/openai@v4.69.0/helpers/zod.ts"; import { zodResponseFormat } from "@openai/openai/helpers/zod";
import { OPENAI_API_KEY } from "@lib/env.ts"; import { OPENAI_API_KEY } from "@lib/env.ts";
import { hashString } from "@lib/helpers.ts"; import { hashString } from "@lib/helpers.ts";
import { createCache } from "@lib/cache.ts"; import { createCache } from "@lib/cache.ts";
import { recipeResponseSchema } from "@lib/recipeSchema.ts"; import { recipeResponseSchema } from "@lib/recipeSchema.ts";
import { articleMetadataSchema } from "./resource/articles.ts"; import { articleMetadataSchema } from "./marka/schema.ts";
const openAI = OPENAI_API_KEY && new OpenAI({ apiKey: OPENAI_API_KEY }); const openAI = OPENAI_API_KEY && new OpenAI({ apiKey: OPENAI_API_KEY });
interface MovieRecommendation { export interface MovieRecommendation {
year: number; year: number;
title: string; title: string;
} }
@@ -35,7 +35,8 @@ export async function summarize(content: string) {
{ {
role: "user", role: "user",
content: content:
`Please summarize the article in one sentence as short as possible: ${content.slice(0, 2000) `Please summarize the article in one sentence as short as possible: ${
content.slice(0, 2000)
}`, }`,
}, },
], ],
@@ -102,7 +103,8 @@ export async function createGenres(
{ {
role: "system", role: "system",
content: content:
`you create some keywords that can be used in a recommendation system. The keywords are based on a ${type} description or title. If you do not know the title, take into account the description aswell. Create a range of keywords from very specific ones that describe the general vibe. ${title ? `The name of the ${type} is ${title}` : "" `you create some keywords that can be used in a recommendation system. The keywords are based on a ${type} description or title. If you do not know the title, take into account the description aswell. Create a range of keywords from very specific ones that describe the general vibe. ${
title ? `The name of the ${type} is ${title}` : ""
}. Return a list of around 20 keywords seperated by commas`, }. Return a list of around 20 keywords seperated by commas`,
}, },
{ {
@@ -166,7 +168,8 @@ export const getMovieRecommendations = async (
${keywords} ${keywords}
The movies should be similar to but not include ${exclude.join(", ") The movies should be similar to but not include ${
exclude.join(", ")
} or remakes of that. } or remakes of that.
respond with a plain unordered list each item starting with the year the movie was released and then the title of the movie seperated by a -`, respond with a plain unordered list each item starting with the year the movie was released and then the title of the movie seperated by a -`,
@@ -178,20 +181,37 @@ respond with a plain unordered list each item starting with the year the movie w
if (!res) return; if (!res) return;
const recommendations = res.split("\n").map((entry) => { const recommendations = res.split("\n").map((entry: string) => {
const [year, ...title] = entry.split("-"); const [year, ...title] = entry.split("-");
return { return {
year: parseInt(year.trim()), year: parseInt(year.trim()),
title: title.join(" ").replaceAll('"', "").trim(), title: title.join(" ").replaceAll('"', "").trim(),
}; };
}).filter((y) => !Number.isNaN(y.year)); }).filter((y: { year: number }) => !Number.isNaN(y.year));
cache.set(cacheId, recommendations); cache.set(cacheId, recommendations);
return recommendations; return recommendations;
}; };
export async function createUnsplashSearchTerm(content: string) {
if (!openAI) return;
const chatCompletion = await openAI.chat.completions.create({
model: model,
messages: [
{
role: "system",
content:
"Please respond with a search term for unsplash for the following article",
},
{ role: "user", content: content.slice(0, 10_000) },
],
});
return chatCompletion.choices[0].message.content?.toLowerCase();
}
export async function createTags(content: string) { export async function createTags(content: string) {
if (!openAI) return; if (!openAI) return;
const chatCompletion = await openAI.chat.completions.create({ const chatCompletion = await openAI.chat.completions.create({
@@ -213,7 +233,7 @@ export async function createTags(content: string) {
export async function extractRecipe(content: string) { export async function extractRecipe(content: string) {
if (!openAI) return; if (!openAI) return;
const completion = await openAI.beta.chat.completions.parse({ const completion = await openAI.chat.completions.parse({
model: model, model: model,
temperature: 0.1, temperature: 0.1,
messages: [ messages: [
@@ -231,7 +251,7 @@ export async function extractRecipe(content: string) {
export async function extractArticleMetadata(content: string) { export async function extractArticleMetadata(content: string) {
if (!openAI) return; if (!openAI) return;
const completion = await openAI.beta.chat.completions.parse({ const completion = await openAI.chat.completions.parse({
model: model, model: model,
temperature: 0.1, temperature: 0.1,
messages: [ messages: [
@@ -256,7 +276,7 @@ export async function transcribe(
): Promise<string | undefined> { ): Promise<string | undefined> {
if (!openAI) return; if (!openAI) return;
const file = new File([mp3Data], "audio.mp3", { const file = await toFile(mp3Data, "audio.mp3", {
type: "audio/mpeg", type: "audio/mpeg",
}); });

View File

@@ -1,7 +1,7 @@
import { import {
parseIngredient, parseIngredient,
unitsOfMeasure as _unitsOfMeasure, unitsOfMeasure as _unitsOfMeasure,
} from "https://esm.sh/parse-ingredient@1.2.1"; } from "parse-ingredient";
import { Ingredient, IngredientGroup } from "@lib/recipeSchema.ts"; import { Ingredient, IngredientGroup } from "@lib/recipeSchema.ts";
import { removeMarkdownFormatting } from "@lib/string.ts"; import { removeMarkdownFormatting } from "@lib/string.ts";
@@ -77,7 +77,7 @@ export function parseIngredients(
}; };
const unit = ingredient.unit.toLowerCase() as keyof typeof unitsOfMeasure; const unit = ingredient.unit.toLowerCase() as keyof typeof unitsOfMeasure;
if (unit in unitsOfMeasure && unit !== "cup") { if (unit in unitsOfMeasure) {
ingredient.unit = unitsOfMeasure[unit].short; ingredient.unit = unitsOfMeasure[unit].short;
} }

View File

@@ -1,6 +1,6 @@
import { firefox } from "npm:playwright-extra"; import { firefox } from "playwright-extra";
import { createStreamResponse } from "@lib/helpers.ts"; import { createStreamResponse } from "@lib/helpers.ts";
import StealthPlugin from "npm:puppeteer-extra-plugin-stealth"; import StealthPlugin from "puppeteer-extra-plugin-stealth";
import * as env from "@lib/env.ts"; import * as env from "@lib/env.ts";
firefox.use(StealthPlugin()); firefox.use(StealthPlugin());
@@ -9,7 +9,7 @@ export async function fetchHtmlWithPlaywright(
fetchUrl: string, fetchUrl: string,
streamResponse: ReturnType<typeof createStreamResponse>, streamResponse: ReturnType<typeof createStreamResponse>,
): Promise<string> { ): Promise<string> {
streamResponse.enqueue("booting up playwright"); streamResponse.info("booting up playwright");
const config: Parameters<typeof firefox.launch>[0] = {}; const config: Parameters<typeof firefox.launch>[0] = {};
if (env.PROXY_SERVER) { if (env.PROXY_SERVER) {
@@ -24,7 +24,7 @@ export async function fetchHtmlWithPlaywright(
// Launch the Playwright browser // Launch the Playwright browser
const browser = await firefox.launch(config); const browser = await firefox.launch(config);
streamResponse.enqueue("fetching html"); streamResponse.info("fetching html");
try { try {
// Open a new browser context and page // Open a new browser context and page
@@ -42,7 +42,7 @@ export async function fetchHtmlWithPlaywright(
return html; return html;
} catch (error) { } catch (error) {
streamResponse.enqueue("error fetching html"); streamResponse.error("error fetching html");
console.error(error); console.error(error);
return ""; return "";
} finally { } finally {

View File

@@ -1,105 +0,0 @@
/**
* Interface zur Beschreibung eines eingereihten Promises in der `PromiseQueue`.
*/
interface QueuedPromise<T = any> {
promise: () => Promise<T>;
resolve: (value: T) => void;
reject: (reason?: any) => void;
}
/**
* Eine einfache Promise Queue, die es ermöglicht mehrere Aufgaben in kontrollierter
* Reihenfolge abzuarbeiten.
*
* Lizenz: CC BY-NC-SA 4.0
* (c) Peter Müller <peter@crycode.de> (https://crycode.de/promise-queue-in-typescript)
*/
export class PromiseQueue {
/**
* Eingereihte Promises.
*/
private queue: QueuedPromise[] = [];
/**
* Indikator, dass aktuell ein Promise abgearbeitet wird.
*/
private working = false;
/**
* Ein Promise einreihen.
* Dies fügt das Promise der Warteschlange hinzu. Wenn die Warteschlange leer
* ist, dann wird das Promise sofort gestartet.
* @param promise Funktion, die das Promise zurückgibt.
* @returns Ein Promise, welches eingelöst (oder zurückgewiesen) wird sobald das eingereihte Promise abgearbeitet ist.
*/
public enqueue<T = void>(promise: () => Promise<T>): Promise<T> {
return new Promise((resolve, reject) => {
this.queue.push({
promise,
resolve,
reject,
});
this.dequeue();
});
}
/**
* Das erste Promise aus der Warteschlange holen und starten, sofern nicht
* bereits ein Promise aktiv ist.
* @returns `true` wenn ein Promise aus der Warteschlange gestartet wurde oder `false` wenn bereits ein Promise aktiv oder die Warteschlange leer ist.
*/
private dequeue(): boolean {
if (this.working) {
return false;
}
const item = this.queue.shift();
if (!item) {
return false;
}
try {
this.working = true;
item.promise()
.then((value) => {
item.resolve(value);
})
.catch((err) => {
item.reject(err);
})
.finally(() => {
this.working = false;
this.dequeue();
});
} catch (err) {
item.reject(err);
this.working = false;
this.dequeue();
}
return true;
}
}
export class ConcurrentPromiseQueue {
/**
* Eingereihte Promises.
*/
private queues: PromiseQueue[] = [];
constructor(concurrency: number = 1) {
this.queues = Array.from({ length: concurrency }).map(() => {
return new PromiseQueue();
});
}
private queueIndex = 0;
private getQueue() {
this.queueIndex = (this.queueIndex + 1) % this.queues.length;
return this.queues[this.queueIndex];
}
public enqueue<T = void>(promise: () => Promise<T>): Promise<T> {
return this.getQueue().enqueue(promise);
}
}

View File

@@ -1,4 +1,5 @@
import { z } from "npm:zod"; import { z } from "zod";
import { RecipeResource } from "./marka/schema.ts";
export const IngredientSchema = z.object({ export const IngredientSchema = z.object({
quantity: z.string().describe( quantity: z.string().describe(
@@ -19,7 +20,7 @@ export type IngredientGroup = z.infer<typeof IngredientGroupSchema>;
const recipeSchema = z.object({ const recipeSchema = z.object({
_type: z.literal("Recipe"), _type: z.literal("Recipe"),
name: z.string().describe( name: z.string().describe(
"Title of the Recipe, without the name of the website or author", "Name of the Recipe, without the name of the website or author",
), ),
description: z.string().describe( description: z.string().describe(
"Optional, short description of the recipe", "Optional, short description of the recipe",
@@ -29,6 +30,9 @@ const recipeSchema = z.object({
_type: z.literal("Person"), _type: z.literal("Person"),
name: z.string().describe("author of the Recipe (optional)"), name: z.string().describe("author of the Recipe (optional)"),
}), }),
keywords: z.array(z.string()).describe(
"List of keywords that match the recipe",
),
recipeIngredient: z.array(z.string()) recipeIngredient: z.array(z.string())
.describe("List of ingredients"), .describe("List of ingredients"),
recipeInstructions: z.array(z.string()).describe("List of instructions"), recipeInstructions: z.array(z.string()).describe("List of instructions"),
@@ -38,17 +42,13 @@ const recipeSchema = z.object({
export type Recipe = z.infer<typeof recipeSchema>; export type Recipe = z.infer<typeof recipeSchema>;
const noRecipeSchema = z.object({ const noRecipeSchema = z.literal("none").describe("No Recipe found");
errorMessages: z.array(z.string()).describe(
"List of error messages, if no recipe was found",
),
});
export const recipeResponseSchema = z.union([recipeSchema, noRecipeSchema]); export const recipeResponseSchema = z.union([recipeSchema, noRecipeSchema]);
export function isValidRecipe( export function isValidRecipe(
recipe: recipe:
| Recipe | RecipeResource
| null | null
| undefined, | undefined,
) { ) {

View File

@@ -1,10 +1,11 @@
import * as openai from "@lib/openai.ts"; import * as openai from "@lib/openai.ts";
import { type MovieRecommendation } from "@lib/openai.ts";
import * as tmdb from "@lib/tmdb.ts"; import * as tmdb from "@lib/tmdb.ts";
import { GenericResource } from "@lib/types.ts";
import { parseRating } from "@lib/helpers.ts"; import { parseRating } from "@lib/helpers.ts";
import { createCache } from "@lib/cache.ts"; import { createCache } from "@lib/cache.ts";
import { ReviewResource } from "./marka/schema.ts";
type RecommendationResource = { export type RecommendationResource = {
id: string; id: string;
type: string; type: string;
rating: number; rating: number;
@@ -18,43 +19,51 @@ type RecommendationResource = {
const cache = createCache<RecommendationResource>("recommendations"); const cache = createCache<RecommendationResource>("recommendations");
export async function createRecommendationResource( export async function createRecommendationResource(
res: GenericResource, res: ReviewResource,
description?: string, description?: string,
) { ) {
const cacheId = `${res.type}:${res.id.replaceAll(":", "")}`; const cacheId = `${res.type}:${res.name.replaceAll(":", "")}`;
const resource = cache.get(cacheId) || { const resource = cache.get(cacheId) || {
id: res.id, id: res.name,
type: res.type, type: res.type,
rating: -1, rating: -1,
}; };
if (description && !resource.keywords) { if (description && !resource.keywords) {
const keywords = await openai.createKeywords(res.type, description, res.id); const keywords = await openai.createKeywords(
res.type,
description,
res.name,
);
if (keywords?.length) { if (keywords?.length) {
resource.keywords = keywords; resource.keywords = keywords;
} }
} }
const { author, date, rating } = res.meta || {}; const { author, datePublished, reviewRating } = res.content;
if (res?.tags) { if (res?.content?.keywords) {
resource.tags = res.tags; resource.keywords = res.content.keywords;
} }
if (typeof rating !== "undefined") { if (typeof reviewRating?.ratingValue !== "undefined") {
resource.rating = parseRating(rating); resource.rating = parseRating(reviewRating?.ratingValue);
} }
if (author) { if (author?.name) {
resource.author = author; resource.author = author.name;
} }
if (description) { if (description) {
resource.description = description; resource.description = description;
} }
if (date) { if (datePublished) {
const d = typeof date === "string" ? new Date(date) : date; const d = typeof datePublished === "string"
resource.year = d.getFullYear(); ? new Date(datePublished)
: datePublished;
if (!isNaN(d.getTime())) {
resource.year = d.getFullYear();
}
} }
cache.set(cacheId, JSON.stringify(resource)); cache.set(cacheId, JSON.stringify(resource));
@@ -68,7 +77,7 @@ export function getRecommendation(
} }
export async function getSimilarMovies(id: string) { export async function getSimilarMovies(id: string) {
const recs = getRecommendation(id, "movie"); const recs = getRecommendation(id, "movies");
if (!recs?.keywords?.length) return; if (!recs?.keywords?.length) return;
const recommendations = await openai.getMovieRecommendations( const recommendations = await openai.getMovieRecommendations(
@@ -77,10 +86,12 @@ export async function getSimilarMovies(id: string) {
); );
if (!recommendations) return; if (!recommendations) return;
const movies = await Promise.all(recommendations.map(async (rec) => { const movies = await Promise.all(
const m = await tmdb.searchMovie(rec.title, rec.year); recommendations.map(async (rec: MovieRecommendation) => {
return m?.results?.[0]; const m = await tmdb.searchMovie(rec.title, rec.year);
})); return m?.results?.[0];
}),
);
return movies.filter(Boolean); return movies.filter(Boolean);
} }
@@ -90,5 +101,7 @@ export async function getAllRecommendations(): Promise<
> { > {
const keys = cache.keys(); const keys = cache.keys();
const res = await Promise.all(keys.map((k) => cache.get(k))); const res = await Promise.all(keys.map((k) => cache.get(k)));
return res.map((r) => JSON.parse(r)); return res.filter((s) => !!s).map((r) =>
typeof r === "string" ? JSON.parse(r) : r
);
} }

View File

@@ -1,30 +0,0 @@
import { z } from "zod";
export type Article = {
_type: "Article";
headline?: string;
datePublished?: string;
articleBody?: string;
keywords?: string[];
image?: string;
url?: string;
reviewRating?: {
bestRating?: number;
worstRating?: number;
ratingValue?: number;
};
author?: {
_type: "Person";
name?: string;
};
};
export const articleMetadataSchema = z.object({
headline: z.union([z.null(), z.string()]).describe("Headline of the article"),
author: z.union([z.null(), z.string()]).describe("Author of the article"),
datePublished: z.union([z.null(), z.string()]).describe(
"Date the article was published",
),
keywords: z.union([z.null(), z.array(z.string())]).describe(
"Keywords for the article",
),
});

View File

@@ -1,21 +0,0 @@
export type Movie = {
_type: "Review";
tmdbId?: number;
link?: string;
author?: {
_type: "Person";
name?: string;
};
datePublished?: string;
reviewRating?: {
bestRating?: number;
worstRating?: number;
ratingValue?: number;
};
reviewBody?: string;
itemReviewed?: {
name?: string;
};
keywords?: string[];
image?: string;
};

View File

@@ -1,17 +0,0 @@
export type Recipe = {
_type: "Recipe";
author?: {
_type: "Person";
name?: string;
};
description?: string;
image?: string;
name?: string;
recipeIngredient?: string[];
recipeInstructions?: string[];
datePublished?: string;
totalTime?: string;
recipeYield?: number;
url?: string;
keywords?: string[];
};

View File

@@ -1,3 +0,0 @@
import { Movie } from "./movies.ts";
export type Series = Movie;

View File

@@ -1,26 +1,26 @@
export const resources = { export const resources = {
"home": { "home": {
emoji: "House with Garden.png", emoji: "home_icon.png",
name: "Home", name: "Home",
link: "/", link: "/",
}, },
"recipe": { "recipe": {
emoji: "Fork and Knife with Plate.png", emoji: "recipes_icon.png",
name: "Recipes", name: "Recipes",
link: "/recipes", link: "/recipes",
}, },
"movie": { "movie": {
emoji: "Popcorn.png", emoji: "movies_icon.png",
name: "Movies", name: "Movies",
link: "/movies", link: "/movies",
}, },
"article": { "article": {
emoji: "Writing Hand Medium-Light Skin Tone.png", emoji: "articles_icon.png",
name: "Articles", name: "Articles",
link: "/articles", link: "/articles",
}, },
"series": { "series": {
emoji: "Television.png", emoji: "tv_series_icon.png",
name: "Series", name: "Series",
link: "/series", link: "/series",
}, },

View File

@@ -1,12 +1,9 @@
import { resources } from "@lib/resources.ts"; import { resources } from "@lib/resources.ts";
import fuzzysort from "npm:fuzzysort"; import fuzzysort from "fuzzysort";
import { GenericResource } from "@lib/types.ts";
import { extractHashTags } from "@lib/string.ts"; import { extractHashTags } from "@lib/string.ts";
import { Movie } from "@lib/resource/movies.ts"; import { listResources } from "./marka/index.ts";
import { Article } from "@lib/resource/articles.ts"; import { GenericResource } from "./marka/schema.ts";
import { Recipe } from "@lib/resource/recipes.ts"; import { parseRating } from "./helpers.ts";
import { Series } from "@lib/resource/series.ts";
import { fetchResource } from "./marka.ts";
type ResourceType = keyof typeof resources; type ResourceType = keyof typeof resources;
@@ -22,7 +19,7 @@ export function parseResourceUrl(_url: string | URL): SearchParams | undefined {
try { try {
const url = typeof _url === "string" ? new URL(_url) : _url; const url = typeof _url === "string" ? new URL(_url) : _url;
let query = url.searchParams.get("q") || "*"; let query = url.searchParams.get("q") || "*";
if (!query) { if (!(typeof query === "string")) {
return undefined; return undefined;
} }
query = decodeURIComponent(query); query = decodeURIComponent(query);
@@ -48,8 +45,8 @@ export function parseResourceUrl(_url: string | URL): SearchParams | undefined {
} }
const isResource = ( const isResource = (
item: Movie | Series | Article | Recipe | boolean, item: GenericResource | boolean | undefined,
): item is Movie | Series | Article | Recipe => { ): item is GenericResource => {
return !!item; return !!item;
}; };
@@ -57,38 +54,59 @@ export async function searchResource(
{ q, tags = [], types, rating }: SearchParams, { q, tags = [], types, rating }: SearchParams,
): Promise<GenericResource[]> { ): Promise<GenericResource[]> {
const resources = (await Promise.all([ const resources = (await Promise.all([
(!types || types.includes("movie")) && fetchResource("movies"), (!types || types.includes("movies")) && listResources("movies"),
(!types || types.includes("series")) && fetchResource("series"), (!types || types.includes("series")) && listResources("series"),
(!types || types.includes("article")) && fetchResource("articles"), (!types || types.includes("articles")) && listResources("articles"),
(!types || types.includes("recipe")) && fetchResource("recipes"), (!types || types.includes("recipes")) && listResources("recipes"),
])).flat().filter(isResource); ])).flat().filter(isResource);
console.log({ types, rating, tags, q, resourceLength: resources.length });
const results: Record<string, GenericResource> = {}; const results: Record<string, GenericResource> = {};
for (const resource of resources) { for (const resource of resources) {
if ( if (
!(resource.name in results) && !(resource.name in results) &&
tags?.length && resource.tags.length && tags?.length && resource.content.keywords?.length &&
tags.every((t) => resource.tags.includes(t)) tags.every((t) => resource.content.keywords?.includes(t))
) { ) {
results[resource.id] = resource; results[resource.name] = resource;
}
// Select not-rated resources
if (
rating === 0 &&
resource.content?.reviewRating?.ratingValue === undefined
) {
results[resource.name] = resource;
} }
if ( if (
!(resource.id in results) && typeof rating === "number" &&
rating && resource?.meta?.rating && resource.meta.rating >= rating rating !== 0 &&
resource.content.reviewRating?.ratingValue &&
parseRating(resource.content.reviewRating.ratingValue) >= rating
) { ) {
results[resource.id] = resource; results[resource.name] = resource;
} }
} }
if (q.length && q !== "*") { if (q.length && q !== "*") {
q = decodeURIComponent(q);
const fuzzyResult = fuzzysort.go(q, resources, { const fuzzyResult = fuzzysort.go(q, resources, {
keys: ["content", "name", "description", "meta.author"], keys: [
"name",
"content.articleBody",
"content.reviewBody",
"content.name",
"content.description",
"content.author.name",
],
threshold: 0.3, threshold: 0.3,
}); });
console.log({ fuzzyResult });
for (const result of fuzzyResult) { for (const result of fuzzyResult) {
results[result.obj.id] = result.obj; results[result.obj.name] = result.obj;
} }
} }

View File

@@ -1,25 +1,37 @@
import { resources } from "@lib/resources.ts"; export function formatDate(date?: string | Date): string {
if (!date) return "";
export function formatDate(date: Date): string { if (typeof date === "string") {
try {
const d = new Date(date);
return formatDate(d);
} catch (_e) {
return "";
}
}
const options = { year: "numeric", month: "long", day: "numeric" } as const; const options = { year: "numeric", month: "long", day: "numeric" } as const;
return new Intl.DateTimeFormat("en-US", options).format(date); return new Intl.DateTimeFormat("en-US", options).format(date);
} }
export function safeFileName(inputString: string): string { export function safeFileName(input: string): string {
let fileName = inputString.toLowerCase(); return input
fileName = fileName.replace(/ /g, "_"); .normalize("NFKD")
fileName = fileName.replace(/[^\w.-]/g, ""); .replace(/[\u0300-\u036f]/g, "")
fileName = fileName.replaceAll(":", ""); .replace(/[\s-]+/g, "_")
return fileName; .replace(/-+/g, "-")
.replace(/[^A-Za-z0-9_]+/g, "")
.replace(/_+/g, "_")
// Trim underscores/dots from ends and prevent leading dots
.replace(/^[_\.]+|[_\.]+$/g, "").replace(/^\.+/, "")
.toLowerCase();
} }
export function toUrlSafeString(input: string): string { export function toUrlSafeString(input: string): string {
return input return input
.trim() // Remove leading and trailing whitespace .normalize("NFKD")
.toLowerCase() // Convert to lowercase .replace(/[\u0300-\u036f]/g, "")
.replace(/[^a-z0-9\s-]/g, "") // Remove non-alphanumeric characters except spaces and hyphens .replace(/[^A-Za-z0-9 _-]+/g, "")
.replace(/\s+/g, "-") // Replace spaces with hyphens .replace(/\s+/g, " ")
.replace(/-+/g, "-"); // Remove consecutive hyphens .trim();
} }
export function extractHashTags(inputString: string) { export function extractHashTags(inputString: string) {
@@ -174,3 +186,7 @@ export function removeMarkdownFormatting(text: string): string {
return text; return text;
} }
export function fileExtension(fname: string) {
return fname.slice((fname.lastIndexOf(".") - 1 >>> 0) + 2);
}

View File

@@ -1,110 +0,0 @@
import { transcribe } from "@lib/openai.ts";
import { createDocument } from "@lib/documents.ts";
import { createLogger } from "./log/index.ts";
import { convertOggToMp3 } from "./helpers.ts";
const log = createLogger("taskManager");
// In-memory task state
const activeTasks: Record<
string,
{
noteName: string;
entries: Array<
{ type: string; content: string | ArrayBufferLike; fileName?: string }
>;
}
> = {};
export function startTask(chatId: string, noteName: string) {
activeTasks[chatId] = { noteName, entries: [] };
log.info(`Started note: ${noteName}`);
}
export async function endTask(chatId: string): Promise<string | null> {
const task = activeTasks[chatId];
if (!task) return null;
log.info("Ending note", task.noteName);
let finalNote = `# ${task.noteName}\n\n`;
const photoTasks: { content: ArrayBuffer; path: string }[] = [];
let photoIndex = 0;
for (const entry of task.entries) {
if (entry.type === "text") {
finalNote += entry.content + "\n\n";
} else if (entry.type === "voice") {
try {
log.info("Converting OGG to MP3");
const mp3Data = await convertOggToMp3(entry.content as ArrayBuffer);
log.info("Finished converting OGG to MP3, transcribing...");
const transcript = await transcribe(mp3Data);
finalNote += `**Voice Transcript:**\n${transcript}\n\n`;
log.info("Finished transcribing");
} catch (error) {
log.error(error);
finalNote += "**[Voice message could not be transcribed]**\n\n";
}
} else if (entry.type === "photo") {
const photoUrl = `${
task.noteName.replace(/\.md$/, "")
}/photo-${photoIndex++}.jpg`;
finalNote += `**Photo**:\n ${photoUrl}\n\n`;
photoTasks.push({
content: entry.content as ArrayBuffer,
path: photoUrl,
});
}
}
try {
for (const entry of photoTasks) {
await createDocument(entry.path, entry.content, "image/jpeg");
}
} catch (err) {
log.error("Error creating photo document:", err);
}
try {
await createDocument(task.noteName, finalNote, "text/markdown");
} catch (error) {
log.error("Error creating document:", error);
return error instanceof Error
? error.toString()
: "Error creating document";
}
delete activeTasks[chatId];
log.debug({ finalNote });
return finalNote;
}
export function addTextEntry(chatId: string, text: string) {
const task = activeTasks[chatId];
if (!task) return;
const entry = { type: "text", content: text };
log.debug("New Entry", entry);
task.entries.push(entry);
}
export function addVoiceEntry(chatId: string, buffer: ArrayBufferLike) {
const task = activeTasks[chatId];
if (!task) return;
const entry = { type: "voice", content: buffer };
log.debug("New Entry", entry);
task.entries.push(entry);
}
export function addPhotoEntry(
chatId: string,
buffer: ArrayBufferLike,
fileName: string,
) {
const task = activeTasks[chatId];
if (!task) return;
const entry = { type: "photo", content: buffer, fileName };
log.debug("New Entry", entry);
task.entries.push(entry);
}

View File

@@ -1,65 +0,0 @@
import { Bot } from "https://deno.land/x/grammy@v1.36.1/mod.ts";
import { TELEGRAM_API_KEY } from "@lib/env.ts";
import { createLogger } from "./log/index.ts";
const bot = new Bot(TELEGRAM_API_KEY);
const log = createLogger("telegram");
import * as manager from "./taskManager.ts";
async function downloadFile(filePath: string): Promise<Uint8Array> {
log.info(`Downloading file from path: ${filePath}`);
const url = `https://api.telegram.org/file/bot${bot.token}/${filePath}`;
const response = await fetch(url);
if (!response.ok) {
throw new Error("Failed to download file: " + response.statusText);
}
log.info("File downloaded successfully");
const buffer = await response.arrayBuffer();
return new Uint8Array(buffer);
}
bot.command("start", async (ctx) => {
log.info("Received /start command");
const [_, noteName] = ctx.message?.text?.split(" ") || [];
if (!noteName) {
return ctx.reply("Please provide a note name. Usage: /start NoteName");
}
manager.startTask(ctx.chat.id.toString(), noteName);
await ctx.reply(`Started note: ${noteName}`);
});
bot.command("end", async (ctx) => {
log.info("Received /end command");
const finalNote = await manager.endTask(ctx.chat.id.toString());
if (!finalNote) return ctx.reply("No active note found.");
try {
await ctx.reply("Note complete. Here is your markdown:");
await ctx.reply(finalNote, { parse_mode: "MarkdownV2" });
} catch (err) {
console.error("Error sending final note:", err);
}
});
bot.on("message:text", (ctx) => {
log.info("Received text message");
manager.addTextEntry(ctx.chat.id.toString(), ctx.message.text);
});
bot.on("message:voice", async (ctx) => {
log.info("Received photo message");
log.info("Received voice message");
const file = await ctx.getFile();
const buffer = await downloadFile(file.file_path!);
manager.addVoiceEntry(ctx.chat.id.toString(), buffer.buffer);
});
bot.on("message:photo", async (ctx) => {
const file = await ctx.getFile();
const buffer = await downloadFile(file.file_path!);
const fileName = file.file_path!.split("/").pop()!;
manager.addPhotoEntry(ctx.chat.id.toString(), buffer.buffer, fileName);
});
bot.start();

View File

@@ -1,6 +1,10 @@
import * as thumbhash from "https://esm.sh/thumbhash@0.1.1"; import * as thumbhash from "thumbhash";
export function generateThumbhash(buffer: Uint8Array, w: number, h: number) { export function generateThumbhash(
buffer: ArrayLike<number>,
w: number,
h: number,
) {
const hash = thumbhash.rgbaToThumbHash(w, h, buffer); const hash = thumbhash.rgbaToThumbHash(w, h, buffer);
return [hash, thumbhash.thumbHashToAverageRGBA(hash)] as const; return [hash, thumbhash.thumbHashToAverageRGBA(hash)] as const;
} }

View File

@@ -5,7 +5,7 @@ import {
MovieResultsResponse, MovieResultsResponse,
ShowResponse, ShowResponse,
TvResultsResponse, TvResultsResponse,
} from "https://esm.sh/moviedb-promise@3.4.1"; } from "moviedb-promise";
import { createCache } from "@lib/cache.ts"; import { createCache } from "@lib/cache.ts";
const moviedb = new MovieDb(Deno.env.get("TMDB_API_KEY") || ""); const moviedb = new MovieDb(Deno.env.get("TMDB_API_KEY") || "");

Some files were not shown because too many files have changed in this diff Show More