2020-11-18 18:24:18 +00:00
|
|
|
import { expect } from 'chai';
|
2022-10-26 22:12:35 +00:00
|
|
|
import { stub, spy, SinonStub, SinonSpy } from 'sinon';
|
2020-11-18 18:24:18 +00:00
|
|
|
import * as supertest from 'supertest';
|
2021-02-03 10:04:37 +00:00
|
|
|
import * as path from 'path';
|
|
|
|
import { promises as fs } from 'fs';
|
2020-11-18 18:24:18 +00:00
|
|
|
|
2022-08-17 23:35:08 +00:00
|
|
|
import { exists, unlinkAll } from '~/lib/fs-utils';
|
|
|
|
import * as appMock from '~/test-lib/application-state-mock';
|
|
|
|
import * as mockedDockerode from '~/test-lib/mocked-dockerode';
|
|
|
|
import mockedAPI = require('~/test-lib/mocked-device-api');
|
|
|
|
import sampleResponses = require('~/test-data/device-api-responses.json');
|
|
|
|
import * as config from '~/src/config';
|
|
|
|
import * as logger from '~/src/logger';
|
2022-09-16 01:51:17 +00:00
|
|
|
import SupervisorAPI from '~/src/device-api';
|
2022-10-18 02:07:40 +00:00
|
|
|
import * as deviceApi from '~/src/device-api';
|
2022-08-17 23:35:08 +00:00
|
|
|
import * as apiBinder from '~/src/api-binder';
|
|
|
|
import * as deviceState from '~/src/device-state';
|
|
|
|
import * as dbus from '~/lib/dbus';
|
|
|
|
import * as updateLock from '~/lib/update-lock';
|
|
|
|
import * as TargetState from '~/src/device-state/target-state';
|
|
|
|
import * as targetStateCache from '~/src/device-state/target-state-cache';
|
|
|
|
import constants = require('~/lib/constants');
|
|
|
|
import { UpdatesLockedError } from '~/lib/errors';
|
|
|
|
import { SchemaTypeKey } from '~/src/config/schema-type';
|
|
|
|
import log from '~/lib/supervisor-console';
|
|
|
|
import * as applicationManager from '~/src/compose/application-manager';
|
|
|
|
import App from '~/src/compose/app';
|
2020-11-18 18:24:18 +00:00
|
|
|
|
|
|
|
describe('SupervisorAPI [V1 Endpoints]', () => {
|
|
|
|
let api: SupervisorAPI;
|
2020-12-03 21:19:44 +00:00
|
|
|
let targetStateCacheMock: SinonStub;
|
2020-11-19 01:37:34 +00:00
|
|
|
const request = supertest(
|
|
|
|
`http://127.0.0.1:${mockedAPI.mockedOptions.listenPort}`,
|
|
|
|
);
|
2020-12-03 21:19:44 +00:00
|
|
|
const services = [
|
2021-09-10 20:35:56 +00:00
|
|
|
{ appId: 2, appUuid: 'deadbeef', serviceId: 640681, serviceName: 'one' },
|
|
|
|
{ appId: 2, appUuid: 'deadbeef', serviceId: 640682, serviceName: 'two' },
|
|
|
|
{ appId: 2, appUuid: 'deadbeef', serviceId: 640683, serviceName: 'three' },
|
2020-11-26 06:11:47 +00:00
|
|
|
];
|
2020-12-03 21:19:44 +00:00
|
|
|
const containers = services.map((service) => mockedAPI.mockService(service));
|
|
|
|
const images = services.map((service) => mockedAPI.mockImage(service));
|
|
|
|
|
|
|
|
let loggerStub: SinonStub;
|
2020-11-26 06:11:47 +00:00
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
// Mock a 3 container release
|
|
|
|
appMock.mockManagers(containers, [], []);
|
|
|
|
appMock.mockImages([], false, images);
|
|
|
|
appMock.mockSupervisorNetwork(true);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
|
|
|
targetStateCacheMock.resolves({
|
|
|
|
appId: 2,
|
2021-09-10 20:35:56 +00:00
|
|
|
appUuid: 'deadbeef',
|
2020-12-03 21:19:44 +00:00
|
|
|
commit: 'abcdef2',
|
|
|
|
name: 'test-app2',
|
|
|
|
source: 'https://api.balena-cloud.com',
|
|
|
|
releaseId: 1232,
|
|
|
|
services: JSON.stringify(services),
|
2021-03-03 20:30:14 +00:00
|
|
|
networks: '[]',
|
|
|
|
volumes: '[]',
|
2020-12-03 21:19:44 +00:00
|
|
|
});
|
2020-11-26 06:11:47 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
// Clear Dockerode actions recorded for each test
|
|
|
|
mockedDockerode.resetHistory();
|
2021-02-12 21:16:37 +00:00
|
|
|
appMock.unmockAll();
|
2020-11-26 06:11:47 +00:00
|
|
|
});
|
2020-11-18 18:24:18 +00:00
|
|
|
|
|
|
|
before(async () => {
|
2022-09-06 18:03:23 +00:00
|
|
|
await apiBinder.initialized();
|
|
|
|
await deviceState.initialized();
|
|
|
|
await targetStateCache.initialized();
|
2020-11-18 18:24:18 +00:00
|
|
|
|
2022-01-26 21:56:08 +00:00
|
|
|
// Do not apply target state
|
|
|
|
stub(deviceState, 'applyStep').resolves();
|
|
|
|
|
2020-11-18 18:24:18 +00:00
|
|
|
// The mockedAPI contains stubs that might create unexpected results
|
|
|
|
// See the module to know what has been stubbed
|
2022-10-26 21:59:10 +00:00
|
|
|
api = await mockedAPI.create([]);
|
2020-11-18 18:24:18 +00:00
|
|
|
|
|
|
|
// Start test API
|
2020-11-19 01:37:34 +00:00
|
|
|
await api.listen(
|
|
|
|
mockedAPI.mockedOptions.listenPort,
|
|
|
|
mockedAPI.mockedOptions.timeout,
|
|
|
|
);
|
2020-11-18 18:24:18 +00:00
|
|
|
|
2020-12-03 21:19:44 +00:00
|
|
|
// Mock target state cache
|
|
|
|
targetStateCacheMock = stub(targetStateCache, 'getTargetApp');
|
|
|
|
|
|
|
|
// Stub logs for all API methods
|
|
|
|
loggerStub = stub(logger, 'attach');
|
|
|
|
loggerStub.resolves();
|
2020-11-18 18:24:18 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
after(async () => {
|
|
|
|
try {
|
|
|
|
await api.stop();
|
2022-09-19 15:08:16 +00:00
|
|
|
} catch (e: any) {
|
2020-11-18 18:24:18 +00:00
|
|
|
if (e.message !== 'Server is not running.') {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
2022-01-26 21:56:08 +00:00
|
|
|
(deviceState.applyStep as SinonStub).restore();
|
2020-11-18 18:24:18 +00:00
|
|
|
// Remove any test data generated
|
|
|
|
await mockedAPI.cleanUp();
|
2020-12-03 21:19:44 +00:00
|
|
|
targetStateCacheMock.restore();
|
|
|
|
loggerStub.restore();
|
2020-11-19 01:37:34 +00:00
|
|
|
});
|
|
|
|
|
2020-11-18 18:24:18 +00:00
|
|
|
describe('GET /v1/apps/:appId', () => {
|
2020-11-26 06:11:47 +00:00
|
|
|
it('does not return information for an application when there is more than 1 container', async () => {
|
|
|
|
await request
|
|
|
|
.get('/v1/apps/2')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2020-11-26 06:11:47 +00:00
|
|
|
.expect(
|
|
|
|
sampleResponses.V1.GET['/apps/2 [Multiple containers running]']
|
|
|
|
.statusCode,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2020-11-18 18:24:18 +00:00
|
|
|
it('returns information about a specific application', async () => {
|
2020-11-26 06:11:47 +00:00
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
releaseId: 77777,
|
|
|
|
});
|
|
|
|
const image = mockedAPI.mockImage({
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
// Make request
|
|
|
|
await request
|
|
|
|
.get('/v1/apps/2')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(sampleResponses.V1.GET['/apps/2'].statusCode)
|
|
|
|
.expect('Content-Type', /json/)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V1.GET['/apps/2'].body,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
);
|
2020-11-18 18:24:18 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
// TODO: setup for this test is wrong, which leads to inconsistent data being passed to
|
|
|
|
// manager methods. A refactor is needed
|
|
|
|
describe.skip('POST /v1/apps/:appId/stop', () => {
|
2020-11-26 06:11:47 +00:00
|
|
|
it('does not allow stopping an application when there is more than 1 container', async () => {
|
|
|
|
// Every test case in this suite has a 3 service release mocked so just make the request
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData({ containers, images }, async () => {
|
|
|
|
await request
|
|
|
|
.post('/v1/apps/2/stop')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(
|
|
|
|
sampleResponses.V1.GET['/apps/2/stop [Multiple containers running]']
|
|
|
|
.statusCode,
|
|
|
|
);
|
|
|
|
});
|
2020-11-26 06:11:47 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('stops a SPECIFIC application and returns a containerId', async () => {
|
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
const image = mockedAPI.mockImage({
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
|
|
|
// Perform the test with our mocked release
|
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
await request
|
|
|
|
.post('/v1/apps/2/stop')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2020-11-26 06:11:47 +00:00
|
|
|
.expect(sampleResponses.V1.GET['/apps/2/stop'].statusCode)
|
|
|
|
.expect('Content-Type', /json/)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V1.GET['/apps/2/stop'].body,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
);
|
2020-11-18 18:24:18 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-07-07 22:01:48 +00:00
|
|
|
// TODO: setup for this test is wrong, which leads to inconsistent data being passed to
|
|
|
|
// manager methods. A refactor is needed
|
|
|
|
describe.skip('POST /v1/apps/:appId/start', () => {
|
2020-12-03 21:19:44 +00:00
|
|
|
it('does not allow starting an application when there is more than 1 container', async () => {
|
|
|
|
// Every test case in this suite has a 3 service release mocked so just make the request
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData({ containers, images }, async () => {
|
|
|
|
await request
|
|
|
|
.post('/v1/apps/2/start')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(400);
|
|
|
|
});
|
2020-12-03 21:19:44 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('starts a SPECIFIC application and returns a containerId', async () => {
|
|
|
|
const service = {
|
|
|
|
serviceName: 'main',
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
serviceId: 640681,
|
|
|
|
};
|
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService(service);
|
|
|
|
const image = mockedAPI.mockImage(service);
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
|
|
|
|
|
|
|
// Target state returns single service
|
|
|
|
targetStateCacheMock.resolves({
|
|
|
|
appId: 2,
|
|
|
|
commit: 'abcdef2',
|
|
|
|
name: 'test-app2',
|
|
|
|
source: 'https://api.balena-cloud.com',
|
|
|
|
releaseId: 1232,
|
|
|
|
services: JSON.stringify([service]),
|
2021-03-03 20:30:14 +00:00
|
|
|
volumes: '[]',
|
|
|
|
networks: '[]',
|
2020-12-03 21:19:44 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// Perform the test with our mocked release
|
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
await request
|
|
|
|
.post('/v1/apps/2/start')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2020-12-03 21:19:44 +00:00
|
|
|
.expect(200)
|
|
|
|
.expect('Content-Type', /json/)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal({ containerId: 'abc123' });
|
|
|
|
});
|
|
|
|
},
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-11-18 18:24:18 +00:00
|
|
|
describe('GET /v1/device', () => {
|
|
|
|
it('returns MAC address', async () => {
|
|
|
|
const response = await request
|
|
|
|
.get('/v1/device')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2020-11-18 18:24:18 +00:00
|
|
|
.expect(200);
|
|
|
|
|
|
|
|
expect(response.body).to.have.property('mac_address').that.is.not.empty;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-12-03 21:19:44 +00:00
|
|
|
describe('POST /v1/reboot', () => {
|
|
|
|
let rebootMock: SinonStub;
|
|
|
|
before(() => {
|
|
|
|
rebootMock = stub(dbus, 'reboot').resolves((() => void 0) as any);
|
|
|
|
});
|
|
|
|
|
|
|
|
after(() => {
|
|
|
|
rebootMock.restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
rebootMock.resetHistory();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return 202 and reboot if no locks are set', async () => {
|
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
releaseId: 77777,
|
|
|
|
});
|
|
|
|
const image = mockedAPI.mockImage({
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
const response = await request
|
|
|
|
.post('/v1/reboot')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(202);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
expect(response.body).to.have.property('Data').that.is.not.empty;
|
|
|
|
expect(rebootMock).to.have.been.calledOnce;
|
|
|
|
},
|
|
|
|
);
|
2020-12-03 21:19:44 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should return 423 and reject the reboot if no locks are set', async () => {
|
2022-11-11 16:11:47 +00:00
|
|
|
stub(updateLock, 'lock').callsFake(async (__, opts, fn) => {
|
2020-12-03 21:19:44 +00:00
|
|
|
if (opts.force) {
|
2022-11-11 16:11:47 +00:00
|
|
|
return fn();
|
2020-12-03 21:19:44 +00:00
|
|
|
}
|
|
|
|
throw new UpdatesLockedError('Updates locked');
|
|
|
|
});
|
|
|
|
|
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
releaseId: 77777,
|
|
|
|
});
|
|
|
|
const image = mockedAPI.mockImage({
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
const response = await request
|
|
|
|
.post('/v1/reboot')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(423);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
expect(updateLock.lock).to.be.calledOnce;
|
|
|
|
expect(response.body).to.have.property('Error').that.is.not.empty;
|
|
|
|
expect(rebootMock).to.not.have.been.called;
|
|
|
|
},
|
|
|
|
);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
|
|
|
(updateLock.lock as SinonStub).restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return 202 and reboot if force is set to true', async () => {
|
2022-11-11 16:11:47 +00:00
|
|
|
stub(updateLock, 'lock').callsFake(async (__, opts, fn) => {
|
2020-12-03 21:19:44 +00:00
|
|
|
if (opts.force) {
|
2022-11-11 16:11:47 +00:00
|
|
|
return fn();
|
2020-12-03 21:19:44 +00:00
|
|
|
}
|
|
|
|
throw new UpdatesLockedError('Updates locked');
|
|
|
|
});
|
|
|
|
|
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
releaseId: 77777,
|
|
|
|
});
|
|
|
|
const image = mockedAPI.mockImage({
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
const response = await request
|
|
|
|
.post('/v1/reboot')
|
|
|
|
.send({ force: true })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(202);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
expect(updateLock.lock).to.be.calledOnce;
|
|
|
|
expect(response.body).to.have.property('Data').that.is.not.empty;
|
|
|
|
expect(rebootMock).to.have.been.calledOnce;
|
|
|
|
},
|
|
|
|
);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
|
|
|
(updateLock.lock as SinonStub).restore();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('POST /v1/shutdown', () => {
|
|
|
|
let shutdownMock: SinonStub;
|
|
|
|
before(() => {
|
|
|
|
shutdownMock = stub(dbus, 'shutdown').resolves((() => void 0) as any);
|
|
|
|
});
|
|
|
|
|
|
|
|
after(async () => {
|
|
|
|
shutdownMock.restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return 202 and shutdown if no locks are set', async () => {
|
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
releaseId: 77777,
|
|
|
|
});
|
|
|
|
const image = mockedAPI.mockImage({
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
const response = await request
|
|
|
|
.post('/v1/shutdown')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(202);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
expect(response.body).to.have.property('Data').that.is.not.empty;
|
|
|
|
expect(shutdownMock).to.have.been.calledOnce;
|
|
|
|
},
|
|
|
|
);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
|
|
|
shutdownMock.resetHistory();
|
|
|
|
});
|
|
|
|
|
2022-05-26 21:23:08 +00:00
|
|
|
it('should lock all applications before trying to shutdown', async () => {
|
|
|
|
// Setup 2 applications running
|
|
|
|
const twoContainers = [
|
|
|
|
mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 1000,
|
|
|
|
releaseId: 55555,
|
|
|
|
}),
|
|
|
|
mockedAPI.mockService({
|
|
|
|
containerId: 'def456',
|
|
|
|
appId: 2000,
|
|
|
|
releaseId: 77777,
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
const twoImages = [
|
|
|
|
mockedAPI.mockImage({
|
|
|
|
appId: 1000,
|
|
|
|
}),
|
|
|
|
mockedAPI.mockImage({
|
|
|
|
appId: 2000,
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
appMock.mockManagers(twoContainers, [], []);
|
|
|
|
appMock.mockImages([], false, twoImages);
|
|
|
|
|
|
|
|
const lockSpy = spy(updateLock, 'lock');
|
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: twoContainers, images: twoImages },
|
|
|
|
async () => {
|
|
|
|
const response = await request
|
|
|
|
.post('/v1/shutdown')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2022-05-26 21:23:08 +00:00
|
|
|
.expect(202);
|
|
|
|
|
|
|
|
expect(lockSpy.callCount).to.equal(1);
|
|
|
|
// Check that lock was passed both application Ids
|
|
|
|
expect(lockSpy.lastCall.args[0]).to.deep.equal([1000, 2000]);
|
|
|
|
expect(response.body).to.have.property('Data').that.is.not.empty;
|
|
|
|
expect(shutdownMock).to.have.been.calledOnce;
|
|
|
|
},
|
|
|
|
);
|
|
|
|
|
|
|
|
shutdownMock.resetHistory();
|
|
|
|
lockSpy.restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return 423 and reject the reboot if locks are set', async () => {
|
2022-11-11 16:11:47 +00:00
|
|
|
stub(updateLock, 'lock').callsFake(async (__, opts, fn) => {
|
2020-12-03 21:19:44 +00:00
|
|
|
if (opts.force) {
|
2022-11-11 16:11:47 +00:00
|
|
|
return fn();
|
2020-12-03 21:19:44 +00:00
|
|
|
}
|
|
|
|
throw new UpdatesLockedError('Updates locked');
|
|
|
|
});
|
|
|
|
|
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
releaseId: 77777,
|
|
|
|
});
|
|
|
|
const image = mockedAPI.mockImage({
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
const response = await request
|
|
|
|
.post('/v1/shutdown')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(423);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
expect(updateLock.lock).to.be.calledOnce;
|
|
|
|
expect(response.body).to.have.property('Error').that.is.not.empty;
|
|
|
|
expect(shutdownMock).to.not.have.been.called;
|
|
|
|
},
|
|
|
|
);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
|
|
|
(updateLock.lock as SinonStub).restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return 202 and shutdown if force is set to true', async () => {
|
2022-11-11 16:11:47 +00:00
|
|
|
stub(updateLock, 'lock').callsFake(async (__, opts, fn) => {
|
2020-12-03 21:19:44 +00:00
|
|
|
if (opts.force) {
|
2022-11-11 16:11:47 +00:00
|
|
|
return fn();
|
2020-12-03 21:19:44 +00:00
|
|
|
}
|
|
|
|
throw new UpdatesLockedError('Updates locked');
|
|
|
|
});
|
|
|
|
|
|
|
|
// Setup single container application
|
|
|
|
const container = mockedAPI.mockService({
|
|
|
|
containerId: 'abc123',
|
|
|
|
appId: 2,
|
|
|
|
releaseId: 77777,
|
|
|
|
});
|
|
|
|
const image = mockedAPI.mockImage({
|
|
|
|
appId: 2,
|
|
|
|
});
|
|
|
|
appMock.mockManagers([container], [], []);
|
|
|
|
appMock.mockImages([], false, [image]);
|
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
await mockedDockerode.testWithData(
|
|
|
|
{ containers: [container], images: [image] },
|
|
|
|
async () => {
|
|
|
|
const response = await request
|
|
|
|
.post('/v1/shutdown')
|
|
|
|
.send({ force: true })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-12 21:16:37 +00:00
|
|
|
.expect(202);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
2021-02-12 21:16:37 +00:00
|
|
|
expect(updateLock.lock).to.be.calledOnce;
|
|
|
|
expect(response.body).to.have.property('Data').that.is.not.empty;
|
|
|
|
expect(shutdownMock).to.have.been.calledOnce;
|
|
|
|
},
|
|
|
|
);
|
2020-12-03 21:19:44 +00:00
|
|
|
|
|
|
|
(updateLock.lock as SinonStub).restore();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-12-18 20:10:04 +00:00
|
|
|
describe('POST /v1/update', () => {
|
|
|
|
let configStub: SinonStub;
|
|
|
|
let targetUpdateSpy: SinonSpy;
|
2022-09-26 22:19:37 +00:00
|
|
|
let readyForUpdatesStub: SinonStub;
|
2020-12-18 20:10:04 +00:00
|
|
|
|
|
|
|
before(() => {
|
|
|
|
configStub = stub(config, 'get');
|
|
|
|
targetUpdateSpy = spy(TargetState, 'update');
|
2022-09-26 22:19:37 +00:00
|
|
|
readyForUpdatesStub = stub(apiBinder, 'isReadyForUpdates').returns(true);
|
2020-12-18 20:10:04 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
targetUpdateSpy.resetHistory();
|
|
|
|
});
|
|
|
|
|
|
|
|
after(() => {
|
|
|
|
configStub.restore();
|
|
|
|
targetUpdateSpy.restore();
|
2022-09-26 22:19:37 +00:00
|
|
|
readyForUpdatesStub.restore();
|
2020-12-18 20:10:04 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('returns 204 with no parameters', async () => {
|
|
|
|
// Stub response for getting instantUpdates
|
|
|
|
configStub.resolves(true);
|
|
|
|
// Make request
|
|
|
|
await request
|
|
|
|
.post('/v1/update')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2020-12-18 20:10:04 +00:00
|
|
|
.expect(sampleResponses.V1.POST['/update [204 Response]'].statusCode);
|
|
|
|
// Check that TargetState.update was called
|
|
|
|
expect(targetUpdateSpy).to.be.called;
|
|
|
|
expect(targetUpdateSpy).to.be.calledWith(undefined, true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('returns 204 with force: true in body', async () => {
|
|
|
|
// Stub response for getting instantUpdates
|
|
|
|
configStub.resolves(true);
|
|
|
|
// Make request with force: true in the body
|
|
|
|
await request
|
|
|
|
.post('/v1/update')
|
|
|
|
.send({ force: true })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2020-12-18 20:10:04 +00:00
|
|
|
.expect(sampleResponses.V1.POST['/update [204 Response]'].statusCode);
|
|
|
|
// Check that TargetState.update was called
|
|
|
|
expect(targetUpdateSpy).to.be.called;
|
|
|
|
expect(targetUpdateSpy).to.be.calledWith(true, true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('returns 202 when instantUpdates are disabled', async () => {
|
|
|
|
// Stub response for getting instantUpdates
|
|
|
|
configStub.resolves(false);
|
|
|
|
// Make request
|
|
|
|
await request
|
|
|
|
.post('/v1/update')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2020-12-18 20:10:04 +00:00
|
|
|
.expect(sampleResponses.V1.POST['/update [202 Response]'].statusCode);
|
|
|
|
// Check that TargetState.update was not called
|
|
|
|
expect(targetUpdateSpy).to.not.be.called;
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-02-03 10:04:37 +00:00
|
|
|
describe('/v1/device/host-config', () => {
|
|
|
|
// Wrap GET and PATCH /v1/device/host-config tests in the same block to share
|
|
|
|
// common scoped variables, namely file paths and file content
|
|
|
|
const hostnamePath: string = path.join(
|
|
|
|
process.env.ROOT_MOUNTPOINT!,
|
|
|
|
'/etc/hostname',
|
|
|
|
);
|
|
|
|
const proxyBasePath: string = path.join(
|
|
|
|
process.env.ROOT_MOUNTPOINT!,
|
|
|
|
process.env.BOOT_MOUNTPOINT!,
|
|
|
|
'system-proxy',
|
|
|
|
);
|
|
|
|
const redsocksPath: string = path.join(proxyBasePath, 'redsocks.conf');
|
|
|
|
const noProxyPath: string = path.join(proxyBasePath, 'no_proxy');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copies contents of hostname, redsocks.conf, and no_proxy test files with `.template`
|
|
|
|
* endings to test files without `.template` endings to ensure the same data always
|
|
|
|
* exists for /v1/device/host-config test suites
|
|
|
|
*/
|
|
|
|
const restoreConfFileTemplates = async (): Promise<void[]> => {
|
|
|
|
return Promise.all([
|
|
|
|
fs.writeFile(
|
|
|
|
hostnamePath,
|
|
|
|
await fs.readFile(`${hostnamePath}.template`),
|
|
|
|
),
|
|
|
|
fs.writeFile(
|
|
|
|
redsocksPath,
|
|
|
|
await fs.readFile(`${redsocksPath}.template`),
|
|
|
|
),
|
|
|
|
fs.writeFile(noProxyPath, await fs.readFile(`${noProxyPath}.template`)),
|
|
|
|
]);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Set hostname & proxy file content to expected defaults
|
|
|
|
before(async () => await restoreConfFileTemplates());
|
|
|
|
afterEach(async () => await restoreConfFileTemplates());
|
|
|
|
|
|
|
|
// Store GET responses for endpoint in variables so we can be less verbose in tests
|
|
|
|
const hostnameOnlyRes =
|
|
|
|
sampleResponses.V1.GET['/device/host-config [Hostname only]'];
|
|
|
|
const hostnameProxyRes =
|
|
|
|
sampleResponses.V1.GET['/device/host-config [Hostname and proxy]'];
|
|
|
|
|
|
|
|
describe('GET /v1/device/host-config', () => {
|
|
|
|
it('returns current host config (hostname and proxy)', async () => {
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(hostnameProxyRes.statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(hostnameProxyRes.body);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('returns current host config (hostname only)', async () => {
|
2021-04-26 19:54:04 +00:00
|
|
|
await unlinkAll(redsocksPath, noProxyPath);
|
2021-02-03 10:04:37 +00:00
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(hostnameOnlyRes.statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(hostnameOnlyRes.body);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('errors if no hostname file exists', async () => {
|
2021-04-26 19:54:04 +00:00
|
|
|
await unlinkAll(hostnamePath);
|
2021-02-03 10:04:37 +00:00
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(503);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('PATCH /v1/device/host-config', () => {
|
|
|
|
let configSetStub: SinonStub;
|
|
|
|
let logWarnStub: SinonStub;
|
|
|
|
let restartServiceSpy: SinonSpy;
|
|
|
|
|
|
|
|
const validProxyReqs: { [key: string]: number[] | string[] } = {
|
|
|
|
ip: ['proxy.example.org', 'proxy.foo.org'],
|
|
|
|
port: [5128, 1080],
|
|
|
|
type: constants.validRedsocksProxyTypes,
|
|
|
|
login: ['user', 'user2'],
|
|
|
|
password: ['foo', 'bar'],
|
|
|
|
};
|
|
|
|
|
|
|
|
// Mock to short-circuit config.set, allowing writing hostname directly to test file
|
|
|
|
const configSetFakeFn = async <T extends SchemaTypeKey>(
|
|
|
|
keyValues: config.ConfigMap<T>,
|
|
|
|
): Promise<void> =>
|
|
|
|
await fs.writeFile(hostnamePath, (keyValues as any).hostname);
|
|
|
|
|
|
|
|
const validatePatchResponse = (res: supertest.Response): void => {
|
|
|
|
expect(res.text).to.equal(
|
|
|
|
sampleResponses.V1.PATCH['/host/device-config'].text,
|
|
|
|
);
|
|
|
|
expect(res.body).to.deep.equal(
|
|
|
|
sampleResponses.V1.PATCH['/host/device-config'].body,
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
before(() => {
|
|
|
|
configSetStub = stub(config, 'set').callsFake(configSetFakeFn);
|
|
|
|
logWarnStub = stub(log, 'warn');
|
2022-06-06 19:14:50 +00:00
|
|
|
stub(applicationManager, 'getCurrentApps').resolves({
|
|
|
|
'1234567': new App(
|
|
|
|
{
|
|
|
|
appId: 1234567,
|
|
|
|
services: [],
|
|
|
|
volumes: {},
|
|
|
|
networks: {},
|
|
|
|
},
|
|
|
|
false,
|
|
|
|
),
|
|
|
|
});
|
2021-02-03 10:04:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
after(() => {
|
|
|
|
configSetStub.restore();
|
|
|
|
logWarnStub.restore();
|
2022-06-06 19:14:50 +00:00
|
|
|
(applicationManager.getCurrentApps as SinonStub).restore();
|
2021-02-03 10:04:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
restartServiceSpy = spy(dbus, 'restartService');
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
restartServiceSpy.restore();
|
|
|
|
});
|
|
|
|
|
2022-02-09 03:05:40 +00:00
|
|
|
it('prevents patch if update locks are present', async () => {
|
2022-11-11 16:11:47 +00:00
|
|
|
stub(updateLock, 'lock').callsFake(async (__, opts, fn) => {
|
2022-02-09 03:05:40 +00:00
|
|
|
if (opts.force) {
|
2022-11-11 16:11:47 +00:00
|
|
|
return fn();
|
2022-02-09 03:05:40 +00:00
|
|
|
}
|
|
|
|
throw new UpdatesLockedError('Updates locked');
|
|
|
|
});
|
|
|
|
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { hostname: 'foobaz' } })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2022-02-09 03:05:40 +00:00
|
|
|
.expect(423);
|
|
|
|
|
|
|
|
expect(updateLock.lock).to.be.calledOnce;
|
|
|
|
(updateLock.lock as SinonStub).restore();
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2022-02-09 03:05:40 +00:00
|
|
|
.then((response) => {
|
|
|
|
expect(response.body.network.hostname).to.deep.equal(
|
|
|
|
'foobardevice',
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('allows patch while update locks are present if force is in req.body', async () => {
|
2022-11-11 16:11:47 +00:00
|
|
|
stub(updateLock, 'lock').callsFake(async (__, opts, fn) => {
|
2022-02-09 03:05:40 +00:00
|
|
|
if (opts.force) {
|
2022-11-11 16:11:47 +00:00
|
|
|
return fn();
|
2022-02-09 03:05:40 +00:00
|
|
|
}
|
|
|
|
throw new UpdatesLockedError('Updates locked');
|
|
|
|
});
|
|
|
|
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { hostname: 'foobaz' }, force: true })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2022-02-09 03:05:40 +00:00
|
|
|
.expect(200);
|
|
|
|
|
|
|
|
expect(updateLock.lock).to.be.calledOnce;
|
|
|
|
(updateLock.lock as SinonStub).restore();
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2022-02-09 03:05:40 +00:00
|
|
|
.then((response) => {
|
|
|
|
expect(response.body.network.hostname).to.deep.equal('foobaz');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-02-03 10:04:37 +00:00
|
|
|
it('updates the hostname with provided string if string is not empty', async () => {
|
2021-08-20 14:10:24 +00:00
|
|
|
// stub servicePartOf to throw exceptions for the new service names
|
|
|
|
stub(dbus, 'servicePartOf').callsFake(
|
|
|
|
async (serviceName: string): Promise<string> => {
|
|
|
|
if (serviceName === 'balena-hostname') {
|
|
|
|
throw new Error('Unit not loaded.');
|
|
|
|
}
|
|
|
|
return '';
|
|
|
|
},
|
|
|
|
);
|
|
|
|
await unlinkAll(redsocksPath, noProxyPath);
|
|
|
|
|
|
|
|
const patchBody = { network: { hostname: 'newdevice' } };
|
|
|
|
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send(patchBody)
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-08-20 14:10:24 +00:00
|
|
|
.expect(sampleResponses.V1.PATCH['/host/device-config'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// should restart services
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(2);
|
|
|
|
expect(restartServiceSpy.args).to.deep.equal([
|
|
|
|
['balena-hostname'],
|
|
|
|
['resin-hostname'],
|
|
|
|
]);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-08-20 14:10:24 +00:00
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(patchBody);
|
|
|
|
});
|
|
|
|
|
|
|
|
(dbus.servicePartOf as SinonStub).restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('skips restarting hostname services if they are part of config-json.target', async () => {
|
|
|
|
// stub servicePartOf to return the config-json.target we are looking for
|
2022-09-19 15:33:52 +00:00
|
|
|
stub(dbus, 'servicePartOf').callsFake(async (): Promise<string> => {
|
|
|
|
return 'config-json.target';
|
|
|
|
});
|
2021-08-20 14:10:24 +00:00
|
|
|
|
2021-04-26 19:54:04 +00:00
|
|
|
await unlinkAll(redsocksPath, noProxyPath);
|
2021-02-03 10:04:37 +00:00
|
|
|
|
|
|
|
const patchBody = { network: { hostname: 'newdevice' } };
|
|
|
|
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send(patchBody)
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(sampleResponses.V1.PATCH['/host/device-config'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
});
|
|
|
|
|
2021-08-20 14:10:24 +00:00
|
|
|
// skips restarting hostname services if they are part of config-json.target
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(0);
|
2021-02-03 10:04:37 +00:00
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(patchBody);
|
|
|
|
});
|
2021-08-20 14:10:24 +00:00
|
|
|
|
|
|
|
(dbus.servicePartOf as SinonStub).restore();
|
2021-02-03 10:04:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('updates hostname to first 7 digits of device uuid when sent invalid hostname', async () => {
|
2021-04-26 19:54:04 +00:00
|
|
|
await unlinkAll(redsocksPath, noProxyPath);
|
2021-02-03 10:04:37 +00:00
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { hostname: '' } })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(sampleResponses.V1.PATCH['/host/device-config'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
});
|
|
|
|
|
2021-08-20 14:10:24 +00:00
|
|
|
// should restart services
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(2);
|
|
|
|
expect(restartServiceSpy.args).to.deep.equal([
|
|
|
|
['balena-hostname'],
|
|
|
|
['resin-hostname'],
|
|
|
|
]);
|
2021-02-03 10:04:37 +00:00
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.then(async (response) => {
|
|
|
|
const uuidHostname = await config
|
|
|
|
.get('uuid')
|
|
|
|
.then((uuid) => uuid?.slice(0, 7));
|
|
|
|
|
|
|
|
expect(response.body).to.deep.equal({
|
|
|
|
network: { hostname: uuidHostname },
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('removes proxy when sent empty proxy object', async () => {
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { proxy: {} } })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(sampleResponses.V1.PATCH['/host/device-config'].statusCode)
|
|
|
|
.then(async (response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
|
2021-04-26 19:54:04 +00:00
|
|
|
expect(await exists(redsocksPath)).to.be.false;
|
|
|
|
expect(await exists(noProxyPath)).to.be.false;
|
2021-02-03 10:04:37 +00:00
|
|
|
});
|
|
|
|
|
2021-08-20 14:10:24 +00:00
|
|
|
// should restart services
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(3);
|
2021-02-03 10:04:37 +00:00
|
|
|
expect(restartServiceSpy.args).to.deep.equal([
|
2021-08-20 14:10:24 +00:00
|
|
|
['balena-proxy-config'],
|
2021-02-03 10:04:37 +00:00
|
|
|
['resin-proxy-config'],
|
|
|
|
['redsocks'],
|
|
|
|
]);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(hostnameOnlyRes.statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(hostnameOnlyRes.body);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('updates proxy type when provided valid values', async () => {
|
2021-08-20 14:10:24 +00:00
|
|
|
// stub servicePartOf to throw exceptions for the new service names
|
|
|
|
stub(dbus, 'servicePartOf').callsFake(
|
|
|
|
async (serviceName: string): Promise<string> => {
|
|
|
|
if (serviceName === 'balena-proxy-config') {
|
|
|
|
throw new Error('Unit not loaded.');
|
|
|
|
}
|
|
|
|
return '';
|
|
|
|
},
|
|
|
|
);
|
2021-02-03 10:04:37 +00:00
|
|
|
// Test each proxy patch sequentially to prevent conflicts when writing to fs
|
|
|
|
let restartCallCount = 0;
|
|
|
|
for (const key of Object.keys(validProxyReqs)) {
|
|
|
|
const patchBodyValuesforKey: string[] | number[] =
|
|
|
|
validProxyReqs[key];
|
|
|
|
for (const value of patchBodyValuesforKey) {
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { proxy: { [key]: value } } })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set(
|
|
|
|
'Authorization',
|
|
|
|
`Bearer ${await deviceApi.getGlobalApiKey()}`,
|
|
|
|
)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(
|
|
|
|
sampleResponses.V1.PATCH['/host/device-config'].statusCode,
|
|
|
|
)
|
|
|
|
.then((response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
});
|
|
|
|
|
2021-08-20 14:10:24 +00:00
|
|
|
// should restart services
|
2021-02-03 10:04:37 +00:00
|
|
|
expect(restartServiceSpy.callCount).to.equal(
|
2021-08-20 14:10:24 +00:00
|
|
|
++restartCallCount * 3,
|
2021-02-03 10:04:37 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set(
|
|
|
|
'Authorization',
|
|
|
|
`Bearer ${await deviceApi.getGlobalApiKey()}`,
|
|
|
|
)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(hostnameProxyRes.statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal({
|
|
|
|
network: {
|
|
|
|
hostname: hostnameProxyRes.body.network.hostname,
|
|
|
|
// All other proxy configs should be unchanged except for any values sent in patch
|
|
|
|
proxy: {
|
|
|
|
...hostnameProxyRes.body.network.proxy,
|
|
|
|
[key]: value,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} // end for (const value of patchBodyValuesforKey)
|
|
|
|
await restoreConfFileTemplates();
|
|
|
|
} // end for (const key in validProxyReqs)
|
2021-08-20 14:10:24 +00:00
|
|
|
(dbus.servicePartOf as SinonStub).restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('skips restarting proxy services when part of redsocks-conf.target', async () => {
|
|
|
|
// stub servicePartOf to return the redsocks-conf.target we are looking for
|
2022-09-19 15:33:52 +00:00
|
|
|
stub(dbus, 'servicePartOf').callsFake(async (): Promise<string> => {
|
|
|
|
return 'redsocks-conf.target';
|
|
|
|
});
|
2021-08-20 14:10:24 +00:00
|
|
|
// Test each proxy patch sequentially to prevent conflicts when writing to fs
|
|
|
|
for (const key of Object.keys(validProxyReqs)) {
|
|
|
|
const patchBodyValuesforKey: string[] | number[] =
|
|
|
|
validProxyReqs[key];
|
|
|
|
for (const value of patchBodyValuesforKey) {
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { proxy: { [key]: value } } })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set(
|
|
|
|
'Authorization',
|
|
|
|
`Bearer ${await deviceApi.getGlobalApiKey()}`,
|
|
|
|
)
|
2021-08-20 14:10:24 +00:00
|
|
|
.expect(
|
|
|
|
sampleResponses.V1.PATCH['/host/device-config'].statusCode,
|
|
|
|
)
|
|
|
|
.then((response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// skips restarting proxy services when part of redsocks-conf.target
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(0);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set(
|
|
|
|
'Authorization',
|
|
|
|
`Bearer ${await deviceApi.getGlobalApiKey()}`,
|
|
|
|
)
|
2021-08-20 14:10:24 +00:00
|
|
|
.expect(hostnameProxyRes.statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal({
|
|
|
|
network: {
|
|
|
|
hostname: hostnameProxyRes.body.network.hostname,
|
|
|
|
// All other proxy configs should be unchanged except for any values sent in patch
|
|
|
|
proxy: {
|
|
|
|
...hostnameProxyRes.body.network.proxy,
|
|
|
|
[key]: value,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} // end for (const value of patchBodyValuesforKey)
|
|
|
|
await restoreConfFileTemplates();
|
|
|
|
} // end for (const key in validProxyReqs)
|
|
|
|
(dbus.servicePartOf as SinonStub).restore();
|
2021-02-03 10:04:37 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('warns on the supervisor console when provided disallowed proxy fields', async () => {
|
|
|
|
const invalidProxyReqs: { [key: string]: string | number } = {
|
|
|
|
// At this time, don't support changing local_ip or local_port
|
|
|
|
local_ip: '0.0.0.0',
|
|
|
|
local_port: 12345,
|
|
|
|
type: 'invalidType',
|
|
|
|
noProxy: 'not a list of addresses',
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const key of Object.keys(invalidProxyReqs)) {
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { proxy: { [key]: invalidProxyReqs[key] } } })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(200)
|
|
|
|
.then(() => {
|
|
|
|
if (key === 'type') {
|
|
|
|
expect(logWarnStub).to.have.been.calledWith(
|
|
|
|
`Invalid redsocks proxy type, must be one of ${validProxyReqs.type.join(
|
|
|
|
', ',
|
|
|
|
)}`,
|
|
|
|
);
|
|
|
|
} else if (key === 'noProxy') {
|
|
|
|
expect(logWarnStub).to.have.been.calledWith(
|
|
|
|
'noProxy field must be an array of addresses',
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
expect(logWarnStub).to.have.been.calledWith(
|
|
|
|
`Invalid proxy field(s): ${key}`,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
it('replaces no_proxy file with noProxy array from PATCH body', async () => {
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { proxy: { noProxy: ['1.2.3.4/5'] } } })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(sampleResponses.V1.PATCH['/host/device-config'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
});
|
|
|
|
|
2021-08-20 14:10:24 +00:00
|
|
|
// should restart services
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(3);
|
2021-02-03 10:04:37 +00:00
|
|
|
expect(restartServiceSpy.args).to.deep.equal([
|
2021-08-20 14:10:24 +00:00
|
|
|
['balena-proxy-config'],
|
2021-02-03 10:04:37 +00:00
|
|
|
['resin-proxy-config'],
|
|
|
|
['redsocks'],
|
|
|
|
]);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(hostnameProxyRes.statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal({
|
|
|
|
network: {
|
|
|
|
hostname: hostnameProxyRes.body.network.hostname,
|
|
|
|
// New noProxy should be only value in no_proxy file
|
|
|
|
proxy: {
|
|
|
|
...hostnameProxyRes.body.network.proxy,
|
|
|
|
noProxy: ['1.2.3.4/5'],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('removes no_proxy file when sent an empty array', async () => {
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: { proxy: { noProxy: [] } } })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(sampleResponses.V1.PATCH['/host/device-config'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
});
|
|
|
|
|
2021-08-20 14:10:24 +00:00
|
|
|
// should restart services
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(3);
|
2021-02-03 10:04:37 +00:00
|
|
|
expect(restartServiceSpy.args).to.deep.equal([
|
2021-08-20 14:10:24 +00:00
|
|
|
['balena-proxy-config'],
|
2021-02-03 10:04:37 +00:00
|
|
|
['resin-proxy-config'],
|
|
|
|
['redsocks'],
|
|
|
|
]);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(hostnameProxyRes.statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal({
|
|
|
|
network: {
|
|
|
|
hostname: hostnameProxyRes.body.network.hostname,
|
|
|
|
// Reference all properties in proxy object EXCEPT noProxy
|
|
|
|
proxy: {
|
|
|
|
ip: hostnameProxyRes.body.network.proxy.ip,
|
|
|
|
login: hostnameProxyRes.body.network.proxy.login,
|
|
|
|
password: hostnameProxyRes.body.network.proxy.password,
|
|
|
|
port: hostnameProxyRes.body.network.proxy.port,
|
|
|
|
type: hostnameProxyRes.body.network.proxy.type,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('does not update hostname or proxy when hostname or proxy are undefined', async () => {
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({ network: {} })
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(sampleResponses.V1.PATCH['/host/device-config'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
validatePatchResponse(response);
|
|
|
|
});
|
|
|
|
|
|
|
|
// As no host configs were patched, no services should be restarted
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(0);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v1/device/host-config')
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(hostnameProxyRes.statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(hostnameProxyRes.body);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('warns on console when sent a malformed patch body', async () => {
|
|
|
|
await request
|
|
|
|
.patch('/v1/device/host-config')
|
|
|
|
.send({})
|
|
|
|
.set('Accept', 'application/json')
|
2022-10-18 02:07:40 +00:00
|
|
|
.set('Authorization', `Bearer ${await deviceApi.getGlobalApiKey()}`)
|
2021-02-03 10:04:37 +00:00
|
|
|
.expect(200)
|
|
|
|
.then(() => {
|
|
|
|
expect(logWarnStub).to.have.been.calledWith(
|
|
|
|
"Key 'network' must exist in PATCH body",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(restartServiceSpy.callCount).to.equal(0);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2020-11-18 18:24:18 +00:00
|
|
|
});
|