diff --git a/package.json b/package.json index 00a1460348a63..9881e46e77575 100644 --- a/package.json +++ b/package.json @@ -17,7 +17,9 @@ "grunt-contrib-watch": "~0.5.3", "grunt-contrib-jade": "~0.10.0", "grunt-contrib-less": "~0.9.0", - "grunt-cli": "~0.1.13" + "grunt-cli": "~0.1.13", + "istanbul": "~0.2.4", + "path-browserify": "0.0.0" }, "scripts": { "test": "grunt test", diff --git a/src/courier/courier.js b/src/courier/courier.js index b38f1a2ceaa23..54d5588f61674 100644 --- a/src/courier/courier.js +++ b/src/courier/courier.js @@ -286,5 +286,17 @@ define(function (require) { this.fetch('doc'); }; + // get the list of open data source objects + // primarily for testing purposes + Courier.prototype._openSources = function (type) { + if (!type) { + return _.transform(this._refs, function (open, refs) { + [].push.apply(open, refs); + }, []); + } + + return this._refs[type] || []; + }; + return Courier; }); \ No newline at end of file diff --git a/src/kibana/require.config.js b/src/kibana/require.config.js index ec9cf26e87ea8..1dafaa38bf415 100644 --- a/src/kibana/require.config.js +++ b/src/kibana/require.config.js @@ -15,7 +15,8 @@ require.config({ lodash: '../bower_components/lodash/dist/lodash', moment: '../bower_components/moment/moment', gridster: '../bower_components/gridster/dist/jquery.gridster', - config: '../config' + configFile: '../config', + bower_components: '../bower_components' }, shim: { angular: { diff --git a/tasks/config/connect.js b/tasks/config/connect.js index 6bc3dde2a8878..b84c760a8225f 100644 --- a/tasks/config/connect.js +++ b/tasks/config/connect.js @@ -1,20 +1,59 @@ module.exports = function (grunt) { + var instrumentationMiddleware = require('../utils/instrumentation'); + var amdWrapMiddleware = require('../utils/amd-wrapper'); + return { dev: { options: { - base: '<%= src %>' - } - }, - test: { - options: { - base: [ - '<%= unitTestDir %>', - '<%= testUtilsDir %>', - '<%= src %>', - '<%= root %>/node_modules/mocha', - '<%= root %>/node_modules/expect.js' - ], - port: 8001 + middleware: function (connect, options, stack) { + stack = stack || []; + + var root = grunt.config.get('root'); + + // when a request for an intrumented file comes in (?instrument=true) + // and it is included in `pattern`, it will be handled + // by this middleware + stack.push(instrumentationMiddleware({ + // root that files should be served from + root: root, + + // make file names easier to read + displayRoot: grunt.config.get('src'), + + // filter the filenames that will be served + filter: function (filename) { + // return true if the filename should be + // included in the coverage report (results are cached) + return grunt.file.isMatch([ + '**/src/**/*.js', + '!**/src/bower_components/**/*', + '!**/src/kibana/utils/{event_emitter,next_tick}.js' + ], filename); + } + })); + + // minimize code duplication (especially in the istanbul reporter) + // by allowing node_modules to be requested in an AMD wrapper + stack.push(amdWrapMiddleware({ + root: root + })); + + // standard static middleware reading from the root + stack.push(connect.static(root)); + + // allow browsing directories + stack.push(connect.directory(root)); + + // redirect requests for '/' to '/src/' + stack.push(function (req, res, next) { + if (req.url !== '/') return next(); + res.statusCode = 303; + res.setHeader('Location', '/src/'); + res.end(); + }); + + return stack; + } } } }; diff --git a/tasks/config/jade.js b/tasks/config/jade.js index fd19f7f151e94..dc03d72e5c250 100644 --- a/tasks/config/jade.js +++ b/tasks/config/jade.js @@ -1,24 +1,41 @@ module.exports = function (grunt) { + var path = require('path'); + return { + options: { + compileDebug: false + }, test: { - src: [ - '<%= unitTestDir %>/**/*.jade', - '<%= app %>/partials/**/*.jade', - '<%= app %>/apps/**/*.jade' - ], - expand: true, - ext: '.html', + files: { + '<%= unitTestDir %>/index.html': '<%= unitTestDir %>/index.jade' + }, options: { data: function (src, dest) { - var pattern = grunt.config.process('<%= unitTestDir %>/**/*.js'); - var tests = grunt.file.expand({}, pattern).map(function (filename) { - return filename.replace(grunt.config.get('unitTestDir'), ''); - }); - return { tests: JSON.stringify(tests) }; - }, - client: false + var unitTestDir = grunt.config.get('unitTestDir'); + + // filter for non unit test related files + if (!~path.dirname(src).indexOf(unitTestDir)) return; + + var pattern = unitTestDir + '/specs/**/*.js'; + var appdir = grunt.config.get('app'); + + return { + tests: grunt.file.expand({}, pattern).map(function (filename) { + return path.relative(appdir, filename).replace(/\.js$/, ''); + }) + }; + } + } + }, + clientside: { + files: { + '<%= testUtilsDir %>/istanbul_reporter/report.jade.js': '<%= testUtilsDir %>/istanbul_reporter/report.clientside-jade' + }, + options: { + client: true, + amd: true, + namespace: false // return the template directly in the amd wrapper } } }; -}; - +}; \ No newline at end of file diff --git a/tasks/config/jshint.js b/tasks/config/jshint.js index 89bf0d1279f53..2a2bf9e68574d 100644 --- a/tasks/config/jshint.js +++ b/tasks/config/jshint.js @@ -3,7 +3,12 @@ module.exports = function (config) { // just lint the source dir source: { files: { - src: ['Gruntfile.js', '<%= src %>/**/*.js', '<%= unitTestDir %>/**/*.js', '<%= root %>/tasks/**/*.js'] + src: [ + 'Gruntfile.js', + '<%= src %>/**/*.js', + '<%= unitTestDir %>/**/*.js', + '<%= root %>/tasks/**/*.js' + ] } }, options: { diff --git a/tasks/config/less.js b/tasks/config/less.js index aa72cedd553d0..eeeb562c4cb70 100644 --- a/tasks/config/less.js +++ b/tasks/config/less.js @@ -1,8 +1,7 @@ module.exports = { src: { src: [ - '<%= app %>/styles/**/*.less', - '<%= app %>/apps/**/*.less', + '<%= app %>/**/styles/**/*.less', '!<%= src %>/**/_*.less' ], expand: true, diff --git a/tasks/config/mocha.js b/tasks/config/mocha.js index 13c593f409463..e8a519756c553 100644 --- a/tasks/config/mocha.js +++ b/tasks/config/mocha.js @@ -1,12 +1,14 @@ module.exports = { + options: { + log: true, + logErrors: true, + run: false + }, unit: { options: { - log: true, - logErrors: true, urls: [ - 'http://localhost:8001/' - ], - run: false + 'http://localhost:8000/test/unit/' + ] } } }; \ No newline at end of file diff --git a/tasks/config/watch.js b/tasks/config/watch.js index 2bfb964239acc..2335d76d64959 100644 --- a/tasks/config/watch.js +++ b/tasks/config/watch.js @@ -1,23 +1,29 @@ module.exports = function (grunt) { return { test: { - files: ['<%= unitTestDir %>/*.jade', '<%= unitTestDir %>/**/*.js'], - tasks: ['jade:test', 'mocha:unit'] + files: [ + '<%= unitTestDir %>/**/*.js' + ], + tasks: ['mocha:unit'] }, less: { files: [ - '<%= app %>/**/*.less', - '<%= src %>/courier/**/*.less' + '<%= app %>/**/styles/**/*.less', + '!<%= src %>/**/_*.less' ], tasks: ['less'] }, jade: { files: [ - '<%= app %>/**/*.jade', - '<%= src %>/courier/**/*.jade', - '!<%= unitTestDir %>/**/*.jade' + '<%= unitTestDir %>/index.jade' + ], + tasks: ['jade:test'] + }, + clientside_jade: { + files: [ + '<%= testUtilsDir %>/istanbul_reporter/report.clientside-jade' ], - tasks: ['jade'] + tasks: ['jade:clientside'] } }; }; diff --git a/tasks/dev.js b/tasks/dev.js index ee4c78feec73a..6afb009e02f04 100644 --- a/tasks/dev.js +++ b/tasks/dev.js @@ -1,3 +1,8 @@ module.exports = function (grunt) { - grunt.registerTask('dev', ['less', 'jade', 'connect:dev', 'watch']); + grunt.registerTask('dev', [ + 'less', + 'jade', + 'connect:dev', + 'watch' + ]); }; \ No newline at end of file diff --git a/tasks/server.js b/tasks/server.js index 17cdd157de2ab..00dc30c9c792a 100644 --- a/tasks/server.js +++ b/tasks/server.js @@ -1,4 +1,3 @@ module.exports = function (grunt) { grunt.registerTask('server', ['connect:dev:keepalive']); - grunt.registerTask('test_server', ['connect:test:keepalive']); }; \ No newline at end of file diff --git a/tasks/test.js b/tasks/test.js index cc19319f8f26b..02481b13babb0 100644 --- a/tasks/test.js +++ b/tasks/test.js @@ -2,13 +2,19 @@ module.exports = function (grunt) { /* jshint scripturl:true */ grunt.registerTask('test', [ 'jshint', - 'connect:test', - 'jade:test', + 'connect:dev', + 'jade', 'mocha:unit' ]); + grunt.registerTask('coverage', [ + 'blanket', + 'connect:dev', + 'mocha:coverage' + ]); + grunt.registerTask('test:watch', [ - 'connect:test', + 'connect:dev', 'watch:test' ]); }; diff --git a/tasks/utils/amd-wrapper.js b/tasks/utils/amd-wrapper.js new file mode 100644 index 0000000000000..56de3fbfb9a73 --- /dev/null +++ b/tasks/utils/amd-wrapper.js @@ -0,0 +1,26 @@ +module.exports = function amdWrapMiddleware(opts) { + opts = opts || {}; + + var root = opts.root || '/'; + var path = require('path'); + var fs = require('fs'); + var pathPrefix = opts.pathPrefix || '/amd-wrap/'; + + return function (req, res, next) { + // only allow prefixed requests + if (req.url.substring(0, pathPrefix.length) !== pathPrefix) return next(); + + // strip the prefix and form the filename + var filename = path.join(root, req._parsedUrl.pathname.replace('/amd-wrap/', '')); + + fs.readFile(filename, 'utf8', function (err, contents) { + // file does not exist + if (err) return next(err.code === 'ENOENT' ? void 0 : err); + + // respond with the wrapped code + res.statusCode = 200; + res.setHeader('Content-Type', 'application/javascript'); + res.end('define(function (require, exports, module) {\n' + contents + '\n});'); + }); + }; +}; \ No newline at end of file diff --git a/tasks/utils/instrumentation.js b/tasks/utils/instrumentation.js new file mode 100644 index 0000000000000..2c47a9a6445c5 --- /dev/null +++ b/tasks/utils/instrumentation.js @@ -0,0 +1,77 @@ +var Istanbul = require('istanbul'); +var i = new Istanbul.Instrumenter({ + embedSource: true, + preserveComments: true, + noAutoWrap: true +}); + +module.exports = function instrumentationMiddleware(opts) { + var fs = require('fs'); + var path = require('path'); + + // for root directory that files will be served from + var root = opts.root || '/'; + + // the root directory used to create a relative file path + // for display in coverage reports + var displayRoot = opts.displayRoot || null; + + // filter the files in root that can be instrumented + var filter = opts.filter || function (filename) { + // by default only instrument *.js files + return /\.js$/.test(filename); + }; + + // cache filename resolution + var fileMap = {}; + + function filenameForReq(req) { + if (!~req.url.indexOf('instrument=true')) return false; + + // expected absolute path to the file + var filename = path.join(root, req._parsedUrl.pathname); + + // shortcut for dev where we could be reloading on every save + if (fileMap[filename] !== void 0) return fileMap[filename]; + + var ret = filename; + + if (!fs.existsSync(filename) || !opts.filter(filename)) { + ret = false; + } + + // cache the return value for next time + fileMap[filename] = ret; + return ret; + } + + return function (req, res, next) { + // resolve the request to a readable filename + var filename = filenameForReq(req); + // the file either doesn't exist of it was filtered out by opts.filter + if (!filename) return next(); + + fs.readFile(filename, 'utf8', function (err, content) { + if (err) { + if (err.code !== 'ENOENT') { + // other issue, report! + return next(err); + } + + // file was deleted, clear cache and move on + delete fileMap[filename]; + return next(); + } + + res.statusCode = 200; + res.setHeader('Content-Type', 'application/javascript'); + res.end(i.instrumentSync( + content, + // make file names easier to read + displayRoot ? path.relative(displayRoot, filename) : filename + )); + }); + }; + + +}; \ No newline at end of file diff --git a/test/.jshintrc b/test/.jshintrc index 0cf87810d60c9..ea222a2c9872b 100644 --- a/test/.jshintrc +++ b/test/.jshintrc @@ -1,8 +1,6 @@ { "extends": "../src/.jshintrc", - "white": false, - "globals": { "module": false, "inject": false, diff --git a/test/unit/index.html b/test/unit/index.html index 8eb19b49f017e..7517defa9867d 100644 --- a/test/unit/index.html +++ b/test/unit/index.html @@ -1,21 +1,42 @@ -Kibana4 Tests
\ No newline at end of file +} + +function runTests() { + require(["../../test/unit/specs/apps/dashboard/index","../../test/unit/specs/apps/dashboard/mocks/modules","../../test/unit/specs/calculate_indices","../../test/unit/specs/courier","../../test/unit/specs/data_source","../../test/unit/specs/mapper"], function () { + window.mochaRunner = mocha.run().on('end', function () { + window.mochaResults = this.stats; + }); + }); +} + +if (COVERAGE) { + setupCoverage(runTests); +} else { + runTests(); +} \ No newline at end of file diff --git a/test/unit/index.jade b/test/unit/index.jade index cfdd8e2bc1045..90e7d705c5295 100644 --- a/test/unit/index.jade +++ b/test/unit/index.jade @@ -2,34 +2,53 @@ doctype html html head title Kibana4 Tests - link(rel="stylesheet", href="mocha.css") + link(rel="stylesheet", href='/node_modules/mocha/mocha.css') body #mocha - script(src="expect.js") - script(src="mocha.js") - script. - mocha.setup('bdd'); - // sauce labs & selenium inject global variables that break this - // mocha.checkLeaks(); - // mocha.globals(['mochaRunner', 'angular']); - script(src="bower_components/requirejs/require.js") - script(src="kibana/require.config.js") + script(src='/node_modules/expect.js/expect.js') + script(src='/node_modules/mocha/mocha.js') + script(src='/src/bower_components/requirejs/require.js') + script(src='/src/kibana/require.config.js') script(type="text/javascript"). + window.COVERAGE = !!(/coverage=true/i.test(location.search)); + mocha.setup('bdd'); + require.config({ - paths: { - sinon: '../sinon' - }, - shim: { - 'sinon/sinon': { - deps: [ - 'sinon/sinon-timers-1.8.2' - ], - exports: 'sinon' - } - } + baseUrl: '/src/kibana', + paths: { + sinon: '../../test/utils/sinon', + istanbul_reporter: '../../test/utils/istanbul_reporter' + }, + shim: { + 'sinon/sinon': { + deps: [ + 'sinon/sinon-timers-1.8.2' + ], + exports: 'sinon' + } + }, + // mark all requested files with instrument query param + urlArgs: COVERAGE ? 'instrument=true' : void 0 }); - require(!{tests}, function () { - window.mochaRunner = mocha.run().on('end', function () { - window.mochaResults = this.stats; + + function setupCoverage(done) { + document.title = document.title.replace('Tests', 'Coverage'); + require(['istanbul_reporter/reporter'], function (IstanbulReporter) { + mocha.reporter(IstanbulReporter); + done(); }); - }); + } + + function runTests() { + require(!{JSON.stringify(tests)}, function () { + window.mochaRunner = mocha.run().on('end', function () { + window.mochaResults = this.stats; + }); + }); + } + + if (COVERAGE) { + setupCoverage(runTests); + } else { + runTests(); + } diff --git a/test/unit/specs/apps/dashboard/index.js b/test/unit/specs/apps/dashboard/index.js index 75b5aeea4b117..9a6ed58665771 100644 --- a/test/unit/specs/apps/dashboard/index.js +++ b/test/unit/specs/apps/dashboard/index.js @@ -4,7 +4,7 @@ define(function (require) { var _ = require('lodash'); var $ = require('jquery'); var sinon = require('sinon/sinon'); - var configFile = require('../../../config.js'); + var configFile = require('configFile'); // Load the kibana app dependencies. require('angular-route'); @@ -19,13 +19,13 @@ define(function (require) { describe('Mapper', function () { var $scope; - beforeEach(function() { + beforeEach(function () { // Start the kibana module module('kibana'); // Create the scope - inject(function($rootScope, $controller) { + inject(function ($rootScope, $controller) { $scope = $rootScope.$new(); var dashCtrl = $controller('dashboard', { $scope: $scope diff --git a/test/unit/specs/apps/dashboard/mocks/modules.js b/test/unit/specs/apps/dashboard/mocks/modules.js index 8de8c1efc50e0..2d43954f6b43a 100644 --- a/test/unit/specs/apps/dashboard/mocks/modules.js +++ b/test/unit/specs/apps/dashboard/mocks/modules.js @@ -2,23 +2,23 @@ define(function (require) { var angular = require('angular'); - angular.module('app/dashboard',[]); + angular.module('app/dashboard', []); // Need this because the controller requires it - angular.module('kibana/directives',[]); + angular.module('kibana/directives', []); // create mock service for courier var mock = { getvalue: function () {} }; - angular.module('kibana/services',[]) - .service('courier',function () { + angular.module('kibana/services', []) + .service('courier', function () { return mock; }); // Could probably get rid of ngRoute if you want to stub it - angular.module('kibana',['ngRoute','kibana/services','app/dashboard']); + angular.module('kibana', ['ngRoute', 'kibana/services', 'app/dashboard']); }); \ No newline at end of file diff --git a/test/unit/specs/calculate_indices.js b/test/unit/specs/calculate_indices.js index 05db6f36584a7..745f9f8fb6e82 100644 --- a/test/unit/specs/calculate_indices.js +++ b/test/unit/specs/calculate_indices.js @@ -4,19 +4,19 @@ define(function (require) { describe('calculateIndices()', function () { - describe('error checking', function() { + describe('error checking', function () { it('should throw an error if start is > end', function () { expect(function () { calculateIndices(moment().add('day', 1), moment()); }).to.throwError(); }); it('should throw an error if interval is not [ hour, day, week, year ]', function () { - expect(function () { calculateIndices(moment().subtract('day', 1), moment(), 'century' ); }).to.throwError(); + expect(function () { calculateIndices(moment().subtract('day', 1), moment(), 'century'); }).to.throwError(); }); it('should throw an error if pattern is not set', function () { - expect(function () { calculateIndices(moment().subtract('day', 1), moment(), 'hour' ); }).to.throwError(); + expect(function () { calculateIndices(moment().subtract('day', 1), moment(), 'hour'); }).to.throwError(); }); }); - describe('hourly interval', function() { + describe('hourly interval', function () { beforeEach(function () { var date = '2014-01-15 04:30:10'; this.start = moment.utc(date).subtract('hours', 4); @@ -36,7 +36,7 @@ define(function (require) { }); }); - describe('daily interval', function() { + describe('daily interval', function () { beforeEach(function () { var date = '2014-01-15 04:30:10'; this.start = moment.utc(date).subtract('days', 4); @@ -56,7 +56,7 @@ define(function (require) { }); }); - describe('weekly interval', function() { + describe('weekly interval', function () { beforeEach(function () { var date = '2014-01-15 04:30:10'; this.start = moment.utc(date).subtract('week', 4); @@ -76,7 +76,7 @@ define(function (require) { }); }); - describe('yearly interval', function() { + describe('yearly interval', function () { beforeEach(function () { var date = '2014-01-15 04:30:10'; this.start = moment.utc(date).subtract('years', 4); diff --git a/test/unit/specs/courier.js b/test/unit/specs/courier.js index 4b755e3c2da41..c570afc0c7dfb 100644 --- a/test/unit/specs/courier.js +++ b/test/unit/specs/courier.js @@ -13,7 +13,18 @@ define(function (require) { expect(courier).to.be.a(Courier); }); - it('knows when a DataSource object has event listeners for the results event'); + it('knows when a DataSource object has event listeners for the results event', function () { + var courier = new Courier(); + var ds = courier.createSource('doc'); + + expect(courier._openSources('doc')).to.have.length(0); + ds.on('results', function () {}); + expect(courier._openSources('doc')).to.have.length(1); + ds.removeAllListeners('results'); + expect(courier._openSources('doc')).to.have.length(0); + }); + + it('executes queries on the interval for searches that have listeners for results'); describe('events', function () { diff --git a/test/unit/specs/mapper.js b/test/unit/specs/mapper.js index 108016a9654f0..7a2c1a0b1e389 100644 --- a/test/unit/specs/mapper.js +++ b/test/unit/specs/mapper.js @@ -1,11 +1,11 @@ define(function (require) { - var elasticsearch = require('../bower_components/elasticsearch/elasticsearch.js'); + var elasticsearch = require('bower_components/elasticsearch/elasticsearch'); var _ = require('lodash'); var sinon = require('sinon/sinon'); var Courier = require('courier/courier'); var DataSource = require('courier/data_source/data_source'); var Mapper = require('courier/mapper'); - var fieldMapping = require('../fixtures/field_mapping.js'); + var fieldMapping = require('../fixtures/field_mapping'); var client = new elasticsearch.Client({ host: 'localhost:9200', @@ -18,30 +18,32 @@ define(function (require) { describe('Mapper', function () { var server, source, mapper; - beforeEach(function() { + beforeEach(function () { source = courier.createSource('search') .index('valid') .size(5); mapper = new Mapper(courier); // Stub out a mini mapping response. - sinon.stub(client.indices, 'getFieldMapping',function (params, callback) { - if(params.index === 'valid') { - setTimeout(callback(undefined, fieldMapping),0); + sinon.stub(client.indices, 'getFieldMapping', function (params, callback) { + if (params.index === 'valid') { + setTimeout(callback(undefined, fieldMapping), 0); } else { - setTimeout(callback('Error: Not Found',undefined)); + setTimeout(callback('Error: Not Found', undefined)); } }); sinon.stub(client, 'getSource', function (params, callback) { - if(params.id === 'valid') { - setTimeout(callback(undefined,{'foo.bar': {'type': 'string'}}),0); + if (params.id === 'valid') { + setTimeout(callback(undefined, {'foo.bar': {'type': 'string'}}), 0); } else { - setTimeout(callback('Error: Not Found',undefined),0); + setTimeout(callback('Error: Not Found', undefined), 0); } }); - sinon.stub(client, 'delete', function (params, callback) {callback(undefined,true);}); + sinon.stub(client, 'delete', function (params, callback) { + callback(undefined, true); + }); }); afterEach(function () { @@ -57,7 +59,7 @@ define(function (require) { }); it('has getFieldsFromMapping function that returns a mapping', function (done) { - mapper.getFieldsFromMapping(source,function (err, mapping) { + mapper.getFieldsFromMapping(source, function (err, mapping) { expect(client.indices.getFieldMapping.called).to.be(true); expect(mapping['foo.bar'].type).to.be('string'); done(); @@ -69,7 +71,7 @@ define(function (require) { .index('invalid') .size(5); - mapper.getFieldsFromCache(source,function (err, mapping) { + mapper.getFieldsFromCache(source, function (err, mapping) { expect(client.getSource.called).to.be(true); expect(err).to.be('Error: Not Found'); done(); @@ -77,7 +79,7 @@ define(function (require) { }); it('has getFieldsFromCache that returns a mapping', function (done) { - mapper.getFieldsFromCache(source,function (err, mapping) { + mapper.getFieldsFromCache(source, function (err, mapping) { expect(client.getSource.called).to.be(true); expect(mapping['foo.bar'].type).to.be('string'); done(); diff --git a/test/utils/istanbul_reporter/report.clientside-jade b/test/utils/istanbul_reporter/report.clientside-jade new file mode 100644 index 0000000000000..3341b399bf1a5 --- /dev/null +++ b/test/utils/istanbul_reporter/report.clientside-jade @@ -0,0 +1,398 @@ +#coverage + h1#overview Coverage + #menu + li + a(href='#overview') + | overview + span.cov(class=coverageClass(cov.coverage)) #{cov.coverage | 0} + + for dir in cov.dirs + li.dirname= dir.name || '.' + for file in dir.files + li + span.cov(class=coverageClass(file.coverage)) #{file.coverage | 0} + a(href='##{file.filename}') + span.basename= file.relname + + a#logo(href='http://visionmedia.github.io/mocha/') m + + + #stats(class=coverageClass(cov.coverage)) + .percentage #{cov.coverage | 0}% + .sloc= cov.sloc + .hits= cov.hits + .misses= cov.misses + + #files + for dir in cov.dirs + for file in dir.files + .file + h2(id=file.filename)= file.filename + #stats(class=coverageClass(file.coverage)) + .percentage #{file.coverage | 0}% + .sloc= file.sloc + .hits= file.hits + .misses= file.misses + + table#source + thead + tr + th Line + th Hits + th Source + tbody + for line, number in file.source + if line.coverage > 0 + tr.hit + td.line= number + td.hits= line.coverage + td.source= line.source + else if 0 === line.coverage + tr.miss + td.line= number + td.hits 0 + td.source= line.source + else + tr + td.line= number + td.hits + td.source= line.source || ' ' + +style. + body { + font: 14px/1.6 "Helvetica Neue", Helvetica, Arial, sans-serif; + margin: 0; + color: #2C2C2C; + border-top: 2px solid #ddd; + } + + .coverage-stats { + font-size: 12px; + margin: 0; + padding: 0; + color: white; + text-align: right; + border: none; + box-shadow: none; + position: fixed; + bottom: 40px; + right: 10px; + } + + #coverage { + padding: 60px 60px 350px; + } + + h1 a { + color: inherit; + font-weight: inherit; + } + + h1 a:hover { + text-decoration: none; + } + + .onload h1 { + opacity: 1; + } + + h2 { + width: 80%; + margin-top: 80px; + margin-bottom: 0; + font-weight: 100; + letter-spacing: 1px; + border-bottom: 1px solid #eee; + } + + a { + color: #8A6343; + font-weight: bold; + text-decoration: none; + } + + a:hover { + text-decoration: underline; + } + + ul { + margin-top: 20px; + padding: 0 15px; + width: 100%; + } + + ul li { + float: left; + width: 40%; + margin-top: 5px; + margin-right: 60px; + list-style: none; + border-bottom: 1px solid #eee; + padding: 5px 0; + font-size: 12px; + } + + ul::after { + content: '.'; + height: 0; + display: block; + visibility: hidden; + clear: both; + } + + code { + font: 12px monaco, monospace; + } + + pre { + margin: 30px; + padding: 30px; + border: 1px solid #eee; + border-bottom-color: #ddd; + -webkit-border-radius: 2px; + -moz-border-radius: 2px; + border-radius: 2px; + -webkit-box-shadow: inset 0 0 10px #eee; + -moz-box-shadow: inset 0 0 10px #eee; + box-shadow: inset 0 0 10px #eee; + overflow-x: auto; + } + + img { + margin: 30px; + padding: 1px; + -webkit-border-radius: 3px; + -moz-border-radius: 3px; + border-radius: 3px; + -webkit-box-shadow: 0 3px 10px #dedede, 0 1px 5px #888; + -moz-box-shadow: 0 3px 10px #dedede, 0 1px 5px #888; + box-shadow: 0 3px 10px #dedede, 0 1px 5px #888; + max-width: 100%; + } + + footer { + background: #eee; + width: 100%; + padding: 50px 0; + text-align: right; + border-top: 1px solid #ddd; + } + + footer span { + display: block; + margin-right: 30px; + color: #888; + font-size: 12px; + } + + #menu { + position: fixed; + font-size: 12px; + overflow-y: auto; + top: 0; + right: 0; + margin: 0; + height: 100%; + padding: 15px 0; + text-align: right; + border-left: 1px solid #eee; + -moz-box-shadow: 0 0 2px #888 + , inset 5px 0 20px rgba(0,0,0,.5) + , inset 5px 0 3px rgba(0,0,0,.3); + -webkit-box-shadow: 0 0 2px #888 + , inset 5px 0 20px rgba(0,0,0,.5) + , inset 5px 0 3px rgba(0,0,0,.3); + box-shadow: 0 0 2px #888 + , inset 5px 0 20px rgba(0,0,0,.5) + , inset 5px 0 3px rgba(0,0,0,.3); + -webkit-font-smoothing: antialiased; + background: url(""); + } + + #menu::after { + display: block; + content: ''; + padding-top: 80px; + } + + #logo { + position: fixed; + bottom: 10px; + right: 10px; + background: rgba(255,255,255,.1); + font-size: 11px; + display: block; + width: 20px; + height: 20px; + line-height: 20px; + text-align: center; + -webkit-border-radius: 20px; + -moz-border-radius: 20px; + border-radius: 20px; + -webkit-box-shadow: 0 0 3px rgba(0,0,0,.2); + -moz-box-shadow: 0 0 3px rgba(0,0,0,.2); + box-shadow: 0 0 3px rgba(0,0,0,.2); + color: inherit; + } + + #menu li a { + display: block; + color: white; + padding: 0 35px 0 25px; + -webkit-transition: background 300ms; + -moz-transition: background 300ms; + } + + #menu li { + position: relative; + list-style: none; + } + + #menu a:hover, + #menu a.active { + text-decoration: none; + background: rgba(255,255,255,.1); + } + + #menu li:hover .cov { + opacity: 1; + } + + #menu li.dirname { + color: white; + opacity: .60; + text-align: left; + padding: 15px 35px 0 20px; + } + + #menu li .dirname { + opacity: .60; + padding-right: 2px; + } + + #menu li .basename { + opacity: 1; + } + + #menu .cov { + background: rgba(0,0,0,.4); + position: absolute; + top: 0; + right: 8px; + font-size: 9px; + opacity: .6; + text-align: left; + width: 17px; + -webkit-border-radius: 10px; + -moz-border-radius: 10px; + border-radius: 10px; + padding: 2px 3px; + text-align: center; + } + + #stats:nth-child(2n) { + display: inline-block; + margin-top: 15px; + border: 1px solid #eee; + padding: 10px; + -webkit-box-shadow: inset 0 0 2px #eee; + -moz-box-shadow: inset 0 0 2px #eee; + box-shadow: inset 0 0 2px #eee; + -webkit-border-radius: 5px; + -moz-border-radius: 5px; + border-radius: 5px; + } + + #stats div { + float: left; + padding: 0 5px; + } + + #stats::after { + display: block; + content: ''; + clear: both; + } + + #stats .sloc::after { + content: ' SLOC'; + color: #b6b6b6; + } + + #stats .percentage::after { + content: ' coverage'; + color: #b6b6b6; + } + + #stats .hits, + #stats .misses { + display: none; + } + + .high { + color: #00d4b4; + } + .medium { + color: #e87d0d; + } + .low { + color: #d4081a; + } + .terrible { + color: #d4081a; + font-weight: bold; + } + + table { + width: 80%; + margin-top: 10px; + border-collapse: collapse; + border: 1px solid #cbcbcb; + color: #363636; + -webkit-border-radius: 3px; + -moz-border-radius: 3px; + border-radius: 3px; + } + + table thead { + display: none; + } + + table td.line, + table td.hits { + width: 20px; + background: #eaeaea; + text-align: center; + font-size: 11px; + padding: 0 10px; + color: #949494; + } + + table td.hits { + width: 10px; + padding: 2px 5px; + color: rgba(0,0,0,.2); + background: #f0f0f0; + } + + tr.miss td.line, + tr.miss td.hits { + background: #e6c3c7; + } + + tr.miss td { + background: #f8d5d8; + } + + td.source { + padding-left: 15px; + line-height: 15px; + white-space: pre; + font: 12px monaco, monospace; + } + + code .comment { color: #ddd } + code .init { color: #2F6FAD } + code .string { color: #5890AD } + code .keyword { color: #8A6343 } + code .number { color: #2F6FAD } \ No newline at end of file diff --git a/test/utils/istanbul_reporter/report.jade.js b/test/utils/istanbul_reporter/report.jade.js new file mode 100644 index 0000000000000..ab79acca6be7e --- /dev/null +++ b/test/utils/istanbul_reporter/report.jade.js @@ -0,0 +1,322 @@ +define(['jade'], function(jade) { if(jade && jade['runtime'] !== undefined) { jade = jade.runtime; } + +return function template(locals) { +var buf = []; +var jade_mixins = {}; +var locals_ = (locals || {}),coverageClass = locals_.coverageClass,cov = locals_.cov; +buf.push("

Coverage

  • overview" + (jade.escape((jade.interp = cov.coverage | 0) == null ? '' : jade.interp)) + "
  • "); +// iterate cov.dirs +;(function(){ + var $$obj = cov.dirs; + if ('number' == typeof $$obj.length) { + + for (var $index = 0, $$l = $$obj.length; $index < $$l; $index++) { + var dir = $$obj[$index]; + +buf.push("
  • " + (jade.escape(null == (jade.interp = dir.name || '.') ? "" : jade.interp)) + "
  • "); +// iterate dir.files +;(function(){ + var $$obj = dir.files; + if ('number' == typeof $$obj.length) { + + for (var $index = 0, $$l = $$obj.length; $index < $$l; $index++) { + var file = $$obj[$index]; + +buf.push("
  • " + (jade.escape((jade.interp = file.coverage | 0) == null ? '' : jade.interp)) + "" + (jade.escape(null == (jade.interp = file.relname) ? "" : jade.interp)) + "
  • "); + } + + } else { + var $$l = 0; + for (var $index in $$obj) { + $$l++; var file = $$obj[$index]; + +buf.push("
  • " + (jade.escape((jade.interp = file.coverage | 0) == null ? '' : jade.interp)) + "" + (jade.escape(null == (jade.interp = file.relname) ? "" : jade.interp)) + "
  • "); + } + + } +}).call(this); + + } + + } else { + var $$l = 0; + for (var $index in $$obj) { + $$l++; var dir = $$obj[$index]; + +buf.push("
  • " + (jade.escape(null == (jade.interp = dir.name || '.') ? "" : jade.interp)) + "
  • "); +// iterate dir.files +;(function(){ + var $$obj = dir.files; + if ('number' == typeof $$obj.length) { + + for (var $index = 0, $$l = $$obj.length; $index < $$l; $index++) { + var file = $$obj[$index]; + +buf.push("
  • " + (jade.escape((jade.interp = file.coverage | 0) == null ? '' : jade.interp)) + "" + (jade.escape(null == (jade.interp = file.relname) ? "" : jade.interp)) + "
  • "); + } + + } else { + var $$l = 0; + for (var $index in $$obj) { + $$l++; var file = $$obj[$index]; + +buf.push("
  • " + (jade.escape((jade.interp = file.coverage | 0) == null ? '' : jade.interp)) + "" + (jade.escape(null == (jade.interp = file.relname) ? "" : jade.interp)) + "
  • "); + } + + } +}).call(this); + + } + + } +}).call(this); + +buf.push("m
    " + (jade.escape((jade.interp = cov.coverage | 0) == null ? '' : jade.interp)) + "%
    " + (jade.escape(null == (jade.interp = cov.sloc) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = cov.hits) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = cov.misses) ? "" : jade.interp)) + "
    "); +// iterate cov.dirs +;(function(){ + var $$obj = cov.dirs; + if ('number' == typeof $$obj.length) { + + for (var $index = 0, $$l = $$obj.length; $index < $$l; $index++) { + var dir = $$obj[$index]; + +// iterate dir.files +;(function(){ + var $$obj = dir.files; + if ('number' == typeof $$obj.length) { + + for (var $index = 0, $$l = $$obj.length; $index < $$l; $index++) { + var file = $$obj[$index]; + +buf.push("
    " + (jade.escape(null == (jade.interp = file.filename) ? "" : jade.interp)) + "
    " + (jade.escape((jade.interp = file.coverage | 0) == null ? '' : jade.interp)) + "%
    " + (jade.escape(null == (jade.interp = file.sloc) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = file.hits) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = file.misses) ? "" : jade.interp)) + "
    "); +// iterate file.source +;(function(){ + var $$obj = file.source; + if ('number' == typeof $$obj.length) { + + for (var number = 0, $$l = $$obj.length; number < $$l; number++) { + var line = $$obj[number]; + +if ( line.coverage > 0) +{ +buf.push(""); +} +else if ( 0 === line.coverage) +{ +buf.push(""); +} +else +{ +buf.push(""); +} + } + + } else { + var $$l = 0; + for (var number in $$obj) { + $$l++; var line = $$obj[number]; + +if ( line.coverage > 0) +{ +buf.push(""); +} +else if ( 0 === line.coverage) +{ +buf.push(""); +} +else +{ +buf.push(""); +} + } + + } +}).call(this); + +buf.push("
    LineHitsSource
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.coverage) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "0" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source || ' ') ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.coverage) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "0" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source || ' ') ? "" : jade.interp)) + "
    "); + } + + } else { + var $$l = 0; + for (var $index in $$obj) { + $$l++; var file = $$obj[$index]; + +buf.push("
    " + (jade.escape(null == (jade.interp = file.filename) ? "" : jade.interp)) + "
    " + (jade.escape((jade.interp = file.coverage | 0) == null ? '' : jade.interp)) + "%
    " + (jade.escape(null == (jade.interp = file.sloc) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = file.hits) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = file.misses) ? "" : jade.interp)) + "
    "); +// iterate file.source +;(function(){ + var $$obj = file.source; + if ('number' == typeof $$obj.length) { + + for (var number = 0, $$l = $$obj.length; number < $$l; number++) { + var line = $$obj[number]; + +if ( line.coverage > 0) +{ +buf.push(""); +} +else if ( 0 === line.coverage) +{ +buf.push(""); +} +else +{ +buf.push(""); +} + } + + } else { + var $$l = 0; + for (var number in $$obj) { + $$l++; var line = $$obj[number]; + +if ( line.coverage > 0) +{ +buf.push(""); +} +else if ( 0 === line.coverage) +{ +buf.push(""); +} +else +{ +buf.push(""); +} + } + + } +}).call(this); + +buf.push("
    LineHitsSource
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.coverage) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "0" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source || ' ') ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.coverage) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "0" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source || ' ') ? "" : jade.interp)) + "
    "); + } + + } +}).call(this); + + } + + } else { + var $$l = 0; + for (var $index in $$obj) { + $$l++; var dir = $$obj[$index]; + +// iterate dir.files +;(function(){ + var $$obj = dir.files; + if ('number' == typeof $$obj.length) { + + for (var $index = 0, $$l = $$obj.length; $index < $$l; $index++) { + var file = $$obj[$index]; + +buf.push("
    " + (jade.escape(null == (jade.interp = file.filename) ? "" : jade.interp)) + "
    " + (jade.escape((jade.interp = file.coverage | 0) == null ? '' : jade.interp)) + "%
    " + (jade.escape(null == (jade.interp = file.sloc) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = file.hits) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = file.misses) ? "" : jade.interp)) + "
    "); +// iterate file.source +;(function(){ + var $$obj = file.source; + if ('number' == typeof $$obj.length) { + + for (var number = 0, $$l = $$obj.length; number < $$l; number++) { + var line = $$obj[number]; + +if ( line.coverage > 0) +{ +buf.push(""); +} +else if ( 0 === line.coverage) +{ +buf.push(""); +} +else +{ +buf.push(""); +} + } + + } else { + var $$l = 0; + for (var number in $$obj) { + $$l++; var line = $$obj[number]; + +if ( line.coverage > 0) +{ +buf.push(""); +} +else if ( 0 === line.coverage) +{ +buf.push(""); +} +else +{ +buf.push(""); +} + } + + } +}).call(this); + +buf.push("
    LineHitsSource
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.coverage) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "0" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source || ' ') ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.coverage) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "0" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source || ' ') ? "" : jade.interp)) + "
    "); + } + + } else { + var $$l = 0; + for (var $index in $$obj) { + $$l++; var file = $$obj[$index]; + +buf.push("
    " + (jade.escape(null == (jade.interp = file.filename) ? "" : jade.interp)) + "
    " + (jade.escape((jade.interp = file.coverage | 0) == null ? '' : jade.interp)) + "%
    " + (jade.escape(null == (jade.interp = file.sloc) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = file.hits) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = file.misses) ? "" : jade.interp)) + "
    "); +// iterate file.source +;(function(){ + var $$obj = file.source; + if ('number' == typeof $$obj.length) { + + for (var number = 0, $$l = $$obj.length; number < $$l; number++) { + var line = $$obj[number]; + +if ( line.coverage > 0) +{ +buf.push(""); +} +else if ( 0 === line.coverage) +{ +buf.push(""); +} +else +{ +buf.push(""); +} + } + + } else { + var $$l = 0; + for (var number in $$obj) { + $$l++; var line = $$obj[number]; + +if ( line.coverage > 0) +{ +buf.push(""); +} +else if ( 0 === line.coverage) +{ +buf.push(""); +} +else +{ +buf.push(""); +} + } + + } +}).call(this); + +buf.push("
    LineHitsSource
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.coverage) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "0" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source || ' ') ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.coverage) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "0" + (jade.escape(null == (jade.interp = line.source) ? "" : jade.interp)) + "
    " + (jade.escape(null == (jade.interp = number) ? "" : jade.interp)) + "" + (jade.escape(null == (jade.interp = line.source || ' ') ? "" : jade.interp)) + "
    "); + } + + } +}).call(this); + + } + + } +}).call(this); + +buf.push("
    ");;return buf.join(""); +} + +}); \ No newline at end of file diff --git a/test/utils/istanbul_reporter/reporter.js b/test/utils/istanbul_reporter/reporter.js new file mode 100644 index 0000000000000..5a618b090bef0 --- /dev/null +++ b/test/utils/istanbul_reporter/reporter.js @@ -0,0 +1,239 @@ +/** + * TODO: move this into it's own module, with it's own dependencies + */ + +require.config({ + paths: { + // jade runtime is required by the AMD wrapped jade templates as "jade" + jade: '/amd-wrap/node_modules/grunt-contrib-jade/node_modules/jade/runtime' + } +}); + +// fake fs module to make jade/runtime.js happy +define('fs', function () {}); + +// fake process obejct to make browserify-path happy +window.process = window.process || { cwd: function () { return '.'; }}; + +// the actual reporter module +define(function (require) { + + var _ = require('lodash'); + var $ = require('jquery'); + + var InsertionText = require('/amd-wrap/node_modules/istanbul/lib/util/insertion-text.js'); + var objUtils = require('/amd-wrap/node_modules/istanbul/lib/object-utils.js'); + // var annotate = require('/amd-wrap/node_modules/istanbul/lib/annotate.js'); + var Progress = require('/amd-wrap/node_modules/mocha/lib/browser/progress.js'); + var path = require('/amd-wrap/node_modules/path-browserify/index.js'); + + var template = require('./report.jade'); + + var Base = window.Mocha.reporters.Base; + + function IstanbulReporter(runner) { + // "inherit" the base reporters characteristics + Base.call(this, runner); + + var stats = this.stats; + var gotoFile = window.location.hash; + if (gotoFile) window.location.hash = ''; + + $(document.body) + .html('
    '); + + var canvas = document.getElementsByTagName('canvas')[0]; + var ctx; + var progress; + if (canvas.getContext) { + var ratio = window.devicePixelRatio || 1; + canvas.style.width = canvas.width; + canvas.style.height = canvas.height; + canvas.width *= ratio; + canvas.height *= ratio; + ctx = canvas.getContext('2d'); + ctx.scale(ratio, ratio); + progress = new Progress(); + progress.size(40); + } + + runner.on('test end', function () { + if (progress) { + progress + .update((stats.tests / this.total) * 100 || 0) + .draw(ctx); + } + }); + + runner.on('end', function () { + var stats = _.omit(this.stats, 'start', 'end', 'suites'); + + stats['create report'] = Date.now(); + $(document.body).empty().append($(createReport())); // attempt to force parsing immediately + stats['create report'] = Date.now() - stats['create report']; + + toSec(stats, 'create report'); + toSec(stats, 'duration'); + + linkNav(); + show(stats); + if (gotoFile) { + var header = document.getElementById(gotoFile.substring(1)); + if (header) { + window.location.hash = gotoFile; + document.body.scrollTop = header.offsetTop; + } + } + }); + } + + function createReport() { + var summary = objUtils.summarizeCoverage(window.__coverage__); + + var dirs = _(window.__coverage__) + .map(convertFile) + .groupBy(function (file) { + var dir = path.dirname(file.filename); + return dir === '.' ? '' : dir; + }) + .transform(function (dirs, files, dirname) { + _.each(files, function (file) { + file.relname = dirname ? path.relative(dirname, file.filename) : file.filename; + }); + + dirs.push({ + name: dirname, + files: files, + coverage: _.reduce(files, function (sum, file) { + return sum + file.coverage; + }, 0) / files.length + }); + }, []) + .sortBy('name') + .value(); + + return template({ + cov: { + dirs: dirs, + coverage: summary.lines.pct, + sloc: summary.lines.total, + hits: summary.lines.covered, + misses: summary.lines.total - summary.lines.covered + }, + dirname: path.dirname, + relative: path.relative, + coverageClass: function (coverage) { + if (coverage >= 75) return 'high'; + if (coverage >= 50) return 'medium'; + if (coverage >= 25) return 'low'; + return 'terrible'; + } + }); + } + + function convertFile(file) { + var summary = objUtils.summarizeFileCoverage(file); + var count = 0; + var structured = file.code.map(function (str) { + count += 1; + return { + line: count, + covered: null, + text: new InsertionText(str, true) + }; + }); + var html = ''; + + structured.unshift({ + line: 0, + covered: null, + text: new InsertionText('') + }); + + _.forOwn(file.l, function (count, lineNumber) { + structured[lineNumber].covered = count > 0 ? true : false; + }); + + // annotate.Lines(file, structured); + //note: order is important, since statements typically result in spanning the whole line and doing branches late + //causes mismatched tags + // annotate.Branches(file, structured); + // annotate.Functions(file, structured); + // annotate.Statements(file, structured); + + structured.shift(); + + var context = { + filename: file.path, + sloc: summary.lines.total, + coverage: summary.lines.pct, + hits: summary.lines.covered, + misses: summary.lines.total - summary.lines.covered, + source: _.map(structured, function (line, lineNumber) { + return { + coverage: file.l[line.line], + source: line.text + '' + }; + }) + }; + + return context; + + // writer.write(detailTemplate(context)); + // writer.write('\n'); + // writer.write(footerTemplate(templateData)); + } + + function linkNav() { + var headings = $('h2').toArray(); + + $(window).scroll(function (e) { + var heading = find(window.scrollY); + if (!heading) return; + var links = document.querySelectorAll('#menu a') + , link; + + for (var i = 0, len = links.length; i < len; ++i) { + link = links[i]; + link.className = link.getAttribute('href') === '#' + heading.id + ? 'active' + : ''; + } + }); + + function find(y) { + var i = headings.length + , heading; + + while (i--) { + heading = headings[i]; + if (y >= heading.offsetTop) { + return heading; + } + } + } + } + + function toSec(stats, prop) { + return stats[prop] = (stats[prop] / 1000).toFixed(2) + ' sec'; + } + + function show(info) { + var width = _(info).keys().sortBy('length').pop().length; + + $('
    ')
    +      .addClass('coverage-stats')
    +      .appendTo('#menu')
    +      .text(
    +        _.map(info, function (val, name) {
    +          var row = val + ' - ' + name;
    +          if (width - name.length) {
    +            row += (new Array(width - name.length + 1)).join(' ');
    +          }
    +          return row;
    +        }).join('\n')
    +      );
    +  }
    +
    +  return IstanbulReporter;
    +});
    \ No newline at end of file