creevey 0.9.0-beta.1 → 0.9.0-non-webpack.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 (267) hide show
  1. package/AUTHORS +15 -15
  2. package/CHANGELOG.md +1275 -1275
  3. package/LICENSE +21 -21
  4. package/README.md +7 -0
  5. package/addon/README.md +3 -0
  6. package/addon/package.json +4 -0
  7. package/docs/config.md +212 -212
  8. package/docs/grid.md +10 -10
  9. package/docs/tests.md +63 -63
  10. package/jest.config.js +6 -0
  11. package/lib/cjs/cli.js +5 -0
  12. package/lib/cjs/client/addon/Manager.js +418 -0
  13. package/lib/cjs/client/addon/components/Addon.js +76 -0
  14. package/lib/cjs/client/addon/components/Icons.js +42 -0
  15. package/lib/cjs/client/addon/components/Panel.js +68 -0
  16. package/lib/cjs/client/addon/components/TestSelect.js +63 -0
  17. package/lib/cjs/client/addon/components/Tools.js +114 -0
  18. package/lib/cjs/client/addon/decorator.js +11 -0
  19. package/lib/cjs/client/addon/index.js +31 -0
  20. package/lib/cjs/client/addon/preset.js +81 -0
  21. package/lib/cjs/client/addon/readyForCapture.js +12 -0
  22. package/lib/cjs/client/addon/register.js +100 -0
  23. package/lib/cjs/client/addon/utils.js +38 -0
  24. package/lib/cjs/client/addon/withCreevey.js +558 -0
  25. package/lib/cjs/client/shared/components/ImagesView/BlendView.js +85 -0
  26. package/lib/cjs/client/shared/components/ImagesView/ImagesView.js +88 -0
  27. package/lib/cjs/client/shared/components/ImagesView/SideBySideView.js +176 -0
  28. package/lib/cjs/client/shared/components/ImagesView/SlideView.js +179 -0
  29. package/lib/cjs/client/shared/components/ImagesView/SwapView.js +110 -0
  30. package/lib/cjs/client/shared/components/ImagesView/index.js +45 -0
  31. package/lib/cjs/client/shared/components/PageFooter/PageFooter.js +46 -0
  32. package/lib/cjs/client/shared/components/PageFooter/Paging.js +98 -0
  33. package/lib/cjs/client/shared/components/PageHeader/ImagePreview.js +78 -0
  34. package/lib/cjs/client/shared/components/PageHeader/PageHeader.js +144 -0
  35. package/lib/cjs/client/shared/components/ResultsPage.js +173 -0
  36. package/lib/cjs/client/shared/creeveyClientApi.js +107 -0
  37. package/lib/cjs/client/shared/helpers.js +482 -0
  38. package/lib/cjs/client/shared/viewMode.js +17 -0
  39. package/lib/cjs/client/web/142.js +2 -0
  40. package/lib/cjs/client/web/142.js.LICENSE.txt +12 -0
  41. package/lib/cjs/client/web/32.js +1 -0
  42. package/lib/cjs/client/web/551.js +1 -0
  43. package/lib/cjs/client/web/566.js +2 -0
  44. package/lib/cjs/client/web/566.js.LICENSE.txt +31 -0
  45. package/lib/cjs/client/web/691.js +2 -0
  46. package/lib/cjs/client/web/691.js.LICENSE.txt +8 -0
  47. package/lib/cjs/client/web/725.js +1 -0
  48. package/lib/cjs/client/web/index.html +19 -0
  49. package/lib/cjs/client/web/main.js +2 -38
  50. package/lib/cjs/client/web/main.js.LICENSE.txt +49 -0
  51. package/lib/cjs/creevey.js +69 -0
  52. package/lib/cjs/index.js +62 -0
  53. package/lib/cjs/server/config.js +96 -0
  54. package/lib/cjs/server/docker.js +146 -0
  55. package/lib/cjs/server/extract.js +50 -0
  56. package/lib/cjs/server/index.js +83 -0
  57. package/lib/cjs/server/loaders/babel/creevey-plugin.js +86 -0
  58. package/lib/cjs/server/loaders/babel/helpers.js +469 -0
  59. package/lib/cjs/server/loaders/babel/register.js +124 -0
  60. package/lib/cjs/server/loaders/hooks/mdx.js +30 -0
  61. package/lib/cjs/server/loaders/hooks/svelte.js +65 -0
  62. package/lib/cjs/server/loaders/webpack/compile.js +293 -0
  63. package/lib/cjs/server/loaders/webpack/creevey-loader.js +179 -0
  64. package/lib/cjs/server/loaders/webpack/dummy-hmr.js +39 -0
  65. package/lib/cjs/server/loaders/webpack/mdx-loader.js +72 -0
  66. package/lib/cjs/server/loaders/webpack/start.js +41 -0
  67. package/lib/cjs/server/logger.js +48 -0
  68. package/lib/cjs/server/master/api.js +71 -0
  69. package/lib/cjs/server/master/index.js +146 -0
  70. package/lib/cjs/server/master/master.js +57 -0
  71. package/lib/cjs/server/master/pool.js +197 -0
  72. package/lib/cjs/server/master/runner.js +281 -0
  73. package/lib/cjs/server/master/server.js +129 -0
  74. package/lib/cjs/server/messages.js +264 -0
  75. package/lib/cjs/server/selenium/browser.js +672 -0
  76. package/lib/cjs/server/selenium/index.js +31 -0
  77. package/lib/cjs/server/selenium/selenoid.js +172 -0
  78. package/lib/cjs/server/stories.js +159 -0
  79. package/lib/cjs/server/storybook/entry.js +70 -0
  80. package/lib/cjs/server/storybook/helpers.js +159 -0
  81. package/lib/cjs/server/storybook/providers/browser.js +74 -0
  82. package/lib/cjs/server/storybook/providers/hybrid.js +84 -0
  83. package/lib/cjs/server/storybook/providers/nodejs.js +239 -0
  84. package/lib/cjs/server/testsFiles/parser.js +72 -0
  85. package/lib/cjs/server/testsFiles/register.js +48 -0
  86. package/lib/cjs/server/update.js +79 -0
  87. package/lib/cjs/server/utils.js +183 -0
  88. package/lib/cjs/server/worker/chai-image.js +142 -0
  89. package/lib/cjs/server/worker/helpers.js +69 -0
  90. package/lib/cjs/server/worker/index.js +15 -0
  91. package/lib/cjs/server/worker/reporter.js +108 -0
  92. package/lib/cjs/server/worker/worker.js +268 -0
  93. package/lib/cjs/shared/index.js +89 -0
  94. package/lib/cjs/shared/serializeRegExp.js +41 -0
  95. package/lib/cjs/types.js +74 -0
  96. package/lib/esm/cli.js +4 -0
  97. package/lib/esm/client/addon/Manager.js +402 -0
  98. package/lib/esm/client/addon/components/Addon.js +58 -0
  99. package/lib/esm/client/addon/components/Icons.js +27 -0
  100. package/lib/esm/client/addon/components/Panel.js +49 -0
  101. package/lib/esm/client/addon/components/TestSelect.js +49 -0
  102. package/lib/esm/client/addon/components/Tools.js +91 -0
  103. package/lib/esm/client/addon/decorator.js +2 -0
  104. package/lib/esm/client/addon/index.js +2 -0
  105. package/lib/esm/client/addon/preset.js +56 -0
  106. package/lib/esm/client/addon/readyForCapture.js +5 -0
  107. package/lib/esm/client/addon/register.js +79 -0
  108. package/lib/esm/client/addon/utils.js +31 -0
  109. package/lib/esm/client/addon/withCreevey.js +534 -0
  110. package/lib/esm/client/shared/components/ImagesView/BlendView.js +63 -0
  111. package/lib/esm/client/shared/components/ImagesView/ImagesView.js +65 -0
  112. package/lib/esm/client/shared/components/ImagesView/SideBySideView.js +151 -0
  113. package/lib/esm/client/shared/components/ImagesView/SlideView.js +154 -0
  114. package/lib/esm/client/shared/components/ImagesView/SwapView.js +88 -0
  115. package/lib/esm/client/shared/components/ImagesView/index.js +5 -0
  116. package/lib/esm/client/shared/components/PageFooter/PageFooter.js +32 -0
  117. package/lib/esm/client/shared/components/PageFooter/Paging.js +84 -0
  118. package/lib/esm/client/shared/components/PageHeader/ImagePreview.js +64 -0
  119. package/lib/esm/client/shared/components/PageHeader/PageHeader.js +120 -0
  120. package/lib/esm/client/shared/components/ResultsPage.js +143 -0
  121. package/lib/esm/client/shared/creeveyClientApi.js +98 -0
  122. package/lib/esm/client/shared/helpers.js +424 -0
  123. package/lib/esm/client/shared/viewMode.js +6 -0
  124. package/lib/esm/creevey.js +54 -0
  125. package/lib/esm/index.js +5 -0
  126. package/lib/esm/server/config.js +73 -0
  127. package/lib/esm/server/docker.js +123 -0
  128. package/lib/esm/server/extract.js +34 -0
  129. package/lib/esm/server/index.js +64 -0
  130. package/lib/esm/server/loaders/babel/creevey-plugin.js +72 -0
  131. package/lib/esm/server/loaders/babel/helpers.js +452 -0
  132. package/lib/esm/server/loaders/babel/register.js +103 -0
  133. package/lib/esm/server/loaders/hooks/mdx.js +15 -0
  134. package/lib/esm/server/loaders/hooks/svelte.js +49 -0
  135. package/lib/esm/server/loaders/webpack/compile.js +270 -0
  136. package/lib/esm/server/loaders/webpack/creevey-loader.js +158 -0
  137. package/lib/esm/server/loaders/webpack/dummy-hmr.js +32 -0
  138. package/lib/esm/server/loaders/webpack/mdx-loader.js +58 -0
  139. package/lib/esm/server/loaders/webpack/start.js +27 -0
  140. package/lib/esm/server/logger.js +20 -0
  141. package/lib/esm/server/master/api.js +60 -0
  142. package/lib/esm/server/master/index.js +125 -0
  143. package/lib/esm/server/master/master.js +38 -0
  144. package/lib/esm/server/master/pool.js +176 -0
  145. package/lib/esm/server/master/runner.js +259 -0
  146. package/lib/esm/server/master/server.js +105 -0
  147. package/lib/esm/server/messages.js +232 -0
  148. package/lib/esm/server/selenium/browser.js +639 -0
  149. package/lib/esm/server/selenium/index.js +2 -0
  150. package/lib/esm/server/selenium/selenoid.js +149 -0
  151. package/lib/esm/server/stories.js +140 -0
  152. package/lib/esm/server/storybook/entry.js +46 -0
  153. package/lib/esm/server/storybook/helpers.js +98 -0
  154. package/lib/esm/server/storybook/providers/browser.js +60 -0
  155. package/lib/esm/server/storybook/providers/hybrid.js +64 -0
  156. package/lib/esm/server/storybook/providers/nodejs.js +217 -0
  157. package/lib/esm/server/testsFiles/parser.js +50 -0
  158. package/lib/esm/server/testsFiles/register.js +35 -0
  159. package/lib/esm/server/update.js +61 -0
  160. package/lib/esm/server/utils.js +142 -0
  161. package/lib/esm/server/worker/chai-image.js +130 -0
  162. package/lib/esm/server/worker/helpers.js +60 -0
  163. package/lib/esm/server/worker/index.js +1 -0
  164. package/lib/esm/server/worker/reporter.js +86 -0
  165. package/lib/esm/server/worker/worker.js +238 -0
  166. package/lib/esm/shared/index.js +66 -0
  167. package/lib/esm/shared/serializeRegExp.js +23 -0
  168. package/lib/esm/types.js +43 -0
  169. package/lib/types/cli.d.ts +1 -1
  170. package/lib/types/client/addon/Manager.d.ts +37 -37
  171. package/lib/types/client/addon/components/Addon.d.ts +8 -8
  172. package/lib/types/client/addon/components/Icons.d.ts +7 -7
  173. package/lib/types/client/addon/components/Panel.d.ts +9 -9
  174. package/lib/types/client/addon/components/TestSelect.d.ts +8 -9
  175. package/lib/types/client/addon/components/Tools.d.ts +6 -6
  176. package/lib/types/client/addon/decorator.d.ts +1 -1
  177. package/lib/types/client/addon/index.d.ts +2 -0
  178. package/lib/types/client/addon/preset.d.ts +23 -24
  179. package/lib/types/client/addon/readyForCapture.d.ts +6 -6
  180. package/lib/types/client/addon/register.d.ts +3 -3
  181. package/lib/types/client/addon/utils.d.ts +2 -2
  182. package/lib/types/client/addon/withCreevey.d.ts +24 -24
  183. package/lib/types/client/shared/components/ImagesView/BlendView.d.ts +3 -3
  184. package/lib/types/client/shared/components/ImagesView/ImagesView.d.ts +24 -25
  185. package/lib/types/client/shared/components/ImagesView/SideBySideView.d.ts +3 -3
  186. package/lib/types/client/shared/components/ImagesView/SlideView.d.ts +3 -3
  187. package/lib/types/client/shared/components/ImagesView/SwapView.d.ts +3 -3
  188. package/lib/types/client/shared/components/ImagesView/index.d.ts +5 -5
  189. package/lib/types/client/shared/components/PageFooter/PageFooter.d.ts +8 -9
  190. package/lib/types/client/shared/components/PageFooter/Paging.d.ts +7 -8
  191. package/lib/types/client/shared/components/PageHeader/ImagePreview.d.ts +12 -12
  192. package/lib/types/client/shared/components/PageHeader/PageHeader.d.ts +16 -17
  193. package/lib/types/client/shared/components/ResultsPage.d.ts +18 -18
  194. package/lib/types/client/shared/creeveyClientApi.d.ts +9 -9
  195. package/lib/types/client/shared/helpers.d.ts +46 -46
  196. package/lib/types/client/shared/viewMode.d.ts +4 -4
  197. package/lib/types/client/web/CreeveyApp.d.ts +11 -12
  198. package/lib/types/client/web/CreeveyContext.d.ts +11 -11
  199. package/lib/types/client/web/CreeveyLoader.d.ts +2 -3
  200. package/lib/types/client/web/CreeveyView/SideBar/Checkbox.d.ts +19 -19
  201. package/lib/types/client/web/CreeveyView/SideBar/Search.d.ts +6 -6
  202. package/lib/types/client/web/CreeveyView/SideBar/SideBar.d.ts +14 -14
  203. package/lib/types/client/web/CreeveyView/SideBar/SideBarHeader.d.ts +12 -13
  204. package/lib/types/client/web/CreeveyView/SideBar/SuiteLink.d.ts +33 -33
  205. package/lib/types/client/web/CreeveyView/SideBar/TestLink.d.ts +7 -8
  206. package/lib/types/client/web/CreeveyView/SideBar/TestStatusIcon.d.ts +10 -10
  207. package/lib/types/client/web/CreeveyView/SideBar/TestsStatus.d.ts +9 -9
  208. package/lib/types/client/web/CreeveyView/SideBar/Toggle.d.ts +6 -6
  209. package/lib/types/client/web/CreeveyView/SideBar/index.d.ts +1 -1
  210. package/lib/types/client/web/KeyboardEventsContext.d.ts +13 -13
  211. package/lib/types/client/web/index.d.ts +4 -4
  212. package/lib/types/creevey.d.ts +1 -1
  213. package/lib/types/index.d.ts +1 -4
  214. package/lib/types/server/config.d.ts +4 -4
  215. package/lib/types/server/docker.d.ts +7 -7
  216. package/lib/types/server/extract.d.ts +2 -2
  217. package/lib/types/server/index.d.ts +2 -2
  218. package/lib/types/server/loaders/babel/creevey-plugin.d.ts +1 -1
  219. package/lib/types/server/loaders/babel/helpers.d.ts +19 -19
  220. package/lib/types/server/loaders/babel/register.d.ts +5 -5
  221. package/lib/types/server/loaders/hooks/mdx.d.ts +1 -1
  222. package/lib/types/server/loaders/hooks/svelte.d.ts +1 -1
  223. package/lib/types/server/loaders/webpack/compile.d.ts +2 -2
  224. package/lib/types/server/loaders/webpack/creevey-loader.d.ts +4 -2
  225. package/lib/types/server/loaders/webpack/dummy-hmr.d.ts +10 -10
  226. package/lib/types/server/loaders/webpack/mdx-loader.d.ts +6 -6
  227. package/lib/types/server/loaders/webpack/start.d.ts +1 -1
  228. package/lib/types/server/logger.d.ts +10 -6
  229. package/lib/types/server/master/api.d.ts +7 -7
  230. package/lib/types/server/master/index.d.ts +3 -3
  231. package/lib/types/server/master/master.d.ts +7 -7
  232. package/lib/types/server/master/pool.d.ts +31 -31
  233. package/lib/types/server/master/runner.d.ts +26 -26
  234. package/lib/types/server/master/server.d.ts +2 -2
  235. package/lib/types/server/messages.d.ts +27 -27
  236. package/lib/types/server/selenium/browser.d.ts +17 -17
  237. package/lib/types/server/selenium/index.d.ts +2 -2
  238. package/lib/types/server/selenium/selenoid.d.ts +3 -3
  239. package/lib/types/server/stories.d.ts +8 -8
  240. package/lib/types/server/storybook/entry.d.ts +18 -18
  241. package/lib/types/server/storybook/helpers.d.ts +24 -24
  242. package/lib/types/server/storybook/providers/browser.d.ts +4 -4
  243. package/lib/types/server/storybook/providers/hybrid.d.ts +4 -4
  244. package/lib/types/server/storybook/providers/nodejs.d.ts +9 -9
  245. package/lib/types/server/testsFiles/parser.d.ts +12 -12
  246. package/lib/types/server/testsFiles/register.d.ts +2 -2
  247. package/lib/types/server/update.d.ts +2 -2
  248. package/lib/types/server/utils.d.ts +20 -20
  249. package/lib/types/server/worker/chai-image.d.ts +6 -6
  250. package/lib/types/server/worker/helpers.d.ts +8 -8
  251. package/lib/types/server/worker/index.d.ts +1 -1
  252. package/lib/types/server/worker/reporter.d.ts +8 -8
  253. package/lib/types/server/worker/worker.d.ts +4 -4
  254. package/lib/types/{shared.d.ts → shared/index.d.ts} +7 -16
  255. package/lib/types/shared/serializeRegExp.d.ts +9 -0
  256. package/lib/types/types.d.ts +490 -489
  257. package/package.json +115 -102
  258. package/preset.js +9 -9
  259. package/types/babel__register.d.ts +1 -1
  260. package/types/chai.d.ts +12 -12
  261. package/types/event-source-polyfill.d.ts +6 -6
  262. package/types/mdx.d.ts +3 -2
  263. package/types/mocha.d.ts +20 -20
  264. package/types/png.d.ts +4 -4
  265. package/lib/cjs/client/web/1.js +0 -13
  266. package/lib/cjs/client/web/2.js +0 -1
  267. 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,140 @@
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
+ import { isStorybookVersionLessThan } from './storybook/helpers';
8
+
9
+ function storyTestFabric(delay, testFn) {
10
+ return async function storyTest() {
11
+ delay ? await new Promise(resolve => setTimeout(resolve, delay)) : void 0;
12
+ await (testFn ? testFn.call(this) : this.screenshots.length > 0 ? this.expect(this.screenshots.reduce((screenshots, {
13
+ imageName,
14
+ screenshot
15
+ }, index) => ({ ...screenshots,
16
+ [imageName ?? `screenshot_${index}`]: screenshot
17
+ }), {})).to.matchImages() : this.expect(await this.takeScreenshot()).to.matchImage());
18
+ };
19
+ }
20
+
21
+ function createCreeveyTest(browser, storyMeta, skipOptions, testName) {
22
+ const {
23
+ kind,
24
+ name: story,
25
+ id: storyId
26
+ } = storyMeta;
27
+ const path = [kind, story, testName, browser].filter(isDefined);
28
+ const skip = skipOptions ? shouldSkip(browser, {
29
+ kind,
30
+ story
31
+ }, skipOptions, testName) : false;
32
+ const id = createHash('sha1').update(path.join('/')).digest('hex');
33
+ return {
34
+ id,
35
+ skip,
36
+ browser,
37
+ testName,
38
+ storyPath: [...kind.split('/').map(x => x.trim()), story],
39
+ storyId
40
+ };
41
+ }
42
+
43
+ function convertStories(browserName, stories) {
44
+ const tests = {};
45
+ (Array.isArray(stories) ? stories : Object.values(stories)).forEach(storyMeta => {
46
+ // TODO Skip docsOnly stories for now
47
+ if (storyMeta.parameters.docsOnly) return;
48
+ const {
49
+ delay: delayParam,
50
+ tests: storyTests,
51
+ skip
52
+ } = storyMeta.parameters.creevey ?? {};
53
+ 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]
54
+ // typeof tests === "function" => rootSuite -> kindSuite -> storyTest -> browser -> [images.png]
55
+ // typeof tests === "object" => rootSuite -> kindSuite -> storySuite -> test -> [browsers.png]
56
+ // typeof tests === "object" => rootSuite -> kindSuite -> storySuite -> test -> browser -> [images.png]
57
+
58
+ if (!storyTests) {
59
+ const test = createCreeveyTest(browserName, storyMeta, skip);
60
+ tests[test.id] = { ...test,
61
+ storyId: storyMeta.id,
62
+ story: storyMeta,
63
+ fn: storyTestFabric(delay)
64
+ };
65
+ return;
66
+ }
67
+
68
+ Object.entries(storyTests).forEach(([testName, testFn]) => {
69
+ const test = createCreeveyTest(browserName, storyMeta, skip, testName);
70
+ tests[test.id] = { ...test,
71
+ storyId: storyMeta.id,
72
+ story: storyMeta,
73
+ fn: storyTestFabric(delay, testFn)
74
+ };
75
+ });
76
+ });
77
+ return tests;
78
+ }
79
+
80
+ export async function loadTestsFromStories(browsers, provider, update) {
81
+ const testIdsByFiles = new Map();
82
+ const stories = await provider(storiesByFiles => {
83
+ const testsDiff = {};
84
+ const tests = {};
85
+ browsers.forEach(browser => {
86
+ Array.from(storiesByFiles.entries()).forEach(([filename, stories]) => {
87
+ var _testIdsByFiles$get;
88
+
89
+ Object.assign(tests, convertStories(browser, stories));
90
+ const changed = Object.keys(tests);
91
+ const removed = ((_testIdsByFiles$get = testIdsByFiles.get(filename)) === null || _testIdsByFiles$get === void 0 ? void 0 : _testIdsByFiles$get.filter(testId => !tests[testId])) ?? [];
92
+ if (changed.length == 0) testIdsByFiles.delete(filename);else testIdsByFiles.set(filename, changed);
93
+ Object.assign(testsDiff, tests);
94
+ removed.forEach(testId => testsDiff[testId] = undefined);
95
+ });
96
+ });
97
+ update === null || update === void 0 ? void 0 : update(testsDiff);
98
+ });
99
+ const tests = browsers.reduce((tests, browser) => Object.assign(tests, convertStories(browser, stories)), {});
100
+ Object.values(tests).filter(isDefined).forEach(({
101
+ id,
102
+ story: {
103
+ parameters: {
104
+ fileName
105
+ }
106
+ }
107
+ }) => // TODO Don't use filename as a key, due possible collisions if two require.context with same structure of modules are defined
108
+ testIdsByFiles.set(fileName, [...(testIdsByFiles.get(fileName) ?? []), id]));
109
+ return tests;
110
+ }
111
+ export function saveStoriesJson(storiesData, extract) {
112
+ const outputDir = typeof extract == 'boolean' ? 'storybook-static' : extract;
113
+
114
+ if (!isStorybookVersionLessThan(6)) {
115
+ // NOTE Copy-pasted from Storybook's `getStoriesJsonData` method
116
+ const allowed = ['fileName', 'docsOnly', 'framework', '__id', '__isArgsStory'];
117
+ storiesData.globalParameters = pick(storiesData.globalParameters, allowed); // @ts-expect-error ignore error
118
+
119
+ storiesData.kindParameters = mapValues(storiesData.kindParameters, v => pick(v, allowed)); // @ts-expect-error ignore error
120
+
121
+ storiesData.stories = mapValues(storiesData.stories, v => ({ ...pick(v, ['id', 'name', 'kind', 'story']),
122
+ parameters: pick(v.parameters, allowed)
123
+ }));
124
+ } // TODO Fix args stories
125
+
126
+
127
+ removeProps(storiesData ?? {}, ['stories', () => true, 'parameters', '__isArgsStory']);
128
+ 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);
129
+ mkdirSync(outputDir, {
130
+ recursive: true
131
+ });
132
+ writeFileSync(path.join(outputDir, 'stories.json'), JSON.stringify(storiesData, null, 2));
133
+ }
134
+ export function saveTestsJson(tests, dstPath = process.cwd()) {
135
+ mkdirSync(dstPath, {
136
+ recursive: true
137
+ });
138
+ writeFileSync(path.join(dstPath, 'tests.json'), // eslint-disable-next-line @typescript-eslint/no-unsafe-return
139
+ JSON.stringify(tests, (_, value) => isFunction(value) ? value.toString() : value, 2));
140
+ }
@@ -0,0 +1,46 @@
1
+ var _api$context;
2
+
3
+ import { addons } from '@storybook/addons';
4
+ import { getStorybookFramework, isStorybookVersionLessThan, resolveFromStorybook } from './helpers';
5
+ const framework = getStorybookFramework(); // eslint-disable-next-line @typescript-eslint/no-var-requires
6
+
7
+ const core = require(resolveFromStorybook('@storybook/core')); //@ts-expect-error: 6.2 use named exports
8
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
9
+
10
+
11
+ const start = isStorybookVersionLessThan(6, 2) ? core.default.start : core.start;
12
+ const api = start(() => void 0);
13
+ export const channel = isStorybookVersionLessThan(6, 4) ? //@ts-expect-error: 6.x has { channel }, but 5.x has { context: { channel } }
14
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
15
+ api.channel ?? ((_api$context = api.context) === null || _api$context === void 0 ? void 0 : _api$context.channel) : addons.getChannel();
16
+ export const clientApi = api.clientApi;
17
+ export const forceReRender = api.forceReRender;
18
+ export const storiesOf = (kind, m) => {
19
+ return clientApi.storiesOf(kind, m).addParameters({
20
+ framework
21
+ });
22
+ };
23
+ export const configure = (...args) => {
24
+ if (isStorybookVersionLessThan(5, 2)) {
25
+ //NOTE: Storybook <= 5.1 pass args as is
26
+ //@ts-expect-error: ignore it
27
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-call
28
+ return api.configApi.configure(...args);
29
+ }
30
+
31
+ if (isStorybookVersionLessThan(6)) {
32
+ //NOTE: Storybook <= 5.3 pass `framework` as last argument
33
+ //@ts-expect-error: ignore it
34
+ return api.configure(...args, framework);
35
+ } //NOTE Storybook 6.x pass `framework` as first argument
36
+ //@ts-expect-error: ignore it
37
+
38
+
39
+ return api.configure(framework, ...args);
40
+ };
41
+ export const addDecorator = clientApi.addDecorator;
42
+ export const addParameters = clientApi.addParameters;
43
+ export const clearDecorators = clientApi.clearDecorators;
44
+ export const setAddon = clientApi.setAddon;
45
+ export const getStorybook = clientApi.getStorybook;
46
+ export const raw = clientApi.raw;
@@ -0,0 +1,98 @@
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 resolveFromStorybookCore = modulePath => resolveFrom(resolveFromStorybook('@storybook/core'), 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 = isStorybookVersionLessThan(6, 2) ? '@storybook/core/dist/server/utils' : '@storybook/core-common/dist/cjs/utils';
77
+ const serverRequireModule = isStorybookVersionLessThan(6, 2) ? 'server-require' : '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
87
+
88
+ const mainConfigFile = isStorybookVersionLessThan(6, 1) ? configPath : // eslint-disable-next-line @typescript-eslint/no-unsafe-call
89
+ getInterpretedFile(configPath); // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call
90
+
91
+ return storybookConfigRef.current = serverRequire(mainConfigFile);
92
+ }
93
+ }
94
+ export async function isCSFv3Enabled() {
95
+ var _await$importStoryboo, _await$importStoryboo2;
96
+
97
+ 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;
98
+ }
@@ -0,0 +1,60 @@
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 async function loadStories(_config, {
7
+ port
8
+ }, storiesListener) {
9
+ if (cluster.isPrimary) {
10
+ return new Promise(resolve => {
11
+ const worker = Object.values(cluster.workers ?? {}).filter(isDefined).find(worker => worker.isConnected());
12
+
13
+ if (worker) {
14
+ const unsubscribe = subscribeOnWorker(worker, 'stories', message => {
15
+ if (message.type == 'set') {
16
+ const {
17
+ stories,
18
+ oldTests
19
+ } = message.payload;
20
+ 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'));
21
+ unsubscribe();
22
+ resolve(stories);
23
+ }
24
+ });
25
+ sendStoriesMessage(worker, {
26
+ type: 'get'
27
+ });
28
+ }
29
+
30
+ subscribeOn('stories', message => {
31
+ // TODO updates only one browser :(
32
+ if (message.type == 'update') storiesListener(new Map(message.payload));
33
+ });
34
+ });
35
+ } else {
36
+ subscribeOn('stories', message => {
37
+ if (message.type == 'get') emitStoriesMessage({
38
+ type: 'set',
39
+ payload: {
40
+ stories,
41
+ oldTests: storiesWithOldTests
42
+ }
43
+ });
44
+ if (message.type == 'update') storiesListener(new Map(message.payload));
45
+ });
46
+ const stories = await loadStoriesFromBrowser(port);
47
+ const storiesWithOldTests = [];
48
+ Object.values(stories).forEach(story => {
49
+ var _parameters, _parameters$creevey;
50
+
51
+ if ((_parameters = story.parameters) !== null && _parameters !== void 0 && (_parameters$creevey = _parameters.creevey) !== null && _parameters$creevey !== void 0 && _parameters$creevey.tests) {
52
+ var _parameters2, _parameters2$creevey;
53
+
54
+ (_parameters2 = story.parameters) === null || _parameters2 === void 0 ? true : (_parameters2$creevey = _parameters2.creevey) === null || _parameters2$creevey === void 0 ? true : delete _parameters2$creevey.tests;
55
+ storiesWithOldTests.push(`${story.kind}/${story.name}`);
56
+ }
57
+ });
58
+ return stories;
59
+ }
60
+ }
@@ -0,0 +1,64 @@
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 async function loadStories(_config, {
8
+ port
9
+ }, storiesListener) {
10
+ let creeveyParamsByStoryId = {};
11
+
12
+ const mergeParamsFromTestsToStory = (story, creeveyParams) => {
13
+ if (story.parameters) {
14
+ story.parameters.creevey = combineParameters(story.parameters.creevey || {}, creeveyParams);
15
+ }
16
+ };
17
+
18
+ const stories = await browserProvider(_config, {
19
+ port
20
+ }, updatedStoriesByFiles => {
21
+ Array.from(updatedStoriesByFiles.entries()).forEach(([, storiesArray]) => {
22
+ storiesArray.forEach(story => {
23
+ const creeveyParams = creeveyParamsByStoryId[story.id];
24
+
25
+ if (creeveyParams) {
26
+ mergeParamsFromTestsToStory(story, creeveyParams);
27
+ }
28
+ });
29
+ });
30
+ storiesListener(updatedStoriesByFiles);
31
+ }); // TODO fix test files hot reloading
32
+
33
+ creeveyParamsByStoryId = await parseParams(_config
34
+ /*, (data) => console.log(data) */
35
+ );
36
+ Object.entries(stories).forEach(([storyId, story]) => {
37
+ mergeParamsFromTestsToStory(story, creeveyParamsByStoryId[storyId]);
38
+ });
39
+ return stories;
40
+ }
41
+
42
+ async function parseParams(config, listener) {
43
+ if (!config.testsDir) {
44
+ return Promise.resolve({});
45
+ }
46
+
47
+ const testFiles = readDirRecursive(config.testsDir).filter(file => {
48
+ var _config$testsRegex;
49
+
50
+ return (_config$testsRegex = config.testsRegex) === null || _config$testsRegex === void 0 ? void 0 : _config$testsRegex.test(file);
51
+ });
52
+ await (await import('../../testsFiles/register')).default(config);
53
+
54
+ if (listener) {
55
+ chokidar.watch(testFiles).on('change', filePath => {
56
+ logger.debug(`changed: ${filePath}`); // doesn't work, always returns {} due modules caching
57
+ // see https://github.com/nodejs/modules/issues/307
58
+
59
+ void parse(testFiles).then(data => listener(data));
60
+ });
61
+ }
62
+
63
+ return parse(testFiles);
64
+ }