Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/jbuget/nodejs-clean-architecture-app
A simple API built with Hapi.js that follows DDD + Clean Architecture principles
https://github.com/jbuget/nodejs-clean-architecture-app
Last synced: about 19 hours ago
JSON representation
A simple API built with Hapi.js that follows DDD + Clean Architecture principles
- Host: GitHub
- URL: https://github.com/jbuget/nodejs-clean-architecture-app
- Owner: jbuget
- License: mit
- Created: 2018-01-05T10:38:19.000Z (about 7 years ago)
- Default Branch: master
- Last Pushed: 2023-01-06T01:32:37.000Z (about 2 years ago)
- Last Synced: 2025-01-05T06:10:04.155Z (8 days ago)
- Language: JavaScript
- Homepage: https://hapijs-v17-app.herokuapp.com/users
- Size: 1.34 MB
- Stars: 1,512
- Watchers: 50
- Forks: 264
- Open Issues: 21
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
- Awesome-List - NodeJS Clean Architecture App - JS - A simple API built with Hapi.js that follows DDD + Clean Architecture principles. (💖 Admired Examples)
- awesome-list - NodeJS Clean Architecture App - JS - A simple API built with Hapi.js that follows DDD + Clean Architecture principles. (💖 Admired Examples)
- awesome-list - NodeJS Clean Architecture App - JS - A simple API built with Hapi.js that follows DDD + Clean Architecture principles. (💖 Admired Examples)
README
# A basic Hapi.js API following Clean Architecture principles
## Getting started (< 2mn)
```
git clone [email protected]:jbuget/nodejs-clean-architecture-app.git
cd nodejs-clean-architecture-app
npm install
npm test
npm start
```In a browser, open [http://localhost:3000/hello](http://localhost:3000/hello).
## Domain Driven Architectures
Software design is a very hard thing. From years, a trend has appeared to put the business logic, a.k.a. the (Business) Domain, and with it the User, in the heart of the overall system. Based on this concept, different architectural patterns was imaginated.
One of the first and main ones was introduced by E. Evans in its [Domain Driven Design approach](http://dddsample.sourceforge.net/architecture.html).
![DDD Architecture](/doc/DDD_architecture.jpg)
Based on it or in the same time, other applicative architectures appeared like [Onion Architecture](https://jeffreypalermo.com/2008/07/the-onion-architecture-part-1/) (by. J. Palermo), [Hexagonal Architecture](https://alistair.cockburn.us/hexagonal-architecture/) (by A. Cockburn) or [Clean Architecture](https://8thlight.com/blog/uncle-bob/2012/08/13/the-clean-architecture.html) (by. R. Martin).
This repository is an exploration of this type of architecture, mainly based on DDD and Clean Architecture, on a concrete and modern JavaScript application.
## DDD and Clean ArchitectureThe application follows the Uncle Bob "[Clean Architecture](https://8thlight.com/blog/uncle-bob/2012/08/13/the-clean-architecture.html)" principles and project structure :
### Clean Architecture layers
![Schema of flow of Clean Architecture](/doc/Uncle_Bob_Clean_Architecture.jpg)
### Project anatomy
```
app
└ lib → Application sources
└ application → Application services layer
└ security → Security tools interfaces (ex: AccessTokenManager.js, to generate and decode OAuth access token)
└ use_cases → Application business rules
└ domain → Enterprise core business layer such as domain model objects (Aggregates, Entities, Value Objects) and repository interfaces
└ infrastructure → Frameworks, drivers and tools such as Database, the Web Framework, mailing/logging/glue code etc.
└ config → Application configuration files, modules and services
└ service-locator.js → Module that manage service implementations by environment
└ orm → Database ORMs middleware (Sequelize for SQLite3 or PostgreSQL, Mongoose for MongoDB)
└ mongoose → Mongoose client and schemas
└ sequelize → Sequelize client and models
└ repositories → Implementation of domain repository interfaces
└ security → Security tools implementations (ex: JwtAccessTokenManager)
└ webserver → Hapi.js Web server configuration (server, routes, plugins, etc.)
└ oauth → Hapi.js authentication strategies and schemas
└ server.js → Hapi.js server definition
└ interfaces → Adapters and formatters for use cases and entities to external agency such as Database or the Web
└ controllers → Hapi.js route handlers
└ routes → Hapi.js route definitions
└ serializers → Converter objects that transform outside objects (ex: HTTP request payload) to inside objects (ex: Use Case request object)
└ node_modules (generated) → NPM dependencies
└ test → Source folder for unit or functional tests
└ index.js → Main application entry point
```### Flow of Control
![Schema of flow of Control](/doc/Hapijs_Clean_Architecture.svg)
### The Dependency Rule
> The overriding rule that makes this architecture work is The Dependency Rule. This rule says that source code dependencies can only point inwards. Nothing in an inner circle can know anything at all about something in an outer circle. In particular, the name of something declared in an outer circle must not be mentioned by the code in the an inner circle. That includes, functions, classes. variables, or any other named software entity.
src. https://blog.cleancoder.com/uncle-bob/2012/08/13/the-clean-architecture.html#the-dependency-rule### Server, Routes and Plugins
Server, routes and plugins can be considered as "plumbery-code" that exposes the API to the external world, via an instance of Hapi.js server.
The role of the server is to intercept the HTTP request and match the corresponding route.
Routes are configuration objects whose responsibilities are to check the request format and params, and then to call the good controller (with the received request). They are registered as Plugins.
Plugins are configuration object that package an assembly of features (ex: authentication & security concerns, routes, pre-handlers, etc.) and are registered at the server startup.
### Controllers (a.k.a Route Handlers)
Controllers are the entry points to the application context.
They have 3 main responsibilities :
1. Extract the parameters (query or body) from the request
2. Call the good Use Case (application layer)
3. Return an HTTP response (with status code and serialized data)### Use Cases
A use case is a business logic unit.
It is a class that must have an `execute` method which will be called by controllers.
It may have a constructor to define its dependencies (concrete implementations - a.k.a. _adapters_ - of the _port_ objects) or its execution context.
**Be careful! A use case must have only one precise business responsibility!**
A use case can call objects in the same layer (such as data repositories) or in the domain layer.