creevey 0.9.0-beta.1 → 0.9.0-beta.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (145) hide show
  1. package/lib/cjs/cli.js +5 -0
  2. package/lib/cjs/client/addon/Manager.js +412 -0
  3. package/lib/cjs/client/addon/components/Addon.js +76 -0
  4. package/lib/cjs/client/addon/components/Icons.js +42 -0
  5. package/lib/cjs/client/addon/components/Panel.js +68 -0
  6. package/lib/cjs/client/addon/components/TestSelect.js +63 -0
  7. package/lib/cjs/client/addon/components/Tools.js +114 -0
  8. package/lib/cjs/client/addon/decorator.js +11 -0
  9. package/lib/cjs/client/addon/preset.js +81 -0
  10. package/lib/cjs/client/addon/readyForCapture.js +12 -0
  11. package/lib/cjs/client/addon/register.js +96 -0
  12. package/lib/cjs/client/addon/utils.js +38 -0
  13. package/lib/cjs/client/addon/withCreevey.js +531 -0
  14. package/lib/cjs/client/shared/components/ImagesView/BlendView.js +85 -0
  15. package/lib/cjs/client/shared/components/ImagesView/ImagesView.js +88 -0
  16. package/lib/cjs/client/shared/components/ImagesView/SideBySideView.js +176 -0
  17. package/lib/cjs/client/shared/components/ImagesView/SlideView.js +179 -0
  18. package/lib/cjs/client/shared/components/ImagesView/SwapView.js +110 -0
  19. package/lib/cjs/client/shared/components/ImagesView/index.js +45 -0
  20. package/lib/cjs/client/shared/components/PageFooter/PageFooter.js +46 -0
  21. package/lib/cjs/client/shared/components/PageFooter/Paging.js +98 -0
  22. package/lib/cjs/client/shared/components/PageHeader/ImagePreview.js +78 -0
  23. package/lib/cjs/client/shared/components/PageHeader/PageHeader.js +144 -0
  24. package/lib/cjs/client/shared/components/ResultsPage.js +173 -0
  25. package/lib/cjs/client/shared/creeveyClientApi.js +103 -0
  26. package/lib/cjs/client/shared/helpers.js +482 -0
  27. package/lib/cjs/client/shared/viewMode.js +17 -0
  28. package/lib/cjs/client/web/index.html +19 -0
  29. package/lib/cjs/creevey.js +71 -0
  30. package/lib/cjs/index.js +62 -0
  31. package/lib/cjs/server/config.js +96 -0
  32. package/lib/cjs/server/docker.js +150 -0
  33. package/lib/cjs/server/extract.js +50 -0
  34. package/lib/cjs/server/index.js +83 -0
  35. package/lib/cjs/server/loaders/babel/creevey-plugin.js +88 -0
  36. package/lib/cjs/server/loaders/babel/helpers.js +479 -0
  37. package/lib/cjs/server/loaders/babel/register.js +126 -0
  38. package/lib/cjs/server/loaders/hooks/mdx.js +30 -0
  39. package/lib/cjs/server/loaders/hooks/svelte.js +65 -0
  40. package/lib/cjs/server/loaders/webpack/compile.js +286 -0
  41. package/lib/cjs/server/loaders/webpack/creevey-loader.js +174 -0
  42. package/lib/cjs/server/loaders/webpack/dummy-hmr.js +44 -0
  43. package/lib/cjs/server/loaders/webpack/mdx-loader.js +72 -0
  44. package/lib/cjs/server/loaders/webpack/start.js +41 -0
  45. package/lib/cjs/server/logger.js +47 -0
  46. package/lib/cjs/server/master/api.js +71 -0
  47. package/lib/cjs/server/master/index.js +146 -0
  48. package/lib/cjs/server/master/master.js +57 -0
  49. package/lib/cjs/server/master/pool.js +206 -0
  50. package/lib/cjs/server/master/runner.js +294 -0
  51. package/lib/cjs/server/master/server.js +129 -0
  52. package/lib/cjs/server/messages.js +266 -0
  53. package/lib/cjs/server/selenium/browser.js +680 -0
  54. package/lib/cjs/server/selenium/index.js +31 -0
  55. package/lib/cjs/server/selenium/selenoid.js +174 -0
  56. package/lib/cjs/server/stories.js +170 -0
  57. package/lib/cjs/server/storybook/entry.js +68 -0
  58. package/lib/cjs/server/storybook/helpers.js +165 -0
  59. package/lib/cjs/server/storybook/providers/browser.js +78 -0
  60. package/lib/cjs/server/storybook/providers/hybrid.js +84 -0
  61. package/lib/cjs/server/storybook/providers/nodejs.js +239 -0
  62. package/lib/cjs/server/testsFiles/parser.js +72 -0
  63. package/lib/cjs/server/testsFiles/register.js +48 -0
  64. package/lib/cjs/server/update.js +83 -0
  65. package/lib/cjs/server/utils.js +185 -0
  66. package/lib/cjs/server/worker/chai-image.js +142 -0
  67. package/lib/cjs/server/worker/helpers.js +69 -0
  68. package/lib/cjs/server/worker/index.js +15 -0
  69. package/lib/cjs/server/worker/reporter.js +120 -0
  70. package/lib/cjs/server/worker/worker.js +278 -0
  71. package/lib/cjs/shared.js +124 -0
  72. package/lib/cjs/types.js +74 -0
  73. package/lib/esm/cli.js +4 -0
  74. package/lib/esm/client/addon/Manager.js +396 -0
  75. package/lib/esm/client/addon/components/Addon.js +58 -0
  76. package/lib/esm/client/addon/components/Icons.js +27 -0
  77. package/lib/esm/client/addon/components/Panel.js +49 -0
  78. package/lib/esm/client/addon/components/TestSelect.js +49 -0
  79. package/lib/esm/client/addon/components/Tools.js +91 -0
  80. package/lib/esm/client/addon/decorator.js +2 -0
  81. package/lib/esm/client/addon/preset.js +56 -0
  82. package/lib/esm/client/addon/readyForCapture.js +5 -0
  83. package/lib/esm/client/addon/register.js +75 -0
  84. package/lib/esm/client/addon/utils.js +31 -0
  85. package/lib/esm/client/addon/withCreevey.js +509 -0
  86. package/lib/esm/client/shared/components/ImagesView/BlendView.js +63 -0
  87. package/lib/esm/client/shared/components/ImagesView/ImagesView.js +65 -0
  88. package/lib/esm/client/shared/components/ImagesView/SideBySideView.js +151 -0
  89. package/lib/esm/client/shared/components/ImagesView/SlideView.js +154 -0
  90. package/lib/esm/client/shared/components/ImagesView/SwapView.js +88 -0
  91. package/lib/esm/client/shared/components/ImagesView/index.js +5 -0
  92. package/lib/esm/client/shared/components/PageFooter/PageFooter.js +32 -0
  93. package/lib/esm/client/shared/components/PageFooter/Paging.js +84 -0
  94. package/lib/esm/client/shared/components/PageHeader/ImagePreview.js +64 -0
  95. package/lib/esm/client/shared/components/PageHeader/PageHeader.js +120 -0
  96. package/lib/esm/client/shared/components/ResultsPage.js +143 -0
  97. package/lib/esm/client/shared/creeveyClientApi.js +94 -0
  98. package/lib/esm/client/shared/helpers.js +424 -0
  99. package/lib/esm/client/shared/viewMode.js +6 -0
  100. package/lib/esm/creevey.js +56 -0
  101. package/lib/esm/index.js +7 -0
  102. package/lib/esm/server/config.js +73 -0
  103. package/lib/esm/server/docker.js +123 -0
  104. package/lib/esm/server/extract.js +34 -0
  105. package/lib/esm/server/index.js +64 -0
  106. package/lib/esm/server/loaders/babel/creevey-plugin.js +74 -0
  107. package/lib/esm/server/loaders/babel/helpers.js +462 -0
  108. package/lib/esm/server/loaders/babel/register.js +105 -0
  109. package/lib/esm/server/loaders/hooks/mdx.js +15 -0
  110. package/lib/esm/server/loaders/hooks/svelte.js +49 -0
  111. package/lib/esm/server/loaders/webpack/compile.js +263 -0
  112. package/lib/esm/server/loaders/webpack/creevey-loader.js +153 -0
  113. package/lib/esm/server/loaders/webpack/dummy-hmr.js +36 -0
  114. package/lib/esm/server/loaders/webpack/mdx-loader.js +58 -0
  115. package/lib/esm/server/loaders/webpack/start.js +27 -0
  116. package/lib/esm/server/logger.js +20 -0
  117. package/lib/esm/server/master/api.js +60 -0
  118. package/lib/esm/server/master/index.js +125 -0
  119. package/lib/esm/server/master/master.js +38 -0
  120. package/lib/esm/server/master/pool.js +187 -0
  121. package/lib/esm/server/master/runner.js +272 -0
  122. package/lib/esm/server/master/server.js +105 -0
  123. package/lib/esm/server/messages.js +234 -0
  124. package/lib/esm/server/selenium/browser.js +647 -0
  125. package/lib/esm/server/selenium/index.js +2 -0
  126. package/lib/esm/server/selenium/selenoid.js +151 -0
  127. package/lib/esm/server/stories.js +151 -0
  128. package/lib/esm/server/storybook/entry.js +44 -0
  129. package/lib/esm/server/storybook/helpers.js +106 -0
  130. package/lib/esm/server/storybook/providers/browser.js +61 -0
  131. package/lib/esm/server/storybook/providers/hybrid.js +64 -0
  132. package/lib/esm/server/storybook/providers/nodejs.js +217 -0
  133. package/lib/esm/server/testsFiles/parser.js +50 -0
  134. package/lib/esm/server/testsFiles/register.js +35 -0
  135. package/lib/esm/server/update.js +65 -0
  136. package/lib/esm/server/utils.js +146 -0
  137. package/lib/esm/server/worker/chai-image.js +130 -0
  138. package/lib/esm/server/worker/helpers.js +60 -0
  139. package/lib/esm/server/worker/index.js +1 -0
  140. package/lib/esm/server/worker/reporter.js +98 -0
  141. package/lib/esm/server/worker/worker.js +248 -0
  142. package/lib/esm/shared.js +93 -0
  143. package/lib/esm/types.js +43 -0
  144. package/lib/types/index.d.ts +2 -4
  145. package/package.json +1 -1
