@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,363 @@
1
+ # Organizing Reusable Test Code
2
+
3
+ ## Table of Contents
4
+
5
+ 1. [Pattern Comparison](#pattern-comparison)
6
+ 2. [Selection Flowchart](#selection-flowchart)
7
+ 3. [Page Objects](#page-objects)
8
+ 4. [Custom Fixtures](#custom-fixtures)
9
+ 5. [Helper Functions](#helper-functions)
10
+ 6. [Combined Project Structure](#combined-project-structure)
11
+ 7. [Anti-Patterns](#anti-patterns)
12
+
13
+ Use all three patterns together. Most projects benefit from a hybrid approach:
14
+
15
+ - **Page objects** for UI interaction (pages/components with 5+ interactions)
16
+ - **Custom fixtures** for test infrastructure (auth state, database, API clients, anything with lifecycle)
17
+ - **Helper functions** for stateless utilities (generate data, format values, simple waits)
18
+
19
+ If only using one pattern, choose **custom fixtures** — they handle setup/teardown, compose well, and Playwright is built around them.
20
+
21
+ ## Pattern Comparison
22
+
23
+ | Aspect | Page Objects | Custom Fixtures | Helper Functions |
24
+ |---|---|---|---|
25
+ | **Purpose** | Encapsulate UI interactions | Provide resources with setup/teardown | Stateless utilities |
26
+ | **Lifecycle** | Manual (constructor/methods) | Built-in (`use()` with automatic teardown) | None |
27
+ | **Composability** | Constructor injection or fixture wiring | Depend on other fixtures | Call other functions |
28
+ | **Best for** | Pages with many reused interactions | Resources needing setup AND teardown | Simple logic with no side effects |
29
+
30
+ ## Selection Flowchart
31
+
32
+ ```text
33
+ What kind of reusable code?
34
+ |
35
+ +-- Interacts with browser page/component?
36
+ | |
37
+ | +-- Has 5+ interactions (fill, click, navigate, assert)?
38
+ | | +-- YES: Used in 3+ test files?
39
+ | | | +-- YES --> PAGE OBJECT
40
+ | | | +-- NO --> Inline or small helper
41
+ | | +-- NO --> HELPER FUNCTION
42
+ | |
43
+ | +-- Needs setup before AND cleanup after test?
44
+ | +-- YES --> CUSTOM FIXTURE
45
+ | +-- NO --> PAGE OBJECT method or HELPER
46
+ |
47
+ +-- Manages resource with lifecycle (create/destroy)?
48
+ | +-- Examples: auth state, DB connection, API client, test user
49
+ | +-- YES --> CUSTOM FIXTURE (always)
50
+ |
51
+ +-- Stateless utility? (no browser, no side effects)
52
+ | +-- Examples: random email, format date, build URL, parse response
53
+ | +-- YES --> HELPER FUNCTION
54
+ |
55
+ +-- Not sure?
56
+ +-- Start with HELPER FUNCTION
57
+ +-- Promote to PAGE OBJECT when interactions grow
58
+ +-- Promote to FIXTURE when lifecycle needed
59
+ ```
60
+
61
+ ## Page Objects
62
+
63
+ Best for pages/components with 5+ interactions appearing in 3+ test files.
64
+
65
+ ```typescript
66
+ // page-objects/booking.page.ts
67
+ import { type Page, type Locator, expect } from '@playwright/test';
68
+
69
+ export class BookingPage {
70
+ readonly page: Page;
71
+ readonly dateField: Locator;
72
+ readonly guestCount: Locator;
73
+ readonly roomType: Locator;
74
+ readonly reserveBtn: Locator;
75
+ readonly totalPrice: Locator;
76
+
77
+ constructor(page: Page) {
78
+ this.page = page;
79
+ this.dateField = page.getByLabel('Check-in date');
80
+ this.guestCount = page.getByLabel('Guests');
81
+ this.roomType = page.getByLabel('Room type');
82
+ this.reserveBtn = page.getByRole('button', { name: 'Reserve' });
83
+ this.totalPrice = page.getByTestId('total-price');
84
+ }
85
+
86
+ async goto() {
87
+ await this.page.goto('/booking');
88
+ }
89
+
90
+ async fillDetails(opts: { date: string; guests: number; room: string }) {
91
+ await this.dateField.fill(opts.date);
92
+ await this.guestCount.fill(String(opts.guests));
93
+ await this.roomType.selectOption(opts.room);
94
+ }
95
+
96
+ async reserve() {
97
+ await this.reserveBtn.click();
98
+ await this.page.waitForURL('**/confirmation');
99
+ }
100
+
101
+ async expectPrice(amount: string) {
102
+ await expect(this.totalPrice).toHaveText(amount);
103
+ }
104
+ }
105
+ ```
106
+
107
+ ```typescript
108
+ // tests/booking/reservation.spec.ts
109
+ import { test, expect } from '@playwright/test';
110
+ import { BookingPage } from '../page-objects/booking.page';
111
+
112
+ test('complete reservation with standard room', async ({ page }) => {
113
+ const booking = new BookingPage(page);
114
+ await booking.goto();
115
+ await booking.fillDetails({ date: '2026-03-15', guests: 2, room: 'standard' });
116
+ await booking.reserve();
117
+ await expect(page.getByText('Reservation confirmed')).toBeVisible();
118
+ });
119
+ ```
120
+
121
+ **Page object principles:**
122
+ - One class per logical page/component, not per URL
123
+ - Constructor takes `Page`
124
+ - Locators as `readonly` properties in constructor
125
+ - Methods represent user intent (`reserve`, `fillDetails`), not low-level clicks
126
+ - Navigation methods (`goto`) belong on the page object
127
+
128
+ ## Custom Fixtures
129
+
130
+ Best for resources needing setup before and teardown after tests — auth state, database connections, API clients, test users.
131
+
132
+ ```typescript
133
+ // fixtures/base.fixture.ts
134
+ import { test as base, expect } from '@playwright/test';
135
+ import { BookingPage } from '../page-objects/booking.page';
136
+ import { generateMember } from '../helpers/data';
137
+
138
+ type Fixtures = {
139
+ bookingPage: BookingPage;
140
+ member: { email: string; password: string; id: string };
141
+ loggedInPage: import('@playwright/test').Page;
142
+ };
143
+
144
+ export const test = base.extend<Fixtures>({
145
+ bookingPage: async ({ page }, use) => {
146
+ await use(new BookingPage(page));
147
+ },
148
+
149
+ member: async ({ request }, use) => {
150
+ const data = generateMember();
151
+ const res = await request.post('/api/test/members', { data });
152
+ const member = await res.json();
153
+ await use(member);
154
+ await request.delete(`/api/test/members/${member.id}`);
155
+ },
156
+
157
+ loggedInPage: async ({ page, member }, use) => {
158
+ await page.goto('/login');
159
+ await page.getByLabel('Email').fill(member.email);
160
+ await page.getByLabel('Password').fill(member.password);
161
+ await page.getByRole('button', { name: 'Sign in' }).click();
162
+ await expect(page).toHaveURL('/dashboard');
163
+ await use(page);
164
+ },
165
+ });
166
+
167
+ export { expect } from '@playwright/test';
168
+ ```
169
+
170
+ ```typescript
171
+ // tests/dashboard/overview.spec.ts
172
+ import { test, expect } from '../../fixtures/base.fixture';
173
+
174
+ test('member sees dashboard widgets', async ({ loggedInPage }) => {
175
+ await expect(loggedInPage.getByRole('heading', { name: 'Dashboard' })).toBeVisible();
176
+ await expect(loggedInPage.getByTestId('stats-widget')).toBeVisible();
177
+ });
178
+
179
+ test('new member sees welcome prompt', async ({ loggedInPage, member }) => {
180
+ await expect(loggedInPage.getByText(`Welcome, ${member.email}`)).toBeVisible();
181
+ });
182
+ ```
183
+
184
+ **Fixture principles:**
185
+ - Use `test.extend()` — never module-level variables
186
+ - `use()` callback separates setup from teardown
187
+ - Teardown runs even if test fails
188
+ - Fixtures compose: one can depend on another
189
+ - Fixtures are lazy: created only when requested
190
+ - Wrap page objects in fixtures for lifecycle management
191
+
192
+ ## Helper Functions
193
+
194
+ Best for stateless utilities — generating test data, formatting values, building URLs, parsing responses.
195
+
196
+ ```typescript
197
+ // helpers/data.ts
198
+ import { randomUUID } from 'node:crypto';
199
+
200
+ export function generateEmail(prefix = 'user'): string {
201
+ return `${prefix}-${Date.now()}-${randomUUID().slice(0, 8)}@test.local`;
202
+ }
203
+
204
+ export function generateMember(overrides: Partial<Member> = {}): Member {
205
+ return {
206
+ email: generateEmail(),
207
+ password: 'SecurePass456!',
208
+ name: 'Test Member',
209
+ ...overrides,
210
+ };
211
+ }
212
+
213
+ interface Member {
214
+ email: string;
215
+ password: string;
216
+ name: string;
217
+ }
218
+
219
+ export function formatPrice(cents: number): string {
220
+ return `$${(cents / 100).toFixed(2)}`;
221
+ }
222
+ ```
223
+
224
+ ```typescript
225
+ // helpers/assertions.ts
226
+ import { type Page, expect } from '@playwright/test';
227
+
228
+ export async function expectNotification(page: Page, message: string): Promise<void> {
229
+ const notification = page.getByRole('alert').filter({ hasText: message });
230
+ await expect(notification).toBeVisible();
231
+ await expect(notification).toBeHidden({ timeout: 10000 });
232
+ }
233
+ ```
234
+
235
+ ```typescript
236
+ // tests/settings/account.spec.ts
237
+ import { test, expect } from '@playwright/test';
238
+ import { generateEmail } from '../../helpers/data';
239
+ import { expectNotification } from '../../helpers/assertions';
240
+
241
+ test('update account email', async ({ page }) => {
242
+ const newEmail = generateEmail('updated');
243
+ await page.goto('/settings/account');
244
+ await page.getByLabel('Email').fill(newEmail);
245
+ await page.getByRole('button', { name: 'Save' }).click();
246
+ await expectNotification(page, 'Account updated');
247
+ await expect(page.getByLabel('Email')).toHaveValue(newEmail);
248
+ });
249
+ ```
250
+
251
+ **Helper principles:**
252
+ - Pure functions with no side effects
253
+ - No browser state — take `page` as parameter if needed
254
+ - Promote to fixture if setup/teardown needed
255
+ - Promote to page object if many page interactions grow
256
+ - Keep small and focused
257
+
258
+ ## Combined Project Structure
259
+
260
+ ```text
261
+ tests/
262
+ +-- fixtures/
263
+ | +-- auth.fixture.ts
264
+ | +-- db.fixture.ts
265
+ | +-- base.fixture.ts
266
+ +-- page-objects/
267
+ | +-- login.page.ts
268
+ | +-- booking.page.ts
269
+ | +-- components/
270
+ | +-- data-table.component.ts
271
+ +-- helpers/
272
+ | +-- data.ts
273
+ | +-- assertions.ts
274
+ +-- e2e/
275
+ | +-- auth/
276
+ | | +-- login.spec.ts
277
+ | +-- booking/
278
+ | +-- reservation.spec.ts
279
+ playwright.config.ts
280
+ ```
281
+
282
+ **Layer responsibilities:**
283
+
284
+ | Layer | Pattern | Responsibility |
285
+ |---|---|---|
286
+ | **Test file** | `test()` | Describes behavior, orchestrates layers |
287
+ | **Fixtures** | `test.extend()` | Resource lifecycle — setup, provide, teardown |
288
+ | **Page objects** | Classes | UI interaction — navigation, actions, locators |
289
+ | **Helpers** | Functions | Utilities — data generation, formatting, assertions |
290
+
291
+ ## Anti-Patterns
292
+
293
+ ### Page object managing resources
294
+
295
+ ```typescript
296
+ // BAD: page object handling API calls and database
297
+ class LoginPage {
298
+ async createUser() { /* API call */ }
299
+ async deleteUser() { /* API call */ }
300
+ async signIn(email: string, password: string) { /* UI */ }
301
+ }
302
+ ```
303
+
304
+ Resource lifecycle belongs in fixtures where teardown is guaranteed. Keep only `signIn` in the page object.
305
+
306
+ ### Locator-only page objects
307
+
308
+ ```typescript
309
+ // BAD: no methods, just locators
310
+ class LoginPage {
311
+ emailInput = this.page.getByLabel('Email');
312
+ passwordInput = this.page.getByLabel('Password');
313
+ submitBtn = this.page.getByRole('button', { name: 'Sign in' });
314
+ constructor(private page: Page) {}
315
+ }
316
+ ```
317
+
318
+ Add intent-revealing methods or skip the page object entirely.
319
+
320
+ ### Monolithic fixtures
321
+
322
+ ```typescript
323
+ // BAD: one fixture doing everything
324
+ test.extend({
325
+ everything: async ({ page, request }, use) => {
326
+ const user = await createUser(request);
327
+ const products = await seedProducts(request, 50);
328
+ await setupPayment(request, user.id);
329
+ await page.goto('/dashboard');
330
+ await use({ user, products, page });
331
+ // massive teardown...
332
+ },
333
+ });
334
+ ```
335
+
336
+ Break into small, composable fixtures. Each fixture does one thing.
337
+
338
+ ### Helpers with side effects
339
+
340
+ ```typescript
341
+ // BAD: module-level state
342
+ let createdUserId: string;
343
+
344
+ export async function createTestUser(request: APIRequestContext) {
345
+ const res = await request.post('/api/users', { data: { email: 'test@example.com' } });
346
+ const user = await res.json();
347
+ createdUserId = user.id; // shared across tests!
348
+ return user;
349
+ }
350
+ ```
351
+
352
+ Module-level state leaks between parallel tests. If it has side effects and needs cleanup, make it a fixture.
353
+
354
+ ### Over-abstracting simple operations
355
+
356
+ ```typescript
357
+ // BAD: helper for one-liner
358
+ export async function clickButton(page: Page, name: string) {
359
+ await page.getByRole('button', { name }).click();
360
+ }
361
+ ```
362
+
363
+ Only abstract when there is real duplication (3+ usages) or complexity (5+ interactions).