react-native-harness 0.0.0 → 1.0.0-alpha.1

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 (103) hide show
  1. package/LICENSE +20 -0
  2. package/README.md +7 -0
  3. package/bin.js +3 -0
  4. package/dist/bundlers/metro.d.ts +5 -0
  5. package/dist/bundlers/metro.d.ts.map +1 -0
  6. package/dist/bundlers/metro.js +52 -0
  7. package/dist/bundlers/webpack.d.ts +2 -0
  8. package/dist/bundlers/webpack.d.ts.map +1 -0
  9. package/dist/bundlers/webpack.js +49 -0
  10. package/dist/commands/test.d.ts +2 -0
  11. package/dist/commands/test.d.ts.map +1 -0
  12. package/dist/commands/test.js +111 -0
  13. package/dist/errors/appNotInstalledError.d.ts +7 -0
  14. package/dist/errors/appNotInstalledError.d.ts.map +1 -0
  15. package/dist/errors/appNotInstalledError.js +12 -0
  16. package/dist/errors/bridgeTimeoutError.d.ts +7 -0
  17. package/dist/errors/bridgeTimeoutError.d.ts.map +1 -0
  18. package/dist/errors/bridgeTimeoutError.js +12 -0
  19. package/dist/errors/errorHandler.d.ts +2 -0
  20. package/dist/errors/errorHandler.d.ts.map +1 -0
  21. package/dist/errors/errorHandler.js +138 -0
  22. package/dist/errors/errors.d.ts +41 -0
  23. package/dist/errors/errors.d.ts.map +1 -0
  24. package/dist/errors/errors.js +83 -0
  25. package/dist/index.d.ts +3 -0
  26. package/dist/index.d.ts.map +1 -0
  27. package/dist/index.js +37 -0
  28. package/dist/platforms/android/build.d.ts +5 -0
  29. package/dist/platforms/android/build.d.ts.map +1 -0
  30. package/dist/platforms/android/build.js +29 -0
  31. package/dist/platforms/android/device.d.ts +5 -0
  32. package/dist/platforms/android/device.d.ts.map +1 -0
  33. package/dist/platforms/android/device.js +36 -0
  34. package/dist/platforms/android/emulator.d.ts +11 -0
  35. package/dist/platforms/android/emulator.d.ts.map +1 -0
  36. package/dist/platforms/android/emulator.js +110 -0
  37. package/dist/platforms/android/index.d.ts +4 -0
  38. package/dist/platforms/android/index.d.ts.map +1 -0
  39. package/dist/platforms/android/index.js +56 -0
  40. package/dist/platforms/ios/build.d.ts +7 -0
  41. package/dist/platforms/ios/build.d.ts.map +1 -0
  42. package/dist/platforms/ios/build.js +51 -0
  43. package/dist/platforms/ios/device.d.ts +7 -0
  44. package/dist/platforms/ios/device.d.ts.map +1 -0
  45. package/dist/platforms/ios/device.js +51 -0
  46. package/dist/platforms/ios/index.d.ts +4 -0
  47. package/dist/platforms/ios/index.d.ts.map +1 -0
  48. package/dist/platforms/ios/index.js +38 -0
  49. package/dist/platforms/ios/simulator.d.ts +11 -0
  50. package/dist/platforms/ios/simulator.d.ts.map +1 -0
  51. package/dist/platforms/ios/simulator.js +133 -0
  52. package/dist/platforms/platform-adapter.d.ts +10 -0
  53. package/dist/platforms/platform-adapter.d.ts.map +1 -0
  54. package/dist/platforms/platform-adapter.js +1 -0
  55. package/dist/platforms/platform-registry.d.ts +3 -0
  56. package/dist/platforms/platform-registry.d.ts.map +1 -0
  57. package/dist/platforms/platform-registry.js +19 -0
  58. package/dist/platforms/web/index.d.ts +4 -0
  59. package/dist/platforms/web/index.d.ts.map +1 -0
  60. package/dist/platforms/web/index.js +73 -0
  61. package/dist/process.d.ts +3 -0
  62. package/dist/process.d.ts.map +1 -0
  63. package/dist/process.js +28 -0
  64. package/dist/reporters/default-reporter.d.ts +3 -0
  65. package/dist/reporters/default-reporter.d.ts.map +1 -0
  66. package/dist/reporters/default-reporter.js +116 -0
  67. package/dist/reporters/junit-reporter.d.ts +3 -0
  68. package/dist/reporters/junit-reporter.d.ts.map +1 -0
  69. package/dist/reporters/junit-reporter.js +119 -0
  70. package/dist/reporters/live-reporter.d.ts +20 -0
  71. package/dist/reporters/live-reporter.d.ts.map +1 -0
  72. package/dist/reporters/live-reporter.js +176 -0
  73. package/dist/src/reporters/default-reporter.js +135 -0
  74. package/dist/test-reporter-demo.js +95 -0
  75. package/dist/tsconfig.lib.tsbuildinfo +1 -0
  76. package/dist/utils.d.ts +5 -0
  77. package/dist/utils.d.ts.map +1 -0
  78. package/dist/utils.js +11 -0
  79. package/eslint.config.mjs +19 -0
  80. package/package.json +16 -8
  81. package/src/bundlers/metro.ts +79 -0
  82. package/src/commands/test.ts +189 -0
  83. package/src/errors/errorHandler.ts +184 -0
  84. package/src/errors/errors.ts +109 -0
  85. package/src/index.ts +52 -0
  86. package/src/platforms/android/build.ts +48 -0
  87. package/src/platforms/android/device.ts +48 -0
  88. package/src/platforms/android/emulator.ts +139 -0
  89. package/src/platforms/android/index.ts +87 -0
  90. package/src/platforms/ios/build.ts +76 -0
  91. package/src/platforms/ios/device.ts +76 -0
  92. package/src/platforms/ios/index.ts +55 -0
  93. package/src/platforms/ios/simulator.ts +177 -0
  94. package/src/platforms/platform-adapter.ts +11 -0
  95. package/src/platforms/platform-registry.ts +24 -0
  96. package/src/platforms/web/index.ts +95 -0
  97. package/src/process.ts +33 -0
  98. package/src/reporters/default-reporter.ts +149 -0
  99. package/src/reporters/junit-reporter.ts +179 -0
  100. package/src/utils.ts +12 -0
  101. package/tsconfig.json +16 -0
  102. package/tsconfig.lib.json +33 -0
  103. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,179 @@
