🎓 Initial commit: Math2 Platform - Plataforma de Álgebra Lineal PRO
✨ Características: - 45 ejercicios universitarios (Basic → Advanced) - Renderizado LaTeX profesional - IA generativa (Z.ai/DashScope) - Docker 9 servicios - Tests 123/123 pasando - Seguridad enterprise (JWT, XSS, Rate limiting) 🐳 Infraestructura: - Next.js 14 + Node.js 20 - PostgreSQL 15 + Redis 7 - Docker Compose completo - Nginx + SSL ready 📚 Documentación: - 5 informes técnicos completos - README profesional - Scripts de deployment automatizados Estado: Producción lista ✅
This commit is contained in:
79
e2e/playwright.config.ts
Normal file
79
e2e/playwright.config.ts
Normal file
@@ -0,0 +1,79 @@
|
||||
import { defineConfig, devices } from '@playwright/test';
|
||||
|
||||
/**
|
||||
* Playwright configuration for E2E tests
|
||||
* @see https://playwright.dev/docs/test-configuration
|
||||
*/
|
||||
export default defineConfig({
|
||||
testDir: './e2e',
|
||||
|
||||
/* Run tests in files in parallel */
|
||||
fullyParallel: true,
|
||||
|
||||
/* Fail the build on CI if you accidentally left test.only in the source code */
|
||||
forbidOnly: !!process.env.CI,
|
||||
|
||||
/* Retry on CI only */
|
||||
retries: process.env.CI ? 2 : 0,
|
||||
|
||||
/* Opt out of parallel tests on CI */
|
||||
workers: process.env.CI ? 1 : undefined,
|
||||
|
||||
/* Reporter to use */
|
||||
reporter: [
|
||||
['html', { outputFolder: 'playwright-report' }],
|
||||
['list'],
|
||||
['json', { outputFile: 'playwright-report/test-results.json' }],
|
||||
],
|
||||
|
||||
/* Shared settings for all the projects below */
|
||||
use: {
|
||||
/* Base URL to use in actions like `await page.goto('/')` */
|
||||
baseURL: process.env.PLAYWRIGHT_BASE_URL || 'http://localhost:3000',
|
||||
|
||||
/* Collect trace when retrying the failed test */
|
||||
trace: 'on-first-retry',
|
||||
|
||||
/* Screenshot on failure */
|
||||
screenshot: 'only-on-failure',
|
||||
|
||||
/* Video recording */
|
||||
video: 'retain-on-failure',
|
||||
|
||||
/* Viewport size */
|
||||
viewport: { width: 1280, height: 720 },
|
||||
},
|
||||
|
||||
/* Configure projects for major browsers */
|
||||
projects: [
|
||||
{
|
||||
name: 'chromium',
|
||||
use: { ...devices['Desktop Chrome'] },
|
||||
},
|
||||
{
|
||||
name: 'firefox',
|
||||
use: { ...devices['Desktop Firefox'] },
|
||||
},
|
||||
{
|
||||
name: 'webkit',
|
||||
use: { ...devices['Desktop Safari'] },
|
||||
},
|
||||
/* Test against mobile viewports */
|
||||
{
|
||||
name: 'Mobile Chrome',
|
||||
use: { ...devices['Pixel 5'] },
|
||||
},
|
||||
{
|
||||
name: 'Mobile Safari',
|
||||
use: { ...devices['iPhone 12'] },
|
||||
},
|
||||
],
|
||||
|
||||
/* Run your local dev server before starting the tests */
|
||||
webServer: {
|
||||
command: 'npm run dev',
|
||||
url: 'http://localhost:3000',
|
||||
reuseExistingServer: !process.env.CI,
|
||||
timeout: 120000,
|
||||
},
|
||||
});
|
||||
213
e2e/tests/auth.spec.ts
Normal file
213
e2e/tests/auth.spec.ts
Normal file
@@ -0,0 +1,213 @@
|
||||
import { test, expect } from '@playwright/test';
|
||||
|
||||
test.describe('Authentication Flow', () => {
|
||||
test.beforeEach(async ({ page }) => {
|
||||
// Navigate to homepage
|
||||
await page.goto('/');
|
||||
});
|
||||
|
||||
test('user can register and login', async ({ page }) => {
|
||||
// Navigate to register
|
||||
await page.click('text=Register');
|
||||
await expect(page).toHaveURL('/register');
|
||||
|
||||
// Fill registration form
|
||||
const testEmail = `test-e2e-${Date.now()}@example.com`;
|
||||
await page.fill('[name="email"]', testEmail);
|
||||
await page.fill('[name="username"]', `testuser${Date.now()}`);
|
||||
await page.fill('[name="password"]', 'SecurePass123!');
|
||||
await page.fill('[name="confirmPassword"]', 'SecurePass123!');
|
||||
|
||||
// Submit
|
||||
await page.click('button[type="submit"]');
|
||||
|
||||
// Should redirect to dashboard
|
||||
await expect(page).toHaveURL('/dashboard', { timeout: 10000 });
|
||||
await expect(page.locator('text=Bienvenido')).toBeVisible();
|
||||
});
|
||||
|
||||
test('should reject weak passwords', async ({ page }) => {
|
||||
await page.goto('/register');
|
||||
|
||||
await page.fill('[name="email"]', 'weak@test.com');
|
||||
await page.fill('[name="username"]', 'weakuser');
|
||||
await page.fill('[name="password"]', '123');
|
||||
await page.fill('[name="confirmPassword"]', '123');
|
||||
|
||||
await page.click('button[type="submit"]');
|
||||
|
||||
// Should show error
|
||||
await expect(page.locator('text=Password must be at least')).toBeVisible();
|
||||
});
|
||||
|
||||
test('user can login with valid credentials', async ({ page }) => {
|
||||
await page.goto('/login');
|
||||
|
||||
await page.fill('[name="email"]', 'test@example.com');
|
||||
await page.fill('[name="password"]', 'SecurePass123!');
|
||||
|
||||
await page.click('button[type="submit"]');
|
||||
|
||||
// Should redirect to dashboard
|
||||
await expect(page).toHaveURL('/dashboard', { timeout: 10000 });
|
||||
});
|
||||
|
||||
test('should reject invalid credentials', async ({ page }) => {
|
||||
await page.goto('/login');
|
||||
|
||||
await page.fill('[name="email"]', 'wrong@example.com');
|
||||
await page.fill('[name="password"]', 'WrongPass123!');
|
||||
|
||||
await page.click('button[type="submit"]');
|
||||
|
||||
// Should show error
|
||||
await expect(page.locator('text=Invalid credentials')).toBeVisible();
|
||||
});
|
||||
});
|
||||
|
||||
test.describe('Exercise Flow', () => {
|
||||
test.beforeEach(async ({ page }) => {
|
||||
// Login first
|
||||
await page.goto('/login');
|
||||
await page.fill('[name="email"]', 'test@example.com');
|
||||
await page.fill('[name="password"]', 'SecurePass123!');
|
||||
await page.click('button[type="submit"]');
|
||||
await expect(page).toHaveURL('/dashboard', { timeout: 10000 });
|
||||
});
|
||||
|
||||
test('user can navigate to exercises', async ({ page }) => {
|
||||
// Go to modules
|
||||
await page.click('text=Módulos');
|
||||
await expect(page).toHaveURL('/modules');
|
||||
|
||||
// Click on first module
|
||||
await page.click('[data-testid="module-card"]');
|
||||
await expect(page).toHaveURL(/\/modules\//);
|
||||
});
|
||||
|
||||
test('user can solve an exercise', async ({ page }) => {
|
||||
// Navigate to an exercise
|
||||
await page.goto('/modules/fundamentos/exercises/ex-1');
|
||||
|
||||
// Wait for exercise to load
|
||||
await expect(page.locator('[data-testid="exercise-question"]')).toBeVisible();
|
||||
|
||||
// Fill answer
|
||||
await page.fill('[data-testid="answer-input"]', '4');
|
||||
|
||||
// Submit
|
||||
await page.click('button:has-text("Enviar")');
|
||||
|
||||
// Verify success
|
||||
await expect(page.locator('text=¡Correcto!')).toBeVisible({ timeout: 5000 });
|
||||
});
|
||||
|
||||
test('should handle incorrect answers', async ({ page }) => {
|
||||
await page.goto('/modules/fundamentos/exercises/ex-1');
|
||||
|
||||
await expect(page.locator('[data-testid="exercise-question"]')).toBeVisible();
|
||||
|
||||
// Wrong answer
|
||||
await page.fill('[data-testid="answer-input"]', '5');
|
||||
await page.click('button:has-text("Enviar")');
|
||||
|
||||
// Should show try again
|
||||
await expect(page.locator('text=Incorrecto')).toBeVisible({ timeout: 5000 });
|
||||
await expect(page.locator('button:has-text("Intentar de nuevo")')).toBeVisible();
|
||||
});
|
||||
|
||||
test('should prevent XSS in answer input', async ({ page }) => {
|
||||
await page.goto('/modules/fundamentos/exercises/ex-1');
|
||||
|
||||
await expect(page.locator('[data-testid="exercise-question"]')).toBeVisible();
|
||||
|
||||
// Try XSS
|
||||
await page.fill('[data-testid="answer-input"]', '<script>alert(1)</script>');
|
||||
await page.click('button:has-text("Enviar")');
|
||||
|
||||
// Should show security error or be sanitized
|
||||
await expect(page.locator('text=security|XSS|inválido', {
|
||||
hasText: /seguridad|security|XSS|inválido/i
|
||||
})).toBeVisible({ timeout: 5000 });
|
||||
});
|
||||
|
||||
test('hint system works correctly', async ({ page }) => {
|
||||
await page.goto('/modules/fundamentos/exercises/ex-1');
|
||||
|
||||
await expect(page.locator('[data-testid="exercise-question"]')).toBeVisible();
|
||||
|
||||
// Click reveal hint
|
||||
await page.click('button:has-text("Mostrar pista")');
|
||||
|
||||
// Hint should be visible
|
||||
await expect(page.locator('[data-testid="hint-content"]')).toBeVisible();
|
||||
});
|
||||
});
|
||||
|
||||
test.describe('Progress Tracking', () => {
|
||||
test.beforeEach(async ({ page }) => {
|
||||
await page.goto('/login');
|
||||
await page.fill('[name="email"]', 'test@example.com');
|
||||
await page.fill('[name="password"]', 'SecurePass123!');
|
||||
await page.click('button[type="submit"]');
|
||||
await expect(page).toHaveURL('/dashboard', { timeout: 10000 });
|
||||
});
|
||||
|
||||
test('user can view progress', async ({ page }) => {
|
||||
await page.goto('/progress');
|
||||
|
||||
await expect(page.locator('text=Progreso')).toBeVisible();
|
||||
await expect(page.locator('[data-testid="progress-chart"]')).toBeVisible();
|
||||
});
|
||||
|
||||
test('streak is displayed correctly', async ({ page }) => {
|
||||
await page.goto('/dashboard');
|
||||
|
||||
await expect(page.locator('[data-testid="streak-display"]')).toBeVisible();
|
||||
});
|
||||
});
|
||||
|
||||
test.describe('Admin Panel', () => {
|
||||
test('admin can access admin panel', async ({ page }) => {
|
||||
// Login as admin
|
||||
await page.goto('/login');
|
||||
await page.fill('[name="email"]', 'admin@mathplatform.com');
|
||||
await page.fill('[name="password"]', 'admin123');
|
||||
await page.click('button[type="submit"]');
|
||||
|
||||
await expect(page).toHaveURL('/dashboard', { timeout: 10000 });
|
||||
|
||||
// Navigate to admin
|
||||
await page.goto('/admin');
|
||||
await expect(page).toHaveURL('/admin');
|
||||
});
|
||||
|
||||
test('non-admin cannot access admin panel', async ({ page }) => {
|
||||
// Login as regular user
|
||||
await page.goto('/login');
|
||||
await page.fill('[name="email"]', 'test@example.com');
|
||||
await page.fill('[name="password"]', 'SecurePass123!');
|
||||
await page.click('button[type="submit"]');
|
||||
|
||||
// Try to access admin
|
||||
await page.goto('/admin');
|
||||
|
||||
// Should redirect or show forbidden
|
||||
await expect(page).not.toHaveURL('/admin');
|
||||
});
|
||||
});
|
||||
|
||||
test.describe('Responsive Design', () => {
|
||||
test('mobile menu works', async ({ page }) => {
|
||||
// Set mobile viewport
|
||||
await page.setViewportSize({ width: 375, height: 667 });
|
||||
|
||||
await page.goto('/');
|
||||
|
||||
// Open mobile menu
|
||||
await page.click('[data-testid="mobile-menu-button"]');
|
||||
|
||||
// Menu should be visible
|
||||
await expect(page.locator('[data-testid="mobile-nav"]')).toBeVisible();
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user