Agent Skill
2/7/2026

test-patterns

This skill provides patterns and best practices for generating and organizing tests. It covers unit testing, integration testing, test data factories, and coverage strategies across multiple languages and frameworks.

J
jayteealao
0GitHub Stars
1Views
npx skills add jayteealao/gitt-gaff

SKILL.md

Nametest-patterns
DescriptionThis skill provides patterns and best practices for generating and organizing tests. It covers unit testing, integration testing, test data factories, and coverage strategies across multiple languages and frameworks.

name: test-patterns description: This skill provides patterns and best practices for generating and organizing tests. It covers unit testing, integration testing, test data factories, and coverage strategies across multiple languages and frameworks.

Test Patterns Skill

Generate and organize tests following project conventions and industry best practices.

When to Use

  • Generating unit tests for new or existing code
  • Creating integration/API tests
  • Setting up test data factories
  • Analyzing and improving test coverage
  • Establishing testing conventions in a project

Reference Documents

Core Principles

1. Test Behavior, Not Implementation

WRONG: Testing internal method calls
RIGHT: Testing observable behavior and outputs

Tests should verify what code does, not how it does it. This makes tests resilient to refactoring.

2. Arrange-Act-Assert (AAA) Pattern

Every test should have three distinct sections:

# Arrange - Set up test data and conditions
user = create_user(name="Alice")
order = create_order(user=user, items=[item1, item2])

# Act - Execute the behavior being tested
result = order.calculate_total()

# Assert - Verify the expected outcome
assert result == 150.00

3. One Assertion Per Test (Logical)

Each test should verify one logical concept, though it may have multiple assertions for that concept:

def test_user_creation_sets_defaults():
    user = User.create(email="test@example.com")

    # Multiple assertions for one concept: default values
    assert user.status == "pending"
    assert user.role == "member"
    assert user.created_at is not None

4. Descriptive Test Names

Test names should describe the scenario and expected outcome:

# WRONG
def test_order():
def test_calculate():

# RIGHT
def test_order_with_discount_applies_percentage_reduction():
def test_calculate_total_includes_tax_for_taxable_items():

5. Test Independence

Tests must not depend on each other or on execution order:

  • Each test sets up its own data
  • Each test cleans up after itself (or uses transactions)
  • No shared mutable state between tests

Workflow: Generating Tests

Step 1: Analyze the Code Under Test

  1. Read the file/function to be tested
  2. Identify public interfaces and behaviors
  3. List edge cases and error conditions
  4. Note dependencies that need mocking

Step 2: Determine Test Type

Code TypeTest TypeFocus
Pure functionUnit testInput/output
Class with dependenciesUnit test with mocksBehavior
API endpointIntegration testRequest/response
Database operationIntegration testData persistence
External service callUnit test with mocksContract

Step 3: Create Test Structure

tests/
├── unit/
│   └── [module]/
│       └── test_[file].py
├── integration/
│   └── test_[feature].py
└── fixtures/
    └── [shared fixtures]

Step 4: Write Tests

Follow the patterns in reference documents for specific test types.

Step 5: Verify Coverage

Run coverage analysis and add tests for uncovered critical paths.

Language-Specific Patterns

Python (pytest)

import pytest
from unittest.mock import Mock, patch

class TestOrderCalculation:
    @pytest.fixture
    def order(self):
        return Order(items=[Item(price=100), Item(price=50)])

    def test_calculates_subtotal(self, order):
        assert order.subtotal == 150

    @pytest.mark.parametrize("discount,expected", [
        (0, 150),
        (10, 135),
        (50, 75),
    ])
    def test_applies_discount(self, order, discount, expected):
        order.apply_discount(discount)
        assert order.total == expected

TypeScript (Jest/Vitest)

import { describe, it, expect, vi } from 'vitest';

describe('OrderService', () => {
  it('calculates total with tax', () => {
    const order = new Order([
      { price: 100 },
      { price: 50 }
    ]);

    expect(order.totalWithTax(0.1)).toBe(165);
  });

  it('sends confirmation email on completion', async () => {
    const emailService = { send: vi.fn() };
    const order = new Order([], { emailService });

    await order.complete();

    expect(emailService.send).toHaveBeenCalledWith(
      expect.objectContaining({ type: 'confirmation' })
    );
  });
});

Ruby (RSpec)

RSpec.describe Order do
  subject(:order) { described_class.new(items: items) }
  let(:items) { [Item.new(price: 100), Item.new(price: 50)] }

  describe '#total' do
    it 'sums item prices' do
      expect(order.total).to eq(150)
    end

    context 'with discount applied' do
      before { order.apply_discount(10) }

      it 'reduces total by percentage' do
        expect(order.total).to eq(135)
      end
    end
  end
end

Quick Reference

PatternWhen to Use
FactoryCreating test objects with defaults
BuilderCreating complex test objects step-by-step
MockReplacing dependencies
StubProviding canned responses
SpyVerifying method calls
FakeLightweight implementation for testing
FixtureShared test data setup
ParametrizeTesting multiple inputs
Skills Info
Original Name:test-patternsAuthor:jayteealao