2020-04-30 01:54:51 +00:00
|
|
|
import { expect } from 'chai';
|
2020-05-20 01:27:36 +00:00
|
|
|
import { spy, stub, SinonStub } from 'sinon';
|
2020-05-08 20:04:21 +00:00
|
|
|
import * as supertest from 'supertest';
|
2020-04-30 01:54:51 +00:00
|
|
|
|
2020-07-21 09:45:37 +00:00
|
|
|
import * as apiBinder from '../src/api-binder';
|
2020-07-21 15:25:47 +00:00
|
|
|
import * as deviceState from '../src/device-state';
|
2020-04-30 01:54:51 +00:00
|
|
|
import Log from '../src/lib/supervisor-console';
|
|
|
|
import SupervisorAPI from '../src/supervisor-api';
|
2020-05-08 20:04:21 +00:00
|
|
|
import sampleResponses = require('./data/device-api-responses.json');
|
|
|
|
import mockedAPI = require('./lib/mocked-device-api');
|
2020-04-30 01:54:51 +00:00
|
|
|
|
2020-08-13 12:25:39 +00:00
|
|
|
import * as applicationManager from '../src/compose/application-manager';
|
|
|
|
import { InstancedAppState } from '../src/types/state';
|
|
|
|
|
2020-09-16 14:19:23 +00:00
|
|
|
import * as apiKeys from '../src/lib/api-keys';
|
|
|
|
import * as db from '../src/db';
|
|
|
|
|
2020-04-30 01:54:51 +00:00
|
|
|
const mockedOptions = {
|
2020-05-08 20:04:21 +00:00
|
|
|
listenPort: 54321,
|
2020-04-30 01:54:51 +00:00
|
|
|
timeout: 30000,
|
|
|
|
};
|
|
|
|
|
|
|
|
describe('SupervisorAPI', () => {
|
2020-05-08 20:04:21 +00:00
|
|
|
let api: SupervisorAPI;
|
2020-05-20 01:27:36 +00:00
|
|
|
let healthCheckStubs: SinonStub[];
|
2020-05-08 20:04:21 +00:00
|
|
|
const request = supertest(`http://127.0.0.1:${mockedOptions.listenPort}`);
|
2020-04-30 01:54:51 +00:00
|
|
|
|
2020-05-08 20:04:21 +00:00
|
|
|
before(async () => {
|
2020-07-21 09:45:37 +00:00
|
|
|
await apiBinder.initialized;
|
2020-07-21 15:25:47 +00:00
|
|
|
await deviceState.initialized;
|
2020-07-21 09:45:37 +00:00
|
|
|
|
2020-05-20 01:27:36 +00:00
|
|
|
// Stub health checks so we can modify them whenever needed
|
|
|
|
healthCheckStubs = [
|
2020-07-21 09:45:37 +00:00
|
|
|
stub(apiBinder, 'healthcheck'),
|
2020-07-21 15:25:47 +00:00
|
|
|
stub(deviceState, 'healthcheck'),
|
2020-05-20 01:27:36 +00:00
|
|
|
];
|
2020-08-13 12:25:39 +00:00
|
|
|
|
2020-05-15 19:18:20 +00:00
|
|
|
// The mockedAPI contains stubs that might create unexpected results
|
|
|
|
// See the module to know what has been stubbed
|
2020-05-08 20:04:21 +00:00
|
|
|
api = await mockedAPI.create();
|
2020-06-09 13:43:45 +00:00
|
|
|
|
2020-05-08 20:04:21 +00:00
|
|
|
// Start test API
|
2020-07-21 09:45:37 +00:00
|
|
|
await api.listen(mockedOptions.listenPort, mockedOptions.timeout);
|
2020-09-16 14:19:23 +00:00
|
|
|
|
|
|
|
// Create a scoped key
|
|
|
|
await apiKeys.initialized;
|
|
|
|
await apiKeys.generateCloudKey();
|
2020-05-08 20:04:21 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
after(async () => {
|
|
|
|
try {
|
|
|
|
await api.stop();
|
|
|
|
} catch (e) {
|
|
|
|
if (e.message !== 'Server is not running.') {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
2020-05-20 01:27:36 +00:00
|
|
|
// Restore healthcheck stubs
|
|
|
|
healthCheckStubs.forEach((hc) => hc.restore);
|
2020-05-08 20:04:21 +00:00
|
|
|
// Remove any test data generated
|
|
|
|
await mockedAPI.cleanUp();
|
|
|
|
});
|
|
|
|
|
2020-09-16 14:19:23 +00:00
|
|
|
describe('API Key Scope', () => {
|
|
|
|
it('should generate a key which is scoped for a single application', async () => {
|
|
|
|
// single app scoped key...
|
|
|
|
const appScopedKey = await apiKeys.generateScopedKey(1, 1);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/1/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${appScopedKey}`)
|
|
|
|
.expect(200);
|
|
|
|
});
|
|
|
|
it('should generate a key which is scoped for multiple applications', async () => {
|
|
|
|
// multi-app scoped key...
|
|
|
|
const multiAppScopedKey = await apiKeys.generateScopedKey(1, 2, {
|
|
|
|
scopes: [1, 2].map((appId) => {
|
|
|
|
return { type: 'app', appId };
|
|
|
|
}),
|
|
|
|
});
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/1/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${multiAppScopedKey}`)
|
|
|
|
.expect(200);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/2/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${multiAppScopedKey}`)
|
|
|
|
.expect(200);
|
|
|
|
});
|
|
|
|
it('should generate a key which is scoped for all applications', async () => {
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/1/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
|
|
|
.expect(200);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/2/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
|
|
|
.expect(200);
|
|
|
|
});
|
|
|
|
it('should have a cached lookup of the key scopes to save DB loading', async () => {
|
|
|
|
const scopes = await apiKeys.getScopesForKey(apiKeys.cloudApiKey);
|
|
|
|
|
|
|
|
const key = 'not-a-normal-key';
|
|
|
|
await db.initialized;
|
|
|
|
await db
|
|
|
|
.models('apiSecret')
|
|
|
|
.update({
|
|
|
|
key,
|
|
|
|
})
|
|
|
|
.where({
|
|
|
|
key: apiKeys.cloudApiKey,
|
|
|
|
});
|
|
|
|
|
|
|
|
// the key we had is now gone, but the cache should return values
|
|
|
|
const cachedScopes = await apiKeys.getScopesForKey(apiKeys.cloudApiKey);
|
|
|
|
expect(cachedScopes).to.deep.equal(scopes);
|
|
|
|
|
|
|
|
// this should bust the cache...
|
|
|
|
await apiKeys.generateCloudKey(true);
|
|
|
|
|
|
|
|
// the key we changed should be gone now, and the new key should have the cloud scopes
|
|
|
|
const missingScopes = await apiKeys.getScopesForKey(key);
|
|
|
|
const freshScopes = await apiKeys.getScopesForKey(apiKeys.cloudApiKey);
|
|
|
|
|
|
|
|
expect(missingScopes).to.be.null;
|
|
|
|
expect(freshScopes).to.deep.equal(scopes);
|
|
|
|
});
|
|
|
|
it('should regenerate a key and invalidate the old one', async () => {
|
|
|
|
// single app scoped key...
|
|
|
|
const appScopedKey = await apiKeys.generateScopedKey(1, 1);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/1/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${appScopedKey}`)
|
|
|
|
.expect(200);
|
|
|
|
|
|
|
|
const newScopedKey = await apiKeys.refreshKey(appScopedKey);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/1/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${appScopedKey}`)
|
|
|
|
.expect(401);
|
|
|
|
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/1/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${newScopedKey}`)
|
|
|
|
.expect(200);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-05-08 20:04:21 +00:00
|
|
|
describe('/ping', () => {
|
|
|
|
it('responds with OK (without auth)', async () => {
|
2020-05-15 11:01:51 +00:00
|
|
|
await request.get('/ping').set('Accept', 'application/json').expect(200);
|
2020-05-08 20:04:21 +00:00
|
|
|
});
|
|
|
|
it('responds with OK (with auth)', async () => {
|
|
|
|
await request
|
|
|
|
.get('/ping')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-05-08 20:04:21 +00:00
|
|
|
.expect(200);
|
2020-04-30 01:54:51 +00:00
|
|
|
});
|
2020-05-08 20:04:21 +00:00
|
|
|
});
|
2020-04-30 01:54:51 +00:00
|
|
|
|
2020-05-20 01:27:36 +00:00
|
|
|
describe('V1 endpoints', () => {
|
|
|
|
describe('GET /v1/healthy', () => {
|
|
|
|
it('returns OK because all checks pass', async () => {
|
|
|
|
// Make all healthChecks pass
|
|
|
|
healthCheckStubs.forEach((hc) => hc.resolves(true));
|
|
|
|
await request
|
|
|
|
.get('/v1/healthy')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-05-20 01:27:36 +00:00
|
|
|
.expect(sampleResponses.V1.GET['/healthy'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V1.GET['/healthy'].body,
|
|
|
|
);
|
|
|
|
expect(response.text).to.deep.equal(
|
|
|
|
sampleResponses.V1.GET['/healthy'].text,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
it('Fails because some checks did not pass', async () => {
|
|
|
|
// Make one of the healthChecks fail
|
|
|
|
healthCheckStubs[0].resolves(false);
|
|
|
|
await request
|
|
|
|
.get('/v1/healthy')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.expect(sampleResponses.V1.GET['/healthy [2]'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V1.GET['/healthy [2]'].body,
|
|
|
|
);
|
|
|
|
expect(response.text).to.deep.equal(
|
|
|
|
sampleResponses.V1.GET['/healthy [2]'].text,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2020-07-29 11:46:02 +00:00
|
|
|
|
2020-08-13 12:25:39 +00:00
|
|
|
before(() => {
|
|
|
|
const appState = {
|
|
|
|
[sampleResponses.V1.GET['/apps/2'].body.appId]: {
|
|
|
|
...sampleResponses.V1.GET['/apps/2'].body,
|
|
|
|
services: [
|
|
|
|
{
|
|
|
|
...sampleResponses.V1.GET['/apps/2'].body,
|
|
|
|
serviceId: 1,
|
|
|
|
serviceName: 'main',
|
|
|
|
config: {},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
stub(applicationManager, 'getCurrentApps').resolves(
|
|
|
|
(appState as unknown) as InstancedAppState,
|
|
|
|
);
|
|
|
|
stub(applicationManager, 'executeStep').resolves();
|
|
|
|
});
|
|
|
|
|
|
|
|
after(() => {
|
|
|
|
(applicationManager.executeStep as SinonStub).restore();
|
|
|
|
(applicationManager.getCurrentApps as SinonStub).restore();
|
|
|
|
});
|
|
|
|
|
2020-05-08 20:04:21 +00:00
|
|
|
// TODO: add tests for V1 endpoints
|
2020-07-29 11:46:02 +00:00
|
|
|
describe('GET /v1/apps/:appId', () => {
|
|
|
|
it('returns information about a SPECIFIC application', async () => {
|
|
|
|
await request
|
|
|
|
.get('/v1/apps/2')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-07-29 11:46:02 +00:00
|
|
|
.expect(sampleResponses.V1.GET['/apps/2'].statusCode)
|
2020-08-13 12:25:39 +00:00
|
|
|
.expect('Content-Type', /json/)
|
2020-07-29 11:46:02 +00:00
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V1.GET['/apps/2'].body,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('POST /v1/apps/:appId/stop', () => {
|
|
|
|
it('stops a SPECIFIC application and returns a containerId', async () => {
|
|
|
|
await request
|
|
|
|
.post('/v1/apps/2/stop')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-07-29 11:46:02 +00:00
|
|
|
.expect(sampleResponses.V1.GET['/apps/2/stop'].statusCode)
|
2020-08-13 12:25:39 +00:00
|
|
|
.expect('Content-Type', /json/)
|
2020-07-29 11:46:02 +00:00
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V1.GET['/apps/2/stop'].body,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2020-06-05 13:16:43 +00:00
|
|
|
describe('GET /v1/device', () => {
|
|
|
|
it('returns MAC address', async () => {
|
|
|
|
const response = await request
|
|
|
|
.get('/v1/device')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-06-05 13:16:43 +00:00
|
|
|
.expect(200);
|
|
|
|
|
|
|
|
expect(response.body).to.have.property('mac_address').that.is.not.empty;
|
|
|
|
});
|
|
|
|
});
|
2020-05-08 20:04:21 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('V2 endpoints', () => {
|
|
|
|
describe('GET /v2/device/vpn', () => {
|
|
|
|
it('returns information about VPN connection', async () => {
|
|
|
|
await request
|
|
|
|
.get('/v2/device/vpn')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-05-08 20:04:21 +00:00
|
|
|
.expect('Content-Type', /json/)
|
|
|
|
.expect(sampleResponses.V2.GET['/device/vpn'].statusCode)
|
2020-05-15 11:01:51 +00:00
|
|
|
.then((response) => {
|
2020-05-08 20:04:21 +00:00
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V2.GET['/device/vpn'].body,
|
|
|
|
);
|
|
|
|
});
|
2020-04-30 01:54:51 +00:00
|
|
|
});
|
2020-05-08 20:04:21 +00:00
|
|
|
});
|
2020-05-15 19:18:20 +00:00
|
|
|
|
|
|
|
describe('GET /v2/applications/:appId/state', () => {
|
|
|
|
it('returns information about a SPECIFIC application', async () => {
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/1/state')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-05-15 19:18:20 +00:00
|
|
|
.expect(sampleResponses.V2.GET['/applications/1/state'].statusCode)
|
2020-09-16 14:19:23 +00:00
|
|
|
.expect('Content-Type', /json/)
|
2020-05-15 19:18:20 +00:00
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V2.GET['/applications/1/state'].body,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('returns 400 for invalid appId', async () => {
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/123invalid/state')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-05-15 19:18:20 +00:00
|
|
|
.expect('Content-Type', /json/)
|
|
|
|
.expect(
|
|
|
|
sampleResponses.V2.GET['/applications/123invalid/state'].statusCode,
|
|
|
|
)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V2.GET['/applications/123invalid/state'].body,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('returns 409 because app does not exist', async () => {
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/9000/state')
|
|
|
|
.set('Accept', 'application/json')
|
2020-09-16 14:19:23 +00:00
|
|
|
.set('Authorization', `Bearer ${apiKeys.cloudApiKey}`)
|
2020-05-15 19:18:20 +00:00
|
|
|
.expect(sampleResponses.V2.GET['/applications/9000/state'].statusCode)
|
|
|
|
.then((response) => {
|
|
|
|
expect(response.body).to.deep.equal(
|
|
|
|
sampleResponses.V2.GET['/applications/9000/state'].body,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
2020-09-16 14:19:23 +00:00
|
|
|
|
|
|
|
describe('Scoped API Keys', () => {
|
|
|
|
it('returns 409 because app is out of scope of the key', async () => {
|
|
|
|
const apiKey = await apiKeys.generateScopedKey(3, 1);
|
|
|
|
await request
|
|
|
|
.get('/v2/applications/2/state')
|
|
|
|
.set('Accept', 'application/json')
|
|
|
|
.set('Authorization', `Bearer ${apiKey}`)
|
|
|
|
.expect(409);
|
|
|
|
});
|
|
|
|
});
|
2020-05-15 19:18:20 +00:00
|
|
|
});
|
|
|
|
|
2020-05-08 20:04:21 +00:00
|
|
|
// TODO: add tests for rest of V2 endpoints
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('State change logging', () => {
|
|
|
|
before(() => {
|
|
|
|
// Spy on functions we will be testing
|
2020-04-30 01:54:51 +00:00
|
|
|
spy(Log, 'info');
|
|
|
|
spy(Log, 'error');
|
|
|
|
});
|
|
|
|
|
2020-05-08 20:04:21 +00:00
|
|
|
beforeEach(async () => {
|
|
|
|
// Start each case with API stopped
|
2020-04-30 01:54:51 +00:00
|
|
|
try {
|
|
|
|
await api.stop();
|
|
|
|
} catch (e) {
|
|
|
|
if (e.message !== 'Server is not running.') {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
after(async () => {
|
2020-05-08 20:04:21 +00:00
|
|
|
// @ts-ignore
|
|
|
|
Log.info.restore();
|
|
|
|
// @ts-ignore
|
|
|
|
Log.error.restore();
|
|
|
|
// Resume API for other test suites
|
2020-06-15 16:46:33 +00:00
|
|
|
return api.listen(mockedOptions.listenPort, mockedOptions.timeout);
|
2020-04-30 01:54:51 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it('logs successful start', async () => {
|
|
|
|
// Start API
|
2020-06-15 16:46:33 +00:00
|
|
|
await api.listen(mockedOptions.listenPort, mockedOptions.timeout);
|
2020-04-30 01:54:51 +00:00
|
|
|
// Check if success start was logged
|
|
|
|
// @ts-ignore
|
|
|
|
expect(Log.info.lastCall?.lastArg).to.equal(
|
|
|
|
`Supervisor API successfully started on port ${mockedOptions.listenPort}`,
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('logs shutdown', async () => {
|
|
|
|
// Start API
|
2020-06-15 16:46:33 +00:00
|
|
|
await api.listen(mockedOptions.listenPort, mockedOptions.timeout);
|
2020-04-30 01:54:51 +00:00
|
|
|
// Stop API
|
|
|
|
await api.stop();
|
|
|
|
// Check if stopped with info was logged
|
|
|
|
// @ts-ignore
|
|
|
|
expect(Log.info.lastCall?.lastArg).to.equal('Stopped Supervisor API');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('logs errored shutdown', async () => {
|
|
|
|
// Start API
|
2020-06-15 16:46:33 +00:00
|
|
|
await api.listen(mockedOptions.listenPort, mockedOptions.timeout);
|
2020-04-30 01:54:51 +00:00
|
|
|
// Stop API with error
|
|
|
|
await api.stop({ errored: true });
|
|
|
|
// Check if stopped with error was logged
|
|
|
|
// @ts-ignore
|
|
|
|
expect(Log.error.lastCall?.lastArg).to.equal('Stopped Supervisor API');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|