mirror of
https://github.com/balena-os/balena-supervisor.git
synced 2025-01-07 06:18:46 +00:00
161 lines
3.6 KiB
TypeScript
161 lines
3.6 KiB
TypeScript
|
import App from '~/src/compose/app';
|
||
|
import * as imageManager from '~/src/compose/images';
|
||
|
import { Image } from '~/src/compose/images';
|
||
|
import Network from '~/src/compose/network';
|
||
|
import Service from '~/src/compose/service';
|
||
|
import { ServiceComposeConfig } from '~/src/compose/types/service';
|
||
|
import Volume from '~/src/compose/volume';
|
||
|
import { InstancedAppState } from '~/src/types/state';
|
||
|
|
||
|
export const DEFAULT_NETWORK = Network.fromComposeObject(
|
||
|
'default',
|
||
|
1,
|
||
|
'appuuid',
|
||
|
{},
|
||
|
);
|
||
|
|
||
|
export async function createService(
|
||
|
{
|
||
|
appId = 1,
|
||
|
appUuid = 'appuuid',
|
||
|
serviceName = 'main',
|
||
|
commit = 'main-commit',
|
||
|
...conf
|
||
|
} = {} as Partial<ServiceComposeConfig>,
|
||
|
{ state = {} as Partial<Service>, options = {} as any } = {},
|
||
|
) {
|
||
|
const svc = await Service.fromComposeObject(
|
||
|
{
|
||
|
appId,
|
||
|
appUuid,
|
||
|
serviceName,
|
||
|
commit,
|
||
|
// db ids should not be used for target state calculation, but images
|
||
|
// are compared using _.isEqual so leaving this here to have image comparisons
|
||
|
// match
|
||
|
serviceId: 1,
|
||
|
imageId: 1,
|
||
|
releaseId: 1,
|
||
|
...conf,
|
||
|
},
|
||
|
options,
|
||
|
);
|
||
|
|
||
|
// Add additonal configuration
|
||
|
for (const k of Object.keys(state)) {
|
||
|
(svc as any)[k] = (state as any)[k];
|
||
|
}
|
||
|
return svc;
|
||
|
}
|
||
|
|
||
|
export function createImage(
|
||
|
{
|
||
|
appId = 1,
|
||
|
appUuid = 'appuuid',
|
||
|
name = 'test-image',
|
||
|
serviceName = 'main',
|
||
|
commit = 'main-commit',
|
||
|
...extra
|
||
|
} = {} as Partial<Image>,
|
||
|
) {
|
||
|
return {
|
||
|
appId,
|
||
|
appUuid,
|
||
|
name,
|
||
|
serviceName,
|
||
|
commit,
|
||
|
// db ids should not be used for target state calculation, but images
|
||
|
// are compared using _.isEqual so leaving this here to have image comparisons
|
||
|
// match
|
||
|
imageId: 1,
|
||
|
releaseId: 1,
|
||
|
serviceId: 1,
|
||
|
dependent: 0,
|
||
|
...extra,
|
||
|
} as Image;
|
||
|
}
|
||
|
|
||
|
export function createApps(
|
||
|
{
|
||
|
services = [] as Service[],
|
||
|
networks = [] as Network[],
|
||
|
volumes = [] as Volume[],
|
||
|
},
|
||
|
target = false,
|
||
|
) {
|
||
|
const servicesByAppId = services.reduce(
|
||
|
(svcs, s) => ({ ...svcs, [s.appId]: [s].concat(svcs[s.appId] || []) }),
|
||
|
{} as Dictionary<Service[]>,
|
||
|
);
|
||
|
const volumesByAppId = volumes.reduce(
|
||
|
(vols, v) => ({ ...vols, [v.appId]: [v].concat(vols[v.appId] || []) }),
|
||
|
{} as Dictionary<Volume[]>,
|
||
|
);
|
||
|
const networksByAppId = networks.reduce(
|
||
|
(nets, n) => ({ ...nets, [n.appId]: [n].concat(nets[n.appId] || []) }),
|
||
|
{} as Dictionary<Network[]>,
|
||
|
);
|
||
|
|
||
|
const allAppIds = [
|
||
|
...new Set([
|
||
|
...Object.keys(servicesByAppId),
|
||
|
...Object.keys(networksByAppId),
|
||
|
...Object.keys(volumesByAppId),
|
||
|
]),
|
||
|
].map((i) => parseInt(i, 10));
|
||
|
|
||
|
const apps: InstancedAppState = {};
|
||
|
for (const appId of allAppIds) {
|
||
|
apps[appId] = new App(
|
||
|
{
|
||
|
appId,
|
||
|
services: servicesByAppId[appId] ?? [],
|
||
|
networks: (networksByAppId[appId] ?? []).reduce(
|
||
|
(nets, n) => ({ ...nets, [n.name]: n }),
|
||
|
{},
|
||
|
),
|
||
|
volumes: (volumesByAppId[appId] ?? []).reduce(
|
||
|
(vols, v) => ({ ...vols, [v.name]: v }),
|
||
|
{},
|
||
|
),
|
||
|
},
|
||
|
target,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
return apps;
|
||
|
}
|
||
|
|
||
|
export function createCurrentState({
|
||
|
services = [] as Service[],
|
||
|
networks = [] as Network[],
|
||
|
volumes = [] as Volume[],
|
||
|
images = services.map((s) => ({
|
||
|
// Infer images from services by default
|
||
|
dockerImageId: s.dockerImageId,
|
||
|
...imageManager.imageFromService(s),
|
||
|
})) as Image[],
|
||
|
downloading = [] as string[],
|
||
|
}) {
|
||
|
const currentApps = createApps({ services, networks, volumes });
|
||
|
|
||
|
const containerIdsByAppId = services.reduce(
|
||
|
(ids, s) => ({
|
||
|
...ids,
|
||
|
[s.appId]: {
|
||
|
...ids[s.appId],
|
||
|
...(s.serviceName &&
|
||
|
s.containerId && { [s.serviceName]: s.containerId }),
|
||
|
},
|
||
|
}),
|
||
|
{} as { [appId: number]: Dictionary<string> },
|
||
|
);
|
||
|
|
||
|
return {
|
||
|
currentApps,
|
||
|
availableImages: images,
|
||
|
downloading,
|
||
|
containerIdsByAppId,
|
||
|
};
|
||
|
}
|