mirror of
https://github.com/balena-os/balena-supervisor.git
synced 2025-01-12 07:52:55 +00:00
758f3caa48
Change-type: patch
540 lines
12 KiB
TypeScript
540 lines
12 KiB
TypeScript
import { assert, expect } from 'chai';
|
|
import { SinonStub, stub } from 'sinon';
|
|
|
|
import { child_process } from 'mz';
|
|
import * as semver from 'semver';
|
|
|
|
import * as constants from '../src/lib/constants';
|
|
import {
|
|
containerContractsFulfilled,
|
|
intialiseContractRequirements,
|
|
validateContract,
|
|
} from '../src/lib/contracts';
|
|
import * as osRelease from '../src/lib/os-release';
|
|
import supervisorVersion = require('../src/lib/supervisor-version');
|
|
|
|
describe('Container contracts', () => {
|
|
before(() => {
|
|
intialiseContractRequirements({
|
|
supervisorVersion: '11.0.0',
|
|
deviceType: 'intel-nuc',
|
|
l4tVersion: '32.2',
|
|
});
|
|
});
|
|
|
|
describe('Contract validation', () => {
|
|
it('should correctly validate a contract with no requirements', () =>
|
|
expect(() =>
|
|
validateContract({
|
|
slug: 'user-container',
|
|
}),
|
|
).to.be.not.throw());
|
|
|
|
it('should correctly validate a contract with extra fields', () =>
|
|
expect(() =>
|
|
validateContract({
|
|
slug: 'user-container',
|
|
name: 'user-container',
|
|
version: '3.0.0',
|
|
}),
|
|
).to.be.not.throw());
|
|
|
|
it('should not validate a contract without the minimum required fields', () => {
|
|
return Promise.all([
|
|
expect(() => validateContract({})).to.throw(),
|
|
expect(() => validateContract({ name: 'test' })).to.throw(),
|
|
expect(() => validateContract({ requires: [] })).to.throw(),
|
|
]);
|
|
});
|
|
|
|
it('should correctly validate a contract with requirements', () =>
|
|
expect(() =>
|
|
validateContract({
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.l4t',
|
|
version: '32.2',
|
|
},
|
|
{
|
|
type: 'sw.supervisor',
|
|
},
|
|
],
|
|
}),
|
|
).to.not.throw());
|
|
|
|
it('should not validate a contract with requirements without the minimum required fields', () => {
|
|
return expect(() =>
|
|
validateContract({
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
version: '>3.0.0',
|
|
},
|
|
],
|
|
}),
|
|
).to.throw();
|
|
});
|
|
});
|
|
|
|
describe('Requirement resolution', () => {
|
|
// Because the supervisor version will change whenever the
|
|
// package.json will, we generate values which are above
|
|
// and below the current value, and use these to reason
|
|
// about the contract engine results
|
|
const supervisorVersionGreater = `${
|
|
semver.major(supervisorVersion)! + 1
|
|
}.0.0`;
|
|
const supervisorVersionLesser = `${
|
|
semver.major(supervisorVersion)! - 1
|
|
}.0.0`;
|
|
|
|
before(async () => {
|
|
// We ensure that the versions we're using for testing
|
|
// are the same as the time of implementation, otherwise
|
|
// these tests could fail or succeed when they shouldn't
|
|
expect(await osRelease.getOSSemver(constants.hostOSVersionPath)).to.equal(
|
|
'2.0.6',
|
|
);
|
|
assert(semver.gt(supervisorVersionGreater, supervisorVersion));
|
|
assert(semver.lt(supervisorVersionLesser, supervisorVersion));
|
|
});
|
|
|
|
it('Should correctly run containers with no requirements', async () => {
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'user-container',
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'user-container1',
|
|
},
|
|
optional: false,
|
|
},
|
|
service2: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'user-container2',
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
});
|
|
|
|
it('should correctly run containers whose requirements are satisfied', async () => {
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `>${supervisorVersionLesser}`,
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `<${supervisorVersionGreater}`,
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `>${supervisorVersionLesser}`,
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `>${supervisorVersionLesser}`,
|
|
},
|
|
{
|
|
type: 'sw.l4t',
|
|
version: '32.2',
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container1',
|
|
slug: 'user-container1',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `>${supervisorVersionLesser}`,
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
service2: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container1',
|
|
slug: 'user-container1',
|
|
requires: [
|
|
{
|
|
type: 'sw.os',
|
|
version: '<3.0.0',
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
});
|
|
|
|
it('Should refuse to run containers whose requirements are not satisfied', async () => {
|
|
let fulfilled = containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `>=${supervisorVersionGreater}`,
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
});
|
|
expect(fulfilled).to.have.property('valid').that.equals(false);
|
|
expect(fulfilled)
|
|
.to.have.property('unmetServices')
|
|
.that.deep.equals(['service']);
|
|
|
|
fulfilled = containerContractsFulfilled({
|
|
service2: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container2',
|
|
slug: 'user-container2',
|
|
requires: [
|
|
{
|
|
type: 'sw.l4t',
|
|
version: '28.2',
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
});
|
|
expect(fulfilled).to.have.property('valid').that.equals(false);
|
|
expect(fulfilled)
|
|
.to.have.property('unmetServices')
|
|
.that.deep.equals(['service2']);
|
|
|
|
fulfilled = containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container1',
|
|
slug: 'user-container1',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `>=${supervisorVersionLesser}`,
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
service2: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
name: 'user-container2',
|
|
slug: 'user-container2',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `<=${supervisorVersionLesser}`,
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
});
|
|
expect(fulfilled).to.have.property('valid').that.equals(false);
|
|
expect(fulfilled)
|
|
.to.have.property('unmetServices')
|
|
.that.deep.equals(['service2']);
|
|
});
|
|
|
|
describe('Optional containers', () => {
|
|
it('should correctly run passing optional containers', async () => {
|
|
const {
|
|
valid,
|
|
unmetServices,
|
|
fulfilledServices,
|
|
} = containerContractsFulfilled({
|
|
service1: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'service1',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `<${supervisorVersionGreater}`,
|
|
},
|
|
],
|
|
},
|
|
optional: true,
|
|
},
|
|
});
|
|
expect(valid).to.equal(true);
|
|
expect(unmetServices).to.deep.equal([]);
|
|
expect(fulfilledServices).to.deep.equal(['service1']);
|
|
});
|
|
|
|
it('should corrrectly omit failing optional containers', async () => {
|
|
const {
|
|
valid,
|
|
unmetServices,
|
|
fulfilledServices,
|
|
} = containerContractsFulfilled({
|
|
service1: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'service1',
|
|
requires: [
|
|
{
|
|
type: 'sw.supervisor',
|
|
version: `>${supervisorVersionGreater}`,
|
|
},
|
|
],
|
|
},
|
|
optional: true,
|
|
},
|
|
service2: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'service2',
|
|
},
|
|
optional: false,
|
|
},
|
|
});
|
|
expect(valid).to.equal(true);
|
|
expect(unmetServices).to.deep.equal(['service1']);
|
|
expect(fulfilledServices).to.deep.equal(['service2']);
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('L4T version detection', () => {
|
|
let execStub: SinonStub;
|
|
|
|
const seedExec = (version: string) => {
|
|
execStub = stub(child_process, 'exec').returns(
|
|
Promise.resolve([Buffer.from(version), Buffer.from('')]),
|
|
);
|
|
};
|
|
|
|
afterEach(() => {
|
|
execStub.restore();
|
|
});
|
|
|
|
it('should correctly parse L4T version strings', async () => {
|
|
seedExec('4.9.140-l4t-r32.2+g3dcbed5');
|
|
expect(await osRelease.getL4tVersion()).to.equal('32.2.0');
|
|
expect(execStub.callCount).to.equal(1);
|
|
execStub.restore();
|
|
|
|
seedExec('4.4.38-l4t-r28.2+g174510d');
|
|
expect(await osRelease.getL4tVersion()).to.equal('28.2.0');
|
|
expect(execStub.callCount).to.equal(1);
|
|
});
|
|
|
|
it('should correctly handle l4t versions which contain three numbers', async () => {
|
|
seedExec('4.4.38-l4t-r32.3.1+g174510d');
|
|
expect(await osRelease.getL4tVersion()).to.equal('32.3.1');
|
|
expect(execStub.callCount).to.equal(1);
|
|
});
|
|
|
|
it('should return undefined when there is no l4t string in uname', async () => {
|
|
seedExec('4.18.14-yocto-standard');
|
|
expect(await osRelease.getL4tVersion()).to.be.undefined;
|
|
});
|
|
|
|
describe('L4T comparison', () => {
|
|
const seedEngine = async (version: string) => {
|
|
if (execStub != null) {
|
|
execStub.restore();
|
|
}
|
|
seedExec(version);
|
|
intialiseContractRequirements({
|
|
supervisorVersion: '11.0.0',
|
|
deviceType: 'intel-nuc',
|
|
l4tVersion: await osRelease.getL4tVersion(),
|
|
});
|
|
};
|
|
|
|
it('should allow semver matching even when l4t does not fulfill semver', async () => {
|
|
await seedEngine('4.4.38-l4t-r31.0');
|
|
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.l4t',
|
|
version: '>=31.0.0',
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.l4t',
|
|
version: '<31.0.0',
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(false);
|
|
});
|
|
|
|
it('should allow semver matching when l4t does fulfill semver', async () => {
|
|
await seedEngine('4.4.38-l4t-r31.0.1');
|
|
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.l4t',
|
|
version: '>=31.0.0',
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(true);
|
|
|
|
expect(
|
|
containerContractsFulfilled({
|
|
service: {
|
|
contract: {
|
|
type: 'sw.container',
|
|
slug: 'user-container',
|
|
requires: [
|
|
{
|
|
type: 'sw.l4t',
|
|
version: '<31.0.0',
|
|
},
|
|
],
|
|
},
|
|
optional: false,
|
|
},
|
|
}),
|
|
)
|
|
.to.have.property('valid')
|
|
.that.equals(false);
|
|
});
|
|
});
|
|
});
|