You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
356 lines
12 KiB
JavaScript
356 lines
12 KiB
JavaScript
5 years ago
|
'use strict';
|
||
|
|
||
|
const assert = require('assert');
|
||
|
const path = require('path');
|
||
4 years ago
|
const fs = require('fs');
|
||
5 years ago
|
const SwaggerParser = require('@apidevtools/swagger-parser');
|
||
5 years ago
|
const request = require('request-promise-native');
|
||
|
const nconf = require('nconf');
|
||
5 years ago
|
const util = require('util');
|
||
|
const wait = util.promisify(setTimeout);
|
||
5 years ago
|
|
||
5 years ago
|
const db = require('./mocks/databasemock');
|
||
|
const helpers = require('./helpers');
|
||
4 years ago
|
const meta = require('../src/meta');
|
||
5 years ago
|
const user = require('../src/user');
|
||
|
const groups = require('../src/groups');
|
||
|
const categories = require('../src/categories');
|
||
|
const topics = require('../src/topics');
|
||
|
const plugins = require('../src/plugins');
|
||
|
const flags = require('../src/flags');
|
||
|
const messaging = require('../src/messaging');
|
||
4 years ago
|
const utils = require('../src/utils');
|
||
5 years ago
|
|
||
|
describe('Read API', async () => {
|
||
|
let readApi = false;
|
||
4 years ago
|
let writeApi = false;
|
||
|
const readApiPath = path.resolve(__dirname, '../public/openapi/read.yaml');
|
||
|
const writeApiPath = path.resolve(__dirname, '../public/openapi/write.yaml');
|
||
5 years ago
|
let jar;
|
||
4 years ago
|
let csrfToken;
|
||
5 years ago
|
let setup = false;
|
||
|
const unauthenticatedRoutes = ['/api/login', '/api/register']; // Everything else will be called with the admin user
|
||
|
|
||
4 years ago
|
const mocks = {
|
||
|
get: {},
|
||
|
post: {},
|
||
|
put: {},
|
||
|
delete: {
|
||
|
'/users/{uid}/tokens/{token}': [
|
||
|
{
|
||
|
in: 'path',
|
||
|
name: 'uid',
|
||
|
example: 1,
|
||
|
},
|
||
|
{
|
||
|
in: 'path',
|
||
|
name: 'token',
|
||
|
example: utils.generateUUID(),
|
||
|
},
|
||
|
],
|
||
|
},
|
||
|
};
|
||
|
|
||
5 years ago
|
async function dummySearchHook(data) {
|
||
|
return [1];
|
||
|
}
|
||
|
|
||
|
after(async function () {
|
||
|
plugins.unregisterHook('core', 'filter:search.query', dummySearchHook);
|
||
|
});
|
||
|
|
||
|
async function setupData() {
|
||
|
if (setup) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Create sample users
|
||
|
const adminUid = await user.create({ username: 'admin', password: '123456', email: '[email protected]' });
|
||
|
const unprivUid = await user.create({ username: 'unpriv', password: '123456', email: '[email protected]' });
|
||
4 years ago
|
for (let x = 0; x < 3; x++) {
|
||
|
// eslint-disable-next-line no-await-in-loop
|
||
|
await user.create({ username: 'deleteme', password: '123456' }); // for testing of user deletion routes (uids 4-6)
|
||
|
}
|
||
5 years ago
|
await groups.join('administrators', adminUid);
|
||
|
|
||
4 years ago
|
// Create sample group
|
||
|
await groups.create({
|
||
|
name: 'Test Group',
|
||
|
});
|
||
|
|
||
|
await meta.settings.set('core.api', {
|
||
|
tokens: [{
|
||
|
token: mocks.delete['/users/{uid}/tokens/{token}'][1].example,
|
||
|
uid: 1,
|
||
|
description: 'for testing of token deletion rotue',
|
||
|
timestamp: Date.now(),
|
||
|
}],
|
||
|
});
|
||
|
|
||
5 years ago
|
// Create a category
|
||
|
const testCategory = await categories.create({ name: 'test' });
|
||
|
|
||
|
// Post a new topic
|
||
|
const testTopic = await topics.post({
|
||
|
uid: adminUid,
|
||
|
cid: testCategory.cid,
|
||
|
title: 'Test Topic',
|
||
|
content: 'Test topic content',
|
||
|
});
|
||
4 years ago
|
const unprivTopic = await topics.post({
|
||
|
uid: unprivUid,
|
||
|
cid: testCategory.cid,
|
||
|
title: 'Test Topic 2',
|
||
|
content: 'Test topic 2 content',
|
||
|
});
|
||
5 years ago
|
|
||
|
// Create a sample flag
|
||
|
await flags.create('post', 1, unprivUid, 'sample reasons', Date.now());
|
||
|
|
||
|
// Create a new chat room
|
||
|
await messaging.newRoom(1, [2]);
|
||
|
|
||
4 years ago
|
// Create an empty file to test DELETE /files
|
||
|
fs.closeSync(fs.openSync(path.resolve(nconf.get('upload_path'), 'files/test.txt'), 'w'));
|
||
|
|
||
4 years ago
|
const socketUser = require('../src/socket.io/user');
|
||
5 years ago
|
// export data for admin user
|
||
|
await socketUser.exportProfile({ uid: adminUid }, { uid: adminUid });
|
||
|
await socketUser.exportPosts({ uid: adminUid }, { uid: adminUid });
|
||
|
await socketUser.exportUploads({ uid: adminUid }, { uid: adminUid });
|
||
|
// wait for export child process to complete
|
||
4 years ago
|
await wait(5000);
|
||
5 years ago
|
|
||
5 years ago
|
// Attach a search hook so /api/search is enabled
|
||
|
plugins.registerHook('core', {
|
||
|
hook: 'filter:search.query',
|
||
|
method: dummySearchHook,
|
||
|
});
|
||
|
|
||
|
jar = await helpers.loginUser('admin', '123456');
|
||
4 years ago
|
|
||
|
// Retrieve CSRF token using cookie, to test Write API
|
||
|
const config = await request({
|
||
|
url: nconf.get('url') + '/api/config',
|
||
|
json: true,
|
||
|
jar: jar,
|
||
|
});
|
||
|
csrfToken = config.csrf_token;
|
||
|
|
||
5 years ago
|
setup = true;
|
||
|
}
|
||
5 years ago
|
|
||
|
it('should pass OpenAPI v3 validation', async () => {
|
||
|
try {
|
||
4 years ago
|
await SwaggerParser.validate(readApiPath);
|
||
|
await SwaggerParser.validate(writeApiPath);
|
||
5 years ago
|
} catch (e) {
|
||
|
assert.ifError(e);
|
||
|
}
|
||
|
});
|
||
5 years ago
|
|
||
4 years ago
|
readApi = await SwaggerParser.dereference(readApiPath);
|
||
|
writeApi = await SwaggerParser.dereference(writeApiPath);
|
||
5 years ago
|
|
||
4 years ago
|
generateTests(readApi, Object.keys(readApi.paths));
|
||
4 years ago
|
generateTests(writeApi, Object.keys(writeApi.paths), writeApi.servers[0].url);
|
||
|
|
||
|
function generateTests(api, paths, prefix) {
|
||
|
// Iterate through all documented paths, make a call to it, and compare the result body with what is defined in the spec
|
||
|
paths.forEach((path) => {
|
||
|
const context = api.paths[path];
|
||
|
let schema;
|
||
|
let response;
|
||
|
let url;
|
||
|
let method;
|
||
|
const headers = {};
|
||
|
const qs = {};
|
||
|
|
||
|
Object.keys(context).forEach((_method) => {
|
||
4 years ago
|
if (api.info.title === 'NodeBB Read API' && _method !== 'get') {
|
||
|
return;
|
||
|
}
|
||
4 years ago
|
|
||
|
it('should have examples when parameters are present', () => {
|
||
|
method = _method;
|
||
|
let parameters = context[method].parameters;
|
||
|
let testPath = path;
|
||
|
|
||
|
if (parameters) {
|
||
|
// Use mock data if provided
|
||
|
parameters = mocks[method][path] || parameters;
|
||
|
|
||
|
parameters.forEach((param) => {
|
||
|
assert(param.example !== null && param.example !== undefined, `${method.toUpperCase()} ${path} has parameters without examples`);
|
||
|
|
||
|
switch (param.in) {
|
||
|
case 'path':
|
||
|
testPath = testPath.replace('{' + param.name + '}', param.example);
|
||
|
break;
|
||
|
case 'header':
|
||
|
headers[param.name] = param.example;
|
||
|
break;
|
||
|
case 'query':
|
||
|
qs[param.name] = param.example;
|
||
|
break;
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
|
||
4 years ago
|
url = nconf.get('url') + (prefix || '') + testPath;
|
||
4 years ago
|
});
|
||
|
|
||
|
it('may contain a request body with application/json type if POST/PUT/DELETE', () => {
|
||
|
if (['post', 'put', 'delete'].includes(method) && context[method].hasOwnProperty('requestBody')) {
|
||
|
assert(context[method].requestBody);
|
||
|
assert(context[method].requestBody.content);
|
||
|
assert(context[method].requestBody.content['application/json']);
|
||
|
assert(context[method].requestBody.content['application/json'].schema);
|
||
|
assert(context[method].requestBody.content['application/json'].schema.properties);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
it('should resolve with a 200 when called', async () => {
|
||
|
await setupData();
|
||
|
|
||
|
if (csrfToken) {
|
||
|
headers['x-csrf-token'] = csrfToken;
|
||
|
}
|
||
|
|
||
|
let body = {};
|
||
|
if (context[method].hasOwnProperty('requestBody')) {
|
||
|
body = buildBody(context[method].requestBody.content['application/json'].schema.properties);
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
// console.log(`calling ${method} ${url} with`, body);
|
||
|
response = await request(url, {
|
||
|
method: method,
|
||
|
jar: !unauthenticatedRoutes.includes(path) ? jar : undefined,
|
||
|
json: true,
|
||
|
headers: headers,
|
||
|
qs: qs,
|
||
|
body: body,
|
||
|
});
|
||
|
} catch (e) {
|
||
|
assert(!e, `${method.toUpperCase()} ${path} resolved with ${e.message}`);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
// Recursively iterate through schema properties, comparing type
|
||
|
it('response should match schema definition', () => {
|
||
|
const has200 = context[method].responses['200'];
|
||
|
if (!has200) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const hasJSON = has200.content && has200.content['application/json'];
|
||
|
if (hasJSON) {
|
||
|
schema = context[method].responses['200'].content['application/json'].schema;
|
||
|
compare(schema, response, method.toUpperCase(), path, 'root');
|
||
|
}
|
||
|
|
||
|
// TODO someday: text/csv, binary file type checking?
|
||
|
});
|
||
|
|
||
|
it('should successfully re-login if needed', async () => {
|
||
|
const reloginPaths = ['/users/{uid}/password'];
|
||
|
if (method === 'put' && reloginPaths.includes(path)) {
|
||
|
jar = await helpers.loginUser('admin', '123456');
|
||
|
|
||
|
// Retrieve CSRF token using cookie, to test Write API
|
||
|
const config = await request({
|
||
|
url: nconf.get('url') + '/api/config',
|
||
|
json: true,
|
||
|
jar: jar,
|
||
|
});
|
||
|
csrfToken = config.csrf_token;
|
||
|
}
|
||
|
});
|
||
4 years ago
|
});
|
||
5 years ago
|
});
|
||
4 years ago
|
}
|
||
|
|
||
|
function buildBody(schema) {
|
||
|
return Object.keys(schema).reduce((memo, cur) => {
|
||
|
memo[cur] = schema[cur].example;
|
||
|
return memo;
|
||
|
}, {});
|
||
|
}
|
||
4 years ago
|
|
||
4 years ago
|
function compare(schema, response, method, path, context) {
|
||
4 years ago
|
let required = [];
|
||
|
const additionalProperties = schema.hasOwnProperty('additionalProperties');
|
||
|
|
||
|
if (schema.allOf) {
|
||
|
schema = schema.allOf.reduce((memo, obj) => {
|
||
|
required = required.concat(obj.required ? obj.required : Object.keys(obj.properties));
|
||
|
memo = { ...memo, ...obj.properties };
|
||
|
return memo;
|
||
|
}, {});
|
||
|
} else if (schema.properties) {
|
||
|
required = schema.required || Object.keys(schema.properties);
|
||
|
schema = schema.properties;
|
||
|
} else {
|
||
|
// If schema contains no properties, check passes
|
||
|
return;
|
||
4 years ago
|
}
|
||
4 years ago
|
|
||
4 years ago
|
// Compare the schema to the response
|
||
|
required.forEach((prop) => {
|
||
|
if (schema.hasOwnProperty(prop)) {
|
||
4 years ago
|
assert(response.hasOwnProperty(prop), '"' + prop + '" is a required property (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
4 years ago
|
|
||
|
// Don't proceed with type-check if the value could possibly be unset (nullable: true, in spec)
|
||
|
if (response[prop] === null && schema[prop].nullable === true) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// Therefore, if the value is actually null, that's a problem (nullable is probably missing)
|
||
4 years ago
|
assert(response[prop] !== null, '"' + prop + '" was null, but schema does not specify it to be a nullable property (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
4 years ago
|
|
||
|
switch (schema[prop].type) {
|
||
|
case 'string':
|
||
4 years ago
|
assert.strictEqual(typeof response[prop], 'string', '"' + prop + '" was expected to be a string, but was ' + typeof response[prop] + ' instead (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
4 years ago
|
break;
|
||
|
case 'boolean':
|
||
4 years ago
|
assert.strictEqual(typeof response[prop], 'boolean', '"' + prop + '" was expected to be a boolean, but was ' + typeof response[prop] + ' instead (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
4 years ago
|
break;
|
||
|
case 'object':
|
||
4 years ago
|
assert.strictEqual(typeof response[prop], 'object', '"' + prop + '" was expected to be an object, but was ' + typeof response[prop] + ' instead (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
|
compare(schema[prop], response[prop], method, path, context ? [context, prop].join('.') : prop);
|
||
4 years ago
|
break;
|
||
|
case 'array':
|
||
4 years ago
|
assert.strictEqual(Array.isArray(response[prop]), true, '"' + prop + '" was expected to be an array, but was ' + typeof response[prop] + ' instead (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
4 years ago
|
|
||
|
if (schema[prop].items) {
|
||
4 years ago
|
// Ensure the array items have a schema defined
|
||
4 years ago
|
assert(schema[prop].items.type || schema[prop].items.allOf, '"' + prop + '" is defined to be an array, but its items have no schema defined (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
4 years ago
|
|
||
|
// Compare types
|
||
|
if (schema[prop].items.type === 'object' || Array.isArray(schema[prop].items.allOf)) {
|
||
|
response[prop].forEach((res) => {
|
||
4 years ago
|
compare(schema[prop].items, res, method, path, context ? [context, prop].join('.') : prop);
|
||
4 years ago
|
});
|
||
|
} else if (response[prop].length) { // for now
|
||
|
response[prop].forEach((item) => {
|
||
4 years ago
|
assert.strictEqual(typeof item, schema[prop].items.type, '"' + prop + '" should have ' + schema[prop].items.type + ' items, but found ' + typeof items + ' instead (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
4 years ago
|
});
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
});
|
||
|
|
||
|
// Compare the response to the schema
|
||
|
Object.keys(response).forEach((prop) => {
|
||
|
if (additionalProperties) { // All bets are off
|
||
|
return;
|
||
|
}
|
||
|
|
||
4 years ago
|
assert(schema[prop], '"' + prop + '" was found in response, but is not defined in schema (path: ' + method + ' ' + path + ', context: ' + context + ')');
|
||
4 years ago
|
});
|
||
|
}
|
||
5 years ago
|
});
|