feat: cache marka api responses
This commit is contained in:
@@ -26,7 +26,7 @@ function generateJsonLd(resource: GenericResource): string {
|
|||||||
if (resource.content?.datePublished) {
|
if (resource.content?.datePublished) {
|
||||||
try {
|
try {
|
||||||
baseSchema.datePublished = formatDate(
|
baseSchema.datePublished = formatDate(
|
||||||
new Date(resource.content.datePublished),
|
resource.content.datePublished,
|
||||||
);
|
);
|
||||||
} catch (_) {
|
} catch (_) {
|
||||||
// Ignore invalid date
|
// Ignore invalid date
|
||||||
|
|||||||
@@ -50,13 +50,17 @@ export const createNewSeries: MenuEntry = {
|
|||||||
return {
|
return {
|
||||||
title: `${r.name} - ${r.first_air_date}`,
|
title: `${r.name} - ${r.first_air_date}`,
|
||||||
cb: async () => {
|
cb: async () => {
|
||||||
state.activeState.value = "loading";
|
try {
|
||||||
const response = await fetch("/api/series/" + r.id, {
|
state.activeState.value = "loading";
|
||||||
method: "POST",
|
const response = await fetch("/api/series/" + r.id, {
|
||||||
});
|
method: "POST",
|
||||||
const series = await response.json() as ReviewResource;
|
});
|
||||||
unsub();
|
const series = await response.json() as ReviewResource;
|
||||||
globalThis.location.href = "/series/" + series.name;
|
unsub();
|
||||||
|
globalThis.location.href = "/series/" + series.name;
|
||||||
|
} catch (_e) {
|
||||||
|
state.activeState.value = "normal";
|
||||||
|
}
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { createCache } from "../cache.ts";
|
||||||
import { MARKA_API_KEY } from "../env.ts";
|
import { MARKA_API_KEY } from "../env.ts";
|
||||||
import { getImage } from "../image.ts";
|
import { getImage } from "../image.ts";
|
||||||
import { GenericResource } from "./schema.ts";
|
import { GenericResource } from "./schema.ts";
|
||||||
@@ -24,32 +25,62 @@ async function addImageToResource<T extends GenericResource>(
|
|||||||
return resource as T;
|
return resource as T;
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function fetchResource<T extends GenericResource>(
|
type Resource = GenericResource & {
|
||||||
|
content: GenericResource["content"] & Array<GenericResource>;
|
||||||
|
};
|
||||||
|
|
||||||
|
const fetchCache = createCache<Resource>("marka");
|
||||||
|
const cacheLock = new Map<string, Promise<Resource>>();
|
||||||
|
async function cachedFetch(
|
||||||
|
url: string,
|
||||||
|
): Promise<Resource | undefined> {
|
||||||
|
if (fetchCache.has(url)) {
|
||||||
|
return fetchCache.get(url);
|
||||||
|
}
|
||||||
|
if (cacheLock.has(url)) {
|
||||||
|
return cacheLock.get(url);
|
||||||
|
}
|
||||||
|
const response = (async () => {
|
||||||
|
const response = await fetch(url);
|
||||||
|
const res = await response.json();
|
||||||
|
fetchCache.set(url, res);
|
||||||
|
return res;
|
||||||
|
})();
|
||||||
|
cacheLock.set(
|
||||||
|
url,
|
||||||
|
response,
|
||||||
|
);
|
||||||
|
const res = await response;
|
||||||
|
if (!res) {
|
||||||
|
cacheLock.delete(url);
|
||||||
|
}
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchResource<T extends Resource>(
|
||||||
resource: string,
|
resource: string,
|
||||||
): Promise<T | undefined> {
|
): Promise<T | undefined> {
|
||||||
try {
|
try {
|
||||||
const response = await fetch(
|
const d = `${url}/resources/${resource}`;
|
||||||
`${url}/resources/${resource}`,
|
const res = await cachedFetch(d);
|
||||||
);
|
if (!res) return;
|
||||||
const res = await response.json();
|
|
||||||
return addImageToResource<T>(res);
|
return addImageToResource<T>(res);
|
||||||
} catch (_e) {
|
} catch (_e) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function listResources<T = GenericResource>(
|
export async function listResources<T extends GenericResource>(
|
||||||
resource: string,
|
resource: string,
|
||||||
): Promise<T[]> {
|
): Promise<T[]> {
|
||||||
try {
|
try {
|
||||||
const response = await fetch(
|
const d = `${url}/resources/${resource}`;
|
||||||
`${url}/resources/${resource}`,
|
const list = await cachedFetch(d);
|
||||||
);
|
if (!list) return [];
|
||||||
const list = await response.json();
|
|
||||||
return Promise.all(
|
return Promise.all(
|
||||||
list?.content
|
list?.content
|
||||||
.filter((a: GenericResource) => a?.content?._type)
|
.filter((a) => a?.content?._type)
|
||||||
.map((res: GenericResource) => addImageToResource(res)),
|
.map((res) => addImageToResource(res) as Promise<T>),
|
||||||
);
|
);
|
||||||
} catch (_e) {
|
} catch (_e) {
|
||||||
return [];
|
return [];
|
||||||
|
|||||||
@@ -1,4 +1,13 @@
|
|||||||
export function formatDate(date: Date): string {
|
export function formatDate(date?: string | Date): string {
|
||||||
|
if (!date) return "";
|
||||||
|
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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,13 +6,15 @@ import * as openai from "@lib/openai.ts";
|
|||||||
import { getYoutubeVideoDetails } from "@lib/youtube.ts";
|
import { getYoutubeVideoDetails } from "@lib/youtube.ts";
|
||||||
import {
|
import {
|
||||||
extractYoutubeId,
|
extractYoutubeId,
|
||||||
|
formatDate,
|
||||||
isYoutubeLink,
|
isYoutubeLink,
|
||||||
toUrlSafeString,
|
safeFileName,
|
||||||
} from "@lib/string.ts";
|
} from "@lib/string.ts";
|
||||||
import { createLogger } from "@lib/log/index.ts";
|
import { createLogger } from "@lib/log/index.ts";
|
||||||
import { createResource } from "@lib/marka/index.ts";
|
import { createResource } from "@lib/marka/index.ts";
|
||||||
import { webScrape } from "@lib/webScraper.ts";
|
import { webScrape } from "@lib/webScraper.ts";
|
||||||
import { ArticleResource } from "@lib/marka/schema.ts";
|
import { ArticleResource } from "@lib/marka/schema.ts";
|
||||||
|
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
|
||||||
|
|
||||||
const log = createLogger("api/article");
|
const log = createLogger("api/article");
|
||||||
|
|
||||||
@@ -44,16 +46,34 @@ async function processCreateArticle(
|
|||||||
streamResponse.enqueue("postprocessing article");
|
streamResponse.enqueue("postprocessing article");
|
||||||
|
|
||||||
const title = result?.title || aiMeta?.headline || "";
|
const title = result?.title || aiMeta?.headline || "";
|
||||||
const id = toUrlSafeString(title);
|
|
||||||
|
let finalPath = result.image;
|
||||||
|
if (result?.image) {
|
||||||
|
const extension = fileExtension(result?.image);
|
||||||
|
const imagePath = `resources/articles/images/${
|
||||||
|
safeFileName(title)
|
||||||
|
}_cover.${extension}`;
|
||||||
|
try {
|
||||||
|
streamResponse.enqueue("downloading image");
|
||||||
|
const res = await fetch(result.image);
|
||||||
|
streamResponse.enqueue("saving image");
|
||||||
|
const buffer = await res.arrayBuffer();
|
||||||
|
await createResource(imagePath, buffer);
|
||||||
|
finalPath = imagePath;
|
||||||
|
} catch (err) {
|
||||||
|
console.log("Failed to save image", err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const newArticle: ArticleResource["content"] = {
|
const newArticle: ArticleResource["content"] = {
|
||||||
_type: "Article",
|
_type: "Article",
|
||||||
headline: title,
|
headline: title,
|
||||||
articleBody: result.content,
|
articleBody: result.content,
|
||||||
url: fetchUrl,
|
url: fetchUrl,
|
||||||
datePublished: result?.published || aiMeta?.datePublished ||
|
datePublished: formatDate(
|
||||||
new Date().toISOString(),
|
result?.published || aiMeta?.datePublished || undefined,
|
||||||
image: result?.image,
|
),
|
||||||
|
image: finalPath,
|
||||||
author: {
|
author: {
|
||||||
_type: "Person",
|
_type: "Person",
|
||||||
name: (result.schemaOrgData?.author?.name || aiMeta?.author || "")
|
name: (result.schemaOrgData?.author?.name || aiMeta?.author || "")
|
||||||
@@ -66,9 +86,9 @@ async function processCreateArticle(
|
|||||||
|
|
||||||
streamResponse.enqueue("writing to disk");
|
streamResponse.enqueue("writing to disk");
|
||||||
|
|
||||||
await createResource(`articles/${id}.md`, newArticle);
|
await createResource(`articles/${title}.md`, newArticle);
|
||||||
|
|
||||||
streamResponse.enqueue("id: " + id);
|
streamResponse.enqueue("id: " + title);
|
||||||
}
|
}
|
||||||
|
|
||||||
async function processCreateYoutubeVideo(
|
async function processCreateYoutubeVideo(
|
||||||
|
|||||||
@@ -2,8 +2,8 @@ import { Handlers } from "$fresh/server.ts";
|
|||||||
import { json } from "@lib/helpers.ts";
|
import { json } from "@lib/helpers.ts";
|
||||||
import * as tmdb from "@lib/tmdb.ts";
|
import * as tmdb from "@lib/tmdb.ts";
|
||||||
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
|
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
|
||||||
import { isString, safeFileName } from "@lib/string.ts";
|
import { formatDate, isString, safeFileName } from "@lib/string.ts";
|
||||||
import { AccessDeniedError } from "@lib/errors.ts";
|
import { AccessDeniedError, BadRequestError } from "@lib/errors.ts";
|
||||||
import { createResource, fetchResource } from "@lib/marka/index.ts";
|
import { createResource, fetchResource } from "@lib/marka/index.ts";
|
||||||
import { ReviewResource } from "@lib/marka/schema.ts";
|
import { ReviewResource } from "@lib/marka/schema.ts";
|
||||||
|
|
||||||
@@ -14,26 +14,22 @@ export const handler: Handlers = {
|
|||||||
},
|
},
|
||||||
async POST(_, ctx) {
|
async POST(_, ctx) {
|
||||||
const session = ctx.state.session;
|
const session = ctx.state.session;
|
||||||
if (!session) {
|
if (!session) throw new AccessDeniedError();
|
||||||
throw new AccessDeniedError();
|
|
||||||
}
|
|
||||||
|
|
||||||
const tmdbId = parseInt(ctx.params.name);
|
const tmdbId = parseInt(ctx.params.name);
|
||||||
|
if (Number.isNaN(tmdbId)) throw new BadRequestError();
|
||||||
|
|
||||||
const movieDetails = await tmdb.getMovie(tmdbId);
|
const [movieDetails, movieCredits] = await Promise.all([
|
||||||
const movieCredits = await tmdb.getMovieCredits(tmdbId);
|
tmdb.getMovie(tmdbId),
|
||||||
|
tmdb.getMovieCredits(tmdbId),
|
||||||
|
]);
|
||||||
|
|
||||||
const releaseDate = movieDetails.release_date;
|
const name = movieDetails.title ||
|
||||||
const posterPath = movieDetails.poster_path;
|
movieDetails.original_title ||
|
||||||
const director = movieCredits?.crew?.filter?.((person) =>
|
|
||||||
person.job === "Director"
|
|
||||||
)[0];
|
|
||||||
|
|
||||||
movieDetails.overview;
|
|
||||||
|
|
||||||
let finalPath = "";
|
|
||||||
const name = movieDetails.title || movieDetails.original_title ||
|
|
||||||
ctx.params.name;
|
ctx.params.name;
|
||||||
|
|
||||||
|
const posterPath = movieDetails.poster_path;
|
||||||
|
let finalPath = "";
|
||||||
if (posterPath) {
|
if (posterPath) {
|
||||||
const poster = await tmdb.getMoviePoster(posterPath);
|
const poster = await tmdb.getMoviePoster(posterPath);
|
||||||
const extension = fileExtension(posterPath);
|
const extension = fileExtension(posterPath);
|
||||||
@@ -41,33 +37,32 @@ export const handler: Handlers = {
|
|||||||
await createResource(finalPath, poster);
|
await createResource(finalPath, poster);
|
||||||
}
|
}
|
||||||
|
|
||||||
const tags: string[] = [];
|
const keywords = movieDetails.genres
|
||||||
if (movieDetails.genres) {
|
?.map((g) => g.name?.toLowerCase())
|
||||||
tags.push(
|
.filter(isString) || [];
|
||||||
...movieDetails.genres.map((g) => g.name?.toLowerCase()).filter(
|
|
||||||
isString,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const movie: ReviewResource["content"] = {
|
const movie: ReviewResource["content"] = {
|
||||||
_type: "Review",
|
_type: "Review",
|
||||||
image: `resources/${finalPath}`,
|
image: `resources/${finalPath}`,
|
||||||
datePublished: releaseDate,
|
datePublished: formatDate(movieDetails.release_date),
|
||||||
tmdbId,
|
tmdbId,
|
||||||
author: {
|
author: {
|
||||||
_type: "Person",
|
_type: "Person",
|
||||||
name: director?.name,
|
name: movieCredits.crew?.filter?.((person) =>
|
||||||
|
person.job === "Director"
|
||||||
|
)[0]?.name,
|
||||||
},
|
},
|
||||||
itemReviewed: {
|
itemReviewed: {
|
||||||
name: name,
|
name,
|
||||||
},
|
},
|
||||||
reviewBody: "",
|
reviewBody: "",
|
||||||
keywords: tags,
|
keywords,
|
||||||
};
|
};
|
||||||
|
|
||||||
await createResource(`movies/${safeFileName(name)}.md`, movie);
|
const fileName = `${safeFileName(name)}.md`;
|
||||||
|
|
||||||
return json(movie);
|
await createResource(`movies/${fileName}`, movie);
|
||||||
|
|
||||||
|
return json({ name: fileName });
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -39,15 +39,16 @@ const POST = async (
|
|||||||
const movieCredits = !movie.content?.author &&
|
const movieCredits = !movie.content?.author &&
|
||||||
await tmdb.getMovieCredits(tmdbId);
|
await tmdb.getMovieCredits(tmdbId);
|
||||||
|
|
||||||
const releaseDate = movieDetails.release_date;
|
|
||||||
if (releaseDate && !movie.content?.datePublished) {
|
|
||||||
movie.content = movie.content || {};
|
|
||||||
movie.content.datePublished = formatDate(new Date(releaseDate));
|
|
||||||
}
|
|
||||||
const director = movieCredits &&
|
const director = movieCredits &&
|
||||||
movieCredits?.crew?.filter?.((person) => person.job === "Director")[0];
|
movieCredits?.crew?.filter?.((person) => person.job === "Director")[0];
|
||||||
|
|
||||||
|
movie.content ??= {
|
||||||
|
_type: "Review",
|
||||||
|
};
|
||||||
|
|
||||||
|
movie.content.datePublished ??= formatDate(movieDetails.release_date);
|
||||||
|
|
||||||
if (director && !movie.content?.author) {
|
if (director && !movie.content?.author) {
|
||||||
movie.content = movie.content || {};
|
|
||||||
movie.content.author = {
|
movie.content.author = {
|
||||||
_type: "Person",
|
_type: "Person",
|
||||||
name: director.name,
|
name: director.name,
|
||||||
@@ -65,18 +66,15 @@ const POST = async (
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!movie.name) {
|
movie.content.tmdbId ??= tmdbId;
|
||||||
movie.name = tmdbId;
|
|
||||||
}
|
|
||||||
|
|
||||||
let finalPath = "";
|
let finalPath = "";
|
||||||
const posterPath = movieDetails.poster_path;
|
const posterPath = movieDetails.poster_path;
|
||||||
if (posterPath && !movie.content?.image) {
|
if (posterPath && !movie.content.image) {
|
||||||
const poster = await tmdb.getMoviePoster(posterPath);
|
const poster = await tmdb.getMoviePoster(posterPath);
|
||||||
const extension = fileExtension(posterPath);
|
const extension = fileExtension(posterPath);
|
||||||
finalPath = `movies/images/${safeFileName(name)}_cover.${extension}`;
|
finalPath = `movies/images/${safeFileName(name)}_cover.${extension}`;
|
||||||
await createResource(finalPath, poster);
|
await createResource(finalPath, poster);
|
||||||
movie.content = movie.content || {};
|
|
||||||
movie.content.image = finalPath;
|
movie.content.image = finalPath;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -76,7 +76,7 @@ async function processCreateRecipeFromUrl(
|
|||||||
};
|
};
|
||||||
|
|
||||||
if (newRecipe?.image && newRecipe.image.length > 5) {
|
if (newRecipe?.image && newRecipe.image.length > 5) {
|
||||||
const extension = fileExtension(new URL(newRecipe.image).pathname);
|
const extension = fileExtension(newRecipe.image);
|
||||||
const finalPath = `resources/recipes/images/${
|
const finalPath = `resources/recipes/images/${
|
||||||
safeFileName(id)
|
safeFileName(id)
|
||||||
}_cover.${extension}`;
|
}_cover.${extension}`;
|
||||||
|
|||||||
@@ -2,11 +2,19 @@ import { Handlers } from "$fresh/server.ts";
|
|||||||
import { json } from "@lib/helpers.ts";
|
import { json } from "@lib/helpers.ts";
|
||||||
import * as tmdb from "@lib/tmdb.ts";
|
import * as tmdb from "@lib/tmdb.ts";
|
||||||
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
|
import { fileExtension } from "https://deno.land/x/file_extension@v2.1.0/mod.ts";
|
||||||
import { isString, safeFileName } from "@lib/string.ts";
|
import { formatDate, isString, safeFileName } from "@lib/string.ts";
|
||||||
import { AccessDeniedError } from "@lib/errors.ts";
|
import { AccessDeniedError, BadRequestError } from "@lib/errors.ts";
|
||||||
import { createResource, fetchResource } from "@lib/marka/index.ts";
|
import { createResource, fetchResource } from "@lib/marka/index.ts";
|
||||||
import { ReviewResource } from "@lib/marka/schema.ts";
|
import { ReviewResource } from "@lib/marka/schema.ts";
|
||||||
|
|
||||||
|
function pickDirector(
|
||||||
|
credits: Awaited<ReturnType<typeof tmdb.getSeriesCredits>>,
|
||||||
|
createdBy?: { name?: string }[],
|
||||||
|
): string | undefined {
|
||||||
|
const crewDirector = credits?.crew?.find?.((p) => p.job === "Director");
|
||||||
|
return crewDirector?.name ?? createdBy?.[0]?.name;
|
||||||
|
}
|
||||||
|
|
||||||
export const handler: Handlers = {
|
export const handler: Handlers = {
|
||||||
async GET(_, ctx) {
|
async GET(_, ctx) {
|
||||||
const series = await fetchResource(`series/${ctx.params.name}`);
|
const series = await fetchResource(`series/${ctx.params.name}`);
|
||||||
@@ -19,54 +27,54 @@ export const handler: Handlers = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const tmdbId = parseInt(ctx.params.name);
|
const tmdbId = parseInt(ctx.params.name);
|
||||||
|
if (Number.isNaN(tmdbId)) throw new BadRequestError();
|
||||||
|
|
||||||
const seriesDetails = await tmdb.getSeries(tmdbId);
|
const [seriesDetails, seriesCredits] = await Promise.all([
|
||||||
const seriesCredits = await tmdb.getSeriesCredits(tmdbId);
|
tmdb.getSeries(tmdbId),
|
||||||
|
tmdb.getSeriesCredits(tmdbId),
|
||||||
|
]);
|
||||||
|
|
||||||
const releaseDate = seriesDetails.first_air_date;
|
const name = seriesDetails.name ||
|
||||||
const posterPath = seriesDetails.poster_path;
|
seriesDetails.original_name ||
|
||||||
const director =
|
ctx.params.name;
|
||||||
seriesCredits?.crew?.filter?.((person) => person.job === "Director")[0] ||
|
|
||||||
seriesDetails.created_by?.[0];
|
|
||||||
|
|
||||||
let finalPath = "";
|
let finalPath = "";
|
||||||
const name = seriesDetails.name || seriesDetails.original_name ||
|
const posterPath = seriesDetails.poster_path;
|
||||||
ctx.params.name;
|
|
||||||
if (posterPath) {
|
if (posterPath) {
|
||||||
const poster = await tmdb.getMoviePoster(posterPath);
|
const poster = await tmdb.getMoviePoster(posterPath);
|
||||||
const extension = fileExtension(posterPath);
|
const extension = fileExtension(posterPath);
|
||||||
|
const imagePath = `series/images/${
|
||||||
finalPath = `series/images/${safeFileName(name)}_cover.${extension}`;
|
safeFileName(name)
|
||||||
await createResource(finalPath, poster);
|
}_cover.${extension}`;
|
||||||
|
await createResource(imagePath, poster);
|
||||||
|
finalPath = imagePath;
|
||||||
}
|
}
|
||||||
|
|
||||||
const tags: string[] = [];
|
const keywords = seriesDetails.genres
|
||||||
if (seriesDetails.genres) {
|
?.map((g) => g.name?.toLowerCase())
|
||||||
tags.push(
|
.filter(isString) ??
|
||||||
...seriesDetails.genres.map((g) => g.name?.toLowerCase()).filter(
|
[];
|
||||||
isString,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
const series: ReviewResource["content"] = {
|
const series: ReviewResource["content"] = {
|
||||||
_type: "Review",
|
_type: "Review",
|
||||||
image: `resources/${finalPath}`,
|
image: `resources/${finalPath}`,
|
||||||
datePublished: releaseDate,
|
datePublished: formatDate(seriesDetails.first_air_date),
|
||||||
tmdbId,
|
tmdbId,
|
||||||
author: {
|
author: {
|
||||||
_type: "Person",
|
_type: "Person",
|
||||||
name: director?.name,
|
name: pickDirector(seriesCredits, seriesDetails?.created_by),
|
||||||
},
|
},
|
||||||
itemReviewed: {
|
itemReviewed: {
|
||||||
name: name,
|
name: name,
|
||||||
},
|
},
|
||||||
reviewBody: "",
|
reviewBody: "",
|
||||||
keywords: tags,
|
keywords: keywords,
|
||||||
};
|
};
|
||||||
|
|
||||||
await createResource(`series/${safeFileName(name)}.md`, series);
|
const fileName = `${safeFileName(name)}.md`;
|
||||||
|
|
||||||
return json(series);
|
await createResource(`series/${fileName}`, series);
|
||||||
|
|
||||||
|
return json({ name: fileName });
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user