@arcadialdev/arcality 2.2.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 (97) hide show
  1. package/.agents/skills/e2e-testing-expert/SKILL.md +28 -0
  2. package/.agents/skills/frontend-design/LICENSE.txt +177 -0
  3. package/.agents/skills/frontend-design/SKILL.md +42 -0
  4. package/.agents/skills/nodejs-backend-patterns/SKILL.md +639 -0
  5. package/.agents/skills/nodejs-backend-patterns/references/advanced-patterns.md +430 -0
  6. package/.agents/skills/playwright-best-practices/LICENSE.md +7 -0
  7. package/.agents/skills/playwright-best-practices/README.md +147 -0
  8. package/.agents/skills/playwright-best-practices/SKILL.md +303 -0
  9. package/.agents/skills/playwright-best-practices/advanced/authentication-flows.md +360 -0
  10. package/.agents/skills/playwright-best-practices/advanced/authentication.md +871 -0
  11. package/.agents/skills/playwright-best-practices/advanced/clock-mocking.md +364 -0
  12. package/.agents/skills/playwright-best-practices/advanced/mobile-testing.md +409 -0
  13. package/.agents/skills/playwright-best-practices/advanced/multi-context.md +288 -0
  14. package/.agents/skills/playwright-best-practices/advanced/multi-user.md +393 -0
  15. package/.agents/skills/playwright-best-practices/advanced/network-advanced.md +452 -0
  16. package/.agents/skills/playwright-best-practices/advanced/third-party.md +464 -0
  17. package/.agents/skills/playwright-best-practices/architecture/pom-vs-fixtures.md +363 -0
  18. package/.agents/skills/playwright-best-practices/architecture/test-architecture.md +369 -0
  19. package/.agents/skills/playwright-best-practices/architecture/when-to-mock.md +383 -0
  20. package/.agents/skills/playwright-best-practices/browser-apis/browser-apis.md +391 -0
  21. package/.agents/skills/playwright-best-practices/browser-apis/iframes.md +403 -0
  22. package/.agents/skills/playwright-best-practices/browser-apis/service-workers.md +504 -0
  23. package/.agents/skills/playwright-best-practices/browser-apis/websockets.md +403 -0
  24. package/.agents/skills/playwright-best-practices/core/annotations.md +424 -0
  25. package/.agents/skills/playwright-best-practices/core/assertions-waiting.md +361 -0
  26. package/.agents/skills/playwright-best-practices/core/configuration.md +452 -0
  27. package/.agents/skills/playwright-best-practices/core/fixtures-hooks.md +417 -0
  28. package/.agents/skills/playwright-best-practices/core/global-setup.md +434 -0
  29. package/.agents/skills/playwright-best-practices/core/locators.md +242 -0
  30. package/.agents/skills/playwright-best-practices/core/page-object-model.md +315 -0
  31. package/.agents/skills/playwright-best-practices/core/projects-dependencies.md +453 -0
  32. package/.agents/skills/playwright-best-practices/core/test-data.md +492 -0
  33. package/.agents/skills/playwright-best-practices/core/test-suite-structure.md +361 -0
  34. package/.agents/skills/playwright-best-practices/core/test-tags.md +298 -0
  35. package/.agents/skills/playwright-best-practices/debugging/console-errors.md +420 -0
  36. package/.agents/skills/playwright-best-practices/debugging/debugging.md +504 -0
  37. package/.agents/skills/playwright-best-practices/debugging/error-testing.md +360 -0
  38. package/.agents/skills/playwright-best-practices/debugging/flaky-tests.md +496 -0
  39. package/.agents/skills/playwright-best-practices/frameworks/angular.md +530 -0
  40. package/.agents/skills/playwright-best-practices/frameworks/nextjs.md +469 -0
  41. package/.agents/skills/playwright-best-practices/frameworks/react.md +531 -0
  42. package/.agents/skills/playwright-best-practices/frameworks/vue.md +574 -0
  43. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/ci-cd.md +468 -0
  44. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/docker.md +283 -0
  45. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/github-actions.md +546 -0
  46. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/gitlab.md +397 -0
  47. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/other-providers.md +521 -0
  48. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/parallel-sharding.md +371 -0
  49. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/performance.md +453 -0
  50. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/reporting.md +424 -0
  51. package/.agents/skills/playwright-best-practices/infrastructure-ci-cd/test-coverage.md +497 -0
  52. package/.agents/skills/playwright-best-practices/testing-patterns/accessibility.md +359 -0
  53. package/.agents/skills/playwright-best-practices/testing-patterns/api-testing.md +719 -0
  54. package/.agents/skills/playwright-best-practices/testing-patterns/browser-extensions.md +506 -0
  55. package/.agents/skills/playwright-best-practices/testing-patterns/canvas-webgl.md +493 -0
  56. package/.agents/skills/playwright-best-practices/testing-patterns/component-testing.md +500 -0
  57. package/.agents/skills/playwright-best-practices/testing-patterns/drag-drop.md +576 -0
  58. package/.agents/skills/playwright-best-practices/testing-patterns/electron.md +509 -0
  59. package/.agents/skills/playwright-best-practices/testing-patterns/file-operations.md +377 -0
  60. package/.agents/skills/playwright-best-practices/testing-patterns/file-upload-download.md +562 -0
  61. package/.agents/skills/playwright-best-practices/testing-patterns/forms-validation.md +561 -0
  62. package/.agents/skills/playwright-best-practices/testing-patterns/graphql-testing.md +331 -0
  63. package/.agents/skills/playwright-best-practices/testing-patterns/i18n.md +508 -0
  64. package/.agents/skills/playwright-best-practices/testing-patterns/performance-testing.md +476 -0
  65. package/.agents/skills/playwright-best-practices/testing-patterns/security-testing.md +430 -0
  66. package/.agents/skills/playwright-best-practices/testing-patterns/visual-regression.md +634 -0
  67. package/.env.example +21 -0
  68. package/README.md +30 -0
  69. package/bin/arcality.mjs +86 -0
  70. package/package.json +66 -0
  71. package/playwright.config.ts +12 -0
  72. package/scripts/cleanup-qmsdev.mjs +63 -0
  73. package/scripts/discover-view.mjs +52 -0
  74. package/scripts/extract-view.mjs +64 -0
  75. package/scripts/gen-and-run.mjs +838 -0
  76. package/scripts/init.mjs +290 -0
  77. package/scripts/migrate-to-central-out.mjs +157 -0
  78. package/scripts/postinstall.mjs +63 -0
  79. package/scripts/rebrand-report.mjs +241 -0
  80. package/scripts/setup.mjs +166 -0
  81. package/src/KnowledgeService.ts +239 -0
  82. package/src/arcalityClient.mjs +266 -0
  83. package/src/configLoader.mjs +179 -0
  84. package/src/configManager.mjs +172 -0
  85. package/src/consoleBanner.ts +32 -0
  86. package/src/envSetup.ts +205 -0
  87. package/src/index.ts +25 -0
  88. package/src/projectInspector.ts +42 -0
  89. package/src/services/collectiveMemoryService.ts +178 -0
  90. package/src/testRunner.ts +201 -0
  91. package/tests/_helpers/ArcalityReporter.ts +490 -0
  92. package/tests/_helpers/agentic-runner.spec.ts +741 -0
  93. package/tests/_helpers/ai-agent-helper.ts +1573 -0
  94. package/tests/_helpers/discover-view.spec.ts +238 -0
  95. package/tests/_helpers/extract-view.spec.ts +118 -0
  96. package/tests/_helpers/qa-tools.ts +333 -0
  97. package/tests/_helpers/smart-action.spec.ts +1458 -0
