A comprehensive Blog REST API built with Spring Boot that allows users to create, manage, and interact with posts, comments, and categories. The project includes user-based filtering, sorting, searching, pagination, and implements role-based access control with JWT authentication.
-
User Registration & Authentication
- JWT-based authentication
- Role-based access control (e.g., Admin, User)
-
Posts & Comments
- Users can create, update, delete, and view posts
- Any user can comment on posts
- Filter posts by user or category
-
Categories
- Posts can be organized into categories
- Users can retrieve posts based on categories
-
Sorting, Searching & Pagination
- Search for posts by title or content
- Sort posts by different fields (e.g., date, popularity)
- Paginate through results
-
Exception Handling
- Custom exception handling with detailed error messages
- Hibernate Validation for request payloads
-
Custom Response Handling
- Utilizes a custom
ApiResponse<T>
class for API responses instead ofResponseEntity<T>
- Utilizes a custom
-
File Service
- Handles file uploads related to posts
-
Database
- MySQL used as the database
- Relationships implemented:
- One-to-Many: A user can have multiple posts, and a post can have multiple comments
- Many-to-Many: Posts can belong to multiple categories, and a category can have multiple posts
-
Architecture
- 3-layer architecture: Controller, Service, DAO layers
- Data Transfer Objects (DTOs) for cleaner communication between layers
-
Documentation
- API is documented using Swagger
-
Testing
- Unit and integration tests using JUnit5
- Backend: Spring Boot
- Security: JWT Authentication, Role-based access control
- Database: MySQL
- ORM: Hibernate
- Validation: Hibernate Validator
- Testing: JUnit5
- API Documentation: Swagger
- File Handling: Custom File Service
- Clone the repository:
https://github.com/SakibvHossain/Blogging_Application_Backend.git
- Navigate to the project directory:
cd blog-rest-api
- Configure the database in
application.properties
:spring.datasource.url=jdbc:mysql://localhost:3306/your-database spring.datasource.username=your-username spring.datasource.password=your-password
- Run the application:
mvn spring-boot:run
POST /api/auth/login
: Authenticate and receive JWTPOST /api/auth/register
: Register a new user
GET /api/users/all
: Get all users (Admin user only)PUT /api/users/updateUser/{id}
: Update user (authenticated user only)DELETE /api/users/deleteUser/{id}
: Delete user (Admin user or authenticated user only)GET /api/users/{id}
: Get specific User (Admin user only)PATCH /api/users/update/specific/{id}
: Update specific field (Admin or authenticated user only)
GET /api/post/all?pageNumber=1&pageSize=3
: Get all posts (with pagination, sorting, and searching)POST /api/user/{userID}/category/{categoryID}
: Create a new post (authenticated users only)PUT /api/post/update/{id}
: Update a post (authenticated users only)DELETE /api/post/delete/{id}
: Delete a post (admin users only)GET /api/postBy/category/{id}?pageNumber=1&pageSize=2
: Get post by ID also based on page number and size (authenticated users only)GET /api/postBy/author/{id}
: Get post based on author (authenticated user only)GET /api/post/{id}
: Get post by ID (authenticated user only)GET /api/post/search/{title}
: Get post by title (authenticated user only)PUT /api/post/upload/image/{postID}
: Update post image by ID (authenticated user only)
GET /api/category/all
: Get all categoriesPOST /api/category/create
: Create a new category (admin users only)PUT /api/category/update/{id}
: Update a category (admin users only)DELETE /api/category/delete/{id}
: Delete a category (admin users only)GET /api/category/byId/{id}
: Get categories by ID (authenticated users only)
POST /api/comment/add/post/{id}
: Add a comment to a post (authenticated users only)DELETE /api/comment/delete/{id}
: Delete post specific post (authenticated users only)
Feel free to fork this repository, create a new branch, and submit a pull request. All contributions are welcome!
The project is done by following Multi-tier Architecture (such as separating concerns into layers like Controller, Service, and Repository), the main design principle I'm following is Separation of Concerns (SoC).
Separation of Concerns (SoC): This is the primary principle I'm following with multi-tier architecture. Each layer (Controller, Service, Repository) has its own responsibility:
- Controller: Handles incoming HTTP requests.
- Service: Contains business logic.
- Repository/DAO: Manages data persistence and interaction with the database.
DRY principle aims to reduce duplication in code, ensuring that any piece of logic, data, or behavior is implemented in only one place. This makes code easier to maintain, less error-prone, and more scalable.
UserServiceImpl.java class not fully following DRY principle. Because I'm using ResourceNotFoundException multiple place where duplication occurs.
CategoryServiceImpl.java class completely following DRY principle. Instead of repeating this logic in every method (e.g., getCategoryById, deleteCategory, updateCategory), I call findCategoryByIdOrThrow method to streamline the process and ensure consistency.
The goal is to avoid over-engineering. The KISS principle (Keep It Simple, Stupid) is about focusing on simplicity, clarity, and efficiency, using only what is necessary to meet the requirements. The hole project follows the KISS principle. No over engineering uses on the project.
This project not following all 5 of them but 2 of them are following such as:
- Following:
- SRP (Single Responsibility Principle): Each class has one responsibility.
- LSP (Liskov Substitution Principle): Interfaces are implemented correctly.
- ISP (Interface Segregation Principle): No unnecessary methods in interfaces.
- Partially Following:
- DIP (Dependency Inversion Principle): Good use of dependency injection, but could benefit from more abstraction in some areas.
- Not Fully Following:
- OCP (Open/Closed Principle): Modifying classes may be necessary for adding new features. Introducing extensions could improve this.
Note: However, don’t force SOLID into every corner of your project, especially when simpler, more straightforward solutions suffice.
- Add dependency:
<!-- Core API dependency --> <dependency> <groupId>io.jsonwebtoken</groupId> <artifactId>jjwt-api</artifactId> <version>0.12.6</version> </dependency> <!-- Implementation for runtime functionality --> <dependency> <groupId>io.jsonwebtoken</groupId> <artifactId>jjwt-impl</artifactId> <version>0.12.6</version> <scope>runtime</scope> </dependency> <!-- JSON processing (optional, recommended for Claims) --> <dependency> <groupId>io.jsonwebtoken</groupId> <artifactId>jjwt-jackson</artifactId> <version>0.12.6</version> <scope>runtime</scope> </dependency>
- Create class JwtAuthenticationEntryPoint implements AuthenticationEntryPoint: This class will be called when unauthorized user trying to access our APIs.
- Create class JwtTokenHelper: This class will contain lots of methods to perform operations over token like:
- Generating Token
- Validating Token
- Token expired or not
- Create class JwtAuthenticationFilter extends OnceRequestFilter: This class will work like filter before reaching to API.
- Create class JwtAuthResponse: We will hold the token here.
- Configure JWT in Spring Security config: We will configure and tell when to start, where to go & where to take action.
- Create Login API to return token: If user authenticated we will generate token and return it.