Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/muslehud77/deep-dive-mongoose

https://mongoose-deep-dive.vercel.app/
https://github.com/muslehud77/deep-dive-mongoose

dotenv eslint expressjs husky mongoose nodejs prettier ts-node-dev typescript zod

Last synced: 15 days ago
JSON representation

https://mongoose-deep-dive.vercel.app/

Awesome Lists containing this project

README

        

# Deep Dive To Mongoose

This project is a basic server setup using Node.js, Express, TypeScript, and Mongoose. It provides a starting point for developing a server-side application with a structured and maintainable codebase.

## Table of Contents

- [Technologies Used](#technologies-used)
- [Project Structure](#project-structure)
- [Setup Instructions](#setup-instructions)
- [Scripts](#scripts)
- [Environment Variables](#environment-variables)
- [Code Quality Configuration](#code-quality-configuration)
- [Implementing the Application Code](#implementing-the-application-code)
- [Creating a Product](#creating-a-product)
- [Listing and Searching Products](#listing-and-searching-products)
- [Retrieve a Specific Product by ID](#retrieve-a-specific-product-by-id)
- [Listing and Searching Products](#listing-and-searching-products)
- [Delete a Product](#delete-a-product)
- [Create a New Order](#create-a-new-order)
- [Retrieve All Orders](#retrieve-all-orders)
- [Retrieve Orders by User Email](#retrieve-orders-by-user-email)

## Technologies Used

- **Node.js**: A JavaScript runtime built on Chrome's V8 JavaScript engine.
- **Express**: A fast, unopinionated, minimalist web framework for Node.js.
- **TypeScript**: A strongly typed programming language that builds on JavaScript, giving you better tooling at any scale.
- **Mongoose**: A MongoDB object modeling tool designed to work in an asynchronous environment.
- **Zod**: A TypeScript-first schema declaration and validation library.
- **dotenv**: A module that loads environment variables from a `.env` file into `process.env`.
- **ts-node-dev**: A development tool that compiles TypeScript files on the fly and restarts the server upon file changes.
- **ESLint**: A pluggable and configurable linter tool for identifying and reporting on patterns in JavaScript.
- **Prettier**: An opinionated code formatter.
- **Husky**: A tool for managing Git hooks.

## Project Structure

```

├── .vscode/
│ └── ...
├── dist/
│ ├── app/
│ │ ├── Order/
│ │ │ ├── order.controller.js
│ │ │ ├── order.interface.js
│ │ │ ├── order.model.js
│ │ │ ├── order.route.js
│ │ │ ├── order.service.js
│ │ │ └── order.validation.js
│ │ └── product/
│ │ ├── product.controller.js
│ │ ├── product.interface.js
│ │ ├── product.model.js
│ │ ├── product.route.js
│ │ ├── product.service.js
│ │ └── product.validation.js
│ └── ...
├── src/
│ ├── app/
│ │ ├── configs/
│ │ │ └── index.ts
│ │ ├── Order/
│ │ │ ├── order.controller.ts
│ │ │ ├── order.interface.ts
│ │ │ ├── order.model.ts
│ │ │ ├── order.route.ts
│ │ │ ├── order.service.ts
│ │ │ └── order.validation.ts
│ │ └── product/
│ │ ├── product.controller.ts
│ │ ├── product.interface.ts
│ │ ├── product.model.ts
│ │ ├── product.route.ts
│ │ ├── product.service.ts
│ │ └── product.validation.ts
│ ├── app.ts
│ └── server.ts
├── .env
├── .gitignore
├── .prettierrc.json
├── eslint.config.mjs
├── package.json
├── package-lock.json
├── README.md
├── tsconfig.json
└── vercel.json

```

## Setup Instructions

1. **Clone the Repository**

```sh
git clone https://github.com/Muslehud77/deep-dive-mongoose.git
cd deep-dive-mongoose
```

2. **Install Dependencies**

Make sure you have Node.js and npm installed. Then run:

```sh
npm install
```

3. **Setup Environment Variables**

Create a `.env` file in the root directory and add the following variables:

```plaintext
PORT=your_port_number
DB_URI=your_mongodb_connection_string
```

Replace `your_port_number` with the port number you want the server to run on (e.g., 6000) and `your_mongodb_connection_string` with your actual MongoDB connection string.

4. **Build the Project**

Compile the TypeScript code to JavaScript:

```sh
npm run build
```

5. **Start the Server**

- For development:

```sh
npm run start:dev
```

- For production:

```sh
npm run start:prod
```

## Scripts

- **`build`**: Compiles TypeScript files to JavaScript.
- **`lint`**: Runs ESLint to analyze the code for potential errors and code style issues.
- **`lint:fix`**: Automatically fixes linting issues.
- **`prettier`**: Formats the code using Prettier.
- **`prettier:fix`**: Automatically formats the code using Prettier.
- **`start:dev`**: Starts the server in development mode with auto-reloading.
- **`start:prod`**: Starts the server in production mode.
- **`test`**: Placeholder for running tests.

## Environment Variables

The following environment variables are used in the project:

- **`PORT`**: The port on which the server will run.
- **`DB_URI`**: The MongoDB connection string.

Make sure to replace the placeholder values in the `.env` file with your own configurations.

Certainly! Below is a detailed README section demonstrating how to create a product in your application.

## Code Quality Configuration

### ESLint and Prettier Setup

This project uses ESLint and Prettier to maintain code quality and consistency. Below are the configurations used for both tools.

### ESLint Configuration

The ESLint configuration is set up to enforce a set of rules and integrates with Prettier for code formatting. Here is the `.eslintrc.mjs` configuration file:

```javascript
import globals from 'globals';
import pluginJs from '@eslint/js';
import tseslint from 'typescript-eslint';

export default [
{
ignores: ['**/node_modules/', '.dist/'],
languageOptions: {
globals: {
...globals.browser,
process: 'readonly',
},
},

rules: {
'no-unused-vars': 'error',
'no-unused-expressions': 'error',
'prefer-const': 'error',
'no-console': 'warn',
'no-undef': 'error',
'prettier/prettier': 'error',
},
},

pluginJs.configs.recommended,
...tseslint.configs.recommended,
'prettier',
];
```

### Prettier Configuration

The Prettier configuration is used to format code consistently across the project. Here is the `.prettierrc.json` configuration file:

```json
{
"semi": true,
"singleQuote": true,
"arrowParens": "avoid"
}
```

## Implementing the Application Code

#### app.ts

This file sets up the Express application, configures middleware, and defines routes.

```typescript
import express, { Request, Response } from 'express';
import cors from 'cors';
import { productRoutes } from './models/product/product.route';
import { orderRoutes } from './models/Order/order.route';

const app = express();

// Parsers
app.use(express.json());
app.use(cors());

// Handling routes
app.use('/api/products', productRoutes);
app.use('/api/orders', orderRoutes);

app.get('/', (req: Request, res: Response) => {
res.send('Hello World!');
});

app.use('*', (req: Request, res: Response) => {
res.status(404).json({
success: false,
message: 'Route not found',
});
});

export default app;
```

#### server.ts

This file starts the server and connects to the MongoDB database.

```typescript
import app from './app';
import config from './configs';
import { connect } from 'mongoose';

async function run() {
try {
await connect(config.dbUri);
app.listen(config.port, () => {
console.log(`Server is listening on port ${config.port}`);
});
} catch (err) {
console.error('Error connecting to the database', err);
}
}

run();
```

#### index.ts in Configs

This file loads environment variables and exports the configuration.

```typescript
import dotenv from 'dotenv';

dotenv.config();

const port = process.env.PORT;
const dbUri = process.env.DB_URI as string;

export default { port, dbUri };
```

## Creating a Product

This section demonstrates how to create a product in your application using a POST request.

### Endpoint

**URL:** `/api/products`

**Method:** `POST`

### Request Body

The request body should be a JSON object containing the following fields:

- `name` (string): The name of the product. This field is required and must be unique.
- `description` (string): A description of the product. This field is required.
- `price` (number): The price of the product. This field is required.
- `category` (string): The category of the product. This field is required.
- `tags` (array of strings): A list of tags associated with the product. This field is required.
- `variants` (array of objects): A list of variants for the product. Each variant object should contain:
- `type` (string): The type of the variant (e.g., color). This field is required.
- `value` (string): The value of the variant (e.g., Black). This field is required.
- `inventory` (object): An object representing the inventory of the product. This field should contain:
- `quantity` (number): The quantity of the product in stock. This field is required.
- `inStock` (boolean): Whether the product is in stock. This field is required.
- `isDeleted` (boolean): Whether the product is deleted. This field is optional and defaults to `false`.

### Example Request Body

```json
{
"name": "Wireless Mouse",
"description": "Ergonomic wireless mouse with adjustable DPI settings.",
"price": 29.99,
"category": "Electronics",
"tags": ["computer", "peripherals", "wireless", "ergonomic"],
"variants": [
{ "type": "color", "value": "Black" },
{ "type": "color", "value": "White" }
],
"inventory": {
"quantity": 150,
"inStock": true
}
}
```

### Response

#### Success Response

- **Status Code:** `200 OK`
- **Content:** A JSON object containing the created product.

```json
{
"success": true,
"message": "Product created successfully!",
"data": {
"_id": "60c72b2f9b1d8b3c88b1e7a1",
"name": "Wireless Mouse",
"description": "Ergonomic wireless mouse with adjustable DPI settings.",
"price": 29.99,
"category": "Electronics",
"tags": ["computer", "peripherals", "wireless", "ergonomic"],
"variants": [
{ "type": "color", "value": "Black" },
{ "type": "color", "value": "White" }
],
"inventory": {
"quantity": 150,
"inStock": true
},
"isDeleted": false,
"__v": 0
}
}
```

#### Error Response

- **Status Code:** `500 Internal Server Error`
- **Content:** A JSON object containing the error message.

```json
{
"success": false,
"message": "Request Could Not Complete",
"data": "Error details here"
}
```

### Implementation Details

#### Controller (`product.controller.ts`)

The `addProduct` function in the controller handles the logic for creating a new product:

```typescript
import { Request, Response } from 'express';
import services from './product.service';
import productValidation from './product.validation';

const addProduct = async function (req: Request, res: Response) {
try {
const product = await productValidation.parseAsync(req.body);

const result = await services.addProductIntoDB(product);

res.status(200).json({
success: true,
message: 'Product created successfully!',
data: result,
});
} catch (err) {
res.status(500).json({
success: false,
message: 'Request Could Not Complete',
data: err,
});
}
};

export default {
addProduct,
// other controller methods...
};
```

#### Service (`product.service.ts`)

The `addProductIntoDB` function in the service handles the database interaction:

```typescript
import { TProduct } from './product.interface';
import { productModel } from './product.model';

const addProductIntoDB = async (product: TProduct) => {
const result = await productModel.create(product);
return result;
};

export default {
addProductIntoDB,
// other service methods...
};
```

#### Validation (`product.validation.ts`)

The `productValidation` schema ensures the request body is valid:

```typescript
import { z } from 'zod';

const productValidation = z.object({
name: z.string().min(1, { message: 'Name is required' }),
description: z.string().min(1, { message: 'Description is required' }),
price: z.number().positive({ message: 'Price must be a positive number' }),
category: z.string().min(1, { message: 'Category is required' }),
tags: z.array(z.string()).min(1, { message: 'At least one tag is required' }),
variants: z.array(
z.object({
type: z.string().min(1, { message: 'Variant type is required' }),
value: z.string().min(1, { message: 'Variant value is required' }),
})
).min(1, { message: 'At least one variant is required' }),
inventory: z.object({
quantity: z.number().nonnegative({ message: 'Quantity must be a non-negative number' }),
inStock: z.boolean(),
}),
isDeleted: z.boolean().default(false),
});

export default productValidation;
```

## Listing and Searching Products

This section demonstrates how to retrieve a list of all products and search for products using a query parameter with a GET request.

### Endpoints

#### List All Products

- **URL:** `/api/products`
- **Method:** `GET`

#### Search Products by Query

- **URL:** `/api/products?searchTerm={searchTerm}`
- **Method:** `GET`

### Success Response

#### List All Products

- **Status Code:** `200 OK`
- **Content:** Array of all products.

```json
{
"success": true,
"message": "Products fetched successfully!",
"data": [
// Product objects here...
]
}
```

#### Search Products

- **Status Code:** `200 OK`
- **Content:** Array of products matching the search term.

```json
{
"success": true,
"message": "Products matching search term 'mouse' fetched successfully!",
"data": [
// Matching product objects here...
]
}
```

### Error Response

- **Status Code:** `500 Internal Server Error`
- **Content:** Error message.

```json
{
"success": false,
"message": "Request Could Not Complete",
"data": "Error details here"
}
```

### Implementation Details

#### Controller (`product.controller.ts`)

```typescript
import { Request, Response } from 'express';
import services from './product.service';

const getAllProducts = async (req: Request, res: Response) => {
try {
const searchTerm = req.query.searchTerm as string;
const result = searchTerm
? await services.getSearchedProductFromDB(searchTerm)
: await services.getAllProductsFromDB();

res.status(200).json({
success: true,
message: searchTerm
? `Products matching search term '${searchTerm}' fetched successfully!`
: 'Products fetched successfully!',
data: result,
});
} catch (err) {
res.status(500).json({
success: false,
message: 'Request Could Not Complete',
data: err,
});
}
};

export default { getAllProducts };
```

#### Service (`product.service.ts`)

```typescript
import { productModel } from './product.model';

const getAllProductsFromDB = async () => {
return await productModel.find();
};

const getSearchedProductFromDB = async (searchTerm: string) => {
return await productModel.find({
$or: [
{ name: { $regex: searchTerm, $options: 'i' } },
{ description: { $regex: searchTerm, $options: 'i' } },
{ category: { $regex: searchTerm, $options: 'i' } },
],
});
};

export default { getAllProductsFromDB, getSearchedProductFromDB };
```

#### Route (`product.route.ts`)

```typescript
import express from 'express';
import productController from './product.controller';
const route = express.Router();

route.get('/', productController.getAllProducts);

export const productRoutes = route;
```

### Example Usage

To list all products, send a GET request to:

```
GET /api/products
```

To search for products with a specific term, send a GET request to:

```
GET /api/products?searchTerm=mouse
```

Replace `mouse` with your desired search term.

## Retrieve a Specific Product by ID

This section demonstrates how to retrieve a specific product by its ID using a GET request.

### Endpoint

- **URL:** `/api/products/:productId`
- **Method:** `GET`

### Success Response

- **Status Code:** `200 OK`
- **Content:** The product object.

```json
{
"success": true,
"message": "Product fetched successfully!",
"data": {
"_id": "product_id_here",
"name": "Product Name",
"description": "Product Description",
"price": 29.99,
"category": "Category",
"tags": ["tag1", "tag2"],
"variants": [
{ "type": "color", "value": "Black" }
],
"inventory": {
"quantity": 150,
"inStock": true
},
"isDeleted": false
}
}
```

### Error Response

- **Status Code:** `400 Bad Request`
- **Content:** Error message when product is not found or deleted.

```json
{
"success": false,
"message": "Could not found or deleted",
"data": null
}
```

- **Status Code:** `500 Internal Server Error`
- **Content:** Error message.

```json
{
"success": false,
"message": "Request Could Not Complete",
"data": "Error details here"
}
```

### Implementation Details

#### Controller (`product.controller.ts`)

```typescript
import { Request, Response } from 'express';
import services from './product.service';

const getProductById = async (req: Request, res: Response) => {
try {
const productId = { _id: req.params.productId };
const result = await services.getProductByIdFromDB(productId);

if (!result) {
res.status(400).json({
success: false,
message: 'Could not found or deleted',
data: null,
});
} else {
res.status(200).json({
success: true,
message: 'Product fetched successfully!',
data: result,
});
}
} catch (err) {
res.status(500).json({
success: false,
message: 'Request Could Not Complete',
data: err,
});
}
};

export default { getProductById };
```

#### Service (`product.service.ts`)

```typescript
import { productModel } from './product.model';

const getProductByIdFromDB = async (id: { _id: string }) => {
return await productModel.findOne(id);
};

export default { getProductByIdFromDB };
```

#### Route (`product.route.ts`)

```typescript
import express from 'express';
import productController from './product.controller';
const route = express.Router();

route.get('/:productId', productController.getProductById);

export const productRoutes = route;
```

### Example Usage

To retrieve a specific product by its ID, send a GET request to:

```
GET /api/products/{productId}
```

Replace `{productId}` with the actual ID of the product you want to retrieve.

## Update Product Information

This section demonstrates how to update an existing product's information using a PUT request.

### Endpoint

- **URL:** `/api/products/:productId`
- **Method:** `PUT`

### Request Body

- **Content-Type:** `application/json`
- **Body:**
```json
{
"name": "Updated Product Name",
"description": "Updated Description",
"price": 39.99,
"category": "Updated Category",
"tags": ["updatedTag1", "updatedTag2"],
"variants": [
{ "type": "color", "value": "Red" }
],
"inventory": {
"quantity": 200,
"inStock": true
},
"isDeleted": false
}
```

### Success Response

- **Status Code:** `200 OK`
- **Content:** The updated product object.

```json
{
"success": true,
"message": "Product updated successfully!",
"data": {
"_id": "product_id_here",
"name": "Updated Product Name",
"description": "Updated Description",
"price": 39.99,
"category": "Updated Category",
"tags": ["updatedTag1", "updatedTag2"],
"variants": [
{ "type": "color", "value": "Red" }
],
"inventory": {
"quantity": 200,
"inStock": true
},
"isDeleted": false
}
}
```

### Error Response

- **Status Code:** `500 Internal Server Error`
- **Content:** Error message.

```json
{
"success": false,
"message": "Request Could Not Complete",
"data": "Error details here"
}
```

### Implementation Details

#### Controller (`product.controller.ts`)

```typescript
import { Request, Response } from 'express';
import services from './product.service';
import productValidation from './product.validation';

const updateProduct = async (req: Request, res: Response) => {
try {
const productId = { _id: req.params.productId };
const product = await productValidation.parseAsync(req.body);

await services.updateProductInDB(productId, product);
const updatedProduct = await services.getProductByIdFromDB(productId);

res.status(200).json({
success: true,
message: 'Product updated successfully!',
data: updatedProduct,
});
} catch (err) {
res.status(500).json({
success: false,
message: 'Request Could Not Complete',
data: err,
});
}
};

export default { updateProduct };
```

#### Service (`product.service.ts`)

```typescript
import { productModel } from './product.model';
import { TProduct } from './product.interface';

const updateProductInDB = async (id: { _id: string }, product: TProduct) => {
return await productModel.updateOne(id, product);
};

const getProductByIdFromDB = async (id: { _id: string }) => {
return await productModel.findOne(id);
};

export default { updateProductInDB, getProductByIdFromDB };
```

#### Route (`product.route.ts`)

```typescript
import express from 'express';
import productController from './product.controller';
const route = express.Router();

route.put('/:productId', productController.updateProduct);

export const productRoutes = route;
```

### Example Usage

To update a product's information, send a PUT request to:

```
PUT /api/products/{productId}
```

Replace `{productId}` with the actual ID of the product you want to update. The request body should contain the updated product details in JSON format.

## Delete a Product

This section demonstrates how to delete a product by marking it as deleted in the database using a DELETE request. The product is not actually removed from the database but is flagged as deleted.

### Endpoint

- **URL:** `/api/products/:productId`
- **Method:** `DELETE`

### Success Response

- **Status Code:** `200 OK`
- **Content:** Confirmation message.

```json
{
"success": true,
"message": "Product deleted successfully!",
"data": null
}
```

### Error Response

- **Status Code:** `500 Internal Server Error`
- **Content:** Error message.

```json
{
"success": false,
"message": "Request Could Not Complete",
"data": "Error details here"
}
```

### Implementation Details

#### Controller (`product.controller.ts`)

```typescript
import { Request, Response } from 'express';
import services from './product.service';

const deleteProduct = async (req: Request, res: Response) => {
try {
const productId = { _id: req.params.productId };

await services.deleteProductFromDB(productId);

res.status(200).json({
success: true,
message: 'Product deleted successfully!',
data: null,
});
} catch (err) {
res.status(500).json({
success: false,
message: 'Request Could Not Complete',
data: err,
});
}
};

export default { deleteProduct };
```

#### Service (`product.service.ts`)

```typescript
import { productModel } from './product.model';
import { TProduct } from './product.interface';

const deleteProductFromDB = async (id: { _id: string }) => {
return await productModel.findByIdAndUpdate(id, { isDeleted: true });
};

export default { deleteProductFromDB };
```

#### Model (`product.model.ts`)

Ensure the schema includes the `isDeleted` field and middleware to exclude deleted records from queries.

```typescript
import { Schema, model } from 'mongoose';
import { TProduct } from './product.interface';

const isRequiredString = { type: String, required: true };
const variantSchema = new Schema({
type: isRequiredString,
value: isRequiredString,
});

const productSchema = new Schema({
name: { ...isRequiredString, unique: true },
description: isRequiredString,
price: { type: Number, required: true },
category: isRequiredString,
tags: { type: [String], required: true },
variants: {
type: [variantSchema],
required: true,
},
inventory: { quantity: Number, inStock: Boolean },
isDeleted: { type: Boolean, default: false },
});

productSchema.pre('find', function (next) {
this.find({ isDeleted: { $ne: true } });
next();
});

productSchema.pre('findOne', function (next) {
this.find({ isDeleted: { $ne: true } });
next();
});

export const productModel = model('products', productSchema);
```

#### Route (`product.route.ts`)

```typescript
import express from 'express';
import productController from './product.controller';
const route = express.Router();

route.delete('/:productId', productController.deleteProduct);

export const productRoutes = route;
```

### Example Usage

To delete a product, send a DELETE request to:

```
DELETE /api/products/{productId}
```

Replace `{productId}` with the actual ID of the product you want to delete. This will mark the product as deleted in the database without physically removing the record.

Certainly! Here's a structured guide for your README.md:

---

## Create a New Order
To create a new order, the system checks if the product is in stock and updates the stock accordingly.

```typescript
// order.controller.ts
import { Request, Response } from 'express';
import orderValidation from './order.validation';
import services from './order.service';

const addOrders = async function (req: Request, res: Response) {
try {
const order = await orderValidation.parseAsync(req.body);
const result = await services.addOrdersToDB(order);
res.status(200).json({
success: true,
message: 'Order created successfully!',
data: result,
});
} catch (err: any) {
res.status(500).json({
success: false,
message: err.message || 'Request could not complete',
});
}
};
export default addOrders;
```

## Retrieve All Orders
This section retrieves all orders from the database.

```typescript
// order.controller.ts
import { Request, Response } from 'express';
import services from './order.service';

const getAllOrders = async function (req: Request, res: Response) {
try {
const result = await services.getAllOrdersFromDB();
if (!result.length) {
throw new Error('Order not found');
}
res.status(200).json({
success: true,
message: 'Orders fetched successfully!',
data: result,
});
} catch (err: any) {
res.status(500).json({
success: false,
message: err.message || 'Request could not complete',
});
}
};
export default getAllOrders;
```

## Retrieve Orders by User Email
This section retrieves orders by user email.

```typescript
// order.controller.ts
import { Request, Response } from 'express';
import services from './order.service';

const getOrderById = async function (req: Request, res: Response) {
try {
const email = req.query.email as string;
const result = await services.getOrderByIdFromDB(email);
if (!result.length) {
throw new Error('Order not found for this email');
}
res.status(200).json({
success: true,
message: 'Orders fetched successfully for user email!',
data: result,
});
} catch (err: any) {
res.status(400).json({
success: false,
message: err.message || 'Request could not complete',
});
}
};
export default getOrderById;
```
## Api Endpoints

1. **Create a New Product**
- **URL:** `/api/products`
- **Method:** `POST`
- **Request Body:** JSON object containing product details
- **Success Response:** `200 OK` with the created product object
- **Error Response:** `500 Internal Server Error` with error message

2. **List All Products**
- **URL:** `/api/products`
- **Method:** `GET`
- **Success Response:** `200 OK` with an array of all products
- **Error Response:** `500 Internal Server Error` with error message

3. **Search Products**
- **URL:** `/api/products?searchTerm={searchTerm}`
- **Method:** `GET`
- **Success Response:** `200 OK` with an array of products matching the search term
- **Error Response:** `500 Internal Server Error` with error message

4. **Retrieve a Specific Product by ID**
- **URL:** `/api/products/:productId`
- **Method:** `GET`
- **Success Response:** `200 OK` with the product object
- **Error Response:** `400 Bad Request` or `500 Internal Server Error` with error message

5. **Update Product Information**
- **URL:** `/api/products/:productId`
- **Method:** `PUT`
- **Request Body:** JSON object containing updated product details
- **Success Response:** `200 OK` with the updated product object
- **Error Response:** `500 Internal Server Error` with error message

6. **Delete a Product**
- **URL:** `/api/products/:productId`
- **Method:** `DELETE`
- **Success Response:** `200 OK` with confirmation message
- **Error Response:** `500 Internal Server Error` with error message

7. **Create a New Order**
- **URL:** `/api/orders`
- **Method:** `POST`
- **Request Body:** JSON object containing order details
- **Success Response:** `200 OK` with the created order object
- **Error Response:** `500 Internal Server Error` with error message

8. **Retrieve All Orders**
- **URL:** `/api/orders`
- **Method:** `GET`
- **Success Response:** `200 OK` with an array of all orders
- **Error Response:** `500 Internal Server Error` with error message

9. **Retrieve Orders by User Email**
- **URL:** `/api/orders?email={userEmail}`
- **Method:** `GET`
- **Success Response:** `200 OK` with an array of orders matching the user email
- **Error Response:** `400 Bad Request` or `500 Internal Server Error` with error message

These endpoints cover all the functionalities of the project, allowing users to manage products and orders efficiently.

---

## Conclusion

The project offers a comprehensive setup for developing a server-side application with structured code and well-defined API endpoints. By leveraging technologies like Node.js, Express, TypeScript, and Mongoose, it ensures robustness, scalability, and maintainability. Here are some key highlights:

- **Technologies Used:** Leveraging Node.js for its efficient runtime, Express for web framework capabilities, TypeScript for strong typing and scalability, and Mongoose for MongoDB object modeling.
- **Project Structure:** A well-organized project structure separates concerns, making it easier to maintain and scale the application.
- **Setup Instructions:** Clear setup instructions guide developers through cloning the repository, installing dependencies, setting up environment variables, building the project, and starting the server.
- **Code Quality Configuration:** ESLint and Prettier configurations ensure consistent code style and identify potential errors, enhancing code quality.
- **API Endpoints:** The project provides a range of API endpoints for CRUD operations on products and orders, facilitating efficient management of data.
- **Error Handling:** Comprehensive error handling ensures that users receive informative responses in case of errors, enhancing the overall user experience.
- **Conclusion:** With a solid foundation and clear documentation, the project serves as an excellent starting point for developers looking to build server-side applications with Node.js and TypeScript.

Overall, the project offers a structured, maintainable, and scalable solution for developing server-side applications, empowering developers to focus on building robust features and delivering value to users.

---