Develop an Express application with TypeScript, integrating MongoDB with Mongoose to manage a Stationery Shop. Ensure data integrity using Mongoose schema validation.
- Create an Express project with TypeScript.
- Set up a MongoDB database to store Stationery Products and Orders.
- Use Mongoose for schema definition and data operations.
- Implement CRUD operations for both stationery products and orders.
- Stationery Product Model
- name (string): The name of the product (e.g., Pen, Notebook, Eraser).
- brand (string): The brand of the product (e.g., Pilot, Moleskine, Faber-Castell).
- price (number): Price of the product.
- category (string): The type of product, using an
enum
(e.g., Writing, Office Supplies). useenum
, exact value (Writing, Office Supplies, Art Supplies, Educational, Technology) - description (string): A brief description of the product.
- quantity (number): Quantity of the product available.
- inStock (boolean): Indicates if the product is in stock.
- Order Model
- email (string): The email address of the customer.
- product (ObjectId): The stationery product ordered. (
unused ref
) (enter the created productId from your database which product you would love to buy). - quantity (number): The quantity of the ordered product.
- totalPrice (number): The total price (product price * quantity).
message
: A brief error message explaining what went wrong.success
: Set tofalse
for error responses.error
: The error message or error object returned by the application (e.g.,"ValidationError"
,"Resource not found"
).stack
: The stack trace showing where the error occurred in the code.
{
"message": "Validation failed",
"success": false,
"error": {
"name": "ValidationError",
"errors": {
"price": {
"message": "Price must be a positive number",
"name": "ValidatorError",
"properties": {
"message": "Price must be a positive number",
"type": "min",
"min": 0
},
"kind": "min",
"path": "price",
"value": -5
}
}
},
"stack": "Error: Something went wrong\n at app.js:23:13\n at..."
}
- Endpoint:
/api/products
- Method:
POST
- Request Body:
{
"name": "Notebook",
"brand": "Moleskine",
"price": 15,
"category": "Office Supplies",
"description": "A high-quality notebook for professionals.",
"quantity": 200,
"inStock": true
}
- Response: Success message and created product details.
{
"message": "Product created successfully",
"success": true,
"data": {
"_id": "648a45e5f0123c45678d9012",
"name": "Notebook",
"brand": "Moleskine",
"price": 15,
"category": "Office Supplies",
"description": "A high-quality notebook for professionals.",
"quantity": 200,
"inStock": true,
"createdAt": "2024-11-19T10:23:45.123Z",
"updatedAt": "2024-11-19T10:23:45.123Z"
}
}
- Endpoint:
/api/products
- Method:
GET
- Response: A list of all products with details like name, brand, price, category, etc.
- Query: A list of all products from the same category, you’ll take this as
/api/products?searchTerm=category
searchTerm can bename
,brand
,category
{
"message": "Products retrieved successfully",
"status": true,
"data": [
{
"_id": "648a45e5f0123c45678d9012",
"name": "Notebook",
"brand": "Moleskine",
"price": 15,
"category": "Office Supplies",
"description": "A high-quality notebook for professionals.",
"quantity": 200,
"inStock": true,
"createdAt": "2024-11-19T10:23:45.123Z",
"updatedAt": "2024-11-19T10:23:45.123Z"
},
// ... rest data
]
}
- Endpoint:
/api/products/:productId
- Method:
GET
- Response: The details of a specific product by ID.
{
"message": "Product retrieved successfully",
"status": true,
"data": {
"_id": "648a45e5f0123c45678d9012",
"name": "Notebook",
"brand": "Moleskine",
"price": 15,
"category": "Office Supplies",
"description": "A high-quality notebook for professionals.",
"quantity": 200,
"inStock": true,
"createdAt": "2024-11-19T10:23:45.123Z",
"updatedAt": "2024-11-19T10:23:45.123Z"
}
}
- Endpoint:
/api/products/:productId
- Method:
PUT
- Request Body: (Product details to update)
{
"price": 18,
"quantity": 180
}
- Response: Success message and updated product details.
{
"message": "Product updated successfully",
"status": true,
"data": {
"_id": "648a45e5f0123c45678d9012",
"name": "Notebook",
"brand": "Moleskine",
"price": 18, // Price updated
"category": "Office Supplies",
"description": "A high-quality notebook for professionals.",
"quantity": 180, // Quantity updated
"inStock": true,
"createdAt": "2024-11-19T10:23:45.123Z",
"updatedAt": "2024-11-19T11:00:00.000Z" // Updated timestamp
}
}
- Endpoint:
/api/products/:productId
- Method:
DELETE
- Response: Success message confirming the product has been deleted.
{
"message": "Product deleted successfully",
"status": true,
"data": {}
}
- Endpoint:
/api/orders
- Method:
POST
- Inventory Management Logic:
- When an order is placed, reduce the quantity in the product model.
- If the inventory quantity goes to zero, set inStock to
false
. - Handle insufficient stock cases by returning an appropriate error message.
- Request Body:
{
"email": "customer@example.com",
"product": "648a45e5f0123c45678d9012",
"quantity": 2,
"totalPrice": 36
}
- Response: Success message confirming the order.
{
"message": "Order created successfully",
"status": true,
"data": {
"_id": "648b45f5e1234b56789a6789",
"email": "customer@example.com",
"product": "648a45e5f0123c45678d9012",
"quantity": 2,
"totalPrice": 36,
"createdAt": "2024-11-19T12:00:00.000Z",
"updatedAt": "2024-11-19T12:00:00.000Z"
}
}
- Endpoint:
/api/orders/revenue
- Method:
GET
- Aggregation Suggestion:
- Use MongoDB aggregation pipeline to calculate the total revenue from
all orders
. - Calculate the total price by multiplying the price of each product by the quantity ordered.
- Use MongoDB aggregation pipeline to calculate the total revenue from
- Response: The total revenue from all orders.
{
"message": "Revenue calculated successfully",
"status": true,
"data": {
"totalRevenue": 720 // Total revenue calculated from all orders
}
}
- Code Quality:
- Write clean, well-documented code.
- Use meaningful variable and function names.
- API Structure:
- API endpoints should be the same as we have provided. If you don't follow the API structure and response structure your mark will be deducted.
- Follow the API structure exactly as outlined above.
- Ensure request and response formats match the specifications.
- Error Handling:
- Implement proper error handling for scenarios like invalid input, missing data, and insufficient stock.
- Not Found: If a product or order is not found, return a
404
error with an appropriate message. - Validation Errors: Return specific error messages for validation failures (e.g., invalid email, insufficient stock).
- Video Explanation:
- Record a video explaining the key features of your Stationery Shop API and the logic behind its design and test APIs.
- GitHub Repository Link
- Live Deployment Link
- Video Explanation (Public Link)
- Professional README file with features of your application and instructions on setting up the project locally.
- 60 Marks: November 24, 2024 - 11:59 PM
- 50 Marks: November 25, 2024 - 11:59 PM
- 30 Marks: After November 25, 2024