diff --git a/src/__tests__/unit/features.controller.unit.ts b/src/__tests__/unit/features.controller.unit.ts index d0e6cb92..79b6132a 100644 --- a/src/__tests__/unit/features.controller.unit.ts +++ b/src/__tests__/unit/features.controller.unit.ts @@ -3,15 +3,17 @@ import { } from '@loopback/testlab'; import { FeaturesController } from '../../controllers/features.controller'; import { FeaturesDataService } from '../../services'; +import { TermsDataService } from '../../services/terms-data.service'; describe('FeaturesController (unit)', () => { const mockedService = {}; + const mockedTermsService = {}; let context = stubExpressContext(); describe('get()',() => { it('retrieves the features flags Information', async() => { - const controller = new FeaturesController(context.response, mockedService); + const controller = new FeaturesController(context.response, mockedService, mockedTermsService); await controller.get(); let result = await context.result; expect(result.payload).not.null(); diff --git a/src/controllers/features.controller.ts b/src/controllers/features.controller.ts index 705e5ef7..831b0ab5 100644 --- a/src/controllers/features.controller.ts +++ b/src/controllers/features.controller.ts @@ -4,18 +4,23 @@ import { RestBindings, get, getModelSchemaRef, Response, } from '@loopback/rest' import { ServicesBindings } from '../dependency-injection-bindings'; import { FeaturesDataService } from '../services/features-data.service'; import { FeaturesDbDataModel } from '../models/features-data.model'; +import { TermsDataService } from '../services/terms-data.service'; export class FeaturesController { logger: Logger; private featuresDatService: FeaturesDataService; + private termsDatService: TermsDataService; HTTP_SUCCESS_OK = 200; HTTP_ERROR = 500; constructor( @inject(RestBindings.Http.RESPONSE) private response: Response, @inject(ServicesBindings.FEATURES_SERVICE) featuresDatService: FeaturesDataService, + @inject(ServicesBindings.TERMS_SERVICE) + termsDatService: TermsDataService, ) { this.featuresDatService = featuresDatService; + this.termsDatService = termsDatService; this.logger = getLogger('features-controller'); } @@ -51,17 +56,21 @@ export class FeaturesController { }) public async get(): Promise { this.logger.debug('[get] started'); - let retorno = [new FeaturesDbDataModel()]; + let features = [new FeaturesDbDataModel()]; let responseCode = this.HTTP_ERROR; try { - retorno = await this.featuresDatService.getAll(); + features = await this.featuresDatService.getAll(); + const termsIdx = features.findIndex((feature) => feature.name === 'terms_and_conditions'); + this.logger.info(`[get] Retrieved terms idx: ${termsIdx}`); + const terms = await this.termsDatService.getVersion(features[termsIdx].version); + features[termsIdx].value = terms.value; responseCode = this.HTTP_SUCCESS_OK; - this.logger.info(`[get] Retrieved the features: ${JSON.stringify(retorno)}`); + this.logger.info(`[get] Retrieved the features: ${JSON.stringify(features)}`); } catch (e) { this.logger.warn(`[get] Got an error: ${e}`); } this.response.contentType('application/json').status(responseCode).send( - retorno + features ); return this.response; } diff --git a/src/dependency-injection-bindings.ts b/src/dependency-injection-bindings.ts index fecac51b..e8066190 100644 --- a/src/dependency-injection-bindings.ts +++ b/src/dependency-injection-bindings.ts @@ -38,4 +38,5 @@ export const ServicesBindings = { REGISTER_SERVICE: 'services.RegisterService', FEATURES_SERVICE: 'services.FeaturesDataService', FLYOVER_SERVICE: 'services.FlyoverService', + TERMS_SERVICE: 'services.TermsDataService', }; diff --git a/src/dependency-injection-handler.ts b/src/dependency-injection-handler.ts index 4df8edc7..1b29e04a 100644 --- a/src/dependency-injection-handler.ts +++ b/src/dependency-injection-handler.ts @@ -21,6 +21,7 @@ import {RskNodeService} from './services/rsk-node.service'; import {SyncStatusMongoService} from './services/sync-status-mongo.service'; import { PegoutDataProcessor } from './services/pegout-data.processor'; import { FeaturesMongoDbDataService } from './services/features-mongo.service'; +import { TermsMongoDbDataService } from './services/terms-mongo.service'; export class DependencyInjectionHandler { public static configureDependencies(app: Application): void { @@ -155,9 +156,14 @@ export class DependencyInjectionHandler { .toClass(RegisterService) .inScope(BindingScope.SINGLETON); - app - .bind(ServicesBindings.FEATURES_SERVICE) - .toClass(FeaturesMongoDbDataService) - .inScope(BindingScope.SINGLETON); + app + .bind(ServicesBindings.FEATURES_SERVICE) + .toClass(FeaturesMongoDbDataService) + .inScope(BindingScope.SINGLETON); + + app + .bind(ServicesBindings.TERMS_SERVICE) + .toClass(TermsMongoDbDataService) + .inScope(BindingScope.SINGLETON); } } diff --git a/src/models/features-data.model.ts b/src/models/features-data.model.ts index a07fc336..0cb9e069 100644 --- a/src/models/features-data.model.ts +++ b/src/models/features-data.model.ts @@ -5,7 +5,7 @@ export interface FeaturesDataModel { creationDate: Date; lastUpdateDate: Date; name: string; - value: string; + enabled: boolean; version: number; } @@ -17,7 +17,7 @@ export class FeaturesAppDataModel implements FeaturesDataModel{ creationDate: Date; lastUpdateDate: Date; name: string; - value: string; + enabled: boolean; version: number; } @@ -42,15 +42,18 @@ export class FeaturesDbDataModel implements SearchableModel, FeaturesDataModel { name: string; @property({ - type: 'string', + type: 'boolean', }) - value: string; + enabled: boolean; @property({ type: 'number', }) version: number; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + [prop: string]: any; + getId() { return this.name; } @@ -63,7 +66,7 @@ export class FeaturesDbDataModel implements SearchableModel, FeaturesDataModel { features.creationDate = other.creationDate; features.lastUpdateDate = other.lastUpdateDate; features.name = other.name; - features.value = other.value; + features.enabled = other.enabled; features.version = other.version; return features; } diff --git a/src/models/index.ts b/src/models/index.ts index 088a3de3..7ebc80c0 100644 --- a/src/models/index.ts +++ b/src/models/index.ts @@ -19,3 +19,4 @@ export * from './fee-amount.model'; export * from './register-payload.model'; export * from './features.model'; export * from './pegout-status.model'; +export * from './terms-db-data.model'; diff --git a/src/models/terms-db-data.model.ts b/src/models/terms-db-data.model.ts new file mode 100644 index 00000000..aa312e1d --- /dev/null +++ b/src/models/terms-db-data.model.ts @@ -0,0 +1,31 @@ +import { model, property} from '@loopback/repository'; +import { SearchableModel } from "./rsk/searchable-model"; + +@model() +export class TermsDbDataModel implements SearchableModel { + + @property({ + type: 'number', + required: true, + }) + version: number; + + @property({ + type: 'string', + required: true, + }) + value: string; + + + constructor(data?: Partial) { + Object.assign(this, data);; + } + + getId() { + return this.version; + } + // eslint-disable-next-line class-methods-use-this + getIdFieldName(): string { + return "version"; + } +} diff --git a/src/services/features-mongo.service.ts b/src/services/features-mongo.service.ts index 57b8ce28..335792bc 100644 --- a/src/services/features-mongo.service.ts +++ b/src/services/features-mongo.service.ts @@ -15,7 +15,7 @@ const FeaturesSchema = new mongoose.Schema({ creationDate: {type: Date}, lastUpdateDate: {type: Date}, name: {type: String, required: true}, - value: {type: String, required: true}, + enabled: {type: Boolean, required: true}, version: {type: Number, required: true}, }); diff --git a/src/services/terms-data.service.ts b/src/services/terms-data.service.ts new file mode 100644 index 00000000..5059ee72 --- /dev/null +++ b/src/services/terms-data.service.ts @@ -0,0 +1,6 @@ +import { GenericDataService } from './generic-data-service'; +import { TermsDbDataModel } from '../models'; + +export interface TermsDataService extends GenericDataService { + getVersion(version: number): Promise; +} diff --git a/src/services/terms-mongo.service.ts b/src/services/terms-mongo.service.ts new file mode 100644 index 00000000..a25b3d4e --- /dev/null +++ b/src/services/terms-mongo.service.ts @@ -0,0 +1,44 @@ +/* eslint-disable @typescript-eslint/no-floating-promises */ +import mongoose from 'mongoose'; +import {MongoDbDataService} from './mongodb-data.service'; +import { TermsDbDataModel } from '../models'; +import { TermsDataService } from './terms-data.service'; + +/* +- THESE MODEL INTERFACES AND CLASSES ARE REQUIRED FOR MONGO BUT WE DON'T WANT THEM EXPOSED OUT OF THIS LAYER +*/ +interface TermsMongoModel extends mongoose.Document, TermsDbDataModel { +} + +const TermsSchema = new mongoose.Schema({ + version: {type: Number, required: true}, + value: {type: String, required: true}, +}); + +const TermsConnector = mongoose.model("Terms", TermsSchema); + +export class TermsMongoDbDataService extends MongoDbDataService implements TermsDataService { + protected getByIdFilter(id: any) { + throw new Error('Method not implemented.'); + } + protected getManyFilter(filter?: any) { + throw new Error('Method not implemented.'); + } + protected getLoggerName(): string { + return 'TermsMongoService'; + } + protected getConnector(): mongoose.Model { + this.verifyAndCreateConnectionIfIsNecessary(); + return TermsConnector; + } + async verifyAndCreateConnectionIfIsNecessary() { + await this.ensureConnection(); + } + public async getVersion(version: number): Promise { + const [document] = await this.getConnector() + .find({ version }) + .exec(); + return document; + } + +}