Few months ago I've created a repository to gather some architectural patterns and best practices I've been using for the last few years, both at work and for personal projects. At first It was a project for myself, but it grew and have reached a point that I can share it and hopefully more people can learn some useful practices from it.
Main emphasis of this project is to provide recommendations on how to design software applications. In this repository are presented some of the techniques, tools, best practices, architectural patterns and guidelines gathered from different sources.
Code examples are written using NodeJS, TypeScript, NestJS framework and Slonik for the database access.
Though patterns and principles presented in this repository are framework/language agnostic, so above technologies can be easily replaced with any alternative. No matter what language or framework is used, any application can benefit from principles described in this repository.
This project's architecture is mainly based on:
- Domain-Driven Design (DDD)
- Hexagonal (Ports and Adapters) Architecture
- Secure by Design
- Clean Architecture
- Onion Architecture
- SOLID Principles
- Software Design Patterns
And many other sources and best practices (links to additonal resources in every section).
Check out my GitHub repository to see full tutorial with code examples:
Also, check out my other repositories:
- Backend best practices - Best practices, tools and guidelines for backend development.
- System Design Patterns - list of topics and resources related to distributed systems, system design, microservices, scalability and performance, etc.
Top comments (8)
hello my freind , thank you for this great project . but when i was reading the source code of UserEntity in create methode , you used
const user = new UserEntity({ id , props })
but you haven't used the constructor function in this UserEntity class , so how it can get an object of id and props in arguments ??? i think you forget to add the constructor !
Thank you for sharing, it's very interesting. I would like to ask you a question. I have seen that events are emitted before you know if they have been successful. For example, in DeleteUserService, "user.delete()" shouldn't be after "this.userRepo.delete(user)":
user.delete();
const result = await this.userRepo.delete(user);
return Ok(result);
Sorry for the late response, dev.to does not notify by email so I didn't see your comment.
Events are first only stored in memory to be emitted later, and are actually emitted after the entity is saved to the database (check out repository base class
entity.publishEvents
call).If you have any other questions please create a discussion in the repository, I will get a notification and answer faster :)
Awesome
Thank you for your work and for sharing this!
Thank you for sharing, really great job.
Thank you for the great work
Thank you for your work. I'm using it in our projects.