@@ -0,0 +1,185 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.shouldSkip = shouldSkip;
7
+ exports.shutdownWorkers = shutdownWorkers;
8
+ exports.shutdown = shutdown;
9
+ exports.getCreeveyCache = getCreeveyCache;
10
+ exports.runSequence = runSequence;
11
+ exports.testsToImages = testsToImages;
12
+ exports.removeProps = removeProps;
13
+ exports.readDirRecursive = readDirRecursive;
14
+ exports.downloadBinary = exports.isInsideDocker = exports.skipOptionKeys = exports.extensions = exports.LOCALHOST_REGEXP = exports.isShuttingDown = void 0;
15
+
16
+ var _fs = require("fs");
17
+
18
+ var _cluster = _interopRequireDefault(require("cluster"));
19
+
20
+ var _types = require("../types");
21
+
22
+ var _messages = require("./messages");
23
+
24
+ var _findCacheDir = _interopRequireDefault(require("find-cache-dir"));
25
+
26
+ var _https = require("https");
27
+
28
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
29
+
30
+ const isShuttingDown = {
31
+ current: false
32
+ };
33
+ exports.isShuttingDown = isShuttingDown;
34
+ const LOCALHOST_REGEXP = /(localhost|127\.0\.0\.1)/i;
35
+ exports.LOCALHOST_REGEXP = LOCALHOST_REGEXP;
36
+ const extensions = ['.js', '.jsx', '.ts', '.tsx'];
37
+ exports.extensions = extensions;
38
+ const skipOptionKeys = ['in', 'kinds', 'stories', 'tests', 'reason'];
39
+ exports.skipOptionKeys = skipOptionKeys;
40
+
41
+ function matchBy(pattern, value) {
42
+ return typeof pattern == 'string' && pattern == value || Array.isArray(pattern) && pattern.includes(value) || pattern instanceof RegExp && pattern.test(value) || !(0, _types.isDefined)(pattern);
43
+ }
44
+
45
+ function shouldSkip(browser, meta, skipOptions, test) {
46
+ if (typeof skipOptions != 'object') {
47
+ return skipOptions;
48
+ }
49
+
50
+ if (Array.isArray(skipOptions)) {
51
+ for (const skip of skipOptions) {
52
+ const reason = shouldSkip(browser, meta, skip, test);
53
+ if (reason) return reason;
54
+ }
55
+
56
+ return false;
57
+ }
58
+
59
+ let hasSkipOptionKeys = false;
60
+
61
+ for (const skipKey in skipOptions) {
62
+ if (skipOptionKeys.includes(skipKey)) {
63
+ hasSkipOptionKeys = true;
64
+ continue;
65
+ }
66
+
67
+ const reason = shouldSkip(browser, meta, {
68
+ reason: skipKey,
69
+ ...skipOptions[skipKey]
70
+ }, test);
71
+ if (reason) return reason;
72
+ }
73
+
74
+ if (!hasSkipOptionKeys) return false;
75
+ const {
76
+ in: browsers,
77
+ kinds,
78
+ stories,
79
+ tests,
80
+ reason = true
81
+ } = skipOptions;
82
+ const {
83
+ kind,
84
+ story
85
+ } = meta;
86
+ const skipByBrowser = matchBy(browsers, browser);
87
+ const skipByKind = matchBy(kinds, kind);
88
+ const skipByStory = matchBy(stories, story);
89
+ const skipByTest = !(0, _types.isDefined)(test) || matchBy(tests, test);
90
+ return skipByBrowser && skipByKind && skipByStory && skipByTest && reason;
91
+ }
92
+
93
+ async function shutdownWorkers() {
94
+ isShuttingDown.current = true;
95
+ await Promise.all(Object.values(_cluster.default.workers).filter(_types.isDefined).filter(worker => worker.isConnected()).map(worker => new Promise(resolve => {
96
+ const timeout = setTimeout(() => worker.kill(), 10000);
97
+ worker.on('exit', () => {
98
+ clearTimeout(timeout);
99
+ resolve();
100
+ });
101
+ (0, _messages.sendShutdownMessage)(worker);
102
+ })));
103
+ (0, _messages.emitShutdownMessage)();
104
+ }
105
+
106
+ function shutdown() {
107
+ // eslint-disable-next-line no-process-exit
108
+ process.exit();
109
+ }
110
+
111
+ function getCreeveyCache() {
112
+ return (0, _findCacheDir.default)({
113
+ name: 'creevey',
114
+ cwd: __dirname
115
+ });
116
+ }
117
+
118
+ async function runSequence(seq, predicate) {
119
+ for (const fn of seq) {
120
+ if (predicate()) await fn();
121
+ }
122
+ }
123
+
124
+ function testsToImages(tests) {
125
+ return new Set([].concat(...tests.filter(_types.isDefined).map(({
126
+ browser,
127
+ testName,
128
+ storyPath,
129
+ results
130
+ }) => {
131
+ var _results$slice$0$imag, _results$slice$;
132
+
133
+ return Object.keys((_results$slice$0$imag = results === null || results === void 0 ? void 0 : (_results$slice$ = results.slice(-1)[0]) === null || _results$slice$ === void 0 ? void 0 : _results$slice$.images) !== null && _results$slice$0$imag !== void 0 ? _results$slice$0$imag : {}).map(image => `${[...storyPath, testName, browser, browser == image ? undefined : image].filter(_types.isDefined).join('/')}.png`);
134
+ })));
135
+ } // https://tuhrig.de/how-to-know-you-are-inside-a-docker-container/
136
+
137
+
138
+ const isInsideDocker = (0, _fs.existsSync)('/proc/1/cgroup') && /docker/.test((0, _fs.readFileSync)('/proc/1/cgroup', 'utf8'));
139
+ exports.isInsideDocker = isInsideDocker;
140
+
141
+ const downloadBinary = (downloadUrl, destination) => new Promise((resolve, reject) => (0, _https.get)(downloadUrl, response => {
142
+ var _response$statusCode2;
143
+
144
+ if (response.statusCode == 302) {
145
+ var _response$statusCode;
146
+
147
+ const {
148
+ location
149
+ } = response.headers;
150
+ if (!location) return reject(new Error(`Couldn't download selenoid. Status code: ${(_response$statusCode = response.statusCode) !== null && _response$statusCode !== void 0 ? _response$statusCode : 'UNKNOWN'}`));
151
+ return resolve(downloadBinary(location, destination));
152
+ }
153
+
154
+ if (response.statusCode != 200) return reject(new Error(`Couldn't download selenoid. Status code: ${(_response$statusCode2 = response.statusCode) !== null && _response$statusCode2 !== void 0 ? _response$statusCode2 : 'UNKNOWN'}`));
155
+ const fileStream = (0, _fs.createWriteStream)(destination);
156
+ response.pipe(fileStream);
157
+ fileStream.on('finish', () => {
158
+ fileStream.close();
159
+ resolve();
160
+ });
161
+ fileStream.on('error', error => {
162
+ (0, _fs.unlink)(destination, _types.noop);
163
+ reject(error);
164
+ });
165
+ }));
166
+
167
+ exports.downloadBinary = downloadBinary;
168
+
169
+ function removeProps(obj, propPath) {
170
+ const [prop, ...restPath] = propPath;
171
+
172
+ if (restPath.length > 0) {
173
+ if (typeof prop == 'string') obj[prop] && removeProps(obj[prop], restPath);
174
+ if ((0, _types.isFunction)(prop)) Object.keys(obj).filter(prop).forEach(key => obj[key] && removeProps(obj[key], restPath));
175
+ } else {
176
+ if (typeof prop == 'string') delete obj[prop];
177
+ if ((0, _types.isFunction)(prop)) Object.keys(obj).filter(prop).forEach(key => delete obj[key]);
178
+ }
179
+ }
180
+
181
+ function readDirRecursive(dirPath) {
182
+ return [].concat(...(0, _fs.readdirSync)(dirPath, {
183
+ withFileTypes: true
184
+ }).map(dirent => dirent.isDirectory() ? readDirRecursive(`${dirPath}/${dirent.name}`) : [`${dirPath}/${dirent.name}`]));
185
+ }
@@ -0,0 +1,142 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = _default;
7
+
8
+ var _pngjs = require("pngjs");
9
+
10
+ var _pixelmatch = _interopRequireDefault(require("pixelmatch"));
11
+
12
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
13
+
14
+ function normalizeImageSize(image, width, height) {
15
+ const normalizedImage = Buffer.alloc(4 * width * height);
16
+
17
+ for (let y = 0; y < height; y++) {
18
+ for (let x = 0; x < width; x++) {
19
+ const i = (y * width + x) * 4;
20
+
21
+ if (x < image.width && y < image.height) {
22
+ const j = (y * image.width + x) * 4;
23
+ normalizedImage[i + 0] = image.data[j + 0];
24
+ normalizedImage[i + 1] = image.data[j + 1];
25
+ normalizedImage[i + 2] = image.data[j + 2];
26
+ normalizedImage[i + 3] = image.data[j + 3];
27
+ } else {
28
+ normalizedImage[i + 0] = 0;
29
+ normalizedImage[i + 1] = 0;
30
+ normalizedImage[i + 2] = 0;
31
+ normalizedImage[i + 3] = 0;
32
+ }
33
+ }
34
+ }
35
+
36
+ return normalizedImage;
37
+ }
38
+
39
+ function hasDiffPixels(diff) {
40
+ for (let i = 0; i < diff.length; i += 4) {
41
+ if (diff[i + 0] == 255 && diff[i + 1] == 0 && diff[i + 2] == 0 && diff[i + 3] == 255) return true;
42
+ }
43
+
44
+ return false;
45
+ }
46
+
47
+ function compareImages(expect, actual, diffOptions) {
48
+ const expectImage = _pngjs.PNG.sync.read(expect);
49
+
50
+ const actualImage = _pngjs.PNG.sync.read(actual);
51
+
52
+ const width = Math.max(actualImage.width, expectImage.width);
53
+ const height = Math.max(actualImage.height, expectImage.height);
54
+ const diffImage = new _pngjs.PNG({
55
+ width,
56
+ height
57
+ });
58
+ let actualImageData = actualImage.data;
59
+
60
+ if (actualImage.width < width || actualImage.height < height) {
61
+ actualImageData = normalizeImageSize(actualImage, width, height);
62
+ }
63
+
64
+ let expectImageData = expectImage.data;
65
+
66
+ if (expectImage.width < width || expectImage.height < height) {
67
+ expectImageData = normalizeImageSize(expectImage, width, height);
68
+ }
69
+
70
+ (0, _pixelmatch.default)(expectImageData, actualImageData, diffImage.data, width, height, diffOptions);
71
+ return {
72
+ isEqual: !hasDiffPixels(diffImage.data),
73
+ diff: _pngjs.PNG.sync.write(diffImage)
74
+ };
75
+ }
76
+
77
+ function _default(getExpected, diffOptions) {
78
+ return function chaiImage({
79
+ Assertion
80
+ }, utils) {
81
+ async function assertImage(actual, imageName) {
82
+ let onCompare = () => Promise.resolve();
83
+
84
+ let expected = await getExpected(imageName);
85
+ if (!(expected instanceof Buffer) && expected != null) ({
86
+ expected,
87
+ onCompare
88
+ } = expected);
89
+
90
+ if (expected == null) {
91
+ await onCompare(actual);
92
+ return imageName ? `Expected image '${imageName}' does not exists` : 'Expected image does not exists';
93
+ }
94
+
95
+ if (actual.equals(expected)) return await onCompare(actual);
96
+ const {
97
+ isEqual,
98
+ diff
99
+ } = compareImages(expected, actual, diffOptions);
100
+ if (isEqual) return await onCompare(actual);
101
+ await onCompare(actual, expected, diff);
102
+ return imageName ? `Expected image '${imageName}' to match` : 'Expected image to match';
103
+ }
104
+
105
+ utils.addMethod(Assertion.prototype, 'matchImage', async function matchImage(imageName) {
106
+ const actual = utils.flag(this, 'object');
107
+ const errorMessage = await assertImage(typeof actual == 'string' ? Buffer.from(actual, 'base64') : actual, imageName);
108
+
109
+ if (errorMessage) {
110
+ throw createImageError(imageName ? {
111
+ [imageName]: errorMessage
112
+ } : errorMessage);
113
+ }
114
+ });
115
+ utils.addMethod(Assertion.prototype, 'matchImages', async function matchImages() {
116
+ const errors = {};
117
+ await Promise.all(Object.entries(utils.flag(this, 'object')).map(async ([imageName, imageOrBase64]) => {
118
+ let errorMessage;
119
+
120
+ try {
121
+ errorMessage = await assertImage(typeof imageOrBase64 == 'string' ? Buffer.from(imageOrBase64, 'base64') : imageOrBase64, imageName);
122
+ } catch (error) {
123
+ errorMessage = error.stack;
124
+ }
125
+
126
+ if (errorMessage) {
127
+ errors[imageName] = errorMessage;
128
+ }
129
+ }));
130
+
131
+ if (Object.keys(errors).length > 0) {
132
+ throw createImageError(errors);
133
+ }
134
+ });
135
+ };
136
+ }
137
+
138
+ function createImageError(imageErrors) {
139
+ const error = new Error('Expected image to match');
140
+ error.images = imageErrors;
141
+ return error;
142
+ }
@@ -0,0 +1,69 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.addTestsFromStories = addTestsFromStories;
7
+
8
+ var _mocha = require("mocha");
9
+
10
+ var _types = require("../../types");
11
+
12
+ var _stories = require("../stories");
13
+
14
+ function findOrCreateSuite(name, parent) {
15
+ const suite = parent.suites.find(({
16
+ title
17
+ }) => title == name) || new _mocha.Suite(name, parent.ctx);
18
+
19
+ if (!suite.parent) {
20
+ suite.parent = parent;
21
+ parent.addSuite(suite);
22
+ }
23
+
24
+ return suite;
25
+ }
26
+
27
+ function createTest(name, fn, skip = false) {
28
+ const test = new _mocha.Test(name, skip ? undefined : fn);
29
+ test.pending = Boolean(skip); // NOTE Can't define skip reason in mocha https://github.com/mochajs/mocha/issues/2026
30
+
31
+ test.skipReason = skip;
32
+ return test;
33
+ }
34
+
35
+ function addTest(rootSuite, test) {
36
+ const [testName, ...suitePath] = [...test.storyPath, test.testName].reverse().filter(_types.isDefined);
37
+ const suite = suitePath.reduceRight((subSuite, suiteName) => findOrCreateSuite(suiteName, subSuite), rootSuite);
38
+ const mochaTest = createTest(testName, test.fn, test.skip);
39
+ suite.addTest(mochaTest);
40
+ mochaTest.ctx = Object.setPrototypeOf({
41
+ id: test.id,
42
+ story: test.story
43
+ }, suite.ctx);
44
+ return mochaTest;
45
+ }
46
+
47
+ function removeTestOrSuite(testOrSuite) {
48
+ const {
49
+ parent
50
+ } = testOrSuite;
51
+ if (!parent) return;
52
+ if (testOrSuite instanceof _mocha.Test) parent.tests = parent.tests.filter(test => test != testOrSuite);
53
+ if (testOrSuite instanceof _mocha.Suite) parent.suites = parent.suites.filter(suite => suite != testOrSuite);
54
+ if (parent.tests.length == 0 && parent.suites.length == 0) removeTestOrSuite(parent);
55
+ }
56
+
57
+ async function addTestsFromStories(rootSuite, config, {
58
+ browser,
59
+ ...options
60
+ }) {
61
+ const mochaTestsById = new Map();
62
+ const tests = await (0, _stories.loadTestsFromStories)([browser], listener => config.storiesProvider(config, options, listener), testsDiff => Object.entries(testsDiff).forEach(([id, newTest]) => {
63
+ const oldTest = mochaTestsById.get(id);
64
+ mochaTestsById.delete(id);
65
+ if (oldTest) removeTestOrSuite(oldTest);
66
+ if (newTest) mochaTestsById.set(id, addTest(rootSuite, newTest));
67
+ }));
68
+ Object.values(tests).filter(_types.isDefined).forEach(test => mochaTestsById.set(test.id, addTest(rootSuite, test)));
69
+ }
@@ -0,0 +1,15 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ Object.defineProperty(exports, "default", {
7
+ enumerable: true,
8
+ get: function () {
9
+ return _worker.default;
10
+ }
11
+ });
12
+
13
+ var _worker = _interopRequireDefault(require("./worker"));
14
+
15
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
@@ -0,0 +1,120 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.TeamcityReporter = exports.CreeveyReporter = void 0;
7
+
8
+ var _chalk = _interopRequireDefault(require("chalk"));
9
+
10
+ var _mocha = require("mocha");
11
+
12
+ var _loglevelPluginPrefix = _interopRequireDefault(require("loglevel-plugin-prefix"));
13
+
14
+ var _types = require("../../types");
15
+
16
+ var _logger = require("../logger");
17
+
18
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
19
+
20
+ function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
21
+
22
+ const testLevels = {
23
+ INFO: _chalk.default.green('PASS'),
24
+ WARN: _chalk.default.yellow('START'),
25
+ ERROR: _chalk.default.red('FAIL')
26
+ };
27
+
28
+ class CreeveyReporter extends _mocha.reporters.Base {
29
+ constructor(runner, options) {
30
+ super(runner);
31
+ const {
32
+ sessionId,
33
+ topLevelSuite
34
+ } = options.reporterOptions;
35
+ const testLogger = (0, _logger.getLogger)(topLevelSuite);
36
+
37
+ _loglevelPluginPrefix.default.apply(testLogger, {
38
+ format(level) {
39
+ return `${testLevels[level]} => (${topLevelSuite}:${_chalk.default.gray(sessionId)})`;
40
+ }
41
+
42
+ });
43
+
44
+ runner.on('test', test => testLogger.warn(_chalk.default.cyan(test.titlePath().join('/'))));
45
+ runner.on('pass', test => testLogger.info(_chalk.default.cyan(test.titlePath().join('/'))));
46
+ runner.on('fail', (test, error) => testLogger.error(_chalk.default.cyan(test.titlePath().join('/')), '\n ', getErrors(error, (error, imageName) => `${_chalk.default.bold(imageName !== null && imageName !== void 0 ? imageName : topLevelSuite)}:${error}`, error => {
47
+ var _error$stack;
48
+
49
+ return `${(_error$stack = error.stack) !== null && _error$stack !== void 0 ? _error$stack : error.message}`;
50
+ }).join('\n ')));
51
+ }
52
+
53
+ }
54
+
55
+ exports.CreeveyReporter = CreeveyReporter;
56
+
57
+ class TeamcityReporter extends _mocha.reporters.Base {
58
+ constructor(runner, options) {
59
+ super(runner);
60
+
61
+ _defineProperty(this, "escape", str => {
62
+ if (!str) return '';
63
+ return str.toString() // eslint-disable-next-line no-control-regex
64
+ .replace(/\x1B.*?m/g, '').replace(/\|/g, '||').replace(/\n/g, '|n').replace(/\r/g, '|r').replace(/\[/g, '|[').replace(/\]/g, '|]').replace(/\u0085/g, '|x').replace(/\u2028/g, '|l').replace(/\u2029/g, '|p').replace(/'/g, "|'");
65
+ });
66
+
67
+ const topLevelSuite = this.escape(options.reporterOptions.topLevelSuite);
68
+ const reporterOptions = options.reporterOptions;
69
+ runner.on('suite', suite => suite.root ? console.log(`##teamcity[testSuiteStarted name='${topLevelSuite}' flowId='${process.pid}']`) : console.log(`##teamcity[testSuiteStarted name='${this.escape(suite.title)}' flowId='${process.pid}']`));
70
+ runner.on('test', test => console.log(`##teamcity[testStarted name='${this.escape(test.title)}' flowId='${process.pid}']`));
71
+ runner.on('fail', (test, error) => {
72
+ var _error$stack2;
73
+
74
+ Object.entries(reporterOptions.images).forEach(([name, image]) => {
75
+ if (!image) return;
76
+ const filePath = test.titlePath().concat(name == topLevelSuite ? [] : [topLevelSuite]).map(this.escape).join('/'); // eslint-disable-next-line @typescript-eslint/no-unused-vars
77
+
78
+ const {
79
+ error,
80
+ ...rest
81
+ } = image;
82
+ Object.values(rest).filter(_types.isDefined).forEach(fileName => {
83
+ console.log(`##teamcity[publishArtifacts '${reporterOptions.reportDir}/${filePath}/${fileName} => report/${filePath}']`);
84
+ console.log(`##teamcity[testMetadata testName='${this.escape(test.title)}' type='image' value='report/${filePath}/${fileName}' flowId='${process.pid}']`);
85
+ });
86
+ }); // Output failed test as passed due TC don't support retry mechanic
87
+ // https://teamcity-support.jetbrains.com/hc/en-us/community/posts/207216829-Count-test-as-successful-if-at-least-one-try-is-successful?page=1#community_comment_207394125
88
+
89
+ reporterOptions.willRetry ? console.log(`##teamcity[testFinished name='${this.escape(test.title)}' flowId='${process.pid}']`) : console.log(`##teamcity[testFailed name='${this.escape(test.title)}' message='${this.escape(error.message)}' details='${this.escape((_error$stack2 = error.stack) !== null && _error$stack2 !== void 0 ? _error$stack2 : '')}' flowId='${process.pid}']`);
90
+ });
91
+ runner.on('pending', test => console.log(`##teamcity[testIgnored name='${this.escape(test.title)}' message='${this.escape(typeof test.skipReason == 'boolean' ? test.title : test.skipReason)}' flowId='${process.pid}']`));
92
+ runner.on('test end', test => console.log(`##teamcity[testFinished name='${this.escape(test.title)}' flowId='${process.pid}']`));
93
+ runner.on('suite end', suite => suite.root || console.log(`##teamcity[testSuiteFinished name='${this.escape(suite.title)}' flowId='${process.pid}']`));
94
+ runner.on('end', () => console.log(`##teamcity[testSuiteFinished name='${topLevelSuite}' flowId='${process.pid}']`));
95
+ }
96
+
97
+ }
98
+
99
+ exports.TeamcityReporter = TeamcityReporter;
100
+
101
+ function getErrors(error, imageErrorToString, errorToString) {
102
+ const errors = [];
103
+
104
+ if (!(error instanceof Error)) {
105
+ errors.push(error);
106
+ } else if (!(0, _types.isImageError)(error)) {
107
+ errors.push(errorToString(error));
108
+ } else if (typeof error.images == 'string') {
109
+ errors.push(imageErrorToString(error.images));
110
+ } else {
111
+ const imageErrors = error.images;
112
+ Object.keys(imageErrors).forEach(imageName => {
113
+ var _imageErrors$imageNam;
114
+
115
+ errors.push(imageErrorToString((_imageErrors$imageNam = imageErrors[imageName]) !== null && _imageErrors$imageNam !== void 0 ? _imageErrors$imageNam : '', imageName));
116
+ });
117
+ }
118
+
119
+ return errors;
120
+ }