js-frontend-repository v2.0.0
js-frontend-repository
The Repository pattern is a well-documented way of working with a data source. In this library I used this pattern for manage API calls from javascript base frontend applications
Benefits
- It centralizes API calls
- It gives a substitution point for the unit tests.
- Provides a flexible architecture
- It reduces redundancy of code
- It force programmer to work using the same pattern
- If you use this pattern then it is easy to maintain the centralized data access logic
- centralized API call
- centralized error handeling
- Can use diffrents API Gatways in same project with no change in Service level
- Adding SOLID principals to your project
Diagram
npm i js-frontend-repository
How to use it
export class Todo {
title? : string;
isDone? : boolean;
}
IRepository
For each model you should add one generic IRepository model of that entity, this Interface will be use on Service concretes of the entity and with help of this IRepository you can avoid changes in Service layer if you want to change Repository layer
import { IRepository } from 'js-frontend-repository/interfaces/IRepository';
import { Todo } from '@/src/models/Todo';
export interface ITodoRepository extends IRepository<Todo> {
}
Repository concrete
This file will contain all logic related to API calls for example if you are using GraphQL API, all queries should be developed in this class. Or if you are using Firestore API all related queries should be developed in this class.
Usually instead of inherit from IRepository in this class it is better practice to inherit from one base repository. current project have already three base repository developed, you can use this base repository in scenario you want to fetch data against Firestore , Azure Cosmos or Azure Cosmos Graph Gremlin if you want to use another service or your own custom API call you should develop the base Repository related to that first
For more information about implemented Repository base class please refer to each Repository base page :
- Firestore Repository base
- Azure cosmos and Azure cosmos Gremlin Repository base
import { FirestoreRepository } from 'react-native-firesotre-repository/FirestoreRepository';
import { Todo } from '@/src/models/Todo';
import { FirestoreEntityMetaData } from 'react-native-firesotre-repository/FirestoreEntityMetaData';
import { FirestoreEntityMetaDataContext } from 'react-native-firesotre-repository/FirestoreEntityMetaDataContext';
export class TodoRepository extends FirestoreRepository<Todo> {
constructor() {
super(Todo, new FirestoreEntityMetaData(new FirestoreEntityMetaDataContext()));
}
}
you can find example project : Here
Service
For each entity you should add one service class too. Service class is responsible for logic across one entity, for example if you want to fetch Todo with Id = 2 and then change the isDone to true and update the result you should develop the logic of that in this class. Service is for logic across one entity, and Repository class is for query of those logic.
import { ITodoRepository } from '@/src/repositories/interfaces/ITodoRepository';
import { Todo } from '@/src/models/Todo';
import { BaseService } from 'js-frontend-repository/BaseSerivce';
export class TodoService extends BaseService<Todo, ITodoRepository> {
async updateTodo(id: string) {
const todo = await (await this.repository.getById(id)).entity;
todo.isDone = true;
await this.repository.update(todo)
}
}
As you can see if you change the repository behind the Todo model this file will not change
UnitOfWork.ts
This singleton file holds all instances of services, and also in this file you can assign a concert repository for each service. With help of this file you can change the repository of services easily and this change will not need to change any logic on service and other layers of application.
// new service import placeholder
// new service import placeholder
import { TodoRepository } from '@/src/repositories/concretes/TodoRepository';
import { TodoService } from '@/src/services/TodoService';
export class UnitOfWork {
// new service placeholder
todoService: TodoService
private static innerInstance: UnitOfWork;
public static instance(): UnitOfWork {
if (!UnitOfWork.innerInstance) {
UnitOfWork.innerInstance = new UnitOfWork();
}
return UnitOfWork.innerInstance;
}
private constructor() {
// new service init placeholder
this.todoService = new TodoService(TodoRepository);
}
}
Code Generator
As you can see for adding new Entity in the repository pattern you have to add 3 files and modify one, to develop faster and avoid duplicate work you can use the “Repository Code Generator” package.
Install
npm i -g rpcodegen
For more information please see Repository Code Generator page : Repository Code Generator
Redux
If you want to use Redux in your application and you want to manage the API call with repository pattern and automatically dispatch them on State and also if your looking for better solution to design your Actions and Routers in Redux you can use this package
npm i react-redux-repository
For more information ablut Redux Repository please visit: React Redux Repository