@@ -0,0 +1,383 @@
1
+ # Mocking Strategy: Real vs Mock Services
2
+
3
+ ## Table of Contents
4
+
5
+ 1. [Core Principle](#core-principle)
6
+ 2. [Decision Matrix](#decision-matrix)
7
+ 3. [Decision Flowchart](#decision-flowchart)
8
+ 4. [Mocking Techniques](#mocking-techniques)
9
+ 5. [Real Service Strategies](#real-service-strategies)
10
+ 6. [Hybrid Approach: Fixture-Based Mock Control](#hybrid-approach-fixture-based-mock-control)
11
+ 7. [Validating Mock Accuracy](#validating-mock-accuracy)
12
+ 8. [Anti-Patterns](#anti-patterns)
13
+
14
+ > **When to use**: Deciding whether to mock API calls, intercept network requests, or hit real services in Playwright tests.
15
+
16
+ ## Core Principle
17
+
18
+ **Mock at the boundary, test your stack end-to-end.** Mock third-party services you don't own (payment gateways, email providers, OAuth). Never mock your own frontend-to-backend communication. Tests should prove YOUR code works, not that third-party APIs are available.
19
+
20
+ ## Decision Matrix
21
+
22
+ | Scenario | Mock? | Strategy |
23
+ | --- | --- | --- |
24
+ | Your own REST/GraphQL API | Never | Hit real API against staging or local dev |
25
+ | Your database (through your API) | Never | Seed via API or fixtures |
26
+ | Authentication (your auth system) | Mostly no | Use `storageState` to skip login in most tests |
27
+ | Stripe / payment gateway | Always | `route.fulfill()` with expected responses |
28
+ | SendGrid / email service | Always | Mock the API call, verify request payload |
29
+ | OAuth providers (Google, GitHub) | Always | Mock token exchange, test your callback handler |
30
+ | Analytics (Segment, Mixpanel) | Always | `route.abort()` or `route.fulfill()` |
31
+ | Maps / geocoding APIs | Always | Mock with static responses |
32
+ | Feature flags (LaunchDarkly) | Usually | Mock to force specific flag states |
33
+ | CDN / static assets | Never | Let them load normally |
34
+ | Flaky external dependency | CI: mock, local: real | Conditional mocking based on environment |
35
+ | Slow external dependency | Dev: mock, nightly: real | Separate test projects in config |
36
+
37
+ ## Decision Flowchart
38
+
39
+ ```text
40
+ Is this service part of YOUR codebase?
41
+ ├── YES → Do NOT mock. Test the real integration.
42
+ │ ├── Is it slow? → Optimize the service, not the test.
43
+ │ └── Is it flaky? → Fix the service. Flaky infra is a bug.
44
+ └── NO → It's a third-party service.
45
+ ├── Is it paid per call? → ALWAYS mock.
46
+ ├── Is it rate-limited? → ALWAYS mock.
47
+ ├── Is it slow or unreliable? → ALWAYS mock.
48
+ └── Is it a complex multi-step flow? → Mock with HAR recording.
49
+ ```
50
+
51
+ ## Mocking Techniques
52
+
53
+ ### Blocking Unwanted Requests
54
+
55
+ Block third-party scripts that slow tests and add no coverage:
56
+
57
+ ```typescript
58
+ test.beforeEach(async ({ page }) => {
59
+ await page.route('**/{analytics,tracking,segment,hotjar}.{com,io}/**', (route) => {
60
+ route.abort();
61
+ });
62
+ });
63
+
64
+ test('dashboard renders without tracking scripts', async ({ page }) => {
65
+ await page.goto('/dashboard');
66
+ await expect(page.getByRole('heading', { name: 'Dashboard' })).toBeVisible();
67
+ });
68
+ ```
69
+
70
+ ### Full Mock (route.fulfill)
71
+
72
+ Completely replace a third-party API response:
73
+
74
+ ```typescript
75
+ test('order flow with mocked payment service', async ({ page }) => {
76
+ await page.route('**/api/charge', (route) => {
77
+ route.fulfill({
78
+ status: 200,
79
+ contentType: 'application/json',
80
+ body: JSON.stringify({
81
+ transactionId: 'txn_mock_abc',
82
+ status: 'completed',
83
+ }),
84
+ });
85
+ });
86
+
87
+ await page.goto('/order/confirm');
88
+ await page.getByRole('button', { name: 'Complete Purchase' }).click();
89
+ await expect(page.getByText('Order confirmed')).toBeVisible();
90
+ });
91
+
92
+ test('display error on payment decline', async ({ page }) => {
93
+ await page.route('**/api/charge', (route) => {
94
+ route.fulfill({
95
+ status: 402,
96
+ contentType: 'application/json',
97
+ body: JSON.stringify({
98
+ error: { code: 'insufficient_funds', message: 'Card declined.' },
99
+ }),
100
+ });
101
+ });
102
+
103
+ await page.goto('/order/confirm');
104
+ await page.getByRole('button', { name: 'Complete Purchase' }).click();
105
+ await expect(page.getByRole('alert')).toContainText('Card declined');
106
+ });
107
+ ```
108
+
109
+ ### Partial Mock (Modify Responses)
110
+
111
+ Let the real API call happen but tweak the response:
112
+
113
+ ```typescript
114
+ test('display low inventory warning', async ({ page }) => {
115
+ await page.route('**/api/inventory/*', async (route) => {
116
+ const response = await route.fetch();
117
+ const data = await response.json();
118
+
119
+ data.quantity = 1;
120
+ data.lowStock = true;
121
+
122
+ await route.fulfill({
123
+ response,
124
+ body: JSON.stringify(data),
125
+ });
126
+ });
127
+
128
+ await page.goto('/products/widget-pro');
129
+ await expect(page.getByText('Only 1 remaining')).toBeVisible();
130
+ });
131
+
132
+ test('inject test notification into real response', async ({ page }) => {
133
+ await page.route('**/api/alerts', async (route) => {
134
+ const response = await route.fetch();
135
+ const data = await response.json();
136
+
137
+ data.items.push({
138
+ id: 'test-alert',
139
+ text: 'Report generated',
140
+ category: 'info',
141
+ });
142
+
143
+ await route.fulfill({
144
+ response,
145
+ body: JSON.stringify(data),
146
+ });
147
+ });
148
+
149
+ await page.goto('/home');
150
+ await expect(page.getByText('Report generated')).toBeVisible();
151
+ });
152
+ ```
153
+
154
+ ### Record and Replay (HAR Files)
155
+
156
+ For complex API sequences (OAuth flows, multi-step wizards):
157
+
158
+ **Recording:**
159
+
160
+ ```typescript
161
+ test('capture API traffic for admin panel', async ({ page }) => {
162
+ await page.routeFromHAR('tests/fixtures/admin-panel.har', {
163
+ url: '**/api/**',
164
+ update: true,
165
+ });
166
+
167
+ await page.goto('/admin');
168
+ await page.getByRole('tab', { name: 'Reports' }).click();
169
+ await page.getByRole('tab', { name: 'Settings' }).click();
170
+ });
171
+ ```
172
+
173
+ **Replaying:**
174
+
175
+ ```typescript
176
+ test('admin panel loads with recorded data', async ({ page }) => {
177
+ await page.routeFromHAR('tests/fixtures/admin-panel.har', {
178
+ url: '**/api/**',
179
+ update: false,
180
+ });
181
+
182
+ await page.goto('/admin');
183
+ await expect(page.getByRole('heading', { name: 'Reports' })).toBeVisible();
184
+ });
185
+ ```
186
+
187
+ **HAR maintenance:**
188
+
189
+ - Record against a known-good staging environment
190
+ - Commit `.har` files to version control
191
+ - Re-record when APIs change
192
+ - Scope HAR to specific URL patterns
193
+
194
+ ## Real Service Strategies
195
+
196
+ ### Local Dev Server
197
+
198
+ ```typescript
199
+ // playwright.config.ts
200
+ export default defineConfig({
201
+ webServer: {
202
+ command: 'npm run dev',
203
+ url: 'http://localhost:3000',
204
+ reuseExistingServer: !process.env.CI,
205
+ timeout: 30_000,
206
+ },
207
+ use: {
208
+ baseURL: 'http://localhost:3000',
209
+ },
210
+ });
211
+ ```
212
+
213
+ ### Staging Environment
214
+
215
+ ```typescript
216
+ // playwright.config.ts
217
+ export default defineConfig({
218
+ use: {
219
+ baseURL: process.env.CI
220
+ ? 'https://staging.example.com'
221
+ : 'http://localhost:3000',
222
+ },
223
+ });
224
+ ```
225
+
226
+ ### Test Containers
227
+
228
+ ```typescript
229
+ // playwright.config.ts
230
+ export default defineConfig({
231
+ webServer: {
232
+ command: 'docker compose -f docker-compose.test.yml up --wait',
233
+ url: 'http://localhost:3000/health',
234
+ reuseExistingServer: !process.env.CI,
235
+ timeout: 120_000,
236
+ },
237
+ globalTeardown: './tests/global-teardown.ts',
238
+ });
239
+ ```
240
+
241
+ ```typescript
242
+ // tests/global-teardown.ts
243
+ import { execSync } from 'child_process';
244
+
245
+ export default function globalTeardown() {
246
+ if (process.env.CI) {
247
+ execSync('docker compose -f docker-compose.test.yml down -v');
248
+ }
249
+ }
250
+ ```
251
+
252
+ ## Hybrid Approach: Fixture-Based Mock Control
253
+
254
+ Create fixtures that let individual tests opt into mocking specific services:
255
+
256
+ ```typescript
257
+ // tests/fixtures/service-mocks.ts
258
+ import { test as base } from '@playwright/test';
259
+
260
+ type MockConfig = {
261
+ mockPayments: boolean;
262
+ mockNotifications: boolean;
263
+ mockAnalytics: boolean;
264
+ };
265
+
266
+ export const test = base.extend<MockConfig>({
267
+ mockPayments: [true, { option: true }],
268
+ mockNotifications: [true, { option: true }],
269
+ mockAnalytics: [true, { option: true }],
270
+
271
+ page: async ({ page, mockPayments, mockNotifications, mockAnalytics }, use) => {
272
+ if (mockPayments) {
273
+ await page.route('**/api/billing/**', (route) => {
274
+ route.fulfill({
275
+ status: 200,
276
+ contentType: 'application/json',
277
+ body: JSON.stringify({ status: 'paid', id: 'inv_mock_789' }),
278
+ });
279
+ });
280
+ }
281
+
282
+ if (mockNotifications) {
283
+ await page.route('**/api/notify', (route) => {
284
+ route.fulfill({
285
+ status: 200,
286
+ contentType: 'application/json',
287
+ body: JSON.stringify({ delivered: true }),
288
+ });
289
+ });
290
+ }
291
+
292
+ if (mockAnalytics) {
293
+ await page.route('**/{segment,mixpanel,amplitude}.**/**', (route) => {
294
+ route.abort();
295
+ });
296
+ }
297
+
298
+ await use(page);
299
+ },
300
+ });
301
+
302
+ export { expect } from '@playwright/test';
303
+ ```
304
+
305
+ ```typescript
306
+ // tests/billing.spec.ts
307
+ import { test, expect } from './fixtures/service-mocks';
308
+
309
+ test('subscription renewal sends notification', async ({ page }) => {
310
+ await page.goto('/account/billing');
311
+ await page.getByRole('button', { name: 'Renew Now' }).click();
312
+ await expect(page.getByText('Subscription renewed')).toBeVisible();
313
+ });
314
+
315
+ test.describe('integration suite', () => {
316
+ test.use({ mockPayments: false });
317
+
318
+ test('real billing flow against test gateway', async ({ page }) => {
319
+ await page.goto('/account/billing');
320
+ await page.getByRole('button', { name: 'Renew Now' }).click();
321
+ await expect(page.getByText('Subscription renewed')).toBeVisible();
322
+ });
323
+ });
324
+ ```
325
+
326
+ ### Environment-Based Test Projects
327
+
328
+ ```typescript
329
+ // playwright.config.ts
330
+ export default defineConfig({
331
+ projects: [
332
+ {
333
+ name: 'ci-fast',
334
+ testMatch: '**/*.spec.ts',
335
+ use: { baseURL: 'http://localhost:3000' },
336
+ },
337
+ {
338
+ name: 'nightly-full',
339
+ testMatch: '**/*.integration.spec.ts',
340
+ use: { baseURL: 'https://staging.example.com' },
341
+ timeout: 120_000,
342
+ },
343
+ ],
344
+ });
345
+ ```
346
+
347
+ ## Validating Mock Accuracy
348
+
349
+ Guard against mock drift from real APIs:
350
+
351
+ ```typescript
352
+ test.describe('contract validation', () => {
353
+ test('billing mock matches real API shape', async ({ request }) => {
354
+ const realResponse = await request.post('/api/billing/charge', {
355
+ data: { amount: 5000, currency: 'usd' },
356
+ });
357
+ const realBody = await realResponse.json();
358
+
359
+ const mockBody = {
360
+ status: 'paid',
361
+ id: 'inv_mock_789',
362
+ };
363
+
364
+ expect(Object.keys(mockBody).sort()).toEqual(Object.keys(realBody).sort());
365
+
366
+ for (const key of Object.keys(mockBody)) {
367
+ expect(typeof mockBody[key]).toBe(typeof realBody[key]);
368
+ }
369
+ });
370
+ });
371
+ ```
372
+
373
+ ## Anti-Patterns
374
+
375
+ | Don't Do This | Problem | Do This Instead |
376
+ | --- | --- | --- |
377
+ | Mock your own API | Tests pass, app breaks. Zero integration coverage. | Hit your real API. Mock only third-party services. |
378
+ | Mock everything for speed | You test a fiction. Frontend and backend may be incompatible. | Mock only external boundaries. |
379
+ | Never mock anything | Tests are slow, flaky, fail when third parties have outages. | Mock third-party services. |
380
+ | Use outdated mocks | Mock returns different shape than real API. | Run contract validation tests. Re-record HAR files regularly. |
381
+ | Mock with `page.evaluate()` to stub fetch | Fragile, doesn't survive navigation. | Use `page.route()` which intercepts at network layer. |
382
+ | Copy-paste mocks across files | One API change requires updating many files. | Centralize mocks in fixtures. |
383
+ | Block all network and whitelist | Extremely brittle. Every new endpoint requires update. | Allow all by default. Selectively mock third-party services. |