testeranto 0.177.0 → 0.197.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 (546) hide show
  1. package/ALL_LICENSES.txt +11246 -0
  2. package/LICENSE +21 -0
  3. package/README.md +24 -4
  4. package/default-project.json +44 -0
  5. package/design-editor/DesignEditor.tsx +247 -0
  6. package/design-editor/index.ts +2 -0
  7. package/design-editor/server.ts +121 -0
  8. package/design-editor/types.ts +16 -0
  9. package/designs/default-project.json +210 -0
  10. package/dist/common/design-editor/DesignEditor.js +239 -0
  11. package/dist/common/design-editor/index.js +18 -0
  12. package/dist/common/design-editor/server.js +98 -0
  13. package/dist/common/design-editor/types.js +2 -0
  14. package/dist/common/src/App.js +53 -0
  15. package/dist/common/src/Node.js +4 -8
  16. package/dist/common/src/PM/PM_WithEslintAndTsc.js +10 -34
  17. package/dist/common/src/PM/base.js +1 -1
  18. package/dist/common/src/PM/main.js +581 -113
  19. package/dist/common/src/PM/node.js +4 -2
  20. package/dist/common/src/PM/nodeSidecar.js +1 -0
  21. package/dist/common/src/PM/pure.js +20 -57
  22. package/dist/common/src/PM/pureSidecar.js +1 -1
  23. package/dist/common/src/PM/types.js +1 -0
  24. package/dist/common/src/PM/web.js +7 -4
  25. package/dist/common/src/Pure.js +1 -1
  26. package/dist/common/src/Pure.test.js +49 -79
  27. package/dist/common/src/ReportServer.js +8 -2
  28. package/dist/common/src/build.js +31 -40
  29. package/dist/common/src/components/DesignEditorPage.js +169 -0
  30. package/dist/common/src/components/SunriseAnimation.js +291 -0
  31. package/dist/common/src/components/SunriseAnimation.test/interface.js +67 -0
  32. package/dist/common/src/components/SunriseAnimation.test/types.js +2 -0
  33. package/dist/common/src/components/TestStatusBadge.js +1 -21
  34. package/dist/common/src/components/pure/AppFrame.js +69 -0
  35. package/dist/common/src/components/pure/AppFrame.test/implementation.js +63 -0
  36. package/dist/common/src/components/pure/AppFrame.test/index.js +14 -0
  37. package/dist/common/src/components/pure/AppFrame.test/specification.js +25 -0
  38. package/dist/common/src/components/pure/AppFrame.test/types.js +3 -0
  39. package/dist/common/src/components/pure/FeaturesReporterView.js +23 -0
  40. package/dist/common/src/components/pure/FeaturesReporterView.test/implementation.js +84 -0
  41. package/dist/common/src/components/pure/FeaturesReporterView.test/index.js +14 -0
  42. package/dist/common/src/components/pure/FeaturesReporterView.test/specification.js +27 -0
  43. package/dist/common/src/components/pure/FeaturesReporterView.test/types.js +2 -0
  44. package/dist/common/src/components/pure/ModalContent.js +86 -0
  45. package/dist/common/src/components/pure/ModalContent.test/implementation.js +35 -0
  46. package/dist/common/src/components/pure/ModalContent.test/index.js +58 -0
  47. package/dist/common/src/components/pure/ModalContent.test/specification.js +19 -0
  48. package/dist/common/src/components/pure/ModalContent.test/types.js +4 -0
  49. package/dist/common/src/{NavBar.js → components/pure/NavBar.js} +9 -10
  50. package/dist/common/src/components/pure/ProcessManager.js +112 -0
  51. package/dist/common/src/components/pure/ProcessManagerView.js +237 -0
  52. package/dist/common/src/components/pure/ProjectPageView.js +53 -76
  53. package/dist/common/src/components/pure/ProjectPageView.test/implementation.js +167 -54
  54. package/dist/common/src/components/pure/ProjectPageView.test/index.js +5 -1
  55. package/dist/common/src/components/pure/ProjectPageView.test/specification.js +15 -2
  56. package/dist/common/src/components/pure/ProjectPageView.test/types.js +2 -0
  57. package/dist/common/src/components/pure/ProjectsPageView.js +72 -0
  58. package/dist/common/src/components/pure/SettingsButton.js +13 -0
  59. package/dist/common/src/components/pure/SingleProcessView.js +214 -0
  60. package/dist/common/src/components/pure/TestPageView.js +513 -0
  61. package/dist/common/src/components/pure/TestPageView.test/implementation.js +157 -0
  62. package/dist/common/src/components/pure/TestPageView.test/index.js +15 -0
  63. package/dist/common/src/components/pure/TestPageView.test/specification.js +26 -0
  64. package/dist/common/src/components/pure/TestPageView.test/types.js +4 -0
  65. package/dist/common/src/components/pure/ThemeCard.js +15 -0
  66. package/dist/common/src/components/stateful/FeaturesReporter.js +59 -0
  67. package/dist/common/src/components/stateful/FileTree.js +40 -0
  68. package/dist/common/src/components/stateful/ProcessManagerPage.js +112 -0
  69. package/dist/common/src/components/stateful/ProjectPage.js +101 -0
  70. package/dist/common/src/components/stateful/ProjectsPage.js +118 -0
  71. package/dist/common/src/components/stateful/SettingsPage.js +72 -0
  72. package/dist/common/src/components/stateful/SingleProcessPage.js +147 -0
  73. package/dist/common/src/components/stateful/TestPage.js +229 -0
  74. package/dist/common/src/components/stateful/TextEditorPage.js +154 -0
  75. package/dist/common/src/defaultConfig.js +1 -0
  76. package/dist/common/src/esbuildConfigs/inputFilesPlugin.js +3 -4
  77. package/dist/common/src/esbuildConfigs/web.js +1 -1
  78. package/dist/common/src/lib/BaseSuite.js +7 -3
  79. package/dist/common/src/lib/BaseSuite.test/mock.js +17 -41
  80. package/dist/common/src/lib/BaseSuite.test/test.js +33 -42
  81. package/dist/common/src/lib/Sidecar.js +1 -0
  82. package/dist/common/src/lib/abstractBase.js +40 -13
  83. package/dist/common/src/lib/baseBuilder.test/baseBuilder.test.adapter.js +12 -12
  84. package/dist/common/src/lib/baseBuilder.test/baseBuilder.test.implementation.js +15 -13
  85. package/dist/common/src/lib/baseBuilder.test/baseBuilder.test.mock.js +28 -17
  86. package/dist/common/src/lib/baseBuilder.test/baseBuilder.test.pure.js +2 -2
  87. package/dist/common/src/lib/basebuilder.js +29 -35
  88. package/dist/common/src/lib/classBuilder.js +5 -3
  89. package/dist/common/src/lib/core.js +3 -6
  90. package/dist/common/src/lib/core.test/MockCore.js +0 -14
  91. package/dist/common/src/lib/core.test/core.test.adapter.js +2 -9
  92. package/dist/common/src/lib/core.test/core.test.implementation.js +3 -7
  93. package/dist/common/src/lib/index.js +13 -14
  94. package/dist/common/src/lib/pmProxy.js +37 -16
  95. package/dist/common/src/lib/pmProxy.test/adapter.js +20 -4
  96. package/dist/common/src/lib/pmProxy.test/implementation.js +93 -34
  97. package/dist/common/src/lib/pmProxy.test/mockPMBase.js +9 -0
  98. package/dist/common/src/lib/pmProxy.test/specification.js +3 -0
  99. package/dist/common/src/mothership/index.js +5 -0
  100. package/dist/common/src/mothership/test.js +44 -25
  101. package/dist/common/src/types/features.js +34 -0
  102. package/dist/common/src/utils/api.js +57 -27
  103. package/dist/common/src/utils/featureUtils.js +29 -0
  104. package/dist/common/src/utils/logFiles.js +51 -0
  105. package/dist/common/src/utils/makePrompt.js +116 -0
  106. package/dist/common/src/web.html.js +2 -6
  107. package/dist/common/testeranto.config.js +54 -24
  108. package/dist/common/tsconfig.common.tsbuildinfo +1 -1
  109. package/dist/module/design-editor/DesignEditor.js +203 -0
  110. package/dist/module/design-editor/index.js +2 -0
  111. package/dist/module/design-editor/server.js +92 -0
  112. package/dist/module/design-editor/types.js +1 -0
  113. package/dist/module/src/App.js +19 -17
  114. package/dist/module/src/Node.js +4 -8
  115. package/dist/module/src/PM/PM_WithEslintAndTsc.js +11 -35
  116. package/dist/module/src/PM/base.js +1 -1
  117. package/dist/module/src/PM/main.js +577 -109
  118. package/dist/module/src/PM/node.js +4 -2
  119. package/dist/module/src/PM/nodeSidecar.js +1 -0
  120. package/dist/module/src/PM/pure.js +20 -57
  121. package/dist/module/src/PM/pureSidecar.js +1 -1
  122. package/dist/module/src/PM/types.js +1 -0
  123. package/dist/module/src/PM/web.js +7 -4
  124. package/dist/module/src/Pure.js +1 -1
  125. package/dist/module/src/Pure.test.js +49 -79
  126. package/dist/module/src/ReportServer.js +8 -2
  127. package/dist/module/src/build.js +26 -38
  128. package/dist/module/src/components/DesignEditorPage.js +132 -0
  129. package/dist/module/src/components/SunriseAnimation.test/interface.js +32 -1
  130. package/dist/module/src/components/SunriseAnimation.test/types.js +1 -0
  131. package/dist/module/src/components/TestStatusBadge.js +1 -21
  132. package/dist/module/src/components/pure/AppFrame.js +32 -0
  133. package/dist/module/src/components/pure/AppFrame.test/implementation.js +57 -0
  134. package/dist/module/src/components/pure/AppFrame.test/index.js +9 -0
  135. package/dist/module/src/components/pure/AppFrame.test/specification.js +21 -0
  136. package/dist/module/src/components/pure/AppFrame.test/types.js +2 -0
  137. package/dist/module/src/components/pure/FeaturesReporterView.js +16 -0
  138. package/dist/module/src/components/pure/FeaturesReporterView.test/implementation.js +81 -0
  139. package/dist/module/src/components/pure/FeaturesReporterView.test/index.js +9 -0
  140. package/dist/module/src/components/pure/FeaturesReporterView.test/specification.js +23 -0
  141. package/dist/module/src/components/pure/FeaturesReporterView.test/types.js +1 -0
  142. package/dist/module/src/components/pure/ModalContent.js +79 -0
  143. package/dist/module/src/components/pure/ModalContent.test/implementation.js +32 -0
  144. package/dist/module/src/components/pure/ModalContent.test/index.js +53 -0
  145. package/dist/module/src/components/pure/ModalContent.test/specification.js +15 -0
  146. package/dist/module/src/components/pure/ModalContent.test/types.js +3 -0
  147. package/dist/module/src/{NavBar.js → components/pure/NavBar.js} +10 -11
  148. package/dist/module/src/components/pure/ProcessManager.js +75 -0
  149. package/dist/module/src/components/pure/ProcessManagerView.js +200 -0
  150. package/dist/module/src/components/pure/ProjectPageView.js +54 -77
  151. package/dist/module/src/components/pure/ProjectPageView.test/implementation.js +167 -54
  152. package/dist/module/src/components/pure/ProjectPageView.test/index.js +5 -1
  153. package/dist/module/src/components/pure/ProjectPageView.test/specification.js +15 -2
  154. package/dist/module/src/components/pure/ProjectPageView.test/types.js +2 -0
  155. package/dist/module/src/components/pure/ProjectsPageView.js +21 -14
  156. package/dist/module/src/components/pure/SettingsButton.js +6 -0
  157. package/dist/module/src/components/pure/SingleProcessView.js +214 -0
  158. package/dist/module/src/components/pure/TestPageView.js +470 -142
  159. package/dist/module/src/components/pure/TestPageView.test/implementation.js +121 -0
  160. package/dist/module/src/components/pure/TestPageView.test/index.js +10 -0
  161. package/dist/module/src/components/pure/TestPageView.test/specification.js +22 -0
  162. package/dist/module/src/components/pure/TestPageView.test/types.js +3 -0
  163. package/dist/module/src/components/pure/ThemeCard.js +8 -0
  164. package/dist/module/src/components/stateful/FeaturesReporter.js +22 -0
  165. package/dist/module/src/components/stateful/FileTree.js +33 -0
  166. package/dist/module/src/components/stateful/ProcessManagerPage.js +75 -0
  167. package/dist/module/src/components/stateful/ProjectPage.js +7 -6
  168. package/dist/module/src/components/stateful/ProjectsPage.js +32 -6
  169. package/dist/module/src/components/stateful/SettingsPage.js +35 -0
  170. package/dist/module/src/components/stateful/SingleProcessPage.js +110 -0
  171. package/dist/module/src/components/stateful/TestPage.js +123 -13
  172. package/dist/module/src/components/stateful/TextEditorPage.js +117 -0
  173. package/dist/module/src/defaultConfig.js +1 -0
  174. package/dist/module/src/esbuildConfigs/inputFilesPlugin.js +3 -4
  175. package/dist/module/src/esbuildConfigs/web.js +1 -1
  176. package/dist/module/src/lib/BaseSuite.js +7 -3
  177. package/dist/module/src/lib/BaseSuite.test/mock.js +17 -41
  178. package/dist/module/src/lib/BaseSuite.test/test.js +33 -42
  179. package/dist/module/src/lib/Sidecar.js +1 -0
  180. package/dist/module/src/lib/abstractBase.js +40 -13
  181. package/dist/module/src/lib/baseBuilder.test/baseBuilder.test.adapter.js +12 -12
  182. package/dist/module/src/lib/baseBuilder.test/baseBuilder.test.implementation.js +15 -13
  183. package/dist/module/src/lib/baseBuilder.test/baseBuilder.test.mock.js +28 -17
  184. package/dist/module/src/lib/baseBuilder.test/baseBuilder.test.pure.js +1 -1
  185. package/dist/module/src/lib/basebuilder.js +29 -35
  186. package/dist/module/src/lib/classBuilder.js +5 -3
  187. package/dist/module/src/lib/core.js +3 -6
  188. package/dist/module/src/lib/core.test/MockCore.js +0 -14
  189. package/dist/module/src/lib/core.test/core.test.adapter.js +2 -9
  190. package/dist/module/src/lib/core.test/core.test.implementation.js +3 -7
  191. package/dist/module/src/lib/index.js +13 -14
  192. package/dist/module/src/lib/pmProxy.js +37 -16
  193. package/dist/module/src/lib/pmProxy.test/adapter.js +20 -4
  194. package/dist/module/src/lib/pmProxy.test/implementation.js +93 -34
  195. package/dist/module/src/lib/pmProxy.test/mockPMBase.js +9 -0
  196. package/dist/module/src/lib/pmProxy.test/specification.js +3 -0
  197. package/dist/module/src/mothership/index.js +5 -0
  198. package/dist/module/src/mothership/test.js +44 -25
  199. package/dist/module/src/types/features.js +31 -0
  200. package/dist/module/src/utils/api.js +56 -25
  201. package/dist/module/src/utils/featureUtils.js +24 -0
  202. package/dist/module/src/utils/logFiles.js +46 -0
  203. package/dist/module/src/utils/makePrompt.js +109 -0
  204. package/dist/module/src/web.html.js +2 -6
  205. package/dist/module/testeranto.config.js +54 -24
  206. package/dist/module/tsconfig.module.tsbuildinfo +1 -1
  207. package/dist/prebuild/App.css +53 -80
  208. package/dist/prebuild/App.js +15520 -4397
  209. package/dist/prebuild/ReportServer.mjs +104 -6
  210. package/dist/prebuild/build.mjs +95 -108
  211. package/dist/prebuild/mothership/index.mjs +5 -0
  212. package/dist/prebuild/run.mjs +704 -221
  213. package/dist/types/design-editor/DesignEditor.d.ts +18 -0
  214. package/dist/types/design-editor/server.d.ts +1 -0
  215. package/dist/types/src/App.d.ts +1 -0
  216. package/dist/types/src/PM/index.d.ts +1 -1
  217. package/dist/types/src/PM/main.d.ts +37 -3
  218. package/dist/types/src/PM/pure.d.ts +10 -7
  219. package/dist/types/src/PM/web.d.ts +1 -1
  220. package/dist/types/src/Pure.test.d.ts +13 -1
  221. package/dist/types/src/Types.d.ts +1 -0
  222. package/dist/types/src/components/DesignEditorPage.d.ts +1 -0
  223. package/dist/types/src/components/SunriseAnimation.d.ts +5 -0
  224. package/dist/types/src/components/SunriseAnimation.test/interface.d.ts +11 -0
  225. package/dist/types/src/components/SunriseAnimation.test/types.d.ts +39 -0
  226. package/dist/types/src/components/pure/AppFrame.d.ts +11 -0
  227. package/dist/types/src/components/pure/AppFrame.test/implementation.d.ts +3 -0
  228. package/dist/types/src/components/pure/AppFrame.test/index.d.ts +5 -0
  229. package/dist/types/src/components/pure/AppFrame.test/specification.d.ts +3 -0
  230. package/dist/types/src/components/pure/AppFrame.test/types.d.ts +33 -0
  231. package/dist/types/src/components/pure/FeaturesReporterView.d.ts +7 -0
  232. package/dist/types/src/components/pure/FeaturesReporterView.test/implementation.d.ts +3 -0
  233. package/dist/types/src/components/pure/FeaturesReporterView.test/index.d.ts +2 -0
  234. package/dist/types/src/components/pure/FeaturesReporterView.test/specification.d.ts +3 -0
  235. package/dist/types/src/components/pure/FeaturesReporterView.test/types.d.ts +54 -0
  236. package/dist/types/src/components/pure/ModalContent.d.ts +7 -0
  237. package/dist/types/src/components/pure/ModalContent.test/implementation.d.ts +3 -0
  238. package/dist/types/src/components/pure/ModalContent.test/index.d.ts +3 -0
  239. package/dist/types/src/components/pure/ModalContent.test/specification.d.ts +3 -0
  240. package/dist/types/src/components/pure/ModalContent.test/types.d.ts +45 -0
  241. package/dist/types/src/{NavBar.d.ts → components/pure/NavBar.d.ts} +4 -0
  242. package/dist/types/src/components/pure/ProcessManager.d.ts +8 -0
  243. package/dist/types/src/components/pure/ProcessManagerView.d.ts +20 -0
  244. package/dist/types/src/components/pure/ProjectPageView.d.ts +7 -7
  245. package/dist/types/src/components/pure/ProjectPageView.test/index.d.ts +1 -2
  246. package/dist/types/src/components/pure/ProjectPageView.test/types.d.ts +29 -11
  247. package/dist/types/src/components/pure/ProjectsPageView.d.ts +29 -0
  248. package/dist/types/src/components/pure/SettingsButton.d.ts +2 -0
  249. package/dist/types/src/components/pure/SingleProcessView.d.ts +0 -0
  250. package/dist/types/src/components/pure/TestPageView.d.ts +15 -0
  251. package/dist/types/src/components/pure/TestPageView.test/implementation.d.ts +12 -0
  252. package/dist/types/src/components/pure/TestPageView.test/index.d.ts +3 -0
  253. package/dist/types/src/components/pure/TestPageView.test/specification.d.ts +11 -0
  254. package/dist/types/src/components/pure/TestPageView.test/types.d.ts +65 -0
  255. package/dist/types/src/components/pure/ThemeCard.d.ts +9 -0
  256. package/dist/types/src/components/stateful/FeaturesReporter.d.ts +2 -0
  257. package/dist/types/src/components/stateful/FileTree.d.ts +8 -0
  258. package/dist/types/src/components/stateful/ProcessManagerPage.d.ts +2 -0
  259. package/dist/types/src/components/stateful/ProjectPage.d.ts +1 -0
  260. package/dist/types/src/components/stateful/ProjectsPage.d.ts +1 -0
  261. package/dist/types/src/components/stateful/SettingsPage.d.ts +2 -0
  262. package/dist/types/src/components/stateful/SingleProcessPage.d.ts +2 -0
  263. package/dist/types/src/components/stateful/TestPage.d.ts +1 -0
  264. package/dist/types/src/components/stateful/TextEditorPage.d.ts +1 -0
  265. package/dist/types/src/lib/BaseSuite.test/mock.d.ts +3 -3
  266. package/dist/types/src/lib/abstractBase.d.ts +1 -1
  267. package/dist/types/src/lib/abstractBase.test/MockGiven.d.ts +1 -1
  268. package/dist/types/src/lib/baseBuilder.test/baseBuilder.test.pure.d.ts +1 -1
  269. package/dist/types/src/lib/basebuilder.d.ts +1 -1
  270. package/dist/types/src/lib/index.d.ts +8 -8
  271. package/dist/types/src/lib/pmProxy.test/mockPMBase.d.ts +1 -1
  272. package/dist/types/src/types/features.d.ts +7 -0
  273. package/dist/types/src/utils/api.d.ts +1 -5
  274. package/dist/types/src/utils/featureUtils.d.ts +6 -0
  275. package/dist/types/src/utils/logFiles.d.ts +71 -0
  276. package/dist/types/src/utils/makePrompt.d.ts +2 -0
  277. package/dist/types/src/web.html.d.ts +1 -1
  278. package/dist/types/tsconfig.types.tsbuildinfo +1 -1
  279. package/docs/index.md +17 -0
  280. package/docs/test-page-components.md +91 -0
  281. package/docs/testing.prompt.txt +1 -3
  282. package/index.d.ts +38 -13
  283. package/marketing.md +166 -0
  284. package/package.json +38 -11
  285. package/src/App.tsx +22 -25
  286. package/src/Node.ts +6 -9
  287. package/src/PM/PM_WithEslintAndTsc.ts +16 -97
  288. package/src/PM/__tests__/nodeSidecar.testeranto.ts +2 -0
  289. package/src/PM/base.ts +1 -3
  290. package/src/PM/index.ts +1 -1
  291. package/src/PM/main.ts +673 -128
  292. package/src/PM/node.ts +5 -2
  293. package/src/PM/nodeSidecar.ts +1 -0
  294. package/src/PM/pure.ts +28 -72
  295. package/src/PM/pureSidecar.ts +1 -1
  296. package/src/PM/types.ts +0 -0
  297. package/src/PM/web.ts +9 -4
  298. package/src/Pure.test.ts +66 -101
  299. package/src/Pure.ts +1 -1
  300. package/src/README.md +85 -0
  301. package/src/ReportServer.ts +10 -3
  302. package/src/Types.ts +1 -0
  303. package/src/app.scss +169 -0
  304. package/src/build.ts +33 -55
  305. package/src/components/DesignEditorPage.tsx +169 -0
  306. package/src/components/{pure/ProjectPageView.test/index.ts → SunriseAnimation.test/index.tsx} +7 -4
  307. package/src/components/SunriseAnimation.test/interface.ts +49 -0
  308. package/src/components/SunriseAnimation.test/types.ts +53 -0
  309. package/src/components/TestStatusBadge.tsx +2 -23
  310. package/src/components/pure/AppFrame.test/implementation.tsx +72 -0
  311. package/src/components/pure/AppFrame.test/index.tsx +22 -0
  312. package/src/components/pure/AppFrame.test/specification.ts +35 -0
  313. package/src/components/pure/AppFrame.test/types.ts +65 -0
  314. package/src/components/pure/AppFrame.tsx +134 -0
  315. package/src/components/pure/FeaturesReporterView.test/implementation.tsx +106 -0
  316. package/src/components/pure/FeaturesReporterView.test/index.tsx +18 -0
  317. package/src/components/pure/FeaturesReporterView.test/specification.ts +39 -0
  318. package/src/components/pure/FeaturesReporterView.test/types.ts +77 -0
  319. package/src/components/pure/FeaturesReporterView.tsx +37 -0
  320. package/src/components/pure/ModalContent.test/implementation.tsx +45 -0
  321. package/src/components/pure/ModalContent.test/index.tsx +67 -0
  322. package/src/components/pure/ModalContent.test/specification.ts +27 -0
  323. package/src/components/pure/ModalContent.test/types.ts +72 -0
  324. package/src/components/pure/ModalContent.tsx +182 -0
  325. package/src/{NavBar.tsx → components/pure/NavBar.tsx} +31 -15
  326. package/src/components/pure/ProcessManager.tsx +117 -0
  327. package/src/components/pure/ProcessManagerView.tsx +414 -0
  328. package/src/components/pure/ProjectPageView.test/implementation.tsx +186 -57
  329. package/src/components/pure/ProjectPageView.test/index.tsx +18 -0
  330. package/src/components/pure/ProjectPageView.test/specification.ts +15 -2
  331. package/src/components/pure/ProjectPageView.test/types.ts +57 -13
  332. package/src/components/pure/ProjectPageView.tsx +155 -121
  333. package/src/components/pure/ProjectsPageView.tsx +73 -32
  334. package/src/components/pure/SettingsButton.md +1 -0
  335. package/src/components/pure/SettingsButton.tsx +11 -0
  336. package/src/components/pure/SingleProcessView.tsx +235 -0
  337. package/src/components/pure/TestPageView.test/implementation.ts +162 -0
  338. package/src/components/pure/TestPageView.test/index.tsx +20 -0
  339. package/src/components/pure/TestPageView.test/specification.ts +58 -0
  340. package/src/components/pure/TestPageView.test/types.ts +92 -0
  341. package/src/components/pure/TestPageView.tsx +1009 -299
  342. package/src/components/pure/ThemeCard.tsx +34 -0
  343. package/src/components/stateful/FeaturesReporter.tsx +24 -0
  344. package/src/components/stateful/FileTree.tsx +66 -0
  345. package/src/components/stateful/ProcessManagerPage.tsx +108 -0
  346. package/src/components/stateful/ProjectPage.tsx +10 -8
  347. package/src/components/stateful/ProjectsPage.tsx +36 -7
  348. package/src/components/stateful/SettingsPage.tsx +82 -0
  349. package/src/components/stateful/SingleProcessPage.tsx +155 -0
  350. package/src/components/stateful/TestPage.tsx +147 -30
  351. package/src/components/stateful/TextEditorPage.tsx +170 -0
  352. package/src/defaultConfig.ts +1 -0
  353. package/src/esbuildConfigs/inputFilesPlugin.ts +3 -4
  354. package/src/esbuildConfigs/web.ts +2 -1
  355. package/src/lib/BaseSuite.test/mock.ts +21 -68
  356. package/src/lib/BaseSuite.test/test.ts +39 -65
  357. package/src/lib/BaseSuite.ts +9 -3
  358. package/src/lib/Sidecar.ts +2 -0
  359. package/src/lib/abstractBase.test/MockGiven.ts +1 -1
  360. package/src/lib/abstractBase.ts +47 -20
  361. package/src/lib/baseBuilder.test/baseBuilder.test.adapter.ts +14 -13
  362. package/src/lib/baseBuilder.test/baseBuilder.test.implementation.ts +17 -14
  363. package/src/lib/baseBuilder.test/baseBuilder.test.mock.ts +27 -17
  364. package/src/lib/baseBuilder.test/baseBuilder.test.pure.ts +1 -1
  365. package/src/lib/basebuilder.ts +48 -54
  366. package/src/lib/classBuilder.ts +11 -10
  367. package/src/lib/core.test/MockCore.ts +0 -23
  368. package/src/lib/core.test/core.test.adapter.ts +6 -12
  369. package/src/lib/core.test/core.test.implementation.ts +7 -10
  370. package/src/lib/core.ts +5 -20
  371. package/src/lib/index.ts +28 -27
  372. package/src/lib/pmProxy.test/adapter.ts +18 -4
  373. package/src/lib/pmProxy.test/implementation.ts +130 -46
  374. package/src/lib/pmProxy.test/mockPMBase.ts +12 -1
  375. package/src/lib/pmProxy.test/specification.ts +11 -0
  376. package/src/lib/pmProxy.ts +42 -17
  377. package/src/lib/types.ts +2 -0
  378. package/src/mothership/index.ts +6 -0
  379. package/src/mothership/test.ts +53 -26
  380. package/src/style.md +2 -0
  381. package/src/style.scss +3 -486
  382. package/src/templates/frontpage.html +331 -0
  383. package/src/templates/frontpage.md +79 -0
  384. package/src/themes.scss +48 -16
  385. package/src/types/features.ts +38 -0
  386. package/src/utils/api.ts +66 -33
  387. package/src/utils/featureUtils.tsx +42 -0
  388. package/src/utils/logFiles.ts +60 -0
  389. package/src/utils/makePrompt.ts +149 -0
  390. package/src/web.html.ts +6 -6
  391. package/stargazers.txt +15 -0
  392. package/stargzers.js +68 -0
  393. package/testeranto/App.css +53 -80
  394. package/testeranto/App.js +15520 -4397
  395. package/testeranto/bundles/node/core/chunk-4CSH4UJE.mjs +872 -0
  396. package/testeranto/bundles/node/core/chunk-4JTDLQVA.mjs +253 -0
  397. package/testeranto/bundles/node/core/chunk-C3APFDUV.mjs +70 -0
  398. package/testeranto/bundles/node/core/src/lib/BaseSuite.test/node.test.mjs +295 -0
  399. package/testeranto/bundles/node/core/src/lib/baseBuilder.test/baseBuilder.test.node.mjs +243 -0
  400. package/testeranto/bundles/node/core/src/lib/classBuilder.test/classBuilder.test.mjs +411 -0
  401. package/testeranto/bundles/node/core/src/lib/core.test/core.test.mjs +494 -0
  402. package/testeranto/bundles/node/core/src/lib/pmProxy.test/index.mjs +4755 -0
  403. package/testeranto/bundles/pure/core/chunk-62UVCSQC.mjs +1022 -0
  404. package/testeranto/bundles/pure/core/src/Pure.test.mjs +410 -0
  405. package/testeranto/bundles/pure/core/src/lib/BaseSuite.test/pure.test.mjs +352 -0
  406. package/testeranto/bundles/pure/core/src/lib/baseBuilder.test/baseBuilder.test.pure.mjs +241 -0
  407. package/testeranto/bundles/web/core/src/components/pure/FeaturesReporterView.test/index.html +15 -0
  408. package/testeranto/bundles/web/core/src/components/pure/ProjectPageView.test/index.html +1 -5
  409. package/testeranto/bundles/web/core/src/components/pure/TestPageView.test/index.html +15 -0
  410. package/testeranto/bundles/web/core/src/lib/BaseSuite.test/web.test.html +15 -0
  411. package/testeranto/bundles/web/core/src/lib/baseBuilder.test/baseBuilder.test.web.html +15 -0
  412. package/testeranto/metafiles/node/core.json +2906 -0
  413. package/testeranto/metafiles/pure/core.json +860 -0
  414. package/testeranto/metafiles/web/core.json +546 -0
  415. package/testeranto/reports/core/config.json +103 -2
  416. package/testeranto/reports/core/src/Pure.test/pure/exit.log +0 -0
  417. package/testeranto/reports/core/src/Pure.test/pure/lint_errors.txt +0 -0
  418. package/testeranto/reports/core/src/Pure.test/pure/message.txt +17 -0
  419. package/testeranto/reports/core/src/Pure.test/pure/prompt.txt +14 -0
  420. package/testeranto/reports/core/src/Pure.test/pure/type_errors.txt +71 -0
  421. package/testeranto/reports/core/src/components/pure/FeaturesReporterView.test/index/web/debug.log +0 -0
  422. package/testeranto/reports/core/src/components/pure/FeaturesReporterView.test/index/web/error.log +3 -0
  423. package/testeranto/reports/core/src/components/pure/FeaturesReporterView.test/index/web/exit.log +1 -0
  424. package/testeranto/reports/core/src/components/pure/FeaturesReporterView.test/index/web/info.log +0 -0
  425. package/testeranto/reports/core/src/components/pure/FeaturesReporterView.test/index/web/warn.log +0 -0
  426. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/debug.log +0 -0
  427. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/error.log +3 -0
  428. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/exit.log +1 -0
  429. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/info.log +0 -0
  430. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/warn.log +0 -0
  431. package/testeranto/reports/core/src/components/pure/TestPageView.test/index/web/debug.log +0 -0
  432. package/testeranto/reports/core/src/components/pure/TestPageView.test/index/web/error.log +3 -0
  433. package/testeranto/reports/core/src/components/pure/TestPageView.test/index/web/exit.log +1 -0
  434. package/testeranto/reports/core/src/components/pure/TestPageView.test/index/web/info.log +0 -0
  435. package/testeranto/reports/core/src/components/pure/TestPageView.test/index/web/warn.log +0 -0
  436. package/testeranto/reports/core/src/lib/BaseSuite.test/node.test/node/exit.log +1 -0
  437. package/testeranto/reports/core/src/lib/BaseSuite.test/node.test/node/lint_errors.txt +0 -0
  438. package/testeranto/reports/core/src/lib/BaseSuite.test/node.test/node/message.txt +17 -0
  439. package/testeranto/reports/core/src/lib/BaseSuite.test/node.test/node/prompt.txt +14 -0
  440. package/testeranto/reports/core/src/lib/BaseSuite.test/node.test/node/stderr.log +0 -0
  441. package/testeranto/reports/core/src/lib/BaseSuite.test/node.test/node/stdout.log +0 -0
  442. package/testeranto/reports/core/src/lib/BaseSuite.test/node.test/node/tests.json +31 -0
  443. package/testeranto/reports/core/src/lib/BaseSuite.test/node.test/node/type_errors.txt +61 -0
  444. package/testeranto/reports/core/src/lib/BaseSuite.test/pure.test/pure/exit.log +0 -0
  445. package/testeranto/reports/core/src/lib/BaseSuite.test/pure.test/pure/lint_errors.txt +0 -0
  446. package/testeranto/reports/core/src/lib/BaseSuite.test/pure.test/pure/message.txt +17 -0
  447. package/testeranto/reports/core/src/lib/BaseSuite.test/pure.test/pure/prompt.txt +15 -0
  448. package/testeranto/reports/core/src/lib/BaseSuite.test/pure.test/pure/type_errors.txt +61 -0
  449. package/testeranto/reports/core/src/lib/BaseSuite.test/web.test/web/debug.log +0 -0
  450. package/testeranto/reports/core/src/lib/BaseSuite.test/web.test/web/error.log +3 -0
  451. package/testeranto/reports/core/src/lib/BaseSuite.test/web.test/web/exit.log +1 -0
  452. package/testeranto/reports/core/src/lib/BaseSuite.test/web.test/web/info.log +0 -0
  453. package/testeranto/reports/core/src/lib/BaseSuite.test/web.test/web/warn.log +0 -0
  454. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/exit.log +1 -0
  455. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/lint_errors.txt +10 -0
  456. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/message.txt +17 -0
  457. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/prompt.txt +17 -0
  458. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/stderr.log +0 -0
  459. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/stdout.log +6 -0
  460. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/suite-0/given-testInitialization/then-0/butThen/hello.txt +1 -0
  461. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/suite-0/given-testInitialization/then-1/butThen/artifact_test.txt +1 -0
  462. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/tests.json +71 -0
  463. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.node/node/type_errors.txt +42 -0
  464. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.pure/pure/exit.log +0 -0
  465. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.pure/pure/lint_errors.txt +10 -0
  466. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.pure/pure/message.txt +17 -0
  467. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.pure/pure/prompt.txt +17 -0
  468. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.pure/pure/type_errors.txt +42 -0
  469. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.web/web/debug.log +0 -0
  470. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.web/web/error.log +3 -0
  471. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.web/web/exit.log +1 -0
  472. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.web/web/info.log +0 -0
  473. package/testeranto/reports/core/src/lib/baseBuilder.test/baseBuilder.test.web/web/warn.log +0 -0
  474. package/testeranto/reports/core/src/lib/classBuilder.test/classBuilder.test/node/exit.log +1 -0
  475. package/testeranto/reports/core/src/lib/classBuilder.test/classBuilder.test/node/lint_errors.txt +0 -0
  476. package/testeranto/reports/core/src/lib/classBuilder.test/classBuilder.test/node/message.txt +17 -0
  477. package/testeranto/reports/core/src/lib/classBuilder.test/classBuilder.test/node/prompt.txt +17 -0
  478. package/testeranto/reports/core/src/lib/classBuilder.test/classBuilder.test/node/stderr.log +0 -0
  479. package/testeranto/reports/core/src/lib/classBuilder.test/classBuilder.test/node/stdout.log +619 -0
  480. package/testeranto/reports/core/src/lib/classBuilder.test/classBuilder.test/node/tests.json +165 -0
  481. package/testeranto/reports/core/src/lib/classBuilder.test/classBuilder.test/node/type_errors.txt +88 -0
  482. package/testeranto/reports/core/src/lib/core.test/core.test/node/exit.log +1 -0
  483. package/testeranto/reports/core/src/lib/core.test/core.test/node/lint_errors.txt +21 -0
  484. package/testeranto/reports/core/src/lib/core.test/core.test/node/message.txt +17 -0
  485. package/testeranto/reports/core/src/lib/core.test/core.test/node/prompt.txt +19 -0
  486. package/testeranto/reports/core/src/lib/core.test/core.test/node/stderr.log +0 -0
  487. package/testeranto/reports/core/src/lib/core.test/core.test/node/stdout.log +0 -0
  488. package/testeranto/reports/core/src/lib/core.test/core.test/node/type_errors.txt +45 -0
  489. package/testeranto/reports/core/src/lib/pmProxy.test/index/node/exit.log +1 -0
  490. package/testeranto/reports/core/src/lib/pmProxy.test/index/node/lint_errors.txt +15 -0
  491. package/testeranto/reports/core/src/lib/pmProxy.test/index/node/message.txt +17 -0
  492. package/testeranto/reports/core/src/lib/pmProxy.test/index/node/prompt.txt +17 -0
  493. package/testeranto/reports/core/src/lib/pmProxy.test/index/node/stderr.log +88 -0
  494. package/testeranto/reports/core/src/lib/pmProxy.test/index/node/stdout.log +10 -0
  495. package/testeranto/reports/core/src/lib/pmProxy.test/index/node/tests.json +152 -0
  496. package/testeranto/reports/core/src/lib/pmProxy.test/index/node/type_errors.txt +52 -0
  497. package/testeranto/reports/core/summary.json +74 -5
  498. package/testeranto/reportsweb_build_errors +546 -0
  499. package/testeranto.config.ts +58 -25
  500. package/tsc.log +378 -211
  501. package/tsconfig.common.json +4 -2
  502. package/tsconfig.json +9 -6
  503. package/tsconfig.module.json +3 -2
  504. package/tsconfig.types.json +3 -1
  505. package/dist/common/src/components/pure/ProjectPageView.test/adapter.js +0 -20
  506. package/dist/common/src/lib/dailyAnimation.js +0 -130
  507. package/dist/module/src/Footer.js +0 -5
  508. package/dist/module/src/ProjectPage.js +0 -319
  509. package/dist/module/src/ProjectsPage.js +0 -1
  510. package/dist/module/src/SettingsButton.js +0 -157
  511. package/dist/module/src/TestPage.js +0 -271
  512. package/dist/module/src/TestReport.js +0 -368
  513. package/dist/module/src/components/pure/ProjectPageView.test/adapter.js +0 -17
  514. package/dist/module/src/lib/dailyAnimation.js +0 -130
  515. package/dist/types/src/components/pure/ProjectPageView.test/adapter.d.ts +0 -3
  516. package/dist/types/testeranto.config.d.ts +0 -3
  517. package/docs/testing.ai.txt +0 -106
  518. package/docs.html +0 -572
  519. package/example.css +0 -351
  520. package/index.html +0 -284
  521. package/scripts/compile-docs.js +0 -140
  522. package/src/App.scss +0 -132
  523. package/src/Footer.tsx +0 -8
  524. package/src/Project.scss +0 -1
  525. package/src/ProjectPage.tsx +0 -459
  526. package/src/ProjectsPage.tsx +0 -1
  527. package/src/ReportApp.scss +0 -1
  528. package/src/SettingsButton.tsx +0 -268
  529. package/src/TestPage.tsx +0 -476
  530. package/src/TestReport.scss +0 -24
  531. package/src/TestReport.tsx +0 -411
  532. package/src/components/pure/ProjectPageView.test/adapter.ts +0 -21
  533. package/src/fonts.scss +0 -55
  534. package/src/lib/dailyAnimation.ts +0 -147
  535. package/style.css +0 -12156
  536. package/testeranto/bundles/node/core/metafile.json +0 -8
  537. package/testeranto/bundles/pure/core/metafile.json +0 -8
  538. package/testeranto/bundles/web/core/metafile.json +0 -15086
  539. package/testeranto/bundles/web/core/src/components/pure/ProjectPageView.test/index.mjs +0 -39991
  540. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/lint_errors.txt +0 -18
  541. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/logs.txt +0 -59
  542. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/message.txt +0 -2
  543. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/prompt.txt +0 -27
  544. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/suite-0/given-basicRender/then-0/butThen/happyPath.png +0 -0
  545. package/testeranto/reports/core/src/components/pure/ProjectPageView.test/index/web/type_errors.txt +0 -50
  546. /package/dist/types/src/{lib/dailyAnimation.d.ts → PM/types.d.ts} +0 -0
