creevey 0.9.1 → 0.10.0-beta.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 (259) hide show
  1. package/chromatic.config.json +5 -0
  2. package/dist/client/addon/components/Addon.d.ts +1 -0
  3. package/dist/client/addon/components/Addon.js.map +1 -1
  4. package/dist/client/addon/components/Icons.d.ts +1 -0
  5. package/dist/client/addon/components/Icons.js.map +1 -1
  6. package/dist/client/addon/components/Panel.d.ts +1 -0
  7. package/dist/client/addon/components/Panel.js.map +1 -1
  8. package/dist/client/addon/components/TestSelect.d.ts +1 -0
  9. package/dist/client/addon/components/TestSelect.js +4 -3
  10. package/dist/client/addon/components/TestSelect.js.map +1 -1
  11. package/dist/client/addon/components/Tools.d.ts +1 -0
  12. package/dist/client/addon/components/Tools.js +7 -8
  13. package/dist/client/addon/components/Tools.js.map +1 -1
  14. package/dist/client/addon/controller.d.ts +1 -1
  15. package/dist/client/addon/controller.js.map +1 -1
  16. package/dist/client/addon/decorator.d.ts +1 -1
  17. package/dist/client/addon/manager.js +3 -2
  18. package/dist/client/addon/manager.js.map +1 -1
  19. package/dist/client/addon/preview.d.ts +1 -1
  20. package/dist/client/addon/withCreevey.d.ts +6 -8
  21. package/dist/client/addon/withCreevey.js +21 -19
  22. package/dist/client/addon/withCreevey.js.map +1 -1
  23. package/dist/client/shared/components/ImagesView/BlendView.d.ts +1 -1
  24. package/dist/client/shared/components/ImagesView/BlendView.js.map +1 -1
  25. package/dist/client/shared/components/ImagesView/ImagesView.d.ts +1 -0
  26. package/dist/client/shared/components/ImagesView/ImagesView.js.map +1 -1
  27. package/dist/client/shared/components/ImagesView/SideBySideView.d.ts +1 -1
  28. package/dist/client/shared/components/ImagesView/SideBySideView.js.map +1 -1
  29. package/dist/client/shared/components/ImagesView/SlideView.d.ts +1 -1
  30. package/dist/client/shared/components/ImagesView/SlideView.js.map +1 -1
  31. package/dist/client/shared/components/ImagesView/SwapView.d.ts +1 -1
  32. package/dist/client/shared/components/ImagesView/SwapView.js.map +1 -1
  33. package/dist/client/shared/components/PageFooter/PageFooter.d.ts +1 -0
  34. package/dist/client/shared/components/PageFooter/PageFooter.js +1 -1
  35. package/dist/client/shared/components/PageFooter/PageFooter.js.map +1 -1
  36. package/dist/client/shared/components/PageFooter/Paging.d.ts +2 -2
  37. package/dist/client/shared/components/PageFooter/Paging.js +8 -6
  38. package/dist/client/shared/components/PageFooter/Paging.js.map +1 -1
  39. package/dist/client/shared/components/PageHeader/ImagePreview.js.map +1 -1
  40. package/dist/client/shared/components/PageHeader/PageHeader.d.ts +1 -0
  41. package/dist/client/shared/components/PageHeader/PageHeader.js +2 -1
  42. package/dist/client/shared/components/PageHeader/PageHeader.js.map +1 -1
  43. package/dist/client/shared/components/ResultsPage.d.ts +2 -2
  44. package/dist/client/shared/components/ResultsPage.js.map +1 -1
  45. package/dist/client/web/CreeveyApp.d.ts +1 -0
  46. package/dist/client/web/CreeveyApp.js.map +1 -1
  47. package/dist/client/web/CreeveyLoader.d.ts +1 -0
  48. package/dist/client/web/CreeveyLoader.js.map +1 -1
  49. package/dist/client/web/CreeveyView/SideBar/Checkbox.d.ts +1 -1
  50. package/dist/client/web/CreeveyView/SideBar/Checkbox.js +4 -4
  51. package/dist/client/web/CreeveyView/SideBar/Checkbox.js.map +1 -1
  52. package/dist/client/web/CreeveyView/SideBar/Search.d.ts +1 -0
  53. package/dist/client/web/CreeveyView/SideBar/Search.js +4 -4
  54. package/dist/client/web/CreeveyView/SideBar/Search.js.map +1 -1
  55. package/dist/client/web/CreeveyView/SideBar/SideBar.d.ts +1 -1
  56. package/dist/client/web/CreeveyView/SideBar/SideBar.js +1 -7
  57. package/dist/client/web/CreeveyView/SideBar/SideBar.js.map +1 -1
  58. package/dist/client/web/CreeveyView/SideBar/SideBarFooter.d.ts +1 -0
  59. package/dist/client/web/CreeveyView/SideBar/SideBarFooter.js +5 -4
  60. package/dist/client/web/CreeveyView/SideBar/SideBarFooter.js.map +1 -1
  61. package/dist/client/web/CreeveyView/SideBar/SideBarHeader.d.ts +1 -0
  62. package/dist/client/web/CreeveyView/SideBar/SideBarHeader.js +4 -3
  63. package/dist/client/web/CreeveyView/SideBar/SideBarHeader.js.map +1 -1
  64. package/dist/client/web/CreeveyView/SideBar/SuiteLink.d.ts +3 -7
  65. package/dist/client/web/CreeveyView/SideBar/SuiteLink.js +6 -5
  66. package/dist/client/web/CreeveyView/SideBar/SuiteLink.js.map +1 -1
  67. package/dist/client/web/CreeveyView/SideBar/TestLink.d.ts +1 -0
  68. package/dist/client/web/CreeveyView/SideBar/TestLink.js +5 -1
  69. package/dist/client/web/CreeveyView/SideBar/TestLink.js.map +1 -1
  70. package/dist/client/web/CreeveyView/SideBar/TestStatusIcon.js +15 -8
  71. package/dist/client/web/CreeveyView/SideBar/TestStatusIcon.js.map +1 -1
  72. package/dist/client/web/CreeveyView/SideBar/TestsStatus.js +5 -4
  73. package/dist/client/web/CreeveyView/SideBar/TestsStatus.js.map +1 -1
  74. package/dist/client/web/CreeveyView/SideBar/Toggle.d.ts +1 -0
  75. package/dist/client/web/CreeveyView/SideBar/Toggle.js.map +1 -1
  76. package/dist/client/web/KeyboardEventsContext.d.ts +3 -4
  77. package/dist/client/web/KeyboardEventsContext.js.map +1 -1
  78. package/dist/client/web/assets/index-DkmZfG9C.js +591 -0
  79. package/dist/client/web/index.html +1 -1
  80. package/dist/client/web/index.js +5 -6
  81. package/dist/client/web/index.js.map +1 -1
  82. package/dist/creevey.js +21 -9
  83. package/dist/creevey.js.map +1 -1
  84. package/dist/index.js +7 -3
  85. package/dist/index.js.map +1 -1
  86. package/dist/server/config.d.ts +1 -1
  87. package/dist/server/config.js +9 -5
  88. package/dist/server/config.js.map +1 -1
  89. package/dist/server/docker.d.ts +2 -2
  90. package/dist/server/docker.js +46 -40
  91. package/dist/server/docker.js.map +1 -1
  92. package/dist/server/index.js +54 -15
  93. package/dist/server/index.js.map +1 -1
  94. package/dist/server/master/master.d.ts +1 -5
  95. package/dist/server/master/master.js +3 -3
  96. package/dist/server/master/master.js.map +1 -1
  97. package/dist/server/master/pool.d.ts +2 -1
  98. package/dist/server/master/pool.js +13 -7
  99. package/dist/server/master/pool.js.map +1 -1
  100. package/dist/server/master/runner.d.ts +1 -1
  101. package/dist/server/master/runner.js +4 -2
  102. package/dist/server/master/runner.js.map +1 -1
  103. package/dist/server/master/server.js +1 -0
  104. package/dist/server/master/server.js.map +1 -1
  105. package/dist/server/master/start.d.ts +3 -0
  106. package/dist/server/master/{index.js → start.js} +6 -9
  107. package/dist/server/master/start.js.map +1 -0
  108. package/dist/server/messages.d.ts +4 -10
  109. package/dist/server/messages.js +4 -58
  110. package/dist/server/messages.js.map +1 -1
  111. package/dist/server/playwright/docker-file.d.ts +1 -0
  112. package/dist/server/playwright/docker-file.js +26 -0
  113. package/dist/server/playwright/docker-file.js.map +1 -0
  114. package/dist/server/playwright/docker.d.ts +1 -0
  115. package/dist/server/playwright/docker.js +31 -0
  116. package/dist/server/playwright/docker.js.map +1 -0
  117. package/dist/server/playwright/internal.d.ts +25 -0
  118. package/dist/server/playwright/internal.js +319 -0
  119. package/dist/server/playwright/internal.js.map +1 -0
  120. package/dist/server/playwright/webdriver.d.ts +16 -0
  121. package/dist/server/playwright/webdriver.js +105 -0
  122. package/dist/server/playwright/webdriver.js.map +1 -0
  123. package/dist/server/providers/browser.d.ts +2 -0
  124. package/dist/server/{storybook/providers → providers}/browser.js +6 -7
  125. package/dist/server/providers/browser.js.map +1 -0
  126. package/dist/server/providers/hybrid.d.ts +2 -0
  127. package/dist/server/{storybook/providers → providers}/hybrid.js +8 -8
  128. package/dist/server/providers/hybrid.js.map +1 -0
  129. package/dist/server/reporter.d.ts +26 -0
  130. package/dist/server/{worker/reporter.js → reporter.js} +34 -56
  131. package/dist/server/reporter.js.map +1 -0
  132. package/dist/server/selenium/internal.d.ts +31 -0
  133. package/dist/server/selenium/internal.js +606 -0
  134. package/dist/server/selenium/internal.js.map +1 -0
  135. package/dist/server/selenium/selenoid.js +6 -13
  136. package/dist/server/selenium/selenoid.js.map +1 -1
  137. package/dist/server/selenium/webdriver.d.ts +24 -0
  138. package/dist/server/selenium/webdriver.js +106 -0
  139. package/dist/server/selenium/webdriver.js.map +1 -0
  140. package/dist/server/stories.js +16 -9
  141. package/dist/server/stories.js.map +1 -1
  142. package/dist/server/telemetry.d.ts +1 -1
  143. package/dist/server/telemetry.js +4 -4
  144. package/dist/server/telemetry.js.map +1 -1
  145. package/dist/server/utils.d.ts +3 -4
  146. package/dist/server/utils.js +10 -9
  147. package/dist/server/utils.js.map +1 -1
  148. package/dist/server/webdriver.d.ts +19 -0
  149. package/dist/server/webdriver.js +79 -0
  150. package/dist/server/webdriver.js.map +1 -0
  151. package/dist/server/worker/chai-image.d.ts +2 -5
  152. package/dist/server/worker/chai-image.js +14 -102
  153. package/dist/server/worker/chai-image.js.map +1 -1
  154. package/dist/server/worker/match-image.d.ts +14 -0
  155. package/dist/server/worker/match-image.js +231 -0
  156. package/dist/server/worker/match-image.js.map +1 -0
  157. package/dist/server/worker/start.d.ts +2 -0
  158. package/dist/server/worker/start.js +258 -0
  159. package/dist/server/worker/start.js.map +1 -0
  160. package/dist/types.d.ts +127 -64
  161. package/dist/types.js +15 -9
  162. package/dist/types.js.map +1 -1
  163. package/package.json +108 -110
  164. package/src/client/addon/components/Addon.tsx +1 -1
  165. package/src/client/addon/components/Icons.tsx +1 -1
  166. package/src/client/addon/components/Panel.tsx +1 -1
  167. package/src/client/addon/components/TestSelect.tsx +5 -5
  168. package/src/client/addon/components/Tools.tsx +9 -9
  169. package/src/client/addon/controller.ts +1 -1
  170. package/src/client/addon/manager.ts +4 -4
  171. package/src/client/addon/withCreevey.ts +26 -28
  172. package/src/client/shared/components/ImagesView/BlendView.tsx +1 -1
  173. package/src/client/shared/components/ImagesView/ImagesView.tsx +2 -2
  174. package/src/client/shared/components/ImagesView/SideBySideView.tsx +1 -1
  175. package/src/client/shared/components/ImagesView/SlideView.tsx +1 -1
  176. package/src/client/shared/components/ImagesView/SwapView.tsx +1 -1
  177. package/src/client/shared/components/PageFooter/PageFooter.tsx +2 -2
  178. package/src/client/shared/components/PageFooter/Paging.tsx +13 -13
  179. package/src/client/shared/components/PageHeader/ImagePreview.tsx +1 -1
  180. package/src/client/shared/components/PageHeader/PageHeader.tsx +4 -3
  181. package/src/client/shared/components/ResultsPage.tsx +1 -1
  182. package/src/client/web/CreeveyApp.tsx +1 -1
  183. package/src/client/web/CreeveyLoader.tsx +1 -1
  184. package/src/client/web/CreeveyView/SideBar/Checkbox.tsx +6 -7
  185. package/src/client/web/CreeveyView/SideBar/Search.tsx +4 -4
  186. package/src/client/web/CreeveyView/SideBar/SideBar.tsx +3 -10
  187. package/src/client/web/CreeveyView/SideBar/SideBarFooter.tsx +7 -6
  188. package/src/client/web/CreeveyView/SideBar/SideBarHeader.tsx +7 -6
  189. package/src/client/web/CreeveyView/SideBar/SuiteLink.tsx +8 -6
  190. package/src/client/web/CreeveyView/SideBar/TestLink.tsx +8 -3
  191. package/src/client/web/CreeveyView/SideBar/TestStatusIcon.tsx +18 -10
  192. package/src/client/web/CreeveyView/SideBar/TestsStatus.tsx +7 -10
  193. package/src/client/web/CreeveyView/SideBar/Toggle.tsx +1 -2
  194. package/src/client/web/KeyboardEventsContext.tsx +3 -4
  195. package/src/client/web/index.html +1 -1
  196. package/src/client/web/index.tsx +4 -3
  197. package/src/creevey.ts +25 -8
  198. package/src/index.ts +4 -2
  199. package/src/server/config.ts +12 -8
  200. package/src/server/docker.ts +58 -44
  201. package/src/server/index.ts +57 -18
  202. package/src/server/master/master.ts +3 -6
  203. package/src/server/master/pool.ts +25 -9
  204. package/src/server/master/runner.ts +4 -2
  205. package/src/server/master/server.ts +1 -0
  206. package/src/server/master/{index.ts → start.ts} +13 -11
  207. package/src/server/messages.ts +11 -75
  208. package/src/server/playwright/docker-file.ts +21 -0
  209. package/src/server/playwright/docker.ts +41 -0
  210. package/src/server/playwright/internal.ts +387 -0
  211. package/src/server/playwright/webdriver.ts +126 -0
  212. package/src/server/{storybook/providers → providers}/browser.ts +7 -8
  213. package/src/server/{storybook/providers → providers}/hybrid.ts +19 -19
  214. package/src/server/{worker/reporter.ts → reporter.ts} +40 -72
  215. package/src/server/selenium/internal.ts +785 -0
  216. package/src/server/selenium/selenoid.ts +12 -17
  217. package/src/server/selenium/webdriver.ts +136 -0
  218. package/src/server/stories.ts +18 -11
  219. package/src/server/telemetry.ts +2 -2
  220. package/src/server/utils.ts +9 -9
  221. package/src/server/webdriver.ts +127 -0
  222. package/src/server/worker/chai-image.ts +21 -133
  223. package/src/server/worker/match-image.ts +303 -0
  224. package/src/server/worker/start.ts +303 -0
  225. package/src/types.ts +162 -60
  226. package/dist/client/web/202.js +0 -1
  227. package/dist/client/web/270.js +0 -43
  228. package/dist/client/web/752.js +0 -1
  229. package/dist/client/web/main.js +0 -79
  230. package/dist/client/web/main.js.LICENSE.txt +0 -34
  231. package/dist/server/master/index.d.ts +0 -3
  232. package/dist/server/master/index.js.map +0 -1
  233. package/dist/server/selenium/browser.d.ts +0 -19
  234. package/dist/server/selenium/browser.js +0 -640
  235. package/dist/server/selenium/browser.js.map +0 -1
  236. package/dist/server/selenium/index.d.ts +0 -2
  237. package/dist/server/selenium/index.js +0 -19
  238. package/dist/server/selenium/index.js.map +0 -1
  239. package/dist/server/storybook/providers/browser.d.ts +0 -2
  240. package/dist/server/storybook/providers/browser.js.map +0 -1
  241. package/dist/server/storybook/providers/hybrid.d.ts +0 -2
  242. package/dist/server/storybook/providers/hybrid.js.map +0 -1
  243. package/dist/server/worker/helpers.d.ts +0 -8
  244. package/dist/server/worker/helpers.js +0 -57
  245. package/dist/server/worker/helpers.js.map +0 -1
  246. package/dist/server/worker/index.d.ts +0 -1
  247. package/dist/server/worker/index.js +0 -6
  248. package/dist/server/worker/index.js.map +0 -1
  249. package/dist/server/worker/reporter.d.ts +0 -8
  250. package/dist/server/worker/reporter.js.map +0 -1
  251. package/dist/server/worker/worker.d.ts +0 -4
  252. package/dist/server/worker/worker.js +0 -212
  253. package/dist/server/worker/worker.js.map +0 -1
  254. package/src/server/selenium/browser.ts +0 -840
  255. package/src/server/selenium/index.ts +0 -2
  256. package/src/server/worker/helpers.ts +0 -61
  257. package/src/server/worker/index.ts +0 -1
  258. package/src/server/worker/worker.ts +0 -240
  259. package/types/mocha.d.ts +0 -20
