creevey 0.10.0-beta.8 → 0.10.0-rc.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 (350) hide show
  1. package/AUTHORS +2 -0
  2. package/CHANGELOG.md +281 -0
  3. package/README.md +19 -41
  4. package/dist/client/addon/components/Addon.js +18 -8
  5. package/dist/client/addon/components/Addon.js.map +1 -1
  6. package/dist/client/addon/components/Panel.js +4 -4
  7. package/dist/client/addon/components/Panel.js.map +1 -1
  8. package/dist/client/addon/components/TestSelect.js +2 -2
  9. package/dist/client/addon/components/TestSelect.js.map +1 -1
  10. package/dist/client/addon/components/Tools.js +19 -9
  11. package/dist/client/addon/components/Tools.js.map +1 -1
  12. package/dist/client/addon/controller.d.ts +1 -1
  13. package/dist/client/addon/controller.js +3 -3
  14. package/dist/client/addon/controller.js.map +1 -1
  15. package/dist/client/addon/decorator.d.ts +1 -1
  16. package/dist/client/addon/makeDecorator.d.ts +9 -0
  17. package/dist/client/addon/makeDecorator.js +48 -0
  18. package/dist/client/addon/makeDecorator.js.map +1 -0
  19. package/dist/client/addon/manager.js +38 -39
  20. package/dist/client/addon/manager.js.map +1 -1
  21. package/dist/client/addon/preset.d.ts +0 -1
  22. package/dist/client/addon/preset.js +3 -2
  23. package/dist/client/addon/preset.js.map +1 -1
  24. package/dist/client/addon/preview.d.ts +1 -1
  25. package/dist/client/addon/withCreevey.d.ts +5 -3
  26. package/dist/client/addon/withCreevey.js +14 -21
  27. package/dist/client/addon/withCreevey.js.map +1 -1
  28. package/dist/client/shared/components/ImagesView/BlendView.d.ts +2 -2
  29. package/dist/client/shared/components/ImagesView/BlendView.js +18 -8
  30. package/dist/client/shared/components/ImagesView/BlendView.js.map +1 -1
  31. package/dist/client/shared/components/ImagesView/ImagesView.js +1 -1
  32. package/dist/client/shared/components/ImagesView/ImagesView.js.map +1 -1
  33. package/dist/client/shared/components/ImagesView/SideBySideView.d.ts +2 -2
  34. package/dist/client/shared/components/ImagesView/SideBySideView.js +19 -9
  35. package/dist/client/shared/components/ImagesView/SideBySideView.js.map +1 -1
  36. package/dist/client/shared/components/ImagesView/SlideView.d.ts +2 -2
  37. package/dist/client/shared/components/ImagesView/SlideView.js +19 -9
  38. package/dist/client/shared/components/ImagesView/SlideView.js.map +1 -1
  39. package/dist/client/shared/components/ImagesView/SwapView.d.ts +2 -2
  40. package/dist/client/shared/components/ImagesView/SwapView.js +31 -9
  41. package/dist/client/shared/components/ImagesView/SwapView.js.map +1 -1
  42. package/dist/client/shared/components/ImagesView/common.d.ts +1 -1
  43. package/dist/client/shared/components/PageFooter/PageFooter.js +1 -1
  44. package/dist/client/shared/components/PageFooter/PageFooter.js.map +1 -1
  45. package/dist/client/shared/components/PageFooter/Paging.js +1 -1
  46. package/dist/client/shared/components/PageFooter/Paging.js.map +1 -1
  47. package/dist/client/shared/components/PageHeader/ImagePreview.d.ts +2 -2
  48. package/dist/client/shared/components/PageHeader/ImagePreview.js +2 -1
  49. package/dist/client/shared/components/PageHeader/ImagePreview.js.map +1 -1
  50. package/dist/client/shared/components/PageHeader/PageHeader.js +34 -13
  51. package/dist/client/shared/components/PageHeader/PageHeader.js.map +1 -1
  52. package/dist/client/shared/components/ResultsPage.d.ts +2 -2
  53. package/dist/client/shared/components/ResultsPage.js +45 -15
  54. package/dist/client/shared/components/ResultsPage.js.map +1 -1
  55. package/dist/client/shared/creeveyClientApi.js +18 -1
  56. package/dist/client/shared/creeveyClientApi.js.map +1 -1
  57. package/dist/client/shared/helpers.d.ts +1 -3
  58. package/dist/client/shared/helpers.js +4 -19
  59. package/dist/client/shared/helpers.js.map +1 -1
  60. package/dist/client/web/CreeveyApp.d.ts +1 -0
  61. package/dist/client/web/CreeveyApp.js +44 -15
  62. package/dist/client/web/CreeveyApp.js.map +1 -1
  63. package/dist/client/web/CreeveyContext.d.ts +6 -0
  64. package/dist/client/web/CreeveyContext.js +21 -7
  65. package/dist/client/web/CreeveyContext.js.map +1 -1
  66. package/dist/client/web/CreeveyLoader.d.ts +1 -1
  67. package/dist/client/web/CreeveyLoader.js +3 -3
  68. package/dist/client/web/CreeveyLoader.js.map +1 -1
  69. package/dist/client/web/CreeveyView/SideBar/Checkbox.d.ts +4 -4
  70. package/dist/client/web/CreeveyView/SideBar/Checkbox.js +36 -6
  71. package/dist/client/web/CreeveyView/SideBar/Checkbox.js.map +1 -1
  72. package/dist/client/web/CreeveyView/SideBar/Search.js +20 -10
  73. package/dist/client/web/CreeveyView/SideBar/Search.js.map +1 -1
  74. package/dist/client/web/CreeveyView/SideBar/SideBar.js +26 -12
  75. package/dist/client/web/CreeveyView/SideBar/SideBar.js.map +1 -1
  76. package/dist/client/web/CreeveyView/SideBar/SideBarFooter.js +67 -13
  77. package/dist/client/web/CreeveyView/SideBar/SideBarFooter.js.map +1 -1
  78. package/dist/client/web/CreeveyView/SideBar/SideBarHeader.js +32 -12
  79. package/dist/client/web/CreeveyView/SideBar/SideBarHeader.js.map +1 -1
  80. package/dist/client/web/CreeveyView/SideBar/SuiteLink.d.ts +6 -6
  81. package/dist/client/web/CreeveyView/SideBar/SuiteLink.js +20 -13
  82. package/dist/client/web/CreeveyView/SideBar/SuiteLink.js.map +1 -1
  83. package/dist/client/web/CreeveyView/SideBar/TestLink.js +20 -13
  84. package/dist/client/web/CreeveyView/SideBar/TestLink.js.map +1 -1
  85. package/dist/client/web/CreeveyView/SideBar/TestStatusIcon.d.ts +2 -2
  86. package/dist/client/web/CreeveyView/SideBar/TestStatusIcon.js +2 -2
  87. package/dist/client/web/CreeveyView/SideBar/TestStatusIcon.js.map +1 -1
  88. package/dist/client/web/CreeveyView/SideBar/TestsStatus.d.ts +2 -2
  89. package/dist/client/web/CreeveyView/SideBar/TestsStatus.js +3 -2
  90. package/dist/client/web/CreeveyView/SideBar/TestsStatus.js.map +1 -1
  91. package/dist/client/web/CreeveyView/SideBar/Toggle.js +1 -1
  92. package/dist/client/web/CreeveyView/SideBar/Toggle.js.map +1 -1
  93. package/dist/client/web/KeyboardEventsContext.d.ts +1 -8
  94. package/dist/client/web/KeyboardEventsContext.js +79 -64
  95. package/dist/client/web/KeyboardEventsContext.js.map +1 -1
  96. package/dist/client/web/assets/index-CtSq3IhG.js +518 -0
  97. package/dist/client/web/index.html +1 -1
  98. package/dist/client/web/index.js +26 -11
  99. package/dist/client/web/index.js.map +1 -1
  100. package/dist/client/web/themes.d.ts +2 -0
  101. package/dist/client/web/themes.js +22 -0
  102. package/dist/client/web/themes.js.map +1 -0
  103. package/dist/creevey.d.ts +1 -1
  104. package/dist/creevey.js +122 -41
  105. package/dist/creevey.js.map +1 -1
  106. package/dist/index.d.ts +1 -0
  107. package/dist/playwright/generator.d.ts +25 -0
  108. package/dist/playwright/generator.js +243 -0
  109. package/dist/playwright/generator.js.map +1 -0
  110. package/dist/playwright/helpers.d.ts +2 -0
  111. package/dist/playwright/helpers.js +29 -0
  112. package/dist/playwright/helpers.js.map +1 -0
  113. package/dist/playwright/reporter.d.ts +83 -0
  114. package/dist/playwright/reporter.js +334 -0
  115. package/dist/playwright/reporter.js.map +1 -0
  116. package/dist/playwright/setup.d.ts +3 -0
  117. package/dist/playwright/setup.js +72 -0
  118. package/dist/playwright/setup.js.map +1 -0
  119. package/dist/playwright.d.ts +1 -0
  120. package/dist/playwright.js +3 -1
  121. package/dist/playwright.js.map +1 -1
  122. package/dist/server/compare.d.ts +18 -0
  123. package/dist/server/compare.js +182 -0
  124. package/dist/server/compare.js.map +1 -0
  125. package/dist/server/config.d.ts +3 -3
  126. package/dist/server/config.js +75 -8
  127. package/dist/server/config.js.map +1 -1
  128. package/dist/server/connection.d.ts +3 -0
  129. package/dist/server/connection.js +28 -0
  130. package/dist/server/connection.js.map +1 -0
  131. package/dist/server/docker.d.ts +1 -1
  132. package/dist/server/docker.js +54 -32
  133. package/dist/server/docker.js.map +1 -1
  134. package/dist/server/index.d.ts +2 -2
  135. package/dist/server/index.js +165 -64
  136. package/dist/server/index.js.map +1 -1
  137. package/dist/server/master/api.d.ts +11 -6
  138. package/dist/server/master/api.js +88 -25
  139. package/dist/server/master/api.js.map +1 -1
  140. package/dist/server/master/handlers/capture-handler.d.ts +5 -0
  141. package/dist/server/master/handlers/capture-handler.js +25 -0
  142. package/dist/server/master/handlers/capture-handler.js.map +1 -0
  143. package/dist/server/master/handlers/index.d.ts +4 -0
  144. package/dist/server/master/handlers/index.js +21 -0
  145. package/dist/server/master/handlers/index.js.map +1 -0
  146. package/dist/server/master/handlers/ping-handler.d.ts +2 -0
  147. package/dist/server/master/handlers/ping-handler.js +8 -0
  148. package/dist/server/master/handlers/ping-handler.js.map +1 -0
  149. package/dist/server/master/handlers/static-handler.d.ts +1 -0
  150. package/dist/server/master/handlers/static-handler.js +20 -0
  151. package/dist/server/master/handlers/static-handler.js.map +1 -0
  152. package/dist/server/master/handlers/stories-handler.d.ts +4 -0
  153. package/dist/server/master/handlers/stories-handler.js +24 -0
  154. package/dist/server/master/handlers/stories-handler.js.map +1 -0
  155. package/dist/server/master/master.js +7 -24
  156. package/dist/server/master/master.js.map +1 -1
  157. package/dist/server/master/pool.d.ts +1 -0
  158. package/dist/server/master/pool.js +5 -3
  159. package/dist/server/master/pool.js.map +1 -1
  160. package/dist/server/master/queue.d.ts +1 -1
  161. package/dist/server/master/queue.js +14 -6
  162. package/dist/server/master/queue.js.map +1 -1
  163. package/dist/server/master/runner.d.ts +6 -6
  164. package/dist/server/master/runner.js +98 -130
  165. package/dist/server/master/runner.js.map +1 -1
  166. package/dist/server/master/server.d.ts +1 -1
  167. package/dist/server/master/server.js +193 -88
  168. package/dist/server/master/server.js.map +1 -1
  169. package/dist/server/master/start.d.ts +1 -2
  170. package/dist/server/master/start.js +13 -29
  171. package/dist/server/master/start.js.map +1 -1
  172. package/dist/server/master/testsManager.d.ts +81 -0
  173. package/dist/server/master/testsManager.js +282 -0
  174. package/dist/server/master/testsManager.js.map +1 -0
  175. package/dist/server/playwright/docker-file.d.ts +1 -1
  176. package/dist/server/playwright/docker-file.js +17 -8
  177. package/dist/server/playwright/docker-file.js.map +1 -1
  178. package/dist/server/playwright/docker.d.ts +2 -1
  179. package/dist/server/playwright/docker.js +10 -2
  180. package/dist/server/playwright/docker.js.map +1 -1
  181. package/dist/server/playwright/index-source.mjs +16 -0
  182. package/dist/server/playwright/internal.d.ts +7 -7
  183. package/dist/server/playwright/internal.js +144 -84
  184. package/dist/server/playwright/internal.js.map +1 -1
  185. package/dist/server/playwright/webdriver.d.ts +3 -3
  186. package/dist/server/playwright/webdriver.js +0 -6
  187. package/dist/server/playwright/webdriver.js.map +1 -1
  188. package/dist/server/providers/browser.js +4 -3
  189. package/dist/server/providers/browser.js.map +1 -1
  190. package/dist/server/providers/hybrid.js +2 -2
  191. package/dist/server/providers/hybrid.js.map +1 -1
  192. package/dist/server/report.d.ts +10 -0
  193. package/dist/server/report.js +45 -0
  194. package/dist/server/report.js.map +1 -0
  195. package/dist/server/reporters/creevey.d.ts +7 -0
  196. package/dist/server/reporters/creevey.js +63 -0
  197. package/dist/server/reporters/creevey.js.map +1 -0
  198. package/dist/server/reporters/index.d.ts +2 -0
  199. package/dist/server/reporters/index.js +16 -0
  200. package/dist/server/reporters/index.js.map +1 -0
  201. package/dist/server/reporters/junit.d.ts +16 -0
  202. package/dist/server/reporters/junit.js +167 -0
  203. package/dist/server/reporters/junit.js.map +1 -0
  204. package/dist/server/reporters/teamcity.d.ts +7 -0
  205. package/dist/server/reporters/teamcity.js +60 -0
  206. package/dist/server/reporters/teamcity.js.map +1 -0
  207. package/dist/server/selenium/internal.d.ts +4 -4
  208. package/dist/server/selenium/internal.js +56 -40
  209. package/dist/server/selenium/internal.js.map +1 -1
  210. package/dist/server/selenium/selenoid.js +12 -6
  211. package/dist/server/selenium/selenoid.js.map +1 -1
  212. package/dist/server/selenium/webdriver.d.ts +3 -3
  213. package/dist/server/selenium/webdriver.js +4 -8
  214. package/dist/server/selenium/webdriver.js.map +1 -1
  215. package/dist/server/shutdown.d.ts +1 -0
  216. package/dist/server/shutdown.js +23 -0
  217. package/dist/server/shutdown.js.map +1 -0
  218. package/dist/server/stories.d.ts +0 -1
  219. package/dist/server/stories.js +0 -12
  220. package/dist/server/stories.js.map +1 -1
  221. package/dist/server/telemetry.js +3 -3
  222. package/dist/server/telemetry.js.map +1 -1
  223. package/dist/server/testsFiles/parser.js +45 -5
  224. package/dist/server/testsFiles/parser.js.map +1 -1
  225. package/dist/server/utils.d.ts +23 -0
  226. package/dist/server/utils.js +114 -15
  227. package/dist/server/utils.js.map +1 -1
  228. package/dist/server/webdriver.d.ts +1 -1
  229. package/dist/server/worker/context.d.ts +3 -0
  230. package/dist/server/worker/context.js +15 -0
  231. package/dist/server/worker/context.js.map +1 -0
  232. package/dist/server/worker/match-image.d.ts +8 -12
  233. package/dist/server/worker/match-image.js +11 -178
  234. package/dist/server/worker/match-image.js.map +1 -1
  235. package/dist/server/worker/start.d.ts +2 -2
  236. package/dist/server/worker/start.js +41 -64
  237. package/dist/server/worker/start.js.map +1 -1
  238. package/dist/shared/index.d.ts +1 -1
  239. package/dist/shared/index.js +9 -7
  240. package/dist/shared/index.js.map +1 -1
  241. package/dist/types.d.ts +84 -43
  242. package/dist/types.js +65 -1
  243. package/dist/types.js.map +1 -1
  244. package/docs/cli.md +80 -0
  245. package/docs/config.md +179 -165
  246. package/docs/examples/playwright-reporer/playwright.config.ts +37 -0
  247. package/docs/migration-0.9-to-0.10.md +144 -0
  248. package/docs/playwright-reporter.md +357 -0
  249. package/docs/storybook.md +60 -0
  250. package/docs/tests.md +50 -45
  251. package/package.json +78 -83
  252. package/playwright.config.mts +46 -0
  253. package/src/client/addon/components/Addon.tsx +1 -1
  254. package/src/client/addon/components/Panel.tsx +4 -4
  255. package/src/client/addon/components/TestSelect.tsx +2 -2
  256. package/src/client/addon/components/Tools.tsx +2 -2
  257. package/src/client/addon/controller.ts +4 -4
  258. package/src/client/addon/makeDecorator.ts +69 -0
  259. package/src/client/addon/manager.ts +38 -37
  260. package/src/client/addon/preset.ts +2 -1
  261. package/src/client/addon/withCreevey.ts +16 -19
  262. package/src/client/shared/components/ImagesView/BlendView.tsx +1 -1
  263. package/src/client/shared/components/ImagesView/ImagesView.tsx +1 -1
  264. package/src/client/shared/components/ImagesView/SideBySideView.tsx +2 -2
  265. package/src/client/shared/components/ImagesView/SlideView.tsx +2 -2
  266. package/src/client/shared/components/ImagesView/SwapView.tsx +20 -2
  267. package/src/client/shared/components/ImagesView/common.ts +1 -1
  268. package/src/client/shared/components/PageFooter/PageFooter.tsx +1 -1
  269. package/src/client/shared/components/PageFooter/Paging.tsx +1 -1
  270. package/src/client/shared/components/PageHeader/ImagePreview.tsx +2 -1
  271. package/src/client/shared/components/PageHeader/PageHeader.tsx +23 -7
  272. package/src/client/shared/components/ResultsPage.tsx +33 -10
  273. package/src/client/shared/creeveyClientApi.ts +19 -1
  274. package/src/client/shared/helpers.ts +4 -24
  275. package/src/client/web/CreeveyApp.tsx +30 -9
  276. package/src/client/web/CreeveyContext.tsx +11 -0
  277. package/src/client/web/CreeveyLoader.tsx +2 -2
  278. package/src/client/web/CreeveyView/SideBar/Checkbox.tsx +3 -3
  279. package/src/client/web/CreeveyView/SideBar/Search.tsx +4 -4
  280. package/src/client/web/CreeveyView/SideBar/SideBar.tsx +11 -6
  281. package/src/client/web/CreeveyView/SideBar/SideBarFooter.tsx +48 -15
  282. package/src/client/web/CreeveyView/SideBar/SideBarHeader.tsx +20 -5
  283. package/src/client/web/CreeveyView/SideBar/SuiteLink.tsx +12 -12
  284. package/src/client/web/CreeveyView/SideBar/TestLink.tsx +10 -10
  285. package/src/client/web/CreeveyView/SideBar/TestStatusIcon.tsx +2 -2
  286. package/src/client/web/CreeveyView/SideBar/TestsStatus.tsx +3 -2
  287. package/src/client/web/CreeveyView/SideBar/Toggle.tsx +1 -1
  288. package/src/client/web/KeyboardEventsContext.tsx +61 -73
  289. package/src/client/web/index.tsx +10 -5
  290. package/src/client/web/themes.ts +24 -0
  291. package/src/creevey.ts +92 -38
  292. package/src/playwright/generator.ts +322 -0
  293. package/src/playwright/helpers.ts +31 -0
  294. package/src/playwright/reporter.ts +381 -0
  295. package/src/playwright/setup.ts +84 -0
  296. package/src/playwright.ts +1 -0
  297. package/src/server/compare.ts +260 -0
  298. package/src/server/config.ts +52 -9
  299. package/src/server/connection.ts +26 -0
  300. package/src/server/docker.ts +62 -34
  301. package/src/server/index.ts +166 -79
  302. package/src/server/master/api.ts +94 -28
  303. package/src/server/master/handlers/capture-handler.ts +20 -0
  304. package/src/server/master/handlers/index.ts +4 -0
  305. package/src/server/master/handlers/ping-handler.ts +6 -0
  306. package/src/server/master/handlers/static-handler.ts +16 -0
  307. package/src/server/master/handlers/stories-handler.ts +20 -0
  308. package/src/server/master/master.ts +10 -27
  309. package/src/server/master/pool.ts +7 -3
  310. package/src/server/master/queue.ts +21 -7
  311. package/src/server/master/runner.ts +123 -134
  312. package/src/server/master/server.ts +214 -101
  313. package/src/server/master/start.ts +19 -41
  314. package/src/server/master/testsManager.ts +316 -0
  315. package/src/server/playwright/docker-file.ts +20 -8
  316. package/src/server/playwright/docker.ts +16 -3
  317. package/src/server/playwright/index-source.mjs +16 -0
  318. package/src/server/playwright/internal.ts +176 -103
  319. package/src/server/playwright/webdriver.ts +4 -10
  320. package/src/server/providers/browser.ts +4 -3
  321. package/src/server/providers/hybrid.ts +2 -3
  322. package/src/server/report.ts +51 -0
  323. package/src/server/reporters/creevey.ts +71 -0
  324. package/src/server/reporters/index.ts +11 -0
  325. package/src/server/reporters/junit.ts +207 -0
  326. package/src/server/reporters/teamcity.ts +74 -0
  327. package/src/server/selenium/internal.ts +70 -53
  328. package/src/server/selenium/selenoid.ts +13 -6
  329. package/src/server/selenium/webdriver.ts +8 -12
  330. package/src/server/shutdown.ts +19 -0
  331. package/src/server/stories.ts +1 -12
  332. package/src/server/telemetry.ts +3 -3
  333. package/src/server/testsFiles/parser.ts +52 -4
  334. package/src/server/utils.ts +124 -16
  335. package/src/server/webdriver.ts +1 -1
  336. package/src/server/worker/context.ts +14 -0
  337. package/src/server/worker/match-image.ts +16 -248
  338. package/src/server/worker/start.ts +49 -79
  339. package/src/shared/index.ts +10 -8
  340. package/src/types.ts +91 -58
  341. package/types/global.d.ts +1 -0
  342. package/dist/client/web/assets/index-DB8lHlJw.js +0 -591
  343. package/dist/server/reporter.d.ts +0 -26
  344. package/dist/server/reporter.js +0 -108
  345. package/dist/server/reporter.js.map +0 -1
  346. package/dist/server/update.d.ts +0 -2
  347. package/dist/server/update.js +0 -53
  348. package/dist/server/update.js.map +0 -1
  349. package/src/server/reporter.ts +0 -139
  350. package/src/server/update.ts +0 -74