@@ -1,338 +1,1048 @@
1
- import React from 'react';
2
- import { Tab, Container, Alert, Button } from 'react-bootstrap';
3
- import { NavBar } from '../../NavBar';
4
- import { TestStatusBadge } from '../TestStatusBadge';
1
+ /* eslint-disable @typescript-eslint/no-unused-vars */
2
+ /* eslint-disable @typescript-eslint/no-explicit-any */
3
+ import React, { ReactElement, useState, useMemo, useEffect } from "react";
4
+ import { Toast, ToastContainer } from "react-bootstrap";
5
+ import { useNavigate } from "react-router-dom";
6
+ import {
7
+ RuntimeName,
8
+ STANDARD_LOGS,
9
+ RUNTIME_SPECIFIC_LOGS
10
+ } from "../../utils/logFiles";
11
+ import { Container, Row, Col, Nav, Button, Modal } from "react-bootstrap";
12
+ import { Editor } from "@monaco-editor/react";
13
+ import { NavBar } from "./NavBar";
14
+ import { TestStatusBadge } from "../TestStatusBadge";
15
+
16
+ type TestData = {
17
+ name: string;
18
+ givens: {
19
+ name: string;
20
+ whens: {
21
+ name: string;
22
+ error?: string;
23
+ features?: string[];
24
+ artifacts?: string[];
25
+ }[];
26
+ thens: {
27
+ name: string;
28
+ error?: string;
29
+ features?: string[];
30
+ artifacts?: string[];
31
+ }[];
32
+ features?: string[];
33
+ artifacts?: string[];
34
+ }[];
35
+ };
36
+
37
+ type TestPageViewProps = {
38
+ projectName: string;
39
+ testName: string;
40
+ decodedTestPath: string;
41
+ runtime: string;
42
+ logs: Record<string, string>;
43
+ testsExist: boolean;
44
+ errorCounts: {
45
+ runTimeErrors: number;
46
+ typeErrors: number;
47
+ staticErrors: number;
48
+ };
49
+ };
50
+
51
+ const FileTree = ({ data, onSelect, level = 0, selectedSourcePath }) => {
52
+ const [expanded, setExpanded] = useState({});
53
+
54
+ const toggleExpand = (path) => {
55
+ setExpanded((prev) => ({ ...prev, [path]: !prev[path] }));
56
+ };
57
+
58
+ return (
59
+ <div>
60
+ {Object.entries(data).map(([name, node]) => {
61
+ const path = Object.keys(expanded).find((k) => k.endsWith(name)) || name;
62
+ const isExpanded = expanded[path];
63
+
64
+ if (node.__isFile) {
65
+ return (
66
+ <FileTreeItem
67
+ key={name}
68
+ name={name}
69
+ isFile={true}
70
+ level={level}
71
+ isSelected={selectedSourcePath === path}
72
+ onClick={() => onSelect(path, node.content)}
73
+ />
74
+ );
75
+ } else {
76
+ return (
77
+ <div key={name}>
78
+ <div
79
+ className="d-flex align-items-center py-1 text-dark"
80
+ style={{
81
+ paddingLeft: `${level * 16}px`,
82
+ cursor: 'pointer',
83
+ fontSize: '0.875rem'
84
+ }}
85
+ onClick={() => toggleExpand(path)}
86
+ >
87
+ <i className={`bi ${isExpanded ? 'bi-folder2-open' : 'bi-folder'} me-1`}></i>
88
+ <span>{name}</span>
89
+ </div>
90
+ {isExpanded && (
91
+ <FileTree
92
+ data={node}
93
+ onSelect={onSelect}
94
+ level={level + 1}
95
+ selectedSourcePath={selectedSourcePath}
96
+ />
97
+ )}
98
+ </div>
99
+ );
100
+ }
101
+ })}
102
+ </div>
103
+ );
104
+ };
5
105
 