1
+ import { Reporter } from '@react-native-harness/config';
2
+ import type { TestSuiteResult, TestResult } from '@react-native-harness/bridge';
3
+ import { writeFileSync } from 'fs';
4
+ import { join } from 'path';
5
+
6
+ export const junitReporter: Reporter = {
7
+ report: async (results) => {
8
+ const xml = generateJUnitXML(results);
9
+
10
+ // Write to junit.xml file
11
+ const outputPath = join(process.cwd(), 'junit.xml');
12
+ writeFileSync(outputPath, xml, 'utf8');
13
+
14
+ console.log(`📄 JUnit report written to: ${outputPath}`);
15
+ },
16
+ };
17
+
18
+ const generateJUnitXML = (results: TestSuiteResult[]): string => {
19
+ const { totalTests, totalFailures, totalSkipped, totalTime } =
20
+ calculateTotals(results);
21
+
22
+ let xml = '<?xml version="1.0" encoding="UTF-8"?>\n';
23
+ xml += `<testsuites tests="${totalTests}" failures="${totalFailures}" skipped="${totalSkipped}" time="${
24
+ totalTime / 1000
25
+ }">\n`;
26
+
27
+ for (const suite of results) {
28
+ xml += generateTestSuiteXML(suite, ' ');
29
+ }
30
+
31
+ xml += '</testsuites>\n';
32
+
33
+ return xml;
34
+ };
35
+
36
+ const generateTestSuiteXML = (
37
+ suite: TestSuiteResult,
38
+ indent: string
39
+ ): string => {
40
+ const { tests, failures, skipped, time } = getSuiteStats(suite);
41
+
42
+ let xml = `${indent}<testsuite name="${escapeXML(
43
+ suite.name
44
+ )}" tests="${tests}" failures="${failures}" skipped="${skipped}" time="${
45
+ time / 1000
46
+ }"`;
47
+
48
+ if (suite.error) {
49
+ xml += ` errors="1"`;
50
+ }
51
+
52
+ xml += '>\n';
53
+
54
+ // Add suite-level error if present
55
+ if (suite.error) {
56
+ xml += `${indent} <error message="${escapeXML(
57
+ suite.error.message || 'Suite failed'
58
+ )}"`;
59
+ if (suite.error.name) {
60
+ xml += ` type="${escapeXML(suite.error.name)}"`;
61
+ }
62
+ xml += '>';
63
+ if (suite.error.stack) {
64
+ xml += escapeXML(suite.error.stack);
65
+ }
66
+ xml += '</error>\n';
67
+ }
68
+
69
+ // Add individual test cases
70
+ for (const test of suite.tests) {
71
+ xml += generateTestCaseXML(test, indent + ' ');
72
+ }
73
+
74
+ // Add nested suites
75
+ for (const nestedSuite of suite.suites) {
76
+ xml += generateTestSuiteXML(nestedSuite, indent + ' ');
77
+ }
78
+
79
+ xml += `${indent}</testsuite>\n`;
80
+
81
+ return xml;
82
+ };
83
+
84
+ const generateTestCaseXML = (test: TestResult, indent: string): string => {
85
+ const time = (test.duration || 0) / 1000;
86
+ let xml = `${indent}<testcase name="${escapeXML(test.name)}" time="${time}"`;
87
+
88
+ if (test.status === 'passed') {
89
+ xml += '/>\n';
90
+ } else {
91
+ xml += '>\n';
92
+
93
+ switch (test.status) {
94
+ case 'failed':
95
+ xml += `${indent} <failure message="${escapeXML(
96
+ test.error?.message || 'Test failed'
97
+ )}"`;
98
+ if (test.error?.name) {
99
+ xml += ` type="${escapeXML(test.error.name)}"`;
100
+ }
101
+ xml += '>';
102
+ if (test.error?.stack) {
103
+ xml += escapeXML(test.error.stack);
104
+ }
105
+ xml += '</failure>\n';
106
+ break;
107
+ case 'skipped':
108
+ xml += `${indent} <skipped/>\n`;
109
+ break;
110
+ case 'todo':
111
+ xml += `${indent} <skipped message="TODO: Test not implemented"/>\n`;
112
+ break;
113
+ }
114
+
115
+ xml += `${indent}</testcase>\n`;
116
+ }
117
+
118
+ return xml;
119
+ };
120
+
121
+ const calculateTotals = (
122
+ results: TestSuiteResult[]
123
+ ): {
124
+ totalTests: number;
125
+ totalFailures: number;
126
+ totalSkipped: number;
127
+ totalTime: number;
128
+ } => {
129
+ let totalTests = 0;
130
+ let totalFailures = 0;
131
+ let totalSkipped = 0;
132
+ let totalTime = 0;
133
+
134
+ for (const suite of results) {
135
+ const stats = getSuiteStats(suite);
136
+ totalTests += stats.tests;
137
+ totalFailures += stats.failures;
138
+ totalSkipped += stats.skipped;
139
+ totalTime += stats.time;
140
+ }
141
+
142
+ return { totalTests, totalFailures, totalSkipped, totalTime };
143
+ };
144
+
145
+ const getSuiteStats = (
146
+ suite: TestSuiteResult
147
+ ): {
148
+ tests: number;
149
+ failures: number;
150
+ skipped: number;
151
+ time: number;
152
+ } => {
153
+ let tests = suite.tests.length;
154
+ let failures = suite.tests.filter((t) => t.status === 'failed').length;
155
+ let skipped = suite.tests.filter(
156
+ (t) => t.status === 'skipped' || t.status === 'todo'
157
+ ).length;
158
+ let time = suite.duration || 0;
159
+
160
+ // Add stats from nested suites
161
+ for (const nestedSuite of suite.suites) {
162
+ const nestedStats = getSuiteStats(nestedSuite);
163
+ tests += nestedStats.tests;
164
+ failures += nestedStats.failures;
165
+ skipped += nestedStats.skipped;
166
+ time += nestedStats.time;
167
+ }
168
+
169
+ return { tests, failures, skipped, time };
170
+ };
171
+
172
+ const escapeXML = (str: string): string => {
173
+ return str
174
+ .replace(/&/g, '&amp;')
175
+ .replace(/</g, '&lt;')
176
+ .replace(/>/g, '&gt;')
177
+ .replace(/"/g, '&quot;')
178
+ .replace(/'/g, '&apos;');
179
+ };
package/src/utils.ts ADDED
@@ -0,0 +1,12 @@
1
+ export function assert(condition: boolean, message: string): asserts condition {
2
+ if (!condition) {
3
+ throw new AssertionError(message);
4
+ }
5
+ }
6
+
7
+ export class AssertionError extends Error {
8
+ constructor(message: string) {
9
+ super(message);
10
+ this.name = 'AssertionError';
11
+ }
12
+ }
package/tsconfig.json ADDED
@@ -0,0 +1,16 @@
1
+ {
2
+ "extends": "../../tsconfig.base.json",
3
+ "files": [],
4
+ "include": [],
5
+ "references": [
6
+ {
7
+ "path": "../config"
8
+ },
9
+ {
10
+ "path": "../bridge"
11
+ },
12
+ {
13
+ "path": "./tsconfig.lib.json"
14
+ }
15
+ ]
16
+ }
@@ -0,0 +1,33 @@
1
+ {
2
+ "extends": "../../tsconfig.base.json",
3
+ "compilerOptions": {
4
+ "baseUrl": ".",
5
+ "rootDir": "src",
6
+ "outDir": "dist",
7
+ "tsBuildInfoFile": "dist/tsconfig.lib.tsbuildinfo",
8
+ "emitDeclarationOnly": false,
9
+ "forceConsistentCasingInFileNames": true,
10
+ "types": ["node"],
11
+ "lib": ["DOM"]
12
+ },
13
+ "include": [
14
+ "src/**/*.ts",
15
+ "../tools/src/abort.ts",
16
+ "../tools/src/color.ts",
17
+ "../tools/src/isInteractive.ts",
18
+ "../tools/src/logger.ts",
19
+ "../tools/src/prompts.ts",
20
+ "../tools/src/spawn.ts"
21
+ ],
22
+ "references": [
23
+ {
24
+ "path": "../tools/tsconfig.lib.json"
25
+ },
26
+ {
27
+ "path": "../config/tsconfig.lib.json"
28
+ },
29
+ {
30
+ "path": "../bridge/tsconfig.lib.json"
31
+ }
32
+ ]
33
+ }
@@ -0,0 +1 @@
1
+ {"fileNames":[],"fileInfos":[],"root":[],"options":{"composite":true,"declarationMap":true,"emitDeclarationOnly":true,"importHelpers":true,"module":199,"noEmitOnError":true,"noFallthroughCasesInSwitch":true,"noImplicitOverride":true,"noImplicitReturns":true,"noUnusedLocals":true,"skipLibCheck":true,"strict":true,"target":9},"version":"5.8.3"}