@solidxai/core 0.1.1 → 0.1.4

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 (295) hide show
  1. package/dist/commands/run-tests.command.d.ts +37 -0
  2. package/dist/commands/run-tests.command.d.ts.map +1 -0
  3. package/dist/commands/run-tests.command.js +345 -0
  4. package/dist/commands/run-tests.command.js.map +1 -0
  5. package/dist/commands/test-data.command.d.ts +6 -6
  6. package/dist/commands/test-data.command.d.ts.map +1 -1
  7. package/dist/commands/test-data.command.js +25 -25
  8. package/dist/commands/test-data.command.js.map +1 -1
  9. package/dist/commands/test.command.d.ts +5 -0
  10. package/dist/commands/test.command.d.ts.map +1 -0
  11. package/dist/commands/test.command.js +26 -0
  12. package/dist/commands/test.command.js.map +1 -0
  13. package/dist/controllers/service.controller.d.ts +0 -9
  14. package/dist/controllers/service.controller.d.ts.map +1 -1
  15. package/dist/controllers/service.controller.js +0 -45
  16. package/dist/controllers/service.controller.js.map +1 -1
  17. package/dist/dtos/basic-filters.dto.d.ts.map +1 -1
  18. package/dist/dtos/basic-filters.dto.js.map +1 -1
  19. package/dist/dtos/create-user.dto.d.ts +1 -0
  20. package/dist/dtos/create-user.dto.d.ts.map +1 -1
  21. package/dist/dtos/create-user.dto.js +2 -1
  22. package/dist/dtos/create-user.dto.js.map +1 -1
  23. package/dist/helpers/schematic.service.js +1 -1
  24. package/dist/helpers/schematic.service.js.map +1 -1
  25. package/dist/index.d.ts +3 -0
  26. package/dist/index.d.ts.map +1 -1
  27. package/dist/index.js +3 -0
  28. package/dist/index.js.map +1 -1
  29. package/dist/seeders/module-metadata-seeder.service.d.ts.map +1 -1
  30. package/dist/seeders/module-metadata-seeder.service.js +3 -21
  31. package/dist/seeders/module-metadata-seeder.service.js.map +1 -1
  32. package/dist/seeders/module-test-data.service.d.ts.map +1 -1
  33. package/dist/seeders/module-test-data.service.js +3 -3
  34. package/dist/seeders/module-test-data.service.js.map +1 -1
  35. package/dist/seeders/seed-data/solid-core-metadata.json +34 -9
  36. package/dist/services/chatter-message.service.d.ts +2 -0
  37. package/dist/services/chatter-message.service.d.ts.map +1 -1
  38. package/dist/services/chatter-message.service.js +18 -2
  39. package/dist/services/chatter-message.service.js.map +1 -1
  40. package/dist/services/crud.service.d.ts.map +1 -1
  41. package/dist/services/crud.service.js.map +1 -1
  42. package/dist/services/model-metadata.service.d.ts.map +1 -1
  43. package/dist/services/model-metadata.service.js +2 -1
  44. package/dist/services/model-metadata.service.js.map +1 -1
  45. package/dist/services/module-metadata.service.d.ts.map +1 -1
  46. package/dist/services/module-metadata.service.js +2 -1
  47. package/dist/services/module-metadata.service.js.map +1 -1
  48. package/dist/services/queues/common.d.ts +3 -0
  49. package/dist/services/queues/common.d.ts.map +1 -0
  50. package/dist/services/queues/common.js +39 -0
  51. package/dist/services/queues/common.js.map +1 -0
  52. package/dist/services/queues/database-publisher.service.d.ts.map +1 -1
  53. package/dist/services/queues/database-publisher.service.js +3 -1
  54. package/dist/services/queues/database-publisher.service.js.map +1 -1
  55. package/dist/services/queues/database-subscriber.service.d.ts.map +1 -1
  56. package/dist/services/queues/database-subscriber.service.js +5 -2
  57. package/dist/services/queues/database-subscriber.service.js.map +1 -1
  58. package/dist/services/queues/rabbitmq-publisher.service.d.ts.map +1 -1
  59. package/dist/services/queues/rabbitmq-publisher.service.js +13 -6
  60. package/dist/services/queues/rabbitmq-publisher.service.js.map +1 -1
  61. package/dist/services/queues/rabbitmq-subscriber.service.d.ts +14 -1
  62. package/dist/services/queues/rabbitmq-subscriber.service.d.ts.map +1 -1
  63. package/dist/services/queues/rabbitmq-subscriber.service.js +197 -65
  64. package/dist/services/queues/rabbitmq-subscriber.service.js.map +1 -1
  65. package/dist/solid-core.module.d.ts.map +1 -1
  66. package/dist/solid-core.module.js +4 -0
  67. package/dist/solid-core.module.js.map +1 -1
  68. package/dist/testing/__examples__/register-example-specs.d.ts +3 -0
  69. package/dist/testing/__examples__/register-example-specs.d.ts.map +1 -0
  70. package/dist/testing/__examples__/register-example-specs.js +8 -0
  71. package/dist/testing/__examples__/register-example-specs.js.map +1 -0
  72. package/dist/testing/__examples__/specs/custom-health.spec.d.ts +17 -0
  73. package/dist/testing/__examples__/specs/custom-health.spec.d.ts.map +1 -0
  74. package/dist/testing/__examples__/specs/custom-health.spec.js +30 -0
  75. package/dist/testing/__examples__/specs/custom-health.spec.js.map +1 -0
  76. package/dist/testing/adapters/api/api-adapter.d.ts +9 -0
  77. package/dist/testing/adapters/api/api-adapter.d.ts.map +1 -0
  78. package/dist/testing/adapters/api/api-adapter.js +76 -0
  79. package/dist/testing/adapters/api/api-adapter.js.map +1 -0
  80. package/dist/testing/adapters/api/api.types.d.ts +14 -0
  81. package/dist/testing/adapters/api/api.types.d.ts.map +1 -0
  82. package/dist/testing/adapters/api/api.types.js +3 -0
  83. package/dist/testing/adapters/api/api.types.js.map +1 -0
  84. package/dist/testing/adapters/ui/playwright-adapter.d.ts +14 -0
  85. package/dist/testing/adapters/ui/playwright-adapter.d.ts.map +1 -0
  86. package/dist/testing/adapters/ui/playwright-adapter.js +47 -0
  87. package/dist/testing/adapters/ui/playwright-adapter.js.map +1 -0
  88. package/dist/testing/adapters/ui/ui.types.d.ts +5 -0
  89. package/dist/testing/adapters/ui/ui.types.d.ts.map +1 -0
  90. package/dist/testing/adapters/ui/ui.types.js +3 -0
  91. package/dist/testing/adapters/ui/ui.types.js.map +1 -0
  92. package/dist/testing/contracts/runtime-context.types.d.ts +35 -0
  93. package/dist/testing/contracts/runtime-context.types.d.ts.map +1 -0
  94. package/dist/testing/contracts/runtime-context.types.js +3 -0
  95. package/dist/testing/contracts/runtime-context.types.js.map +1 -0
  96. package/dist/testing/contracts/test-spec.types.d.ts +21 -0
  97. package/dist/testing/contracts/test-spec.types.d.ts.map +1 -0
  98. package/dist/testing/contracts/test-spec.types.js +3 -0
  99. package/dist/testing/contracts/test-spec.types.js.map +1 -0
  100. package/dist/testing/contracts/testing-metadata.types.d.ts +41 -0
  101. package/dist/testing/contracts/testing-metadata.types.d.ts.map +1 -0
  102. package/dist/testing/contracts/testing-metadata.types.js +3 -0
  103. package/dist/testing/contracts/testing-metadata.types.js.map +1 -0
  104. package/dist/testing/core/interpolation.d.ts +4 -0
  105. package/dist/testing/core/interpolation.d.ts.map +1 -0
  106. package/dist/testing/core/interpolation.js +180 -0
  107. package/dist/testing/core/interpolation.js.map +1 -0
  108. package/dist/testing/core/normalize-steps.d.ts +7 -0
  109. package/dist/testing/core/normalize-steps.d.ts.map +1 -0
  110. package/dist/testing/core/normalize-steps.js +20 -0
  111. package/dist/testing/core/normalize-steps.js.map +1 -0
  112. package/dist/testing/core/resource-store.d.ts +8 -0
  113. package/dist/testing/core/resource-store.d.ts.map +1 -0
  114. package/dist/testing/core/resource-store.js +41 -0
  115. package/dist/testing/core/resource-store.js.map +1 -0
  116. package/dist/testing/core/spec-registry.d.ts +10 -0
  117. package/dist/testing/core/spec-registry.d.ts.map +1 -0
  118. package/dist/testing/core/spec-registry.js +32 -0
  119. package/dist/testing/core/spec-registry.js.map +1 -0
  120. package/dist/testing/core/step-registry.d.ts +10 -0
  121. package/dist/testing/core/step-registry.d.ts.map +1 -0
  122. package/dist/testing/core/step-registry.js +26 -0
  123. package/dist/testing/core/step-registry.js.map +1 -0
  124. package/dist/testing/core/testing-engine.d.ts +14 -0
  125. package/dist/testing/core/testing-engine.d.ts.map +1 -0
  126. package/dist/testing/core/testing-engine.js +97 -0
  127. package/dist/testing/core/testing-engine.js.map +1 -0
  128. package/dist/testing/core/timeout.d.ts +2 -0
  129. package/dist/testing/core/timeout.d.ts.map +1 -0
  130. package/dist/testing/core/timeout.js +18 -0
  131. package/dist/testing/core/timeout.js.map +1 -0
  132. package/dist/testing/reporter/attachments.d.ts +4 -0
  133. package/dist/testing/reporter/attachments.d.ts.map +1 -0
  134. package/dist/testing/reporter/attachments.js +25 -0
  135. package/dist/testing/reporter/attachments.js.map +1 -0
  136. package/dist/testing/reporter/console-reporter.d.ts +45 -0
  137. package/dist/testing/reporter/console-reporter.d.ts.map +1 -0
  138. package/dist/testing/reporter/console-reporter.js +189 -0
  139. package/dist/testing/reporter/console-reporter.js.map +1 -0
  140. package/dist/testing/reporter/reporter.types.d.ts +37 -0
  141. package/dist/testing/reporter/reporter.types.d.ts.map +1 -0
  142. package/dist/testing/reporter/reporter.types.js +3 -0
  143. package/dist/testing/reporter/reporter.types.js.map +1 -0
  144. package/dist/testing/runner/lifecycle.d.ts +9 -0
  145. package/dist/testing/runner/lifecycle.d.ts.map +1 -0
  146. package/dist/testing/runner/lifecycle.js +33 -0
  147. package/dist/testing/runner/lifecycle.js.map +1 -0
  148. package/dist/testing/runner/run-from-metadata.d.ts +24 -0
  149. package/dist/testing/runner/run-from-metadata.d.ts.map +1 -0
  150. package/dist/testing/runner/run-from-metadata.js +70 -0
  151. package/dist/testing/runner/run-from-metadata.js.map +1 -0
  152. package/dist/testing/runner/scenario-filter.d.ts +9 -0
  153. package/dist/testing/runner/scenario-filter.d.ts.map +1 -0
  154. package/dist/testing/runner/scenario-filter.js +22 -0
  155. package/dist/testing/runner/scenario-filter.js.map +1 -0
  156. package/dist/testing/steps/api/auth.step.d.ts +3 -0
  157. package/dist/testing/steps/api/auth.step.d.ts.map +1 -0
  158. package/dist/testing/steps/api/auth.step.js +38 -0
  159. package/dist/testing/steps/api/auth.step.js.map +1 -0
  160. package/dist/testing/steps/api/index.d.ts +3 -0
  161. package/dist/testing/steps/api/index.d.ts.map +1 -0
  162. package/dist/testing/steps/api/index.js +10 -0
  163. package/dist/testing/steps/api/index.js.map +1 -0
  164. package/dist/testing/steps/api/request.step.d.ts +3 -0
  165. package/dist/testing/steps/api/request.step.d.ts.map +1 -0
  166. package/dist/testing/steps/api/request.step.js +281 -0
  167. package/dist/testing/steps/api/request.step.js.map +1 -0
  168. package/dist/testing/steps/assert/http.step.d.ts +3 -0
  169. package/dist/testing/steps/assert/http.step.d.ts.map +1 -0
  170. package/dist/testing/steps/assert/http.step.js +27 -0
  171. package/dist/testing/steps/assert/http.step.js.map +1 -0
  172. package/dist/testing/steps/assert/index.d.ts +3 -0
  173. package/dist/testing/steps/assert/index.d.ts.map +1 -0
  174. package/dist/testing/steps/assert/index.js +12 -0
  175. package/dist/testing/steps/assert/index.js.map +1 -0
  176. package/dist/testing/steps/assert/jsonpath.step.d.ts +3 -0
  177. package/dist/testing/steps/assert/jsonpath.step.d.ts.map +1 -0
  178. package/dist/testing/steps/assert/jsonpath.step.js +40 -0
  179. package/dist/testing/steps/assert/jsonpath.step.js.map +1 -0
  180. package/dist/testing/steps/assert/primitives.step.d.ts +3 -0
  181. package/dist/testing/steps/assert/primitives.step.d.ts.map +1 -0
  182. package/dist/testing/steps/assert/primitives.step.js +43 -0
  183. package/dist/testing/steps/assert/primitives.step.js.map +1 -0
  184. package/dist/testing/steps/test/index.d.ts +3 -0
  185. package/dist/testing/steps/test/index.d.ts.map +1 -0
  186. package/dist/testing/steps/test/index.js +8 -0
  187. package/dist/testing/steps/test/index.js.map +1 -0
  188. package/dist/testing/steps/test/test-spec.step.d.ts +3 -0
  189. package/dist/testing/steps/test/test-spec.step.d.ts.map +1 -0
  190. package/dist/testing/steps/test/test-spec.step.js +41 -0
  191. package/dist/testing/steps/test/test-spec.step.js.map +1 -0
  192. package/dist/testing/steps/ui/actions.step.d.ts +3 -0
  193. package/dist/testing/steps/ui/actions.step.d.ts.map +1 -0
  194. package/dist/testing/steps/ui/actions.step.js +31 -0
  195. package/dist/testing/steps/ui/actions.step.js.map +1 -0
  196. package/dist/testing/steps/ui/assertions.step.d.ts +3 -0
  197. package/dist/testing/steps/ui/assertions.step.d.ts.map +1 -0
  198. package/dist/testing/steps/ui/assertions.step.js +41 -0
  199. package/dist/testing/steps/ui/assertions.step.js.map +1 -0
  200. package/dist/testing/steps/ui/form.step.d.ts +3 -0
  201. package/dist/testing/steps/ui/form.step.d.ts.map +1 -0
  202. package/dist/testing/steps/ui/form.step.js +34 -0
  203. package/dist/testing/steps/ui/form.step.js.map +1 -0
  204. package/dist/testing/steps/ui/index.d.ts +3 -0
  205. package/dist/testing/steps/ui/index.d.ts.map +1 -0
  206. package/dist/testing/steps/ui/index.js +14 -0
  207. package/dist/testing/steps/ui/index.js.map +1 -0
  208. package/dist/testing/steps/ui/navigation.step.d.ts +3 -0
  209. package/dist/testing/steps/ui/navigation.step.d.ts.map +1 -0
  210. package/dist/testing/steps/ui/navigation.step.js +39 -0
  211. package/dist/testing/steps/ui/navigation.step.js.map +1 -0
  212. package/dist/testing/steps/util/index.d.ts +3 -0
  213. package/dist/testing/steps/util/index.d.ts.map +1 -0
  214. package/dist/testing/steps/util/index.js +12 -0
  215. package/dist/testing/steps/util/index.js.map +1 -0
  216. package/dist/testing/steps/util/log.step.d.ts +3 -0
  217. package/dist/testing/steps/util/log.step.d.ts.map +1 -0
  218. package/dist/testing/steps/util/log.step.js +18 -0
  219. package/dist/testing/steps/util/log.step.js.map +1 -0
  220. package/dist/testing/steps/util/require.step.d.ts +3 -0
  221. package/dist/testing/steps/util/require.step.d.ts.map +1 -0
  222. package/dist/testing/steps/util/require.step.js +16 -0
  223. package/dist/testing/steps/util/require.step.js.map +1 -0
  224. package/dist/testing/steps/util/sleep.step.d.ts +3 -0
  225. package/dist/testing/steps/util/sleep.step.d.ts.map +1 -0
  226. package/dist/testing/steps/util/sleep.step.js +13 -0
  227. package/dist/testing/steps/util/sleep.step.js.map +1 -0
  228. package/docs/test-data-workflow.md +51 -11
  229. package/package.json +4 -2
  230. package/src/commands/run-tests.command.ts +278 -0
  231. package/src/commands/test-data.command.ts +26 -26
  232. package/src/commands/test.command.ts +14 -0
  233. package/src/controllers/service.controller.ts +58 -59
  234. package/src/dtos/basic-filters.dto.ts +0 -2
  235. package/src/dtos/create-user.dto.ts +1 -0
  236. package/src/helpers/schematic.service.ts +1 -1
  237. package/src/index.ts +3 -0
  238. package/src/seeders/module-metadata-seeder.service.ts +5 -25
  239. package/src/seeders/module-test-data.service.ts +5 -3
  240. package/src/seeders/seed-data/solid-core-metadata.json +34 -9
  241. package/src/services/chatter-message.service.ts +18 -1
  242. package/src/services/crud.service.ts +1 -0
  243. package/src/services/model-metadata.service.ts +2 -1
  244. package/src/services/module-metadata.service.ts +2 -1
  245. package/src/services/queues/common.ts +75 -0
  246. package/src/services/queues/database-publisher.service.ts +4 -1
  247. package/src/services/queues/database-subscriber.service.ts +5 -3
  248. package/src/services/queues/rabbitmq-publisher.service.ts +17 -7
  249. package/src/services/queues/rabbitmq-subscriber.service.ts +223 -95
  250. package/src/solid-core.module.ts +4 -0
  251. package/src/testing/README.md +364 -0
  252. package/src/testing/__examples__/register-example-specs.ts +6 -0
  253. package/src/testing/__examples__/specs/custom-health.spec.ts +29 -0
  254. package/src/testing/__examples__/testing.sample.json +82 -0
  255. package/src/testing/adapters/api/api-adapter.ts +85 -0
  256. package/src/testing/adapters/api/api.types.ts +15 -0
  257. package/src/testing/adapters/ui/playwright-adapter.ts +54 -0
  258. package/src/testing/adapters/ui/ui.types.ts +4 -0
  259. package/src/testing/contracts/runtime-context.types.ts +36 -0
  260. package/src/testing/contracts/test-spec.types.ts +24 -0
  261. package/src/testing/contracts/testing-metadata.types.ts +46 -0
  262. package/src/testing/core/interpolation.ts +189 -0
  263. package/src/testing/core/normalize-steps.ts +21 -0
  264. package/src/testing/core/resource-store.ts +38 -0
  265. package/src/testing/core/spec-registry.ts +33 -0
  266. package/src/testing/core/step-registry.ts +27 -0
  267. package/src/testing/core/testing-engine.ts +127 -0
  268. package/src/testing/core/timeout.ts +19 -0
  269. package/src/testing/reporter/attachments.ts +25 -0
  270. package/src/testing/reporter/console-reporter.ts +229 -0
  271. package/src/testing/reporter/reporter.types.ts +36 -0
  272. package/src/testing/runner/lifecycle.ts +31 -0
  273. package/src/testing/runner/run-from-metadata.ts +87 -0
  274. package/src/testing/runner/scenario-filter.ts +33 -0
  275. package/src/testing/steps/api/auth.step.ts +66 -0
  276. package/src/testing/steps/api/index.ts +10 -0
  277. package/src/testing/steps/api/request.step.ts +358 -0
  278. package/src/testing/steps/assert/http.step.ts +33 -0
  279. package/src/testing/steps/assert/index.ts +12 -0
  280. package/src/testing/steps/assert/jsonpath.step.ts +50 -0
  281. package/src/testing/steps/assert/primitives.step.ts +69 -0
  282. package/src/testing/steps/test/index.ts +8 -0
  283. package/src/testing/steps/test/test-spec.step.ts +52 -0
  284. package/src/testing/steps/ui/actions.step.ts +36 -0
  285. package/src/testing/steps/ui/assertions.step.ts +54 -0
  286. package/src/testing/steps/ui/form.step.ts +39 -0
  287. package/src/testing/steps/ui/index.ts +12 -0
  288. package/src/testing/steps/ui/navigation.step.ts +53 -0
  289. package/src/testing/steps/util/index.ts +10 -0
  290. package/src/testing/steps/util/log.step.ts +19 -0
  291. package/src/testing/steps/util/require.step.ts +16 -0
  292. package/src/testing/steps/util/sleep.step.ts +15 -0
  293. package/tsconfig.json +35 -25
  294. package/tsconfig.tests.json +14 -0
  295. package/dist/tsconfig.tsbuildinfo +0 -1