@@ -0,0 +1,785 @@
1
+ import { Args } from '@storybook/csf';
2
+ import { SET_GLOBALS, UPDATE_STORY_ARGS, STORY_RENDERED } from '@storybook/core-events';
3
+ import chalk from 'chalk';
4
+ import http from 'http';
5
+ import https from 'https';
6
+ import Logger from 'loglevel';
7
+ import prefix from 'loglevel-plugin-prefix';
8
+ import { PNG } from 'pngjs';
9
+ import { Builder, By, Capabilities, Origin, WebDriver, WebElement, logging } from 'selenium-webdriver';
10
+ // import { Options as IeOptions } from 'selenium-webdriver/ie';
11
+ // import { Options as EdgeOptions } from 'selenium-webdriver/edge';
12
+ // import { Options as ChromeOptions } from 'selenium-webdriver/chrome';
13
+ // import { Options as SafariOptions } from 'selenium-webdriver/safari';
14
+ // import { Options as FirefoxOptions } from 'selenium-webdriver/firefox';
15
+ import { PageLoadStrategy } from 'selenium-webdriver/lib/capabilities.js';
16
+ import { BrowserConfigObject, Config, noop, StorybookGlobals, StoryInput, StoriesRaw, Options } from '../../types.js';
17
+ import { colors, logger } from '../logger.js';
18
+ import { subscribeOn } from '../messages.js';
19
+ import { isShuttingDown, runSequence } from '../utils.js';
20
+ import {
21
+ appendIframePath,
22
+ getAddresses,
23
+ LOCALHOST_REGEXP,
24
+ resolveStorybookUrl,
25
+ storybookRootID,
26
+ } from '../webdriver.js';
27
+
28
+ interface ElementRect {
29
+ top: number;
30
+ left: number;
31
+ width: number;
32
+ height: number;
33
+ }
34
+
35
+ // type UnPromise<P> = P extends Promise<infer T> ? T : never;
36
+ // let context: UnPromise<ReturnType<typeof BrowsingContext>> | null = null;
37
+
38
+ function getSessionData(grid: string, sessionId = ''): Promise<Record<string, unknown>> {
39
+ const gridUrl = new URL(grid);
40
+ gridUrl.pathname = `/host/${sessionId}`;
41
+
42
+ return new Promise((resolve, reject) =>
43
+ (gridUrl.protocol == 'https:' ? https : http).get(gridUrl.toString(), (res) => {
44
+ if (res.statusCode !== 200) {
45
+ reject(new Error(`Couldn't get session data for ${sessionId}. Status code: ${res.statusCode ?? 'Unknown'}`));
46
+ return;
47
+ }
48
+
49
+ let data = '';
50
+
51
+ res.setEncoding('utf-8');
52
+ res.on('data', (chunk: string) => (data += chunk));
53
+ res.on('end', () => {
54
+ try {
55
+ resolve(JSON.parse(data) as Record<string, unknown>);
56
+ } catch (error) {
57
+ reject(
58
+ new Error(
59
+ `Couldn't get session data for ${sessionId}. ${
60
+ error instanceof Error ? (error.stack ?? error.message) : (error as string)
61
+ }`,
62
+ ),
63
+ );
64
+ }
65
+ });
66
+ }),
67
+ );
68
+ }
69
+
70
+ async function openUrlAndWaitForPageSource(
71
+ browser: WebDriver,
72
+ url: string,
73
+ predicate: (source: string) => boolean,
74
+ ): Promise<string> {
75
+ let source = '';
76
+ await browser.get(url);
77
+ do {
78
+ try {
79
+ source = await browser.getPageSource();
80
+ } catch {
81
+ // NOTE: Firefox can raise exception "curContainer.frame.document.documentElement is null"
82
+ }
83
+ } while (predicate(source));
84
+ return source;
85
+ }
86
+
87
+ async function buildWebdriver(
88
+ browserName: string,
89
+ gridUrl: string,
90
+ config: Config,
91
+ options: Options,
92
+ ): Promise<WebDriver | null> {
93
+ const browserConfig = config.browsers[browserName] as BrowserConfigObject;
94
+ const { _storybookGlobals, /*customizeBuilder,*/ ...userCapabilities } = browserConfig;
95
+
96
+ const url = new URL(gridUrl);
97
+ url.username = url.username ? '********' : '';
98
+ url.password = url.password ? '********' : '';
99
+ logger.debug(`(${browserName}) Connecting to Selenium ${chalk.magenta(url.toString())}`);
100
+
101
+ // TODO Define some capabilities explicitly and define typings
102
+ const capabilities = new Capabilities({ ...userCapabilities, pageLoadStrategy: PageLoadStrategy.EAGER });
103
+ const prefs = new logging.Preferences();
104
+
105
+ if (options.trace) {
106
+ for (const type of Object.values(logging.Type)) {
107
+ prefs.setLevel(type as string, logging.Level.ALL);
108
+ }
109
+ }
110
+
111
+ // TODO Fetch selenium grid capabilities
112
+ // TODO Validate browsers, versions, and platform
113
+ // TODO Use `customizeBuilder`
114
+
115
+ let browser: WebDriver;
116
+
117
+ try {
118
+ // const ie = new IeOptions();
119
+ // const edge = new EdgeOptions();
120
+ // const chrome = new ChromeOptions();
121
+ // const safari = new SafariOptions();
122
+ // const firefox = new FirefoxOptions();
123
+ // edge.enableBidi();
124
+ // chrome.enableBidi();
125
+ // firefox.enableBidi();
126
+
127
+ browser = await new Builder()
128
+ // .setIeOptions(ie)
129
+ // .setEdgeOptions(edge)
130
+ // .setChromeOptions(chrome)
131
+ // .setSafariOptions(safari)
132
+ // .setFirefoxOptions(firefox)
133
+ .usingServer(gridUrl)
134
+ .withCapabilities(capabilities)
135
+ .setLoggingPrefs(prefs) // NOTE: Should go last
136
+ .build();
137
+
138
+ // const id = await browser.getWindowHandle();
139
+ // context = await BrowsingContext(browser, { browsingContextId: id });
140
+ } catch (error) {
141
+ logger.error(`(${browserName}) Failed to start browser:`, error);
142
+ return null;
143
+ }
144
+
145
+ return browser;
146
+ }
147
+
148
+ export class InternalBrowser {
149
+ #isShuttingDown = false;
150
+ #browser: WebDriver;
151
+ #serverHost: string | null = null;
152
+ #serverPort: number;
153
+ #logger: Logger.Logger;
154
+ #unsubscribe: () => void = noop;
155
+ #keepAliveInterval: NodeJS.Timeout | null = null;
156
+ constructor(browser: WebDriver, port: number, logger: Logger.Logger) {
157
+ this.#browser = browser;
158
+ this.#serverPort = port;
159
+ this.#logger = logger;
160
+ this.#unsubscribe = subscribeOn('shutdown', () => {
161
+ void this.closeBrowser();
162
+ });
163
+ }
164
+
165
+ get browser() {
166
+ return this.#browser;
167
+ }
168
+
169
+ async closeBrowser(): Promise<void> {
170
+ if (this.#isShuttingDown) return;
171
+
172
+ this.#isShuttingDown = true;
173
+ this.#unsubscribe();
174
+ if (this.#keepAliveInterval !== null) clearInterval(this.#keepAliveInterval);
175
+
176
+ try {
177
+ await this.#browser.quit();
178
+ } catch (_) {
179
+ /* noop */
180
+ }
181
+ }
182
+
183
+ async takeScreenshot(captureElement?: string | null, ignoreElements?: string | string[] | null): Promise<Buffer> {
184
+ let screenshot: string | Buffer;
185
+
186
+ const ignoreStyles = await this.insertIgnoreStyles(ignoreElements);
187
+
188
+ if (this.#logger.getLevel() <= Logger.levels.DEBUG) {
189
+ const { innerWidth, innerHeight } = await this.#browser.executeScript<{
190
+ innerWidth: number;
191
+ innerHeight: number;
192
+ }>(function () {
193
+ return {
194
+ innerWidth: window.innerWidth,
195
+ innerHeight: window.innerHeight,
196
+ };
197
+ });
198
+ this.#logger.debug(`Viewport size is: ${innerWidth}x${innerHeight}`);
199
+ }
200
+
201
+ try {
202
+ if (!captureElement) {
203
+ this.#logger.debug('Capturing viewport screenshot');
204
+ screenshot = await this.#browser.takeScreenshot();
205
+ this.#logger.debug('Viewport screenshot is captured');
206
+ } else {
207
+ this.#logger.debug(`Checking is element ${chalk.cyan(captureElement)} fit into viewport`);
208
+ const rects = await this.#browser.executeScript<
209
+ { elementRect: ElementRect; windowRect: ElementRect } | undefined
210
+ >(function (selector: string): { elementRect: ElementRect; windowRect: ElementRect } | undefined {
211
+ window.scrollTo(0, 0); // TODO Maybe we should remove same code from `resetMousePosition`
212
+ // eslint-disable-next-line no-var
213
+ var element = document.querySelector(selector);
214
+ if (!element) return;
215
+
216
+ // eslint-disable-next-line no-var
217
+ var elementRect = element.getBoundingClientRect();
218
+
219
+ return {
220
+ elementRect: {
221
+ top: elementRect.top,
222
+ left: elementRect.left,
223
+ width: elementRect.width,
224
+ height: elementRect.height,
225
+ },
226
+ // NOTE page_Offset is used only for IE9-11
227
+ windowRect: {
228
+ // eslint-disable-next-line @typescript-eslint/no-deprecated
229
+ top: Math.round(window.scrollY || window.pageYOffset),
230
+ // eslint-disable-next-line @typescript-eslint/no-deprecated
231
+ left: Math.round(window.scrollX || window.pageXOffset),
232
+ width: window.innerWidth,
233
+ height: window.innerHeight,
234
+ },
235
+ };
236
+ }, captureElement);
237
+ const { elementRect, windowRect } = rects ?? {};
238
+
239
+ if (!elementRect || !windowRect) throw new Error(`Couldn't find element with selector: '${captureElement}'`);
240
+
241
+ const isFitIntoViewport =
242
+ elementRect.width + elementRect.left <= windowRect.width &&
243
+ elementRect.height + elementRect.top <= windowRect.height;
244
+
245
+ if (isFitIntoViewport) {
246
+ this.#logger.debug(
247
+ `Capturing ${chalk.cyan(captureElement)} with size: ${elementRect.width}x${elementRect.height}`,
248
+ );
249
+ } else
250
+ this.#logger.debug(
251
+ `Capturing composite screenshot image of ${chalk.cyan(captureElement)} with size: ${elementRect.width}x${elementRect.height}`,
252
+ );
253
+
254
+ // const element = await browser.findElement(By.css(captureElement));
255
+ // screenshot = isFitIntoViewport
256
+ // ? context
257
+ // ? await context.captureElementScreenshot(await element.getId())
258
+ // : await browser.findElement(By.css(captureElement)).takeScreenshot()
259
+ // : // TODO pointer-events: none, need to research
260
+ // await takeCompositeScreenshot(browser, windowRect, elementRect);
261
+ screenshot = isFitIntoViewport
262
+ ? await this.#browser.findElement(By.css(captureElement)).takeScreenshot()
263
+ : // TODO pointer-events: none, need to research
264
+ await this.takeCompositeScreenshot(windowRect, elementRect);
265
+
266
+ this.#logger.debug(`${chalk.cyan(captureElement)} is captured`);
267
+ }
268
+ } finally {
269
+ await this.removeIgnoreStyles(ignoreStyles);
270
+ }
271
+
272
+ return typeof screenshot === 'string' ? Buffer.from(screenshot, 'base64') : screenshot;
273
+ }
274
+
275
+ waitForComplete(callback: (isCompleted: boolean) => void): void {
276
+ void this.#browser
277
+ .executeAsyncScript<boolean>(function (callback: (isCompleted: boolean) => void) {
278
+ void window.__CREEVEY_HAS_PLAY_COMPLETED_YET__().then(callback);
279
+ })
280
+ .then(callback);
281
+ }
282
+
283
+ async selectStory(id: string, waitForReady = false): Promise<boolean> {
284
+ // NOTE: Global variables might be reset after hot reload. I think it's workaround, maybe we need better solution
285
+ await this.updateBrowserGlobalVariables();
286
+ await this.resetMousePosition();
287
+
288
+ this.#logger.debug(`Triggering 'SetCurrentStory' event with storyId ${chalk.magenta(id)}`);
289
+
290
+ const result = await this.#browser.executeAsyncScript<[error?: string | null, isCaptureCalled?: boolean] | null>(
291
+ function (
292
+ storyId: string,
293
+ shouldWaitForReady: boolean,
294
+ callback: (response: [error?: string | null, isCaptureCalled?: boolean]) => void,
295
+ ) {
296
+ if (typeof window.__CREEVEY_SELECT_STORY__ == 'undefined') {
297
+ callback([
298
+ "Creevey can't switch story. This may happened if forget to add `creevey` addon to your storybook config, or storybook not loaded in browser due syntax error.",
299
+ ]);
300
+ return;
301
+ }
302
+ void window.__CREEVEY_SELECT_STORY__(storyId, shouldWaitForReady).then(callback);
303
+ },
304
+ id,
305
+ waitForReady,
306
+ );
307
+
308
+ const [errorMessage, isCaptureCalled = false] = result ?? [];
309
+
310
+ if (errorMessage) throw new Error(errorMessage);
311
+
312
+ return isCaptureCalled;
313
+ }
314
+
315
+ async updateStoryArgs(story: StoryInput, updatedArgs: Args): Promise<void> {
316
+ await this.#browser.executeAsyncScript<undefined>(
317
+ function (
318
+ storyId: string,
319
+ updatedArgs: Args,
320
+ UPDATE_STORY_ARGS: string,
321
+ STORY_RENDERED: string,
322
+ callback: () => void,
323
+ ) {
324
+ window.__STORYBOOK_ADDONS_CHANNEL__.once(STORY_RENDERED, callback);
325
+ window.__STORYBOOK_ADDONS_CHANNEL__.emit(UPDATE_STORY_ARGS, {
326
+ storyId,
327
+ updatedArgs,
328
+ });
329
+ },
330
+ story.id,
331
+ updatedArgs,
332
+ UPDATE_STORY_ARGS,
333
+ STORY_RENDERED,
334
+ );
335
+ }
336
+
337
+ async loadStoriesFromBrowser(): Promise<StoriesRaw> {
338
+ const stories = await this.#browser.executeAsyncScript<StoriesRaw | undefined>(function (
339
+ callback: (stories: StoriesRaw | undefined) => void,
340
+ ) {
341
+ void window.__CREEVEY_GET_STORIES__().then(callback);
342
+ });
343
+
344
+ if (!stories) throw new Error("Can't get stories, it seems creevey or storybook API isn't available");
345
+
346
+ return stories;
347
+ }
348
+
349
+ static async getBrowser(
350
+ browserName: string,
351
+ gridUrl: string,
352
+ config: Config,
353
+ options: Options,
354
+ ): Promise<InternalBrowser | null> {
355
+ const browserConfig = config.browsers[browserName] as BrowserConfigObject;
356
+ const { storybookUrl: address = config.storybookUrl, limit, viewport, _storybookGlobals } = browserConfig;
357
+ void limit;
358
+
359
+ const browser = await buildWebdriver(browserName, gridUrl, config, options);
360
+
361
+ if (!browser) return null;
362
+
363
+ const internalBrowser = new InternalBrowser(browser, options.port, logger);
364
+
365
+ try {
366
+ if (isShuttingDown.current) return null;
367
+
368
+ const done = await internalBrowser.init({
369
+ browserName,
370
+ gridUrl,
371
+ viewport,
372
+ storybookUrl: address,
373
+ storybookGlobals: _storybookGlobals,
374
+ resolveStorybookUrl: config.resolveStorybookUrl,
375
+ });
376
+
377
+ return done ? internalBrowser : null;
378
+ } catch (originalError) {
379
+ void internalBrowser.closeBrowser();
380
+
381
+ const message = originalError instanceof Error ? originalError.message : (originalError as string);
382
+ const error = new Error(`Can't load storybook root page by URL ${await browser.getCurrentUrl()}: ${message}`);
383
+ if (originalError instanceof Error) error.stack = originalError.stack;
384
+
385
+ logger.error(error);
386
+
387
+ return null;
388
+ }
389
+ }
390
+
391
+ private async init({
392
+ browserName,
393
+ gridUrl,
394
+ viewport,
395
+ storybookUrl,
396
+ storybookGlobals,
397
+ resolveStorybookUrl,
398
+ }: {
399
+ browserName: string;
400
+ gridUrl: string;
401
+ viewport?: { width: number; height: number };
402
+ storybookUrl: string;
403
+ storybookGlobals?: StorybookGlobals;
404
+ resolveStorybookUrl?: () => Promise<string>;
405
+ }): Promise<boolean> {
406
+ const sessionId = (await this.#browser.getSession()).getId();
407
+ let browserHost = '';
408
+
409
+ try {
410
+ const { Name } = await getSessionData(gridUrl, sessionId);
411
+ if (typeof Name == 'string') browserHost = Name;
412
+ } catch {
413
+ /* noop */
414
+ }
415
+
416
+ this.#logger = Logger.getLogger(sessionId);
417
+
418
+ prefix.apply(this.#logger, {
419
+ format(level) {
420
+ const levelColor = colors[level.toUpperCase() as keyof typeof colors];
421
+ return `[${browserName}:${chalk.gray(sessionId)}] ${levelColor(level)} =>`;
422
+ },
423
+ });
424
+
425
+ this.#logger.debug(`Connected successful with ${chalk.green(browserHost)}`);
426
+
427
+ return await runSequence(
428
+ [
429
+ () => this.#browser.manage().setTimeouts({ pageLoad: 10000, script: 60000 }),
430
+ () => this.openStorybookPage(storybookUrl, resolveStorybookUrl),
431
+ () => this.waitForStorybook(),
432
+ () => this.updateStorybookGlobals(storybookGlobals),
433
+ () => this.resolveCreeveyHost(),
434
+ () => this.updateBrowserGlobalVariables(),
435
+ // NOTE: Selenium draws automation toolbar with some delay after webdriver initialization
436
+ // NOTE: So if we resize window right after getting webdriver instance we might get situation
437
+ // NOTE: When the toolbar appears after resize and final viewport size become smaller than we set
438
+ () => this.resizeViewport(viewport),
439
+ () => {
440
+ this.keepAlive();
441
+ },
442
+ ],
443
+ () => !this.#isShuttingDown,
444
+ );
445
+ }
446
+
447
+ private async openStorybookPage(storybookUrl: string, resolver?: () => Promise<string>): Promise<void> {
448
+ if (!LOCALHOST_REGEXP.test(storybookUrl)) {
449
+ return this.#browser.get(appendIframePath(storybookUrl));
450
+ }
451
+
452
+ try {
453
+ if (resolver) {
454
+ this.#logger.debug('Resolving storybook url with custom resolver');
455
+
456
+ const resolvedUrl = await resolver();
457
+
458
+ this.#logger.debug(`Resolver storybook url ${resolvedUrl}`);
459
+
460
+ await this.#browser.get(appendIframePath(resolvedUrl));
461
+ } else {
462
+ // NOTE: getUrlChecker already calls `browser.get` so we don't need another one
463
+ await resolveStorybookUrl(appendIframePath(storybookUrl), (url) => this.checkUrl(url), this.#logger);
464
+ }
465
+ } catch (error) {
466
+ this.#logger.error('Failed to resolve storybook URL', error instanceof Error ? error.message : '');
467
+ throw error;
468
+ }
469
+ }
470
+
471
+ private async checkUrl(url: string): Promise<boolean> {
472
+ try {
473
+ // NOTE: Before trying a new url, reset the current one
474
+ this.#logger.debug(`Opening ${chalk.magenta('about:blank')} page`);
475
+ await openUrlAndWaitForPageSource(
476
+ this.#browser,
477
+ 'about:blank',
478
+ (source: string) => !source.includes('<body></body>'),
479
+ );
480
+ this.#logger.debug(`Opening ${chalk.magenta(url)} and checking the page source`);
481
+ const source = await openUrlAndWaitForPageSource(
482
+ this.#browser,
483
+ url,
484
+ // NOTE: IE11 can return only `head` without body
485
+ (source: string) => source.length == 0 || !/<body([^>]*>).+<\/body>/s.test(source),
486
+ );
487
+ // NOTE: This is the most optimal way to check if we in storybook or not
488
+ // We don't use any page load strategies except `NONE`
489
+ // because other add significant delay and some of them don't work in earlier chrome versions
490
+ // Browsers always load page successful even it's failed
491
+ // So we just check `root` element
492
+ this.#logger.debug(`Checking ${chalk.cyan(`#${storybookRootID}`)} existence on ${chalk.magenta(url)}`);
493
+ return source.includes(`id="${storybookRootID}"`);
494
+ } catch {
495
+ return false;
496
+ }
497
+ }
498
+
499
+ private async waitForStorybook(): Promise<void> {
500
+ this.#logger.debug('Waiting for `setStories` event to make sure that storybook is initiated');
501
+
502
+ const isTimeout = await Promise.race([
503
+ new Promise<boolean>((resolve) => {
504
+ setTimeout(() => {
505
+ resolve(true);
506
+ }, 60000);
507
+ }),
508
+ (async () => {
509
+ let wait = true;
510
+ do {
511
+ try {
512
+ // TODO Research a different way to ensure storybook is initiated
513
+ wait = await this.#browser.executeScript<boolean>(function (SET_GLOBALS: string): boolean {
514
+ // TODO Maybe use
515
+ // import { global } from '@storybook/global';
516
+ // global.IS_STORYBOOK
517
+ if (typeof window.__STORYBOOK_ADDONS_CHANNEL__ == 'undefined') return true;
518
+ if (window.__STORYBOOK_ADDONS_CHANNEL__.last(SET_GLOBALS) == undefined) return true;
519
+ return false;
520
+ }, SET_GLOBALS);
521
+ } catch (e: unknown) {
522
+ this.#logger.debug('An error has been caught during the script:', e);
523
+ }
524
+ } while (wait);
525
+ return false;
526
+ })(),
527
+ ]);
528
+
529
+ // TODO Change the message to describe a reason why it might happen
530
+ if (isTimeout) throw new Error('Failed to wait `setStories` event');
531
+ }
532
+
533
+ private async updateStorybookGlobals(globals?: StorybookGlobals): Promise<void> {
534
+ if (!globals) return;
535
+
536
+ this.#logger.debug('Applying storybook globals');
537
+ await this.#browser.executeScript(function (globals: StorybookGlobals) {
538
+ window.__CREEVEY_UPDATE_GLOBALS__(globals);
539
+ }, globals);
540
+ }
541
+
542
+ private async resolveCreeveyHost(): Promise<void> {
543
+ const addresses = getAddresses();
544
+
545
+ this.#serverHost = await this.#browser.executeAsyncScript(
546
+ function (hosts: string[], port: number, callback: (host?: string | null) => void) {
547
+ void Promise.all(
548
+ hosts.map(function (host) {
549
+ return Promise.race([
550
+ // eslint-disable-next-line @typescript-eslint/restrict-plus-operands
551
+ fetch('http://' + host + ':' + port + '/ping').then(function (response) {
552
+ return response.text();
553
+ }),
554
+ new Promise((_resolve, reject) => {
555
+ setTimeout(reject, 5000);
556
+ }),
557
+ ])
558
+ .then(function (pong) {
559
+ return pong == 'pong' ? host : null;
560
+ })
561
+ .catch(function () {
562
+ return null;
563
+ });
564
+ }),
565
+ ).then(function (hosts) {
566
+ callback(
567
+ hosts.find(function (host) {
568
+ return host != null;
569
+ }),
570
+ );
571
+ });
572
+ },
573
+ addresses,
574
+ this.#serverPort,
575
+ );
576
+
577
+ if (this.#serverHost == null) throw new Error("Can't reach creevey server from a browser");
578
+ }
579
+
580
+ private async updateBrowserGlobalVariables() {
581
+ await this.#browser.executeScript(
582
+ function (workerId: number, creeveyHost: string, creeveyPort: number) {
583
+ window.__CREEVEY_WORKER_ID__ = workerId;
584
+ window.__CREEVEY_SERVER_HOST__ = creeveyHost;
585
+ window.__CREEVEY_SERVER_PORT__ = creeveyPort;
586
+ },
587
+ process.pid,
588
+ this.#serverHost,
589
+ this.#serverPort,
590
+ );
591
+ }
592
+
593
+ private async resizeViewport(viewport?: { width: number; height: number }): Promise<void> {
594
+ if (!viewport) return;
595
+
596
+ const windowRect = await this.#browser.manage().window().getRect();
597
+ const { innerWidth, innerHeight } = await this.#browser.executeScript<{ innerWidth: number; innerHeight: number }>(
598
+ function () {
599
+ return {
600
+ innerWidth: window.innerWidth,
601
+ innerHeight: window.innerHeight,
602
+ };
603
+ },
604
+ );
605
+
606
+ this.#logger.debug(`Resizing viewport from ${innerWidth}x${innerHeight} to ${viewport.width}x${viewport.height}`);
607
+
608
+ const dWidth = windowRect.width - innerWidth;
609
+ const dHeight = windowRect.height - innerHeight;
610
+ await this.#browser
611
+ .manage()
612
+ .window()
613
+ .setRect({
614
+ width: viewport.width + dWidth,
615
+ height: viewport.height + dHeight,
616
+ });
617
+ }
618
+
619
+ private async resetMousePosition(): Promise<void> {
620
+ this.#logger.debug('Resetting mouse position to the top-left corner');
621
+ const browserName = (await this.#browser.getCapabilities()).getBrowserName();
622
+ const [browserVersion] =
623
+ (await this.#browser.getCapabilities()).getBrowserVersion()?.split('.') ??
624
+ ((await this.#browser.getCapabilities()).get('version') as string | undefined)?.split('.') ??
625
+ [];
626
+
627
+ // NOTE Reset mouse position to support keweb selenium grid browser versions
628
+ if (browserName == 'chrome' && browserVersion == '70') {
629
+ const { top, left, width, height } = await this.#browser.executeScript<ElementRect>(function (): ElementRect {
630
+ const bodyRect = document.body.getBoundingClientRect();
631
+
632
+ return {
633
+ top: bodyRect.top,
634
+ left: bodyRect.left,
635
+ width: bodyRect.width,
636
+ height: bodyRect.height,
637
+ };
638
+ });
639
+ // NOTE Bridge mode doesn't support `Origin.VIEWPORT`, move mouse relative
640
+ await this.#browser
641
+ .actions({ bridge: true })
642
+ .move({
643
+ origin: this.#browser.findElement(By.css('body')),
644
+ x: Math.ceil((-1 * width) / 2) - left,
645
+ y: Math.ceil((-1 * height) / 2) - top,
646
+ })
647
+ .perform();
648
+ } else if (browserName == 'firefox' && browserVersion == '61') {
649
+ // NOTE Firefox for some reason moving by 0 x 0 move cursor in bottom left corner :sad:
650
+ await this.#browser.actions().move({ origin: Origin.VIEWPORT, x: 0, y: 1 }).perform();
651
+ } else {
652
+ // NOTE IE don't emit move events until force window focus or connect by RDP on virtual machine
653
+ await this.#browser.actions().move({ origin: Origin.VIEWPORT, x: 0, y: 0 }).perform();
654
+ }
655
+ }
656
+
657
+ private async insertIgnoreStyles(ignoreElements?: string | string[] | null): Promise<WebElement | null> {
658
+ const ignoreSelectors = Array.prototype.concat(ignoreElements).filter(Boolean);
659
+ if (!ignoreSelectors.length) return null;
660
+
661
+ this.#logger.debug('Hiding ignored elements before capturing');
662
+
663
+ return await this.#browser.executeScript(function (ignoreSelectors: string[]) {
664
+ return window.__CREEVEY_INSERT_IGNORE_STYLES__(ignoreSelectors);
665
+ }, ignoreSelectors);
666
+ }
667
+
668
+ private async takeCompositeScreenshot(windowRect: ElementRect, elementRect: ElementRect): Promise<Buffer> {
669
+ const screens = [];
670
+ const isScreenshotWithoutScrollBar = !(await this.hasScrollBar());
671
+ const scrollBarWidth = await this.getScrollBarWidth();
672
+ // NOTE Sometimes viewport has been scrolled somewhere
673
+ const normalizedElementRect = {
674
+ left: elementRect.left - windowRect.left,
675
+ right: elementRect.left + elementRect.width - windowRect.left,
676
+ top: elementRect.top - windowRect.top,
677
+ bottom: elementRect.top + elementRect.height - windowRect.top,
678
+ };
679
+ const isFitHorizontally = windowRect.width >= elementRect.width + normalizedElementRect.left;
680
+ const isFitVertically = windowRect.height >= elementRect.height + normalizedElementRect.top;
681
+ const viewportWidth = windowRect.width - (isFitVertically ? 0 : scrollBarWidth);
682
+ const viewportHeight = windowRect.height - (isFitHorizontally ? 0 : scrollBarWidth);
683
+ const cols = Math.ceil(elementRect.width / viewportWidth);
684
+ const rows = Math.ceil(elementRect.height / viewportHeight);
685
+ const xOffset = Math.round(
686
+ isFitHorizontally ? normalizedElementRect.left : Math.max(0, cols * viewportWidth - elementRect.width),
687
+ );
688
+ const yOffset = Math.round(
689
+ isFitVertically ? normalizedElementRect.top : Math.max(0, rows * viewportHeight - elementRect.height),
690
+ );
691
+
692
+ for (let row = 0; row < rows; row += 1) {
693
+ for (let col = 0; col < cols; col += 1) {
694
+ const dx = Math.min(
695
+ viewportWidth * col + normalizedElementRect.left,
696
+ Math.max(0, normalizedElementRect.right - viewportWidth),
697
+ );
698
+ const dy = Math.min(
699
+ viewportHeight * row + normalizedElementRect.top,
700
+ Math.max(0, normalizedElementRect.bottom - viewportHeight),
701
+ );
702
+ await this.#browser.executeScript(
703
+ function (x: number, y: number) {
704
+ window.scrollTo(x, y);
705
+ },
706
+ dx,
707
+ dy,
708
+ );
709
+ screens.push(await this.#browser.takeScreenshot());
710
+ }
711
+ }
712
+
713
+ const images = screens.map((s) => Buffer.from(s, 'base64')).map((b) => PNG.sync.read(b));
714
+ const compositeImage = new PNG({ width: Math.round(elementRect.width), height: Math.round(elementRect.height) });
715
+
716
+ for (let y = 0; y < compositeImage.height; y += 1) {
717
+ for (let x = 0; x < compositeImage.width; x += 1) {
718
+ const col = Math.floor(x / viewportWidth);
719
+ const row = Math.floor(y / viewportHeight);
720
+ const isLastCol = cols - col == 1;
721
+ const isLastRow = rows - row == 1;
722
+ const scrollOffset = isFitVertically || isScreenshotWithoutScrollBar ? 0 : scrollBarWidth;
723
+ const i = (y * compositeImage.width + x) * 4;
724
+ const j =
725
+ // NOTE compositeImage(x, y) => image(x, y)
726
+ ((y % viewportHeight) * (viewportWidth + scrollOffset) + (x % viewportWidth)) * 4 +
727
+ // NOTE Offset for last row/col image
728
+ (isLastRow ? yOffset * (viewportWidth + scrollOffset) * 4 : 0) +
729
+ (isLastCol ? xOffset * 4 : 0);
730
+ const image = images[row * cols + col];
731
+ compositeImage.data[i + 0] = image.data[j + 0];
732
+ compositeImage.data[i + 1] = image.data[j + 1];
733
+ compositeImage.data[i + 2] = image.data[j + 2];
734
+ compositeImage.data[i + 3] = image.data[j + 3];
735
+ }
736
+ }
737
+ return compositeImage.data;
738
+ }
739
+
740
+ private async removeIgnoreStyles(ignoreStyles: WebElement | null): Promise<void> {
741
+ if (ignoreStyles) {
742
+ this.#logger.debug('Revert hiding ignored elements');
743
+ await this.#browser.executeScript(function (ignoreStyles: HTMLStyleElement) {
744
+ window.__CREEVEY_REMOVE_IGNORE_STYLES__(ignoreStyles);
745
+ }, ignoreStyles);
746
+ }
747
+ }
748
+
749
+ // NOTE Firefox and Safari take viewport screenshot without scrollbars
750
+ private async hasScrollBar(): Promise<boolean> {
751
+ const browserName = (await this.#browser.getCapabilities()).getBrowserName();
752
+ const [browserVersion] = (await this.#browser.getCapabilities()).getBrowserVersion()?.split('.') ?? [];
753
+
754
+ return (
755
+ browserName != 'Safari' &&
756
+ // NOTE This need to work with keweb selenium grid
757
+ !(browserName == 'firefox' && browserVersion == '61')
758
+ );
759
+ }
760
+
761
+ private async getScrollBarWidth(): Promise<number> {
762
+ const scrollBarWidth = await this.#browser.executeScript<number>(function () {
763
+ // eslint-disable-next-line no-var
764
+ var div = document.createElement('div');
765
+ div.innerHTML = 'a'; // NOTE: In IE clientWidth is 0 if this div is empty.
766
+ div.style.overflowY = 'scroll';
767
+ document.body.appendChild(div);
768
+ // eslint-disable-next-line no-var
769
+ var widthDiff = div.offsetWidth - div.clientWidth;
770
+ document.body.removeChild(div);
771
+
772
+ return widthDiff;
773
+ });
774
+ return scrollBarWidth;
775
+ }
776
+
777
+ private keepAlive(): void {
778
+ this.#keepAliveInterval = setInterval(() => {
779
+ // NOTE Simple way to keep session alive
780
+ void this.#browser.getCurrentUrl().then((url) => {
781
+ logger.debug('current url', chalk.magenta(url));
782
+ });
783
+ }, 10 * 1000);
784
+ }
785
+ }