2021-05-05 20:32:31 +00:00
|
|
|
import { expect } from 'chai';
|
|
|
|
|
2022-08-17 23:35:08 +00:00
|
|
|
import * as imageManager from '~/src/compose/images';
|
|
|
|
import * as dbHelper from '~/test-lib/db-helper';
|
|
|
|
import { createImage, withMockerode } from '~/test-lib/mockerode';
|
2021-05-05 20:32:31 +00:00
|
|
|
import * as sinon from 'sinon';
|
|
|
|
|
2022-08-17 23:35:08 +00:00
|
|
|
import log from '~/lib/supervisor-console';
|
2021-05-05 20:32:31 +00:00
|
|
|
|
2021-08-03 23:12:47 +00:00
|
|
|
// TODO: this code is duplicated in multiple tests
|
|
|
|
// create a test module with all helper functions like this
|
|
|
|
function createDBImage(
|
|
|
|
{
|
|
|
|
appId = 1,
|
|
|
|
name = 'test-image',
|
|
|
|
serviceName = 'test',
|
|
|
|
dependent = 0,
|
|
|
|
...extra
|
|
|
|
} = {} as Partial<imageManager.Image>,
|
|
|
|
) {
|
|
|
|
return {
|
|
|
|
appId,
|
|
|
|
dependent,
|
|
|
|
name,
|
|
|
|
serviceName,
|
|
|
|
...extra,
|
|
|
|
} as imageManager.Image;
|
|
|
|
}
|
|
|
|
|
2021-05-05 20:32:31 +00:00
|
|
|
describe('compose/images', () => {
|
|
|
|
let testDb: dbHelper.TestDatabase;
|
|
|
|
before(async () => {
|
|
|
|
testDb = await dbHelper.createDB();
|
|
|
|
|
|
|
|
// disable log output during testing
|
|
|
|
sinon.stub(log, 'debug');
|
|
|
|
sinon.stub(log, 'warn');
|
|
|
|
sinon.stub(log, 'info');
|
|
|
|
sinon.stub(log, 'event');
|
|
|
|
sinon.stub(log, 'success');
|
|
|
|
});
|
|
|
|
|
|
|
|
after(async () => {
|
|
|
|
try {
|
|
|
|
await testDb.destroy();
|
|
|
|
} catch (e) {
|
|
|
|
/* noop */
|
|
|
|
}
|
|
|
|
|
|
|
|
// Restore stubbed methods
|
|
|
|
sinon.restore();
|
|
|
|
});
|
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
afterEach(async () => {
|
|
|
|
await testDb.reset();
|
2021-05-05 20:32:31 +00:00
|
|
|
});
|
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
it('finds image by matching digest on the database', async () => {
|
2021-08-03 23:12:47 +00:00
|
|
|
const dbImage = createDBImage({
|
2021-05-05 20:32:31 +00:00
|
|
|
name:
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/aaaaa@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
2021-05-05 20:32:31 +00:00
|
|
|
dockerImageId:
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
2021-08-03 23:12:47 +00:00
|
|
|
});
|
2021-05-05 20:32:31 +00:00
|
|
|
await testDb.models('image').insert([dbImage]);
|
|
|
|
|
|
|
|
const images = [
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id:
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: [
|
2021-07-07 22:01:48 +00:00
|
|
|
// Different image repo
|
|
|
|
'registry2.balena-cloud.com/v2/bbbb@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
2021-05-05 20:32:31 +00:00
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
// Looking by name should fail, if not, this is a mockerode issue
|
|
|
|
await expect(mockerode.getImage(dbImage.name).inspect()).to.be.rejected;
|
|
|
|
|
|
|
|
// Looking up the image by id should succeed
|
2021-08-03 23:12:47 +00:00
|
|
|
await expect(mockerode.getImage(dbImage.dockerImageId!).inspect()).to
|
|
|
|
.not.be.rejected;
|
2021-05-05 20:32:31 +00:00
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
// The image is found
|
|
|
|
expect(await imageManager.inspectByName(dbImage.name))
|
|
|
|
.to.have.property('Id')
|
|
|
|
.that.equals(
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
);
|
|
|
|
|
|
|
|
// It couldn't find the image by name so it finds it by matching digest
|
|
|
|
expect(mockerode.getImage).to.have.been.calledWith(
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
);
|
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('finds image by tag on the engine', async () => {
|
|
|
|
const images = [
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id:
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: ['some-image:some-tag'],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
expect(await imageManager.inspectByName('some-image:some-tag'))
|
|
|
|
.to.have.property('Id')
|
|
|
|
.that.equals(
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(mockerode.getImage).to.have.been.calledWith(
|
|
|
|
'some-image:some-tag',
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check that non existing tags are not found
|
|
|
|
await expect(
|
|
|
|
imageManager.inspectByName('non-existing-image:non-existing-tag'),
|
|
|
|
).to.be.rejected;
|
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('finds image by tag on the database', async () => {
|
2021-08-03 23:12:47 +00:00
|
|
|
const dbImage = createDBImage({
|
2021-07-07 22:01:48 +00:00
|
|
|
name: 'some-image:some-tag',
|
|
|
|
dockerImageId:
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
2021-08-03 23:12:47 +00:00
|
|
|
});
|
2021-07-07 22:01:48 +00:00
|
|
|
await testDb.models('image').insert([dbImage]);
|
|
|
|
|
|
|
|
const images = [
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id:
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: [
|
|
|
|
// Reference is different but there is a matching name on the database
|
|
|
|
'registry2.balena-cloud.com/v2/bbbb@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
expect(await imageManager.inspectByName(dbImage.name))
|
|
|
|
.to.have.property('Id')
|
|
|
|
.that.equals(
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
);
|
2021-05-05 20:32:31 +00:00
|
|
|
|
|
|
|
expect(mockerode.getImage).to.have.been.calledWith(
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
);
|
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
// Check that non existing tags are not found
|
|
|
|
await expect(
|
|
|
|
imageManager.inspectByName('non-existing-image:non-existing-tag'),
|
|
|
|
).to.be.rejected;
|
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('finds image by reference on the engine', async () => {
|
|
|
|
const images = [
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id:
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: [
|
|
|
|
// the reference we expect to look for is registry2.balena-cloud.com/v2/one
|
|
|
|
'registry2.balena-cloud.com/v2/one:delta-one@sha256:12345a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
'registry2.balena-cloud.com/v2/one:latest@sha256:12345a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
// This is really testing mockerode functionality
|
|
|
|
expect(
|
|
|
|
await mockerode.listImages({
|
|
|
|
filters: {
|
|
|
|
reference: ['registry2.balena-cloud.com/v2/one'],
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
).to.have.lengthOf(1);
|
|
|
|
|
|
|
|
expect(
|
|
|
|
await imageManager.inspectByName(
|
|
|
|
// different target digest but same imageName
|
|
|
|
'registry2.balena-cloud.com/v2/one@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.to.have.property('Id')
|
|
|
|
.that.equals(
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(mockerode.getImage).to.have.been.calledWith(
|
|
|
|
'sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc90e8e7',
|
|
|
|
);
|
|
|
|
|
|
|
|
// Looking for the reference with correct name tag shoud not throw
|
|
|
|
await expect(
|
|
|
|
imageManager.inspectByName(
|
|
|
|
// different target digest but same tag
|
|
|
|
'registry2.balena-cloud.com/v2/one:delta-one@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
),
|
|
|
|
).to.not.be.rejected;
|
|
|
|
|
|
|
|
// Looking for a non existing reference should throw
|
|
|
|
await expect(
|
|
|
|
imageManager.inspectByName(
|
|
|
|
'registry2.balena-cloud.com/v2/two@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
),
|
|
|
|
).to.be.rejected;
|
|
|
|
await expect(
|
|
|
|
imageManager.inspectByName(
|
|
|
|
'registry2.balena-cloud.com/v2/one:some-tag@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
),
|
|
|
|
).to.be.rejected;
|
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('returns all images in both the database and the engine', async () => {
|
|
|
|
await testDb.models('image').insert([
|
2021-08-03 23:12:47 +00:00
|
|
|
createDBImage({
|
2021-07-07 22:01:48 +00:00
|
|
|
name: 'first-image-name:first-image-tag',
|
|
|
|
serviceName: 'app_1',
|
|
|
|
dockerImageId: 'sha256:first-image-id',
|
2021-08-03 23:12:47 +00:00
|
|
|
}),
|
|
|
|
createDBImage({
|
2021-07-07 22:01:48 +00:00
|
|
|
name: 'second-image-name:second-image-tag',
|
|
|
|
serviceName: 'app_2',
|
|
|
|
dockerImageId: 'sha256:second-image-id',
|
2021-08-03 23:12:47 +00:00
|
|
|
}),
|
|
|
|
createDBImage({
|
2021-07-07 22:01:48 +00:00
|
|
|
name:
|
|
|
|
'registry2.balena-cloud.com/v2/three@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf558',
|
|
|
|
serviceName: 'app_3',
|
|
|
|
// Third image has different name but same docker id
|
|
|
|
dockerImageId: 'sha256:second-image-id',
|
2021-08-03 23:12:47 +00:00
|
|
|
}),
|
|
|
|
createDBImage({
|
2021-07-07 22:01:48 +00:00
|
|
|
name: 'fourth-image-name:fourth-image-tag',
|
|
|
|
serviceName: 'app_4',
|
|
|
|
// The fourth image exists on the engine but with the wrong id
|
|
|
|
dockerImageId: 'sha256:fourth-image-id',
|
2021-08-03 23:12:47 +00:00
|
|
|
}),
|
2021-07-07 22:01:48 +00:00
|
|
|
]);
|
|
|
|
|
|
|
|
const images = [
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id: 'sha256:first-image-id',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: ['first-image-name:first-image-tag'],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id: 'sha256:second-image-id',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: [
|
|
|
|
// The tag for the second image does not exist on the engine but it should be found
|
|
|
|
'not-second-image-name:some-image-tag',
|
|
|
|
'fourth-image-name:fourth-image-tag',
|
|
|
|
'registry2.balena-cloud.com/v2/three@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf558',
|
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
// Perform the test with our specially crafted data
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
// failsafe to check for mockerode problems
|
|
|
|
expect(
|
|
|
|
await mockerode.listImages(),
|
|
|
|
'images exist on the engine before test',
|
|
|
|
).to.have.lengthOf(2);
|
|
|
|
|
|
|
|
const availableImages = await imageManager.getAvailable();
|
|
|
|
expect(availableImages).to.have.lengthOf(4);
|
2021-05-05 20:32:31 +00:00
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('removes a single legacy db images without dockerImageId', async () => {
|
|
|
|
// Legacy images don't have a dockerImageId so they are queried by name
|
2021-08-03 23:12:47 +00:00
|
|
|
const imageToRemove = createDBImage({
|
2021-07-07 22:01:48 +00:00
|
|
|
name: 'image-name:image-tag',
|
2021-08-03 23:12:47 +00:00
|
|
|
});
|
2021-05-05 20:32:31 +00:00
|
|
|
|
|
|
|
await testDb.models('image').insert([imageToRemove]);
|
|
|
|
|
|
|
|
// Engine image state
|
|
|
|
const images = [
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id: 'deadbeef',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
// Image references
|
|
|
|
References: [
|
2021-07-07 22:01:48 +00:00
|
|
|
'image-name:image-tag@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
2021-05-05 20:32:31 +00:00
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id: 'deadca1f',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: ['balena/aarch64-supervisor:11.11.11'],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
// Perform the test with our specially crafted data
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
// Check that our legacy image exists
|
|
|
|
// failsafe to check for mockerode problems
|
|
|
|
await expect(
|
|
|
|
mockerode.getImage(imageToRemove.name).inspect(),
|
|
|
|
'image exists on the engine before test',
|
|
|
|
).to.not.be.rejected;
|
|
|
|
|
|
|
|
// Check that the image exists on the db
|
|
|
|
expect(
|
|
|
|
await testDb.models('image').select().where(imageToRemove),
|
|
|
|
).to.have.lengthOf(1);
|
|
|
|
|
|
|
|
// Now remove this image...
|
|
|
|
await imageManager.remove(imageToRemove);
|
|
|
|
|
|
|
|
// This checks that the remove method was ultimately called
|
|
|
|
expect(mockerode.removeImage).to.have.been.calledOnceWith(
|
|
|
|
imageToRemove.name,
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check that the image was removed from the db
|
|
|
|
expect(await testDb.models('image').select().where(imageToRemove)).to.be
|
|
|
|
.empty;
|
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
it('removes image from DB and engine when there is a single DB image with matching name', async () => {
|
2021-05-05 20:32:31 +00:00
|
|
|
// Newer image
|
2021-08-03 23:12:47 +00:00
|
|
|
const imageToRemove = createDBImage({
|
2021-05-05 20:32:31 +00:00
|
|
|
name:
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/one@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
dockerImageId: 'sha256:image-id-one',
|
2021-08-03 23:12:47 +00:00
|
|
|
});
|
2021-05-05 20:32:31 +00:00
|
|
|
|
|
|
|
// Insert images into the db
|
|
|
|
await testDb.models('image').insert([
|
|
|
|
imageToRemove,
|
2021-08-03 23:12:47 +00:00
|
|
|
createDBImage({
|
2021-05-05 20:32:31 +00:00
|
|
|
name:
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/two@sha256:12345a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
dockerImageId: 'sha256:image-id-two',
|
2021-08-03 23:12:47 +00:00
|
|
|
}),
|
2021-05-05 20:32:31 +00:00
|
|
|
]);
|
|
|
|
|
|
|
|
// Engine image state
|
|
|
|
const images = [
|
|
|
|
// The image to remove
|
|
|
|
createImage(
|
|
|
|
{
|
2021-07-07 22:01:48 +00:00
|
|
|
Id: 'sha256:image-id-one',
|
2021-05-05 20:32:31 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
References: [
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/one:delta-one@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
2021-05-05 20:32:31 +00:00
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
// Other images to test
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id: 'aaa',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: ['balena/aarch64-supervisor:11.11.11'],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
// The other image on the database
|
|
|
|
createImage(
|
|
|
|
{
|
2021-07-07 22:01:48 +00:00
|
|
|
Id: 'sha256:image-id-two',
|
2021-05-05 20:32:31 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
References: [
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/two:delta-two@sha256:12345a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
2021-05-05 20:32:31 +00:00
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
// Perform the test with our specially crafted data
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
// Check that the image exists
|
|
|
|
// this is really checking that mockerode works ok
|
|
|
|
await expect(
|
|
|
|
mockerode.getImage(imageToRemove.name).inspect(),
|
|
|
|
'image exists on the engine before test',
|
|
|
|
).to.not.be.rejected;
|
|
|
|
|
|
|
|
// Check that only one image with this dockerImageId exists in the db
|
|
|
|
// in memory db is a bit flaky sometimes, this checks for issues
|
|
|
|
expect(
|
|
|
|
await testDb.models('image').where(imageToRemove).select(),
|
|
|
|
'image exists on db before the test',
|
|
|
|
).to.have.lengthOf(1);
|
|
|
|
|
|
|
|
// Now remove this image...
|
|
|
|
await imageManager.remove(imageToRemove);
|
|
|
|
|
|
|
|
// Check that the remove method was only called once
|
|
|
|
expect(mockerode.removeImage).to.have.been.calledOnceWith(
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/one:delta-one',
|
2021-05-05 20:32:31 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
// Check that the database no longer has this image
|
|
|
|
expect(await testDb.models('image').select().where(imageToRemove)).to.be
|
|
|
|
.empty;
|
|
|
|
|
|
|
|
// Expect 1 entry left on the database
|
|
|
|
expect(await testDb.models('image').select()).to.have.lengthOf(1);
|
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
it('removes the requested image even when there are multiple DB images with same docker ID', async () => {
|
2021-08-03 23:12:47 +00:00
|
|
|
const imageToRemove = createDBImage({
|
2021-05-05 20:32:31 +00:00
|
|
|
name:
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/one@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
dockerImageId: 'sha256:image-id-one',
|
2021-08-03 23:12:47 +00:00
|
|
|
});
|
2021-05-05 20:32:31 +00:00
|
|
|
|
2021-08-03 23:12:47 +00:00
|
|
|
const imageWithSameDockerImageId = createDBImage({
|
2021-05-05 20:32:31 +00:00
|
|
|
name:
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/two@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
2021-05-05 20:32:31 +00:00
|
|
|
// Same imageId
|
2021-07-07 22:01:48 +00:00
|
|
|
dockerImageId: 'sha256:image-id-one',
|
2021-08-03 23:12:47 +00:00
|
|
|
});
|
2021-05-05 20:32:31 +00:00
|
|
|
|
|
|
|
// Insert images into the db
|
|
|
|
await testDb.models('image').insert([
|
|
|
|
imageToRemove,
|
|
|
|
// Another image from the same app
|
|
|
|
imageWithSameDockerImageId,
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Engine image state
|
|
|
|
const images = [
|
|
|
|
// The image to remove
|
|
|
|
createImage(
|
|
|
|
{
|
2021-08-03 23:12:47 +00:00
|
|
|
Id: imageToRemove.dockerImageId!,
|
2021-05-05 20:32:31 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
References: [imageToRemove.name, imageWithSameDockerImageId.name],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
// Other images to test
|
|
|
|
createImage(
|
|
|
|
{
|
|
|
|
Id: 'aaa',
|
|
|
|
},
|
|
|
|
{
|
|
|
|
References: ['balena/aarch64-supervisor:11.11.11'],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
// Perform the test with our specially crafted data
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
// Check that the image is on the engine
|
|
|
|
// really checking mockerode behavior
|
|
|
|
await expect(
|
2021-08-03 23:12:47 +00:00
|
|
|
mockerode.getImage(imageToRemove.dockerImageId!).inspect(),
|
2021-05-05 20:32:31 +00:00
|
|
|
'image exists on the engine before the test',
|
|
|
|
).to.not.be.rejected;
|
|
|
|
|
|
|
|
// Check that multiple images with the same dockerImageId are returned
|
|
|
|
expect(
|
|
|
|
await testDb
|
|
|
|
.models('image')
|
|
|
|
.where({ dockerImageId: imageToRemove.dockerImageId })
|
|
|
|
.select(),
|
|
|
|
).to.have.lengthOf(2);
|
|
|
|
|
|
|
|
// Now remove these images
|
|
|
|
await imageManager.remove(imageToRemove);
|
|
|
|
|
|
|
|
// Check that only the image with the right name was removed
|
|
|
|
expect(mockerode.removeImage).to.have.been.calledOnceWith(
|
|
|
|
imageToRemove.name,
|
|
|
|
);
|
|
|
|
|
|
|
|
// Check that the database no longer has this image
|
|
|
|
expect(await testDb.models('image').select().where(imageToRemove)).to.be
|
|
|
|
.empty;
|
|
|
|
|
|
|
|
// Check that the image with the same dockerImageId is still on the database
|
|
|
|
expect(
|
|
|
|
await testDb
|
|
|
|
.models('image')
|
|
|
|
.select()
|
|
|
|
.where({ dockerImageId: imageWithSameDockerImageId.dockerImageId }),
|
|
|
|
).to.have.lengthOf(1);
|
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
it('removes image from DB by tag when deltas are being used', async () => {
|
2021-08-03 23:12:47 +00:00
|
|
|
const imageToRemove = createDBImage({
|
2021-05-05 20:32:31 +00:00
|
|
|
name:
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/one@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
dockerImageId: 'sha256:image-one-id',
|
2021-08-03 23:12:47 +00:00
|
|
|
});
|
2021-05-05 20:32:31 +00:00
|
|
|
|
2021-08-03 23:12:47 +00:00
|
|
|
const imageWithSameDockerImageId = createDBImage({
|
2021-05-05 20:32:31 +00:00
|
|
|
name:
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/two@sha256:2c969a1ba1c6bc10df53481f48c6a74dbd562cfb41ba58f81beabd03facf5582',
|
|
|
|
// Same docker id
|
|
|
|
dockerImageId: 'sha256:image-one-id',
|
2021-08-03 23:12:47 +00:00
|
|
|
});
|
2021-05-05 20:32:31 +00:00
|
|
|
|
|
|
|
// Insert images into the db
|
|
|
|
await testDb.models('image').insert([
|
|
|
|
imageToRemove,
|
|
|
|
// Another image from the same app
|
|
|
|
imageWithSameDockerImageId,
|
|
|
|
]);
|
|
|
|
|
|
|
|
// Engine image state
|
|
|
|
const images = [
|
|
|
|
// The image to remove
|
|
|
|
createImage(
|
|
|
|
{
|
2021-08-03 23:12:47 +00:00
|
|
|
Id: imageToRemove.dockerImageId!,
|
2021-05-05 20:32:31 +00:00
|
|
|
},
|
|
|
|
{
|
|
|
|
References: [
|
|
|
|
// The image has two deltas with different digests than those in image.name
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/one:delta-one@sha256:6eb712fc797ff68f258d9032cf292c266cb9bd8be4cbdaaafeb5a8824bb104fd',
|
|
|
|
'registry2.balena-cloud.com/v2/two:delta-two@sha256:f1154d76c731f04711e5856b6e6858730e3023d9113124900ac65c2ccc901234',
|
2021-05-05 20:32:31 +00:00
|
|
|
],
|
|
|
|
},
|
|
|
|
),
|
|
|
|
];
|
|
|
|
|
|
|
|
// Perform the test with our specially crafted data
|
|
|
|
await withMockerode(
|
|
|
|
async (mockerode) => {
|
|
|
|
// Check that the image is on the engine
|
|
|
|
await expect(
|
2021-08-03 23:12:47 +00:00
|
|
|
mockerode.getImage(imageToRemove.dockerImageId!).inspect(),
|
2021-05-05 20:32:31 +00:00
|
|
|
'image can be found by id before the test',
|
|
|
|
).to.not.be.rejected;
|
|
|
|
|
|
|
|
// Check that a single image is returned when given entire object
|
|
|
|
expect(
|
|
|
|
await testDb.models('image').select().where(imageToRemove),
|
|
|
|
).to.have.lengthOf(1);
|
|
|
|
|
|
|
|
// Check that multiple images with the same dockerImageId are returned
|
|
|
|
expect(
|
|
|
|
await testDb
|
|
|
|
.models('image')
|
|
|
|
.where({ dockerImageId: imageToRemove.dockerImageId })
|
|
|
|
.select(),
|
|
|
|
).to.have.lengthOf(2);
|
|
|
|
|
|
|
|
// Now remove these images
|
|
|
|
await imageManager.remove(imageToRemove);
|
|
|
|
|
|
|
|
// This tests the behavior
|
|
|
|
expect(mockerode.removeImage).to.have.been.calledOnceWith(
|
2021-07-07 22:01:48 +00:00
|
|
|
'registry2.balena-cloud.com/v2/one:delta-one',
|
2021-05-05 20:32:31 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
// Check that the database no longer has this image
|
|
|
|
expect(await testDb.models('image').select().where(imageToRemove)).to.be
|
|
|
|
.empty;
|
|
|
|
|
|
|
|
// Check that the image with the same dockerImageId is still on the database
|
|
|
|
expect(
|
|
|
|
await testDb
|
|
|
|
.models('image')
|
|
|
|
.select()
|
|
|
|
.where(imageWithSameDockerImageId),
|
|
|
|
).to.have.lengthOf(1);
|
|
|
|
},
|
|
|
|
{ images },
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|