import { Injectable } from '@angular/core' import { pauseFor } from '../../util/misc.util' import { ApiService } from './embassy-api.service' import { PatchOp, Update, Operation, RemoveOperation } from 'patch-db-client' import { DataModel, DependencyErrorType, InstallProgress, PackageDataEntry, PackageMainStatus, PackageState, ServerStatus, } from 'src/app/services/patch-db/data-model' import { CifsBackupTarget, Log, RR, WithRevision } from './api.types' import { parsePropertiesPermissive } from 'src/app/util/properties.util' import { Mock } from './api.fixures' import markdown from 'raw-loader!../../../../../../assets/markdown/md-sample.md' import { BehaviorSubject } from 'rxjs' import { LocalStorageBootstrap } from '../patch-db/local-storage-bootstrap' import { mockPatchData } from './mock-patch' @Injectable() export class MockApiService extends ApiService { readonly mockPatch$ = new BehaviorSubject>(undefined) private readonly revertTime = 4000 sequence: number constructor(private readonly bootstrapper: LocalStorageBootstrap) { super() } async getStatic(url: string): Promise { await pauseFor(2000) return markdown } // db async getRevisions(since: number): Promise { return this.getDump() } async getDump(): Promise { const cache = await this.bootstrapper.init() return { id: cache.sequence, value: cache.data, expireId: null, } } async setDbValueRaw(params: RR.SetDBValueReq): Promise { await pauseFor(2000) const patch = [ { op: PatchOp.REPLACE, path: '/ui' + params.pointer, value: params.value, }, ] return this.withRevision(patch) } // auth async login(params: RR.LoginReq): Promise { await pauseFor(2000) setTimeout(() => { this.mockPatch$.next({ id: 1, value: mockPatchData, expireId: null }) }, 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 getServerLogs( params: RR.GetServerLogsReq, ): Promise { await pauseFor(2000) let entries: Log[] if (Math.random() < 0.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, } setTimeout(() => { this.updateOSProgress(initialProgress.size) }, 500) const patch = [ { op: PatchOp.REPLACE, path: '/server-info/update-progress', value: initialProgress, }, ] return this.withRevision(patch, 'updating') } async restartServer( params: RR.RestartServerReq, ): Promise { await pauseFor(2000) return null } async shutdownServer( params: RR.ShutdownServerReq, ): Promise { await pauseFor(2000) return null } async systemRebuild( params: RR.RestartServerReq, ): Promise { await pauseFor(2000) return null } // marketplace URLs async marketplaceProxy(path: string, params: {}, url: string): Promise { await pauseFor(2000) if (path === '/package/v0/info') { return { name: 'Dark9', categories: [ 'featured', 'bitcoin', 'lightning', 'data', 'messaging', 'social', 'alt coin', ], } } else if (path === '/package/v0/index') { return Mock.MarketplacePkgsList } else if (path === '/package/v0/release-notes') { return Mock.ReleaseNotes } } async getEos( params: RR.GetMarketplaceEOSReq, ): Promise { await pauseFor(2000) return Mock.MarketplaceEos } // 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, }, ] return this.withRevision(patch, Mock.Notifications) } 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 { 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 getBackupTargets( params: RR.GetBackupTargetsReq, ): Promise { await pauseFor(2000) return Mock.BackupTargets } async addBackupTarget( params: RR.AddBackupTargetReq, ): Promise { await pauseFor(2000) const { hostname, path, username } = params return { latfgvwdbhjsndmk: { type: 'cifs', hostname, path: path.replace(/\\/g, '/'), username, mountable: true, 'embassy-os': null, }, } } async updateBackupTarget( params: RR.UpdateBackupTargetReq, ): Promise { await pauseFor(2000) const { id, hostname, path, username } = params return { [id]: { ...(Mock.BackupTargets[id] as CifsBackupTarget), hostname, path, username, }, } } async removeBackupTarget( params: RR.RemoveBackupTargetReq, ): Promise { await pauseFor(2000) return null } async getBackupInfo( params: RR.GetBackupInfoReq, ): Promise { await pauseFor(2000) return Mock.BackupInfo } async createBackupRaw( params: RR.CreateBackupReq, ): Promise { await pauseFor(2000) const path = '/server-info/status' const ids = ['bitcoind', 'lnd'] setTimeout(async () => { for (let i = 0; i < ids.length; i++) { const appPath = `/package-data/${ids[i]}/installed/status/main/status` const appPatch = [ { op: PatchOp.REPLACE, path: appPath, value: PackageMainStatus.BackingUp, }, ] this.updateMock(appPatch) await pauseFor(8000) appPatch[0].value = PackageMainStatus.Stopped this.updateMock(appPatch) } await pauseFor(1000) // set server back to running const lastPatch = [ { op: PatchOp.REPLACE, path, value: ServerStatus.Running, }, ] this.updateMock(lastPatch) }, 500) const originalPatch = [ { op: PatchOp.REPLACE, path, value: ServerStatus.BackingUp, }, ] return this.withRevision(originalPatch) } // 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() < 0.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, } setTimeout(async () => { this.updateProgress(params.id, initialProgress) }, 1000) 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, }, ] return this.withRevision(patch) } async dryUpdatePackage( params: RR.DryUpdatePackageReq, ): Promise { await pauseFor(2000) return {} } async getPackageConfig( params: RR.GetPackageConfigReq, ): Promise { await pauseFor(2000) return { config: Mock.MockConfig, spec: Mock.ConfigSpec, } } 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.withRevision(patch) } async restorePackagesRaw( params: RR.RestorePackagesReq, ): Promise { await pauseFor(2000) const patch: Operation[] = params.ids.map(id => { const initialProgress: InstallProgress = { size: 120, downloaded: 120, 'download-complete': true, validated: 0, 'validation-complete': false, unpacked: 0, 'unpack-complete': false, } const pkg: PackageDataEntry = { ...Mock.LocalPkgs[id], state: PackageState.Restoring, 'install-progress': initialProgress, installed: undefined, } setTimeout(async () => { this.updateProgress(id, initialProgress) }, 2000) return { op: PatchOp.ADD, path: `/package-data/${id}`, value: pkg, } }) return this.withRevision(patch) } async executePackageAction( params: RR.ExecutePackageActionReq, ): Promise { await pauseFor(2000) return Mock.ActionResponse } async startPackageRaw( params: RR.StartPackageReq, ): Promise { const path = `/package-data/${params.id}/installed/status/main` await pauseFor(2000) setTimeout(async () => { const patch2 = [ { op: PatchOp.REPLACE, path: path + '/status', value: PackageMainStatus.Running, }, { op: PatchOp.REPLACE, path: path + '/started', value: new Date().toISOString(), }, ] this.updateMock(patch2) const patch3 = [ { op: PatchOp.REPLACE, path: path + '/health', value: { 'ephemeral-health-check': { result: 'starting', }, 'unnecessary-health-check': { result: 'disabled', }, }, }, ] this.updateMock(patch3) await pauseFor(2000) const patch4 = [ { op: PatchOp.REPLACE, path: path + '/health', value: { 'ephemeral-health-check': { result: 'starting', }, 'unnecessary-health-check': { result: 'disabled', }, 'chain-state': { result: 'loading', message: 'Bitcoin is syncing from genesis', }, 'p2p-interface': { result: 'success', }, 'rpc-interface': { result: 'failure', error: 'RPC interface unreachable.', }, }, }, ] this.updateMock(patch4) }, 2000) const originalPatch = [ { op: PatchOp.REPLACE, path: path + '/status', value: PackageMainStatus.Starting, }, ] return this.withRevision(originalPatch) } 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` setTimeout(() => { const patch2 = [ { op: PatchOp.REPLACE, path: path + '/status', value: PackageMainStatus.Stopped, }, ] this.updateMock(patch2) }, this.revertTime) const patch = [ { op: PatchOp.REPLACE, path: path + '/status', value: PackageMainStatus.Stopping, }, { op: PatchOp.REPLACE, path: path + '/health', value: {}, }, ] return this.withRevision(patch) } async dryUninstallPackage( params: RR.DryUninstallPackageReq, ): Promise { await pauseFor(2000) return {} } async uninstallPackageRaw( params: RR.UninstallPackageReq, ): Promise { await pauseFor(2000) setTimeout(async () => { const patch2 = [ { op: PatchOp.REMOVE, path: `/package-data/${params.id}`, } as RemoveOperation, ] this.updateMock(patch2) }, this.revertTime) const patch = [ { op: PatchOp.REPLACE, path: `/package-data/${params.id}/state`, value: PackageState.Removing, }, ] return this.withRevision(patch) } async deleteRecoveredPackageRaw( params: RR.DeleteRecoveredPackageReq, ): Promise { await pauseFor(2000) const patch = [ { op: PatchOp.REMOVE, path: `/recovered-packages/${params.id}`, } as RemoveOperation, ] return this.withRevision(patch) } async dryConfigureDependency( params: RR.DryConfigureDependencyReq, ): Promise { await pauseFor(2000) return { 'old-config': Mock.MockConfig, 'new-config': Mock.MockDependencyConfig, spec: Mock.ConfigSpec, } } private async updateProgress( id: string, initialProgress: InstallProgress, ): Promise { 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 } const patch = [ { op: PatchOp.REPLACE, path: `/package-data/${id}/install-progress`, value: initialProgress, }, ] this.updateMock(patch) } } setTimeout(() => { const patch2: any = [ { op: PatchOp.REPLACE, path: `/package-data/${id}`, value: { ...Mock.LocalPkgs[id] }, }, { op: PatchOp.REMOVE, path: `/package-data/${id}/install-progress`, }, ] this.updateMock(patch2) }, 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, }, ] this.updateMock(patch) } const patch2 = [ { op: PatchOp.REPLACE, path: `/server-info/update-progress/downloaded`, value: size, }, ] this.updateMock(patch2) setTimeout(async () => { const patch3: Operation[] = [ { op: PatchOp.REPLACE, path: '/server-info/status', value: ServerStatus.Updated, }, { op: PatchOp.REMOVE, path: '/server-info/update-progress', }, ] this.updateMock(patch3) // quickly revert server to "running" for continued testing await pauseFor(100) const patch4 = [ { op: PatchOp.REPLACE, path: '/server-info/status', value: ServerStatus.Running, }, ] this.updateMock(patch4) }, 1000) } private async updateMock(patch: Operation[]): Promise { if (!this.sequence) { const { sequence } = await this.bootstrapper.init() this.sequence = sequence } const revision = { id: ++this.sequence, patch, expireId: null, } this.mockPatch$.next(revision) } private async withRevision( patch: Operation[], response: T = null, ): Promise> { if (!this.sequence) { const { sequence } = await this.bootstrapper.init() this.sequence = sequence } const revision = { id: ++this.sequence, patch, expireId: null, } return { response, revision } } }