import { Injectable } from '@angular/core' import { pauseFor } from '../../util/misc.util' import { ApiService } from './embassy-api.service' import { PatchOp } from 'patch-db-client' import { DependencyErrorType, InstallProgress, PackageDataEntry, PackageMainStatus, PackageState, ServerStatus } from 'src/app/services/patch-db/data-model' import { RR, WithRevision } from './api.types' import { parsePropertiesPermissive } from 'src/app/util/properties.util' import { Mock } from './api.fixures' import { HttpService } from '../http.service' import markdown from 'raw-loader!src/assets/markdown/md-sample.md' @Injectable() export class MockApiService extends ApiService { private readonly revertTime = 4000 constructor ( private readonly http: HttpService, ) { super() } async getStatic (url: string): Promise { await pauseFor(2000) return markdown } // db async getRevisions (since: number): Promise { return this.http.rpcRequest({ method: 'db.revisions', params: { since } }) } async getDump (): Promise { return this.http.rpcRequest({ method: 'db.dump' }) } async setDbValueRaw (params: RR.SetDBValueReq): Promise { await pauseFor(2000) return this.http.rpcRequest>({ method: 'db.put.ui', params }) } // auth async login (params: RR.LoginReq): Promise { await pauseFor(2000) return null } async logout (params: RR.LogoutReq): Promise { await pauseFor(2000) return null } async getSessions (params: RR.GetSessionsReq): Promise { await pauseFor(2000) return Mock.Sessions } async killSessions (params: RR.KillSessionsReq): Promise { await pauseFor(2000) return null } // server async setShareStatsRaw (params: RR.SetShareStatsReq): Promise { await pauseFor(2000) const patch = [ { op: PatchOp.REPLACE, path: '/server-info/share-stats', value: params.value, }, ] return this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) } async getServerLogs (params: RR.GetServerLogsReq): Promise { await pauseFor(2000) let entries if (Math.random() < .2) { entries = Mock.ServerLogs } else { const arrLength = params.limit ? Math.ceil(params.limit / Mock.ServerLogs.length) : 10 entries = new Array(arrLength).fill(Mock.ServerLogs).reduce((acc, val) => acc.concat(val), []) } return { entries, 'start-cursor': 'startCursor', 'end-cursor': 'endCursor', } } async getServerMetrics (params: RR.GetServerMetricsReq): Promise { await pauseFor(2000) return Mock.getServerMetrics() } async getPkgMetrics (params: RR.GetServerMetricsReq): Promise { await pauseFor(2000) return Mock.getAppMetrics() } async updateServerRaw (params: RR.UpdateServerReq): Promise { await pauseFor(2000) const initialProgress = { size: 10000, downloaded: 0, } const patch = [ { op: PatchOp.REPLACE, path: '/server-info/status', value: ServerStatus.Updating, }, { op: PatchOp.REPLACE, path: '/server-info/update-progress', value: initialProgress, }, ] const res = await this.http.rpcRequest({ method: 'db.patch', params: { patch } }) res.response = 'updating' this.updateOSProgress(initialProgress.size) return res } async restartServer (params: RR.RestartServerReq): Promise { await pauseFor(2000) return null } async shutdownServer (params: RR.ShutdownServerReq): Promise { await pauseFor(2000) return null } // marketplace URLs async getEos (params: RR.GetMarketplaceEOSReq): Promise { await pauseFor(2000) return Mock.MarketplaceEos } async getMarketplaceData (params: RR.GetMarketplaceDataReq): Promise { await pauseFor(2000) return { categories: ['featured', 'bitcoin', 'lightning', 'data', 'messaging', 'social', 'alt coin'], } } async getMarketplacePkgs (params: RR.GetMarketplacePackagesReq): Promise { await pauseFor(2000) return Mock.MarketplacePkgsList } async getReleaseNotes (params: RR.GetReleaseNotesReq): Promise { await pauseFor(2000) return Mock.ReleaseNotes } async getLatestVersion (params: RR.GetLatestVersionReq): Promise { await pauseFor(2000) return params.ids.reduce((obj, id) => { obj[id] = '1.3.0' return obj }, { }) } // async setPackageMarketplaceRaw (params: RR.SetPackageMarketplaceReq): Promise { // await pauseFor(2000) // const patch = [ // { // op: PatchOp.REPLACE, // path: '/server-info/package-marketplace', // value: params.url, // }, // ] // return this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) // } // password // async updatePassword (params: RR.UpdatePasswordReq): Promise { // await pauseFor(2000) // return null // } // notification async getNotificationsRaw (params: RR.GetNotificationsReq): Promise { await pauseFor(2000) const patch = [ { op: PatchOp.REPLACE, path: '/server-info/unread-notification-count', value: 0, }, ] const { revision } = await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) as WithRevision return { response: Mock.Notifications, revision, } } async deleteNotification (params: RR.DeleteNotificationReq): Promise { await pauseFor(2000) return null } async deleteAllNotifications (params: RR.DeleteAllNotificationsReq): Promise { await pauseFor(2000) return null } // wifi async getWifi (params: RR.GetWifiReq): Promise < RR.GetWifiRes > { await pauseFor(2000) return Mock.Wifi } async setWifiCountry (params: RR.SetWifiCountryReq): Promise { await pauseFor(2000) return null } async addWifi (params: RR.AddWifiReq): Promise { await pauseFor(2000) return null } async connectWifi (params: RR.ConnectWifiReq): Promise { await pauseFor(2000) return null } async deleteWifi (params: RR.DeleteWifiReq): Promise { await pauseFor(2000) return null } // ssh async getSshKeys (params: RR.GetSSHKeysReq): Promise { await pauseFor(2000) return Mock.SshKeys } async addSshKey (params: RR.AddSSHKeyReq): Promise { await pauseFor(2000) return Mock.SshKey } async deleteSshKey (params: RR.DeleteSSHKeyReq): Promise { await pauseFor(2000) return null } // backup async createBackupRaw (params: RR.CreateBackupReq): Promise { await pauseFor(2000) const path = '/server-info/status' const patch = [ { op: PatchOp.REPLACE, path, value: ServerStatus.BackingUp, }, ] const res = await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) setTimeout(() => { const patch = [ { op: PatchOp.REPLACE, path, value: ServerStatus.Running, }, ] this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) }, this.revertTime) return res } // disk async getDisks (params: RR.GetDisksReq): Promise { await pauseFor(2000) return Mock.Disks } async getBackupInfo (params: RR.GetBackupInfoReq): Promise { await pauseFor(2000) return Mock.BackupInfo } // package async getPackageProperties (params: RR.GetPackagePropertiesReq): Promise['data']> { await pauseFor(2000) return parsePropertiesPermissive(Mock.PackageProperties) } async getPackageLogs (params: RR.GetPackageLogsReq): Promise { await pauseFor(2000) let entries if (Math.random() < .2) { entries = Mock.PackageLogs } else { const arrLength = params.limit ? Math.ceil(params.limit / Mock.PackageLogs.length) : 10 entries = new Array(arrLength).fill(Mock.PackageLogs).reduce((acc, val) => acc.concat(val), []) } return { entries, 'start-cursor': 'startCursor', 'end-cursor': 'endCursor', } } async installPackageRaw (params: RR.InstallPackageReq): Promise { await pauseFor(2000) const initialProgress: InstallProgress = { size: 120, downloaded: 0, 'download-complete': false, validated: 0, 'validation-complete': false, unpacked: 0, 'unpack-complete': false, } const pkg: PackageDataEntry = { ...Mock.LocalPkgs[params.id], state: PackageState.Installing, 'install-progress': initialProgress, installed: undefined, } const patch = [ { op: PatchOp.ADD, path: `/package-data/${params.id}`, value: pkg, }, ] const res = await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) setTimeout(async () => { this.updateProgress(params.id, initialProgress) }, 1000) return res } async dryUpdatePackage (params: RR.DryUpdatePackageReq): Promise { await pauseFor(2000) return { } } async getPackageConfig (params: RR.GetPackageConfigReq): Promise { await pauseFor(2000) return Mock.PackageConfig } async drySetPackageConfig (params: RR.DrySetPackageConfigReq): Promise { await pauseFor(2000) return { } } async setPackageConfigRaw (params: RR.SetPackageConfigReq): Promise { await pauseFor(2000) const patch = [ { op: PatchOp.REPLACE, path: `/package-data/${params.id}/installed/status/configured`, value: true, }, ] return this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) } async restorePackageRaw (params: RR.RestorePackageReq): Promise { await pauseFor(2000) const path = `/package-data/${params.id}/installed/status/main/status` const patch = [ { op: PatchOp.REPLACE, path, value: PackageMainStatus.Restoring, }, ] const res = await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) setTimeout(() => { const patch = [ { op: PatchOp.REPLACE, path, value: PackageMainStatus.Stopped, }, ] this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) }, this.revertTime) return res } async executePackageAction (params: RR.ExecutePackageActionReq): Promise { await pauseFor(2000) return Mock.ActionResponse } async startPackageRaw (params: RR.StartPackageReq): Promise { await pauseFor(2000) const path = `/package-data/${params.id}/installed/status/main` const patch = [ { op: PatchOp.REPLACE, path: path + '/status', value: PackageMainStatus.Running, }, { op: PatchOp.REPLACE, path: path + '/started', value: new Date().toISOString(), }, ] return this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) } async dryStopPackage (params: RR.DryStopPackageReq): Promise { await pauseFor(2000) return { 'lnd': { dependency: 'bitcoind', error: { type: DependencyErrorType.NotRunning, }, }, } } async stopPackageRaw (params: RR.StopPackageReq): Promise { await pauseFor(2000) const path = `/package-data/${params.id}/installed/status/main/status` const patch = [ { op: PatchOp.REPLACE, path, value: PackageMainStatus.Stopping, }, ] const res = await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) setTimeout(() => { const patch = [ { op: PatchOp.REPLACE, path, value: PackageMainStatus.Stopped, }, ] this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) }, this.revertTime) return res } async dryUninstallPackage (params: RR.DryUninstallPackageReq): Promise { await pauseFor(2000) return { } } async uninstallPackageRaw (params: RR.UninstallPackageReq): Promise { await pauseFor(2000) const patch = [ { op: PatchOp.REPLACE, path: `/package-data/${params.id}/state`, value: PackageState.Removing, }, ] let res: any res = await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) setTimeout(async () => { const patch = [ { op: PatchOp.REMOVE, path: `/package-data/${params.id}`, }, ] this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) }, this.revertTime) return res } async deleteRecoveredPackageRaw (params: RR.DeleteRecoveredPackageReq): Promise { await pauseFor(2000) let res: any const patch = [ { op: PatchOp.REMOVE, path: `/recovered-packages/${params.id}`, }, ] res = await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) return res } async dryConfigureDependency (params: RR.DryConfigureDependencyReq): Promise { await pauseFor(2000) return { testnet: true, // objectList: [], // unionList: [], randomEnum: 'option2', favoriteNumber: 9, secondaryNumbers: [2, 3, 5, 6], rpcsettings: { laws: { law1: 'The 1st law', law2: 'The 2nd law', }, rpcpass: null, rpcuser: '123', rulemakers: [], }, advanced: { notifications: ['call', 'text'], }, // bitcoinNode: undefined, port: 22, // rpcallowip: undefined, // rpcauth: ['matt: 8273gr8qwoidm1uid91jeh8y23gdio1kskmwejkdnm'], } } private async updateProgress (id: string, initialProgress: InstallProgress) { const phases = [ { progress: 'downloaded', completion: 'download-complete'}, { progress: 'validated', completion: 'validation-complete'}, { progress: 'unpacked', completion: 'unpack-complete'}, ] for (let phase of phases) { let i = initialProgress[phase.progress] while (i < initialProgress.size) { await pauseFor(250) i = Math.min(i + 5, initialProgress.size) initialProgress[phase.progress] = i if (i === initialProgress.size) { initialProgress[phase.completion] = true } let patch: any if (initialProgress['unpack-complete']) { patch = [ { op: PatchOp.REMOVE, path: `/package-data/${id}/install-progress`, }, { op: PatchOp.REMOVE, path: `/recovered-packages/${id}`, }, ] } else { patch = [ { op: PatchOp.REPLACE, path: `/package-data/${id}/install-progress`, value: initialProgress, }, ] } try { await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) } catch (e) { console.error('Insufficient Mocks, happens when installing a service that does not exist in recovered-package') } } } setTimeout(() => { const patch = [ { op: PatchOp.REPLACE, path: `/package-data/${id}`, value: Mock.LocalPkgs[id], }, ] this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) }, 1000) } private async updateOSProgress (size: number) { let downloaded = 0 while (downloaded < size) { await pauseFor(250) downloaded += 500 const patch = [ { op: PatchOp.REPLACE, path: `/server-info/update-progress/downloaded`, value: downloaded, }, ] await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) } const patch = [ { op: PatchOp.REPLACE, path: `/server-info/update-progress/downloaded`, value: size, }, ] await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) setTimeout(async () => { const patch = [ { op: PatchOp.REPLACE, path: '/server-info/status', value: ServerStatus.Updated, }, { op: PatchOp.REMOVE, path: '/server-info/update-progress', }, ] await this.http.rpcRequest>({ method: 'db.patch', params: { patch } }) // quickly revert server to "running" for continued testing const patch2 = [ { op: PatchOp.REPLACE, path: '/server-info/status', value: ServerStatus.Running, }, ] this.http.rpcRequest>({ method: 'db.patch', params: { patch: patch2 } }) }, 1000) } }