@@ -0,0 +1,97 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TestingEngine = void 0;
4
+ const interpolation_1 = require("./interpolation");
5
+ const normalize_steps_1 = require("./normalize-steps");
6
+ const timeout_1 = require("./timeout");
7
+ class TestingEngine {
8
+ constructor(registry, defaults) {
9
+ this.registry = registry;
10
+ this.defaults = defaults;
11
+ }
12
+ async runScenario(scenario, ctxBase) {
13
+ const retries = scenario.retries ?? this.defaults?.retries ?? 0;
14
+ const maxAttempts = Math.max(0, retries) + 1;
15
+ const scenarioTimeoutMs = scenario.timeoutMs ?? this.defaults?.timeoutMs;
16
+ for (let attempt = 1; attempt <= maxAttempts; attempt += 1) {
17
+ const ctx = {
18
+ ...ctxBase,
19
+ scenarioId: scenario.id,
20
+ scenarioType: scenario.type,
21
+ params: scenario.params ?? {},
22
+ };
23
+ const reporter = ctx.reporter;
24
+ const scenarioStart = Date.now();
25
+ let scenarioError;
26
+ reporter.onScenarioStart(scenario);
27
+ try {
28
+ const execute = async () => {
29
+ for (const block of scenario.steps) {
30
+ const normalized = (0, normalize_steps_1.normalizeBlock)(block);
31
+ await this.runBlock(normalized.phase, normalized.steps, ctx);
32
+ }
33
+ };
34
+ if (scenarioTimeoutMs !== undefined) {
35
+ await (0, timeout_1.withTimeout)(execute(), scenarioTimeoutMs, `Scenario timeout after ${scenarioTimeoutMs}ms: "${scenario.id}"`);
36
+ }
37
+ else {
38
+ await execute();
39
+ }
40
+ }
41
+ catch (err) {
42
+ scenarioError = err;
43
+ }
44
+ finally {
45
+ const durationMs = Date.now() - scenarioStart;
46
+ reporter.onScenarioEnd(scenario, {
47
+ ok: !scenarioError,
48
+ error: scenarioError,
49
+ durationMs,
50
+ });
51
+ }
52
+ if (!scenarioError) {
53
+ return;
54
+ }
55
+ if (attempt >= maxAttempts) {
56
+ throw scenarioError;
57
+ }
58
+ }
59
+ }
60
+ async runBlock(phase, steps, ctx) {
61
+ for (const step of steps) {
62
+ const reporter = ctx.reporter;
63
+ const stepStart = Date.now();
64
+ let stepError;
65
+ let resolvedStep;
66
+ reporter.onStepStart({ scenarioId: ctx.scenarioId, phase, step });
67
+ try {
68
+ resolvedStep = (0, interpolation_1.interpolateDeep)(step, ctx);
69
+ const handler = this.registry.get(resolvedStep.op);
70
+ const run = handler(ctx, resolvedStep);
71
+ const result = resolvedStep.timeoutMs !== undefined
72
+ ? await (0, timeout_1.withTimeout)(run, resolvedStep.timeoutMs, `Step timeout after ${resolvedStep.timeoutMs}ms: "${resolvedStep.op}"`)
73
+ : await run;
74
+ if (resolvedStep.saveAs) {
75
+ ctx.resources.set(resolvedStep.saveAs, result);
76
+ }
77
+ }
78
+ catch (err) {
79
+ stepError = err;
80
+ throw err;
81
+ }
82
+ finally {
83
+ const durationMs = Date.now() - stepStart;
84
+ reporter.onStepEnd({
85
+ scenarioId: ctx.scenarioId,
86
+ phase,
87
+ step: resolvedStep ?? step,
88
+ ok: !stepError,
89
+ error: stepError,
90
+ durationMs,
91
+ });
92
+ }
93
+ }
94
+ }
95
+ }
96
+ exports.TestingEngine = TestingEngine;
97
+ //# sourceMappingURL=testing-engine.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testing-engine.js","sourceRoot":"","sources":["../../../src/testing/core/testing-engine.ts"],"names":[],"mappings":"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","sourcesContent":["import type { TestContext, StepPhase } from \"../contracts/runtime-context.types\";\nimport type { OpStep, ScenarioSpec } from \"../contracts/testing-metadata.types\";\nimport { interpolateDeep } from \"./interpolation\";\nimport { normalizeBlock } from \"./normalize-steps\";\nimport { StepRegistry } from \"./step-registry\";\nimport { withTimeout } from \"./timeout\";\n\nexport class TestingEngine {\n private readonly registry: StepRegistry;\n private readonly defaults?: { timeoutMs?: number; retries?: number };\n\n constructor(\n registry: StepRegistry,\n defaults?: { timeoutMs?: number; retries?: number },\n ) {\n this.registry = registry;\n this.defaults = defaults;\n }\n\n async runScenario(\n scenario: ScenarioSpec,\n ctxBase: Omit<TestContext, \"scenarioId\" | \"scenarioType\" | \"params\">,\n ): Promise<void> {\n const retries = scenario.retries ?? this.defaults?.retries ?? 0;\n const maxAttempts = Math.max(0, retries) + 1;\n const scenarioTimeoutMs =\n scenario.timeoutMs ?? this.defaults?.timeoutMs;\n\n for (let attempt = 1; attempt <= maxAttempts; attempt += 1) {\n const ctx: TestContext = {\n ...ctxBase,\n scenarioId: scenario.id,\n scenarioType: scenario.type,\n params: scenario.params ?? {},\n };\n\n const reporter = ctx.reporter;\n const scenarioStart = Date.now();\n let scenarioError: unknown;\n\n reporter.onScenarioStart(scenario);\n\n try {\n const execute = async () => {\n for (const block of scenario.steps) {\n const normalized = normalizeBlock(block);\n await this.runBlock(normalized.phase, normalized.steps, ctx);\n }\n };\n\n if (scenarioTimeoutMs !== undefined) {\n await withTimeout(\n execute(),\n scenarioTimeoutMs,\n `Scenario timeout after ${scenarioTimeoutMs}ms: \"${scenario.id}\"`,\n );\n } else {\n await execute();\n }\n } catch (err) {\n scenarioError = err;\n } finally {\n const durationMs = Date.now() - scenarioStart;\n reporter.onScenarioEnd(scenario, {\n ok: !scenarioError,\n error: scenarioError,\n durationMs,\n });\n }\n\n if (!scenarioError) {\n return;\n }\n\n if (attempt >= maxAttempts) {\n throw scenarioError;\n }\n }\n }\n\n private async runBlock(\n phase: StepPhase,\n steps: OpStep[],\n ctx: TestContext,\n ): Promise<void> {\n for (const step of steps) {\n const reporter = ctx.reporter;\n const stepStart = Date.now();\n let stepError: unknown;\n let resolvedStep: OpStep | undefined;\n\n reporter.onStepStart({ scenarioId: ctx.scenarioId, phase, step });\n\n try {\n resolvedStep = interpolateDeep(step, ctx) as OpStep;\n const handler = this.registry.get(resolvedStep.op);\n const run = handler(ctx, resolvedStep);\n const result =\n resolvedStep.timeoutMs !== undefined\n ? await withTimeout(\n run,\n resolvedStep.timeoutMs,\n `Step timeout after ${resolvedStep.timeoutMs}ms: \"${resolvedStep.op}\"`,\n )\n : await run;\n\n if (resolvedStep.saveAs) {\n // console.log(`Step ${resolvedStep.name} attempting to saveAs ${resolvedStep.saveAs}`, JSON.stringify(result));\n ctx.resources.set(resolvedStep.saveAs, result);\n }\n } catch (err) {\n stepError = err;\n throw err;\n } finally {\n const durationMs = Date.now() - stepStart;\n reporter.onStepEnd({\n scenarioId: ctx.scenarioId,\n phase,\n step: resolvedStep ?? step,\n ok: !stepError,\n error: stepError,\n durationMs,\n });\n }\n }\n }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export declare function withTimeout<T>(promise: Promise<T>, ms: number, message: string): Promise<T>;
2
+ //# sourceMappingURL=timeout.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeout.d.ts","sourceRoot":"","sources":["../../../src/testing/core/timeout.ts"],"names":[],"mappings":"AAAA,wBAAsB,WAAW,CAAC,CAAC,EACjC,OAAO,EAAE,OAAO,CAAC,CAAC,CAAC,EACnB,EAAE,EAAE,MAAM,EACV,OAAO,EAAE,MAAM,GACd,OAAO,CAAC,CAAC,CAAC,CAcZ"}
@@ -0,0 +1,18 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.withTimeout = withTimeout;
4
+ async function withTimeout(promise, ms, message) {
5
+ let timeoutId;
6
+ const timeoutPromise = new Promise((_resolve, reject) => {
7
+ timeoutId = setTimeout(() => reject(new Error(message)), ms);
8
+ });
9
+ try {
10
+ return await Promise.race([promise, timeoutPromise]);
11
+ }
12
+ finally {
13
+ if (timeoutId !== undefined) {
14
+ clearTimeout(timeoutId);
15
+ }
16
+ }
17
+ }
18
+ //# sourceMappingURL=timeout.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"timeout.js","sourceRoot":"","sources":["../../../src/testing/core/timeout.ts"],"names":[],"mappings":";;AAAA,kCAkBC;AAlBM,KAAK,UAAU,WAAW,CAC/B,OAAmB,EACnB,EAAU,EACV,OAAe;IAEf,IAAI,SAAoD,CAAC;IAEzD,MAAM,cAAc,GAAG,IAAI,OAAO,CAAQ,CAAC,QAAQ,EAAE,MAAM,EAAE,EAAE;QAC7D,SAAS,GAAG,UAAU,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IAC/D,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC;QACH,OAAO,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC,CAAC;IACvD,CAAC;YAAS,CAAC;QACT,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;YAC5B,YAAY,CAAC,SAAS,CAAC,CAAC;QAC1B,CAAC;IACH,CAAC;AACH,CAAC","sourcesContent":["export async function withTimeout<T>(\n promise: Promise<T>,\n ms: number,\n message: string,\n): Promise<T> {\n let timeoutId: ReturnType<typeof setTimeout> | undefined;\n\n const timeoutPromise = new Promise<never>((_resolve, reject) => {\n timeoutId = setTimeout(() => reject(new Error(message)), ms);\n });\n\n try {\n return await Promise.race([promise, timeoutPromise]);\n } finally {\n if (timeoutId !== undefined) {\n clearTimeout(timeoutId);\n }\n }\n}\n"]}
@@ -0,0 +1,4 @@
1
+ import type { TestContext } from "../contracts/runtime-context.types";
2
+ export declare function attachJson(ctx: TestContext, name: string, obj: unknown): void;
3
+ export declare function attachText(ctx: TestContext, name: string, text: string): void;
4
+ //# sourceMappingURL=attachments.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"attachments.d.ts","sourceRoot":"","sources":["../../../src/testing/reporter/attachments.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,oCAAoC,CAAC;AAEtE,wBAAgB,UAAU,CACxB,GAAG,EAAE,WAAW,EAChB,IAAI,EAAE,MAAM,EACZ,GAAG,EAAE,OAAO,GACX,IAAI,CAQN;AAED,wBAAgB,UAAU,CAAC,GAAG,EAAE,WAAW,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,IAAI,CAQ7E"}
@@ -0,0 +1,25 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.attachJson = attachJson;
4
+ exports.attachText = attachText;
5
+ function attachJson(ctx, name, obj) {
6
+ if (!ctx.reporter.attach)
7
+ return;
8
+ ctx.reporter.attach({
9
+ scenarioId: ctx.scenarioId,
10
+ name,
11
+ contentType: "application/json",
12
+ data: JSON.stringify(obj, null, 2),
13
+ });
14
+ }
15
+ function attachText(ctx, name, text) {
16
+ if (!ctx.reporter.attach)
17
+ return;
18
+ ctx.reporter.attach({
19
+ scenarioId: ctx.scenarioId,
20
+ name,
21
+ contentType: "text/plain",
22
+ data: text,
23
+ });
24
+ }
25
+ //# sourceMappingURL=attachments.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"attachments.js","sourceRoot":"","sources":["../../../src/testing/reporter/attachments.ts"],"names":[],"mappings":";;AAEA,gCAYC;AAED,gCAQC;AAtBD,SAAgB,UAAU,CACxB,GAAgB,EAChB,IAAY,EACZ,GAAY;IAEZ,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,MAAM;QAAE,OAAO;IACjC,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC;QAClB,UAAU,EAAE,GAAG,CAAC,UAAU;QAC1B,IAAI;QACJ,WAAW,EAAE,kBAAkB;QAC/B,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC,CAAC;KACnC,CAAC,CAAC;AACL,CAAC;AAED,SAAgB,UAAU,CAAC,GAAgB,EAAE,IAAY,EAAE,IAAY;IACrE,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,MAAM;QAAE,OAAO;IACjC,GAAG,CAAC,QAAQ,CAAC,MAAM,CAAC;QAClB,UAAU,EAAE,GAAG,CAAC,UAAU;QAC1B,IAAI;QACJ,WAAW,EAAE,YAAY;QACzB,IAAI,EAAE,IAAI;KACX,CAAC,CAAC;AACL,CAAC","sourcesContent":["import type { TestContext } from \"../contracts/runtime-context.types\";\n\nexport function attachJson(\n ctx: TestContext,\n name: string,\n obj: unknown,\n): void {\n if (!ctx.reporter.attach) return;\n ctx.reporter.attach({\n scenarioId: ctx.scenarioId,\n name,\n contentType: \"application/json\",\n data: JSON.stringify(obj, null, 2),\n });\n}\n\nexport function attachText(ctx: TestContext, name: string, text: string): void {\n if (!ctx.reporter.attach) return;\n ctx.reporter.attach({\n scenarioId: ctx.scenarioId,\n name,\n contentType: \"text/plain\",\n data: text,\n });\n}\n"]}
@@ -0,0 +1,45 @@
1
+ import type { Reporter } from "./reporter.types";
2
+ import type { OpStep } from "../contracts/testing-metadata.types";
3
+ export declare class ConsoleReporter implements Reporter {
4
+ onScenarioStart(scenario: {
5
+ id: string;
6
+ name?: string;
7
+ }): void;
8
+ onScenarioEnd(scenario: {
9
+ id: string;
10
+ name?: string;
11
+ }, result: {
12
+ ok: boolean;
13
+ error?: unknown;
14
+ durationMs: number;
15
+ }): void;
16
+ onStepStart(args: {
17
+ scenarioId: string;
18
+ phase: string;
19
+ step: OpStep;
20
+ }): void;
21
+ onStepEnd(args: {
22
+ scenarioId: string;
23
+ phase: string;
24
+ step: OpStep;
25
+ ok: boolean;
26
+ error?: unknown;
27
+ durationMs: number;
28
+ }): void;
29
+ onSpecResult(args: {
30
+ scenarioId: string;
31
+ specId: string;
32
+ stepName?: string;
33
+ result: {
34
+ ok: boolean;
35
+ details?: Record<string, any>;
36
+ };
37
+ }): void;
38
+ attach(args: {
39
+ scenarioId: string;
40
+ name: string;
41
+ contentType: string;
42
+ data: Buffer | string;
43
+ }): void;
44
+ }
45
+ //# sourceMappingURL=console-reporter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"console-reporter.d.ts","sourceRoot":"","sources":["../../../src/testing/reporter/console-reporter.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,kBAAkB,CAAC;AACjD,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,qCAAqC,CAAC;AAgJlE,qBAAa,eAAgB,YAAW,QAAQ;IAC9C,eAAe,CAAC,QAAQ,EAAE;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,IAAI;IAK9D,aAAa,CACX,QAAQ,EAAE;QAAE,EAAE,EAAE,MAAM,CAAC;QAAC,IAAI,CAAC,EAAE,MAAM,CAAA;KAAE,EACvC,MAAM,EAAE;QAAE,EAAE,EAAE,OAAO,CAAC;QAAC,KAAK,CAAC,EAAE,OAAO,CAAC;QAAC,UAAU,EAAE,MAAM,CAAA;KAAE,GAC3D,IAAI;IAMP,WAAW,CAAC,IAAI,EAAE;QAChB,UAAU,EAAE,MAAM,CAAC;QACnB,KAAK,EAAE,MAAM,CAAC;QACd,IAAI,EAAE,MAAM,CAAC;KACd,GAAG,IAAI;IAMR,SAAS,CAAC,IAAI,EAAE;QACd,UAAU,EAAE,MAAM,CAAC;QACnB,KAAK,EAAE,MAAM,CAAC;QACd,IAAI,EAAE,MAAM,CAAC;QACb,EAAE,EAAE,OAAO,CAAC;QACZ,KAAK,CAAC,EAAE,OAAO,CAAC;QAChB,UAAU,EAAE,MAAM,CAAC;KACpB,GAAG,IAAI;IAwBR,YAAY,CAAC,IAAI,EAAE;QACjB,UAAU,EAAE,MAAM,CAAC;QACnB,MAAM,EAAE,MAAM,CAAC;QACf,QAAQ,CAAC,EAAE,MAAM,CAAC;QAClB,MAAM,EAAE;YAAE,EAAE,EAAE,OAAO,CAAC;YAAC,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;SAAE,CAAC;KACxD,GAAG,IAAI;IAQR,MAAM,CAAC,IAAI,EAAE;QACX,UAAU,EAAE,MAAM,CAAC;QACnB,IAAI,EAAE,MAAM,CAAC;QACb,WAAW,EAAE,MAAM,CAAC;QACpB,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC;KACvB,GAAG,IAAI;CAOT"}
@@ -0,0 +1,189 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ConsoleReporter = void 0;
4
+ const INDENT = " ";
5
+ const STEP_INDENT = INDENT;
6
+ const MAX_DETAIL_LEN = 140;
7
+ function formatError(err) {
8
+ if (!err)
9
+ return "";
10
+ if (err instanceof Error) {
11
+ return err.stack || err.message;
12
+ }
13
+ return String(err);
14
+ }
15
+ function indentLines(text, indent) {
16
+ return text
17
+ .split("\n")
18
+ .map((line) => (line.length ? `${indent}${line}` : indent))
19
+ .join("\n");
20
+ }
21
+ function truncate(value, maxLen = MAX_DETAIL_LEN) {
22
+ if (value.length <= maxLen)
23
+ return value;
24
+ return `${value.slice(0, maxLen - 1)}…`;
25
+ }
26
+ function formatHttpBody(body) {
27
+ if (body === null || body === undefined)
28
+ return "";
29
+ if (typeof body === "string") {
30
+ const trimmed = body.trim();
31
+ if (trimmed.startsWith("{") || trimmed.startsWith("[")) {
32
+ try {
33
+ return JSON.stringify(JSON.parse(trimmed), null, 2);
34
+ }
35
+ catch {
36
+ return body;
37
+ }
38
+ }
39
+ return body;
40
+ }
41
+ try {
42
+ return JSON.stringify(body, null, 2);
43
+ }
44
+ catch {
45
+ return String(body);
46
+ }
47
+ }
48
+ function withoutQuery(url) {
49
+ const idx = url.indexOf("?");
50
+ return idx === -1 ? url : url.slice(0, idx);
51
+ }
52
+ function maskIfSensitive(value, hints) {
53
+ const lower = hints.join(" ").toLowerCase();
54
+ if (lower.includes("password") || lower.includes("pwd")) {
55
+ return "••••••";
56
+ }
57
+ return value;
58
+ }
59
+ function stepDetails(step) {
60
+ const withObj = (step.with ?? {});
61
+ switch (step.op) {
62
+ case "api.request": {
63
+ const method = String(withObj.method ?? "").toUpperCase();
64
+ const url = typeof withObj.url === "string" ? withoutQuery(withObj.url) : "";
65
+ if (!method && !url)
66
+ return undefined;
67
+ return `${method} ${url}`.trim();
68
+ }
69
+ case "api.auth.bearerFromLogin": {
70
+ const url = typeof withObj.url === "string" ? withoutQuery(withObj.url) : "";
71
+ return url ? `url: ${url}` : undefined;
72
+ }
73
+ case "ui.goto": {
74
+ return withObj.url ? `url: ${withObj.url}` : undefined;
75
+ }
76
+ case "ui.click":
77
+ case "ui.expectVisible": {
78
+ return withObj.selector ? `selector: ${withObj.selector}` : undefined;
79
+ }
80
+ case "ui.fill":
81
+ case "ui.select": {
82
+ const selector = withObj.selector ? `selector: ${withObj.selector}` : "";
83
+ const value = withObj.value !== undefined
84
+ ? `value: ${maskIfSensitive(String(withObj.value), [String(withObj.selector ?? ""), String(step.op)])}`
85
+ : "";
86
+ return [selector, value].filter(Boolean).join(", ") || undefined;
87
+ }
88
+ case "ui.press": {
89
+ const selector = withObj.selector ? `selector: ${withObj.selector}` : "";
90
+ const key = withObj.key ? `key: ${withObj.key}` : "";
91
+ return [selector, key].filter(Boolean).join(", ") || undefined;
92
+ }
93
+ case "ui.expectUrl": {
94
+ const equals = withObj.equals ? `equals: ${withObj.equals}` : "";
95
+ const contains = withObj.contains ? `contains: ${withObj.contains}` : "";
96
+ return [equals, contains].filter(Boolean).join(", ") || undefined;
97
+ }
98
+ case "ui.expectText": {
99
+ const selector = withObj.selector ? `selector: ${withObj.selector}` : "";
100
+ const equals = withObj.equals ? `equals: ${truncate(String(withObj.equals))}` : "";
101
+ const contains = withObj.contains ? `contains: ${truncate(String(withObj.contains))}` : "";
102
+ return [selector, equals, contains].filter(Boolean).join(", ") || undefined;
103
+ }
104
+ case "assert.httpStatus": {
105
+ return withObj.is !== undefined ? `status: ${withObj.is}` : undefined;
106
+ }
107
+ case "assert.jsonPath": {
108
+ const path = withObj.path ? `path: ${withObj.path}` : "";
109
+ const equals = withObj.equals !== undefined ? `equals: ${truncate(String(withObj.equals))}` : "";
110
+ return [path, equals].filter(Boolean).join(", ") || undefined;
111
+ }
112
+ case "assert.contains":
113
+ case "assert.equals":
114
+ case "assert.matches": {
115
+ const expected = withObj.expected !== undefined
116
+ ? `expected: ${truncate(String(withObj.expected))}`
117
+ : withObj.pattern
118
+ ? `pattern: ${truncate(String(withObj.pattern))}`
119
+ : withObj.equals !== undefined
120
+ ? `equals: ${truncate(String(withObj.equals))}`
121
+ : "";
122
+ return expected || undefined;
123
+ }
124
+ case "util.sleep": {
125
+ return withObj.ms !== undefined ? `ms: ${withObj.ms}` : undefined;
126
+ }
127
+ case "util.log": {
128
+ return withObj.message ? `message: ${truncate(String(withObj.message))}` : undefined;
129
+ }
130
+ default:
131
+ return undefined;
132
+ }
133
+ }
134
+ function formatStepLabel(step) {
135
+ const base = step.name ? `${step.op} (${step.name})` : step.op;
136
+ const details = stepDetails(step);
137
+ if (!details)
138
+ return base;
139
+ return `${base} (${details})`;
140
+ }
141
+ class ConsoleReporter {
142
+ onScenarioStart(scenario) {
143
+ const label = scenario.name ? `${scenario.id} (${scenario.name})` : scenario.id;
144
+ console.log(`\n▶ Scenario: ${label}`);
145
+ }
146
+ onScenarioEnd(scenario, result) {
147
+ const label = scenario.name ? `${scenario.id} (${scenario.name})` : scenario.id;
148
+ const status = result.ok ? "✔" : "✖";
149
+ console.log(`${status} Scenario: ${label} (${result.durationMs}ms)`);
150
+ }
151
+ onStepStart(args) {
152
+ const label = formatStepLabel(args.step);
153
+ const phase = args.phase.toUpperCase();
154
+ console.log(`${STEP_INDENT}↳ ${phase} ${label}`);
155
+ }
156
+ onStepEnd(args) {
157
+ const label = formatStepLabel(args.step);
158
+ const phase = args.phase.toUpperCase();
159
+ const status = args.ok ? "✔" : "✖";
160
+ console.log(`${STEP_INDENT}${status} ${phase} ${label} (${args.durationMs}ms)`);
161
+ if (!args.ok && args.error) {
162
+ const details = formatError(args.error);
163
+ console.error(indentLines(details, `${STEP_INDENT}${INDENT}`));
164
+ const httpBody = args.error?.httpResponseBody;
165
+ if (httpBody !== undefined) {
166
+ const formatted = formatHttpBody(httpBody);
167
+ const bodyText = formatted.length ? formatted : "<empty>";
168
+ console.error(indentLines(`HTTP Response Body:\n${bodyText}`, `${STEP_INDENT}${INDENT}`));
169
+ }
170
+ }
171
+ }
172
+ onSpecResult(args) {
173
+ console.log(`${INDENT}↳ SPEC ${args.specId} ok=${args.result.ok}`);
174
+ if (args.result.details) {
175
+ const details = JSON.stringify(args.result.details, null, 2);
176
+ console.log(indentLines(details, `${INDENT}${INDENT}`));
177
+ }
178
+ }
179
+ attach(args) {
180
+ const header = `ATTACH ${args.name} (${args.contentType})`;
181
+ console.log(indentLines(header, `${STEP_INDENT}${INDENT}`));
182
+ const dataText = Buffer.isBuffer(args.data) ? args.data.toString("utf8") : String(args.data ?? "");
183
+ if (!dataText.length)
184
+ return;
185
+ console.log(indentLines(dataText, `${STEP_INDENT}${INDENT}${INDENT}`));
186
+ }
187
+ }
188
+ exports.ConsoleReporter = ConsoleReporter;
189
+ //# sourceMappingURL=console-reporter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"console-reporter.js","sourceRoot":"","sources":["../../../src/testing/reporter/console-reporter.ts"],"names":[],"mappings":"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","sourcesContent":["// Purpose: Basic console reporter implementation.\n\nimport type { Reporter } from \"./reporter.types\";\nimport type { OpStep } from \"../contracts/testing-metadata.types\";\n\nconst INDENT = \" \";\nconst STEP_INDENT = INDENT;\nconst MAX_DETAIL_LEN = 140;\n\nfunction formatError(err: unknown): string {\n if (!err) return \"\";\n if (err instanceof Error) {\n return err.stack || err.message;\n }\n return String(err);\n}\n\nfunction indentLines(text: string, indent: string): string {\n return text\n .split(\"\\n\")\n .map((line) => (line.length ? `${indent}${line}` : indent))\n .join(\"\\n\");\n}\n\nfunction truncate(value: string, maxLen: number = MAX_DETAIL_LEN): string {\n if (value.length <= maxLen) return value;\n return `${value.slice(0, maxLen - 1)}…`;\n}\n\nfunction formatHttpBody(body: unknown): string {\n if (body === null || body === undefined) return \"\";\n if (typeof body === \"string\") {\n const trimmed = body.trim();\n if (trimmed.startsWith(\"{\") || trimmed.startsWith(\"[\")) {\n try {\n return JSON.stringify(JSON.parse(trimmed), null, 2);\n } catch {\n return body;\n }\n }\n return body;\n }\n try {\n return JSON.stringify(body, null, 2);\n } catch {\n return String(body);\n }\n}\n\nfunction withoutQuery(url: string): string {\n const idx = url.indexOf(\"?\");\n return idx === -1 ? url : url.slice(0, idx);\n}\n\nfunction maskIfSensitive(value: string, hints: string[]): string {\n const lower = hints.join(\" \").toLowerCase();\n if (lower.includes(\"password\") || lower.includes(\"pwd\")) {\n return \"••••••\";\n }\n return value;\n}\n\nfunction stepDetails(step: OpStep): string | undefined {\n const withObj = (step.with ?? {}) as Record<string, any>;\n\n switch (step.op) {\n case \"api.request\": {\n const method = String(withObj.method ?? \"\").toUpperCase();\n const url = typeof withObj.url === \"string\" ? withoutQuery(withObj.url) : \"\";\n if (!method && !url) return undefined;\n return `${method} ${url}`.trim();\n }\n case \"api.auth.bearerFromLogin\": {\n const url = typeof withObj.url === \"string\" ? withoutQuery(withObj.url) : \"\";\n return url ? `url: ${url}` : undefined;\n }\n case \"ui.goto\": {\n return withObj.url ? `url: ${withObj.url}` : undefined;\n }\n case \"ui.click\":\n case \"ui.expectVisible\": {\n return withObj.selector ? `selector: ${withObj.selector}` : undefined;\n }\n case \"ui.fill\":\n case \"ui.select\": {\n const selector = withObj.selector ? `selector: ${withObj.selector}` : \"\";\n const value =\n withObj.value !== undefined\n ? `value: ${maskIfSensitive(String(withObj.value), [String(withObj.selector ?? \"\"), String(step.op)])}`\n : \"\";\n return [selector, value].filter(Boolean).join(\", \") || undefined;\n }\n case \"ui.press\": {\n const selector = withObj.selector ? `selector: ${withObj.selector}` : \"\";\n const key = withObj.key ? `key: ${withObj.key}` : \"\";\n return [selector, key].filter(Boolean).join(\", \") || undefined;\n }\n case \"ui.expectUrl\": {\n const equals = withObj.equals ? `equals: ${withObj.equals}` : \"\";\n const contains = withObj.contains ? `contains: ${withObj.contains}` : \"\";\n return [equals, contains].filter(Boolean).join(\", \") || undefined;\n }\n case \"ui.expectText\": {\n const selector = withObj.selector ? `selector: ${withObj.selector}` : \"\";\n const equals = withObj.equals ? `equals: ${truncate(String(withObj.equals))}` : \"\";\n const contains = withObj.contains ? `contains: ${truncate(String(withObj.contains))}` : \"\";\n return [selector, equals, contains].filter(Boolean).join(\", \") || undefined;\n }\n case \"assert.httpStatus\": {\n return withObj.is !== undefined ? `status: ${withObj.is}` : undefined;\n }\n case \"assert.jsonPath\": {\n const path = withObj.path ? `path: ${withObj.path}` : \"\";\n const equals = withObj.equals !== undefined ? `equals: ${truncate(String(withObj.equals))}` : \"\";\n return [path, equals].filter(Boolean).join(\", \") || undefined;\n }\n case \"assert.contains\":\n case \"assert.equals\":\n case \"assert.matches\": {\n const expected =\n withObj.expected !== undefined\n ? `expected: ${truncate(String(withObj.expected))}`\n : withObj.pattern\n ? `pattern: ${truncate(String(withObj.pattern))}`\n : withObj.equals !== undefined\n ? `equals: ${truncate(String(withObj.equals))}`\n : \"\";\n return expected || undefined;\n }\n case \"util.sleep\": {\n return withObj.ms !== undefined ? `ms: ${withObj.ms}` : undefined;\n }\n case \"util.log\": {\n return withObj.message ? `message: ${truncate(String(withObj.message))}` : undefined;\n }\n default:\n return undefined;\n }\n}\n\nfunction formatStepLabel(step: OpStep): string {\n const base = step.name ? `${step.op} (${step.name})` : step.op;\n const details = stepDetails(step);\n if (!details) return base;\n return `${base} (${details})`;\n}\n\nexport class ConsoleReporter implements Reporter {\n onScenarioStart(scenario: { id: string; name?: string }): void {\n const label = scenario.name ? `${scenario.id} (${scenario.name})` : scenario.id;\n console.log(`\\n▶ Scenario: ${label}`);\n }\n\n onScenarioEnd(\n scenario: { id: string; name?: string },\n result: { ok: boolean; error?: unknown; durationMs: number },\n ): void {\n const label = scenario.name ? `${scenario.id} (${scenario.name})` : scenario.id;\n const status = result.ok ? \"✔\" : \"✖\";\n console.log(`${status} Scenario: ${label} (${result.durationMs}ms)`);\n }\n\n onStepStart(args: {\n scenarioId: string;\n phase: string;\n step: OpStep;\n }): void {\n const label = formatStepLabel(args.step);\n const phase = args.phase.toUpperCase();\n console.log(`${STEP_INDENT}↳ ${phase} ${label}`);\n }\n\n onStepEnd(args: {\n scenarioId: string;\n phase: string;\n step: OpStep;\n ok: boolean;\n error?: unknown;\n durationMs: number;\n }): void {\n const label = formatStepLabel(args.step);\n const phase = args.phase.toUpperCase();\n const status = args.ok ? \"✔\" : \"✖\";\n console.log(\n `${STEP_INDENT}${status} ${phase} ${label} (${args.durationMs}ms)`\n );\n if (!args.ok && args.error) {\n const details = formatError(args.error);\n console.error(indentLines(details, `${STEP_INDENT}${INDENT}`));\n const httpBody = (args.error as any)?.httpResponseBody;\n if (httpBody !== undefined) {\n const formatted = formatHttpBody(httpBody);\n const bodyText = formatted.length ? formatted : \"<empty>\";\n console.error(\n indentLines(\n `HTTP Response Body:\\n${bodyText}`,\n `${STEP_INDENT}${INDENT}`,\n ),\n );\n }\n }\n }\n\n onSpecResult(args: {\n scenarioId: string;\n specId: string;\n stepName?: string;\n result: { ok: boolean; details?: Record<string, any> };\n }): void {\n console.log(`${INDENT}↳ SPEC ${args.specId} ok=${args.result.ok}`);\n if (args.result.details) {\n const details = JSON.stringify(args.result.details, null, 2);\n console.log(indentLines(details, `${INDENT}${INDENT}`));\n }\n }\n\n attach(args: {\n scenarioId: string;\n name: string;\n contentType: string;\n data: Buffer | string;\n }): void {\n const header = `ATTACH ${args.name} (${args.contentType})`;\n console.log(indentLines(header, `${STEP_INDENT}${INDENT}`));\n const dataText = Buffer.isBuffer(args.data) ? args.data.toString(\"utf8\") : String(args.data ?? \"\");\n if (!dataText.length) return;\n console.log(indentLines(dataText, `${STEP_INDENT}${INDENT}${INDENT}`));\n }\n}\n"]}
@@ -0,0 +1,37 @@
1
+ import type { StepPhase } from "../contracts/runtime-context.types";
2
+ import type { OpStep, ScenarioSpec } from "../contracts/testing-metadata.types";
3
+ import type { SolidTestSpecResult } from "../contracts/test-spec.types";
4
+ export interface Reporter {
5
+ onScenarioStart(scenario: ScenarioSpec): void;
6
+ onScenarioEnd(scenario: ScenarioSpec, result: {
7
+ ok: boolean;
8
+ error?: unknown;
9
+ durationMs: number;
10
+ }): void;
11
+ onStepStart(args: {
12
+ scenarioId: string;
13
+ phase: StepPhase;
14
+ step: OpStep;
15
+ }): void;
16
+ onStepEnd(args: {
17
+ scenarioId: string;
18
+ phase: StepPhase;
19
+ step: OpStep;
20
+ ok: boolean;
21
+ error?: unknown;
22
+ durationMs: number;
23
+ }): void;
24
+ onSpecResult?(args: {
25
+ scenarioId: string;
26
+ specId: string;
27
+ stepName?: string;
28
+ result: SolidTestSpecResult;
29
+ }): void;
30
+ attach?(args: {
31
+ scenarioId: string;
32
+ name: string;
33
+ contentType: string;
34
+ data: Buffer | string;
35
+ }): void;
36
+ }
37
+ //# sourceMappingURL=reporter.types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reporter.types.d.ts","sourceRoot":"","sources":["../../../src/testing/reporter/reporter.types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,oCAAoC,CAAC;AACpE,OAAO,KAAK,EAAE,MAAM,EAAE,YAAY,EAAE,MAAM,qCAAqC,CAAC;AAChF,OAAO,KAAK,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AAExE,MAAM,WAAW,QAAQ;IACvB,eAAe,CAAC,QAAQ,EAAE,YAAY,GAAG,IAAI,CAAC;IAC9C,aAAa,CACX,QAAQ,EAAE,YAAY,EACtB,MAAM,EAAE;QAAE,EAAE,EAAE,OAAO,CAAC;QAAC,KAAK,CAAC,EAAE,OAAO,CAAC;QAAC,UAAU,EAAE,MAAM,CAAA;KAAE,GAC3D,IAAI,CAAC;IACR,WAAW,CAAC,IAAI,EAAE;QAChB,UAAU,EAAE,MAAM,CAAC;QACnB,KAAK,EAAE,SAAS,CAAC;QACjB,IAAI,EAAE,MAAM,CAAC;KACd,GAAG,IAAI,CAAC;IACT,SAAS,CAAC,IAAI,EAAE;QACd,UAAU,EAAE,MAAM,CAAC;QACnB,KAAK,EAAE,SAAS,CAAC;QACjB,IAAI,EAAE,MAAM,CAAC;QACb,EAAE,EAAE,OAAO,CAAC;QACZ,KAAK,CAAC,EAAE,OAAO,CAAC;QAChB,UAAU,EAAE,MAAM,CAAC;KACpB,GAAG,IAAI,CAAC;IACT,YAAY,CAAC,CAAC,IAAI,EAAE;QAClB,UAAU,EAAE,MAAM,CAAC;QACnB,MAAM,EAAE,MAAM,CAAC;QACf,QAAQ,CAAC,EAAE,MAAM,CAAC;QAClB,MAAM,EAAE,mBAAmB,CAAC;KAC7B,GAAG,IAAI,CAAC;IACT,MAAM,CAAC,CAAC,IAAI,EAAE;QACZ,UAAU,EAAE,MAAM,CAAC;QACnB,IAAI,EAAE,MAAM,CAAC;QACb,WAAW,EAAE,MAAM,CAAC;QACpB,IAAI,EAAE,MAAM,GAAG,MAAM,CAAC;KACvB,GAAG,IAAI,CAAC;CACV"}
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=reporter.types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reporter.types.js","sourceRoot":"","sources":["../../../src/testing/reporter/reporter.types.ts"],"names":[],"mappings":"","sourcesContent":["import type { StepPhase } from \"../contracts/runtime-context.types\";\nimport type { OpStep, ScenarioSpec } from \"../contracts/testing-metadata.types\";\nimport type { SolidTestSpecResult } from \"../contracts/test-spec.types\";\n\nexport interface Reporter {\n onScenarioStart(scenario: ScenarioSpec): void;\n onScenarioEnd(\n scenario: ScenarioSpec,\n result: { ok: boolean; error?: unknown; durationMs: number },\n ): void;\n onStepStart(args: {\n scenarioId: string;\n phase: StepPhase;\n step: OpStep;\n }): void;\n onStepEnd(args: {\n scenarioId: string;\n phase: StepPhase;\n step: OpStep;\n ok: boolean;\n error?: unknown;\n durationMs: number;\n }): void;\n onSpecResult?(args: {\n scenarioId: string;\n specId: string;\n stepName?: string;\n result: SolidTestSpecResult;\n }): void;\n attach?(args: {\n scenarioId: string;\n name: string;\n contentType: string;\n data: Buffer | string;\n }): void;\n}\n"]}
@@ -0,0 +1,9 @@
1
+ import type { TestContext } from "../contracts/runtime-context.types";
2
+ import type { ScenarioSpec } from "../contracts/testing-metadata.types";
3
+ type StartedFlag = {
4
+ value: boolean;
5
+ };
6
+ export declare function scenarioNeedsUi(scenario: ScenarioSpec): boolean;
7
+ export declare function ensureUiStarted(ctxBase: Omit<TestContext, "scenarioId" | "scenarioType" | "params">, startedFlag: StartedFlag): Promise<void>;
8
+ export {};
9
+ //# sourceMappingURL=lifecycle.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lifecycle.d.ts","sourceRoot":"","sources":["../../../src/testing/runner/lifecycle.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,oCAAoC,CAAC;AACtE,OAAO,KAAK,EAAU,YAAY,EAAa,MAAM,qCAAqC,CAAC;AAE3F,KAAK,WAAW,GAAG;IAAE,KAAK,EAAE,OAAO,CAAA;CAAE,CAAC;AAUtC,wBAAgB,eAAe,CAAC,QAAQ,EAAE,YAAY,GAAG,OAAO,CAQ/D;AAED,wBAAsB,eAAe,CACnC,OAAO,EAAE,IAAI,CAAC,WAAW,EAAE,YAAY,GAAG,cAAc,GAAG,QAAQ,CAAC,EACpE,WAAW,EAAE,WAAW,GACvB,OAAO,CAAC,IAAI,CAAC,CAIf"}
@@ -0,0 +1,33 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.scenarioNeedsUi = scenarioNeedsUi;
4
+ exports.ensureUiStarted = ensureUiStarted;
5
+ function stepsFromBlock(block) {
6
+ if ("given" in block)
7
+ return [block.given];
8
+ if ("when" in block)
9
+ return [block.when];
10
+ if ("and" in block)
11
+ return [block.and];
12
+ if ("then" in block)
13
+ return Array.isArray(block.then) ? block.then : [block.then];
14
+ return [block];
15
+ }
16
+ function scenarioNeedsUi(scenario) {
17
+ if (scenario.type === "ui" || scenario.type === "mixed")
18
+ return true;
19
+ for (const block of scenario.steps) {
20
+ for (const step of stepsFromBlock(block)) {
21
+ if (step.op.startsWith("ui."))
22
+ return true;
23
+ }
24
+ }
25
+ return false;
26
+ }
27
+ async function ensureUiStarted(ctxBase, startedFlag) {
28
+ if (!ctxBase.ui || startedFlag.value)
29
+ return;
30
+ await ctxBase.ui.start();
31
+ startedFlag.value = true;
32
+ }
33
+ //# sourceMappingURL=lifecycle.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lifecycle.js","sourceRoot":"","sources":["../../../src/testing/runner/lifecycle.ts"],"names":[],"mappings":";;AAaA,0CAQC;AAED,0CAOC;AAzBD,SAAS,cAAc,CAAC,KAAgB;IACtC,IAAI,OAAO,IAAI,KAAK;QAAE,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IAC3C,IAAI,MAAM,IAAI,KAAK;QAAE,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACzC,IAAI,KAAK,IAAI,KAAK;QAAE,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IACvC,IAAI,MAAM,IAAI,KAAK;QAAE,OAAO,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IAClF,OAAO,CAAC,KAAK,CAAC,CAAC;AACjB,CAAC;AAED,SAAgB,eAAe,CAAC,QAAsB;IACpD,IAAI,QAAQ,CAAC,IAAI,KAAK,IAAI,IAAI,QAAQ,CAAC,IAAI,KAAK,OAAO;QAAE,OAAO,IAAI,CAAC;IACrE,KAAK,MAAM,KAAK,IAAI,QAAQ,CAAC,KAAK,EAAE,CAAC;QACnC,KAAK,MAAM,IAAI,IAAI,cAAc,CAAC,KAAK,CAAC,EAAE,CAAC;YACzC,IAAI,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,KAAK,CAAC;gBAAE,OAAO,IAAI,CAAC;QAC7C,CAAC;IACH,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAEM,KAAK,UAAU,eAAe,CACnC,OAAoE,EACpE,WAAwB;IAExB,IAAI,CAAC,OAAO,CAAC,EAAE,IAAI,WAAW,CAAC,KAAK;QAAE,OAAO;IAC7C,MAAM,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;IACzB,WAAW,CAAC,KAAK,GAAG,IAAI,CAAC;AAC3B,CAAC","sourcesContent":["import type { TestContext } from \"../contracts/runtime-context.types\";\nimport type { OpStep, ScenarioSpec, StepBlock } from \"../contracts/testing-metadata.types\";\n\ntype StartedFlag = { value: boolean };\n\nfunction stepsFromBlock(block: StepBlock): OpStep[] {\n if (\"given\" in block) return [block.given];\n if (\"when\" in block) return [block.when];\n if (\"and\" in block) return [block.and];\n if (\"then\" in block) return Array.isArray(block.then) ? block.then : [block.then];\n return [block];\n}\n\nexport function scenarioNeedsUi(scenario: ScenarioSpec): boolean {\n if (scenario.type === \"ui\" || scenario.type === \"mixed\") return true;\n for (const block of scenario.steps) {\n for (const step of stepsFromBlock(block)) {\n if (step.op.startsWith(\"ui.\")) return true;\n }\n }\n return false;\n}\n\nexport async function ensureUiStarted(\n ctxBase: Omit<TestContext, \"scenarioId\" | \"scenarioType\" | \"params\">,\n startedFlag: StartedFlag,\n): Promise<void> {\n if (!ctxBase.ui || startedFlag.value) return;\n await ctxBase.ui.start();\n startedFlag.value = true;\n}\n"]}
@@ -0,0 +1,24 @@
1
+ import type { Reporter } from "../reporter/reporter.types";
2
+ import type { TestingMetadata } from "../contracts/testing-metadata.types";
3
+ import type { ApiAdapterOptions } from "../adapters/api/api.types";
4
+ import type { PlaywrightAdapterOptions } from "../adapters/ui/ui.types";
5
+ import { SpecRegistry } from "../core/spec-registry";
6
+ export type RunnerOptions = {
7
+ metadata: TestingMetadata;
8
+ scenarioIds?: string[];
9
+ includeTags?: string[];
10
+ skipScenarioIds?: string[];
11
+ reporter?: Reporter;
12
+ api?: ApiAdapterOptions;
13
+ ui?: PlaywrightAdapterOptions;
14
+ defaults?: {
15
+ timeoutMs?: number;
16
+ retries?: number;
17
+ };
18
+ options?: {
19
+ printApiLogs?: boolean;
20
+ };
21
+ specs?: (registry: SpecRegistry) => void;
22
+ };
23
+ export declare function runFromMetadata(opts: RunnerOptions): Promise<void>;
24
+ //# sourceMappingURL=run-from-metadata.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"run-from-metadata.d.ts","sourceRoot":"","sources":["../../../src/testing/runner/run-from-metadata.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,4BAA4B,CAAC;AAC3D,OAAO,KAAK,EAAE,eAAe,EAAqB,MAAM,qCAAqC,CAAC;AAC9F,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,2BAA2B,CAAC;AACnE,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,yBAAyB,CAAC;AAUxE,OAAO,EAAE,YAAY,EAAE,MAAM,uBAAuB,CAAC;AAoBrD,MAAM,MAAM,aAAa,GAAG;IAC1B,QAAQ,EAAE,eAAe,CAAC;IAC1B,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC;IACvB,WAAW,CAAC,EAAE,MAAM,EAAE,CAAC;IACvB,eAAe,CAAC,EAAE,MAAM,EAAE,CAAC;IAC3B,QAAQ,CAAC,EAAE,QAAQ,CAAC;IACpB,GAAG,CAAC,EAAE,iBAAiB,CAAC;IACxB,EAAE,CAAC,EAAE,wBAAwB,CAAC;IAC9B,QAAQ,CAAC,EAAE;QAAE,SAAS,CAAC,EAAE,MAAM,CAAC;QAAC,OAAO,CAAC,EAAE,MAAM,CAAA;KAAE,CAAC;IACpD,OAAO,CAAC,EAAE;QAAE,YAAY,CAAC,EAAE,OAAO,CAAA;KAAE,CAAC;IACrC,KAAK,CAAC,EAAE,CAAC,QAAQ,EAAE,YAAY,KAAK,IAAI,CAAC;CAC1C,CAAC;AAEF,wBAAsB,eAAe,CAAC,IAAI,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CAwCxE"}