import * as _ from 'lodash'; import { Router } from 'express'; import rewire = require('rewire'); import { unlinkAll } from '../../src/lib/fs-utils'; import * as applicationManager from '../../src/compose/application-manager'; import * as networkManager from '../../src/compose/network-manager'; import * as serviceManager from '../../src/compose/service-manager'; import * as volumeManager from '../../src/compose/volume-manager'; import * as commitStore from '../../src/compose/commit'; import * as config from '../../src/config'; import * as db from '../../src/db'; import { createV1Api } from '../../src/device-api/v1'; import { createV2Api } from '../../src/device-api/v2'; import * as deviceState from '../../src/device-state'; import SupervisorAPI from '../../src/supervisor-api'; import { Service } from '../../src/compose/service'; import { Image } from '../../src/compose/images'; const apiBinder = rewire('../../src/api-binder'); const DB_PATH = './test/data/supervisor-api.sqlite'; // Holds all values used for stubbing const STUBBED_VALUES = { commits: { 1: '7fc9c5bea8e361acd49886fe6cc1e1cd', 2: '4e380136c2cf56cd64197d51a1ab263a', }, services: [ { appId: 1, imageId: 1111, status: 'Running', releaseId: 99999, createdAt: new Date('2020-04-25T04:15:23.111Z'), serviceName: 'main', }, { appId: 1, imageId: 2222, status: 'Running', releaseId: 99999, createdAt: new Date('2020-04-25T04:15:23.111Z'), serviceName: 'redis', }, { appId: 2, imageId: 3333, containerId: 'abc123', status: 'Running', releaseId: 77777, createdAt: new Date('2020-05-15T19:33:06.088Z'), serviceName: 'main', }, ], images: [], networks: [], volumes: [], }; // Useful for creating mock services in the ServiceManager const mockService = (overrides?: Partial) => { return { ...{ appId: 1658654, status: 'Running', serviceName: 'main', imageId: 2885946, serviceId: 640681, containerId: 'f93d386599d1b36e71272d46ad69770cff333842db04e2e4c64dda7b54da07c6', createdAt: '2020-11-13T20:29:44.143Z', config: { labels: {}, }, toDockerContainer: () => { return; }, extraNetworksToJoin: () => { return []; }, isEqualConfig: (service: Service) => { return _.isEqual( _.pick(mockService, ['imageId', 'containerId', 'serviceId']), _.pick(service, ['imageId', 'containerId', 'serviceId']), ); }, }, ...overrides, } as Service; }; // Useful for creating mock images that are returned from Images.getStatus const mockImage = (overrides?: Partial) => { return { ...{ name: 'registry2.balena-cloud.com/v2/e2bf6410ffc30850e96f5071cdd1dca8@sha256:e2e87a8139b8fc14510095b210ad652d7d5badcc64fdc686cbf749d399fba15e', appId: 1658654, serviceName: 'main', imageId: 2885946, dockerImageId: 'sha256:4502983d72e2c72bc292effad1b15b49576da3801356f47fd275ba274d409c1a', status: 'Downloaded', downloadProgress: null, }, ...overrides, } as Image; }; const mockedOptions = { listenPort: 54321, timeout: 30000, }; /** * THIS MOCKED API CONTAINS STUBS THAT MIGHT CAUSE UNEXPECTED RESULTS * IF YOU WANT TO ADD/MODIFY STUBS THAT INVOLVE API OPERATIONS * AND MULTIPLE TEST CASES WILL USE THEM THEN ADD THEM HERE * OTHERWISE YOU CAN ADD STUBS ON A PER TEST CASE BASIS * * EXAMPLE: We stub ApplicationManager so there is at least 1 running app * * You can see all the stubbed values convientiely in STUBBED_VALUES. * */ async function create(): Promise { // Get SupervisorAPI construct options await createAPIOpts(); // Stub functions setupStubs(); // Create SupervisorAPI const api = new SupervisorAPI({ routers: [deviceState.router, buildRoutes()], healthchecks: [deviceState.healthcheck, apiBinder.healthcheck], }); const macAddress = await config.get('macAddress'); deviceState.reportCurrentState({ mac_address: macAddress, }); // Return SupervisorAPI that is not listening yet return api; } async function cleanUp(): Promise { // Clean up test data await unlinkAll(DB_PATH); // Restore created SinonStubs return restoreStubs(); } async function createAPIOpts(): Promise { await db.initialized; await deviceState.initialized; // Initialize and set values for mocked Config await initConfig(); } async function initConfig(): Promise { // Initialize this config await config.initialized; // Set a currentCommit for (const [id, commit] of Object.entries(STUBBED_VALUES.commits)) { await commitStore.upsertCommitForApp(parseInt(id, 10), commit); } } function buildRoutes(): Router { // Add to existing apiBinder router (it contains additional middleware and endpoints) const router = apiBinder.router; // Add V1 routes createV1Api(applicationManager.router); // Add V2 routes createV2Api(applicationManager.router); // Return modified Router return router; } // TO-DO: Create a cleaner way to restore previous values. const originalNetGetAll = networkManager.getAllByAppId; const originalVolGetAll = volumeManager.getAllByAppId; const originalSvcGetAppId = serviceManager.getAllByAppId; const originalSvcGetStatus = serviceManager.getStatus; const originalReadyForUpdates = apiBinder.__get__('readyForUpdates'); function setupStubs() { apiBinder.__set__('readyForUpdates', true); // @ts-expect-error Assigning to a RO property networkManager.getAllByAppId = async () => STUBBED_VALUES.networks; // @ts-expect-error Assigning to a RO property volumeManager.getAllByAppId = async () => STUBBED_VALUES.volumes; // @ts-expect-error Assigning to a RO property serviceManager.getStatus = async () => STUBBED_VALUES.services; // @ts-expect-error Assigning to a RO property serviceManager.getAllByAppId = async (appId) => _.filter(STUBBED_VALUES.services, (service) => service.appId === appId); } function restoreStubs() { apiBinder.__set__('readyForUpdates', originalReadyForUpdates); // @ts-expect-error Assigning to a RO property networkManager.getAllByAppId = originalNetGetAll; // @ts-expect-error Assigning to a RO property volumeManager.getAllByAppId = originalVolGetAll; // @ts-expect-error Assigning to a RO property serviceManager.getStatus = originalSvcGetStatus; // @ts-expect-error Assigning to a RO property serviceManager.getAllByAppId = originalSvcGetAppId; } export = { create, cleanUp, STUBBED_VALUES, mockService, mockImage, mockedOptions, };