creevey 0.9.0-beta.1 → 0.9.0-beta.10.timeouts.0

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 (221) hide show
  1. package/CHANGELOG.md +51 -0
  2. package/README.md +9 -1
  3. package/addon/README.md +3 -0
  4. package/addon/package.json +5 -0
  5. package/docs/config.md +29 -26
  6. package/jest.config.js +6 -0
  7. package/lib/cjs/cli.js +5 -0
  8. package/lib/cjs/client/addon/Manager.js +264 -0
  9. package/lib/cjs/client/addon/components/Addon.js +55 -0
  10. package/lib/cjs/client/addon/components/Icons.js +46 -0
  11. package/lib/cjs/client/addon/components/Panel.js +72 -0
  12. package/lib/cjs/client/addon/components/TestSelect.js +65 -0
  13. package/lib/cjs/client/addon/components/Tools.js +95 -0
  14. package/lib/cjs/client/addon/decorator.js +11 -0
  15. package/lib/cjs/client/addon/index.js +31 -0
  16. package/lib/cjs/client/addon/preset.ie11.js +74 -0
  17. package/lib/cjs/client/addon/preset.js +17 -0
  18. package/lib/cjs/client/addon/preset.sb7.js +19 -0
  19. package/lib/cjs/client/addon/preview.js +14 -0
  20. package/lib/cjs/client/addon/readyForCapture.js +12 -0
  21. package/lib/cjs/client/addon/register.js +72 -0
  22. package/lib/cjs/client/addon/utils.js +42 -0
  23. package/lib/cjs/client/addon/withCreevey.js +351 -0
  24. package/lib/cjs/client/shared/components/ImagesView/BlendView.js +87 -0
  25. package/lib/cjs/client/shared/components/ImagesView/ImagesView.js +92 -0
  26. package/lib/cjs/client/shared/components/ImagesView/SideBySideView.js +154 -0
  27. package/lib/cjs/client/shared/components/ImagesView/SlideView.js +166 -0
  28. package/lib/cjs/client/shared/components/ImagesView/SwapView.js +91 -0
  29. package/lib/cjs/client/shared/components/ImagesView/index.js +45 -0
  30. package/lib/cjs/client/shared/components/PageFooter/PageFooter.js +50 -0
  31. package/lib/cjs/client/shared/components/PageFooter/Paging.js +94 -0
  32. package/lib/cjs/client/shared/components/PageHeader/ImagePreview.js +82 -0
  33. package/lib/cjs/client/shared/components/PageHeader/PageHeader.js +119 -0
  34. package/lib/cjs/client/shared/components/ResultsPage.js +143 -0
  35. package/lib/cjs/client/shared/creeveyClientApi.js +76 -0
  36. package/lib/cjs/client/shared/helpers.js +411 -0
  37. package/lib/cjs/client/shared/viewMode.js +17 -0
  38. package/lib/cjs/client/web/142.js +2 -0
  39. package/lib/cjs/client/web/142.js.LICENSE.txt +12 -0
  40. package/lib/cjs/client/web/32.js +1 -0
  41. package/lib/cjs/client/web/551.js +1 -0
  42. package/lib/cjs/client/web/566.js +2 -0
  43. package/lib/cjs/client/web/566.js.LICENSE.txt +31 -0
  44. package/lib/cjs/client/web/691.js +2 -0
  45. package/lib/cjs/client/web/691.js.LICENSE.txt +8 -0
  46. package/lib/cjs/client/web/725.js +1 -0
  47. package/lib/cjs/client/web/index.html +19 -0
  48. package/lib/cjs/client/web/main.js +2 -38
  49. package/lib/cjs/client/web/main.js.LICENSE.txt +49 -0
  50. package/lib/cjs/creevey.js +69 -0
  51. package/lib/cjs/index.js +62 -0
  52. package/lib/cjs/server/config.js +94 -0
  53. package/lib/cjs/server/docker.js +146 -0
  54. package/lib/cjs/server/extract.js +46 -0
  55. package/lib/cjs/server/index.js +83 -0
  56. package/lib/cjs/server/loaders/babel/creevey-plugin.js +86 -0
  57. package/lib/cjs/server/loaders/babel/helpers.js +469 -0
  58. package/lib/cjs/server/loaders/babel/register.js +124 -0
  59. package/lib/cjs/server/loaders/hooks/mdx.js +30 -0
  60. package/lib/cjs/server/loaders/hooks/svelte.js +65 -0
  61. package/lib/cjs/server/loaders/webpack/compile.js +269 -0
  62. package/lib/cjs/server/loaders/webpack/creevey-loader.js +172 -0
  63. package/lib/cjs/server/loaders/webpack/dummy-hmr.js +39 -0
  64. package/lib/cjs/server/loaders/webpack/mdx-loader.js +72 -0
  65. package/lib/cjs/server/loaders/webpack/start.js +41 -0
  66. package/lib/cjs/server/logger.js +48 -0
  67. package/lib/cjs/server/master/api.js +71 -0
  68. package/lib/cjs/server/master/index.js +146 -0
  69. package/lib/cjs/server/master/master.js +57 -0
  70. package/lib/cjs/server/master/pool.js +197 -0
  71. package/lib/cjs/server/master/runner.js +281 -0
  72. package/lib/cjs/server/master/server.js +131 -0
  73. package/lib/cjs/server/messages.js +264 -0
  74. package/lib/cjs/server/selenium/browser.js +671 -0
  75. package/lib/cjs/server/selenium/index.js +31 -0
  76. package/lib/cjs/server/selenium/selenoid.js +172 -0
  77. package/lib/cjs/server/stories.js +153 -0
  78. package/lib/cjs/server/storybook/entry.js +53 -0
  79. package/lib/cjs/server/storybook/helpers.js +158 -0
  80. package/lib/cjs/server/storybook/providers/browser.js +74 -0
  81. package/lib/cjs/server/storybook/providers/hybrid.js +82 -0
  82. package/lib/cjs/server/storybook/providers/nodejs.js +239 -0
  83. package/lib/cjs/server/testsFiles/parser.js +72 -0
  84. package/lib/cjs/server/testsFiles/register.js +44 -0
  85. package/lib/cjs/server/update.js +79 -0
  86. package/lib/cjs/server/utils.js +176 -0
  87. package/lib/cjs/server/worker/chai-image.js +142 -0
  88. package/lib/cjs/server/worker/helpers.js +69 -0
  89. package/lib/cjs/server/worker/index.js +15 -0
  90. package/lib/cjs/server/worker/reporter.js +108 -0
  91. package/lib/cjs/server/worker/worker.js +268 -0
  92. package/lib/cjs/shared/index.js +101 -0
  93. package/lib/cjs/shared/serializeRegExp.js +42 -0
  94. package/lib/cjs/types.js +75 -0
  95. package/lib/esm/cli.js +4 -0
  96. package/lib/esm/client/addon/Manager.js +248 -0
  97. package/lib/esm/client/addon/components/Addon.js +39 -0
  98. package/lib/esm/client/addon/components/Icons.js +31 -0
  99. package/lib/esm/client/addon/components/Panel.js +53 -0
  100. package/lib/esm/client/addon/components/TestSelect.js +51 -0
  101. package/lib/esm/client/addon/components/Tools.js +74 -0
  102. package/lib/esm/client/addon/decorator.js +2 -0
  103. package/lib/esm/client/addon/index.js +2 -0
  104. package/lib/esm/client/addon/preset.ie11.js +59 -0
  105. package/lib/esm/client/addon/preset.js +8 -0
  106. package/lib/esm/client/addon/preset.sb7.js +8 -0
  107. package/lib/esm/client/addon/preview.js +5 -0
  108. package/lib/esm/client/addon/readyForCapture.js +5 -0
  109. package/lib/esm/client/addon/register.js +51 -0
  110. package/lib/esm/client/addon/utils.js +32 -0
  111. package/lib/esm/client/addon/withCreevey.js +325 -0
  112. package/lib/esm/client/shared/components/ImagesView/BlendView.js +67 -0
  113. package/lib/esm/client/shared/components/ImagesView/ImagesView.js +69 -0
  114. package/lib/esm/client/shared/components/ImagesView/SideBySideView.js +131 -0
  115. package/lib/esm/client/shared/components/ImagesView/SlideView.js +143 -0
  116. package/lib/esm/client/shared/components/ImagesView/SwapView.js +71 -0
  117. package/lib/esm/client/shared/components/ImagesView/index.js +5 -0
  118. package/lib/esm/client/shared/components/PageFooter/PageFooter.js +36 -0
  119. package/lib/esm/client/shared/components/PageFooter/Paging.js +80 -0
  120. package/lib/esm/client/shared/components/PageHeader/ImagePreview.js +68 -0
  121. package/lib/esm/client/shared/components/PageHeader/PageHeader.js +97 -0
  122. package/lib/esm/client/shared/components/ResultsPage.js +115 -0
  123. package/lib/esm/client/shared/creeveyClientApi.js +67 -0
  124. package/lib/esm/client/shared/helpers.js +353 -0
  125. package/lib/esm/client/shared/viewMode.js +6 -0
  126. package/lib/esm/creevey.js +54 -0
  127. package/lib/esm/index.js +5 -0
  128. package/lib/esm/server/config.js +71 -0
  129. package/lib/esm/server/docker.js +123 -0
  130. package/lib/esm/server/extract.js +32 -0
  131. package/lib/esm/server/index.js +64 -0
  132. package/lib/esm/server/loaders/babel/creevey-plugin.js +72 -0
  133. package/lib/esm/server/loaders/babel/helpers.js +452 -0
  134. package/lib/esm/server/loaders/babel/register.js +103 -0
  135. package/lib/esm/server/loaders/hooks/mdx.js +15 -0
  136. package/lib/esm/server/loaders/hooks/svelte.js +49 -0
  137. package/lib/esm/server/loaders/webpack/compile.js +246 -0
  138. package/lib/esm/server/loaders/webpack/creevey-loader.js +152 -0
  139. package/lib/esm/server/loaders/webpack/dummy-hmr.js +32 -0
  140. package/lib/esm/server/loaders/webpack/mdx-loader.js +58 -0
  141. package/lib/esm/server/loaders/webpack/start.js +27 -0
  142. package/lib/esm/server/logger.js +20 -0
  143. package/lib/esm/server/master/api.js +60 -0
  144. package/lib/esm/server/master/index.js +125 -0
  145. package/lib/esm/server/master/master.js +38 -0
  146. package/lib/esm/server/master/pool.js +176 -0
  147. package/lib/esm/server/master/runner.js +259 -0
  148. package/lib/esm/server/master/server.js +107 -0
  149. package/lib/esm/server/messages.js +232 -0
  150. package/lib/esm/server/selenium/browser.js +638 -0
  151. package/lib/esm/server/selenium/index.js +2 -0
  152. package/lib/esm/server/selenium/selenoid.js +149 -0
  153. package/lib/esm/server/stories.js +135 -0
  154. package/lib/esm/server/storybook/entry.js +27 -0
  155. package/lib/esm/server/storybook/helpers.js +97 -0
  156. package/lib/esm/server/storybook/providers/browser.js +58 -0
  157. package/lib/esm/server/storybook/providers/hybrid.js +60 -0
  158. package/lib/esm/server/storybook/providers/nodejs.js +216 -0
  159. package/lib/esm/server/testsFiles/parser.js +50 -0
  160. package/lib/esm/server/testsFiles/register.js +31 -0
  161. package/lib/esm/server/update.js +61 -0
  162. package/lib/esm/server/utils.js +133 -0
  163. package/lib/esm/server/worker/chai-image.js +130 -0
  164. package/lib/esm/server/worker/helpers.js +60 -0
  165. package/lib/esm/server/worker/index.js +1 -0
  166. package/lib/esm/server/worker/reporter.js +86 -0
  167. package/lib/esm/server/worker/worker.js +238 -0
  168. package/lib/esm/shared/index.js +78 -0
  169. package/lib/esm/shared/serializeRegExp.js +24 -0
  170. package/lib/esm/types.js +44 -0
  171. package/lib/types/client/addon/Manager.d.ts +2 -2
  172. package/lib/types/client/addon/components/TestSelect.d.ts +0 -1
  173. package/lib/types/client/addon/index.d.ts +2 -0
  174. package/lib/types/client/addon/preset.d.ts +0 -22
  175. package/lib/types/client/addon/preset.ie11.d.ts +10 -0
  176. package/lib/types/client/addon/preset.sb7.d.ts +2 -0
  177. package/lib/types/client/addon/preview.d.ts +4 -0
  178. package/lib/types/client/addon/utils.d.ts +1 -0
  179. package/lib/types/client/addon/withCreevey.d.ts +1 -1
  180. package/lib/types/client/shared/components/ImagesView/BlendView.d.ts +1 -1
  181. package/lib/types/client/shared/components/ImagesView/ImagesView.d.ts +0 -1
  182. package/lib/types/client/shared/components/ImagesView/SideBySideView.d.ts +1 -1
  183. package/lib/types/client/shared/components/ImagesView/SlideView.d.ts +1 -1
  184. package/lib/types/client/shared/components/ImagesView/SwapView.d.ts +1 -1
  185. package/lib/types/client/shared/components/PageFooter/PageFooter.d.ts +0 -1
  186. package/lib/types/client/shared/components/PageFooter/Paging.d.ts +0 -1
  187. package/lib/types/client/shared/components/PageHeader/ImagePreview.d.ts +1 -1
  188. package/lib/types/client/shared/components/PageHeader/PageHeader.d.ts +0 -1
  189. package/lib/types/client/shared/components/ResultsPage.d.ts +1 -1
  190. package/lib/types/client/web/CreeveyApp.d.ts +0 -1
  191. package/lib/types/client/web/CreeveyLoader.d.ts +1 -2
  192. package/lib/types/client/web/CreeveyView/SideBar/Checkbox.d.ts +1 -1
  193. package/lib/types/client/web/CreeveyView/SideBar/SideBarHeader.d.ts +0 -1
  194. package/lib/types/client/web/CreeveyView/SideBar/SuiteLink.d.ts +4 -4
  195. package/lib/types/client/web/CreeveyView/SideBar/TestLink.d.ts +0 -1
  196. package/lib/types/client/web/CreeveyView/SideBar/TestStatusIcon.d.ts +1 -1
  197. package/lib/types/client/web/CreeveyView/SideBar/TestsStatus.d.ts +1 -1
  198. package/lib/types/index.d.ts +1 -0
  199. package/lib/types/server/loaders/babel/register.d.ts +1 -1
  200. package/lib/types/server/loaders/webpack/creevey-loader.d.ts +4 -2
  201. package/lib/types/server/logger.d.ts +6 -2
  202. package/lib/types/server/messages.d.ts +14 -12
  203. package/lib/types/server/selenium/browser.d.ts +5 -3
  204. package/lib/types/server/storybook/entry.d.ts +2 -3
  205. package/lib/types/server/storybook/helpers.d.ts +1 -1
  206. package/lib/types/server/storybook/providers/browser.d.ts +2 -4
  207. package/lib/types/server/storybook/providers/hybrid.d.ts +2 -4
  208. package/lib/types/server/storybook/providers/nodejs.d.ts +3 -3
  209. package/lib/types/server/utils.d.ts +5 -1
  210. package/lib/types/{shared.d.ts → shared/index.d.ts} +1 -10
  211. package/lib/types/shared/serializeRegExp.d.ts +9 -0
  212. package/lib/types/types.d.ts +4 -7
  213. package/package.json +117 -103
  214. package/preset/ie11.js +5 -0
  215. package/{preset.js → preset/index.js} +2 -2
  216. package/preset/sb7.js +5 -0
  217. package/types/global.d.ts +5 -0
  218. package/types/mdx.d.ts +3 -2
  219. package/lib/cjs/client/web/1.js +0 -13
  220. package/lib/cjs/client/web/2.js +0 -1
  221. package/storybook-static/stories.json +0 -21
