Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/bespoyasov/frontend-clean-architecture
React + TypeScript app built using the clean architecture principles in a more functional way.
https://github.com/bespoyasov/frontend-clean-architecture
adapters application clean-architecture domain functional layered-architecture onion-architecture
Last synced: about 20 hours ago
JSON representation
React + TypeScript app built using the clean architecture principles in a more functional way.
- Host: GitHub
- URL: https://github.com/bespoyasov/frontend-clean-architecture
- Owner: bespoyasov
- Created: 2021-06-07T10:34:43.000Z (over 3 years ago)
- Default Branch: master
- Last Pushed: 2023-12-05T14:12:35.000Z (about 1 year ago)
- Last Synced: 2025-01-18T02:01:55.848Z (8 days ago)
- Topics: adapters, application, clean-architecture, domain, functional, layered-architecture, onion-architecture
- Language: TypeScript
- Homepage: https://bespoyasov.me/blog/clean-architecture-on-frontend/
- Size: 575 KB
- Stars: 2,364
- Watchers: 18
- Forks: 261
- Open Issues: 2
-
Metadata Files:
- Readme: README.md
- Funding: .github/FUNDING.yml
Awesome Lists containing this project
- awesome-codebases - Frontend Clean Architecture - React + TypeScript app built using the clean architecture principles in a more functional way. (**Awesome Codebases** [![Awesome](https://cdn.rawgit.com/sindresorhus/awesome/d7305f38d29fed78fa85652e3a63e154dd8e8829/media/badge.svg)](https://github.com/sindresorhus/awesome))
- jimsghstars - bespoyasov/frontend-clean-architecture - React + TypeScript app built using the clean architecture principles in a more functional way. (TypeScript)
README
> Other languages: [Russian](https://github.com/bespoyasov/frontend-clean-architecture/blob/master/docs/ru.md).
# Frontend Clean Architecture
A React + TypeScript example app built using the clean architecture in a functional(-ish) way.
- [Working app](https://bespoyasov.ru/showcase/frontend-clean-architecture/en/)
- [Huge post about it](https://dev.to/bespoyasov/clean-architecture-on-frontend-4311)## Things to Consider
There are a few compromises and simplifications in the code that are worth to be mentioned.
### Shared Kernel
Shared Kernel is the code and data on which any modules can depend, but _only if this dependency would not increase coupling_. More details about the limitations and application are well described in the article ["DDD, Hexagonal, Onion, Clean, CQRS, ... How I put it all together"](https://herbertograca.com/2017/11/16/explicit-architecture-01-ddd-hexagonal-onion-clean-cqrs-how-i-put-it-all-together/).
In this application, the shared kernel includes global type annotations that can be accessed anywhere in the app and by any module. Such types are collected in [`shared-kernel.d.ts`](https://github.com/bespoyasov/frontend-clean-architecture/blob/master/src/shared-kernel.d.ts).
### Dependency in the Domain
The [`createOrder`](https://github.com/bespoyasov/frontend-clean-architecture/blob/master/src/domain/order.ts#L15) function uses the library-like function `currentDatetime` to specify the order creation date. This is not quite correct, because the domain should not depend on anything.
Ideally, the implementation of the `Order` type should accept all the necessary data, including the date, from outside. The creation of this entity would be in the application layer in `orderProducts`:
```ts
async function orderProducts(user: User, { products }: Cart) {
const datetime = currentDatetime();
const order = new Order(user, products, datetime);// ...
}
```### Use Case Testability
The order creation function [`orderProduct`](https://github.com/bespoyasov/frontend-clean-architecture/blob/master/src/application/orderProducts.ts#L24) itself is framework-dependent right now and cannot be used or tested in isolation outside of React. The hook wrapper though is only used to provide the use case to components and to inject services into the use case itself.
In a canonical implementation, the function of the use case would be extracted outside the hook, and the services would be passed to the use case via a last argument or a DI:
```ts
type Dependencies = {
notifier?: NotificationService;
payment?: PaymentService;
orderStorage?: OrderStorageService;
};async function orderProducts(
user: User,
cart: Cart,
dependencies: Dependencies = defaultDependencies
) {
const { notifier, payment, orderStorage } = dependencies;// ...
}
```Hook would then become an adapter:
```ts
function useOrderProducts() {
const notifier = useNotifier();
const payment = usePayment();
const orderStorage = useOrdersStorage();return (user: User, cart: Cart) =>
orderProducts(user, cart, {
notifier,
payment,
orderStorage,
});
}
```In the sources, I thought it was unnecessary, as it would distract from the essence.
### Crooked DI
In the [application layer](https://github.com/bespoyasov/frontend-clean-architecture/blob/master/src/application/orderProducts.ts) we inject services by hand:
```ts
export function useAuthenticate() {
const storage: UserStorageService = useUserStorage();
const auth: AuthenticationService = useAuth();// ...
}
```In a good way, this should be automated and done through the dependency injection. But in the case of React and hooks, we can use them as a “container” that returns an implementation of the specified interface.
In this particular application, it didn't make much sense to set up the DI because it would distract from the main topic.