@@ -1,125 +1,238 @@
1
+ import fs from 'fs';
1
2
  import path from 'path';
2
- import http from 'http';
3
- import cluster from 'cluster';
4
- import Koa from 'koa';
5
- import cors from '@koa/cors';
6
- import serve from 'koa-static';
7
- import mount from 'koa-mount';
8
- import body from 'koa-bodyparser';
9
- import WebSocket from 'ws';
10
- import { fileURLToPath, pathToFileURL } from 'url';
11
- import { CreeveyApi } from './api.js';
12
- import { emitStoriesMessage, sendStoriesMessage, subscribeOn, subscribeOnWorker } from '../messages.js';
13
- import { CaptureOptions, isDefined, noop, StoryInput } from '../../types.js';
3
+ import { IncomingMessage, ServerResponse, createServer } from 'http';
4
+ import { WebSocketServer, WebSocket, RawData } from 'ws';
5
+ import { parse, fileURLToPath, pathToFileURL } from 'url';
6
+ import { shutdownOnException } from '../utils.js';
7
+ import { subscribeOn } from '../messages.js';
8
+ import { noop } from '../../types.js';
14
9
  import { logger } from '../logger.js';
15
- import { deserializeStory } from '../../shared/index.js';
10
+ import { CreeveyApi } from './api.js';
11
+ import { pingHandler, captureHandler, storiesHandler, staticHandler } from './handlers/index.js';
16
12
 
