Initial commit
This commit is contained in:
11
.claude-plugin/plugin.json
Normal file
11
.claude-plugin/plugin.json
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
{
|
||||||
|
"name": "playwright-skill",
|
||||||
|
"description": "Claude Code Skill for general-purpose browser automation with Playwright. Auto-detects dev servers, writes clean test scripts to /tmp, and autonomously handles any browser automation task.",
|
||||||
|
"version": "4.0.2",
|
||||||
|
"author": {
|
||||||
|
"name": "lackeyjb"
|
||||||
|
},
|
||||||
|
"skills": [
|
||||||
|
"./skills"
|
||||||
|
]
|
||||||
|
}
|
||||||
3
README.md
Normal file
3
README.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
# playwright-skill
|
||||||
|
|
||||||
|
Claude Code Skill for general-purpose browser automation with Playwright. Auto-detects dev servers, writes clean test scripts to /tmp, and autonomously handles any browser automation task.
|
||||||
57
plugin.lock.json
Normal file
57
plugin.lock.json
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
{
|
||||||
|
"$schema": "internal://schemas/plugin.lock.v1.json",
|
||||||
|
"pluginId": "gh:lackeyjb/playwright-skill:",
|
||||||
|
"normalized": {
|
||||||
|
"repo": null,
|
||||||
|
"ref": "refs/tags/v20251128.0",
|
||||||
|
"commit": "cb47e7c22f0702dd37d5a876c4c8df0580a0341a",
|
||||||
|
"treeHash": "e95a97ff554cd8485f73790ee2075249e7fd3bcc9c06d5f2fb0e8b33b9d05b30",
|
||||||
|
"generatedAt": "2025-11-28T10:20:02.597466Z",
|
||||||
|
"toolVersion": "publish_plugins.py@0.2.0"
|
||||||
|
},
|
||||||
|
"origin": {
|
||||||
|
"remote": "git@github.com:zhongweili/42plugin-data.git",
|
||||||
|
"branch": "master",
|
||||||
|
"commit": "aa1497ed0949fd50e99e70d6324a29c5b34f9390",
|
||||||
|
"repoRoot": "/Users/zhongweili/projects/openmind/42plugin-data"
|
||||||
|
},
|
||||||
|
"manifest": {
|
||||||
|
"name": "playwright-skill",
|
||||||
|
"description": "Claude Code Skill for general-purpose browser automation with Playwright. Auto-detects dev servers, writes clean test scripts to /tmp, and autonomously handles any browser automation task.",
|
||||||
|
"version": "4.0.2"
|
||||||
|
},
|
||||||
|
"content": {
|
||||||
|
"files": [
|
||||||
|
{
|
||||||
|
"path": "README.md",
|
||||||
|
"sha256": "b4a49d705042a741ea3e983f62cbea4b5b31a1a8b24c594000b55109692fb144"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": ".claude-plugin/plugin.json",
|
||||||
|
"sha256": "69bb41f3e66d2abe7ab1a233255b2dab6cb5827ddb6806d078ac2a8a4376440c"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/playwright-skill/run.js",
|
||||||
|
"sha256": "a86f72fc97251c520a973ff16351c84afb44d99397e8a0ae60cc30c8270468e0"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/playwright-skill/package.json",
|
||||||
|
"sha256": "7f93bd576b91118cd43a7360eec7d282217fa5eb3ca2887a4b32658910669d09"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/playwright-skill/SKILL.md",
|
||||||
|
"sha256": "0662093a4d482f12a384545a6c9fb2856144e9ab6f5c6b48eed0517926a816d9"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "skills/playwright-skill/lib/helpers.js",
|
||||||
|
"sha256": "109be8cef8254f4ce3f0453f66a5915ef908229eede4af9c3474929fe8f2a6d1"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"dirSha256": "e95a97ff554cd8485f73790ee2075249e7fd3bcc9c06d5f2fb0e8b33b9d05b30"
|
||||||
|
},
|
||||||
|
"security": {
|
||||||
|
"scannedAt": null,
|
||||||
|
"scannerVersion": null,
|
||||||
|
"flags": []
|
||||||
|
}
|
||||||
|
}
|
||||||
406
skills/playwright-skill/SKILL.md
Normal file
406
skills/playwright-skill/SKILL.md
Normal file
@@ -0,0 +1,406 @@
|
|||||||
|
---
|
||||||
|
name: Playwright Browser Automation
|
||||||
|
description: Complete browser automation with Playwright. Auto-detects dev servers, writes clean test scripts to /tmp. Test pages, fill forms, take screenshots, check responsive design, validate UX, test login flows, check links, automate any browser task. Use when user wants to test websites, automate browser interactions, validate web functionality, or perform any browser-based testing.
|
||||||
|
version: 4.0.0
|
||||||
|
author: Claude Assistant
|
||||||
|
tags: [testing, automation, browser, e2e, playwright, web-testing]
|
||||||
|
---
|
||||||
|
|
||||||
|
**IMPORTANT - Path Resolution:**
|
||||||
|
This skill can be installed in different locations (plugin system, manual installation, global, or project-specific). Before executing any commands, determine the skill directory based on where you loaded this SKILL.md file, and use that path in all commands below. Replace `$SKILL_DIR` with the actual discovered path.
|
||||||
|
|
||||||
|
Common installation paths:
|
||||||
|
- Plugin system: `~/.claude/plugins/marketplaces/playwright-skill/skills/playwright-skill`
|
||||||
|
- Manual global: `~/.claude/skills/playwright-skill`
|
||||||
|
- Project-specific: `<project>/.claude/skills/playwright-skill`
|
||||||
|
|
||||||
|
# Playwright Browser Automation
|
||||||
|
|
||||||
|
General-purpose browser automation skill. I'll write custom Playwright code for any automation task you request and execute it via the universal executor.
|
||||||
|
|
||||||
|
**CRITICAL WORKFLOW - Follow these steps in order:**
|
||||||
|
|
||||||
|
1. **Auto-detect dev servers** - For localhost testing, ALWAYS run server detection FIRST:
|
||||||
|
```bash
|
||||||
|
cd $SKILL_DIR && node -e "require('./lib/helpers').detectDevServers().then(servers => console.log(JSON.stringify(servers)))"
|
||||||
|
```
|
||||||
|
- If **1 server found**: Use it automatically, inform user
|
||||||
|
- If **multiple servers found**: Ask user which one to test
|
||||||
|
- If **no servers found**: Ask for URL or offer to help start dev server
|
||||||
|
|
||||||
|
2. **Write scripts to /tmp** - NEVER write test files to skill directory; always use `/tmp/playwright-test-*.js`
|
||||||
|
|
||||||
|
3. **Use visible browser by default** - Always use `headless: false` unless user specifically requests headless mode
|
||||||
|
|
||||||
|
4. **Parameterize URLs** - Always make URLs configurable via environment variable or constant at top of script
|
||||||
|
|
||||||
|
## How It Works
|
||||||
|
|
||||||
|
1. You describe what you want to test/automate
|
||||||
|
2. I auto-detect running dev servers (or ask for URL if testing external site)
|
||||||
|
3. I write custom Playwright code in `/tmp/playwright-test-*.js` (won't clutter your project)
|
||||||
|
4. I execute it via: `cd $SKILL_DIR && node run.js /tmp/playwright-test-*.js`
|
||||||
|
5. Results displayed in real-time, browser window visible for debugging
|
||||||
|
6. Test files auto-cleaned from /tmp by your OS
|
||||||
|
|
||||||
|
## Setup (First Time)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd $SKILL_DIR
|
||||||
|
npm run setup
|
||||||
|
```
|
||||||
|
|
||||||
|
This installs Playwright and Chromium browser. Only needed once.
|
||||||
|
|
||||||
|
## Execution Pattern
|
||||||
|
|
||||||
|
**Step 1: Detect dev servers (for localhost testing)**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd $SKILL_DIR && node -e "require('./lib/helpers').detectDevServers().then(s => console.log(JSON.stringify(s)))"
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 2: Write test script to /tmp with URL parameter**
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// /tmp/playwright-test-page.js
|
||||||
|
const { chromium } = require('playwright');
|
||||||
|
|
||||||
|
// Parameterized URL (detected or user-provided)
|
||||||
|
const TARGET_URL = 'http://localhost:3001'; // <-- Auto-detected or from user
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const browser = await chromium.launch({ headless: false });
|
||||||
|
const page = await browser.newPage();
|
||||||
|
|
||||||
|
await page.goto(TARGET_URL);
|
||||||
|
console.log('Page loaded:', await page.title());
|
||||||
|
|
||||||
|
await page.screenshot({ path: '/tmp/screenshot.png', fullPage: true });
|
||||||
|
console.log('📸 Screenshot saved to /tmp/screenshot.png');
|
||||||
|
|
||||||
|
await browser.close();
|
||||||
|
})();
|
||||||
|
```
|
||||||
|
|
||||||
|
**Step 3: Execute from skill directory**
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd $SKILL_DIR && node run.js /tmp/playwright-test-page.js
|
||||||
|
```
|
||||||
|
|
||||||
|
## Common Patterns
|
||||||
|
|
||||||
|
### Test a Page (Multiple Viewports)
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// /tmp/playwright-test-responsive.js
|
||||||
|
const { chromium } = require('playwright');
|
||||||
|
|
||||||
|
const TARGET_URL = 'http://localhost:3001'; // Auto-detected
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const browser = await chromium.launch({ headless: false, slowMo: 100 });
|
||||||
|
const page = await browser.newPage();
|
||||||
|
|
||||||
|
// Desktop test
|
||||||
|
await page.setViewportSize({ width: 1920, height: 1080 });
|
||||||
|
await page.goto(TARGET_URL);
|
||||||
|
console.log('Desktop - Title:', await page.title());
|
||||||
|
await page.screenshot({ path: '/tmp/desktop.png', fullPage: true });
|
||||||
|
|
||||||
|
// Mobile test
|
||||||
|
await page.setViewportSize({ width: 375, height: 667 });
|
||||||
|
await page.screenshot({ path: '/tmp/mobile.png', fullPage: true });
|
||||||
|
|
||||||
|
await browser.close();
|
||||||
|
})();
|
||||||
|
```
|
||||||
|
|
||||||
|
### Test Login Flow
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// /tmp/playwright-test-login.js
|
||||||
|
const { chromium } = require('playwright');
|
||||||
|
|
||||||
|
const TARGET_URL = 'http://localhost:3001'; // Auto-detected
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const browser = await chromium.launch({ headless: false });
|
||||||
|
const page = await browser.newPage();
|
||||||
|
|
||||||
|
await page.goto(`${TARGET_URL}/login`);
|
||||||
|
|
||||||
|
await page.fill('input[name="email"]', 'test@example.com');
|
||||||
|
await page.fill('input[name="password"]', 'password123');
|
||||||
|
await page.click('button[type="submit"]');
|
||||||
|
|
||||||
|
// Wait for redirect
|
||||||
|
await page.waitForURL('**/dashboard');
|
||||||
|
console.log('✅ Login successful, redirected to dashboard');
|
||||||
|
|
||||||
|
await browser.close();
|
||||||
|
})();
|
||||||
|
```
|
||||||
|
|
||||||
|
### Fill and Submit Form
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// /tmp/playwright-test-form.js
|
||||||
|
const { chromium } = require('playwright');
|
||||||
|
|
||||||
|
const TARGET_URL = 'http://localhost:3001'; // Auto-detected
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const browser = await chromium.launch({ headless: false, slowMo: 50 });
|
||||||
|
const page = await browser.newPage();
|
||||||
|
|
||||||
|
await page.goto(`${TARGET_URL}/contact`);
|
||||||
|
|
||||||
|
await page.fill('input[name="name"]', 'John Doe');
|
||||||
|
await page.fill('input[name="email"]', 'john@example.com');
|
||||||
|
await page.fill('textarea[name="message"]', 'Test message');
|
||||||
|
await page.click('button[type="submit"]');
|
||||||
|
|
||||||
|
// Verify submission
|
||||||
|
await page.waitForSelector('.success-message');
|
||||||
|
console.log('✅ Form submitted successfully');
|
||||||
|
|
||||||
|
await browser.close();
|
||||||
|
})();
|
||||||
|
```
|
||||||
|
|
||||||
|
### Check for Broken Links
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
const { chromium } = require('playwright');
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const browser = await chromium.launch({ headless: false });
|
||||||
|
const page = await browser.newPage();
|
||||||
|
|
||||||
|
await page.goto('http://localhost:3000');
|
||||||
|
|
||||||
|
const links = await page.locator('a[href^="http"]').all();
|
||||||
|
const results = { working: 0, broken: [] };
|
||||||
|
|
||||||
|
for (const link of links) {
|
||||||
|
const href = await link.getAttribute('href');
|
||||||
|
try {
|
||||||
|
const response = await page.request.head(href);
|
||||||
|
if (response.ok()) {
|
||||||
|
results.working++;
|
||||||
|
} else {
|
||||||
|
results.broken.push({ url: href, status: response.status() });
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
results.broken.push({ url: href, error: e.message });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`✅ Working links: ${results.working}`);
|
||||||
|
console.log(`❌ Broken links:`, results.broken);
|
||||||
|
|
||||||
|
await browser.close();
|
||||||
|
})();
|
||||||
|
```
|
||||||
|
|
||||||
|
### Take Screenshot with Error Handling
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
const { chromium } = require('playwright');
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const browser = await chromium.launch({ headless: false });
|
||||||
|
const page = await browser.newPage();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await page.goto('http://localhost:3000', {
|
||||||
|
waitUntil: 'networkidle',
|
||||||
|
timeout: 10000
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.screenshot({
|
||||||
|
path: '/tmp/screenshot.png',
|
||||||
|
fullPage: true
|
||||||
|
});
|
||||||
|
|
||||||
|
console.log('📸 Screenshot saved to /tmp/screenshot.png');
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Error:', error.message);
|
||||||
|
} finally {
|
||||||
|
await browser.close();
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
```
|
||||||
|
|
||||||
|
### Test Responsive Design
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
// /tmp/playwright-test-responsive-full.js
|
||||||
|
const { chromium } = require('playwright');
|
||||||
|
|
||||||
|
const TARGET_URL = 'http://localhost:3001'; // Auto-detected
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
const browser = await chromium.launch({ headless: false });
|
||||||
|
const page = await browser.newPage();
|
||||||
|
|
||||||
|
const viewports = [
|
||||||
|
{ name: 'Desktop', width: 1920, height: 1080 },
|
||||||
|
{ name: 'Tablet', width: 768, height: 1024 },
|
||||||
|
{ name: 'Mobile', width: 375, height: 667 }
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const viewport of viewports) {
|
||||||
|
console.log(`Testing ${viewport.name} (${viewport.width}x${viewport.height})`);
|
||||||
|
|
||||||
|
await page.setViewportSize({
|
||||||
|
width: viewport.width,
|
||||||
|
height: viewport.height
|
||||||
|
});
|
||||||
|
|
||||||
|
await page.goto(TARGET_URL);
|
||||||
|
await page.waitForTimeout(1000);
|
||||||
|
|
||||||
|
await page.screenshot({
|
||||||
|
path: `/tmp/${viewport.name.toLowerCase()}.png`,
|
||||||
|
fullPage: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log('✅ All viewports tested');
|
||||||
|
await browser.close();
|
||||||
|
})();
|
||||||
|
```
|
||||||
|
|
||||||
|
## Inline Execution (Simple Tasks)
|
||||||
|
|
||||||
|
For quick one-off tasks, you can execute code inline without creating files:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Take a quick screenshot
|
||||||
|
cd $SKILL_DIR && node run.js "
|
||||||
|
const browser = await chromium.launch({ headless: false });
|
||||||
|
const page = await browser.newPage();
|
||||||
|
await page.goto('http://localhost:3001');
|
||||||
|
await page.screenshot({ path: '/tmp/quick-screenshot.png', fullPage: true });
|
||||||
|
console.log('Screenshot saved');
|
||||||
|
await browser.close();
|
||||||
|
"
|
||||||
|
```
|
||||||
|
|
||||||
|
**When to use inline vs files:**
|
||||||
|
- **Inline**: Quick one-off tasks (screenshot, check if element exists, get page title)
|
||||||
|
- **Files**: Complex tests, responsive design checks, anything user might want to re-run
|
||||||
|
|
||||||
|
## Available Helpers
|
||||||
|
|
||||||
|
Optional utility functions in `lib/helpers.js`:
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
const helpers = require('./lib/helpers');
|
||||||
|
|
||||||
|
// Detect running dev servers (CRITICAL - use this first!)
|
||||||
|
const servers = await helpers.detectDevServers();
|
||||||
|
console.log('Found servers:', servers);
|
||||||
|
|
||||||
|
// Safe click with retry
|
||||||
|
await helpers.safeClick(page, 'button.submit', { retries: 3 });
|
||||||
|
|
||||||
|
// Safe type with clear
|
||||||
|
await helpers.safeType(page, '#username', 'testuser');
|
||||||
|
|
||||||
|
// Take timestamped screenshot
|
||||||
|
await helpers.takeScreenshot(page, 'test-result');
|
||||||
|
|
||||||
|
// Handle cookie banners
|
||||||
|
await helpers.handleCookieBanner(page);
|
||||||
|
|
||||||
|
// Extract table data
|
||||||
|
const data = await helpers.extractTableData(page, 'table.results');
|
||||||
|
```
|
||||||
|
|
||||||
|
See `lib/helpers.js` for full list.
|
||||||
|
|
||||||
|
## Advanced Usage
|
||||||
|
|
||||||
|
For comprehensive Playwright API documentation, see [API_REFERENCE.md](API_REFERENCE.md):
|
||||||
|
|
||||||
|
- Selectors & Locators best practices
|
||||||
|
- Network interception & API mocking
|
||||||
|
- Authentication & session management
|
||||||
|
- Visual regression testing
|
||||||
|
- Mobile device emulation
|
||||||
|
- Performance testing
|
||||||
|
- Debugging techniques
|
||||||
|
- CI/CD integration
|
||||||
|
|
||||||
|
## Tips
|
||||||
|
|
||||||
|
- **CRITICAL: Detect servers FIRST** - Always run `detectDevServers()` before writing test code for localhost testing
|
||||||
|
- **Use /tmp for test files** - Write to `/tmp/playwright-test-*.js`, never to skill directory or user's project
|
||||||
|
- **Parameterize URLs** - Put detected/provided URL in a `TARGET_URL` constant at the top of every script
|
||||||
|
- **DEFAULT: Visible browser** - Always use `headless: false` unless user explicitly asks for headless mode
|
||||||
|
- **Headless mode** - Only use `headless: true` when user specifically requests "headless" or "background" execution
|
||||||
|
- **Slow down:** Use `slowMo: 100` to make actions visible and easier to follow
|
||||||
|
- **Wait strategies:** Use `waitForURL`, `waitForSelector`, `waitForLoadState` instead of fixed timeouts
|
||||||
|
- **Error handling:** Always use try-catch for robust automation
|
||||||
|
- **Console output:** Use `console.log()` to track progress and show what's happening
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
|
||||||
|
**Playwright not installed:**
|
||||||
|
```bash
|
||||||
|
cd $SKILL_DIR && npm run setup
|
||||||
|
```
|
||||||
|
|
||||||
|
**Module not found:**
|
||||||
|
Ensure running from skill directory via `run.js` wrapper
|
||||||
|
|
||||||
|
**Browser doesn't open:**
|
||||||
|
Check `headless: false` and ensure display available
|
||||||
|
|
||||||
|
**Element not found:**
|
||||||
|
Add wait: `await page.waitForSelector('.element', { timeout: 10000 })`
|
||||||
|
|
||||||
|
## Example Usage
|
||||||
|
|
||||||
|
```
|
||||||
|
User: "Test if the marketing page looks good"
|
||||||
|
|
||||||
|
Claude: I'll test the marketing page across multiple viewports. Let me first detect running servers...
|
||||||
|
[Runs: detectDevServers()]
|
||||||
|
[Output: Found server on port 3001]
|
||||||
|
I found your dev server running on http://localhost:3001
|
||||||
|
|
||||||
|
[Writes custom automation script to /tmp/playwright-test-marketing.js with URL parameterized]
|
||||||
|
[Runs: cd $SKILL_DIR && node run.js /tmp/playwright-test-marketing.js]
|
||||||
|
[Shows results with screenshots from /tmp/]
|
||||||
|
```
|
||||||
|
|
||||||
|
```
|
||||||
|
User: "Check if login redirects correctly"
|
||||||
|
|
||||||
|
Claude: I'll test the login flow. First, let me check for running servers...
|
||||||
|
[Runs: detectDevServers()]
|
||||||
|
[Output: Found servers on ports 3000 and 3001]
|
||||||
|
I found 2 dev servers. Which one should I test?
|
||||||
|
- http://localhost:3000
|
||||||
|
- http://localhost:3001
|
||||||
|
|
||||||
|
User: "Use 3001"
|
||||||
|
|
||||||
|
[Writes login automation to /tmp/playwright-test-login.js]
|
||||||
|
[Runs: cd $SKILL_DIR && node run.js /tmp/playwright-test-login.js]
|
||||||
|
[Reports: ✅ Login successful, redirected to /dashboard]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Notes
|
||||||
|
|
||||||
|
- Each automation is custom-written for your specific request
|
||||||
|
- Not limited to pre-built scripts - any browser task possible
|
||||||
|
- Auto-detects running dev servers to eliminate hardcoded URLs
|
||||||
|
- Test scripts written to `/tmp` for automatic cleanup (no clutter)
|
||||||
|
- Code executes reliably with proper module resolution via `run.js`
|
||||||
|
- Progressive disclosure - API_REFERENCE.md loaded only when advanced features needed
|
||||||
26
skills/playwright-skill/package.json
Normal file
26
skills/playwright-skill/package.json
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
{
|
||||||
|
"name": "playwright-skill",
|
||||||
|
"version": "4.0.2",
|
||||||
|
"description": "General-purpose browser automation with Playwright for Claude Code with auto-detection and smart test management",
|
||||||
|
"author": "lackeyjb",
|
||||||
|
"main": "run.js",
|
||||||
|
"scripts": {
|
||||||
|
"setup": "npm install && npx playwright install chromium",
|
||||||
|
"install-all-browsers": "npx playwright install chromium firefox webkit"
|
||||||
|
},
|
||||||
|
"keywords": [
|
||||||
|
"playwright",
|
||||||
|
"automation",
|
||||||
|
"browser-testing",
|
||||||
|
"web-automation",
|
||||||
|
"claude-skill",
|
||||||
|
"general-purpose"
|
||||||
|
],
|
||||||
|
"dependencies": {
|
||||||
|
"playwright": "^1.48.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=14.0.0"
|
||||||
|
},
|
||||||
|
"license": "MIT"
|
||||||
|
}
|
||||||
208
skills/playwright-skill/run.js
Executable file
208
skills/playwright-skill/run.js
Executable file
@@ -0,0 +1,208 @@
|
|||||||
|
#!/usr/bin/env node
|
||||||
|
/**
|
||||||
|
* Universal Playwright Executor for Claude Code
|
||||||
|
*
|
||||||
|
* Executes Playwright automation code from:
|
||||||
|
* - File path: node run.js script.js
|
||||||
|
* - Inline code: node run.js 'await page.goto("...")'
|
||||||
|
* - Stdin: cat script.js | node run.js
|
||||||
|
*
|
||||||
|
* Ensures proper module resolution by running from skill directory.
|
||||||
|
*/
|
||||||
|
|
||||||
|
const fs = require('fs');
|
||||||
|
const path = require('path');
|
||||||
|
const { execSync } = require('child_process');
|
||||||
|
|
||||||
|
// Change to skill directory for proper module resolution
|
||||||
|
process.chdir(__dirname);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if Playwright is installed
|
||||||
|
*/
|
||||||
|
function checkPlaywrightInstalled() {
|
||||||
|
try {
|
||||||
|
require.resolve('playwright');
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Install Playwright if missing
|
||||||
|
*/
|
||||||
|
function installPlaywright() {
|
||||||
|
console.log('📦 Playwright not found. Installing...');
|
||||||
|
try {
|
||||||
|
execSync('npm install', { stdio: 'inherit', cwd: __dirname });
|
||||||
|
execSync('npx playwright install chromium', { stdio: 'inherit', cwd: __dirname });
|
||||||
|
console.log('✅ Playwright installed successfully');
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
console.error('❌ Failed to install Playwright:', e.message);
|
||||||
|
console.error('Please run manually: cd', __dirname, '&& npm run setup');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get code to execute from various sources
|
||||||
|
*/
|
||||||
|
function getCodeToExecute() {
|
||||||
|
const args = process.argv.slice(2);
|
||||||
|
|
||||||
|
// Case 1: File path provided
|
||||||
|
if (args.length > 0 && fs.existsSync(args[0])) {
|
||||||
|
const filePath = path.resolve(args[0]);
|
||||||
|
console.log(`📄 Executing file: ${filePath}`);
|
||||||
|
return fs.readFileSync(filePath, 'utf8');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Case 2: Inline code provided as argument
|
||||||
|
if (args.length > 0) {
|
||||||
|
console.log('⚡ Executing inline code');
|
||||||
|
return args.join(' ');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Case 3: Code from stdin
|
||||||
|
if (!process.stdin.isTTY) {
|
||||||
|
console.log('📥 Reading from stdin');
|
||||||
|
return fs.readFileSync(0, 'utf8');
|
||||||
|
}
|
||||||
|
|
||||||
|
// No input
|
||||||
|
console.error('❌ No code to execute');
|
||||||
|
console.error('Usage:');
|
||||||
|
console.error(' node run.js script.js # Execute file');
|
||||||
|
console.error(' node run.js "code here" # Execute inline');
|
||||||
|
console.error(' cat script.js | node run.js # Execute from stdin');
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clean up old temporary execution files from previous runs
|
||||||
|
*/
|
||||||
|
function cleanupOldTempFiles() {
|
||||||
|
try {
|
||||||
|
const files = fs.readdirSync(__dirname);
|
||||||
|
const tempFiles = files.filter(f => f.startsWith('.temp-execution-') && f.endsWith('.js'));
|
||||||
|
|
||||||
|
if (tempFiles.length > 0) {
|
||||||
|
tempFiles.forEach(file => {
|
||||||
|
const filePath = path.join(__dirname, file);
|
||||||
|
try {
|
||||||
|
fs.unlinkSync(filePath);
|
||||||
|
} catch (e) {
|
||||||
|
// Ignore errors - file might be in use or already deleted
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Ignore directory read errors
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Wrap code in async IIFE if not already wrapped
|
||||||
|
*/
|
||||||
|
function wrapCodeIfNeeded(code) {
|
||||||
|
// Check if code already has require() and async structure
|
||||||
|
const hasRequire = code.includes('require(');
|
||||||
|
const hasAsyncIIFE = code.includes('(async () => {') || code.includes('(async()=>{');
|
||||||
|
|
||||||
|
// If it's already a complete script, return as-is
|
||||||
|
if (hasRequire && hasAsyncIIFE) {
|
||||||
|
return code;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If it's just Playwright commands, wrap in full template
|
||||||
|
if (!hasRequire) {
|
||||||
|
return `
|
||||||
|
const { chromium, firefox, webkit, devices } = require('playwright');
|
||||||
|
const helpers = require('./lib/helpers');
|
||||||
|
|
||||||
|
(async () => {
|
||||||
|
try {
|
||||||
|
${code}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Automation error:', error.message);
|
||||||
|
if (error.stack) {
|
||||||
|
console.error(error.stack);
|
||||||
|
}
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If has require but no async wrapper
|
||||||
|
if (!hasAsyncIIFE) {
|
||||||
|
return `
|
||||||
|
(async () => {
|
||||||
|
try {
|
||||||
|
${code}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Automation error:', error.message);
|
||||||
|
if (error.stack) {
|
||||||
|
console.error(error.stack);
|
||||||
|
}
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return code;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Main execution
|
||||||
|
*/
|
||||||
|
async function main() {
|
||||||
|
console.log('🎭 Playwright Skill - Universal Executor\n');
|
||||||
|
|
||||||
|
// Clean up old temp files from previous runs
|
||||||
|
cleanupOldTempFiles();
|
||||||
|
|
||||||
|
// Check Playwright installation
|
||||||
|
if (!checkPlaywrightInstalled()) {
|
||||||
|
const installed = installPlaywright();
|
||||||
|
if (!installed) {
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get code to execute
|
||||||
|
const rawCode = getCodeToExecute();
|
||||||
|
const code = wrapCodeIfNeeded(rawCode);
|
||||||
|
|
||||||
|
// Create temporary file for execution
|
||||||
|
const tempFile = path.join(__dirname, `.temp-execution-${Date.now()}.js`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Write code to temp file
|
||||||
|
fs.writeFileSync(tempFile, code, 'utf8');
|
||||||
|
|
||||||
|
// Execute the code
|
||||||
|
console.log('🚀 Starting automation...\n');
|
||||||
|
require(tempFile);
|
||||||
|
|
||||||
|
// Note: Temp file will be cleaned up on next run
|
||||||
|
// This allows long-running async operations to complete safely
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ Execution failed:', error.message);
|
||||||
|
if (error.stack) {
|
||||||
|
console.error('\n📋 Stack trace:');
|
||||||
|
console.error(error.stack);
|
||||||
|
}
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Run main function
|
||||||
|
main().catch(error => {
|
||||||
|
console.error('❌ Fatal error:', error.message);
|
||||||
|
process.exit(1);
|
||||||
|
});
|
||||||
Reference in New Issue
Block a user