Skip to content

Commit 2940a1c

Browse files
amarzaveryvishrutshah
authored andcommittedFeb 6, 2017
Adding semantic and model validator to CI (Azure#910)
* added model validation and more refactoring * Dependency on master of openapi-validation-tools
1 parent 3015ce5 commit 2940a1c

File tree

9 files changed

+374
-257
lines changed

9 files changed

+374
-257
lines changed
 

‎.travis.yml

+12-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
language: node_js
22
node_js:
3-
- "4"
3+
- "6"
44
services:
55
- docker
66
env:
@@ -11,13 +11,21 @@ env:
1111
- MODE=ruby
1212
- MODE=linter PR_ONLY=false
1313
- MODE=linter PR_ONLY=true
14+
- MODE=semantic PR_ONLY=false
15+
- MODE=semantic PR_ONLY=true
16+
- MODE=model PR_ONLY=false
17+
- MODE=model PR_ONLY=true
1418
global:
1519
secure: IwPVExGJQfL6QpTd6Utv3R/qBz7yApgMspzMLSkrq9cMP6t+NHk1vYg0n2evcWaq2rqNZE9xBMfRZi7Yy/RCUT++qALE89xlNvZ/A94PaDUiK3tYYNj1XxsgRauCHshXz6smmLwQMpTKsEl6lyjXeXpwQMDNRehhTu0Hg2tE7ZikfQVb+jjRNcP41RfzzGQyxEhd3fxF2G1G21WXH0cXtZssOd1MYcXhmXk8dR3DyEEMbPzDV1Bk3auXOvbqS79bnZ/Pi4p/7P/aTOm8O2ACKM0XAvww95vAQ0LcPzKnNe3sNjFVAssBiZbIk0Zs30wULBphlTxVoufHKSZuTf+QEBTKpH99v/+SBDDPu9+0q2esq7TKgf8bvzkeXjh54fECdJUqEo9E2gW08+RQxWCqryMJouOPcY2OMs4lZwSMOXQY68a/CYVRWFaFg5s6jntC8sLHtDxV0qem3xyjc+852v8rUfkyvMhOBoZJjWmnqYVqdamHOOfrzjc7AzPUEzSeiN6OPPVli+SzwLHdip0GxdK46pAISCOcbdyYn11VvTIn1QosE66eWhF6SViVH6lNWgSfVTpcQ2zq/qSKh0/zpwn82Ys+wKDOf3EwQAanndgk26npi7Ik4nIuexZ/TE56rQ7qjZqmHoxFz7QMeTZDiVmLFxtR19cTT3GLxDz8nBQ=
1620
matrix:
1721
fast_finish: true
1822
allow_failures:
1923
- env: MODE=linter PR_ONLY=false
2024
- env: MODE=linter PR_ONLY=true
25+
- env: MODE=semantic PR_ONLY=false
26+
- env: MODE=semantic PR_ONLY=true
27+
- env: MODE=model PR_ONLY=false
28+
- env: MODE=model PR_ONLY=true
2129
before_install:
2230
- docker pull lmazuel/swagger-to-sdk
2331
- python -c "import os; print('\n'.join(v for v in os.environ.keys() if v.startswith('TRAVIS')))" > /tmp/env_file
@@ -38,5 +46,7 @@ script:
3846
- if [[ $MODE == 'python' ]]; then $DOCKER_CMD AutorestCI/azure-sdk-for-python --pr-repo-id Azure/azure-sdk-for-python -o master -v; fi
3947
- if [[ $MODE == 'node' ]]; then $DOCKER_CMD AutorestCI/azure-sdk-for-node --pr-repo-id Azure/azure-sdk-for-node -o master -v; fi
4048
- if [[ $MODE == 'ruby' ]]; then $DOCKER_CMD AutorestCI/azure-sdk-for-ruby --pr-repo-id Azure/azure-sdk-for-ruby -o master -v; fi
41-
- if [[ $MODE == 'syntax' ]]; then npm test -- test/test.js; fi
49+
- if [[ $MODE == 'syntax' ]]; then npm test -- test/syntax.js; fi
4250
- if [[ $MODE == 'linter' ]]; then npm test -- test/linter.js; fi
51+
- if [[ $MODE == 'semantic' ]]; then npm test -- test/semantic.js; fi
52+
- if [[ $MODE == 'model' ]]; then npm test -- test/model.js; fi

‎LICENSE

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
The MIT License (MIT)
22

3-
Copyright (c) 2016 Microsoft
3+
Copyright (c) 2017 Microsoft
44

55
Permission is hereby granted, free of charge, to any person obtaining a copy
66
of this software and associated documentation files (the "Software"), to deal

‎package.json

+5-6
Original file line numberDiff line numberDiff line change
@@ -9,16 +9,15 @@
99
"version": "0.1.0",
1010
"description": "Tests for Azure REST API Specifications",
1111
"license": "MIT",
12-
"dependencies": {
12+
"devDependencies": {
13+
"mocha": "*",
1314
"async": "^2.1.4",
1415
"glob": "^5.0.14",
1516
"json-schema-ref-parser": "^3.1.2",
1617
"lodash": "^3.10.1",
1718
"request": "^2.61.0",
18-
"z-schema": "^3.16.1"
19-
},
20-
"devDependencies": {
21-
"mocha": "*"
19+
"z-schema": "^3.16.1",
20+
"openapi-validation-tools": "azure/openapi-validation-tools#master"
2221
},
2322
"homepage": "https://github.com/azure/azure-rest-api-specs",
2423
"repository": {
@@ -31,4 +30,4 @@
3130
"scripts": {
3231
"test": "mocha -t 500000"
3332
}
34-
}
33+
}

‎test/linter.js

+7-48
Original file line numberDiff line numberDiff line change
@@ -2,58 +2,17 @@
22
// Licensed under the MIT License. See License in the project root for license information.
33

44
'use strict';
5-
var glob = require('glob'),
6-
path = require('path'),
7-
_ = require('lodash');
8-
9-
var globPath, swaggers;
10-
var execSync = require('child_process').execSync;
11-
var isWindows = (process.platform.lastIndexOf('win') === 0);
12-
var prOnly = undefined !== process.env['PR_ONLY'] ? process.env['PR_ONLY'] : 'false';
13-
globPath = path.join(__dirname, '../', '/**/swagger/*.json');
14-
swaggers = _(glob.sync(globPath));
15-
16-
/**
17-
* Converts command to OS specific command by prepending `mono` for non-windows prOnlySwaggers
18-
* @returns {string} clr command
19-
*/
20-
function clrCmd(cmd) {
21-
return isWindows ? cmd : ('mono ' + cmd);
22-
};
23-
24-
/**
25-
* Retrieves list of swagger files to be processed for linting
26-
* @returns {Array} list of files to be processed for linting
27-
*/
28-
function getFilesForLinter() {
29-
if (prOnly === 'true') {
30-
// TODO: Currently works for PR into master branch only
31-
var cmd = 'git diff --name-only HEAD $(git merge-base HEAD master)';
32-
let result;
33-
try {
34-
result = execSync(cmd, { encoding: 'utf8' });
35-
console.log(result);
36-
var swaggerFileInPR = result.split('\n').filter(function (item) {
37-
return (item.match(/.*\/swagger\/*/ig) !== null);
38-
});
39-
console.log(`>>>> Number of swaggers found in this PR: ${swaggerFileInPR.length}`);
40-
return swaggerFileInPR;
41-
} catch (err) {
42-
throw err;
43-
}
44-
} else {
45-
// Return all the swagger files for linter processing
46-
return swaggers;
47-
}
48-
}
5+
var _ = require('lodash'),
6+
execSync = require('child_process').execSync,
7+
utils = require('./util/utils');
498

509
describe('AutoRest Linter validation:', function () {
5110
var autoRestLocation = './AutoRest.*/tools/AutoRest.exe';
52-
let swaggersToProcess = getFilesForLinter();
11+
let swaggersToProcess = utils.getFilesChangedInPR();
5312
_(swaggersToProcess).each(function (swagger) {
54-
it(swagger + ' should follow linter rules.', function (done) {
55-
var cmd = clrCmd(autoRestLocation + ' -CodeGenerator None -I ' + swagger);
56-
console.log(cmd);
13+
it(swagger + ' should honor linter validation rules.', function (done) {
14+
var cmd = utils.clrCmd(autoRestLocation + ' -CodeGenerator None -I ' + swagger);
15+
console.log(`Executing: ${cmd}`);
5716
let result;
5817
try {
5918
result = execSync(cmd, { encoding: 'utf8' });

‎test/model.js

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// Copyright (c) Microsoft Corporation. All rights reserved.
2+
// Licensed under the MIT License. See License in the project root for license information.
3+
4+
'use strict';
5+
var _ = require('lodash'),
6+
utils = require('./util/utils'),
7+
oav = require('openapi-validation-tools');
8+
9+
describe('Azure swagger model validation using x-ms-examples and examples in spec', function () {
10+
let swaggersToProcess = utils.getFilesChangedInPR();
11+
_(swaggersToProcess).each(function (swagger) {
12+
it(swagger + ' should have valid examples.', function (done) {
13+
oav.validateExamples(swagger, null, false, 'error').catch(function (err) {
14+
console.log(err);
15+
});
16+
done();
17+
});
18+
}).value();
19+
});

‎test/semantic.js

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
// Copyright (c) Microsoft Corporation. All rights reserved.
2+
// Licensed under the MIT License. See License in the project root for license information.
3+
4+
'use strict';
5+
var _ = require('lodash'),
6+
utils = require('./util/utils'),
7+
oav = require('openapi-validation-tools');
8+
9+
describe('Azure swagger semantic validation:', function () {
10+
let swaggersToProcess = utils.getFilesChangedInPR();
11+
_(swaggersToProcess).each(function (swagger) {
12+
it(swagger + ' should be semantically valid.', function (done) {
13+
oav.validateSpec(swagger, false, 'error').catch(function (err) {
14+
console.log(err);
15+
});
16+
done();
17+
});
18+
}).value();
19+
});

‎test/syntax.js

+142
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,142 @@
1+
// Copyright (c) Microsoft Corporation. All rights reserved.
2+
// Licensed under the MIT License. See License in the project root for license information.
3+
4+
'use strict';
5+
var assert = require("assert"),
6+
fs = require('fs'),
7+
path = require('path'),
8+
_ = require('lodash'),
9+
async = require('async'),
10+
RefParser = require('json-schema-ref-parser'),
11+
util = require('util'),
12+
utils = require('./util/utils');
13+
14+
var context;
15+
16+
17+
// Useful when debugging a test for a particular swagger.
18+
// Just update the regex. That will return an array of filtered items.
19+
// swaggers = swaggers.filter(function(item) {
20+
// return (item.match(/.*arm-redis.*/ig) !== null);
21+
// });
22+
23+
24+
describe('Azure swagger schema validation:', function () {
25+
before(function (done) {
26+
utils.initializeValidator(function (err, result) {
27+
if (err) {
28+
done(err);
29+
}
30+
context = result;
31+
done();
32+
});
33+
});
34+
35+
_(utils.swaggers).each(function (swagger) {
36+
it(swagger + ' should be a valid Swagger document.', function (done) {
37+
utils.parseJsonFromFile(swagger, function (err, parsedData) {
38+
if (err) { done(err); }
39+
if (parsedData.documents && util.isArray(parsedData.documents)) {
40+
console.log(util.format('Skipping the test for \'%s\' document as it seems to be a composite swagger doc.', swagger));
41+
done();
42+
}
43+
var valid = context.validator.validate(parsedData, context.extensionSwaggerSchema);
44+
if (!valid) {
45+
var error = context.validator.getLastErrors();
46+
throw new Error("Schema validation failed: " + util.inspect(error, { depth: null }));
47+
}
48+
assert(valid === true);
49+
done();
50+
});
51+
});
52+
}).value();
53+
54+
describe('Azure composite swagger schema validation:', function () {
55+
_(utils.compositeSwaggers).each(function (compositeSwagger) {
56+
it('composite: ' + compositeSwagger + ' should be a valid Composite Swagger document.', function (done) {
57+
utils.parseJsonFromFile(compositeSwagger, function (err, parsedData) {
58+
if (err) { done(err); }
59+
var valid = context.validator.validate(parsedData, context.compositeSchema);
60+
if (!valid) {
61+
var error = context.validator.getLastErrors();
62+
throw new Error("Schema validation for Composite Swagger failed: " + util.inspect(error, { depth: null }));
63+
}
64+
assert(valid === true);
65+
var compositeSwaggerDir = path.dirname(compositeSwagger);
66+
var messages = [];
67+
if (parsedData.documents && parsedData.documents.length > 0) {
68+
async.eachSeries(parsedData.documents, function (docUrl, loopCallback) {
69+
//construct the absolue path if the item in the documents array is a relative path
70+
if (!path.isAbsolute(docUrl) && !docUrl.startsWith('http')) {
71+
docUrl = path.join(compositeSwaggerDir, docUrl);
72+
}
73+
//make a request if it is a url
74+
if (docUrl.startsWith('http')) {
75+
request({ url: docUrl, json: true }, function (error, response, responseBody) {
76+
if (error) {
77+
messages.push('An error occurred while accessing the swagger doc ' +
78+
docUrl + ' from the documents list. The error is ' + util.inspect(error, { depth: null }));
79+
}
80+
if (response.statusCode !== 200) {
81+
messages.push('\'' + response.statusCode + '\': \'File Not Found\'- error occurred while accessing the swagger doc ' +
82+
docUrl + ' from the documents list.');
83+
}
84+
loopCallback();
85+
});
86+
} else {
87+
//check whether the file exists
88+
if (!fs.existsSync(docUrl)) {
89+
messages.push('\'File Not Found\': error occurred while accessing the swagger doc ' +
90+
docUrl + ' from the documents list on the host filesystem.');
91+
}
92+
loopCallback();
93+
}
94+
}, function (err) {
95+
if (err) {
96+
throw err;
97+
}
98+
if (messages.length > 0) {
99+
throw new Error(JSON.stringify(messages));
100+
}
101+
done();
102+
});
103+
} else {
104+
done();
105+
}
106+
});
107+
});
108+
}).value();
109+
});
110+
111+
describe('Azure x-ms-example schema validation:', function () {
112+
_(utils.examples).each(function (example) {
113+
it('x-ms-examples: ' + example + ' should be a valid x-ms-example.', function (done) {
114+
utils.parseJsonFromFile(example, function (err, parsedData) {
115+
if (err) { done(err); }
116+
var valid = context.validator.validate(parsedData, context.exampleSchema);
117+
if (!valid) {
118+
var error = context.validator.getLastErrors();
119+
throw new Error("Schema validation failed: " + util.inspect(error, { depth: null }));
120+
}
121+
assert(valid === true);
122+
done();
123+
});
124+
});
125+
}).value();
126+
});
127+
});
128+
129+
describe('External file or url references ("$ref") in a swagger spec:', function () {
130+
_(utils.swaggers).each(function (swagger) {
131+
it(swagger + ' should be completely resolvable.', function (done) {
132+
RefParser.bundle(swagger, function (bundleErr, bundleResult) {
133+
if (bundleErr) {
134+
var msg = swagger + ' has references that cannot be resolved. They are as follows: \n' + util.inspect(bundleErr.message, { depth: null });
135+
console.log(msg);
136+
throw new Error(msg);
137+
}
138+
done();
139+
});
140+
});
141+
}).value();
142+
});

0 commit comments

Comments
 (0)
Please sign in to comment.