creevey 0.8.1-sb7.1 → 0.8.1-sb7.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 (235) hide show
  1. package/lib/cjs/cli.js +5 -0
  2. package/lib/cjs/client/addon/Manager.js +264 -0
  3. package/lib/cjs/client/addon/components/Addon.js +55 -0
  4. package/lib/cjs/client/addon/components/Icons.js +46 -0
  5. package/lib/cjs/client/addon/components/Panel.js +72 -0
  6. package/lib/cjs/client/addon/components/TestSelect.js +65 -0
  7. package/lib/cjs/client/addon/components/Tools.js +97 -0
  8. package/lib/cjs/client/addon/decorator.js +11 -0
  9. package/lib/cjs/client/addon/index.js +31 -0
  10. package/lib/cjs/client/addon/preset.ie11.js +74 -0
  11. package/lib/cjs/client/addon/preset.js +62 -0
  12. package/lib/cjs/client/addon/readyForCapture.js +12 -0
  13. package/lib/cjs/client/addon/register.js +74 -0
  14. package/lib/cjs/client/addon/utils.js +42 -0
  15. package/lib/cjs/client/addon/withCreevey.js +354 -0
  16. package/lib/cjs/client/shared/components/ImagesView/BlendView.js +87 -0
  17. package/lib/cjs/client/shared/components/ImagesView/ImagesView.js +92 -0
  18. package/lib/cjs/client/shared/components/ImagesView/SideBySideView.js +154 -0
  19. package/lib/cjs/client/shared/components/ImagesView/SlideView.js +166 -0
  20. package/lib/cjs/client/shared/components/ImagesView/SwapView.js +91 -0
  21. package/lib/cjs/client/shared/components/ImagesView/index.js +45 -0
  22. package/lib/cjs/client/shared/components/PageFooter/PageFooter.js +50 -0
  23. package/lib/cjs/client/shared/components/PageFooter/Paging.js +94 -0
  24. package/lib/cjs/client/shared/components/PageHeader/ImagePreview.js +82 -0
  25. package/lib/cjs/client/shared/components/PageHeader/PageHeader.js +119 -0
  26. package/lib/cjs/client/shared/components/ResultsPage.js +143 -0
  27. package/lib/cjs/client/shared/creeveyClientApi.js +76 -0
  28. package/lib/cjs/client/shared/helpers.js +411 -0
  29. package/lib/cjs/client/shared/viewMode.js +17 -0
  30. package/lib/cjs/client/web/19.js +1 -0
  31. package/lib/cjs/client/web/632.js +43 -0
  32. package/lib/cjs/client/web/794.js +1 -0
  33. package/lib/cjs/client/web/index.html +19 -0
  34. package/lib/cjs/client/web/main.js +79 -0
  35. package/lib/cjs/client/web/main.js.LICENSE.txt +34 -0
  36. package/lib/cjs/creevey.js +69 -0
  37. package/lib/cjs/index.js +39 -0
  38. package/lib/cjs/server/config.js +93 -0
  39. package/lib/cjs/server/docker.js +146 -0
  40. package/lib/cjs/server/extract.js +46 -0
  41. package/lib/cjs/server/index.js +83 -0
  42. package/lib/cjs/server/loaders/babel/creevey-plugin.js +86 -0
  43. package/lib/cjs/server/loaders/babel/helpers.js +469 -0
  44. package/lib/cjs/server/loaders/babel/register.js +124 -0
  45. package/lib/cjs/server/loaders/hooks/mdx.js +30 -0
  46. package/lib/cjs/server/loaders/hooks/svelte.js +65 -0
  47. package/lib/cjs/server/loaders/webpack/compile.js +269 -0
  48. package/lib/cjs/server/loaders/webpack/creevey-loader.js +172 -0
  49. package/lib/cjs/server/loaders/webpack/dummy-hmr.js +39 -0
  50. package/lib/cjs/server/loaders/webpack/mdx-loader.js +72 -0
  51. package/lib/cjs/server/loaders/webpack/start.js +41 -0
  52. package/lib/cjs/server/logger.js +48 -0
  53. package/lib/cjs/server/master/api.js +71 -0
  54. package/lib/cjs/server/master/index.js +152 -0
  55. package/lib/cjs/server/master/master.js +57 -0
  56. package/lib/cjs/server/master/pool.js +197 -0
  57. package/lib/cjs/server/master/runner.js +281 -0
  58. package/lib/cjs/server/master/server.js +131 -0
  59. package/lib/cjs/server/messages.js +264 -0
  60. package/lib/cjs/server/selenium/browser.js +667 -0
  61. package/lib/cjs/server/selenium/index.js +31 -0
  62. package/lib/cjs/server/selenium/selenoid.js +172 -0
  63. package/lib/cjs/server/stories.js +155 -0
  64. package/lib/cjs/server/storybook/entry.js +55 -0
  65. package/lib/cjs/server/storybook/helpers.js +158 -0
  66. package/lib/cjs/server/storybook/providers/browser.js +76 -0
  67. package/lib/cjs/server/storybook/providers/nodejs.js +239 -0
  68. package/lib/cjs/server/update.js +79 -0
  69. package/lib/cjs/server/utils.js +169 -0
  70. package/lib/cjs/server/worker/chai-image.js +142 -0
  71. package/lib/cjs/server/worker/helpers.js +69 -0
  72. package/lib/cjs/server/worker/index.js +15 -0
  73. package/lib/cjs/server/worker/reporter.js +108 -0
  74. package/lib/cjs/server/worker/worker.js +268 -0
  75. package/lib/cjs/shared/index.js +103 -0
  76. package/lib/cjs/shared/serializeRegExp.js +42 -0
  77. package/lib/cjs/types.js +77 -0
  78. package/lib/esm/cli.js +4 -0
  79. package/lib/esm/client/addon/Manager.js +252 -0
  80. package/lib/esm/client/addon/components/Addon.js +39 -0
  81. package/lib/esm/client/addon/components/Icons.js +31 -0
  82. package/lib/esm/client/addon/components/Panel.js +53 -0
  83. package/lib/esm/client/addon/components/TestSelect.js +51 -0
  84. package/lib/esm/client/addon/components/Tools.js +76 -0
  85. package/lib/esm/client/addon/decorator.js +2 -0
  86. package/lib/esm/client/addon/index.js +2 -0
  87. package/lib/esm/client/addon/preset.ie11.js +59 -0
  88. package/lib/esm/client/addon/preset.js +41 -0
  89. package/lib/esm/client/addon/readyForCapture.js +5 -0
  90. package/lib/esm/client/addon/register.js +53 -0
  91. package/lib/esm/client/addon/utils.js +32 -0
  92. package/lib/esm/client/addon/withCreevey.js +327 -0
  93. package/lib/esm/client/shared/components/ImagesView/BlendView.js +67 -0
  94. package/lib/esm/client/shared/components/ImagesView/ImagesView.js +69 -0
  95. package/lib/esm/client/shared/components/ImagesView/SideBySideView.js +131 -0
  96. package/lib/esm/client/shared/components/ImagesView/SlideView.js +143 -0
  97. package/lib/esm/client/shared/components/ImagesView/SwapView.js +71 -0
  98. package/lib/esm/client/shared/components/ImagesView/index.js +5 -0
  99. package/lib/esm/client/shared/components/PageFooter/PageFooter.js +36 -0
  100. package/lib/esm/client/shared/components/PageFooter/Paging.js +80 -0
  101. package/lib/esm/client/shared/components/PageHeader/ImagePreview.js +68 -0
  102. package/lib/esm/client/shared/components/PageHeader/PageHeader.js +97 -0
  103. package/lib/esm/client/shared/components/ResultsPage.js +115 -0
  104. package/lib/esm/client/shared/creeveyClientApi.js +67 -0
  105. package/lib/esm/client/shared/helpers.js +353 -0
  106. package/lib/esm/client/shared/viewMode.js +6 -0
  107. package/lib/esm/creevey.js +54 -0
  108. package/lib/esm/index.js +3 -0
  109. package/lib/esm/server/config.js +70 -0
  110. package/lib/esm/server/docker.js +123 -0
  111. package/lib/esm/server/extract.js +32 -0
  112. package/lib/esm/server/index.js +64 -0
  113. package/lib/esm/server/loaders/babel/creevey-plugin.js +72 -0
  114. package/lib/esm/server/loaders/babel/helpers.js +452 -0
  115. package/lib/esm/server/loaders/babel/register.js +103 -0
  116. package/lib/esm/server/loaders/hooks/mdx.js +15 -0
  117. package/lib/esm/server/loaders/hooks/svelte.js +49 -0
  118. package/lib/esm/server/loaders/webpack/compile.js +246 -0
  119. package/lib/esm/server/loaders/webpack/creevey-loader.js +152 -0
  120. package/lib/esm/server/loaders/webpack/dummy-hmr.js +32 -0
  121. package/lib/esm/server/loaders/webpack/mdx-loader.js +58 -0
  122. package/lib/esm/server/loaders/webpack/start.js +27 -0
  123. package/lib/esm/server/logger.js +20 -0
  124. package/lib/esm/server/master/api.js +60 -0
  125. package/lib/esm/server/master/index.js +131 -0
  126. package/lib/esm/server/master/master.js +38 -0
  127. package/lib/esm/server/master/pool.js +176 -0
  128. package/lib/esm/server/master/runner.js +259 -0
  129. package/lib/esm/server/master/server.js +107 -0
  130. package/lib/esm/server/messages.js +232 -0
  131. package/lib/esm/server/selenium/browser.js +634 -0
  132. package/lib/esm/server/selenium/index.js +2 -0
  133. package/lib/esm/server/selenium/selenoid.js +149 -0
  134. package/lib/esm/server/stories.js +137 -0
  135. package/lib/esm/server/storybook/entry.js +29 -0
  136. package/lib/esm/server/storybook/helpers.js +99 -0
  137. package/lib/esm/server/storybook/providers/browser.js +60 -0
  138. package/lib/esm/server/storybook/providers/nodejs.js +219 -0
  139. package/lib/esm/server/update.js +61 -0
  140. package/lib/esm/server/utils.js +128 -0
  141. package/lib/esm/server/worker/chai-image.js +130 -0
  142. package/lib/esm/server/worker/helpers.js +60 -0
  143. package/lib/esm/server/worker/index.js +1 -0
  144. package/lib/esm/server/worker/reporter.js +86 -0
  145. package/lib/esm/server/worker/worker.js +238 -0
  146. package/lib/esm/shared/index.js +80 -0
  147. package/lib/esm/shared/serializeRegExp.js +24 -0
  148. package/lib/esm/types.js +46 -0
  149. package/lib/types/cli.d.ts +1 -0
  150. package/lib/types/client/addon/Manager.d.ts +37 -0
  151. package/lib/types/client/addon/components/Addon.d.ts +9 -0
  152. package/lib/types/client/addon/components/Icons.d.ts +8 -0
  153. package/lib/types/client/addon/components/Panel.d.ts +10 -0
  154. package/lib/types/client/addon/components/TestSelect.d.ts +9 -0
  155. package/lib/types/client/addon/components/Tools.d.ts +7 -0
  156. package/lib/types/client/addon/decorator.d.ts +1 -0
  157. package/lib/types/client/addon/index.d.ts +2 -0
  158. package/lib/types/client/addon/preset.d.ts +23 -0
  159. package/lib/types/client/addon/preset.ie11.d.ts +10 -0
  160. package/lib/types/client/addon/readyForCapture.d.ts +6 -0
  161. package/lib/types/client/addon/register.d.ts +3 -0
  162. package/lib/types/client/addon/utils.d.ts +3 -0
  163. package/lib/types/client/addon/withCreevey.d.ts +24 -0
  164. package/lib/types/client/shared/components/ImagesView/BlendView.d.ts +5 -0
  165. package/lib/types/client/shared/components/ImagesView/ImagesView.d.ts +25 -0
  166. package/lib/types/client/shared/components/ImagesView/SideBySideView.d.ts +5 -0
  167. package/lib/types/client/shared/components/ImagesView/SlideView.d.ts +5 -0
  168. package/lib/types/client/shared/components/ImagesView/SwapView.d.ts +5 -0
  169. package/lib/types/client/shared/components/ImagesView/index.d.ts +5 -0
  170. package/lib/types/client/shared/components/PageFooter/PageFooter.d.ts +9 -0
  171. package/lib/types/client/shared/components/PageFooter/Paging.d.ts +8 -0
  172. package/lib/types/client/shared/components/PageHeader/ImagePreview.d.ts +14 -0
  173. package/lib/types/client/shared/components/PageHeader/PageHeader.d.ts +17 -0
  174. package/lib/types/client/shared/components/ResultsPage.d.ts +20 -0
  175. package/lib/types/client/shared/creeveyClientApi.d.ts +9 -0
  176. package/lib/types/client/shared/helpers.d.ts +46 -0
  177. package/lib/types/client/shared/viewMode.d.ts +4 -0
  178. package/lib/types/client/web/CreeveyApp.d.ts +12 -0
  179. package/lib/types/client/web/CreeveyContext.d.ts +11 -0
  180. package/lib/types/client/web/CreeveyLoader.d.ts +3 -0
  181. package/lib/types/client/web/CreeveyView/SideBar/Checkbox.d.ts +22 -0
  182. package/lib/types/client/web/CreeveyView/SideBar/Search.d.ts +7 -0
  183. package/lib/types/client/web/CreeveyView/SideBar/SideBar.d.ts +14 -0
  184. package/lib/types/client/web/CreeveyView/SideBar/SideBarHeader.d.ts +13 -0
  185. package/lib/types/client/web/CreeveyView/SideBar/SuiteLink.d.ts +38 -0
  186. package/lib/types/client/web/CreeveyView/SideBar/TestLink.d.ts +8 -0
  187. package/lib/types/client/web/CreeveyView/SideBar/TestStatusIcon.d.ts +12 -0
  188. package/lib/types/client/web/CreeveyView/SideBar/TestsStatus.d.ts +11 -0
  189. package/lib/types/client/web/CreeveyView/SideBar/Toggle.d.ts +7 -0
  190. package/lib/types/client/web/CreeveyView/SideBar/index.d.ts +1 -0
  191. package/lib/types/client/web/KeyboardEventsContext.d.ts +13 -0
  192. package/lib/types/client/web/index.d.ts +4 -0
  193. package/lib/types/creevey.d.ts +1 -0
  194. package/lib/types/index.d.ts +2 -0
  195. package/lib/types/server/config.d.ts +4 -0
  196. package/lib/types/server/docker.d.ts +7 -0
  197. package/lib/types/server/extract.d.ts +2 -0
  198. package/lib/types/server/index.d.ts +2 -0
  199. package/lib/types/server/loaders/babel/creevey-plugin.d.ts +1 -0
  200. package/lib/types/server/loaders/babel/helpers.d.ts +19 -0
  201. package/lib/types/server/loaders/babel/register.d.ts +5 -0
  202. package/lib/types/server/loaders/hooks/mdx.d.ts +1 -0
  203. package/lib/types/server/loaders/hooks/svelte.d.ts +1 -0
  204. package/lib/types/server/loaders/webpack/compile.d.ts +2 -0
  205. package/lib/types/server/loaders/webpack/creevey-loader.d.ts +4 -0
  206. package/lib/types/server/loaders/webpack/dummy-hmr.d.ts +10 -0
  207. package/lib/types/server/loaders/webpack/mdx-loader.d.ts +6 -0
  208. package/lib/types/server/loaders/webpack/start.d.ts +1 -0
  209. package/lib/types/server/logger.d.ts +10 -0
  210. package/lib/types/server/master/api.d.ts +7 -0
  211. package/lib/types/server/master/index.d.ts +3 -0
  212. package/lib/types/server/master/master.d.ts +7 -0
  213. package/lib/types/server/master/pool.d.ts +31 -0
  214. package/lib/types/server/master/runner.d.ts +26 -0
  215. package/lib/types/server/master/server.d.ts +2 -0
  216. package/lib/types/server/messages.d.ts +29 -0
  217. package/lib/types/server/selenium/browser.d.ts +19 -0
  218. package/lib/types/server/selenium/index.d.ts +2 -0
  219. package/lib/types/server/selenium/selenoid.d.ts +3 -0
  220. package/lib/types/server/stories.d.ts +8 -0
  221. package/lib/types/server/storybook/entry.d.ts +16 -0
  222. package/lib/types/server/storybook/helpers.d.ts +24 -0
  223. package/lib/types/server/storybook/providers/browser.d.ts +2 -0
  224. package/lib/types/server/storybook/providers/nodejs.d.ts +6 -0
  225. package/lib/types/server/update.d.ts +2 -0
  226. package/lib/types/server/utils.d.ts +23 -0
  227. package/lib/types/server/worker/chai-image.d.ts +6 -0
  228. package/lib/types/server/worker/helpers.d.ts +8 -0
  229. package/lib/types/server/worker/index.d.ts +1 -0
  230. package/lib/types/server/worker/reporter.d.ts +8 -0
  231. package/lib/types/server/worker/worker.d.ts +4 -0
  232. package/lib/types/shared/index.d.ts +7 -0
  233. package/lib/types/shared/serializeRegExp.d.ts +9 -0
  234. package/lib/types/types.d.ts +486 -0
  235. package/package.json +1 -1
