diff --git a/.gitignore b/.gitignore new file mode 100644 index 00000000..b512c09d --- /dev/null +++ b/.gitignore @@ -0,0 +1 @@ +node_modules \ No newline at end of file diff --git a/Dockerfile b/Dockerfile index baec7ba2..5eacd6f5 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1 +1,14 @@ #TODO Configure o Dockerfile +FROM node:20-alpine AS build + +WORKDIR /usr/app + +COPY package.json package-lock.json* ./ + +RUN npm install + +COPY . . + +EXPOSE 3000 + +CMD ["npm", "run", "dev"] \ No newline at end of file diff --git a/docker-compose.yml b/docker-compose.yml index 2ac8411e..35a82757 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -11,7 +11,8 @@ services: - DB_PASSWORD=password - DB_NAME=test_db depends_on: - - db + db: + condition: service_healthy db: @@ -24,6 +25,12 @@ services: volumes: - mysql-data:/var/lib/mysql - ./init.sql:/docker-entrypoint-initdb.d/init.sql + healthcheck: + test: ["CMD", "mysqladmin", "ping", "-h", "127.0.0.1", "-u", "root", "-p$$MYSQL_ROOT_PASSWORD"] + interval: 5s + timeout: 30s + retries: 5 + start_period: 0s volumes: mysql-data: diff --git a/init.sql b/init.sql index e2c36c6f..a0fa8936 100644 --- a/init.sql +++ b/init.sql @@ -1,5 +1,18 @@ USE test_db; ---TODO Crie a tabela de user; +CREATE TABLE IF NOT EXISTS `user` ( + id int AUTO_INCREMENT NOT NULL, + firstName varchar(100) NOT NULL, + lastName varchar(100) NOT NULL, + email varchar(100) NOT NULL, + PRIMARY KEY (id) +); ---TODO Crie a tabela de posts; +CREATE TABLE IF NOT EXISTS post ( + id int AUTO_INCREMENT NOT NULL, + title varchar(100) NOT NULL, + description varchar(100) NOT NULL, + userId int NOT NULL, + PRIMARY KEY (id), + FOREIGN KEY (userId) REFERENCES `user`(id) +); \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index febfdc4f..768d64f7 100644 --- a/package-lock.json +++ b/package-lock.json @@ -10,7 +10,8 @@ "express": "^4.18.2", "mysql2": "^3.6.1", "reflect-metadata": "^0.1.13", - "typeorm": "^0.3.17" + "typeorm": "^0.3.17", + "zod": "^3.23.8" }, "devDependencies": { "@types/express": "^4.17.17", @@ -2583,6 +2584,15 @@ "engines": { "node": ">=6" } + }, + "node_modules/zod": { + "version": "3.23.8", + "resolved": "https://registry.npmjs.org/zod/-/zod-3.23.8.tgz", + "integrity": "sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==", + "license": "MIT", + "funding": { + "url": "https://github.com/sponsors/colinhacks" + } } } } diff --git a/package.json b/package.json index 46cdba25..3164ccf0 100644 --- a/package.json +++ b/package.json @@ -8,11 +8,12 @@ "test": "ts-node src/db.test.ts" }, "dependencies": { + "axios": "^1.5.0", "express": "^4.18.2", "mysql2": "^3.6.1", - "typeorm": "^0.3.17", "reflect-metadata": "^0.1.13", - "axios": "^1.5.0" + "typeorm": "^0.3.17", + "zod": "^3.23.8" }, "devDependencies": { "@types/express": "^4.17.17", diff --git a/src/controllers/PostController.ts b/src/controllers/PostController.ts new file mode 100644 index 00000000..fe9f5d45 --- /dev/null +++ b/src/controllers/PostController.ts @@ -0,0 +1,18 @@ +import { PostService } from "../services/PostService"; +import { Request, Response } from "express" + +export class PostController { + private postService = new PostService() + + async createPost(req: Request, res: Response) { + const { title, description, userId } = req.body; + + try { + const createPost = await this.postService.createPost(title, description, userId) + return res.status(201).json(createPost) + } catch (error) { + console.error("Error while creating post: ", error) + return res.status(500).json({ message: "Internal Server Error"}) + } + } +} \ No newline at end of file diff --git a/src/controllers/UserController.ts b/src/controllers/UserController.ts new file mode 100644 index 00000000..09db8ab0 --- /dev/null +++ b/src/controllers/UserController.ts @@ -0,0 +1,18 @@ +import { UserService } from "../services/UserService"; +import { Request, Response } from "express" + +export class UserController { + private userService = new UserService() + + async createUser(req: Request, res: Response) { + const { firstName, lastName, email } = req.body; + + try { + const createUser = await this.userService.createUser(firstName, lastName, email) + return res.status(201).json(createUser) + } catch(error) { + console.error("Error while creating user: ", error) + return res.status(500).json({ message: "Internal Server Error"}) + } + } +} \ No newline at end of file diff --git a/src/entity/Post.ts b/src/entity/Post.ts index a1f68038..3322bd57 100644 --- a/src/entity/Post.ts +++ b/src/entity/Post.ts @@ -1,3 +1,20 @@ -import { Entity, PrimaryGeneratedColumn, Column } from "typeorm"; +import { Entity, PrimaryGeneratedColumn, Column, ManyToOne } from "typeorm"; +import { User } from "./User"; -//TODO Crie a entidade de Post +@Entity() +export class Post { + @PrimaryGeneratedColumn() + id: number + + @Column({ type: "varchar", length: 100 }) + title: string + + @Column({ type: "varchar", length: 100 }) + description: string + + @ManyToOne(() => User, (user) => user.posts, { nullable: false }) + user: User + + @Column() + userId: number +} \ No newline at end of file diff --git a/src/entity/User.ts b/src/entity/User.ts index a8e22632..406ac08f 100644 --- a/src/entity/User.ts +++ b/src/entity/User.ts @@ -1,3 +1,20 @@ -import { Entity, PrimaryGeneratedColumn, Column } from "typeorm"; +import { Entity, PrimaryGeneratedColumn, Column, OneToMany } from "typeorm"; +import { Post } from "./Post"; -//TODO Crie a entidade de User +@Entity() +export class User { + @PrimaryGeneratedColumn() + id: number + + @Column({ type: "varchar", length: 100 }) + firstName: string + + @Column({ type: "varchar", length: 100 }) + lastName: string + + @Column({ type: "varchar", length: 100 }) + email: string + + @OneToMany(() => Post, (post) => post.user) + posts: Post[] +} \ No newline at end of file diff --git a/src/index.ts b/src/index.ts index 1af52a84..9389b0d0 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,47 +1,29 @@ import 'reflect-metadata'; import express from 'express'; -import { DataSource } from 'typeorm'; -import { User } from './entity/User'; -import { Post } from './entity/Post'; +import database from './infra/db'; +import userRoutes from './routes/userRoutes'; +import postRoutes from './routes/postRoutes'; const app = express(); app.use(express.json()); -const AppDataSource = new DataSource({ - type: "mysql", - host: process.env.DB_HOST || "localhost", - port: 3306, - username: process.env.DB_USER || "root", - password: process.env.DB_PASSWORD || "password", - database: process.env.DB_NAME || "test_db", - entities: [User,Post], - synchronize: true, -}); - -const wait = (ms: number) => new Promise(resolve => setTimeout(resolve, ms)); - -const initializeDatabase = async () => { - await wait(20000); +const waitForDatabase = async () => { try { - await AppDataSource.initialize(); - console.log("Data Source has been initialized!"); - } catch (err) { - console.error("Error during Data Source initialization:", err); + await database.initializeDatabase(); + console.log('Database initialized'); + } catch (error) { + console.error('Error initializing the database:', error); process.exit(1); - } -}; + }; +} -initializeDatabase(); +waitForDatabase(); -app.post('/users', async (req, res) => { -// Crie o endpoint de users -}); - -app.post('/posts', async (req, res) => { -// Crie o endpoint de posts -}); +app.use("/users", userRoutes) +app.use("/posts", postRoutes) const PORT = process.env.PORT || 3000; + app.listen(PORT, () => { console.log(`Server is running on port ${PORT}`); -}); +}); \ No newline at end of file diff --git a/src/infra/db.ts b/src/infra/db.ts new file mode 100644 index 00000000..69079619 --- /dev/null +++ b/src/infra/db.ts @@ -0,0 +1,34 @@ +import { DataSource } from 'typeorm'; +import { User } from '../entity/User'; +import { Post } from '../entity/Post'; + +const AppDataSource = new DataSource({ + type: "mysql", + host: process.env.DB_HOST || "localhost", + port: 3306, + username: process.env.DB_USER || "root", + password: process.env.DB_PASSWORD || "password", + database: process.env.DB_NAME || "test_db", + entities: [User,Post], + logging: true, + synchronize: true, +}); + +const wait = (ms: number) => new Promise(resolve => setTimeout(resolve, ms)); + +const initializeDatabase = async () => { + try { + await AppDataSource.initialize(); + console.log("Data Source has been initialized!"); + } catch (err) { + console.error("Error during Data Source initialization:", err); + process.exit(1); + } +}; + +const database = { + AppDataSource, + initializeDatabase, +} + +export default database; \ No newline at end of file diff --git a/src/middlewares/validationMiddleware.ts b/src/middlewares/validationMiddleware.ts new file mode 100644 index 00000000..4ab6acbf --- /dev/null +++ b/src/middlewares/validationMiddleware.ts @@ -0,0 +1,20 @@ +import { z, ZodError, ZodIssue } from "zod"; +import { Request, Response, NextFunction } from "express"; + +export function validateFields(schema: z.ZodObject) { + return (req: Request, res: Response, next: NextFunction) => { + try { + schema.parse(req.body) + next() + } catch (error) { + if (error instanceof ZodError) { + const errorformatted = error.flatten((issue: ZodIssue) => ({ + message: issue.message, + })); + res.status(400).json({ error: "Invalid data", details: errorformatted.fieldErrors}) + } else { + res.status(500).json(({ message: "Internal Server Error"})) + } + } + } +} \ No newline at end of file diff --git a/src/routes/postRoutes.ts b/src/routes/postRoutes.ts new file mode 100644 index 00000000..cad904b1 --- /dev/null +++ b/src/routes/postRoutes.ts @@ -0,0 +1,15 @@ +import { Router } from "express"; +import { Request, Response } from "express" +import { PostController } from "../controllers/PostController"; +import { validateFields } from "../middlewares/validationMiddleware"; +import { postSchemaValidation } from "../schemas/postSchema"; + +const postRoutes = Router() +const postController = new PostController(); + +postRoutes.post("/", + validateFields(postSchemaValidation), (req: Request, res: Response) => + postController.createPost(req, res) +) + +export default postRoutes; \ No newline at end of file diff --git a/src/routes/userRoutes.ts b/src/routes/userRoutes.ts new file mode 100644 index 00000000..f93f6b57 --- /dev/null +++ b/src/routes/userRoutes.ts @@ -0,0 +1,15 @@ +import { Router } from "express"; +import { Request, Response } from "express" +import { UserController } from "../controllers/UserController"; +import { validateFields } from "../middlewares/validationMiddleware"; +import { userSchemaValidation } from "../schemas/userSchema"; + +const userRoutes = Router(); +const userController = new UserController(); + +userRoutes.post("/", + validateFields(userSchemaValidation), (req: Request, res: Response) => + userController.createUser(req, res) +) + +export default userRoutes; \ No newline at end of file diff --git a/src/schemas/postSchema.ts b/src/schemas/postSchema.ts new file mode 100644 index 00000000..11a3b751 --- /dev/null +++ b/src/schemas/postSchema.ts @@ -0,0 +1,7 @@ +import { z } from "zod"; + +export const postSchemaValidation = z.object({ + title: z.string().min(1).max(100), + description: z.string().min(1).max(100), + userId: z.number().int().positive().min(1), +}) \ No newline at end of file diff --git a/src/schemas/userSchema.ts b/src/schemas/userSchema.ts new file mode 100644 index 00000000..cd9d6391 --- /dev/null +++ b/src/schemas/userSchema.ts @@ -0,0 +1,7 @@ +import { z } from "zod"; + +export const userSchemaValidation = z.object({ + firstName: z.string().min(1).max(100), + lastName: z.string().min(1).max(100), + email: z.string().email(), +}) \ No newline at end of file diff --git a/src/services/PostService.ts b/src/services/PostService.ts new file mode 100644 index 00000000..6729fb67 --- /dev/null +++ b/src/services/PostService.ts @@ -0,0 +1,13 @@ +import { Post } from "../entity/Post"; +import { User } from "../entity/User"; +import database from "../infra/db"; + +export class PostService { + private PostRepository = database.AppDataSource.getRepository(Post) + + async createPost(title: string, description: string, userId: number) : Promise { + const post = this.PostRepository.create({ title, description, userId }) + + return await this.PostRepository.save(post); + } +} \ No newline at end of file diff --git a/src/services/UserService.ts b/src/services/UserService.ts new file mode 100644 index 00000000..5a36faef --- /dev/null +++ b/src/services/UserService.ts @@ -0,0 +1,12 @@ +import { User } from "../entity/User"; +import database from "../infra/db"; + +export class UserService { + private userRepository = database.AppDataSource.getRepository(User) + + async createUser(firstName: string, lastName: string, email: string) : Promise { + const user = this.userRepository.create({ firstName, lastName, email}) + + return await this.userRepository.save(user); + } +} \ No newline at end of file diff --git a/tsconfig.json b/tsconfig.json index cb1c9da8..64afdc7d 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -9,6 +9,7 @@ "esModuleInterop": true, "emitDecoratorMetadata": true, "experimentalDecorators": true, + "strictPropertyInitialization": false, "skipLibCheck": true }, "include": ["src/**/*", "src/db.test.ts"],