Structuring Express Applications

In module 5, we will dive into the best practices and techniques for
structuring Express.js applications. Proper organization and project
structure are essential for building scalable, maintainable, and
readable applications.

5.1 Organizing Code and Project Structure

Why Structure Matters
Proper organization and project structure are critical for the longterm maintainability and scalability of your Express.js applications. A
well-structured application is easier to understand, modify, and
extend, making it more manageable as your project grows.
Common Project Structure
While there isn't a one-size-fits-all structure for Express.js
applications, many developers follow common patterns and best
practices. Here's a typical project structure for an Express app:
my-express-app/
├── node_modules/
├── public/
│ ├── css/
│ ├── js/
│ └── images/
├── routes/
│ ├── index.js
│ ├── users.js
│ └── ...
├── controllers/
│ ├── indexController.js
│ ├── usersController.js
│ └── ...
├── views/
│ ├── index.ejs
│ ├── user.ejs
│ └── ...
├── app.js
├── package.json
└── ...
In this structure:
 “node_modules”: Contains project dependencies.
 “public”: Holds static assets like CSS, JavaScript, and images.
 “routes”: Defines route handling logic and route-specific
middleware.
 “controllers”: Contains controller functions responsible for
handling route logic.
 “views”: Stores template files used for rendering HTML pages.
 “app.js”: The main application file where Express is initialized
and configured.
 “package.json”: Contains project metadata and dependencies.

5.2 Separating Routes and Controllers


Separation of Concerns

One of the fundamental principles of software design is the
separation of concerns. In the context of an Express.js application,
this means separating the routing logic (how requests are handled)
from the business logic (what happens when a request is received).
Routing in Express
Routing defines how an application responds to client requests. In
Express, you can define routes in the `routes` directory or directly in
the main “app.js” file. However, it's recommended to organize routes
into separate files.
Example of Routing
- javascript
// routes/index.js
const express = require('express');
const router = express.Router();
router.get('/', (req, res) => {
res.render('index');
});
module.exports = router;
In this example, we define a route for the root URL ('/') in the
“index.js” file within the “routes” directory. This route responds by
rendering an 'index' view.

Controllers in Express

Controllers are responsible for handling the business logic associated
with specific routes. They should contain functions that perform
actions related to the route, such as processing data, interacting with
databases, and sending responses.
Example of a Controller
- javascript
// controllers/indexController.js
const indexController = {};
indexController.renderIndex = (req, res) => {
res.render('index');
};
module.exports = indexController;
In this example, we create an “indexController” object with a
function “renderIndex”. This function renders the 'index' view.
Connecting Routes and Controllers
To connect routes with controllers, you can require the controller
module in your route files and invoke the relevant controller
functions when defining routes.
Connecting Route and Controller
- javascript
// routes/index.js
const express = require('express');
const router = express.Router();
const indexController = require('../controllers/indexController');
router.get('/', indexController.renderIndex);
module.exports = router;
In this example, we import the “indexController” and use its
“renderIndex” function as the route handler for the root URL ('/').

5.3 Best Practices for Structuring Express Apps

While the project structure and organization may vary depending on
your specific requirements, adhering to best practices can help
maintain a clean and efficient Express.js application.

1. Use Express Generator

If you're starting a new Express project, consider using the [Express
Generator](https://expressjs.com/en/starter/generator.html). It
provides a basic project structure with sensible defaults, including
routes, controllers, and views.

2. Separate Concerns

Adhere to the separation of concerns principle. Keep your routing
and controller logic separate to enhance code readability and
maintainability.

3. Modularize Your Code

Split your application into modular components, such as routes,
controllers, and middleware. Organize them in separate files and
directories to make the codebase more manageable.

4. Choose a Consistent Naming Convention

Follow a consistent naming convention for your files, routes, and
controllers. This makes it easier to locate and identify components
within your project.

5. Use Middleware Wisely

Leverage middleware for common tasks like authentication, logging,
and error handling. Keep middleware functions organized and avoid
duplicating code.

6. Error Handling

Implement centralized error handling. You can use Express's built-in
error-handling middleware or create custom error handling to
centralize error management and provide consistent error
responses.
- javascript
// Error handling middleware
app.use((err, req, res, next) => {
// Handle errors here
res.status(err.status || 500).send('Something went wrong');
});

7. Maintain a Clean “app.js”

Keep your “app.js” or main application file clean and focused on
configuration and setup. Place your routes, controllers, and other
components in separate files and require them in your main file.

8. Versioning Your API

If you're building a RESTful API, consider versioning your endpoints
from the beginning. This allows you to make changes and updates to
your API without breaking existing clients.

9. Documentation

Document your code, especially if you're working on a team or opensource project. Use comments and README files to explain the
purpose and usage of different components.

10. Testing

Implement testing for your Express application. Tools like Mocha,
Chai, and Supertest can help you write and run tests to ensure your
application functions correctly.

11. Use an ORM/ODM

If your application interacts with a database, consider using an
Object-Relational Mapping (ORM) or Object-Document Mapping
(ODM) library like Sequelize, Mongoose, or TypeORM to simplify
database operations and structure.

12. Keep Security in Mind

Prioritize security by validating and sanitizing user inputs, using
HTTPS, implementing proper authentication, and following security
best practices