@@ -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,137 @@
1
+ /* eslint-disable */
2
+ //@ts-nocheck
3
+ import path from 'path';
4
+ import { mkdirSync, writeFileSync } from 'fs';
5
+ import { createHash } from 'crypto';
6
+ import { mapValues, pick } from 'lodash';
7
+ import { isDefined, isFunction, isObject } from '../types';
8
+ import { shouldSkip, removeProps } from './utils';
9
+
10
+ function storyTestFabric(delay, testFn) {
11
+ return async function storyTest() {
12
+ delay ? await new Promise(resolve => setTimeout(resolve, delay)) : void 0;
13
+ await (testFn ? testFn.call(this) : this.screenshots.length > 0 ? this.expect(this.screenshots.reduce((screenshots, {
14
+ imageName,
15
+ screenshot
16
+ }, index) => ({ ...screenshots,
17
+ [imageName ?? `screenshot_${index}`]: screenshot
18
+ }), {})).to.matchImages() : this.expect(await this.takeScreenshot()).to.matchImage());
19
+ };
20
+ }
21
+
22
+ function createCreeveyTest(browser, storyMeta, skipOptions, testName) {
23
+ const {
24
+ kind,
25
+ name: story,
26
+ id: storyId
27
+ } = storyMeta;
28
+ const path = [kind, story, testName, browser].filter(isDefined);
29
+ const skip = skipOptions ? shouldSkip(browser, {
30
+ kind,
31
+ story
32
+ }, skipOptions, testName) : false;
33
+ const id = createHash('sha1').update(path.join('/')).digest('hex');
34
+ return {
35
+ id,
36
+ skip,
37
+ browser,
38
+ testName,
39
+ storyPath: [...kind.split('/').map(x => x.trim()), story],
40
+ storyId
41
+ };
42
+ }
43
+
44
+ function convertStories(browserName, stories) {
45
+ const tests = {};
46
+ (Array.isArray(stories) ? stories : Object.values(stories)).forEach(storyMeta => {
47
+ // TODO Skip docsOnly stories for now
48
+ if (storyMeta.parameters.docsOnly) return;
49
+ const {
50
+ delay: delayParam,
51
+ tests: storyTests,
52
+ skip
53
+ } = storyMeta.parameters.creevey ?? {};
54
+ 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]
55
+ // typeof tests === "function" => rootSuite -> kindSuite -> storyTest -> browser -> [images.png]
56
+ // typeof tests === "object" => rootSuite -> kindSuite -> storySuite -> test -> [browsers.png]
57
+ // typeof tests === "object" => rootSuite -> kindSuite -> storySuite -> test -> browser -> [images.png]
58
+
59
+ if (!storyTests) {
60
+ const test = createCreeveyTest(browserName, storyMeta, skip);
61
+ tests[test.id] = { ...test,
62
+ storyId: storyMeta.id,
63
+ story: storyMeta,
64
+ fn: storyTestFabric(delay)
65
+ };
66
+ return;
67
+ }
68
+
69
+ Object.entries(storyTests).forEach(([testName, testFn]) => {
70
+ const test = createCreeveyTest(browserName, storyMeta, skip, testName);
71
+ tests[test.id] = { ...test,
72
+ storyId: storyMeta.id,
73
+ story: storyMeta,
74
+ fn: storyTestFabric(delay, testFn)
75
+ };
76
+ });
77
+ });
78
+ return tests;
79
+ }
80
+
81
+ export async function loadTestsFromStories(browsers, provider, update) {
82
+ const testIdsByFiles = new Map();
83
+ const stories = await provider(storiesByFiles => {
84
+ const testsDiff = {};
85
+ const tests = {};
86
+ browsers.forEach(browser => {
87
+ Array.from(storiesByFiles.entries()).forEach(([filename, stories]) => {
88
+ var _testIdsByFiles$get;
89
+
90
+ Object.assign(tests, convertStories(browser, stories));
91
+ const changed = Object.keys(tests);
92
+ const removed = ((_testIdsByFiles$get = testIdsByFiles.get(filename)) === null || _testIdsByFiles$get === void 0 ? void 0 : _testIdsByFiles$get.filter(testId => !tests[testId])) ?? [];
93
+ if (changed.length == 0) testIdsByFiles.delete(filename);else testIdsByFiles.set(filename, changed);
94
+ Object.assign(testsDiff, tests);
95
+ removed.forEach(testId => testsDiff[testId] = undefined);
96
+ });
97
+ });
98
+ update === null || update === void 0 ? void 0 : update(testsDiff);
99
+ });
100
+ const tests = browsers.reduce((tests, browser) => Object.assign(tests, convertStories(browser, stories)), {});
101
+ Object.values(tests).filter(isDefined).forEach(({
102
+ id,
103
+ story: {
104
+ parameters: {
105
+ fileName
106
+ }
107
+ }
108
+ }) => // TODO Don't use filename as a key, due possible collisions if two require.context with same structure of modules are defined
109
+ testIdsByFiles.set(fileName, [...(testIdsByFiles.get(fileName) ?? []), id]));
110
+ return tests;
111
+ }
112
+ export function saveStoriesJson(storiesData, extract) {
113
+ const outputDir = typeof extract == 'boolean' ? 'storybook-static' : extract; // NOTE Copy-pasted from Storybook's `getStoriesJsonData` method
114
+
115
+ const allowed = ['fileName', 'docsOnly', 'framework', '__id', '__isArgsStory'];
116
+ storiesData.globalParameters = pick(storiesData.globalParameters, allowed); // @ts-expect-error ignore error
117
+
118
+ storiesData.kindParameters = mapValues(storiesData.kindParameters, v => pick(v, allowed)); // @ts-expect-error ignore error
119
+
120
+ storiesData.stories = mapValues(storiesData.stories, v => ({ ...pick(v, ['id', 'name', 'kind', 'story']),
121
+ parameters: pick(v.parameters, allowed)
122
+ })); // TODO Fix args stories
123
+
124
+ removeProps(storiesData ?? {}, ['stories', () => true, 'parameters', '__isArgsStory']);
125
+ 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);
126
+ mkdirSync(outputDir, {
127
+ recursive: true
128
+ });
129
+ writeFileSync(path.join(outputDir, 'stories.json'), JSON.stringify(storiesData, null, 2));
130
+ }
131
+ export function saveTestsJson(tests, dstPath = process.cwd()) {
132
+ mkdirSync(dstPath, {
133
+ recursive: true
134
+ });
135
+ writeFileSync(path.join(dstPath, 'tests.json'), // eslint-disable-next-line @typescript-eslint/no-unsafe-return
136
+ JSON.stringify(tests, (_, value) => isFunction(value) ? value.toString() : value, 2));
137
+ }
@@ -0,0 +1,29 @@
1
+ /* eslint-disable */
2
+ //@ts-nocheck
3
+ import { addons } from '@storybook/addons';
4
+ import { getStorybookFramework, resolveFromStorybook } from './helpers';
5
+ const framework = getStorybookFramework(); // eslint-disable-next-line @typescript-eslint/no-var-requires
6
+
7
+ const core = require(resolveFromStorybook('@storybook/core-client'));
8
+
9
+ const start = core.start;
10
+ const api = start(() => void 0);
11
+ export const channel = addons.getChannel();
12
+ export const clientApi = api.clientApi;
13
+ export const forceReRender = api.forceReRender;
14
+ export const storiesOf = (kind, m) => {
15
+ return clientApi.storiesOf(kind, m).addParameters({
16
+ framework
17
+ });
18
+ };
19
+ export const configure = (...args) => {
20
+ //NOTE Storybook 6.x pass `framework` as first argument
21
+ //@ts-expect-error: ignore it
22
+ return api.configure(framework, ...args);
23
+ };
24
+ export const addDecorator = clientApi.addDecorator;
25
+ export const addParameters = clientApi.addParameters;
26
+ export const clearDecorators = clientApi.clearDecorators;
27
+ export const setAddon = clientApi.setAddon;
28
+ export const getStorybook = clientApi.getStorybook;
29
+ export const raw = clientApi.raw;
@@ -0,0 +1,99 @@
1
+ /* eslint-disable */
2
+ //@ts-nocheck
3
+ import path from 'path';
4
+ import resolveFrom from 'resolve-from';
5
+ const supportedFrameworks = ['react', 'vue', 'vue3', 'angular', 'marionette', 'mithril', 'marko', 'html', 'svelte', 'riot', 'ember', 'preact', 'rax', 'aurelia', 'server', 'web-components'];
6
+ export const storybookDirRef = {
7
+ current: path.resolve('.storybook')
8
+ };
9
+ export const resolveFromStorybook = modulePath => resolveFrom(storybookDirRef.current, modulePath);
10
+ export const resolveFromStorybookAddonDocs = modulePath => resolveFrom(resolveFromStorybook('@storybook/addon-docs'), modulePath);
11
+ export const resolveFromStorybookBuilderWebpack4 = modulePath => resolveFrom(resolveFromStorybook('@storybook/builder-webpack4'), modulePath);
12
+ export const resolveFromStorybookCoreClient = modulePath => resolveFrom(resolveFromStorybook('@storybook/core-client'), modulePath);
13
+ export const resolveFromStorybookCoreServer = modulePath => resolveFrom(resolveFromStorybook('@storybook/core-server'), modulePath);
14
+
15
+ const importFromStorybook = modulePath => import(resolveFromStorybook(modulePath));
16
+
17
+ export const importStorybookClientLogger = () => importFromStorybook('@storybook/client-logger');
18
+ export const importStorybookCoreCommon = () => importFromStorybook('@storybook/core-common');
19
+ export const importStorybookCoreEvents = () => importFromStorybook('@storybook/core-events');
20
+ export function hasDocsAddon() {
21
+ try {
22
+ resolveFromStorybook('@storybook/addon-docs');
23
+ return true;
24
+ } catch (_) {
25
+ return false;
26
+ }
27
+ }
28
+ export function hasSvelteCSFAddon() {
29
+ try {
30
+ resolveFromStorybook('@storybook/addon-svelte-csf');
31
+ return true;
32
+ } catch (_) {
33
+ return false;
34
+ }
35
+ }
36
+ export function getStorybookVersion() {
37
+ // eslint-disable-next-line @typescript-eslint/no-var-requires
38
+ const {
39
+ version
40
+ } = require(resolveFromStorybook('@storybook/core-server/package.json'));
41
+
42
+ return version;
43
+ }
44
+ export function isStorybookVersionLessThan(major, minor) {
45
+ const [sbMajor, sbMinor] = (process.env.__CREEVEY_STORYBOOK_VERSION__ ?? getStorybookVersion()).split('.');
46
+ return Number(sbMajor) < major || minor != undefined && Number(sbMajor) == major && Number(sbMinor) < minor;
47
+ }
48
+ export function isStorybookVersionGreaterThan(major, minor) {
49
+ const [sbMajor, sbMinor] = (process.env.__CREEVEY_STORYBOOK_VERSION__ ?? getStorybookVersion()).split('.');
50
+ return Number(sbMajor) > major || minor != undefined && Number(sbMajor) == major && Number(sbMinor) > minor;
51
+ }
52
+ export function isStorybookVersion(major, minor) {
53
+ const [sbMajor, sbMinor] = (process.env.__CREEVEY_STORYBOOK_VERSION__ ?? getStorybookVersion()).split('.');
54
+ return Number(sbMajor) == major || minor != undefined && Number(sbMajor) == major && Number(sbMinor) == minor;
55
+ }
56
+ export function getStorybookFramework() {
57
+ const framework = process.env.__CREEVEY_STORYBOOK_FRAMEWORK__ ?? supportedFrameworks.find(framework => {
58
+ try {
59
+ return require.resolve(resolveFromStorybook(`@storybook/${framework}`));
60
+ } catch (_) {
61
+ return false;
62
+ }
63
+ });
64
+ if (!framework) throw new Error("Couldn't detect used Storybook framework. Please ensure that you install `@storybook/<framework>` package");
65
+ return framework;
66
+ }
67
+ export const storybookConfigRef = {
68
+ current: {
69
+ stories: []
70
+ }
71
+ };
72
+ export async function importStorybookConfig() {
73
+ const configPath = `${storybookDirRef.current}/main`;
74
+
75
+ try {
76
+ return storybookConfigRef.current = (await import(require.resolve(configPath))).default;
77
+ } catch (_) {
78
+ const storybookUtilsPath = '@storybook/core-common/dist/cjs/utils';
79
+ const serverRequireModule = 'interpret-require'; // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
80
+
81
+ const {
82
+ getInterpretedFile
83
+ } = await import(resolveFromStorybook(`${storybookUtilsPath}/interpret-files`)); // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment
84
+
85
+ const {
86
+ default: serverRequireFallback,
87
+ serverRequire = serverRequireFallback
88
+ } = await import(resolveFromStorybook(`${storybookUtilsPath}/${serverRequireModule}`)); // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call
89
+
90
+ const mainConfigFile = getInterpretedFile(configPath); // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call
91
+
92
+ return storybookConfigRef.current = serverRequire(mainConfigFile);
93
+ }
94
+ }
95
+ export async function isCSFv3Enabled() {
96
+ var _await$importStoryboo, _await$importStoryboo2;
97
+
98
+ 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;
99
+ }
@@ -0,0 +1,60 @@
1
+ /* eslint-disable */
2
+ //@ts-nocheck
3
+ import cluster from 'cluster';
4
+ import { loadStoriesFromBrowser } from '../../selenium';
5
+ import { emitStoriesMessage, sendStoriesMessage, subscribeOn, subscribeOnWorker } from '../../messages';
6
+ import { isDefined } from '../../../types';
7
+ import { logger } from '../../logger';
8
+ export const loadStories = async (_config, _options, 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();
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
+ };