import axios from 'axios' export type NewUser = { email: string; firstname: string; lastname: string; password: string; recaptcha: string | null; } export type MapInfo = { id: number; starred: boolean; name: string; labels: [string]; creator: string; modified: number; description: string; } export type BasicMapInfo = { name: string; description?: string; } export type FieldError = { id: string, msg: string } export type ErrorInfo = { msg?: string; fields?: Map; } interface Service { registerNewUser(user: NewUser): Promise; resetPassword(email: string): Promise; fetchAllMaps(): Promise; deleteMap(id: number): Promise; renameMap(id: number, basicInfo: BasicMapInfo): Promise; loadMapInfo(id: number): Promise; } class RestService implements Service { private baseUrl: string; private authFailed: () => void private maps: MapInfo[] = []; constructor(baseUrl: string, authFailed: () => void) { this.baseUrl = baseUrl; // Remove, just for develop .... function createMapInfo( id: number, starred: boolean, name: string, labels: [string], creator: string, modified: number, description: string ): MapInfo { return { id, name, labels, creator, modified, starred, description }; } this.maps = [ createMapInfo(1, true, "El Mapa", [""], "Paulo", 67, ""), createMapInfo(2, false, "El Mapa2", [""], "Paulo2", 67, ""), createMapInfo(3, false, "El Mapa3", [""], "Paulo3", 67, "") ]; } loadMapInfo(id: number): Promise { return Promise.resolve({ name: 'My Map', description: 'My Description' }); } renameMap(id: number, basicInfo: BasicMapInfo): Promise { const exists = this.maps.find(m => m.name == basicInfo.name) != undefined; if (!exists) { this.maps = this.maps.map(m => { const result = m; if (m.id == id) { result.description = basicInfo.description ? basicInfo.description : ''; result.name = basicInfo.name; } return result; }) return Promise.resolve(); } else { const fieldErrors: Map = new Map(); fieldErrors.set('name', 'name already exists ') return Promise.reject({ msg: 'Map already exists ...' + basicInfo.name, fields: fieldErrors }) }; } deleteMap(id: number): Promise { this.maps = this.maps.filter(m => m.id != id); return Promise.resolve(); } registerNewUser(user: NewUser): Promise { const handler = (success: () => void, reject: (error: ErrorInfo) => void) => { axios.post(this.baseUrl + '/service/users', JSON.stringify(user), { headers: { 'Content-Type': 'application/json' } } ).then(response => { // All was ok, let's sent to success page ...; success(); }).catch(error => { const response = error.response; const errorInfo = this.parseResponseOnError(response); reject(errorInfo); }); } return new Promise(handler); } fetchAllMaps(): Promise { return Promise.resolve(this.maps); } resetPassword(email: string): Promise { const handler = (success: () => void, reject: (error: ErrorInfo) => void) => { axios.post(this.baseUrl + '/service/users/resetPassword?email=' + email, null, { headers: { 'Content-Type': 'application/json' } } ).then(response => { // All was ok, let's sent to success page ...; success(); }).catch(error => { const response = error.response; const errorInfo = this.parseResponseOnError(response); reject(errorInfo); }); } return new Promise(handler); } private parseResponseOnError = (response: any): ErrorInfo => { let result: ErrorInfo | undefined; if (response) { const status: number = response.status; const data = response.data; console.log(data); switch (status) { case 401: this.authFailed(); break; default: if (data) { // Set global errorrs ... if (data.globalErrors) { let msg; let errors = data.globalErrors; if (errors.length > 0) { msg = errors[0]; } result = { msg: errors }; } // Set field errors ... if (data.fieldErrors) { // @Todo: Fix this ... result = { msg: data.fieldErrors }; result.fields = new Map(); } } else { result = { msg: response.statusText }; } } } // Network related problem ... if (!result) { result = { msg: 'Unexpected error. Please, try latter' }; } return result; } } export { Service, RestService }