6
106
  export const TestPageView = ({
7
- route,
8
- setRoute,
9
- navigate,
10
107
  projectName,
11
108
  testName,
12
109
  decodedTestPath,
13
110
  runtime,
14
- testData,
15
- logs,
16
- typeErrors,
17
- lintErrors,
18
111
  testsExist,
19
112
  errorCounts,
20
- }) => {
113
+ logs,
114
+ }: TestPageViewProps) => {
115
+ const navigate = useNavigate();
116
+ const [showAiderModal, setShowAiderModal] = useState(false);
117
+ const [messageOption, setMessageOption] = useState<'default' | 'custom'>('default');
118
+ const [customMessage, setCustomMessage] = useState(
119
+ typeof logs['message.txt'] === 'string' ? logs['message.txt'] : 'make a script that prints hello'
120
+ );
121
+ const [showToast, setShowToast] = useState(false);
122
+ const [toastMessage, setToastMessage] = useState('');
123
+ const [toastVariant, setToastVariant] = useState<'success' | 'danger'>('success');
124
+ const [ws, setWs] = useState<WebSocket | null>(null);
125
+ const [expandedSections, setExpandedSections] = useState({
126
+ standardLogs: true,
127
+ runtimeLogs: true,
128
+ sourceFiles: true
129
+ });
130
+ const [isNavbarCollapsed, setIsNavbarCollapsed] = useState(false);
131
+
132
+ // Update customMessage when logs change
133
+ useEffect(() => {
134
+ if (typeof logs['message.txt'] === 'string' && logs['message.txt'].trim()) {
135
+ setCustomMessage(logs['message.txt']);
136
+ }
137
+ }, [logs]);
138
+
139
+ // Set up WebSocket connection
140
+ useEffect(() => {
141
+ const wsProtocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
142
+ const wsUrl = `${wsProtocol}//${window.location.host}`;
143
+ const websocket = new WebSocket(wsUrl);
144
+ setWs(websocket);
145
+
146
+ return () => {
147
+ websocket.close();
148
+ };
149
+ }, []);
150
+ const [activeTab, setActiveTab] = React.useState("tests.json");
151
+ const [selectedFile, setSelectedFile] = useState<{
152
+ path: string;
153
+ content: string;
154
+ language?: string;
155
+ } | null>(null);
156
+ const [selectedSourcePath, setSelectedSourcePath] = useState<string | null>(
157
+ null
158
+ );
159
+ const [editorTheme, setEditorTheme] = useState<"light" | "vs-dark">(
160
+ "vs-dark"
161
+ );
162
+
163
+ // Determine language from file extension
164
+ const getLanguage = (path: string) => {
165
+ const ext = path.split(".").pop()?.toLowerCase();
166
+ switch (ext) {
167
+ case "ts":
168
+ return "typescript";
169
+ case "tsx":
170
+ return "typescript";
171
+ case "js":
172
+ return "javascript";
173
+ case "json":
174
+ return "json";
175
+ case "md":
176
+ return "markdown";
177
+ default:
178
+ return "plaintext";
179
+ }
180
+ };
181
+
182
+ const renderTestResults = (testData: TestData) => {
183
+ return (
184
+ <div className="test-results">
185
+ {testData.givens.map((given, i) => (
186
+ <div key={i} className="mb-4 card">
187
+ <div className="card-header bg-primary text-white">
188
+ <div className="d-flex justify-content-between align-items-center">
189
+ <div>
190
+ <h4>Given: {given.name}</h4>
191
+ {given.features && given.features.length > 0 && (
192
+ <div className="mt-1">
193
+ <small>Features:</small>
194
+ <ul className="list-unstyled">
195
+ {given.features.map((feature, fi) => (
196
+ <li key={fi}>
197
+ {feature.startsWith("http") ? (
198
+ <a
199
+ href={feature}
200
+ target="_blank"
201
+ rel="noopener noreferrer"
202
+ className="text-white"
203
+ >
204
+ {new URL(feature).hostname}
205
+ </a>
206
+ ) : (
207
+ <span className="text-white">{feature}</span>
208
+ )}
209
+ </li>
210
+ ))}
211
+ </ul>
212
+ </div>
213
+ )}
214
+ </div>
215
+ {given.artifacts && given.artifacts.length > 0 && (
216
+ <div className="dropdown">
217
+ <button
218
+ className="btn btn-sm btn-light dropdown-toggle"
219
+ type="button"
220
+ data-bs-toggle="dropdown"
221
+ >
222
+ Artifacts ({given.artifacts.length})
223
+ </button>
224
+ <ul className="dropdown-menu dropdown-menu-end">
225
+ {given.artifacts.map((artifact, ai) => (
226
+ <li key={ai}>
227
+ <a
228
+ className="dropdown-item"
229
+ href={`reports/${projectName}/${testName
230
+ .split(".")
231
+ .slice(0, -1)
232
+ .join(".")}/${runtime}/${artifact}`}
233
+ target="_blank"
234
+ rel="noopener noreferrer"
235
+ >
236
+ {artifact.split("/").pop()}
237
+ </a>
238
+ </li>
239
+ ))}
240
+ </ul>
241
+ </div>
242
+ )}
243
+ </div>
244
+ </div>
245
+ <div className="card-body">
246
+ {given.whens.map((when, j) => (
247
+ <div
248
+ key={`w-${j}`}
249
+ className={`p-3 mb-2 ${when.error
250
+ ? "bg-danger text-white"
251
+ : "bg-success text-white"
252
+ }`}
253
+ >
254
+ <div className="d-flex justify-content-between align-items-start">
255
+ <div>
256
+ <div>
257
+ <strong>When:</strong> {when.name}
258
+ {when.features && when.features.length > 0 && (
259
+ <div className="mt-2">
260
+ <small>Features:</small>
261
+ <ul className="list-unstyled">
262
+ {when.features.map((feature, fi) => (
263
+ <li key={fi}>
264
+ {feature.startsWith("http") ? (
265
+ <a
266
+ href={feature}
267
+ target="_blank"
268
+ rel="noopener noreferrer"
269
+ >
270
+ {new URL(feature).hostname}
271
+ </a>
272
+ ) : (
273
+ feature
274
+ )}
275
+ </li>
276
+ ))}
277
+ </ul>
278
+ </div>
279
+ )}
280
+ {when.error && <pre className="mt-2">{when.error}</pre>}
281
+ </div>
282
+ </div>
283
+ {when.artifacts && when.artifacts.length > 0 && (
284
+ <div className="ms-3">
285
+ <strong>Artifacts:</strong>
286
+ <ul className="list-unstyled">
287
+ {when.artifacts.map((artifact, ai) => (
288
+ <li key={ai}>
289
+ <a
290
+ href={`reports/${projectName}/${testName
291
+ .split(".")
292
+ .slice(0, -1)
293
+ .join(".")}/${runtime}/${artifact}`}
294
+ target="_blank"
295
+ className="text-white"
296
+ rel="noopener noreferrer"
297
+ >
298
+ {artifact.split("/").pop()}
299
+ </a>
300
+ </li>
301
+ ))}
302
+ </ul>
303
+ </div>
304
+ )}
305
+ </div>
306
+ </div>
307
+ ))}
308
+ {given.thens.map((then, k) => (
309
+ <div
310
+ key={`t-${k}`}
311
+ className={`p-3 mb-2 ${then.error
312
+ ? "bg-danger text-white"
313
+ : "bg-success text-white"
314
+ }`}
315
+ >
316
+ <div className="d-flex justify-content-between align-items-start">
317
+ <div>
318
+ <div>
319
+ <strong>Then:</strong> {then.name}
320
+ {then.features && then.features.length > 0 && (
321
+ <div className="mt-2">
322
+ <small>Features:</small>
323
+ <ul className="list-unstyled">
324
+ {then.features.map((feature, fi) => (
325
+ <li key={fi}>
326
+ {feature.startsWith("http") ? (
327
+ <a
328
+ href={feature}
329
+ target="_blank"
330
+ rel="noopener noreferrer"
331
+ >
332
+ {new URL(feature).hostname}
333
+ </a>
334
+ ) : (
335
+ feature
336
+ )}
337
+ </li>
338
+ ))}
339
+ </ul>
340
+ </div>
341
+ )}
342
+ {then.error && <pre className="mt-2">{then.error}</pre>}
343
+ </div>
344
+ </div>
345
+ {then.artifacts && then.artifacts.length > 0 && (
346
+ <div className="ms-3">
347
+ <strong>Artifacts:</strong>
348
+ <ul className="list-unstyled">
349
+ {then.artifacts.map((artifact, ai) => (
350
+ <li key={ai}>
351
+ <a
352
+ href={`reports/${projectName}/${testName
353
+ .split(".")
354
+ .slice(0, -1)
355
+ .join(".")}/${runtime}/${artifact}`}
356
+ target="_blank"
357
+ className="text-white"
358
+ rel="noopener noreferrer"
359
+ >
360
+ {artifact.split("/").pop()}
361
+ </a>
362
+ </li>
363
+ ))}
364
+ </ul>
365
+ </div>
366
+ )}
367
+ </div>
368
+ </div>
369
+ ))}
370
+ </div>
371
+ </div>
372
+ ))}
373
+ </div>
374
+ );
375
+ };
376
+
377
+ console.log("Rendering TestPageView with logs:", {
378
+ logKeys: Object.keys(logs),
379
+ sourceFiles: logs.source_files ? Object.keys(logs.source_files) : null,
380
+ selectedFile,
381
+ activeTab,
382
+ });
383
+
21
384
  return (
22
- <Container fluid={true}>
385
+ <Container fluid className="px-0">
23
386
  <NavBar
24
387
  title={decodedTestPath}
25
388
  backLink={`/projects/${projectName}`}
26
389
  navItems={[
27
390
  {
28
- label: '',
391
+ label: "",
29
392
  badge: {
30
- variant: runtime === 'node' ? 'primary' :
31
- runtime === 'web' ? 'success' :
32
- 'info',
33
- text: runtime
393
+ variant:
394
+ runtime === "node"
395
+ ? "primary"
396
+ : runtime === "web"
397
+ ? "success"
398
+ : "info",
399
+ text: runtime,
34
400
  },
35
- className: 'pe-none d-flex align-items-center gap-2'
36
- },
37
- {
38
- to: `#results`,
39
- label: (
40
- <TestStatusBadge
41
- testName={decodedTestPath}
42
- testsExist={testsExist}
43
- runTimeErrors={errorCounts.runTimeErrors}
44
- variant="compact"
45
- />
46
- ),
47
- className: !testsExist || errorCounts.runTimeErrors > 0
48
- ? 'text-danger fw-bold'
49
- : '',
50
- active: route === 'results'
51
- },
52
- {
53
- to: `#logs`,
54
- label: `Runtime logs`,
55
- active: route === 'logs'
56
- },
57
- {
58
- to: `#types`,
59
- label: errorCounts.typeErrors > 0
60
- ? `tsc (❌ * ${errorCounts.typeErrors})`
61
- : 'tsc ✅ ',
62
- active: route === 'types'
63
- },
64
- {
65
- to: `#lint`,
66
- label: errorCounts.staticErrors > 0
67
- ? `eslint (❌ *${errorCounts.staticErrors}) `
68
- : 'eslint ✅',
69
- active: route === 'lint'
401
+ className: "pe-none d-flex align-items-center gap-2",
70
402
  },
71
403
  ]}
72
404
  rightContent={
73
405
  <Button
74
406
  variant="info"
407
+ onClick={() => setShowAiderModal(true)}
408
+ className="ms-2"
409
+ title="AI Assistant"
410
+ >
411
+ 🤖
412
+ </Button>
413
+ }
414
+ />
415
+
416
+ <Modal show={showAiderModal} onHide={() => setShowAiderModal(false)} size="lg" onShow={() => setMessageOption('default')}>
417
+ <Modal.Header closeButton>
418
+ <Modal.Title>Aider</Modal.Title>
419
+ </Modal.Header>
420
+ <Modal.Body>
421
+
422
+ <div className="mb-3">
423
+
424
+ <div className="form-check">
425
+ <input
426
+ className="form-check-input"
427
+ type="radio"
428
+ name="messageOption"
429
+ id="defaultMessage"
430
+ value="default"
431
+ checked={messageOption === 'default'}
432
+ onChange={() => setMessageOption('default')}
433
+ />
434
+ <label className="form-check-label" htmlFor="defaultMessage">
435
+ Use default message.txt
436
+ </label>
437
+ </div>
438
+ <div className="form-check">
439
+ <input
440
+ className="form-check-input"
441
+ type="radio"
442
+ name="messageOption"
443
+ id="customMessage"
444
+ value="custom"
445
+ checked={messageOption === 'custom'}
446
+ onChange={() => setMessageOption('custom')}
447
+ />
448
+ <label className="form-check-label" htmlFor="customMessage">
449
+ Use custom message
450
+ </label>
451
+ </div>
452
+ {messageOption === 'custom' && (
453
+ <div className="mt-2">
454
+ <textarea
455
+ className="form-control"
456
+ rows={8}
457
+ placeholder="Enter your custom message"
458
+ value={customMessage}
459
+ onChange={(e) => setCustomMessage(e.target.value)}
460
+ style={{ minHeight: '500px' }}
461
+ />
462
+ </div>
463
+ )}
464
+ </div>
465
+ </Modal.Body>
466
+ <Modal.Footer>
467
+ {/* <Button
468
+ variant="secondary"
469
+ onClick={() => setShowAiderModal(false)}
470
+ >
471
+ Close
472
+ </Button> */}
473
+ <Button
474
+ variant="primary"
75
475
  onClick={async () => {
76
476
  try {
77
- const promptPath = `testeranto/reports/${projectName}/${testName.split('.').slice(0, -1).join('.')}/${runtime}/prompt.txt`;
78
- const messagePath = `testeranto/reports/${projectName}/${testName.split('.').slice(0, -1).join('.')}/${runtime}/message.txt`;
79
- const command = `aider --load ${promptPath} --message-file ${messagePath}`;
80
- await navigator.clipboard.writeText(command);
81
- alert("Copied aider command to clipboard!");
477
+ const promptPath = `testeranto/reports/${projectName}/${testName
478
+ .split(".")
479
+ .slice(0, -1)
480
+ .join(".")}/${runtime}/prompt.txt`;
481
+
482
+ let command = `aider --load ${promptPath}`;
483
+
484
+ if (messageOption === 'default') {
485
+ const messagePath = `testeranto/reports/${projectName}/${testName
486
+ .split(".")
487
+ .slice(0, -1)
488
+ .join(".")}/${runtime}/message.txt`;
489
+ command += ` --message-file ${messagePath}`;
490
+ } else {
491
+ command += ` --message "${customMessage}"`;
492
+ }
493
+
494
+ // Send command to server via WebSocket
495
+ const ws = new WebSocket(`ws://${window.location.host}`);
496
+ ws.onopen = () => {
497
+ ws.send(JSON.stringify({
498
+ type: 'executeCommand',
499
+ command: command
500
+ }));
501
+ setToastMessage('Command sent to server');
502
+ setToastVariant('success');
503
+ setShowToast(true);
504
+ setShowAiderModal(false);
505
+ ws.close();
506
+
507
+ // Navigate to process manager page
508
+ setTimeout(() => {
509
+ navigate('/processes');
510
+ }, 1000);
511
+ };
512
+
513
+ ws.onerror = (error) => {
514
+ setToastMessage('Failed to connect to server');
515
+ setToastVariant('danger');
516
+ setShowToast(true);
517
+ };
82
518
  } catch (err) {
83
- alert("Failed to copy command to clipboard");
84
- console.error("Copy failed:", err);
519
+ console.error("WebSocket error:", err);
520
+ setToastMessage('Error preparing command');
521
+ setToastVariant('danger');
522
+ setShowToast(true);
85
523
  }
86
524
  }}
87
- className="ms-2"
88
525
  >
89
- 🤖
526
+ Run Aider Command
90
527
  </Button>
91
- }
92
- />
528
+ </Modal.Footer>
529
+ </Modal>
93
530
 
94
- <Tab.Container activeKey={route} onSelect={(k) => {
95
- if (k) {
96
- setRoute(k);
97
- navigate(`#${k}`, { replace: true });
98
- }
99
- }}>
100
- <Tab.Content className="mt-3">
101
- <Tab.Pane eventKey="results">
102
- {!testsExist ? (
103
- <Alert variant="danger" className="mt-3">
104
- <h4>Tests did not run to completion</h4>
105
- <p>The test results file (tests.json) was not found or could not be loaded.</p>
106
- <div className="mt-3">
107
- <Button
108
- variant="outline-light"
109
- onClick={() => setRoute('logs')}
110
- className="me-2"
111
- >
112
- View Runtime Logs
113
- </Button>
114
- <Button
115
- variant="outline-light"
116
- onClick={() => navigate(`/projects/${projectName}#${runtime}`)}
117
- >
118
- View Build Logs
119
- </Button>
120
- </div>
121
- </Alert>
122
- ) : testData ? (
123
- <div className="test-results">
124
- {testData.givens.map((given, i) => (
125
- <div key={i} className="mb-4 card">
126
- <div className="card-header bg-primary text-white">
127
- <div className="d-flex justify-content-between align-items-center">
128
- <div>
129
- <h4>Given: {given.name}</h4>
130
- {given.features?.length > 0 && (
131
- <div className="mt-1">
132
- <small>Features:</small>
133
- <ul className="list-unstyled">
134
- {given.features.map((feature, fi) => (
135
- <li key={fi}>
136
- {feature.startsWith('http') ? (
137
- <a href={feature} target="_blank" rel="noopener noreferrer" className="text-white">
138
- {new URL(feature).hostname}
139
- </a>
140
- ) : (
141
- <span className="text-white">{feature}</span>
142
- )}
143
- </li>
144
- ))}
145
- </ul>
146
- </div>
147
- )}
148
- </div>
149
- {given.artifacts?.length > 0 && (
150
- <div className="dropdown">
151
- <button
152
- className="btn btn-sm btn-light dropdown-toggle"
153
- type="button"
154
- data-bs-toggle="dropdown"
155
- >
156
- Artifacts ({given.artifacts.length})
157
- </button>
158
- <ul className="dropdown-menu dropdown-menu-end">
159
- {given.artifacts.map((artifact, ai) => (
160
- <li key={ai}>
161
- <a
162
- className="dropdown-item"
163
- href={`/testeranto/reports/${projectName}/${testName.split('.').slice(0, -1).join('.')}/${runtime}/${artifact}`}
164
- target="_blank"
165
- rel="noopener noreferrer"
166
- >
167
- {artifact.split('/').pop()}
168
- </a>
169
- </li>
170
- ))}
171
- </ul>
172
- </div>
173
- )}
174
- </div>
175
- </div>
176
- <div className="card-body">
177
- {given.whens.map((when, j) => (
178
- <div key={`w-${j}`} className={`p-3 mb-2 ${when.error ? 'bg-danger text-white' : 'bg-success text-white'}`}>
179
- <div className="d-flex justify-content-between align-items-start">
180
- <div>
181
- <div>
182
- <strong>When:</strong> {when.name}
183
- {when.features?.length > 0 && (
184
- <div className="mt-2">
185
- <small>Features:</small>
186
- <ul className="list-unstyled">
187
- {when.features.map((feature, fi) => (
188
- <li key={fi}>
189
- {feature.startsWith('http') ? (
190
- <a href={feature} target="_blank" rel="noopener noreferrer">
191
- {new URL(feature).hostname}
192
- </a>
193
- ) : (
194
- feature
195
- )}
196
- </li>
197
- ))}
198
- </ul>
199
- </div>
200
- )}
201
- {when.error && <pre className="mt-2">{when.error}</pre>}
202
- </div>
203
- </div>
204
- {when.artifacts?.length > 0 && (
205
- <div className="ms-3">
206
- <strong>Artifacts:</strong>
207
- <ul className="list-unstyled">
208
- {when.artifacts.map((artifact, ai) => (
209
- <li key={ai}>
210
- <a
211
- href={`/testeranto/reports/${projectName}/${testName.split('.').slice(0, -1).join('.')}/${runtime}/${artifact}`}
212
- target="_blank"
213
- className="text-white"
214
- rel="noopener noreferrer"
215
- >
216
- {artifact.split('/').pop()}
217
- </a>
218
- </li>
219
- ))}
220
- </ul>
221
- </div>
222
- )}
223
- </div>
224
- </div>
225
- ))}
226
- {given.thens.map((then, k) => (
227
- <div key={`t-${k}`} className={`p-3 mb-2 ${then.error ? 'bg-danger text-white' : 'bg-success text-white'}`}>
228
- <div className="d-flex justify-content-between align-items-start">
229
- <div>
230
- <div>
231
- <strong>Then:</strong> {then.name}
232
- {then.features?.length > 0 && (
233
- <div className="mt-2">
234
- <small>Features:</small>
235
- <ul className="list-unstyled">
236
- {then.features.map((feature, fi) => (
237
- <li key={fi}>
238
- {feature.startsWith('http') ? (
239
- <a href={feature} target="_blank" rel="noopener noreferrer">
240
- {new URL(feature).hostname}
241
- </a>
242
- ) : (
243
- feature
244
- )}
245
- </li>
246
- ))}
247
- </ul>
248
- </div>
249
- )}
250
- {then.error && <pre className="mt-2">{then.error}</pre>}
251
- </div>
252
- </div>
253
- {then.artifacts?.length > 0 && (
254
- <div className="ms-3">
255
- <strong>Artifacts:</strong>
256
- <ul className="list-unstyled">
257
- {then.artifacts.map((artifact, ai) => (
258
- <li key={ai}>
259
- <a
260
- href={`/testeranto/reports/${projectName}/${testName.split('.').slice(0, -1).join('.')}/${runtime}/${artifact}`}
261
- target="_blank"
262
- className="text-white"
263
- rel="noopener noreferrer"
264
- >
265
- {artifact.split('/').pop()}
266
- </a>
267
- </li>
268
- ))}
269
- </ul>
270
- </div>
271
- )}
272
- </div>
273
- </div>
274
- ))}
275
- </div>
276
- </div>
277
- ))}
531
+ <Row className="g-0">
532
+ <Col sm={3} className="border-end" style={{
533
+ height: "calc(100vh - 56px)",
534
+ overflow: "auto",
535
+ backgroundColor: '#f8f9fa'
536
+ }}>
537
+ {/* File Tree Header */}
538
+ <div className="p-2 border-bottom">
539
+ <small className="fw-bold text-muted">EXPLORER</small>
540
+ </div>
541
+
542
+ {/* Standard Logs Section */}
543
+ <div className="p-2">
544
+ <div
545
+ className="d-flex align-items-center text-muted mb-1"
546
+ style={{ cursor: 'pointer', fontSize: '0.875rem' }}
547
+ onClick={() => setExpandedSections(prev => ({ ...prev, standardLogs: !prev.standardLogs }))}
548
+ >
549
+ <i className={`bi bi-chevron-${expandedSections.standardLogs ? 'down' : 'right'} me-1`}></i>
550
+ <span>Standard Logs</span>
551
+ </div>
552
+ {expandedSections.standardLogs && (
553
+ <div>
554
+ {Object.values(STANDARD_LOGS).map((logName) => {
555
+ const logContent = logs[logName];
556
+ const exists = logContent !== undefined &&
557
+ ((typeof logContent === "string" && logContent.trim() !== "") ||
558
+ (typeof logContent === "object" && Object.keys(logContent).length > 0));
559
+
560
+ return (
561
+ <FileTreeItem
562
+ key={logName}
563
+ name={logName}
564
+ isFile={true}
565
+ level={1}
566
+ isSelected={activeTab === logName}
567
+ exists={exists}
568
+ onClick={() => {
569
+ if (exists) {
570
+ setActiveTab(logName);
571
+ setSelectedFile({
572
+ path: logName,
573
+ content: typeof logContent === "string" ? logContent : JSON.stringify(logContent, null, 2),
574
+ language: logName.endsWith(".json") ? "json" : "plaintext",
575
+ });
576
+ } else {
577
+ setActiveTab(logName);
578
+ setSelectedFile({
579
+ path: logName,
580
+ content: `// ${logName} not found or empty\nThis file was not generated during the test run.`,
581
+ language: "plaintext",
582
+ });
583
+ }
584
+ }}
585
+ />
586
+ );
587
+ })}
278
588
  </div>
279
- ) : (
280
- <Alert variant="warning">No test results found</Alert>
281
589
  )}
282
- </Tab.Pane>
283
- <Tab.Pane eventKey="logs">
284
- {logs === undefined ? (
285
- <Alert variant="danger">
286
- <h4>Logs file missing</h4>
287
- <p>The runtime logs file (logs.txt) was not found.</p>
288
- <p>This suggests the test may not have executed properly.</p>
289
- </Alert>
290
- ) : logs === '' ? (
291
- <Alert variant="success">
292
- <h4>No runtime logs</h4>
293
- <p>The test executed successfully with no log output.</p>
294
- </Alert>
295
- ) : (
296
- <pre className="bg-dark text-white p-3">{logs}</pre>
590
+ </div>
591
+
592
+ {/* Runtime Logs Section */}
593
+ {Object.values(RUNTIME_SPECIFIC_LOGS[runtime as RuntimeName]).length > 0 && (
594
+ <div className="p-2">
595
+ <div
596
+ className="d-flex align-items-center text-muted mb-1"
597
+ style={{ cursor: 'pointer', fontSize: '0.875rem' }}
598
+ onClick={() => setExpandedSections(prev => ({ ...prev, runtimeLogs: !prev.runtimeLogs }))}
599
+ >
600
+ <i className={`bi bi-chevron-${expandedSections.runtimeLogs ? 'down' : 'right'} me-1`}></i>
601
+ <span>Runtime Logs</span>
602
+ </div>
603
+ {expandedSections.runtimeLogs && (
604
+ <div>
605
+ {Object.values(RUNTIME_SPECIFIC_LOGS[runtime as RuntimeName]).map((logName) => {
606
+ const logContent = logs[logName];
607
+ const exists = logContent !== undefined &&
608
+ ((typeof logContent === "string" && logContent.trim() !== "") ||
609
+ (typeof logContent === "object" && Object.keys(logContent).length > 0));
610
+
611
+ return (
612
+ <FileTreeItem
613
+ key={logName}
614
+ name={logName}
615
+ isFile={true}
616
+ level={1}
617
+ isSelected={activeTab === logName}
618
+ exists={exists}
619
+ onClick={() => {
620
+ if (exists) {
621
+ setActiveTab(logName);
622
+ setSelectedFile({
623
+ path: logName,
624
+ content: typeof logContent === "string" ? logContent : JSON.stringify(logContent, null, 2),
625
+ language: logName.endsWith(".json") ? "json" : "plaintext",
626
+ });
627
+ } else {
628
+ setActiveTab(logName);
629
+ setSelectedFile({
630
+ path: logName,
631
+ content: `// ${logName} not found or empty\nThis file was not generated during the test run.`,
632
+ language: "plaintext",
633
+ });
634
+ }
635
+ }}
636
+ />
637
+ );
638
+ })}
639
+ </div>
640
+ )}
641
+ </div>
642
+ )}
643
+
644
+ {/* Source Files Section */}
645
+ {logs.source_files && (
646
+ <div className="p-2">
647
+ <div
648
+ className="d-flex align-items-center text-muted mb-1"
649
+ style={{ cursor: 'pointer', fontSize: '0.875rem' }}
650
+ onClick={() => setExpandedSections(prev => ({ ...prev, sourceFiles: !prev.sourceFiles }))}
651
+ >
652
+ <i className={`bi bi-chevron-${expandedSections.sourceFiles ? 'down' : 'right'} me-1`}></i>
653
+ <span>Source Files</span>
654
+ </div>
655
+ {expandedSections.sourceFiles && (
656
+ <div>
657
+ <FileTree
658
+ data={logs.source_files}
659
+ onSelect={(path, content) => {
660
+ setActiveTab("source_file");
661
+ setSelectedSourcePath(path);
662
+ setSelectedFile({
663
+ path,
664
+ content,
665
+ language: getLanguage(path),
666
+ });
667
+ }}
668
+ level={1}
669
+ selectedSourcePath={selectedSourcePath}
670
+ />
671
+ </div>
672
+ )}
673
+ </div>
674
+ )}
675
+ </Col>
676
+ <Col
677
+ sm={6}
678
+ className="border-end p-0"
679
+ style={{ height: "calc(100vh - 56px)", overflow: "hidden" }}
680
+ >
681
+ <Editor
682
+ height="100%"
683
+ path={selectedFile?.path || "empty"}
684
+ defaultLanguage={selectedFile?.language || "plaintext"}
685
+ value={
686
+ selectedFile?.content || "// Select a file to view its contents"
687
+ }
688
+ theme={editorTheme}
689
+ options={{
690
+ minimap: { enabled: false },
691
+ fontSize: 14,
692
+ wordWrap: "on",
693
+ automaticLayout: true,
694
+ readOnly: !selectedFile?.path.includes("source_files"),
695
+ }}
696
+ />
697
+ </Col>
698
+ <Col sm={3} className="p-0 border-start" style={{ height: "calc(100vh - 56px)", overflow: "auto" }}>
699
+ <div className="p-3">
700
+ {selectedFile?.path.endsWith("tests.json") && (
701
+ <div className="test-results-preview">
702
+ {typeof selectedFile.content === "string"
703
+ ? renderTestResults(JSON.parse(selectedFile.content))
704
+ : renderTestResults(selectedFile.content)
705
+ }
706
+ </div>
297
707
  )}
298
- </Tab.Pane>
299
- <Tab.Pane eventKey="types">
300
- {typeErrors ? (
301
- <pre className="bg-dark text-white p-3">{typeErrors}</pre>
302
- ) : (
303
- <Alert variant="warning">No type errors found</Alert>
708
+ {selectedFile?.path.match(/\.(png|jpg|jpeg|gif|svg)$/i) && (
709
+ <div className="text-center">
710
+ <img
711
+ src={selectedFile.content}
712
+ alt={selectedFile.path}
713
+ className="img-fluid"
714
+ style={{ maxHeight: '300px' }}
715
+ />
716
+ <div className="mt-2">
717
+ <a
718
+ href={selectedFile.content}
719
+ target="_blank"
720
+ rel="noopener noreferrer"
721
+ className="btn btn-sm btn-outline-primary"
722
+ >
723
+ Open Full Size
724
+ </a>
725
+ </div>
726
+ </div>
304
727
  )}
305
- </Tab.Pane>
306
- <Tab.Pane eventKey="lint">
307
- {lintErrors ? (
308
- <pre className="bg-dark text-white p-3">{lintErrors}</pre>
309
- ) : (
310
- <Alert variant="warning">No lint errors found</Alert>
728
+ {selectedFile?.path.endsWith(".json") && !selectedFile.path.endsWith("tests.json") && (
729
+ <pre className="bg-light p-2 small">
730
+ <code>{selectedFile.content}</code>
731
+ </pre>
311
732
  )}
312
- </Tab.Pane>
313
- <Tab.Pane eventKey="coverage">
314
- <div className="coverage-report">
315
- <Alert variant="info">
316
- Coverage reports coming soon!
317
- </Alert>
318
- <div className="coverage-stats">
319
- <div className="stat-card bg-success text-white">
320
- <h4>85%</h4>
321
- <p>Lines Covered</p>
322
- </div>
323
- <div className="stat-card bg-warning text-dark">
324
- <h4>72%</h4>
325
- <p>Branches Covered</p>
326
- </div>
327
- <div className="stat-card bg-info text-white">
328
- <h4>91%</h4>
329
- <p>Functions Covered</p>
733
+ {selectedFile?.path.includes("source_files") && (
734
+ <div>
735
+ <div className="mb-2 small text-muted">
736
+ <i className="bi bi-file-earmark-text me-1"></i>
737
+ {selectedFile.path.split('/').pop()}
330
738
  </div>
739
+ <Button
740
+ variant="outline-primary"
741
+ size="sm"
742
+ className="mb-2"
743
+ onClick={() => {
744
+ // TODO: Add save functionality
745
+ alert("Save functionality will be implemented here");
746
+ }}
747
+ >
748
+ Save Changes
749
+ </Button>
331
750
  </div>
332
- </div>
333
- </Tab.Pane>
334
- </Tab.Content>
335
- </Tab.Container>
751
+ )}
752
+ </div>
753
+ </Col>
754
+ </Row>
755
+
756
+ <ToastContainer position="top-end" className="p-3">
757
+ <Toast
758
+ show={showToast}
759
+ onClose={() => setShowToast(false)}
760
+ delay={3000}
761
+ autohide
762
+ bg={toastVariant}
763
+ >
764
+ <Toast.Header>
765
+ <strong className="me-auto">Command Status</strong>
766
+ </Toast.Header>
767
+ <Toast.Body className="text-white">
768
+ {toastMessage}
769
+ </Toast.Body>
770
+ </Toast>
771
+ </ToastContainer>
772
+
336
773
  </Container>
337
774
  );
338
775
  };
776
+ // Simple file tree item component
777
+ const FileTreeItem = ({
778
+ name,
779
+ isFile,
780
+ level,
781
+ isSelected,
782
+ exists = true,
783
+ onClick
784
+ }: {
785
+ name: string;
786
+ isFile: boolean;
787
+ level: number;
788
+ isSelected: boolean;
789
+ exists?: boolean;
790
+ onClick: () => void;
791
+ }) => {
792
+ const displayName = name
793
+ .replace(".json", "")
794
+ .replace(".txt", "")
795
+ .replace(".log", "")
796
+ .replace(/_/g, " ")
797
+ .replace(/^std/, "Standard ")
798
+ .replace(/^exit/, "Exit Code")
799
+ .split('/').pop();
800
+
801
+ return (
802
+ <div
803
+ className={`d-flex align-items-center py-1 ${isSelected ? 'text-primary fw-bold' : exists ? 'text-dark' : 'text-muted'}`}
804
+ style={{
805
+ paddingLeft: `${level * 16}px`,
806
+ cursor: exists ? 'pointer' : 'not-allowed',
807
+ fontSize: '0.875rem',
808
+ opacity: exists ? 1 : 0.6
809
+ }}
810
+ onClick={exists ? onClick : undefined}
811
+ title={exists ? undefined : "File not found or empty"}
812
+ >
813
+ <i className={`bi ${isFile ? (exists ? 'bi-file-earmark-text' : 'bi-file-earmark') : 'bi-folder'} me-1`}></i>
814
+ <span>{displayName}</span>
815
+ {!exists && (
816
+ <i className="bi bi-question-circle ms-1" title="File not found or empty"></i>
817
+ )}
818
+ </div>
819
+ );
820
+ };
821
+
822
+ const ArtifactTree = ({
823
+ treeData,
824
+ projectName,
825
+ testName,
826
+ runtime,
827
+ onSelect,
828
+ level = 0,
829
+ basePath = ''
830
+ }: {
831
+ treeData: Record<string, any>;
832
+ projectName: string;
833
+ testName: string;
834
+ runtime: string;
835
+ onSelect: (path: string) => void;
836
+ level?: number;
837
+ basePath?: string;
838
+ }) => {
839
+ const [expanded, setExpanded] = useState<Record<string, boolean>>({});
840
+
841
+ const toggleExpand = (path: string) => {
842
+ setExpanded(prev => ({ ...prev, [path]: !prev[path] }));
843
+ };
844
+
845
+ return (
846
+ <ul className="list-unstyled" style={{ paddingLeft: `${level * 16}px` }}>
847
+ {Object.entries(treeData).map(([name, node]) => {
848
+ const fullPath = basePath ? `${basePath}/${name}` : name;
849
+ const isExpanded = expanded[fullPath];
850
+
851
+ if (node.__isFile) {
852
+ return (
853
+ <li key={fullPath} className="py-1">
854
+ <a
855
+ href={`reports/${projectName}/${testName
856
+ .split('.')
857
+ .slice(0, -1)
858
+ .join('.')}/${runtime}/${node.path}`}
859
+ target="_blank"
860
+ rel="noopener noreferrer"
861
+ className="text-decoration-none"
862
+ onClick={(e) => {
863
+ e.preventDefault();
864
+ onSelect(node.path);
865
+ }}
866
+ >
867
+ <i className="bi bi-file-earmark-text me-2"></i>
868
+ {name}
869
+ </a>
870
+ </li>
871
+ );
872
+ } else {
873
+ return (
874
+ <li key={fullPath} className="py-1">
875
+ <div className="d-flex align-items-center">
876
+ <button
877
+ className="btn btn-link text-start p-0 text-decoration-none me-1"
878
+ onClick={() => toggleExpand(fullPath)}
879
+ >
880
+ <i
881
+ className={`bi ${isExpanded ? 'bi-folder2-open' : 'bi-folder'} me-2`}
882
+ ></i>
883
+ {name}
884
+ </button>
885
+ </div>
886
+ {isExpanded && (
887
+ <ArtifactTree
888
+ treeData={node}
889
+ projectName={projectName}
890
+ testName={testName}
891
+ runtime={runtime}
892
+ onSelect={onSelect}
893
+ level={level + 1}
894
+ basePath={fullPath}
895
+ />
896
+ )}
897
+ </li>
898
+ );
899
+ }
900
+ })}
901
+ </ul>
902
+ );
903
+ };
904
+
905
+ const buildArtifactTree = (testData: any) => {
906
+ const artifactPaths = new Set<string>();
907
+ testData.givens?.forEach((given: any) => {
908
+ given.artifacts?.forEach((artifact: string) => artifactPaths.add(artifact));
909
+ given.whens?.forEach((when: any) =>
910
+ when.artifacts?.forEach((artifact: string) => artifactPaths.add(artifact)));
911
+ given.thens?.forEach((then: any) =>
912
+ then.artifacts?.forEach((artifact: string) => artifactPaths.add(artifact)));
913
+ });
914
+
915
+ const sortedArtifacts = Array.from(artifactPaths).sort();
916
+
917
+ return sortedArtifacts.reduce((tree, artifactPath) => {
918
+ const parts = artifactPath.split('/');
919
+ let currentLevel = tree;
920
+
921
+ parts.forEach((part, i) => {
922
+ if (!currentLevel[part]) {
923
+ currentLevel[part] = i === parts.length - 1
924
+ ? { __isFile: true, path: artifactPath }
925
+ : {};
926
+ }
927
+ currentLevel = currentLevel[part];
928
+ });
929
+
930
+ return tree;
931
+ }, {} as Record<string, any>);
932
+ };
933
+
934
+ const LogNavItem = ({
935
+ logName,
936
+ logContent,
937
+ activeTab,
938
+ setActiveTab,
939
+ setSelectedFile,
940
+ errorCounts,
941
+ decodedTestPath,
942
+ testsExist,
943
+ }: {
944
+ logName: string;
945
+ logContent: any;
946
+ activeTab: string;
947
+ setActiveTab: (tab: string) => void;
948
+ setSelectedFile: (file: {
949
+ path: string;
950
+ content: string;
951
+ language?: string;
952
+ }) => void;
953
+ errorCounts: {
954
+ runTimeErrors: number;
955
+ typeErrors: number;
956
+ staticErrors: number;
957
+ };
958
+ decodedTestPath: string;
959
+ testsExist: boolean;
960
+ }) => {
961
+ const displayName = logName
962
+ .replace(".json", "")
963
+ .replace(".txt", "")
964
+ .replace(".log", "")
965
+ .replace(/_/g, " ")
966
+ .replace(/^std/, "Standard ")
967
+ .replace(/^exit/, "Exit Code");
968
+
969
+ const logValue = typeof logContent === "string" ? logContent.trim() : "";
970
+ let statusIndicator: ReactElement<any, any> | null = null;
971
+
972
+ if (
973
+ logName === STANDARD_LOGS.TYPE_ERRORS &&
974
+ (errorCounts.typeErrors > 0 || (logValue && logValue !== "0"))
975
+ ) {
976
+ statusIndicator = (
977
+ <span className="ms-1">
978
+ ❌ {errorCounts.typeErrors || (logValue ? 1 : 0)}
979
+ </span>
980
+ );
981
+ } else if (
982
+ logName === STANDARD_LOGS.LINT_ERRORS &&
983
+ (errorCounts.staticErrors > 0 || (logValue && logValue !== "0"))
984
+ ) {
985
+ statusIndicator = (
986
+ <span className="ms-1">
987
+ ❌ {errorCounts.staticErrors || (logValue ? 1 : 0)}
988
+ </span>
989
+ );
990
+ } else if (
991
+ logName === RUNTIME_SPECIFIC_LOGS.node.STDERR &&
992
+ (errorCounts.runTimeErrors > 0 || (logValue && logValue !== "0"))
993
+ ) {
994
+ statusIndicator = (
995
+ <span className="ms-1">
996
+ ❌ {errorCounts.runTimeErrors || (logValue ? 1 : 0)}
997
+ </span>
998
+ );
999
+ } else if (logName === STANDARD_LOGS.EXIT && logValue !== "0") {
1000
+ statusIndicator = <span className="ms-1">⚠️ {logValue}</span>;
1001
+ } else if (logName === STANDARD_LOGS.TESTS && logValue) {
1002
+ statusIndicator = (
1003
+ <div className="ms-1">
1004
+ <TestStatusBadge
1005
+ testName={decodedTestPath}
1006
+ testsExist={testsExist}
1007
+ runTimeErrors={errorCounts.runTimeErrors}
1008
+ typeErrors={errorCounts.typeErrors}
1009
+ staticErrors={errorCounts.staticErrors}
1010
+ variant="compact"
1011
+ className="mt-1"
1012
+ />
1013
+ </div>
1014
+ );
1015
+ }
1016
+
1017
+ return (
1018
+ <Nav.Item key={logName}>
1019
+ <Nav.Link
1020
+ eventKey={logName}
1021
+ active={activeTab === logName}
1022
+ onClick={() => {
1023
+ setActiveTab(logName);
1024
+ setSelectedFile({
1025
+ path: logName,
1026
+ content:
1027
+ typeof logContent === "string"
1028
+ ? logContent
1029
+ : JSON.stringify(logContent, null, 2),
1030
+ language: logName.endsWith(".json")
1031
+ ? "json"
1032
+ : logName.endsWith(".txt")
1033
+ ? "plaintext"
1034
+ : logName.endsWith(".log")
1035
+ ? "log"
1036
+ : "plaintext",
1037
+ });
1038
+ }}
1039
+ className="d-flex flex-column align-items-start"
1040
+ >
1041
+ <div className="d-flex justify-content-between w-100">
1042
+ <span className="text-capitalize">{displayName}</span>
1043
+ {statusIndicator}
1044
+ </div>
1045
+ </Nav.Link>
1046
+ </Nav.Item>
1047
+ );
1048
+ };