mirror of
https://github.com/aaronpo97/the-biergarten-app.git
synced 2026-02-16 18:52:06 +00:00
Add create beer, beer post page
This commit is contained in:
@@ -1,13 +1,11 @@
|
|||||||
|
import { NewBeerInfo } from '@/pages/api/beers/create';
|
||||||
import BreweryPostQueryResult from '@/services/BreweryPost/types/BreweryPostQueryResult';
|
import BreweryPostQueryResult from '@/services/BreweryPost/types/BreweryPostQueryResult';
|
||||||
|
import { BeerType } from '@prisma/client';
|
||||||
|
|
||||||
import { FunctionComponent } from 'react';
|
import { FunctionComponent } from 'react';
|
||||||
import { SubmitHandler, useForm } from 'react-hook-form';
|
import { SubmitHandler, useForm } from 'react-hook-form';
|
||||||
// import { useNavigate } from 'react-router-dom';
|
import { z } from 'zod';
|
||||||
// import createBeerPost from '../api/beerPostRoutes/createBeerPost';
|
|
||||||
// import getAllBreweryPosts from '../api/breweryPostRoutes/getAllBreweryPosts';
|
|
||||||
|
|
||||||
// import BreweryPostI from '../types/BreweryPostI';
|
|
||||||
// import isValidUuid from '../util/isValidUuid';
|
|
||||||
import Button from './ui/Button';
|
import Button from './ui/Button';
|
||||||
import FormError from './ui/forms/FormError';
|
import FormError from './ui/forms/FormError';
|
||||||
import FormInfo from './ui/forms/FormInfo';
|
import FormInfo from './ui/forms/FormInfo';
|
||||||
@@ -17,25 +15,37 @@ import FormSelect from './ui/forms/FormSelect';
|
|||||||
import FormTextArea from './ui/forms/FormTextArea';
|
import FormTextArea from './ui/forms/FormTextArea';
|
||||||
import FormTextInput from './ui/forms/FormTextInput';
|
import FormTextInput from './ui/forms/FormTextInput';
|
||||||
|
|
||||||
interface IFormInput {
|
type IFormInput = z.infer<typeof NewBeerInfo>;
|
||||||
name: string;
|
|
||||||
description: string;
|
|
||||||
type: string;
|
|
||||||
abv: number;
|
|
||||||
ibu: number;
|
|
||||||
breweryId: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface BeerFormProps {
|
interface BeerFormProps {
|
||||||
type: 'edit' | 'create';
|
type: 'edit' | 'create';
|
||||||
// eslint-disable-next-line react/require-default-props
|
// eslint-disable-next-line react/require-default-props
|
||||||
defaultValues?: IFormInput;
|
defaultValues?: IFormInput;
|
||||||
breweries?: BreweryPostQueryResult[];
|
breweries?: BreweryPostQueryResult[];
|
||||||
|
types?: BeerType[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const sendCreateBeerPostRequest = async (data: IFormInput) => {
|
||||||
|
// const body = JSON.stringify(data);
|
||||||
|
|
||||||
|
const response = await fetch('/api/beers/create', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
body: JSON.stringify(data),
|
||||||
|
});
|
||||||
|
|
||||||
|
const json = await response.json();
|
||||||
|
|
||||||
|
console.log(json);
|
||||||
|
};
|
||||||
|
|
||||||
const BeerForm: FunctionComponent<BeerFormProps> = ({
|
const BeerForm: FunctionComponent<BeerFormProps> = ({
|
||||||
type,
|
type,
|
||||||
defaultValues,
|
defaultValues,
|
||||||
breweries = [],
|
breweries = [],
|
||||||
|
types = [],
|
||||||
}) => {
|
}) => {
|
||||||
const {
|
const {
|
||||||
register,
|
register,
|
||||||
@@ -78,8 +88,24 @@ const BeerForm: FunctionComponent<BeerFormProps> = ({
|
|||||||
validate: (ibu) => !Number.isNaN(ibu) || 'IBU is invalid.',
|
validate: (ibu) => !Number.isNaN(ibu) || 'IBU is invalid.',
|
||||||
});
|
});
|
||||||
|
|
||||||
const onSubmit: SubmitHandler<IFormInput> = (data) => {
|
const descriptionValidationSchema = register('description', {
|
||||||
console.log(data);
|
required: 'Description is required.',
|
||||||
|
});
|
||||||
|
|
||||||
|
const typeIdValidationSchema = register('typeId', {
|
||||||
|
required: 'Type is required.',
|
||||||
|
});
|
||||||
|
|
||||||
|
const onSubmit: SubmitHandler<IFormInput> = async (data) => {
|
||||||
|
switch (type) {
|
||||||
|
case 'create':
|
||||||
|
await sendCreateBeerPostRequest(data);
|
||||||
|
break;
|
||||||
|
case 'edit':
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -156,27 +182,27 @@ const BeerForm: FunctionComponent<BeerFormProps> = ({
|
|||||||
<FormTextArea
|
<FormTextArea
|
||||||
placeholder="Ratione cumque quas quia aut impedit ea culpa facere. Ut in sit et quas reiciendis itaque."
|
placeholder="Ratione cumque quas quia aut impedit ea culpa facere. Ut in sit et quas reiciendis itaque."
|
||||||
error={!!errors.description}
|
error={!!errors.description}
|
||||||
formValidationSchema={register('description', {
|
formValidationSchema={descriptionValidationSchema}
|
||||||
required: 'Beer description is required.',
|
|
||||||
})}
|
|
||||||
id="description"
|
id="description"
|
||||||
rows={8}
|
rows={8}
|
||||||
/>
|
/>
|
||||||
</FormSegment>
|
</FormSegment>
|
||||||
|
|
||||||
<FormInfo>
|
<FormInfo>
|
||||||
<FormLabel htmlFor="type">Type</FormLabel>
|
<FormLabel htmlFor="typeId">Type</FormLabel>
|
||||||
<FormError>{errors.type?.message}</FormError>
|
<FormError>{errors.typeId?.message}</FormError>
|
||||||
</FormInfo>
|
</FormInfo>
|
||||||
<FormSegment>
|
<FormSegment>
|
||||||
<FormTextInput
|
<FormSelect
|
||||||
placeholder="Lagered Ale"
|
formRegister={typeIdValidationSchema}
|
||||||
error={!!errors.type}
|
error={!!errors.typeId}
|
||||||
formValidationSchema={register('type', {
|
id="typeId"
|
||||||
required: 'Beer type is required.',
|
options={types.map((beerType) => ({
|
||||||
})}
|
value: beerType.id,
|
||||||
id="type"
|
text: beerType.name,
|
||||||
type="text"
|
}))}
|
||||||
|
placeholder="Beer type"
|
||||||
|
message="Pick a beer type"
|
||||||
/>
|
/>
|
||||||
</FormSegment>
|
</FormSegment>
|
||||||
|
|
||||||
|
|||||||
31
package-lock.json
generated
31
package-lock.json
generated
@@ -8,6 +8,7 @@
|
|||||||
"name": "my-project",
|
"name": "my-project",
|
||||||
"version": "0.1.0",
|
"version": "0.1.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@hookform/resolvers": "^2.9.10",
|
||||||
"@next/font": "13.1.2",
|
"@next/font": "13.1.2",
|
||||||
"@prisma/client": "^4.8.1",
|
"@prisma/client": "^4.8.1",
|
||||||
"@types/node": "18.11.18",
|
"@types/node": "18.11.18",
|
||||||
@@ -24,7 +25,8 @@
|
|||||||
"react-dom": "18.2.0",
|
"react-dom": "18.2.0",
|
||||||
"react-hook-form": "^7.42.1",
|
"react-hook-form": "^7.42.1",
|
||||||
"react-icons": "^4.7.1",
|
"react-icons": "^4.7.1",
|
||||||
"typescript": "4.9.4"
|
"typescript": "4.9.4",
|
||||||
|
"zod": "^3.20.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@faker-js/faker": "^7.6.0",
|
"@faker-js/faker": "^7.6.0",
|
||||||
@@ -101,6 +103,14 @@
|
|||||||
"npm": ">=6.0.0"
|
"npm": ">=6.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@hookform/resolvers": {
|
||||||
|
"version": "2.9.10",
|
||||||
|
"resolved": "https://registry.npmjs.org/@hookform/resolvers/-/resolvers-2.9.10.tgz",
|
||||||
|
"integrity": "sha512-JIL1DgJIlH9yuxcNGtyhsWX/PgNltz+5Gr6+8SX9fhXc/hPbEIk6wPI82nhgvp3uUb6ZfAM5mqg/x7KR7NAb+A==",
|
||||||
|
"peerDependencies": {
|
||||||
|
"react-hook-form": "^7.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@humanwhocodes/config-array": {
|
"node_modules/@humanwhocodes/config-array": {
|
||||||
"version": "0.11.8",
|
"version": "0.11.8",
|
||||||
"resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.8.tgz",
|
"resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.8.tgz",
|
||||||
@@ -5216,6 +5226,14 @@
|
|||||||
"funding": {
|
"funding": {
|
||||||
"url": "https://github.com/sponsors/sindresorhus"
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"node_modules/zod": {
|
||||||
|
"version": "3.20.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/zod/-/zod-3.20.2.tgz",
|
||||||
|
"integrity": "sha512-1MzNQdAvO+54H+EaK5YpyEy0T+Ejo/7YLHS93G3RnYWh5gaotGHwGeN/ZO687qEDU2y4CdStQYXVHIgrUl5UVQ==",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/colinhacks"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -5260,6 +5278,12 @@
|
|||||||
"integrity": "sha512-XK6BTq1NDMo9Xqw/YkYyGjSsg44fbNwYRx7QK2CuoQgyy+f1rrTDHoExVM5PsyXCtfl2vs2vVJ0MN0yN6LppRw==",
|
"integrity": "sha512-XK6BTq1NDMo9Xqw/YkYyGjSsg44fbNwYRx7QK2CuoQgyy+f1rrTDHoExVM5PsyXCtfl2vs2vVJ0MN0yN6LppRw==",
|
||||||
"dev": true
|
"dev": true
|
||||||
},
|
},
|
||||||
|
"@hookform/resolvers": {
|
||||||
|
"version": "2.9.10",
|
||||||
|
"resolved": "https://registry.npmjs.org/@hookform/resolvers/-/resolvers-2.9.10.tgz",
|
||||||
|
"integrity": "sha512-JIL1DgJIlH9yuxcNGtyhsWX/PgNltz+5Gr6+8SX9fhXc/hPbEIk6wPI82nhgvp3uUb6ZfAM5mqg/x7KR7NAb+A==",
|
||||||
|
"requires": {}
|
||||||
|
},
|
||||||
"@humanwhocodes/config-array": {
|
"@humanwhocodes/config-array": {
|
||||||
"version": "0.11.8",
|
"version": "0.11.8",
|
||||||
"resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.8.tgz",
|
"resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.11.8.tgz",
|
||||||
@@ -8771,6 +8795,11 @@
|
|||||||
"resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
|
"resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
|
||||||
"integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
|
"integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
|
||||||
"dev": true
|
"dev": true
|
||||||
|
},
|
||||||
|
"zod": {
|
||||||
|
"version": "3.20.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/zod/-/zod-3.20.2.tgz",
|
||||||
|
"integrity": "sha512-1MzNQdAvO+54H+EaK5YpyEy0T+Ejo/7YLHS93G3RnYWh5gaotGHwGeN/ZO687qEDU2y4CdStQYXVHIgrUl5UVQ=="
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,6 +13,7 @@
|
|||||||
"seed": "npx ts-node ./prisma/seed/index.ts"
|
"seed": "npx ts-node ./prisma/seed/index.ts"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@hookform/resolvers": "^2.9.10",
|
||||||
"@next/font": "13.1.2",
|
"@next/font": "13.1.2",
|
||||||
"@prisma/client": "^4.8.1",
|
"@prisma/client": "^4.8.1",
|
||||||
"@types/node": "18.11.18",
|
"@types/node": "18.11.18",
|
||||||
@@ -29,7 +30,8 @@
|
|||||||
"react-dom": "18.2.0",
|
"react-dom": "18.2.0",
|
||||||
"react-hook-form": "^7.42.1",
|
"react-hook-form": "^7.42.1",
|
||||||
"react-icons": "^4.7.1",
|
"react-icons": "^4.7.1",
|
||||||
"typescript": "4.9.4"
|
"typescript": "4.9.4",
|
||||||
|
"zod": "^3.20.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@faker-js/faker": "^7.6.0",
|
"@faker-js/faker": "^7.6.0",
|
||||||
|
|||||||
92
pages/api/beers/create.ts
Normal file
92
pages/api/beers/create.ts
Normal file
@@ -0,0 +1,92 @@
|
|||||||
|
import DBClient from '@/prisma/DBClient';
|
||||||
|
import { NextApiHandler } from 'next';
|
||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
class ServerError extends Error {
|
||||||
|
constructor(message: string, public statusCode: number) {
|
||||||
|
super(message);
|
||||||
|
this.name = 'ServerError';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const NewBeerInfo = z.object({
|
||||||
|
name: z.string().min(1).max(100),
|
||||||
|
description: z.string().min(1).max(1000),
|
||||||
|
typeId: z.string().uuid(),
|
||||||
|
abv: z.number().min(1).max(50),
|
||||||
|
ibu: z.number().min(2),
|
||||||
|
breweryId: z.string().uuid(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ResponseBody = z.object({
|
||||||
|
message: z.string(),
|
||||||
|
statusCode: z.number(),
|
||||||
|
success: z.boolean(),
|
||||||
|
payload: z.unknown(),
|
||||||
|
});
|
||||||
|
|
||||||
|
const handler: NextApiHandler<z.infer<typeof ResponseBody>> = async (req, res) => {
|
||||||
|
try {
|
||||||
|
const { method } = req;
|
||||||
|
|
||||||
|
if (method !== 'POST') {
|
||||||
|
throw new ServerError('Method not allowed', 405);
|
||||||
|
}
|
||||||
|
|
||||||
|
const cleanedReqBody = NewBeerInfo.safeParse(req.body);
|
||||||
|
|
||||||
|
if (!cleanedReqBody.success) {
|
||||||
|
throw new ServerError('Invalid request body', 400);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { name, description, typeId, abv, ibu, breweryId } = cleanedReqBody.data;
|
||||||
|
|
||||||
|
const user = await DBClient.instance.user.findFirstOrThrow();
|
||||||
|
|
||||||
|
const newBeerPost = await DBClient.instance.beerPost.create({
|
||||||
|
data: {
|
||||||
|
name,
|
||||||
|
description,
|
||||||
|
abv,
|
||||||
|
ibu,
|
||||||
|
type: {
|
||||||
|
connect: {
|
||||||
|
id: typeId,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
postedBy: {
|
||||||
|
connect: {
|
||||||
|
id: user.id,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
brewery: {
|
||||||
|
connect: {
|
||||||
|
id: breweryId,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
res
|
||||||
|
.status(200)
|
||||||
|
.json({ message: 'Success', statusCode: 200, payload: newBeerPost, success: true });
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof ServerError) {
|
||||||
|
res.status(error.statusCode).json({
|
||||||
|
message: error.message,
|
||||||
|
statusCode: error.statusCode,
|
||||||
|
payload: null,
|
||||||
|
success: false,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
res.status(500).json({
|
||||||
|
message: 'Internal server error',
|
||||||
|
statusCode: 500,
|
||||||
|
payload: null,
|
||||||
|
success: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export default handler;
|
||||||
@@ -86,7 +86,25 @@ const BeerInfoHeader: React.FC<{ beerPost: BeerPostQueryResult }> = ({ beerPost
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const CommentCard: React.FC<{ comment: BeerPostQueryResult['beerComments'][number] }> = ({
|
||||||
|
comment,
|
||||||
|
}) => {
|
||||||
|
return (
|
||||||
|
<div className="card bg-base-300">
|
||||||
|
<div className="card-body">
|
||||||
|
<h3 className="text-2xl font-semibold">{comment.postedBy.username}</h3>
|
||||||
|
<h4 className="italic">{`posted ${formatDistanceStrict(
|
||||||
|
new Date(comment.createdAt),
|
||||||
|
new Date(),
|
||||||
|
)} ago`}</h4>
|
||||||
|
<p>{comment.content}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
const BeerByIdPage: NextPage<BeerPageProps> = ({ beerPost }) => {
|
const BeerByIdPage: NextPage<BeerPageProps> = ({ beerPost }) => {
|
||||||
|
console.log(beerPost.beerComments);
|
||||||
return (
|
return (
|
||||||
<Layout>
|
<Layout>
|
||||||
<Head>
|
<Head>
|
||||||
@@ -108,7 +126,13 @@ const BeerByIdPage: NextPage<BeerPageProps> = ({ beerPost }) => {
|
|||||||
<div className="mt-4 flex space-x-3">
|
<div className="mt-4 flex space-x-3">
|
||||||
<div className="w-[60%] space-y-3">
|
<div className="w-[60%] space-y-3">
|
||||||
<div className="card h-[22rem] bg-base-300"></div>
|
<div className="card h-[22rem] bg-base-300"></div>
|
||||||
<div className="card h-[44rem] bg-base-300"></div>
|
<div className="card h-[44rem] overflow-y-auto bg-base-300">
|
||||||
|
{/* for each comment make a card */}
|
||||||
|
|
||||||
|
{beerPost.beerComments.map((comment) => (
|
||||||
|
<CommentCard key={comment.id} comment={comment} />
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="w-[40%]">
|
<div className="w-[40%]">
|
||||||
<div className="card h-full bg-base-300"></div>
|
<div className="card h-full bg-base-300"></div>
|
||||||
|
|||||||
@@ -1,19 +1,29 @@
|
|||||||
import BeerForm from '@/components/BeerForm';
|
import BeerForm from '@/components/BeerForm';
|
||||||
import Layout from '@/components/Layout';
|
import Layout from '@/components/Layout';
|
||||||
|
import DBClient from '@/prisma/DBClient';
|
||||||
import getAllBreweryPosts from '@/services/BreweryPost/getAllBreweryPosts';
|
import getAllBreweryPosts from '@/services/BreweryPost/getAllBreweryPosts';
|
||||||
import BreweryPostQueryResult from '@/services/BreweryPost/types/BreweryPostQueryResult';
|
import BreweryPostQueryResult from '@/services/BreweryPost/types/BreweryPostQueryResult';
|
||||||
|
import { BeerType } from '@prisma/client';
|
||||||
import { NextPage } from 'next';
|
import { NextPage } from 'next';
|
||||||
|
|
||||||
|
import { BiBeer } from 'react-icons/bi';
|
||||||
|
|
||||||
interface CreateBeerPageProps {
|
interface CreateBeerPageProps {
|
||||||
breweries: BreweryPostQueryResult[];
|
breweries: BreweryPostQueryResult[];
|
||||||
|
types: BeerType[];
|
||||||
}
|
}
|
||||||
|
|
||||||
const Create: NextPage<CreateBeerPageProps> = ({ breweries }) => {
|
const Create: NextPage<CreateBeerPageProps> = ({ breweries, types }) => {
|
||||||
return (
|
return (
|
||||||
<Layout>
|
<Layout>
|
||||||
<div className="align-center flex h-full flex-col items-center justify-center">
|
<div className="align-center my-12 flex h-full flex-col items-center justify-center">
|
||||||
<div className="w-8/12">
|
<div className="w-8/12">
|
||||||
<BeerForm type="create" breweries={breweries} />
|
<div className="flex flex-col items-center space-y-1">
|
||||||
|
<BiBeer className="text-5xl" />
|
||||||
|
<h1 className="text-3xl font-bold">Create a New Beer</h1>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<BeerForm type="create" breweries={breweries} types={types} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Layout>
|
</Layout>
|
||||||
@@ -22,9 +32,12 @@ const Create: NextPage<CreateBeerPageProps> = ({ breweries }) => {
|
|||||||
|
|
||||||
export const getServerSideProps = async () => {
|
export const getServerSideProps = async () => {
|
||||||
const breweryPosts = await getAllBreweryPosts();
|
const breweryPosts = await getAllBreweryPosts();
|
||||||
|
|
||||||
|
const beerTypes = await DBClient.instance.beerType.findMany();
|
||||||
return {
|
return {
|
||||||
props: {
|
props: {
|
||||||
breweries: breweryPosts,
|
breweries: JSON.parse(JSON.stringify(breweryPosts)),
|
||||||
|
types: JSON.parse(JSON.stringify(beerTypes)),
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -96,7 +96,7 @@ export const getServerSideProps: GetServerSideProps<BeerPageProps> = async (cont
|
|||||||
|
|
||||||
const pageNumber = parseInt(query.page_num as string, 10) || 1;
|
const pageNumber = parseInt(query.page_num as string, 10) || 1;
|
||||||
|
|
||||||
const pageSize = 24;
|
const pageSize = 12;
|
||||||
const numberOfPosts = await DBClient.instance.beerPost.count();
|
const numberOfPosts = await DBClient.instance.beerPost.count();
|
||||||
const pageCount = numberOfPosts ? Math.ceil(numberOfPosts / pageSize) : 0;
|
const pageCount = numberOfPosts ? Math.ceil(numberOfPosts / pageSize) : 0;
|
||||||
|
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ datasource db {
|
|||||||
}
|
}
|
||||||
|
|
||||||
model User {
|
model User {
|
||||||
id String @id @default(cuid())
|
id String @id @default(uuid())
|
||||||
username String
|
username String
|
||||||
firstName String
|
firstName String
|
||||||
lastName String
|
lastName String
|
||||||
@@ -56,7 +56,7 @@ model BeerComment {
|
|||||||
}
|
}
|
||||||
|
|
||||||
model BeerType {
|
model BeerType {
|
||||||
id String @id @default(cuid())
|
id String @id @default(uuid())
|
||||||
name String
|
name String
|
||||||
createdAt DateTime @default(now()) @db.Timestamptz(3)
|
createdAt DateTime @default(now()) @db.Timestamptz(3)
|
||||||
updatedAt DateTime? @updatedAt @db.Timestamptz(3)
|
updatedAt DateTime? @updatedAt @db.Timestamptz(3)
|
||||||
@@ -66,7 +66,7 @@ model BeerType {
|
|||||||
}
|
}
|
||||||
|
|
||||||
model BreweryPost {
|
model BreweryPost {
|
||||||
id String @id @default(cuid())
|
id String @id @default(uuid())
|
||||||
name String
|
name String
|
||||||
location String
|
location String
|
||||||
beers BeerPost[]
|
beers BeerPost[]
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ import createNewUsers from './create/createNewUsers';
|
|||||||
|
|
||||||
const users = await createNewUsers({ numberOfUsers: 1000 });
|
const users = await createNewUsers({ numberOfUsers: 1000 });
|
||||||
const [breweryPosts, beerTypes] = await Promise.all([
|
const [breweryPosts, beerTypes] = await Promise.all([
|
||||||
createNewBreweryPosts({ numberOfPosts: 10000, joinData: { users } }),
|
createNewBreweryPosts({ numberOfPosts: 10, joinData: { users } }),
|
||||||
createNewBeerTypes({ joinData: { users } }),
|
createNewBeerTypes({ joinData: { users } }),
|
||||||
]);
|
]);
|
||||||
const beerPosts = await createNewBeerPosts({
|
const beerPosts = await createNewBeerPosts({
|
||||||
|
|||||||
@@ -32,6 +32,21 @@ const getAllBeerPosts = async (pageNum: number, pageSize: number) => {
|
|||||||
username: true,
|
username: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
beerComments: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
content: true,
|
||||||
|
createdAt: true,
|
||||||
|
postedBy: {
|
||||||
|
select: {
|
||||||
|
username: true,
|
||||||
|
id: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
beerImages: {
|
beerImages: {
|
||||||
select: {
|
select: {
|
||||||
url: true,
|
url: true,
|
||||||
|
|||||||
@@ -6,6 +6,19 @@ const prisma = DBClient.instance;
|
|||||||
const getBeerPostById = async (id: string) => {
|
const getBeerPostById = async (id: string) => {
|
||||||
const beerPost: BeerPostQueryResult | null = await prisma.beerPost.findFirst({
|
const beerPost: BeerPostQueryResult | null = await prisma.beerPost.findFirst({
|
||||||
select: {
|
select: {
|
||||||
|
beerComments: {
|
||||||
|
select: {
|
||||||
|
id: true,
|
||||||
|
content: true,
|
||||||
|
createdAt: true,
|
||||||
|
postedBy: {
|
||||||
|
select: {
|
||||||
|
username: true,
|
||||||
|
id: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
id: true,
|
id: true,
|
||||||
name: true,
|
name: true,
|
||||||
brewery: {
|
brewery: {
|
||||||
|
|||||||
@@ -22,5 +22,15 @@ export default interface BeerPostQueryResult {
|
|||||||
username: string;
|
username: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
beerComments: {
|
||||||
|
id: string;
|
||||||
|
content: string;
|
||||||
|
createdAt: Date;
|
||||||
|
postedBy: {
|
||||||
|
id: string;
|
||||||
|
username: string;
|
||||||
|
};
|
||||||
|
}[];
|
||||||
|
|
||||||
createdAt: Date;
|
createdAt: Date;
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user