Add dinosaur CRUD validation

This commit is contained in:
Tomas Krejci 2026-06-17 19:05:27 +02:00
parent 3739142e6b
commit f432aa7d87
7 changed files with 186 additions and 5 deletions

View File

@ -9,3 +9,29 @@ GET http://localhost:3000/dinosaurs/not-a-number
### Missing dinosaur
GET http://localhost:3000/dinosaurs/999999
### Create a dinosaur
POST http://localhost:3000/dinosaurs
Content-Type: application/json
{
"name": "Triceratops",
"description" : "Triceratops (neboli „třírohá tvář“) …",
"period": "křída",
"wikipediaAddress": "https://cs.wikipedia.org/wiki/Triceratops"
}
### Replace a dinosaur
PUT http://localhost:3000/dinosaurs/16
Content-Type: application/json
{
"name": "Tyrannosaurus",
"description": "Tyrannosaurus …",
"period": "křída",
"wikipediaAddress": "https://en.wikipedia.org/wiki/Tyrannosaurus"
}
### Delete a dinosaur
DELETE http://localhost:3000/dinosaurs/2

View File

@ -13,7 +13,7 @@ const client = new Client({
const app = express();
app.get("/dinosaurs", async (request, response) => {
app.get("/dinosaurs", async (_request, response) => {
const result = await client.query("SELECT * FROM dinosaur ORDER BY id");
const dinosaurs = result.rows.map((row) => ({
id: row.id,

View File

@ -10,7 +10,8 @@
"dependencies": {
"dotenv": "^17.4.2",
"express": "^5.1.0",
"pg": "^8.11.3"
"pg": "^8.11.3",
"zod": "^4.4.3"
},
"devDependencies": {
"@eslint/js": "^10.0.1",
@ -1791,6 +1792,14 @@
"funding": {
"url": "https://github.com/sponsors/sindresorhus"
}
},
"node_modules/zod": {
"version": "4.4.3",
"resolved": "https://registry.npmjs.org/zod/-/zod-4.4.3.tgz",
"integrity": "sha512-ytENFjIJFl2UwYglde2jchW2Hwm4GJFLDiSXWdTrJQBIN9Fcyp7n4DhxJEiWNAJMV1/BqWfW/kkg71UDcHJyTQ==",
"funding": {
"url": "https://github.com/sponsors/colinhacks"
}
}
}
}

View File

@ -11,7 +11,8 @@
"dependencies": {
"dotenv": "^17.4.2",
"express": "^5.1.0",
"pg": "^8.11.3"
"pg": "^8.11.3",
"zod": "^4.4.3"
},
"devDependencies": {
"@eslint/js": "^10.0.1",

View File

@ -55,4 +55,54 @@ export const dinosaurRepository = {
return mapRowToDinosaurDto(result.rows[0]);
},
/**
* Creates a dinosaur and returns its generated ID.
*
* @param {DinosaurWithoutId} dinosaur
* @returns {Promise<number>}
*/
async createDinosaur(dinosaur) {
const result = await client.query(
`INSERT INTO dinosaur (name, description, period, wikipedia_address)
VALUES ($1, $2, $3, $4)
RETURNING id`,
[dinosaur.name, dinosaur.description, dinosaur.period, dinosaur.wikipediaAddress],
);
return result.rows[0].id;
},
/**
* Updates a dinosaur by its ID.
*
* @param {number} id
* @param {DinosaurWithoutId} dinosaur
* @returns {Promise<boolean>}
*/
async updateDinosaurById(id, dinosaur) {
const result = await client.query(
`UPDATE dinosaur
SET name = $2,
description = $3,
period = $4,
wikipedia_address = $5
WHERE id = $1`,
[id, dinosaur.name, dinosaur.description, dinosaur.period, dinosaur.wikipediaAddress],
);
return result.rowCount > 0;
},
/**
* Deletes a dinosaur by its ID.
*
* @param {number} id
* @returns {Promise<boolean>}
*/
async deleteDinosaurById(id) {
const result = await client.query("DELETE FROM dinosaur WHERE id = $1", [id]);
return result.rowCount > 0;
},
};

View File

@ -1,8 +1,16 @@
import express from "express";
import { dinosaurIdParamSchema, dinosaurBodySchema } from "../validation/dinosaurSchemas.js";
import { dinosaurRepository } from "../repositories/dinosaurRepository.js";
export const dinosaurRouter = express.Router();
const formatValidationErrors = (issues) => {
return issues.map((issue) => ({
field: issue.path.join("."),
message: issue.message,
}));
};
// Return all dinosaurs through the repository abstraction.
dinosaurRouter.get("/dinosaurs", async (_request, response) => {
const dinosaurs = await dinosaurRepository.listDinosaurs();
@ -11,14 +19,16 @@ dinosaurRouter.get("/dinosaurs", async (_request, response) => {
// Return one dinosaur identified by a positive integer ID.
dinosaurRouter.get("/dinosaurs/:id", async (request, response) => {
const id = Number(request.params.id);
const parsedParams = dinosaurIdParamSchema.safeParse(request.params);
if (!Number.isInteger(id) || id <= 0) {
if (!parsedParams.success) {
return response.status(400).json({
message: "Dinosaur ID must be a positive integer",
errors: formatValidationErrors(parsedParams.error.issues),
});
}
const { id } = parsedParams.data;
const dinosaur = await dinosaurRepository.getDinosaurById(id);
if (dinosaur === null) {
@ -29,3 +39,71 @@ dinosaurRouter.get("/dinosaurs/:id", async (request, response) => {
response.json(dinosaur);
});
// Insert one dinosaur into the database.
dinosaurRouter.post("/dinosaurs", async (request, response) => {
const parsedBody = dinosaurBodySchema.safeParse(request.body);
if (!parsedBody.success) {
return response.status(400).json({
message: "Invalid dinosaur payload",
errors: formatValidationErrors(parsedBody.error.issues),
});
}
const id = await dinosaurRepository.createDinosaur(parsedBody.data);
response.status(201).json({ id });
});
// Replace one dinosaur identified by a positive integer ID.
dinosaurRouter.put("/dinosaurs/:id", async (request, response) => {
const parsedParams = dinosaurIdParamSchema.safeParse(request.params);
if (!parsedParams.success) {
return response.status(400).json({
message: "Dinosaur ID must be a positive integer",
errors: formatValidationErrors(parsedParams.error.issues),
});
}
const parsedBody = dinosaurBodySchema.safeParse(request.body);
if (!parsedBody.success) {
return response.status(400).json({
message: "Invalid dinosaur payload",
errors: formatValidationErrors(parsedBody.error.issues),
});
}
const updated = await dinosaurRepository.updateDinosaurById(parsedParams.data.id, parsedBody.data);
if (!updated) {
return response.status(404).json({
message: "Dinosaur not found",
});
}
response.sendStatus(204);
});
// Delete one dinosaur identified by a positive integer ID.
dinosaurRouter.delete("/dinosaurs/:id", async (request, response) => {
const parsedParams = dinosaurIdParamSchema.safeParse(request.params);
if (!parsedParams.success) {
return response.status(400).json({
message: "Dinosaur ID must be a positive integer",
errors: formatValidationErrors(parsedParams.error.issues),
});
}
const deleted = await dinosaurRepository.deleteDinosaurById(parsedParams.data.id);
if (!deleted) {
return response.status(404).json({
message: "Dinosaur not found",
});
}
response.sendStatus(204);
});

View File

@ -0,0 +1,17 @@
import { z } from "zod";
// Route schemas keep HTTP input aligned with the database constraints in schema.sql.
export const dinosaurIdParamSchema = z
.object({
id: z.coerce.number().int("id must be an integer").min(1, { message: "id must be greater than or equal to 1" }),
})
.strict();
export const dinosaurBodySchema = z
.object({
name: z.string().min(1, { message: "name must not be an empty string" }).max(256),
description: z.string().min(1, { message: "description must not be an empty string" }).max(4096),
period: z.string().min(1, { message: "period must not be an empty string" }).max(32),
wikipediaAddress: z.url().max(4096),
})
.strict();