Skip to content

Commit

Permalink
Add support for io.balena.update.requires-reboot
Browse files Browse the repository at this point in the history
Services can now indicate via a label that they require a reboot
before starting to ensure the update the application works properly.

Change-type: minor
  • Loading branch information
pipex committed Jan 13, 2025
1 parent 75127c6 commit e8ce10d
Show file tree
Hide file tree
Showing 12 changed files with 238 additions and 27 deletions.
52 changes: 41 additions & 11 deletions src/compose/app.ts
Original file line number Diff line number Diff line change
Expand Up @@ -654,6 +654,7 @@ class AppImpl implements App {
context.targetApp,
needsDownload,
servicesLocked,
context.rebootBreadcrumbSet,
context.appsToLock,
context.availableImages,
context.networkPairs,
Expand Down Expand Up @@ -682,6 +683,8 @@ class AppImpl implements App {
context.appsToLock,
context.targetApp.services,
servicesLocked,
context.rebootBreadcrumbSet,
context.bootTime,
);
}

Expand Down Expand Up @@ -761,6 +764,8 @@ class AppImpl implements App {
appsToLock: AppsToLockMap,
targetServices: Service[],
servicesLocked: boolean,
rebootBreadcrumbSet: boolean,
bootTime: Date,
): CompositionStep[] {
// Update container metadata if service release has changed
if (current.commit !== target.commit) {
Expand All @@ -774,16 +779,38 @@ class AppImpl implements App {
return [];
}
} else if (target.config.running !== current.config.running) {
// Take lock for all services before starting/stopping container
if (!servicesLocked) {
this.services.concat(targetServices).forEach((s) => {
appsToLock[target.appId].add(s.serviceName);
});
return [];
}
if (target.config.running) {
// if the container has a reboot
// required label and the boot time is before the creation time, then
// return a 'noop' to ensure a reboot happens before starting the container
const requiresReboot =
checkTruthy(
target.config.labels?.['io.balena.update.requires-reboot'],
) &&
current.createdAt != null &&
current.createdAt > bootTime;

if (requiresReboot && rebootBreadcrumbSet) {
// Do not return a noop to allow locks to be released by the
// app module
return [];
} else if (requiresReboot) {
return [
generateStep('requireReboot', {
serviceName: target.serviceName,
}),
];
}

return [generateStep('start', { target })];
} else {
// Take lock for all services before stopping container
if (!servicesLocked) {
this.services.concat(targetServices).forEach((s) => {
appsToLock[target.appId].add(s.serviceName);
});
return [];
}
return [generateStep('stop', { current })];
}
} else {
Expand All @@ -796,6 +823,7 @@ class AppImpl implements App {
targetApp: App,
needsDownload: boolean,
servicesLocked: boolean,
rebootBreadcrumbSet: boolean,
appsToLock: AppsToLockMap,
availableImages: UpdateState['availableImages'],
networkPairs: Array<ChangingPair<Network>>,
Expand Down Expand Up @@ -832,8 +860,10 @@ class AppImpl implements App {
}
return [generateStep('start', { target })];
} else {
// Wait for dependencies to be started
return [generateStep('noop', {})];
// Wait for dependencies to be started unless there is a
// reboot breadcrumb set, in which case we need to allow the state
// to settle for the reboot to happen
return rebootBreadcrumbSet ? [] : [generateStep('noop', {})];
}
} else {
return [];
Expand Down Expand Up @@ -897,11 +927,11 @@ class AppImpl implements App {
return false;
}

const depedencyUnmet = _.some(target.dependsOn, (dep) =>
const dependencyUnmet = _.some(target.dependsOn, (dep) =>
_.some(servicePairs, (pair) => pair.target?.serviceName === dep),
);

if (depedencyUnmet) {
if (dependencyUnmet) {
return false;
}

Expand Down
13 changes: 13 additions & 0 deletions src/compose/application-manager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,8 @@ import type {
Image,
InstancedAppState,
} from './types';
import { isRebootBreadcrumbSet } from '../lib/reboot';
import { getBootTime } from '../lib/fs-utils';

type ApplicationManagerEventEmitter = StrictEventEmitter<
EventEmitter,
Expand Down Expand Up @@ -127,6 +129,7 @@ export async function getRequiredSteps(
config.getMany(['localMode', 'delta']),
]);
const containerIdsByAppId = getAppContainerIds(currentApps);
const rebootBreadcrumbSet = await isRebootBreadcrumbSet();

// Local mode sets the image and volume retention only
// if not explicitely set by the caller
Expand All @@ -149,6 +152,7 @@ export async function getRequiredSteps(
availableImages,
containerIdsByAppId,
appLocks: lockRegistry,
rebootBreadcrumbSet,
});
}

Expand All @@ -161,6 +165,7 @@ interface InferNextOpts {
availableImages: UpdateState['availableImages'];
containerIdsByAppId: { [appId: number]: UpdateState['containerIds'] };
appLocks: LockRegistry;
rebootBreadcrumbSet: boolean;
}

// Calculate the required steps from the current to the target state
Expand All @@ -176,6 +181,7 @@ export async function inferNextSteps(
availableImages = [],
containerIdsByAppId = {},
appLocks = {},
rebootBreadcrumbSet = false,
}: Partial<InferNextOpts>,
) {
const currentAppIds = Object.keys(currentApps).map((i) => parseInt(i, 10));
Expand All @@ -184,6 +190,7 @@ export async function inferNextSteps(
const withLeftoverLocks = await Promise.all(
currentAppIds.map((id) => hasLeftoverLocks(id)),
);
const bootTime = getBootTime();

let steps: CompositionStep[] = [];

Expand Down Expand Up @@ -245,6 +252,8 @@ export async function inferNextSteps(
force,
lock: appLocks[id],
hasLeftoverLocks: withLeftoverLocks[id],
rebootBreadcrumbSet,
bootTime,
},
targetApps[id],
),
Expand All @@ -261,6 +270,8 @@ export async function inferNextSteps(
force,
lock: appLocks[id],
hasLeftoverLocks: withLeftoverLocks[id],
rebootBreadcrumbSet,
bootTime,
}),
);
}
Expand All @@ -287,6 +298,8 @@ export async function inferNextSteps(
force,
lock: appLocks[id],
hasLeftoverLocks: false,
rebootBreadcrumbSet,
bootTime,
},
targetApps[id],
),
Expand Down
4 changes: 4 additions & 0 deletions src/compose/composition-steps.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import * as networkManager from './network-manager';
import * as volumeManager from './volume-manager';
import * as commitStore from './commit';
import { Lockable, cleanLocksForApp } from '../lib/update-lock';
import { setRebootBreadcrumb } from '../lib/reboot';
import type { DeviceLegacyReport } from '../types/state';
import type { CompositionStepAction, CompositionStepT } from './types';
import type { Lock } from '../lib/update-lock';
Expand Down Expand Up @@ -157,6 +158,9 @@ export function getExecutors(app: { callbacks: CompositionCallbacks }) {
// Clean up any remaining locks
await cleanLocksForApp(step.appId);
},
requireReboot: async (step) => {
await setRebootBreadcrumb({ serviceName: step.serviceName });
},
};

