stably 4.8.9 → 4.10.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 (47) hide show
  1. package/dist/index.mjs +1 -1
  2. package/dist/stably-plugin-cli/.claude-plugin/plugin.json +5 -0
  3. package/dist/stably-plugin-cli/skills/bash-commands/SKILL.md +65 -0
  4. package/dist/stably-plugin-cli/skills/browser-interaction-guide/SKILL.md +144 -0
  5. package/dist/stably-plugin-cli/skills/bulk-test-handling/SKILL.md +104 -0
  6. package/dist/stably-plugin-cli/skills/debugging-test-failures/SKILL.md +146 -0
  7. package/dist/{stably-plugin → stably-plugin-cli}/skills/playwright-best-practices/SKILL.md +11 -5
  8. package/dist/stably-plugin-cli/skills/playwright-config-auth/SKILL.md +217 -0
  9. package/dist/stably-plugin-cli/skills/stably-sdk-reference/SKILL.md +307 -0
  10. package/dist/stably-plugin-cli/skills/test-creation-workflow/SKILL.md +311 -0
  11. package/package.json +4 -1
  12. package/dist/stably-plugin/.claude-plugin/plugin.json +0 -5
  13. package/dist/stably-plugin/skills/playwright-best-practices/references/accessibility.md +0 -359
  14. package/dist/stably-plugin/skills/playwright-best-practices/references/annotations.md +0 -526
  15. package/dist/stably-plugin/skills/playwright-best-practices/references/assertions-waiting.md +0 -361
  16. package/dist/stably-plugin/skills/playwright-best-practices/references/browser-apis.md +0 -391
  17. package/dist/stably-plugin/skills/playwright-best-practices/references/browser-extensions.md +0 -506
  18. package/dist/stably-plugin/skills/playwright-best-practices/references/canvas-webgl.md +0 -493
  19. package/dist/stably-plugin/skills/playwright-best-practices/references/ci-cd.md +0 -407
  20. package/dist/stably-plugin/skills/playwright-best-practices/references/clock-mocking.md +0 -364
  21. package/dist/stably-plugin/skills/playwright-best-practices/references/component-testing.md +0 -500
  22. package/dist/stably-plugin/skills/playwright-best-practices/references/console-errors.md +0 -420
  23. package/dist/stably-plugin/skills/playwright-best-practices/references/debugging.md +0 -491
  24. package/dist/stably-plugin/skills/playwright-best-practices/references/electron.md +0 -509
  25. package/dist/stably-plugin/skills/playwright-best-practices/references/error-testing.md +0 -360
  26. package/dist/stably-plugin/skills/playwright-best-practices/references/file-operations.md +0 -375
  27. package/dist/stably-plugin/skills/playwright-best-practices/references/fixtures-hooks.md +0 -417
  28. package/dist/stably-plugin/skills/playwright-best-practices/references/flaky-tests.md +0 -494
  29. package/dist/stably-plugin/skills/playwright-best-practices/references/global-setup.md +0 -434
  30. package/dist/stably-plugin/skills/playwright-best-practices/references/i18n.md +0 -508
  31. package/dist/stably-plugin/skills/playwright-best-practices/references/iframes.md +0 -403
  32. package/dist/stably-plugin/skills/playwright-best-practices/references/locators.md +0 -242
  33. package/dist/stably-plugin/skills/playwright-best-practices/references/mobile-testing.md +0 -409
  34. package/dist/stably-plugin/skills/playwright-best-practices/references/multi-context.md +0 -288
  35. package/dist/stably-plugin/skills/playwright-best-practices/references/multi-user.md +0 -393
  36. package/dist/stably-plugin/skills/playwright-best-practices/references/network-advanced.md +0 -452
  37. package/dist/stably-plugin/skills/playwright-best-practices/references/page-object-model.md +0 -315
  38. package/dist/stably-plugin/skills/playwright-best-practices/references/performance-testing.md +0 -476
  39. package/dist/stably-plugin/skills/playwright-best-practices/references/performance.md +0 -453
  40. package/dist/stably-plugin/skills/playwright-best-practices/references/projects-dependencies.md +0 -456
  41. package/dist/stably-plugin/skills/playwright-best-practices/references/security-testing.md +0 -430
  42. package/dist/stably-plugin/skills/playwright-best-practices/references/service-workers.md +0 -504
  43. package/dist/stably-plugin/skills/playwright-best-practices/references/test-coverage.md +0 -495
  44. package/dist/stably-plugin/skills/playwright-best-practices/references/test-data.md +0 -492
  45. package/dist/stably-plugin/skills/playwright-best-practices/references/test-organization.md +0 -361
  46. package/dist/stably-plugin/skills/playwright-best-practices/references/third-party.md +0 -464
  47. package/dist/stably-plugin/skills/playwright-best-practices/references/websockets.md +0 -403
