2022-12-29 22:10:28 +00:00
|
|
|
import { error } from '@sveltejs/kit';
|
2023-05-15 04:08:30 +00:00
|
|
|
import { superValidate } from 'sveltekit-superforms/server';
|
2023-06-26 06:14:26 +00:00
|
|
|
import kebabCase from 'just-kebab-case';
|
|
|
|
|
import { BOARD_GAME_ATLAS_CLIENT_ID } from '$env/static/private';
|
|
|
|
|
import prisma from '$lib/prisma.js';
|
2023-05-15 04:08:30 +00:00
|
|
|
import type { GameType, SearchQuery } from '$lib/types';
|
|
|
|
|
import { mapAPIGameToBoredGame } from '$lib/util/gameMapper';
|
|
|
|
|
import { search_schema } from '$lib/zodValidation';
|
2022-12-25 18:52:36 +00:00
|
|
|
|
2023-05-21 05:18:04 +00:00
|
|
|
async function searchForGames(urlQueryParams: SearchQuery) {
|
2022-12-25 18:52:36 +00:00
|
|
|
try {
|
2023-06-26 06:14:26 +00:00
|
|
|
let dbGames = await prisma.game.findMany({
|
|
|
|
|
where: {
|
|
|
|
|
name: {
|
|
|
|
|
search: urlQueryParams?.name
|
|
|
|
|
},
|
|
|
|
|
min_players: {
|
|
|
|
|
gte: urlQueryParams?.min_players || 0
|
|
|
|
|
},
|
|
|
|
|
max_players: {
|
|
|
|
|
lte: urlQueryParams?.max_players || 100
|
|
|
|
|
},
|
|
|
|
|
min_playtime: {
|
|
|
|
|
gte: urlQueryParams?.min_playtime || 0
|
|
|
|
|
},
|
|
|
|
|
max_playtime: {
|
|
|
|
|
lte: urlQueryParams?.max_playtime || 5000
|
|
|
|
|
},
|
|
|
|
|
min_age: {
|
|
|
|
|
gte: urlQueryParams?.min_age || 130
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
skip: urlQueryParams?.skip,
|
|
|
|
|
take: urlQueryParams?.limit,
|
|
|
|
|
orderBy: {
|
|
|
|
|
name: 'asc'
|
2022-12-25 18:52:36 +00:00
|
|
|
}
|
|
|
|
|
});
|
2023-06-26 06:14:26 +00:00
|
|
|
console.log('dbGames', dbGames);
|
2022-12-25 18:52:36 +00:00
|
|
|
|
2023-06-26 06:14:26 +00:00
|
|
|
if (!dbGames || dbGames.length === 0) {
|
|
|
|
|
const url = new URL(
|
|
|
|
|
`https://api.boardgameatlas.com/api/search${urlQueryParams ? `?${urlQueryParams}` : ''}`
|
|
|
|
|
);
|
|
|
|
|
const headers: HeadersInit = new Headers();
|
|
|
|
|
headers.set('Content-Type', 'application/json');
|
|
|
|
|
const requestInit: RequestInit = {
|
|
|
|
|
method: 'GET',
|
|
|
|
|
headers
|
|
|
|
|
};
|
|
|
|
|
const response = await fetch(url, requestInit);
|
2022-12-25 18:52:36 +00:00
|
|
|
|
2023-06-26 06:14:26 +00:00
|
|
|
if (!response.ok) {
|
|
|
|
|
console.log('Status not 200', response.status);
|
|
|
|
|
throw error(response.status);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const games: GameType[] = [];
|
|
|
|
|
let totalCount = 0;
|
|
|
|
|
if (response.ok) {
|
|
|
|
|
const gameResponse = await response.json();
|
|
|
|
|
const gameList: GameType[] = gameResponse?.games;
|
|
|
|
|
totalCount = gameResponse?.count;
|
|
|
|
|
console.log('totalCount', totalCount);
|
|
|
|
|
gameList.forEach((game) => {
|
|
|
|
|
if (game?.min_players && game?.max_players) {
|
|
|
|
|
game.players = `${game.min_players}-${game.max_players}`;
|
|
|
|
|
game.playtime = `${game.min_playtime}-${game.max_playtime}`;
|
|
|
|
|
}
|
|
|
|
|
const boredGame = mapAPIGameToBoredGame(game);
|
|
|
|
|
createOrUpdateGame(boredGame);
|
|
|
|
|
games.push(boredGame);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return {
|
|
|
|
|
totalCount,
|
|
|
|
|
games
|
|
|
|
|
};
|
|
|
|
|
} else {
|
|
|
|
|
return {
|
|
|
|
|
totalCount: dbGames.length,
|
|
|
|
|
dbGames
|
|
|
|
|
};
|
2022-12-25 18:52:36 +00:00
|
|
|
}
|
|
|
|
|
} catch (e) {
|
|
|
|
|
console.log(`Error searching board games ${e}`);
|
|
|
|
|
}
|
|
|
|
|
return {
|
|
|
|
|
totalCount: 0,
|
2023-05-15 04:08:30 +00:00
|
|
|
games: []
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-26 06:14:26 +00:00
|
|
|
async function createOrUpdateGame(game: GameType) {
|
|
|
|
|
const categoryIds = game.categories.map((category) => ({
|
|
|
|
|
external_id: category.id
|
|
|
|
|
}));
|
|
|
|
|
const mechanicIds = game.mechanics.map((mechanic) => ({
|
|
|
|
|
external_id: mechanic.id
|
|
|
|
|
}));
|
|
|
|
|
return await prisma.game.upsert({
|
|
|
|
|
where: {
|
|
|
|
|
external_id: game.id
|
|
|
|
|
},
|
|
|
|
|
create: {
|
|
|
|
|
name: game.name,
|
|
|
|
|
slug: kebabCase(game.name),
|
|
|
|
|
description: game.description,
|
|
|
|
|
description_preview: game.description_preview,
|
|
|
|
|
external_id: game.id,
|
|
|
|
|
thumb_url: game.thumb_url,
|
|
|
|
|
min_age: game.min_age,
|
|
|
|
|
min_players: game.min_players,
|
|
|
|
|
max_players: game.max_players,
|
|
|
|
|
min_playtime: game.min_playtime,
|
|
|
|
|
max_playtime: game.max_playtime,
|
|
|
|
|
year_published: game.year_published,
|
|
|
|
|
primary_publisher: {
|
|
|
|
|
connectOrCreate: {
|
|
|
|
|
where: {
|
|
|
|
|
external_id: game.primary_publisher.id
|
|
|
|
|
},
|
|
|
|
|
create: {
|
|
|
|
|
external_id: game.primary_publisher.id,
|
|
|
|
|
name: game.primary_publisher.name,
|
|
|
|
|
slug: kebabCase(game.primary_publisher.name)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
categories: {
|
|
|
|
|
connect: categoryIds
|
|
|
|
|
},
|
|
|
|
|
mechanics: {
|
|
|
|
|
connect: mechanicIds
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
update: {
|
|
|
|
|
name: game.name,
|
|
|
|
|
slug: kebabCase(game.name),
|
|
|
|
|
description: game.description,
|
|
|
|
|
description_preview: game.description_preview,
|
|
|
|
|
external_id: game.id,
|
|
|
|
|
thumb_url: game.thumb_url,
|
|
|
|
|
min_age: game.min_age,
|
|
|
|
|
min_players: game.min_players,
|
|
|
|
|
max_players: game.max_players,
|
|
|
|
|
min_playtime: game.min_playtime,
|
|
|
|
|
max_playtime: game.max_playtime,
|
|
|
|
|
year_published: game.year_published,
|
|
|
|
|
primary_publisher: {
|
|
|
|
|
connectOrCreate: {
|
|
|
|
|
where: {
|
|
|
|
|
external_id: game.primary_publisher.id
|
|
|
|
|
},
|
|
|
|
|
create: {
|
|
|
|
|
external_id: game.primary_publisher.id,
|
|
|
|
|
name: game.primary_publisher.name
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
categories: {
|
|
|
|
|
connect: categoryIds
|
|
|
|
|
},
|
|
|
|
|
mechanics: {
|
|
|
|
|
connect: mechanicIds
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export const load = async (event) => {
|
|
|
|
|
const { params, locals, request, fetch, url } = event;
|
2023-05-15 04:08:30 +00:00
|
|
|
const defaults = {
|
|
|
|
|
limit: 10,
|
|
|
|
|
skip: 0
|
|
|
|
|
};
|
|
|
|
|
const searchParams = Object.fromEntries(url?.searchParams);
|
|
|
|
|
searchParams.limit = searchParams.limit || `${defaults.limit}`;
|
|
|
|
|
searchParams.skip = searchParams.skip || `${defaults.skip}`;
|
|
|
|
|
const form = await superValidate(searchParams, search_schema);
|
|
|
|
|
|
|
|
|
|
const queryParams: SearchQuery = {
|
|
|
|
|
order_by: 'rank',
|
|
|
|
|
ascending: false,
|
|
|
|
|
limit: form.data?.limit,
|
|
|
|
|
skip: form.data?.skip,
|
|
|
|
|
client_id: BOARD_GAME_ATLAS_CLIENT_ID,
|
|
|
|
|
fuzzy_match: true,
|
2023-06-20 18:55:21 +00:00
|
|
|
name: form.data?.q
|
2023-05-15 04:08:30 +00:00
|
|
|
};
|
|
|
|
|
|
2023-06-20 18:55:21 +00:00
|
|
|
// fields: ('id,name,min_age,min_players,max_players,thumb_url,min_playtime,max_playtime,min_age,description');
|
|
|
|
|
|
2023-05-15 04:08:30 +00:00
|
|
|
if (form.data?.minAge) {
|
|
|
|
|
if (form.data?.exactMinAge) {
|
|
|
|
|
queryParams.min_age = form.data?.minAge;
|
|
|
|
|
} else {
|
|
|
|
|
queryParams.gt_min_age = form.data?.minAge === 1 ? 0 : form.data?.minAge - 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (form.data?.minPlayers) {
|
|
|
|
|
if (form.data?.exactMinPlayers) {
|
|
|
|
|
queryParams.min_players = form.data?.minPlayers;
|
|
|
|
|
} else {
|
|
|
|
|
queryParams.gt_min_players = form.data?.minPlayers === 1 ? 0 : form.data?.minPlayers - 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (form.data?.maxPlayers) {
|
|
|
|
|
if (form.data?.exactMaxPlayers) {
|
|
|
|
|
queryParams.max_players = form.data?.maxPlayers;
|
|
|
|
|
} else {
|
|
|
|
|
queryParams.lt_max_players = form.data?.maxPlayers + 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const newQueryParams: Record<string, string> = {};
|
|
|
|
|
for (const key in queryParams) {
|
|
|
|
|
newQueryParams[key] = `${queryParams[key as keyof SearchQuery]}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const urlQueryParams = new URLSearchParams(newQueryParams);
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
form,
|
|
|
|
|
searchData: await searchForGames(urlQueryParams)
|
2022-12-25 18:52:36 +00:00
|
|
|
};
|
|
|
|
|
};
|
2022-11-01 00:31:36 +00:00
|
|
|
|
2023-06-02 05:22:34 +00:00
|
|
|
export const actions = {
|
2023-05-21 05:18:04 +00:00
|
|
|
random: async ({ request }): Promise<any> => {
|
|
|
|
|
const form = await superValidate(request, search_schema);
|
2022-12-25 18:52:36 +00:00
|
|
|
const queryParams: SearchQuery = {
|
|
|
|
|
order_by: 'rank',
|
|
|
|
|
ascending: false,
|
|
|
|
|
client_id: BOARD_GAME_ATLAS_CLIENT_ID,
|
2023-01-08 19:46:22 +00:00
|
|
|
random: true,
|
|
|
|
|
fields:
|
|
|
|
|
'id,name,min_age,min_players,max_players,thumb_url,min_playtime,max_playtime,min_age,description'
|
2022-12-25 18:52:36 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const newQueryParams: Record<string, string> = {};
|
|
|
|
|
for (const key in queryParams) {
|
|
|
|
|
newQueryParams[key] = `${queryParams[key as keyof SearchQuery]}`;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const urlQueryParams = new URLSearchParams(newQueryParams);
|
|
|
|
|
|
|
|
|
|
return {
|
2023-05-21 05:18:04 +00:00
|
|
|
form,
|
|
|
|
|
searchData: await searchForGames(urlQueryParams)
|
2022-12-25 18:52:36 +00:00
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
};
|