17
- const importMetaUrl = pathToFileURL(__filename).href;
13
+ function json<T = unknown>(
14
+ handler: (data: T) => void,
15
+ defaultValue: T,
16
+ ): (request: IncomingMessage, response: ServerResponse) => void {
17
+ return (request: IncomingMessage, response: ServerResponse) => {
18
+ const chunks: Buffer[] = [];
18
19
 
19
- export function start(reportDir: string, port: number, ui: boolean): (api: CreeveyApi) => void {
20
- let resolveApi: (api: CreeveyApi) => void = noop;
21
- let setStoriesCounter = 0;
22
- const creeveyApi = new Promise<CreeveyApi>((resolve) => (resolveApi = resolve));
23
- const app = new Koa();
24
- // eslint-disable-next-line @typescript-eslint/no-misused-promises
25
- const server = http.createServer(app.callback());
26
- const wss = new WebSocket.Server({ server });
27
-
28
- app.use(cors());
29
- app.use(body());
30
-
31
- app.use(async (ctx, next) => {
32
- if (ctx.method == 'GET' && ctx.path == '/ping') {
33
- ctx.body = 'pong';
34
- return;
35
- }
36
- await next();
37
- });
20
+ request.on('data', (chunk: Buffer) => {
21
+ chunks.push(chunk);
22
+ });
38
23
 
39
- if (ui) {
40
- app.use(async (_, next) => {
41
- await creeveyApi;
42
- await next();
24
+ request.on('end', () => {
25
+ try {
26
+ const body = Buffer.concat(chunks);
27
+ const value = body.length === 0 ? defaultValue : (JSON.parse(body.toString('utf-8')) as T);
28
+
29
+ handler(value);
30
+ response.end();
31
+ } catch (error) {
32
+ logger().error('Failed to parse JSON', error);
33
+ const errorMessage = error instanceof Error ? error.message : String(error);
34
+ response.statusCode = 500;
35
+ response.setHeader('Content-Type', 'text/plain');
36
+ response.end(`Failed to parse JSON: ${errorMessage}`);
37
+ }
43
38
  });
44
- }
45
39
 
46
- app.use(async (ctx, next) => {
47
- if (ctx.method == 'POST' && ctx.path == '/stories') {
48
- const { setStoriesCounter: counter, stories } = ctx.request.body as {
49
- setStoriesCounter: number;
50
- stories: [string, StoryInput[]][];
51
- };
52
- if (setStoriesCounter >= counter) return;
53
-
54
- const deserializedStories = stories.map<[string, StoryInput[]]>(([file, stories]) => [
55
- file,
56
- stories.map(deserializeStory),
57
- ]);
58
-
59
- setStoriesCounter = counter;
60
- emitStoriesMessage({ type: 'update', payload: deserializedStories });
61
- Object.values(cluster.workers ?? {})
62
- .filter(isDefined)
63
- .filter((worker) => worker.isConnected())
64
- .forEach((worker) => {
65
- sendStoriesMessage(worker, { type: 'update', payload: deserializedStories });
40
+ request.on('error', (error) => {
41
+ logger().error('Failed to parse JSON', error);
42
+ const errorMessage = error instanceof Error ? error.message : String(error);
43
+ response.statusCode = 500;
44
+ response.setHeader('Content-Type', 'text/plain');
45
+ response.end(`Failed to parse JSON: ${errorMessage}`);
46
+ });
47
+ };
48
+ }
49
+
50
+ function file(handler: (requestedPath: string) => string | undefined) {
51
+ return (request: IncomingMessage, response: ServerResponse) => {
52
+ const parsedUrl = parse(request.url ?? '/', true);
53
+ const requestedPath = decodeURIComponent(parsedUrl.pathname ?? '/');
54
+
55
+ try {
56
+ const filePath = handler(requestedPath);
57
+ if (filePath) {
58
+ const stat = fs.statSync(filePath);
59
+ // Set appropriate MIME type
60
+ const ext = path.extname(filePath).toLowerCase();
61
+ const mimeTypes: Record<string, string> = {
62
+ '.html': 'text/html',
63
+ '.js': 'application/javascript',
64
+ '.css': 'text/css',
65
+ '.json': 'application/json',
66
+ '.png': 'image/png',
67
+ '.jpg': 'image/jpeg',
68
+ '.jpeg': 'image/jpeg',
69
+ '.gif': 'image/gif',
70
+ '.svg': 'image/svg+xml',
71
+ '.ico': 'image/x-icon',
72
+ };
73
+
74
+ const contentType = mimeTypes[ext] || 'application/octet-stream';
75
+
76
+ response.statusCode = 200;
77
+ response.setHeader('Content-Type', contentType);
78
+ response.setHeader('Content-Length', stat.size);
79
+
80
+ // Stream the file
81
+ const stream = fs.createReadStream(filePath);
82
+ stream.pipe(response);
83
+
84
+ stream.on('error', (error) => {
85
+ logger().error('Error streaming file', error);
86
+ if (!response.headersSent) {
87
+ const errorMessage = error instanceof Error ? error.message : String(error);
88
+ response.statusCode = 500;
89
+ response.setHeader('Content-Type', 'text/plain');
90
+ response.end(`Internal server error: ${errorMessage}`);
91
+ }
66
92
  });
67
- return;
93
+ } else {
94
+ logger().error('File not found', requestedPath);
95
+ response.statusCode = 404;
96
+ response.setHeader('Content-Type', 'text/plain');
97
+ response.end('File not found');
98
+ }
99
+ } catch (error) {
100
+ logger().error('Failed to serve file', error);
101
+ const errorMessage = error instanceof Error ? error.message : String(error);
102
+ response.statusCode = 500;
103
+ response.setHeader('Content-Type', 'text/plain');
104
+ response.end(`Failed to serve file: ${errorMessage}`);
68
105
  }
69
- await next();
70
- });
106
+ };
107
+ }
71
108
 
72
- app.use(async (ctx, next) => {
73
- if (ctx.method == 'POST' && ctx.path == '/capture') {
74
- const { workerId, options } = ctx.request.body as { workerId: number; options?: CaptureOptions };
75
- const worker = Object.values(cluster.workers ?? {})
76
- .filter(isDefined)
77
- .find((worker) => worker.process.pid == workerId);
78
- // NOTE: Hypothetical case when someone send to us capture req and we don't have a worker with browser session for it
79
- if (!worker) return;
80
- await new Promise<void>((resolve) => {
81
- const unsubscribe = subscribeOnWorker(worker, 'stories', (message) => {
82
- if (message.type != 'capture') return;
83
- unsubscribe();
84
- resolve();
85
- });
86
- sendStoriesMessage(worker, { type: 'capture', payload: options });
87
- });
88
- // TODO Pass screenshot result to show it in inspector
89
- ctx.body = 'Ok';
90
- return;
109
+ const importMetaUrl = pathToFileURL(__filename).href;
110
+
111
+ export function start(reportDir: string, port: number, ui = false): (api: CreeveyApi) => void {
112
+ let wss: WebSocketServer | null = null;
113
+ let creeveyApi: CreeveyApi | null = null;
114
+ let resolveApi: (api: CreeveyApi) => void = noop;
115
+
116
+ const webDir = path.join(path.dirname(fileURLToPath(importMetaUrl)), '../../client/web');
117
+ const server = createServer();
118
+
119
+ const routes = [
120
+ {
121
+ path: '/ping',
122
+ method: 'GET',
123
+ handler: pingHandler,
124
+ },
125
+ {
126
+ path: '/stories',
127
+ method: 'POST',
128
+ handler: json(storiesHandler, { stories: [] }),
129
+ },
130
+ {
131
+ path: '/capture',
132
+ method: 'POST',
133
+ handler: json(captureHandler, { workerId: 0, options: undefined }),
134
+ },
135
+ {
136
+ path: '/report/',
137
+ method: 'GET',
138
+ handler: file(staticHandler(reportDir, '/report/')),
139
+ },
140
+ {
141
+ path: '/',
142
+ method: 'GET',
143
+ handler: file(staticHandler(webDir)),
144
+ },
145
+ ];
146
+
147
+ const router = (request: IncomingMessage, response: ServerResponse): void => {
148
+ const parsedUrl = parse(request.url ?? '/', true);
149
+ const path = parsedUrl.pathname ?? '/';
150
+ const method = request.method ?? 'GET';
151
+
152
+ try {
153
+ const route = routes.find((route) => path.startsWith(route.path) && route.method === method);
154
+ if (route) {
155
+ route.handler(request, response);
156
+ } else {
157
+ response.statusCode = 404;
158
+ response.setHeader('Content-Type', 'text/plain');
159
+ response.end('Not Found');
160
+ }
161
+ } catch (error) {
162
+ logger().error('Request handling error', error);
163
+ response.statusCode = 500;
164
+ response.setHeader('Content-Type', 'text/plain');
165
+ response.end('Internal Server Error');
91
166
  }
92
- await next();
93
- });
167
+ };
94
168
 
95
- app.use(serve(path.join(path.dirname(fileURLToPath(importMetaUrl)), '../../client/web')));
96
- app.use(mount('/report', serve(reportDir)));
169
+ server.on('request', (request: IncomingMessage, response: ServerResponse): void => {
170
+ response.setHeader('Access-Control-Allow-Origin', '*');
171
+ response.setHeader('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE, OPTIONS');
172
+ response.setHeader('Access-Control-Allow-Headers', 'Origin, X-Requested-With, Content-Type, Accept');
173
+
174
+ if (request.method === 'OPTIONS') {
175
+ response.statusCode = 200;
176
+ response.end();
177
+ return;
178
+ }
97
179
 
98
- wss.on('error', (error) => {
99
- logger().error(error);
180
+ router(request, response);
100
181
  });
101
182
 
102
- server.listen(port);
183
+ if (ui) {
184
+ wss = new WebSocketServer({ server });
185
+ wss.on('connection', (ws: WebSocket) => {
186
+ ws.on('message', (message: RawData, isBinary: boolean) => {
187
+ if (creeveyApi) {
188
+ // NOTE Text messages are passed as Buffer https://github.com/websockets/ws/releases/tag/8.0.0
189
+ // eslint-disable-next-line @typescript-eslint/no-base-to-string
190
+ creeveyApi.handleMessage(ws, isBinary ? message : message.toString('utf-8'));
191
+ return;
192
+ }
193
+ });
103
194
 
104
- subscribeOn('shutdown', () => {
105
- server.close();
106
- wss.close();
107
- wss.clients.forEach((ws) => {
108
- ws.close();
195
+ ws.on('error', (error) => {
196
+ logger().error('WebSocket error', error);
197
+ });
109
198
  });
110
- });
111
199
 
112
- void creeveyApi.then((api) => {
113
- api.subscribe(wss);
200
+ wss.on('error', (error) => {
201
+ logger().error('WebSocket error', error);
202
+ });
203
+ }
114
204
 
115
- wss.on('connection', (ws) => {
116
- ws.on('message', (message: WebSocket.RawData, isBinary: boolean) => {
117
- // NOTE Text messages are passed as Buffer https://github.com/websockets/ws/releases/tag/8.0.0
118
- // eslint-disable-next-line @typescript-eslint/no-base-to-string
119
- api.handleMessage(ws, isBinary ? message : message.toString('utf-8'));
205
+ subscribeOn('shutdown', () => {
206
+ if (wss) {
207
+ wss.clients.forEach((ws) => {
208
+ ws.close();
120
209
  });
121
- });
210
+ wss.close(() => {
211
+ server.close();
212
+ });
213
+ } else {
214
+ server.close();
215
+ }
122
216
  });
123
217
 
218
+ server
219
+ .listen(port, () => {
220
+ logger().info(`Server starting on port ${port}`);
221
+ })
222
+ .on('error', (error: unknown) => {
223
+ logger().error('Failed to start server', error);
224
+ process.exit(1);
225
+ });
226
+
227
+ void new Promise<CreeveyApi>((resolve) => (resolveApi = resolve))
228
+ .then((api) => {
229
+ creeveyApi = api;
230
+ if (wss) {
231
+ creeveyApi.subscribe(wss);
232
+ }
233
+ })
234
+ .catch(shutdownOnException);
235
+
236
+ // Return the function to resolve the API
124
237
  return resolveApi;
125
238
  }
@@ -1,40 +1,14 @@
1
1
  import path from 'path';
2
2
  import { existsSync } from 'fs';
3
- import { fileURLToPath, pathToFileURL } from 'url';
4
- import { copyFile, readdir, mkdir, writeFile } from 'fs/promises';
5
3
  import master from './master.js';
6
- import creeveyApi, { CreeveyApi } from './api.js';
7
- import { Config, Options, TestData, isDefined } from '../../types.js';
8
- import { shutdownWorkers, testsToImages, readDirRecursive } from '../utils.js';
4
+ import { CreeveyApi } from './api.js';
5
+ import { Config, Options, isDefined } from '../../types.js';
6
+ import { shutdownWorkers, testsToImages, readDirRecursive, copyStatics } from '../utils.js';
9
7
  import { subscribeOn } from '../messages.js';
10
8
  import Runner from './runner.js';
11
9
  import { logger } from '../logger.js';
12
10
  import { sendScreenshotsCount } from '../telemetry.js';
13
-
14
- const importMetaUrl = pathToFileURL(__filename).href;
15
-
16
- async function copyStatics(reportDir: string): Promise<void> {
17
- const clientDir = path.join(path.dirname(fileURLToPath(importMetaUrl)), '../../client/web');
18
- const files = (await readdir(clientDir, { withFileTypes: true }))
19
- .filter((dirent) => dirent.isFile() && !dirent.name.endsWith('.d.ts') && !dirent.name.endsWith('.tsx'))
20
- .map((dirent) => dirent.name);
21
- await mkdir(reportDir, { recursive: true });
22
- for (const file of files) {
23
- await copyFile(path.join(clientDir, file), path.join(reportDir, file));
24
- }
25
- }
26
-
27
- function reportDataModule(data: Partial<Record<string, TestData>>): string {
28
- return `
29
- (function (root, factory) {
30
- if (typeof module === 'object' && module.exports) {
31
- module.exports = factory();
32
- } else {
33
- root.__CREEVEY_DATA__ = factory();
34
- }
35
- }(this, function () { return ${JSON.stringify(data)} }));
36
- `;
37
- }
11
+ import { start as startServer } from './server.js';
38
12
 
39
13
  function outputUnnecessaryImages(imagesDir: string, images: Set<string>): void {
40
14
  if (!existsSync(imagesDir)) return;
@@ -49,12 +23,9 @@ function outputUnnecessaryImages(imagesDir: string, images: Set<string>): void {
49
23
  }
50
24
  }
51
25
 
52
- export async function start(
53
- gridUrl: string | undefined,
54
- config: Config,
55
- options: Options,
56
- resolveApi: (api: CreeveyApi) => void,
57
- ): Promise<void> {
26
+ export async function start(gridUrl: string | undefined, config: Config, options: Options): Promise<void> {
27
+ const resolveApi = startServer(config.reportDir, options.port, options.ui);
28
+
58
29
  let runner: Runner | null = null;
59
30
  if (config.hooks.before) {
60
31
  await config.hooks.before();
@@ -77,13 +48,19 @@ export async function start(
77
48
  runner = await master(config, gridUrl);
78
49
 
79
50
  runner.on('stop', () => {
80
- void copyStatics(config.reportDir).then(() =>
81
- writeFile(path.join(config.reportDir, 'data.js'), reportDataModule(runner.status.tests)),
82
- );
51
+ void copyStatics(config.reportDir).then(() => runner.testsManager.saveTestData());
83
52
  });
84
53
 
85
54
  if (options.ui) {
86
- resolveApi(creeveyApi(runner));
55
+ // Initialize TestsManager
56
+ const testsManager = runner.testsManager;
57
+
58
+ // Create the CreeveyApi instance using the existing runner
59
+ const api = new CreeveyApi(testsManager, runner);
60
+
61
+ // Resolve the API for the server
62
+ resolveApi(api);
63
+
87
64
  logger().info(`Started on http://localhost:${options.port}`);
88
65
  } else {
89
66
  if (Object.values(runner.status.tests).filter((test) => test && !test.skip).length == 0) {
@@ -107,7 +84,8 @@ export async function start(
107
84
  logger().warn(`Can't send telemetry: ${error}`);
108
85
  })
109
86
  .finally(() => {
110
- void shutdownWorkers().then(() => process.exit());
87
+ // NOTE: Take some time to kill processes
88
+ void shutdownWorkers().then(() => setTimeout(() => process.exit(), 500));
111
89
  });
112
90
  });
113
91
  // TODO grep