mirror of
https://github.com/balena-os/balena-supervisor.git
synced 2025-01-19 03:06:27 +00:00
89175432af
We have seen a few times devices with duplicated network names for some reason. While we don't know the cause the networks get duplicates, this can be disruptive for updates as trying to create a container referencing a duplicate network results in a 400 error from the engine. This commit finds and removes duplicate networks via the state engine, this means that even if somehow a container could be referencing a network that has been duplicated later somehow, this will remove the container first. While thies doesn't solve the problem of duplicate networks being created in the first place, it will fix the state of the system to correct the inconsistency. Change-type: minor Closes: #590
154 lines
3.4 KiB
TypeScript
154 lines
3.4 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,
|
|
...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] ?? [],
|
|
volumes: volumesByAppId[appId] ?? [],
|
|
},
|
|
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,
|
|
};
|
|
}
|