Required:
Recommended:
- MongoDB Compass - To view and work directly with the database
- Postman
To get started use a terminal, cd into the backend directory and install packages.
cd backend
npm install
Next, you can start the server in one of the two ways:
- Running live development server:
npm run dev
(Recommended) - Start the server without live refresh:
npm start
The location where the server is hosted will be displayed on the terminal. By default, it will be live at http://localhost:3000.
We are using a MongoDB database named "LAK" hosted on MongoDB Atlas. Please ping Assaf to get yourself added to the database so that you can start working on the backend.
Our backend is structured into two layers -
- Routes layer - Recieves requests, calls the appropriate service(s), and returns a response (interacts with the client)
- Services layer - Performs the logic for all requests (interacts with the database)
Here is the backend repo structure
├── README.md
├── models/* (DB models to interact with the database)
├── services/* (Methods that utilize models to perform database operations. Logic of the backend)
├── routes/* (Exposes the various services to the clients via API endpoints)
├── tests/* (Please add tests for your routes here)
└── index.js (Main script to run the server and use routes)
- To maintain consistency, we create responses matching the following format:
{
"message": "Resource with $id was created",
"id": "<some-id>"
}
Error handling is especially important in backend programming because you cannot let the server exit because of an unhandled exception. Therefore, our error handling uses ES6 classes to classify between different types of errors (mainly client facing and internal errors)
Errors
- Client facing errors
- Validation Errors (errors detected while validating user requests)
- Service Errors (errors detected while processing requests, eg. perimssion denied)
- Internal Errors (not relevant to users)
Please look at errors.js to see how ES6 classes are used to classify all these errors.
For any errors, the following are the attributes required:
- Error Code: Unique error identification for a particular type (For internal identification only )
- Status Code: Status sent in the response to the client
- message: Message to be sent to the client if this error occurs
Inspired from Golang, our error class is designed to allow addition of context which could be important especially to deal with internal errors. You can use addContext(error | string)
to add context to an error which will be displayed internally for debugging.
For example:
throw InternalError.DOCUMENT_UPLOAD_ERROR.addContext(e.stack)
This throws an internal error of type DOCUMENT_UPLOAD_ERROR
with the e
stack added for context.
Error display Messages for all client facing errors should be sent to user as response in the format:
{
"message": "<YOUR MESSAGE GOES HERE>",
"error": true
}
Using the examples in errors.js, you can see how to create new errors that can be used in your code. An error should be created appropriately, extending the base error class. Such errors can be thrown anywhere in the backend synchronously as the handler middleware at the end will catch all errors and display it to the client (as a response) or to the server's logs (for internal errors).
Don't hesitate to ask if you have any questions!
We don't have proper logging setup yet, but the current practice is to log each service initation using console.debug
and log each route initiation using console.info
. All the logs will be live on the shell running the server.
In addition, the internal errors mentioned above will also be logged here with their context (if any) so be on the lookout for that in case an internal error occurs.