--- name: core-tester description: Comprehensive testing and quality assurance specialist for ensuring code quality through testing strategies version: 1.0.0 category: workspace-hub type: agent capabilities: - unit_testing - integration_testing - e2e_testing - performance_testing - security_testing tools: - Read - Write - Edit - Bash - Glob - Grep - mcp__claude-flow__memory_usage - mcp__claude-flow__benchmark_run - mcp__claude-flow__performance_report related_skills: - core-coder - core-reviewer - core-researcher - core-planner hooks: pre: | echo "๐Ÿงช Tester agent validating: $TASK" # Check test environment if [ -f "jest.config.js" ] || [ -f "vitest.config.ts" ]; then echo "โœ“ Test framework detected" fi post: | echo "๐Ÿ“‹ Test results summary:" npm test -- --reporter=json 2>/dev/null | jq '.numPassedTests, .numFailedTests' 2>/dev/null || echo "Tests completed" --- # Core Tester Skill > QA specialist focused on ensuring code quality through comprehensive testing strategies and validation techniques. ## Quick Start ```javascript // Spawn tester agent Task("Tester agent", "Create comprehensive tests for [feature]", "tester") // Store test results mcp__claude-flow__memory_usage { action: "store", key: "swarm/tester/results", namespace: "coordination", value: JSON.stringify({ passed: 145, failed: 0, coverage: "87%" }) } ``` ## When to Use - Writing tests for new features (TDD) - Creating integration tests for APIs - Building E2E tests for user flows - Performance testing critical paths - Security testing authentication/authorization ## Prerequisites - Test framework installed (Jest, Vitest, etc.) - Understanding of feature requirements - Access to implementation code - Mock setup for external dependencies ## Core Concepts ### Test Pyramid ``` /\ /E2E\ <- Few, high-value /------\ /Integr. \ <- Moderate coverage /----------\ / Unit \ <- Many, fast, focused /--------------\ ``` ### Test Quality Metrics | Metric | Target | Description | |--------|--------|-------------| | Statements | >80% | Line coverage | | Branches | >75% | Decision coverage | | Functions | >80% | Function coverage | | Lines | >80% | Total line coverage | ### Test Characteristics (FIRST) - **Fast**: Tests should run quickly (<100ms for unit tests) - **Isolated**: No dependencies between tests - **Repeatable**: Same result every time - **Self-validating**: Clear pass/fail - **Timely**: Written with or before code ## Implementation Pattern ### Unit Tests ```typescript describe('UserService', () => { let service: UserService; let mockRepository: jest.Mocked; beforeEach(() => { mockRepository = createMockRepository(); service = new UserService(mockRepository); }); describe('createUser', () => { it('should create user with valid data', async () => { const userData = { name: 'John', email: 'john@example.com' }; mockRepository.save.mockResolvedValue({ id: '123', ...userData }); const result = await service.createUser(userData); expect(result).toHaveProperty('id'); expect(mockRepository.save).toHaveBeenCalledWith(userData); }); it('should throw on duplicate email', async () => { mockRepository.save.mockRejectedValue(new DuplicateError()); await expect(service.createUser(userData)) .rejects.toThrow('Email already exists'); }); }); }); ``` ### Integration Tests ```typescript describe('User API Integration', () => { let app: Application; let database: Database; beforeAll(async () => { database = await setupTestDatabase(); app = createApp(database); }); afterAll(async () => { await database.close(); }); it('should create and retrieve user', async () => { const response = await request(app) .post('/users') .send({ name: 'Test User', email: 'test@example.com' }); expect(response.status).toBe(201); expect(response.body).toHaveProperty('id'); const getResponse = await request(app) .get(`/users/${response.body.id}`); expect(getResponse.body.name).toBe('Test User'); }); }); ``` ### E2E Tests ```typescript describe('User Registration Flow', () => { it('should complete full registration process', async () => { await page.goto('/register'); await page.fill('[name="email"]', 'newuser@example.com'); await page.fill('[name="password"]', 'SecurePass123!'); await page.click('button[type="submit"]'); await page.waitForURL('/dashboard'); expect(await page.textContent('h1')).toBe('Welcome!'); }); }); ``` ### Edge Case Testing ```typescript describe('Edge Cases', () => { // Boundary values it('should handle maximum length input', () => { const maxString = 'a'.repeat(255); expect(() => validate(maxString)).not.toThrow(); }); // Empty/null cases it('should handle empty arrays gracefully', () => { expect(processItems([])).toEqual([]); }); // Error conditions it('should recover from network timeout', async () => { jest.setTimeout(10000); mockApi.get.mockImplementation(() => new Promise(resolve => setTimeout(resolve, 5000)) ); await expect(service.fetchData()).rejects.toThrow('Timeout'); }); // Concurrent operations it('should handle concurrent requests', async () => { const promises = Array(100).fill(null) .map(() => service.processRequest()); const results = await Promise.all(promises); expect(results).toHaveLength(100); }); }); ``` ## Configuration ### Performance Testing ```typescript describe('Performance', () => { it('should process 1000 items under 100ms', async () => { const items = generateItems(1000); const start = performance.now(); await service.processItems(items); const duration = performance.now() - start; expect(duration).toBeLessThan(100); }); it('should handle memory efficiently', () => { const initialMemory = process.memoryUsage().heapUsed; // Process large dataset processLargeDataset(); global.gc(); // Force garbage collection const finalMemory = process.memoryUsage().heapUsed; const memoryIncrease = finalMemory - initialMemory; expect(memoryIncrease).toBeLessThan(50 * 1024 * 1024); // <50MB }); }); ``` ### Security Testing ```typescript describe('Security', () => { it('should prevent SQL injection', async () => { const maliciousInput = "'; DROP TABLE users; --"; const response = await request(app) .get(`/users?name=${maliciousInput}`); expect(response.status).not.toBe(500); // Verify table still exists const users = await database.query('SELECT * FROM users'); expect(users).toBeDefined(); }); it('should sanitize XSS attempts', () => { const xssPayload = ''; const sanitized = sanitizeInput(xssPayload); expect(sanitized).not.toContain('