@@ -1,315 +0,0 @@
1
- # Page Object Model (POM)
2
-
3
- ## Table of Contents
4
-
5
- 1. [Overview](#overview)
6
- 2. [Basic Structure](#basic-structure)
7
- 3. [Component Objects](#component-objects)
8
- 4. [Composition Patterns](#composition-patterns)
9
- 5. [Factory Functions](#factory-functions)
10
- 6. [Best Practices](#best-practices)
11
-
12
- ## Overview
13
-
14
- Page Object Model encapsulates page structure and interactions, providing:
15
-
16
- - **Maintainability**: Change selectors in one place
17
- - **Reusability**: Share page interactions across tests
18
- - **Readability**: Tests express intent, not implementation
19
-
20
- ## Basic Structure
21
-
22
- ### Page Class
23
-
24
- ```typescript
25
- // pages/login.page.ts
26
- import { Page, Locator, expect } from "@playwright/test";
27
-
28
- export class LoginPage {
29
- readonly page: Page;
30
- readonly emailInput: Locator;
31
- readonly passwordInput: Locator;
32
- readonly submitButton: Locator;
33
- readonly errorMessage: Locator;
34
-
35
- constructor(page: Page) {
36
- this.page = page;
37
- this.emailInput = page.getByLabel("Email");
38
- this.passwordInput = page.getByLabel("Password");
39
- this.submitButton = page.getByRole("button", { name: "Sign in" });
40
- this.errorMessage = page.getByRole("alert");
41
- }
42
-
43
- async goto() {
44
- await this.page.goto("/login");
45
- }
46
-
47
- async login(email: string, password: string) {
48
- await this.emailInput.fill(email);
49
- await this.passwordInput.fill(password);
50
- await this.submitButton.click();
51
- }
52
-
53
- async expectError(message: string) {
54
- await expect(this.errorMessage).toContainText(message);
55
- }
56
- }
57
- ```
58
-
59
- ### Usage in Tests
60
-
61
- ```typescript
62
- // tests/login.spec.ts
63
- import { test, expect } from "@playwright/test";
64
- import { LoginPage } from "../pages/login.page";
65
-
66
- test.describe("Login", () => {
67
- test("successful login redirects to dashboard", async ({ page }) => {
68
- const loginPage = new LoginPage(page);
69
- await loginPage.goto();
70
- await loginPage.login("user@example.com", "password123");
71
- await expect(page).toHaveURL("/dashboard");
72
- });
73
-
74
- test("shows error for invalid credentials", async ({ page }) => {
75
- const loginPage = new LoginPage(page);
76
- await loginPage.goto();
77
- await loginPage.login("invalid@example.com", "wrong");
78
- await loginPage.expectError("Invalid credentials");
79
- });
80
- });
81
- ```
82
-
83
- ## Component Objects
84
-
85
- For reusable UI components:
86
-
87
- ```typescript
88
- // components/navbar.component.ts
89
- import { Page, Locator } from "@playwright/test";
90
-
91
- export class NavbarComponent {
92
- readonly container: Locator;
93
- readonly logo: Locator;
94
- readonly searchInput: Locator;
95
- readonly userMenu: Locator;
96
-
97
- constructor(page: Page) {
98
- this.container = page.getByRole("navigation");
99
- this.logo = this.container.getByRole("link", { name: "Home" });
100
- this.searchInput = this.container.getByRole("searchbox");
101
- this.userMenu = this.container.getByRole("button", { name: /user menu/i });
102
- }
103
-
104
- async search(query: string) {
105
- await this.searchInput.fill(query);
106
- await this.searchInput.press("Enter");
107
- }
108
-
109
- async openUserMenu() {
110
- await this.userMenu.click();
111
- }
112
- }
113
- ```
114
-
115
- ```typescript
116
- // components/modal.component.ts
117
- import { Locator, expect } from "@playwright/test";
118
-
119
- export class ModalComponent {
120
- readonly container: Locator;
121
- readonly title: Locator;
122
- readonly closeButton: Locator;
123
- readonly confirmButton: Locator;
124
-
125
- constructor(container: Locator) {
126
- this.container = container;
127
- this.title = container.getByRole("heading");
128
- this.closeButton = container.getByRole("button", { name: "Close" });
129
- this.confirmButton = container.getByRole("button", { name: "Confirm" });
130
- }
131
-
132
- async expectTitle(title: string) {
133
- await expect(this.title).toHaveText(title);
134
- }
135
-
136
- async close() {
137
- await this.closeButton.click();
138
- }
139
-
140
- async confirm() {
141
- await this.confirmButton.click();
142
- }
143
- }
144
- ```
145
-
146
- ## Composition Patterns
147
-
148
- ### Page with Components
149
-
150
- ```typescript
151
- // pages/dashboard.page.ts
152
- import { Page, Locator } from "@playwright/test";
153
- import { NavbarComponent } from "../components/navbar.component";
154
- import { ModalComponent } from "../components/modal.component";
155
-
156
- export class DashboardPage {
157
- readonly page: Page;
158
- readonly navbar: NavbarComponent;
159
- readonly newProjectButton: Locator;
160
-
161
- constructor(page: Page) {
162
- this.page = page;
163
- this.navbar = new NavbarComponent(page);
164
- this.newProjectButton = page.getByRole("button", { name: "New Project" });
165
- }
166
-
167
- async goto() {
168
- await this.page.goto("/dashboard");
169
- }
170
-
171
- async createProject() {
172
- await this.newProjectButton.click();
173
- return new ModalComponent(this.page.getByRole("dialog"));
174
- }
175
- }
176
- ```
177
-
178
- ### Page Navigation
179
-
180
- ```typescript
181
- // pages/base.page.ts
182
- import { Page } from "@playwright/test";
183
-
184
- export abstract class BasePage {
185
- constructor(readonly page: Page) {}
186
-
187
- abstract goto(): Promise<void>;
188
-
189
- async getTitle(): Promise<string> {
190
- return this.page.title();
191
- }
192
- }
193
- ```
194
-
195
- ```typescript
196
- // Return new page object on navigation
197
- export class LoginPage extends BasePage {
198
- async login(email: string, password: string): Promise<DashboardPage> {
199
- await this.emailInput.fill(email);
200
- await this.passwordInput.fill(password);
201
- await this.submitButton.click();
202
- return new DashboardPage(this.page);
203
- }
204
- }
205
-
206
- // Usage
207
- const loginPage = new LoginPage(page);
208
- await loginPage.goto();
209
- const dashboardPage = await loginPage.login("user@example.com", "pass");
210
- await dashboardPage.expectWelcomeMessage();
211
- ```
212
-
213
- ## Factory Functions
214
-
215
- Alternative to classes for simpler pages:
216
-
217
- ```typescript
218
- // pages/login.page.ts
219
- import { Page } from "@playwright/test";
220
-
221
- export function createLoginPage(page: Page) {
222
- const emailInput = page.getByLabel("Email");
223
- const passwordInput = page.getByLabel("Password");
224
- const submitButton = page.getByRole("button", { name: "Sign in" });
225
-
226
- return {
227
- goto: () => page.goto("/login"),
228
- login: async (email: string, password: string) => {
229
- await emailInput.fill(email);
230
- await passwordInput.fill(password);
231
- await submitButton.click();
232
- },
233
- emailInput,
234
- passwordInput,
235
- submitButton,
236
- };
237
- }
238
-
239
- // Usage
240
- const loginPage = createLoginPage(page);
241
- await loginPage.goto();
242
- await loginPage.login("user@example.com", "password");
243
- ```
244
-
245
- ## Best Practices
246
-
247
- ### Do
248
-
249
- - **Keep locators in page objects** - Single source of truth
250
- - **Return new page objects** when navigation occurs
251
- - **Expose elements** for custom assertions in tests
252
- - **Use descriptive method names** - `submitOrder()` not `clickButton()`
253
- - **Keep methods focused** - One action per method
254
-
255
- ### Don't
256
-
257
- - **Don't include assertions in page methods** (usually) - Keep in tests
258
- - **Don't expose implementation details** - Hide complex interactions
259
- - **Don't make page objects too large** - Split into components
260
- - **Don't share state** between page object instances
261
-
262
- ### Directory Structure
263
-
264
- ```
265
- tests/
266
- ├── pages/
267
- │ ├── base.page.ts
268
- │ ├── login.page.ts
269
- │ ├── dashboard.page.ts
270
- │ └── settings.page.ts
271
- ├── components/
272
- │ ├── navbar.component.ts
273
- │ ├── modal.component.ts
274
- │ └── table.component.ts
275
- ├── fixtures/
276
- │ └── pages.fixture.ts
277
- └── specs/
278
- ├── login.spec.ts
279
- └── dashboard.spec.ts
280
- ```
281
-
282
- ### Using with Fixtures
283
-
284
- ```typescript
285
- // fixtures/pages.fixture.ts
286
- import { test as base } from "@playwright/test";
287
- import { LoginPage } from "../pages/login.page";
288
- import { DashboardPage } from "../pages/dashboard.page";
289
-
290
- type Pages = {
291
- loginPage: LoginPage;
292
- dashboardPage: DashboardPage;
293
- };
294
-
295
- export const test = base.extend<Pages>({
296
- loginPage: async ({ page }, use) => {
297
- await use(new LoginPage(page));
298
- },
299
- dashboardPage: async ({ page }, use) => {
300
- await use(new DashboardPage(page));
301
- },
302
- });
303
-
304
- // Usage in tests
305
- test("can login", async ({ loginPage }) => {
306
- await loginPage.goto();
307
- await loginPage.login("user@example.com", "password");
308
- });
309
- ```
310
-
311
- ## Related References
312
-
313
- - **Locator strategies**: See [locators.md](locators.md) for selecting elements
314
- - **Fixtures**: See [fixtures-hooks.md](fixtures-hooks.md) for advanced fixture patterns
315
- - **Test organization**: See [test-organization.md](test-organization.md) for structuring test suites