process.env.DOCKER_HOST = 'unix:///your/dockerode/mocks/are/not/working'; import * as dockerode from 'dockerode'; import { Stream } from 'stream'; import _ = require('lodash'); import { NotFoundError } from '~/lib/errors'; const overrides: Dictionary<(...args: any[]) => Resolvable> = {}; interface Action { name: string; parameters: Dictionary; } export let actions: Action[] = []; export function resetHistory() { actions = []; } /** * Tracks actions performed on a mocked dockerode instance * @param name action called * @param parameters data passed */ function addAction(name: string, parameters: Dictionary = {}) { actions.push({ name, parameters, }); } type DockerodeFunction = keyof dockerode; for (const fn of Object.getOwnPropertyNames(dockerode.prototype)) { if ( fn !== 'constructor' && typeof (dockerode.prototype as any)[fn] === 'function' ) { (dockerode.prototype as any)[fn] = async function (...args: any[]) { console.log(`🐳 Calling ${fn}...`); if (overrides[fn] != null) { return overrides[fn](args); } /* Return promise */ return Promise.resolve([]); }; } } // default overrides needed to startup... registerOverride( 'getEvents', async () => new Stream.Readable({ read: () => { return _.noop(); }, }), ); /** * Used to add or modifying functions on the mocked dockerode * @param name function name to override * @param fn function to execute */ export function registerOverride< T extends DockerodeFunction, P extends Parameters, R extends ReturnType, >(name: T, fn: (...args: P) => R) { console.log(`Overriding ${name}...`); overrides[name] = fn; } export function restoreOverride(name: T) { if (Object.hasOwn(overrides, name)) { delete overrides[name]; } } export interface TestData { networks: Dictionary; images: Dictionary; containers: Dictionary; volumes: Dictionary; } function createMockedDockerode(data: TestData) { const mockedDockerode = dockerode.prototype; mockedDockerode.listImages = async () => []; mockedDockerode.listVolumes = async () => { addAction('listVolumes'); return { Volumes: data.volumes as dockerode.VolumeInspectInfo[], Warnings: [], }; }; mockedDockerode.getVolume = (name: string) => { addAction('getVolume'); const picked = data.volumes.filter((v: Dictionary) => v.Name === name); if (picked.length !== 1) { throw new NotFoundError(); } const volume = picked[0]; return { ...volume, inspect: async () => { addAction('inspect'); // TODO fully implement volume inspect. // This should return VolumeInspectInfo not Volume return volume; }, remove: async (options?: any) => { addAction('remove', options); data.volumes = _.reject(data.volumes, { name: volume.name }); }, name: volume.name, modem: {}, } as dockerode.Volume; }; mockedDockerode.createContainer = async ( options: dockerode.ContainerCreateOptions, ) => { addAction('createContainer', { options }); return { start: async () => { addAction('start'); }, } as dockerode.Container; }; mockedDockerode.getContainer = (id: string) => { addAction('getContainer', { id }); return { inspect: async () => { return data.containers.filter((c: Dictionary) => c.id === id); }, start: async () => { addAction('start'); data.containers = data.containers.map((c: any) => { if (c.containerId === id) { c.status = 'Installing'; } return c; }); }, stop: async () => { addAction('stop'); data.containers = data.containers.map((c: any) => { if (c.containerId === id) { c.status = 'Stopping'; } return c; }); }, remove: async () => { addAction('remove'); data.containers = data.containers.map((c: any) => { if (c.containerId === id) { c.status = 'removing'; } return c; }); }, } as dockerode.Container; }; mockedDockerode.getNetwork = (id: string) => { addAction('getNetwork', { id }); return { inspect: async () => { addAction('inspect'); return data.networks[id]; }, } as dockerode.Network; }; mockedDockerode.getImage = (name: string) => { addAction('getImage', { name }); return { inspect: async () => { addAction('inspect'); return data.images[name]; }, remove: async () => { addAction('remove'); data.images = _.reject(data.images, { name, }); }, } as dockerode.Image; }; return mockedDockerode; } type Prototype = { [key: string]: any }; function clonePrototype(prototype: Prototype): Prototype { const clone: Prototype = {}; Object.getOwnPropertyNames(prototype).forEach((fn) => { if (fn !== 'constructor' && typeof prototype[fn] === 'function') { clone[fn] = prototype[fn]; } }); return clone; } function assignPrototype(target: Prototype, source: Prototype) { Object.keys(source).forEach((fn) => { target[fn] = source[fn]; }); } export async function testWithData( data: Partial, test: () => Promise, ) { const mockedData: TestData = { ...{ networks: [], images: [], containers: [], volumes: [], }, ...data, }; // grab the original prototype... const basePrototype = clonePrototype(dockerode.prototype); // @ts-expect-error setting a RO property dockerode.prototype = createMockedDockerode(mockedData); try { // run the test... await test(); } finally { // reset the original prototype... assignPrototype(dockerode.prototype, basePrototype); } }