summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorMike Morris <michael.patrick.morris@gmail.com>2015-02-10 19:18:32 -0800
committerMike Morris <michael.patrick.morris@gmail.com>2015-02-10 19:18:32 -0800
commitb87cff289c1b65757ab0d652f09a5a12470c9cc2 (patch)
treedb5849296075f59827bb5855fe1d2bf49c7f5441
parent83b32ec1db3dce493d79303bae9289b8e652f01b (diff)
downloadqtlocation-mapboxgl-b87cff289c1b65757ab0d652f09a5a12470c9cc2.tar.gz
add mapbox-gl-test-suite render tests
-rw-r--r--Makefile11
-rw-r--r--package.json2
-rwxr-xr-xscripts/travis_test.sh2
-rw-r--r--test/js/file_source.test.js (renamed from test/file_source.test.js)6
-rw-r--r--test/js/map.test.js (renamed from test/map.test.js)8
-rw-r--r--test/render.test.js161
6 files changed, 180 insertions, 10 deletions
diff --git a/Makefile b/Makefile
index 4f83d28679..3df03e09de 100644
--- a/Makefile
+++ b/Makefile
@@ -49,9 +49,16 @@ build/Makefile: $(MBGL)/config/$(HOST).gypi
$(MBGL)/config/%.gypi: $(MBGL) $(MBGL)/configure
make -C $(MBGL) config/$*.gypi
+.PHONY: test-suite
+test-suite: build
+ @(`npm bin`/tape test/render.test.js || true)
+
+.PHONY: test-js
+test-js: build
+ @`npm bin`/tape test/js/**/*.test.js
+
.PHONY: test
-test: build
- @`npm bin`/tape test/**/*.test.js
+test: test-js test-suite
.PHONY: clean
clean:
diff --git a/package.json b/package.json
index 0f53fe51ec..be66045298 100644
--- a/package.json
+++ b/package.json
@@ -6,6 +6,8 @@
"nan": "^1.4.1"
},
"devDependencies": {
+ "aws-sdk": "^2.1.9",
+ "mapbox-gl-test-suite": "git://github.com/mapbox/mapbox-gl-test-suite",
"mkdirp": "^0.5.0",
"tape": "^3.5.0"
},
diff --git a/scripts/travis_test.sh b/scripts/travis_test.sh
index 707293505a..70fff88aa4 100755
--- a/scripts/travis_test.sh
+++ b/scripts/travis_test.sh
@@ -6,5 +6,5 @@ set -o pipefail
# Travis OS X has no GPU
if [[ ${TRAVIS_OS_NAME} == "linux" ]]; then
npm test
- # ./scripts/deploy_results.sh
+ ./scripts/deploy_results.sh
fi
diff --git a/test/file_source.test.js b/test/js/file_source.test.js
index 90a91de970..72f2180f20 100644
--- a/test/file_source.test.js
+++ b/test/js/file_source.test.js
@@ -1,9 +1,9 @@
-/* jshint node: true, unused: false */
'use strict';
-var test = require('tape').test;
+/* jshint node: true */
-var mbgl = require('..');
+var test = require('tape');
+var mbgl = require('../..');
test('FileSource', function(t) {
t.test('constructor', function(t) {
diff --git a/test/map.test.js b/test/js/map.test.js
index c612636949..97daa8d524 100644
--- a/test/map.test.js
+++ b/test/js/map.test.js
@@ -1,16 +1,16 @@
-/* jshint node: true, unused: false */
'use strict';
-var test = require('tape').test;
+/* jshint node: true */
-var mbgl = require('..');
+var test = require('tape');
+var mbgl = require('../..');
var fs = require('fs');
var path = require('path');
var mkdirp = require('mkdirp');
mkdirp.sync('test/results');
-var style = require('./fixtures/style.json');
+var style = require('../fixtures/style.json');
function setup(fileSource, callback) {
callback(new mbgl.Map(fileSource));
diff --git a/test/render.test.js b/test/render.test.js
new file mode 100644
index 0000000000..1329b7ef53
--- /dev/null
+++ b/test/render.test.js
@@ -0,0 +1,161 @@
+'use strict';
+
+/* jshint node:true */
+
+var test = require('tape');
+var mbgl = require('..');
+var fs = require('fs');
+var path = require('path');
+var mkdirp = require('mkdirp');
+var spawn = require('child_process').spawn;
+
+var suitePath = path.dirname(require.resolve('mapbox-gl-test-suite/package.json'));
+
+function template(name) {
+ return fs.readFileSync(require.resolve('mapbox-gl-test-suite/templates/' + name + '.html.tmpl')).toString();
+}
+
+var results = '';
+var resultTemplate = template('result');
+
+function format(tmpl, kwargs) {
+ return tmpl.replace(/\{\{|\}\}|\{([^}]+)\}/g, function(match, key) {
+ if (match === '{{') return '{';
+ if (match === '}}') return '}';
+ return kwargs[key];
+ });
+}
+
+function renderTest(style, info, base, key) {
+ var dir = path.join(suitePath, 'tests', base, key);
+ return function(t) {
+ var watchdog = setTimeout(function() {
+ t.fail('timed out after 20 seconds');
+ }, 20000);
+
+ t.once('end', function() {
+ clearTimeout(watchdog);
+ });
+
+ var fileSource = new mbgl.FileSource();
+ fileSource.request = function(req) {
+ fs.readFile(path.join(suitePath, decodeURIComponent(req.url)), function(err, data) {
+ req.respond(err, { data: data });
+ t.error(err);
+ });
+ };
+ fileSource.cancel = function() {};
+
+ var map = new mbgl.Map(fileSource);
+ map.load(style);
+
+ map.render(info[key], function(err, image) {
+ t.error(err);
+ mkdirp.sync(dir);
+
+ var expected = path.join(dir, 'expected.png');
+ var actual = path.join(dir, 'actual.png');
+ var diff = path.join(dir, 'diff.png');
+
+ if (process.env.UPDATE) {
+ fs.writeFile(expected, image, function(err) {
+ t.error(err);
+ t.end();
+ });
+ } else {
+ fs.writeFile(actual, image, function(err) {
+ t.error(err);
+
+ var compare = spawn('compare', ['-metric', 'MAE', actual, expected, diff]);
+ var error = '';
+
+ compare.stderr.on('data', function (data) {
+ error += data.toString();
+ });
+
+ compare.on('exit', function (code) {
+ // The compare program returns 2 on error otherwise 0 if the images are similar or 1 if they are dissimilar.
+ if (code === 2) {
+ writeResult(error.trim(), Infinity);
+ } else {
+ var match = error.match(/^\d+(?:\.\d+)?\s+\(([^\)]+)\)\s*$/);
+ var difference = match ? parseFloat(match[1]) : Infinity;
+ writeResult(match ? '' : error, difference);
+ }
+ });
+
+ compare.stdin.end();
+
+ function writeResult(error, difference) {
+ var allowedDifference = ('diff' in info) ? info.diff : 0.001;
+ var color = difference <= allowedDifference ? 'green' : 'red';
+
+ results += format(resultTemplate, {
+ name: base,
+ key: key,
+ color: color,
+ error: error ? '<p>' + error + '</p>' : '',
+ difference: difference,
+ zoom: info.zoom || 0,
+ center: info.center || [0, 0],
+ bearing: info.bearing || 0,
+ width: info.width || 512,
+ height: info.height || 512
+ });
+
+ t.ok(difference <= allowedDifference);
+ t.end();
+ }
+ });
+ }
+ });
+ };
+}
+
+function rewriteLocalSchema(uri) {
+ return uri.replace(/^local:\/\//, '');
+}
+
+var tests;
+
+if (process.argv[1] === __filename) {
+ tests = process.argv.slice(2);
+}
+
+test('Render', function(t) {
+ fs.readdirSync(path.join(suitePath, 'tests')).forEach(function(dir) {
+ if (dir === 'index.html' || dir[0] === '.') return;
+ if (tests && tests.length && tests.indexOf(dir) < 0) return;
+
+ var style = require(path.join(suitePath, 'tests', dir, 'style.json')),
+ info = require(path.join(suitePath, 'tests', dir, 'info.json'));
+
+ for (var k in style.sources) {
+ var source = style.sources[k];
+
+ if (source.tiles) {
+ source.tiles = source.tiles.map(rewriteLocalSchema);
+ }
+
+ if (source.url) {
+ source.url = rewriteLocalSchema(source.url);
+ }
+ }
+
+ if (style.sprite) style.sprite = rewriteLocalSchema(style.sprite);
+ if (style.glyphs) style.glyphs = rewriteLocalSchema(style.glyphs);
+
+ style = JSON.stringify(style);
+
+ for (k in info) {
+ (info[k].native === false ? t.skip : t.test)(dir + ' ' + k, renderTest(style, info, dir, k));
+ }
+ });
+
+ t.test('results', function(t) {
+ var p = path.join(suitePath, 'tests', 'index.html');
+ fs.writeFileSync(p, format(template('results'), {results: results}));
+ console.warn('Results at: ' + p);
+ t.end();
+ });
+});