-
Notifications
You must be signed in to change notification settings - Fork 23
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Getting weird coverage reports #56
Comments
If you dig into the code generated by babel at the
|
So the generated code looks like this in case it is useful to you (with 6.5.x version of babel): 'use strict';var _createClass = function () {function defineProperties(target, props) {for (var i = 0; i < props.length; i++) {var descriptor = props[i];descriptor.enumerable = descriptor.enumerable || false;descriptor.configurable = true;if ("value" in descriptor) descriptor.writable = true;Object.defineProperty(target, descriptor.key, descriptor);}}return function (Constructor, protoProps, staticProps) {if (protoProps) defineProperties(Constructor.prototype, protoProps);if (staticProps) defineProperties(Constructor, staticProps);return Constructor;};}();var _react = __webpack_require__(153);var React = _interopRequireWildcard(_react);var _headerComponent = __webpack_require__(254);var _headerComponent2 = _interopRequireDefault(_headerComponent);function _interopRequireDefault(obj) {return obj && obj.__esModule ? obj : { default: obj };}function _interopRequireWildcard(obj) {if (obj && obj.__esModule) {return obj;} else {var newObj = {};if (obj != null) {for (var key in obj) {if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key];}}newObj.default = obj;return newObj;}}function _classCallCheck(instance, Constructor) {if (!(instance instanceof Constructor)) {throw new TypeError("Cannot call a class as a function");}}function _possibleConstructorReturn(self, call) {if (!self) {throw new ReferenceError("this hasn't been initialised - super() hasn't been called");}return call && (typeof call === "object" || typeof call === "function") ? call : self;}function _inherits(subClass, superClass) {if (typeof superClass !== "function" && superClass !== null) {throw new TypeError("Super expression must either be null or a function, not " + typeof superClass);}subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } });if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass;}var
Application = function (_React$Component) {_inherits(Application, _React$Component);
function Application(props) {_classCallCheck(this, Application);return _possibleConstructorReturn(this, Object.getPrototypeOf(Application).call(this,
props));}_createClass(Application, [{ key: 'render', value: function render()
{
return (
React.createElement('div', { className: 'application' },
React.createElement(_headerComponent2.default, null),
this.props.children));} }]);return Application;}(React.Component);
Application.displayName = 'Application';
module.exports = Application; I also tried updating the following libraries to the listed version: and got the same result. As for my other files, they are compilers.js var babel = require('babel-core/register');
var noop = function(){return null;};
//assets the webpack parsers that node can't
require.extensions['.html'] = noop;
require.extensions['.scss'] = noop;
require.extensions['.jpg'] = noop;
require.extensions['.jpeg'] = noop;
require.extensions['.png'] = noop; mocha-generators.js var co = require('co');
var mocha = require('mocha');
var Runnable = mocha.Runnable;
var run = Runnable.prototype.run;
Runnable.prototype.run = function(fn) {
if(this.fn.constructor.name === 'GeneratorFunction') {
this.fn = co(this.fn);
this.sync = !(this.async = true);
}
return run.call(this, fn);
}; setup.js //NOTE: setup expect as global
import {expect} from 'chai';
global.expect = expect;
//NOTE: setup JSDOM for test running
import * as jsdom from 'jsdom';
var exposedProperties = ['window', 'navigator', 'document'];
global.document = jsdom.jsdom('');
global.window = document.defaultView;
Object.keys(document.defaultView).forEach((property) => {
if (typeof global[property] === 'undefined') {
exposedProperties.push(property);
global[property] = document.defaultView[property];
}
});
global.navigator = {
userAgent: 'node.js'
};
//NOTE: need to include the i18n testing file
global.window.i18n = require('./i18n-node.js');
//NOTE: include all files that should have coverage numbers on them in-case there are files the specs don't trigger inclusion
import globArray from 'glob-array';
var filesForCoverage = [
process.cwd() + '/web/app/**/*.js',
process.cwd() + '/web/app/**/*.jsx',
//NOTE: pages are tested through ui tests, not unit tests
'!' + process.cwd() + '/web/app/pages/**/*.js',
'!' + process.cwd() + '/web/app/pages/**/*.jsx',
//NOTE: mocks are not that important to coverage test
'!' + process.cwd() + '/web/app/mock/**/*.js',
//NOTE: 3rd party libraries that there is no point in testing
'!' + process.cwd() + '/web/app/misc/ua-parser.js',
//router based stuff seems to cause issue with needing the history location for react-router so just skip those file
'!' + process.cwd() + '/web/app/application.jsx',
'!' + process.cwd() + '/web/app/router.jsx',
'!' + process.cwd() + '/web/app/routes.jsx'
];
globArray.sync(filesForCoverage).forEach(function(filePath) {
require(filePath);
}); Let me know if you see anything weird that might be causing this issue in these files or if I can provide any other information to help you debug this issue. |
I am suspecting something strange could be happening in your filesForCoverage setup step. Perhaps try omitting that part to see if it helps. I'll go ahead and update the babel dependency also, just to make sure it isn't related. |
Still the same issue with that block of code relating to filesForCoverage removed. I mean since it just includes files it should not cause any issues (not to mention I don't think there is any other way of make sure that a file is included in the report incase the unit tests don't trigger the inclusion of the all the files). |
@ryanzec The preferred method of making sure all files are covered is using a common I have published |
Also, one other thing you can try is |
It was the I do agree that it is generally bad to do import * for the reasons you mentioned but I was unaware I could import react and react-dom in that way. I have updated by code and guess I will close this issue for now. If I find it again, I will re-open. |
Ok, getting this issue again. This seems to be an issue when using and the prevent-double-click.constants.js is:
This is a case where I think using the |
Your only option here is to look at the generated code and try to figure out why it is doing what it is doing. Babel is literally creating statements that are not being run, so Istanbul is doing its job. This project has no control over how Istanbul covers or how Babel compiles. With that being said, you can try a few things:
|
I've had the same issue using |
I guess I will use the 1.x.x branch of the regular istanbul and see how using that goes. |
@ryanzec do you by any chance use retainLines : true in your babelrc and in the context of instrumentation of your sources? Nonetheless, I also get weird coverage results. The report shows 100% branch/function/line whatever coverage, but when digging into the generated HTML report it shows that multiple lines/branches are not being covered at all. This happens with [email protected] (babel-core 6.7.7) My guess is that it is the ternary operator that is causing this in this case. And, after refactoring of that ternary, I get the following, more sane report Looking at the generated code, though
shows that nothing special is happening here. Perhaps istanbul should be upgraded again 👯 |
@silkentrance @ryanzec I actually have not been able to get completely correct coverage from babel-istanbul, isparta, or istanbul's aphla. I believe there is just some inherent weirdness with the source maps generated by babel. Both babel-istanbul and isparta work by skipping coverage in the instrumentation layer based on source maps. Istanbul's alpha works in the reporting layer, but also uses source maps. My hunch is that some of the transformers do not completely play well with source mapping. That is, they probably don't clean up after themselves very well. I've pinpointed babel-plugin-transform-es2015-typeof-symbol as one that causes trouble for me. TLDR: Any implementation that uses source maps for coverage are at the mercy of the quality of the source map. |
@jmcriffey yeah, however, and in my case I think that it maybe has something to do with the HTML reporter, though. Since, when I go through my test cases and sources I think that the counters to the left are correct, but that the HTML reporter somehow gets it all wrong. We will have to wait for an istanbul update, I think. |
While it is a newer project https://github.com/dtinth/babel-plugin-__coverage__ was able to produce accurate HTML reports for us. |
So I am trying to use this to do coverage reports on my es6 code with babel however I am getting weird results. For example, I see this in the html output of:
Line 1 is weird as it is expecting to test a if / else / else statement when there is none. Lines 4 and 5 are weird as I don't know what it is talking about as I do render an Application component in my tests. The command I am using to run my tests is:
and my mocha.opts file is:
Am I missing something are are there false failure showing up?
The text was updated successfully, but these errors were encountered: