feat: add Playwright e2e tests for note linking functionality
This commit is contained in:
parent
6aa3903c1f
commit
096b2fa48e
5 changed files with 96 additions and 1 deletions
5
.gitignore
vendored
5
.gitignore
vendored
|
@ -30,3 +30,8 @@ notes-app
|
||||||
|
|
||||||
# SQLite db
|
# SQLite db
|
||||||
notes.db
|
notes.db
|
||||||
|
|
||||||
|
# Playwright
|
||||||
|
/test-results/
|
||||||
|
/playwright-report/
|
||||||
|
/playwright/.cache/
|
||||||
|
|
13
main.go
13
main.go
|
@ -86,6 +86,19 @@ func main() {
|
||||||
// API routes
|
// API routes
|
||||||
http.HandleFunc("/api/notes", handleNotes)
|
http.HandleFunc("/api/notes", handleNotes)
|
||||||
http.HandleFunc("/api/notes/", handleNote)
|
http.HandleFunc("/api/notes/", handleNote)
|
||||||
|
http.HandleFunc("/api/test/reset", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
if r.Method != "POST" {
|
||||||
|
http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := db.Exec("DELETE FROM notes")
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.WriteHeader(http.StatusOK)
|
||||||
|
})
|
||||||
|
|
||||||
// Serve frontend
|
// Serve frontend
|
||||||
http.HandleFunc("/", handleFrontend)
|
http.HandleFunc("/", handleFrontend)
|
||||||
|
|
|
@ -13,9 +13,13 @@
|
||||||
"format": "prettier --write .",
|
"format": "prettier --write .",
|
||||||
"lint": "prettier --check . && eslint .",
|
"lint": "prettier --check . && eslint .",
|
||||||
"test:unit": "vitest",
|
"test:unit": "vitest",
|
||||||
"test": "npm run test:unit -- --run"
|
"test": "npm run test:unit -- --run",
|
||||||
|
"test:e2e": "playwright test",
|
||||||
|
"test:e2e:ui": "playwright test --ui",
|
||||||
|
"test:e2e:headed": "playwright test --headed"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@playwright/test": "^1.42.1",
|
||||||
"@eslint/compat": "^1.2.5",
|
"@eslint/compat": "^1.2.5",
|
||||||
"@eslint/js": "^9.18.0",
|
"@eslint/js": "^9.18.0",
|
||||||
"@sveltejs/adapter-static": "^3.0.8",
|
"@sveltejs/adapter-static": "^3.0.8",
|
||||||
|
|
20
playwright.config.ts
Normal file
20
playwright.config.ts
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
import { defineConfig, devices } from '@playwright/test';
|
||||||
|
|
||||||
|
export default defineConfig({
|
||||||
|
testDir: './tests',
|
||||||
|
fullyParallel: true,
|
||||||
|
forbidOnly: !!process.env.CI,
|
||||||
|
retries: 0,
|
||||||
|
workers: 1, // Recommended for test isolation with database
|
||||||
|
reporter: 'html',
|
||||||
|
use: {
|
||||||
|
baseURL: 'http://localhost:3000',
|
||||||
|
trace: 'on',
|
||||||
|
},
|
||||||
|
webServer: {
|
||||||
|
command: 'go run main.go',
|
||||||
|
url: 'http://localhost:3000',
|
||||||
|
reuseExistingServer: !process.env.CI,
|
||||||
|
timeout: 120 * 1000,
|
||||||
|
},
|
||||||
|
});
|
53
tests/note-linking.spec.ts
Normal file
53
tests/note-linking.spec.ts
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
|
||||||
|
test.beforeEach(async ({ page }) => {
|
||||||
|
// Reset database before each test
|
||||||
|
await page.request.post('/api/test/reset');
|
||||||
|
});
|
||||||
|
|
||||||
|
test.describe('Note Linking', () => {
|
||||||
|
test('should create new note from broken link', async ({ page }) => {
|
||||||
|
// Create initial note
|
||||||
|
await page.goto('/notes/new');
|
||||||
|
await page.fill('input[name="title"]', 'Parent Note');
|
||||||
|
await page.fill('textarea[name="content"]', 'Content with [[Unlinked Child Note]]');
|
||||||
|
await page.click('button[type="submit"]');
|
||||||
|
|
||||||
|
// Click the unlinked note
|
||||||
|
await page.click('a.note-link');
|
||||||
|
|
||||||
|
// Verify new note form with title pre-filled
|
||||||
|
await expect(page).toHaveURL('/notes/new');
|
||||||
|
await expect(page.locator('input[name="title"]')).toHaveValue('Unlinked Child Note');
|
||||||
|
|
||||||
|
// Create child note
|
||||||
|
await page.fill('textarea[name="content"]', 'Child content');
|
||||||
|
await page.click('button[type="submit"]');
|
||||||
|
|
||||||
|
// Verify child note creation
|
||||||
|
await expect(page.locator('h1')).toHaveText('Unlinked Child Note');
|
||||||
|
await expect(page.locator('.content')).toContainText('Child content');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should navigate between linked notes', async ({ page }) => {
|
||||||
|
// Create two linked notes
|
||||||
|
await page.goto('/notes/new');
|
||||||
|
await page.fill('input[name="title"]', 'First Note');
|
||||||
|
await page.fill('textarea[name="content"]', 'Links to [[Second Note]]');
|
||||||
|
await page.click('button[type="submit"]');
|
||||||
|
const firstNoteUrl = page.url();
|
||||||
|
|
||||||
|
await page.click('a.note-link');
|
||||||
|
await page.fill('textarea[name="content"]', 'Links back to [[First Note]]');
|
||||||
|
await page.click('button[type="submit"]');
|
||||||
|
const secondNoteUrl = page.url();
|
||||||
|
|
||||||
|
// Verify bidirectional linking
|
||||||
|
await page.goto(firstNoteUrl);
|
||||||
|
await page.click('a.note-link:has-text("Second Note")');
|
||||||
|
await expect(page).toHaveURL(secondNoteUrl);
|
||||||
|
|
||||||
|
await page.click('a.note-link:has-text("First Note")');
|
||||||
|
await expect(page).toHaveURL(firstNoteUrl);
|
||||||
|
});
|
||||||
|
});
|
Loading…
Add table
Reference in a new issue