@@ -0,0 +1,149 @@
1
+ import path from 'path';
2
+ import { promisify } from 'util';
3
+ import { mkdir, writeFile, copyFile, lstatSync, existsSync } from 'fs';
4
+ import { downloadBinary, getCreeveyCache } from '../utils';
5
+ import { pullImages, runImage } from '../docker';
6
+ import { Octokit } from '@octokit/core';
7
+ import { subscribeOn } from '../messages';
8
+ import cluster from 'cluster';
9
+ import { chmod, exec } from 'shelljs';
10
+ const mkdirAsync = promisify(mkdir);
11
+ const writeFileAsync = promisify(writeFile);
12
+ const copyFileAsync = promisify(copyFile);
13
+
14
+ async function createSelenoidConfig(browsers, {
15
+ useDocker
16
+ }) {
17
+ const selenoidConfig = {};
18
+ const selenoidConfigDir = path.join(getCreeveyCache(), 'selenoid');
19
+ browsers.forEach(({
20
+ browserName,
21
+ version = 'latest',
22
+ dockerImage = `selenoid/${browserName}:${version}`,
23
+ webdriverCommand = []
24
+ }) => {
25
+ if (!selenoidConfig[browserName]) selenoidConfig[browserName] = {
26
+ default: version,
27
+ versions: {}
28
+ };
29
+ if (!useDocker && webdriverCommand.length == 0) throw new Error('Please specify "webdriverCommand" browser option with path to browser webdriver');
30
+ selenoidConfig[browserName].versions[version] = {
31
+ image: useDocker ? dockerImage : webdriverCommand,
32
+ port: '4444',
33
+ path: !useDocker || ['chrome', 'opera', 'webkit', 'MicrosoftEdge'].includes(browserName) ? '/' : '/wd/hub'
34
+ };
35
+ });
36
+ await mkdirAsync(selenoidConfigDir, {
37
+ recursive: true
38
+ });
39
+ await writeFileAsync(path.join(selenoidConfigDir, 'browsers.json'), JSON.stringify(selenoidConfig));
40
+ return selenoidConfigDir;
41
+ }
42
+
43
+ async function downloadSelenoidBinary(destination) {
44
+ const platformNameMapping = {
45
+ darwin: 'selenoid_darwin_amd64',
46
+ linux: 'selenoid_linux_amd64',
47
+ win32: 'selenoid_windows_amd64.exe'
48
+ };
49
+ const octokit = new Octokit();
50
+ const response = await octokit.request('GET /repos/{owner}/{repo}/releases/latest', {
51
+ owner: 'aerokube',
52
+ repo: 'selenoid'
53
+ });
54
+ const {
55
+ assets
56
+ } = response.data;
57
+ const {
58
+ browser_download_url: downloadUrl,
59
+ size: binarySize
60
+ } = assets.find(({
61
+ name
62
+ }) => platformNameMapping[process.platform] == name) ?? {};
63
+ if (existsSync(destination) && lstatSync(destination).size == binarySize) return;
64
+
65
+ if (!downloadUrl) {
66
+ throw new Error(`Couldn't get download url for selenoid binary. Please download it manually from "https://github.com/aerokube/selenoid/releases/latest" and define "selenoidPath" option in the Creevey config`);
67
+ }
68
+
69
+ return downloadBinary(downloadUrl, destination);
70
+ }
71
+
72
+ export async function startSelenoidStandalone(config, debug) {
73
+ config.gridUrl = 'http://localhost:4444/wd/hub';
74
+ if (cluster.isWorker) return;
75
+ const browsers = Object.values(config.browsers).filter(browser => !browser.gridUrl);
76
+ const selenoidConfigDir = await createSelenoidConfig(browsers, {
77
+ useDocker: false
78
+ });
79
+ const binaryPath = path.join(selenoidConfigDir, process.platform == 'win32' ? 'selenoid.exe' : 'selenoid');
80
+
81
+ if (config.selenoidPath) {
82
+ await copyFileAsync(path.resolve(config.selenoidPath), binaryPath);
83
+ } else {
84
+ await downloadSelenoidBinary(binaryPath);
85
+ } // TODO Download browser webdrivers
86
+
87
+
88
+ try {
89
+ if (process.platform != 'win32') chmod('+x', binaryPath);
90
+ } catch (_) {
91
+ /* noop */
92
+ }
93
+
94
+ const selenoidProcess = exec(`${binaryPath} -conf ./browsers.json -disable-docker`, {
95
+ async: true,
96
+ cwd: selenoidConfigDir
97
+ });
98
+
99
+ if (debug) {
100
+ var _selenoidProcess$stdo, _selenoidProcess$stde;
101
+
102
+ (_selenoidProcess$stdo = selenoidProcess.stdout) === null || _selenoidProcess$stdo === void 0 ? void 0 : _selenoidProcess$stdo.pipe(process.stdout);
103
+ (_selenoidProcess$stde = selenoidProcess.stderr) === null || _selenoidProcess$stde === void 0 ? void 0 : _selenoidProcess$stde.pipe(process.stderr);
104
+ }
105
+
106
+ subscribeOn('shutdown', () => selenoidProcess.kill());
107
+ }
108
+ export async function startSelenoidContainer(config, debug) {
109
+ const browsers = Object.values(config.browsers).filter(browser => !browser.gridUrl);
110
+ const images = [];
111
+ let limit = 0;
112
+ browsers.forEach(({
113
+ browserName,
114
+ version = 'latest',
115
+ limit: browserLimit = 1,
116
+ dockerImage = `selenoid/${browserName}:${version}`
117
+ }) => {
118
+ limit += browserLimit;
119
+ images.push(dockerImage);
120
+ });
121
+ const selenoidImage = config.dockerImage;
122
+ const pullOptions = {
123
+ auth: config.dockerAuth,
124
+ platform: config.dockerImagePlatform
125
+ };
126
+
127
+ if (config.pullImages) {
128
+ await pullImages([selenoidImage], pullOptions);
129
+ await pullImages(images, pullOptions);
130
+ } // TODO Allow pass custom options
131
+
132
+
133
+ const selenoidOptions = {
134
+ ExposedPorts: {
135
+ '4444/tcp': {}
136
+ },
137
+ HostConfig: {
138
+ PortBindings: {
139
+ '4444/tcp': [{
140
+ HostPort: '4444'
141
+ }]
142
+ },
143
+ Binds: ['/var/run/docker.sock:/var/run/docker.sock', `${await createSelenoidConfig(browsers, {
144
+ useDocker: true
145
+ })}:/etc/selenoid/:ro`]
146
+ }
147
+ };
148
+ return runImage(selenoidImage, ['-limit', String(limit)], selenoidOptions, debug);
149
+ }
@@ -0,0 +1,135 @@
1
+ import path from 'path';
2
+ import { mkdirSync, writeFileSync } from 'fs';
3
+ import { createHash } from 'crypto';
4
+ import { mapValues, pick } from 'lodash';
5
+ import { isDefined, isFunction, isObject } from '../types';
6
+ import { shouldSkip, removeProps } from './utils';
7
+
8
+ function storyTestFabric(delay, testFn) {
9
+ return async function storyTest() {
10
+ delay ? await new Promise(resolve => setTimeout(resolve, delay)) : void 0;
11
+ await (testFn ? testFn.call(this) : this.screenshots.length > 0 ? this.expect(this.screenshots.reduce((screenshots, {
12
+ imageName,
13
+ screenshot
14
+ }, index) => ({ ...screenshots,
15
+ [imageName ?? `screenshot_${index}`]: screenshot
16
+ }), {})).to.matchImages() : this.expect(await this.takeScreenshot()).to.matchImage());
17
+ };
18
+ }
19
+
20
+ function createCreeveyTest(browser, storyMeta, skipOptions, testName) {
21
+ const {
22
+ kind,
23
+ name: story,
24
+ id: storyId
25
+ } = storyMeta;
26
+ const path = [kind, story, testName, browser].filter(isDefined);
27
+ const skip = skipOptions ? shouldSkip(browser, {
28
+ kind,
29
+ story
30
+ }, skipOptions, testName) : false;
31
+ const id = createHash('sha1').update(path.join('/')).digest('hex');
32
+ return {
33
+ id,
34
+ skip,
35
+ browser,
36
+ testName,
37
+ storyPath: [...kind.split('/').map(x => x.trim()), story],
38
+ storyId
39
+ };
40
+ }
41
+
42
+ function convertStories(browserName, stories) {
43
+ const tests = {};
44
+ (Array.isArray(stories) ? stories : Object.values(stories)).forEach(storyMeta => {
45
+ // TODO Skip docsOnly stories for now
46
+ if (storyMeta.parameters.docsOnly) return;
47
+ const {
48
+ delay: delayParam,
49
+ tests: storyTests,
50
+ skip
51
+ } = storyMeta.parameters.creevey ?? {};
52
+ const delay = typeof delayParam == 'number' ? delayParam : delayParam !== null && delayParam !== void 0 && delayParam.for.includes(browserName) ? delayParam.ms : 0; // typeof tests === "undefined" => rootSuite -> kindSuite -> storyTest -> [browsers.png]
53
+ // typeof tests === "function" => rootSuite -> kindSuite -> storyTest -> browser -> [images.png]
54
+ // typeof tests === "object" => rootSuite -> kindSuite -> storySuite -> test -> [browsers.png]
55
+ // typeof tests === "object" => rootSuite -> kindSuite -> storySuite -> test -> browser -> [images.png]
56
+
57
+ if (!storyTests) {
58
+ const test = createCreeveyTest(browserName, storyMeta, skip);
59
+ tests[test.id] = { ...test,
60
+ storyId: storyMeta.id,
61
+ story: storyMeta,
62
+ fn: storyTestFabric(delay)
63
+ };
64
+ return;
65
+ }
66
+
67
+ Object.entries(storyTests).forEach(([testName, testFn]) => {
68
+ const test = createCreeveyTest(browserName, storyMeta, skip, testName);
69
+ tests[test.id] = { ...test,
70
+ storyId: storyMeta.id,
71
+ story: storyMeta,
72
+ fn: storyTestFabric(delay, testFn)
73
+ };
74
+ });
75
+ });
76
+ return tests;
77
+ }
78
+
79
+ export async function loadTestsFromStories(browsers, provider, update) {
80
+ const testIdsByFiles = new Map();
81
+ const stories = await provider(storiesByFiles => {
82
+ const testsDiff = {};
83
+ const tests = {};
84
+ browsers.forEach(browser => {
85
+ Array.from(storiesByFiles.entries()).forEach(([filename, stories]) => {
86
+ var _testIdsByFiles$get;
87
+
88
+ Object.assign(tests, convertStories(browser, stories));
89
+ const changed = Object.keys(tests);
90
+ const removed = ((_testIdsByFiles$get = testIdsByFiles.get(filename)) === null || _testIdsByFiles$get === void 0 ? void 0 : _testIdsByFiles$get.filter(testId => !tests[testId])) ?? [];
91
+ if (changed.length == 0) testIdsByFiles.delete(filename);else testIdsByFiles.set(filename, changed);
92
+ Object.assign(testsDiff, tests);
93
+ removed.forEach(testId => testsDiff[testId] = undefined);
94
+ });
95
+ });
96
+ update === null || update === void 0 ? void 0 : update(testsDiff);
97
+ });
98
+ const tests = browsers.reduce((tests, browser) => Object.assign(tests, convertStories(browser, stories)), {});
99
+ Object.values(tests).filter(isDefined).forEach(({
100
+ id,
101
+ story: {
102
+ parameters: {
103
+ fileName
104
+ }
105
+ }
106
+ }) => // TODO Don't use filename as a key, due possible collisions if two require.context with same structure of modules are defined
107
+ testIdsByFiles.set(fileName, [...(testIdsByFiles.get(fileName) ?? []), id]));
108
+ return tests;
109
+ }
110
+ export function saveStoriesJson(storiesData, extract) {
111
+ const outputDir = typeof extract == 'boolean' ? 'storybook-static' : extract; // NOTE Copy-pasted from Storybook's `getStoriesJsonData` method
112
+
113
+ const allowed = ['fileName', 'docsOnly', 'framework', '__id', '__isArgsStory'];
114
+ storiesData.globalParameters = pick(storiesData.globalParameters, allowed); // @ts-expect-error ignore error
115
+
116
+ storiesData.kindParameters = mapValues(storiesData.kindParameters, v => pick(v, allowed)); // @ts-expect-error ignore error
117
+
118
+ storiesData.stories = mapValues(storiesData.stories, v => ({ ...pick(v, ['id', 'name', 'kind', 'story']),
119
+ parameters: pick(v.parameters, allowed)
120
+ })); // TODO Fix args stories
121
+
122
+ removeProps(storiesData ?? {}, ['stories', () => true, 'parameters', '__isArgsStory']);
123
+ Object.values((storiesData === null || storiesData === void 0 ? void 0 : storiesData.stories) ?? {}).forEach(story => isObject(story) && 'parameters' in story && isObject(story.parameters) && delete story.parameters.__isArgsStory);
124
+ mkdirSync(outputDir, {
125
+ recursive: true
126
+ });
127
+ writeFileSync(path.join(outputDir, 'stories.json'), JSON.stringify(storiesData, null, 2));
128
+ }
129
+ export function saveTestsJson(tests, dstPath = process.cwd()) {
130
+ mkdirSync(dstPath, {
131
+ recursive: true
132
+ });
133
+ writeFileSync(path.join(dstPath, 'tests.json'), // eslint-disable-next-line @typescript-eslint/no-unsafe-return
134
+ JSON.stringify(tests, (_, value) => isFunction(value) ? value.toString() : value, 2));
135
+ }
@@ -0,0 +1,27 @@
1
+ import { addons } from '@storybook/addons';
2
+ import { getStorybookFramework, resolveFromStorybook } from './helpers';
3
+ const framework = getStorybookFramework(); // eslint-disable-next-line @typescript-eslint/no-var-requires
4
+
5
+ const core = require(resolveFromStorybook('@storybook/core-client'));
6
+
7
+ const start = core.start;
8
+ const api = start(() => void 0);
9
+ export const channel = addons.getChannel();
10
+ export const clientApi = api.clientApi;
11
+ export const forceReRender = api.forceReRender;
12
+ export const storiesOf = (kind, m) => {
13
+ return clientApi.storiesOf(kind, m).addParameters({
14
+ framework
15
+ });
16
+ };
17
+ export const configure = (...args) => {
18
+ //NOTE Storybook 6.x pass `framework` as first argument
19
+ //@ts-expect-error: ignore it
20
+ return api.configure(framework, ...args);
21
+ };
22
+ export const addDecorator = clientApi.addDecorator;
23
+ export const addParameters = clientApi.addParameters;
24
+ export const clearDecorators = clientApi.clearDecorators;
25
+ export const setAddon = clientApi.setAddon;
26
+ export const getStorybook = clientApi.getStorybook;
27
+ export const raw = clientApi.raw;
@@ -0,0 +1,97 @@
1
+ import path from 'path';
2
+ import resolveFrom from 'resolve-from';
3
+ const supportedFrameworks = ['react', 'vue', 'vue3', 'angular', 'marionette', 'mithril', 'marko', 'html', 'svelte', 'riot', 'ember', 'preact', 'rax', 'aurelia', 'server', 'web-components'];
4
+ export const storybookDirRef = {
5
+ current: path.resolve('.storybook')
6
+ };
7
+ export const resolveFromStorybook = modulePath => resolveFrom(storybookDirRef.current, modulePath);
8
+ export const resolveFromStorybookAddonDocs = modulePath => resolveFrom(resolveFromStorybook('@storybook/addon-docs'), modulePath);
9
+ export const resolveFromStorybookBuilderWebpack4 = modulePath => resolveFrom(resolveFromStorybook('@storybook/builder-webpack4'), modulePath);
10
+ export const resolveFromStorybookCoreClient = modulePath => resolveFrom(resolveFromStorybook('@storybook/core-client'), modulePath);
11
+ export const resolveFromStorybookCoreServer = modulePath => resolveFrom(resolveFromStorybook('@storybook/core-server'), modulePath);
12
+
13
+ const importFromStorybook = modulePath => import(resolveFromStorybook(modulePath));
14
+
15
+ export const importStorybookClientLogger = () => importFromStorybook('@storybook/client-logger');
16
+ export const importStorybookCoreCommon = () => importFromStorybook('@storybook/core-common');
17
+ export const importStorybookCoreEvents = () => importFromStorybook('@storybook/core-events');
18
+ export function hasDocsAddon() {
19
+ try {
20
+ resolveFromStorybook('@storybook/addon-docs');
21
+ return true;
22
+ } catch (_) {
23
+ return false;
24
+ }
25
+ }
26
+ export function hasSvelteCSFAddon() {
27
+ try {
28
+ resolveFromStorybook('@storybook/addon-svelte-csf');
29
+ return true;
30
+ } catch (_) {
31
+ return false;
32
+ }
33
+ }
34
+ export function getStorybookVersion() {
35
+ // eslint-disable-next-line @typescript-eslint/no-var-requires
36
+ const {
37
+ version
38
+ } = require(resolveFromStorybook('@storybook/core-server/package.json'));
39
+
40
+ return version;
41
+ }
42
+ export function isStorybookVersionLessThan(major, minor) {
43
+ const [sbMajor, sbMinor] = (process.env.__CREEVEY_STORYBOOK_VERSION__ ?? getStorybookVersion()).split('.');
44
+ return Number(sbMajor) < major || minor != undefined && Number(sbMajor) == major && Number(sbMinor) < minor;
45
+ }
46
+ export function isStorybookVersionGreaterThan(major, minor) {
47
+ const [sbMajor, sbMinor] = (process.env.__CREEVEY_STORYBOOK_VERSION__ ?? getStorybookVersion()).split('.');
48
+ return Number(sbMajor) > major || minor != undefined && Number(sbMajor) == major && Number(sbMinor) > minor;
49
+ }
50
+ export function isStorybookVersion(major, minor) {
51
+ const [sbMajor, sbMinor] = (process.env.__CREEVEY_STORYBOOK_VERSION__ ?? getStorybookVersion()).split('.');
52
+ return Number(sbMajor) == major || minor != undefined && Number(sbMajor) == major && Number(sbMinor) == minor;
53
+ }
54
+ export function getStorybookFramework() {
55
+ const framework = process.env.__CREEVEY_STORYBOOK_FRAMEWORK__ ?? supportedFrameworks.find(framework => {
56
+ try {
57
+ return require.resolve(resolveFromStorybook(`@storybook/${framework}`));
58
+ } catch (_) {
59
+ return false;
60
+ }
61
+ });
62
+ if (!framework) throw new Error("Couldn't detect used Storybook framework. Please ensure that you install `@storybook/<framework>` package");
63
+ return framework;
64
+ }
65
+ export const storybookConfigRef = {
66
+ current: {
67
+ stories: []
68
+ }
69
+ };
70
+ export async function importStorybookConfig() {
71
+ const configPath = `${storybookDirRef.current}/main`;
72
+
73
+ try {
74
+ return storybookConfigRef.current = (await import(require.resolve(configPath))).default;
75
+ } catch (_) {
76
+ const storybookUtilsPath = '@storybook/core-common/dist/cjs/utils';
77
+ const serverRequireModule = 'interpret-require'; // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
78
+
79
+ const {
80
+ getInterpretedFile
81
+ } = await import(resolveFromStorybook(`${storybookUtilsPath}/interpret-files`)); // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
82
+
83
+ const {
84
+ default: serverRequireFallback,
85
+ serverRequire = serverRequireFallback
86
+ } = await import(resolveFromStorybook(`${storybookUtilsPath}/${serverRequireModule}`)); // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call
87
+
88
+ const mainConfigFile = getInterpretedFile(configPath); // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call
89
+
90
+ return storybookConfigRef.current = serverRequire(mainConfigFile);
91
+ }
92
+ }
93
+ export async function isCSFv3Enabled() {
94
+ var _await$importStoryboo, _await$importStoryboo2;
95
+
96
+ return ((_await$importStoryboo = await importStorybookConfig()) === null || _await$importStoryboo === void 0 ? void 0 : (_await$importStoryboo2 = _await$importStoryboo.features) === null || _await$importStoryboo2 === void 0 ? void 0 : _await$importStoryboo2.previewCsfV3) ?? false;
97
+ }
@@ -0,0 +1,58 @@
1
+ import cluster from 'cluster';
2
+ import { loadStoriesFromBrowser } from '../../selenium';
3
+ import { emitStoriesMessage, sendStoriesMessage, subscribeOn, subscribeOnWorker } from '../../messages';
4
+ import { isDefined } from '../../../types';
5
+ import { logger } from '../../logger';
6
+ export const loadStories = async (_config, _options, storiesListener) => {
7
+ if (cluster.isPrimary) {
8
+ return new Promise(resolve => {
9
+ const worker = Object.values(cluster.workers ?? {}).filter(isDefined).find(worker => worker.isConnected());
10
+
11
+ if (worker) {
12
+ const unsubscribe = subscribeOnWorker(worker, 'stories', message => {
13
+ if (message.type == 'set') {
14
+ const {
15
+ stories,
16
+ oldTests
17
+ } = message.payload;
18
+ if (oldTests.length > 0) logger.warn(`If you use browser stories provider of CSFv3 Storybook feature\n` + `Creevey will not load tests defined in story parameters from following stories:\n` + oldTests.join('\n'));
19
+ unsubscribe();
20
+ resolve(stories);
21
+ }
22
+ });
23
+ sendStoriesMessage(worker, {
24
+ type: 'get'
25
+ });
26
+ }
27
+
28
+ subscribeOn('stories', message => {
29
+ // TODO updates only one browser :(
30
+ if (message.type == 'update') storiesListener(new Map(message.payload));
31
+ });
32
+ });
33
+ } else {
34
+ subscribeOn('stories', message => {
35
+ if (message.type == 'get') emitStoriesMessage({
36
+ type: 'set',
37
+ payload: {
38
+ stories,
39
+ oldTests: storiesWithOldTests
40
+ }
41
+ });
42
+ if (message.type == 'update') storiesListener(new Map(message.payload));
43
+ });
44
+ const stories = await loadStoriesFromBrowser();
45
+ const storiesWithOldTests = [];
46
+ Object.values(stories).forEach(story => {
47
+ var _parameters, _parameters$creevey;
48
+
49
+ if ((_parameters = story.parameters) !== null && _parameters !== void 0 && (_parameters$creevey = _parameters.creevey) !== null && _parameters$creevey !== void 0 && _parameters$creevey.tests) {
50
+ var _parameters2, _parameters2$creevey;
51
+
52
+ (_parameters2 = story.parameters) === null || _parameters2 === void 0 ? true : (_parameters2$creevey = _parameters2.creevey) === null || _parameters2$creevey === void 0 ? true : delete _parameters2$creevey.tests;
53
+ storiesWithOldTests.push(`${story.kind}/${story.name}`);
54
+ }
55
+ });
56
+ return stories;
57
+ }
58
+ };
@@ -0,0 +1,60 @@
1
+ import chokidar from 'chokidar';
2
+ import { loadStories as browserProvider } from './browser';
3
+ import { logger } from '../../logger';
4
+ import parse from '../../testsFiles/parser';
5
+ import { readDirRecursive } from '../../../server/utils';
6
+ import { combineParameters } from '../../../shared';
7
+ export const loadStories = async (_config, _options, storiesListener) => {
8
+ let creeveyParamsByStoryId = {};
9
+
10
+ const mergeParamsFromTestsToStory = (story, creeveyParams) => {
11
+ if (story.parameters) {
12
+ story.parameters.creevey = combineParameters(story.parameters.creevey || {}, creeveyParams);
13
+ }
14
+ };
15
+
16
+ const stories = await browserProvider(_config, {}, updatedStoriesByFiles => {
17
+ Array.from(updatedStoriesByFiles.entries()).forEach(([, storiesArray]) => {
18
+ storiesArray.forEach(story => {
19
+ const creeveyParams = creeveyParamsByStoryId[story.id];
20
+
21
+ if (creeveyParams) {
22
+ mergeParamsFromTestsToStory(story, creeveyParams);
23
+ }
24
+ });
25
+ });
26
+ storiesListener(updatedStoriesByFiles);
27
+ }); // TODO fix test files hot reloading
28
+
29
+ creeveyParamsByStoryId = await parseParams(_config
30
+ /*, (data) => console.log(data) */
31
+ );
32
+ Object.entries(stories).forEach(([storyId, story]) => {
33
+ mergeParamsFromTestsToStory(story, creeveyParamsByStoryId[storyId]);
34
+ });
35
+ return stories;
36
+ };
37
+
38
+ async function parseParams(config, listener) {
39
+ if (!config.testsDir) {
40
+ return Promise.resolve({});
41
+ }
42
+
43
+ const testFiles = readDirRecursive(config.testsDir).filter(file => {
44
+ var _config$testsRegex;
45
+
46
+ return (_config$testsRegex = config.testsRegex) === null || _config$testsRegex === void 0 ? void 0 : _config$testsRegex.test(file);
47
+ });
48
+ await (await import('../../testsFiles/register')).default(config);
49
+
50
+ if (listener) {
51
+ chokidar.watch(testFiles).on('change', filePath => {
52
+ logger.debug(`changed: ${filePath}`); // doesn't work, always returns {} due modules caching
53
+ // see https://github.com/nodejs/modules/issues/307
54
+
55
+ void parse(testFiles).then(data => listener(data));
56
+ });
57
+ }
58
+
59
+ return parse(testFiles);
60
+ }