return executors;
Expand Down
31 changes: 23 additions & 8 deletions src/compose/service-manager.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,14 +19,15 @@ import {
isStatusError,
} from '../lib/errors';
import * as LogTypes from '../lib/log-types';
import { checkInt, isValidDeviceName } from '../lib/validation';
import { checkInt, isValidDeviceName, checkTruthy } from '../lib/validation';
import { Service } from './service';
import type { ServiceStatus } from './types';
import { serviceNetworksToDockerNetworks } from './utils';

import log from '../lib/supervisor-console';
import logMonitor from '../logging/monitor';
import { setTimeout } from 'timers/promises';
import { getBootTime } from '../lib/fs-utils';

interface ServiceManagerEvents {
change: void;
Expand Down Expand Up @@ -233,7 +234,7 @@ export async function remove(service: Service) {
}
}

async function create(service: Service) {
async function create(service: Service): Promise<Service> {
const mockContainerId = config.newUniqueKey();
try {
const existing = await get(service);
Expand All @@ -242,7 +243,7 @@ async function create(service: Service) {
`No containerId provided for service ${service.serviceName} in ServiceManager.updateMetadata. Service: ${service}`,
);
}
return docker.getContainer(existing.containerId);
return existing;
} catch (e: unknown) {
if (!isNotFoundError(e)) {
logger.logSystemEvent(LogTypes.installServiceError, {
Expand Down Expand Up @@ -287,7 +288,9 @@ async function create(service: Service) {
reportNewStatus(mockContainerId, service, 'Installing');

const container = await docker.createContainer(conf);
service.containerId = container.id;
const inspectInfo = await container.inspect();

service = Service.fromDockerContainer(inspectInfo);

await Promise.all(
_.map((nets || {}).EndpointsConfig, (endpointConfig, name) =>
Expand All @@ -299,7 +302,7 @@ async function create(service: Service) {
);

logger.logSystemEvent(LogTypes.installServiceSuccess, { service });
return container;
return service;
} finally {
reportChange(mockContainerId);
}
Expand All @@ -310,13 +313,25 @@ export async function start(service: Service) {
let containerId: string | null = null;

try {
const container = await create(service);
const svc = await create(service);
const container = docker.getContainer(svc.containerId!);

const requiresReboot =
checkTruthy(
service.config.labels?.['io.balena.update.requires-reboot'],
) &&
svc.createdAt != null &&
svc.createdAt > getBootTime();

if (requiresReboot) {
log.warn(`Skipping start of service ${svc.serviceName} until reboot`);
}

// Exit here if the target state of the service
// is set to running: false
// is set to running: false or we are waiting for a reboot
// QUESTION: should we split the service steps into
// 'install' and 'start' instead of doing this?
if (service.config.running === false) {
if (service.config.running === false || requiresReboot) {
return container;
}

Expand Down
1 change: 0 additions & 1 deletion src/compose/service.ts
Original file line number Diff line number Diff line change
Expand Up @@ -128,7 +128,6 @@ class ServiceImpl implements Service {
service.releaseId = parseInt(appConfig.releaseId, 10);
service.serviceId = parseInt(appConfig.serviceId, 10);
service.imageName = appConfig.image;
service.createdAt = appConfig.createdAt;
service.commit = appConfig.commit;
service.appUuid = appConfig.appUuid;

Expand Down
2 changes: 2 additions & 0 deletions src/compose/types/app.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@ export interface UpdateState {
hasLeftoverLocks: boolean;
lock: Lock | null;
force: boolean;
rebootBreadcrumbSet: boolean;
bootTime: Date;
}

export interface App {
Expand Down
1 change: 1 addition & 0 deletions src/compose/types/composition-step.ts
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ export interface CompositionStepArgs {
appId: string | number;
lock: Lock | null;
};
requireReboot: { serviceName: string };
}

export type CompositionStepAction = keyof CompositionStepArgs;
Expand Down
20 changes: 17 additions & 3 deletions src/device-state/device-config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -104,7 +104,13 @@ const actionExecutors: DeviceActionExecutors = {
await setBootConfig(backend, step.target as Dictionary<string>);
}
},
setRebootBreadcrumb,
setRebootBreadcrumb: async (step) => {
const changes =
step != null && step.target != null && typeof step.target === 'object'
? step.target
: {};
return setRebootBreadcrumb(changes);
},
};

const configBackends: ConfigBackend[] = [];
Expand Down Expand Up @@ -396,6 +402,7 @@ function getConfigSteps(
target: Dictionary<string>,
) {
const configChanges: Dictionary<string> = {};
const rebootingChanges: Dictionary<string> = {};
const humanReadableConfigChanges: Dictionary<string> = {};
let reboot = false;
const steps: ConfigStep[] = [];
Expand Down Expand Up @@ -431,6 +438,9 @@ function getConfigSteps(
}
if (changingValue != null) {
configChanges[key] = changingValue;
if ($rebootRequired) {
rebootingChanges[key] = changingValue;
}
humanReadableConfigChanges[envVarName] = changingValue;
reboot = $rebootRequired || reboot;
}
Expand All @@ -440,7 +450,7 @@ function getConfigSteps(

if (!_.isEmpty(configChanges)) {
if (reboot) {
steps.push({ action: 'setRebootBreadcrumb' });
steps.push({ action: 'setRebootBreadcrumb', target: rebootingChanges });
}

steps.push({
Expand Down Expand Up @@ -527,7 +537,11 @@ async function getBackendSteps(
return [
// All backend steps require a reboot except fan control
...(steps.length > 0 && rebootRequired
? [{ action: 'setRebootBreadcrumb' } as ConfigStep]
? [
{
action: 'setRebootBreadcrumb',
} as ConfigStep,
]
: []),
...steps,
];
Expand Down
2 changes: 1 addition & 1 deletion src/device-state/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -657,7 +657,7 @@ export const applyTarget = async ({
// For application manager, the reboot breadcrumb should
// be set after all downloads are ready and target containers
// have been installed
if (_.every(steps, ({ action }) => action === 'noop') && rebootRequired) {
if (steps.every(({ action }) => action === 'noop') && rebootRequired) {
steps.push({
action: 'reboot',
});
Expand Down
3 changes: 1 addition & 2 deletions src/lib/fs-utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -87,5 +87,4 @@ export const touch = (file: string, time = new Date()) =>
);

// Get the system boot time as a Date object
export const getBootTime = () =>
new Date(new Date().getTime() - uptime() * 1000);
export const getBootTime = () => new Date(Date.now() - uptime() * 1000);
Loading

0 comments on commit e8ce10d

Please sign in to comment.