projects/loopers-kotlin-spring-template/skills/testing/SKILL.md
Use when writing tests, generating test skeletons, deciding mock strategies, learning test patterns, or understanding test levels. Triggers include "테스트 작성", "테스트 패턴", "mock 전략", "test skeleton".
npx skillsauth add toongri/oh-my-toong-playground testingInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Test writing standards and quality guidelines following Classical TDD (state verification only).
| Test Level | File Pattern | When to Use | External Deps |
|-------------|-----------------------|---------------------------------------------|--------------------------|
| Unit | *Test.kt | Domain logic, value objects, pure functions | None |
| Integration | *IntegrationTest.kt | Service + Repository, transactions | Real DB |
| Concurrency | *ConcurrencyTest.kt | Locking, race conditions | Real DB |
| Adapter | *AdapterTest.kt | External API clients, queries | WireMock, Testcontainers |
| E2E | *E2ETest.kt | Full API flow, auth | Full stack |
| Batch | *BatchTest.kt | Spring Batch jobs | Real DB |
Tests serve three purposes: verify correctness, document behavior, and enable safe refactoring.
VERIFY STATE, NEVER INTERACTIONS.
NO EXCEPTIONS. NO NEGOTIATIONS.
This applies to ALL tests:
Violating the letter of this rule IS violating the spirit.
These thoughts mean you're rationalizing. STOP and reconsider:
| Thought | Reality | |------------------------------------------|----------------------------------------------------------------------| | "This is too simple for BDD structure" | Simple code deserves consistent structure. Lower cost = less excuse. | | "verify() is fine for external services" | Use WireMock/Adapter test. If impossible, it's design feedback. | | "State verification is impossible here" | Redesign to return verifiable result. "Hard to test" = "bad design". | | "This is just a utility class" | Utilities need BDD too. Consistency > convenience. | | "It's overkill for this case" | Rules exist precisely for these "exception" moments. | | "Factory method is boilerplate" | 5 minutes now saves hours of confusion later. | | "Service has domain logic, needs Unit Test" | Domain logic belongs in Domain model. Service only orchestrates. | | "Mock Unit Test is faster than Integration" | Speed is not the goal. Correct test level is. | | "verify() is just extra insurance" | Any verify() use is forbidden. No "insurance" exceptions. | | "Following the spirit, not the letter" | Violating the letter IS violating the spirit. No exceptions. | | "Happy path + one failure is enough" | Boundaries and class edges are where bugs hide. Systematic coverage required. | | "Any value in the valid range works" | Every test value must represent a named equivalence class. Arbitrary = untested. | | "Too many combinations, test the main ones" | Document which combinations are skipped and why. Silent omission is a defect. | | "BVA only applies to numeric types" | Boundaries exist in dates, string lengths, collection sizes. Apply BVA to all. | | "Commenting test values is over-documenting" | If you can't name the class a value represents, you don't know why you chose it. | | "@CsvSource rows exceed 8 while verifying conditions from 2+ independent responsibilities in a single test" | Eager Test anti-pattern. Split by responsibility so each test fails for exactly one reason. |
All of these mean: Follow the rules anyway.
Common excuses and why they're wrong:
| Excuse | Why It's Wrong | What To Do Instead | |----------------------------------------------|----------------------------------------------|----------------------------------| | "verify() is the only way to test this" | WireMock, Testcontainers exist. Or redesign. | Use Adapter test pattern | | "BDD structure is overhead for simple tests" | Consistency trumps perceived efficiency | Apply same structure everywhere | | "Factory methods are boilerplate" | They're investment, not cost | Create factory with all defaults | | "DRY - share setup between tests" | Test isolation > code reuse | Fresh fixtures per test | | "State verified, verify() is extra safety" | ANY verify() usage is forbidden. No hybrids. | Remove verify(), state only | | "Service does X, so Unit Test for X" | If X is domain logic, test Domain model | Unit Test Domain, Integration Service | | "Mock is faster than real DB" | Speed < correctness. Mocks hide real bugs. | Use real DB via Integration Test | | "One success + one failure covers it" | Boundaries between success/failure are untested | Apply BVA: boundary-1, boundary, boundary+1 | | "ParameterizedTest values don't need explanation" | Every value must represent a named class | Comment which equivalence class each value covers | | "Too many combinations, not practical" | Undocumented reduction = hidden risk | Apply Decision Table, document reduction rationale | | "BVA is only for integers" | Dates, string lengths, prices all have boundaries | Apply BVA to any ordered/comparable type | | "Test value comments clutter the code" | Unnamed values = arbitrary values = untested | Comment which equivalence class each value covers | | "Verifying all combinations in one place is faster" | Eager Test anti-pattern. Separating by responsibility ensures each test fails for exactly one reason | Split @CsvSource by responsibility, one test per concern |
If state verification appears technically impossible:
This project follows Classical TDD (Detroit School). All tests MUST verify outcomes, NOT interactions.
Verify WHAT happened, not HOW it happened.
assertThat(result).isEqualTo(expected)
assertThat(point.balance).isEqualTo(700L)
assertThatThrownBy { point.use(500L) }.isInstanceOf(CoreException::class.java)
verify(repository).save(any())
verify(mock, times(1)).method()
verifyNoInteractions(mock)
// ❌ ALSO FORBIDDEN: "Hybrid approach" - state + verify()
assertThat(order.status).isEqualTo(OrderStatus.PLACED) // state verification ✅
verify(paymentClient).requestPayment(any()) // then verify ❌ STILL FORBIDDEN!
No "extra insurance" verify(): If you did state verification, you're done. Adding verify() for "safety" is still forbidden.
For level classification criteria, decision flow, and file naming conventions, see references/test-level-guide.md
Use @Nested per behavior (method/endpoint). No more than 1 level of nesting.
@Nested
@DisplayName("use")
inner class Use {
// All cases for use()
}
@DisplayName: Korean description[result] when [condition]Two patterns for exception test names:
| Pattern | Korean (DisplayName) | English (Method name) |
|---------|---------------------|----------------------|
| Specific Error Type | [condition]하면 [SPECIFIC_ERROR] 예외가 발생한다 | throws [SPECIFIC_ERROR] when [condition] |
| CoreException | [condition]하면 [ErrorType] CoreException 발생 | throws [ErrorType] CoreException when [condition] |
Examples:
// Specific Error Type - when the exception class name is descriptive
@Test
@DisplayName("잔액이 부족하면 InsufficientBalanceException 예외가 발생한다")
fun `throws InsufficientBalanceException when balance is insufficient`()
// CoreException - when using CoreException with ErrorType enum
@Test
@DisplayName("존재하지 않는 사용자면 NotFound CoreException 발생")
fun `throws NotFound CoreException when user does not exist`()
@Test
@DisplayName("권한이 없으면 Forbidden CoreException 발생")
fun `throws Forbidden CoreException when user has no permission`()
Every test must have comments specifying concrete values and expected results.
@Test
@DisplayName("주문 금액이 올바르게 계산된다")
fun `calculates total correctly`() {
// given
val initialBalance = 1000L
val point = createPoint(balance = initialBalance)
// when
val deductAmount = 300L
point.deduct(deductAmount)
// then
assertThat(point.balance).isEqualTo(initialBalance - deductAmount)
}
Every test class must have private factory methods with all parameters defaulted.
// Unit Test: domain object creation
private fun createPoint(
id: Long = 0L,
userId: Long = 1L,
balance: Long = 1000L,
status: PointStatus = PointStatus.ACTIVE,
): Point = Point.of(id, userId, balance, status)
// Integration Test: includes DB persistence
private fun createProduct(
price: Money = Money.krw(10000),
stockQuantity: Int = 100,
): Product {
val brand = brandRepository.save(Brand.create("Test Brand"))
val product = productRepository.save(Product.create(name = "Test Product", price = price, brand = brand))
stockRepository.save(Stock.create(product.id, stockQuantity))
return product
}
// ❌ Bad: What is this test about?
val point = Point.of(id = 1L, userId = 42L, balance = 1000L, status = PointStatus.ACTIVE)
// ✅ Good: Clearly about balance deduction
val point = createPoint(balance = 1000L)
Each test verifies one behavior. Multiple assertThat is fine if they verify aspects of the same result.
// ❌ Bad
assertThat(result).isEqualTo(700)
// ✅ Good
val initialBalance = 1000L
val deductAmount = 300L
assertThat(point.balance).isEqualTo(initialBalance - deductAmount)
@AfterEachCreateOrderCommand)OrderCreatedEvent)@Scheduled methods that only invoke service methods
Load references based on the current task. Each file provides detailed patterns and real code examples.
references/test-level-guide.md - Level classification criteria and decision flowreferences/test-generation.md - Spec to test skeleton process, quality checklistreferences/unit-test.md - Unit test patterns (state change, validation, ParameterizedTest, domain events)references/integration-test.md - Integration patterns (rollback, Spring Event, Kafka Consumer)references/concurrency-test.md - Concurrency patterns (thread pool, locking, idempotency)references/adapter-test.md - Adapter patterns (WireMock, Circuit Breaker, Retry, complex queries)references/e2e-test.md - E2E patterns (HTTP status codes, auth failures, API contract)references/batch-test.md - Spring Batch patterns (Processor unit test, Step/Job integration test)references/external-dependencies.md - External dependencies by test level (Real DB, WireMock, Testcontainers)| Mistake | Why It's Wrong | Fix |
|------------------------------------|-------------------------------------|------------------------------------------------------|
| verify(mock).save(any()) | Interaction verification, not state | Assert on returned/persisted state |
| Shared mutable state between tests | Test pollution, flaky results | Create fresh fixtures per test |
| Testing implementation details | Breaks on refactor | Test observable behavior only |
| Magic numbers in assertions | Unclear what's being tested | Use named variables: initialBalance - deductAmount |
| Multiple behaviors per test | Hard to diagnose failures | One logical assertion per test |
| Missing @AfterEach cleanup | DB pollution across tests | Clean up created entities |
| Unit Test for pure delegation | Mock returns mock - no real testing | Skip Unit, write Integration Test instead |
| State + verify() hybrid | Any verify() is forbidden | Remove verify(), keep state verification only |
tools
Use at the end of a work session to review the WHOLE session and record entities worth pinning. This is the manual, deliberate complete-sweep review — NOT an automated nudge. Triggers on "wrap up", "wrap-up", "session wrap", "end of session", "what should I pin".
documentation
Use when initializing the pins knowledge graph for the first time in a project. Guides the user through creating pins.yaml (the storage manifest). Triggers on "setup pins", "initialize pins", "create pins.yaml", "first-run pins".
testing
Use when you need to record a single pin entity to the knowledge graph. Invokes lib/pins record() to validate and write a canonical .md file. Triggers on "record pin", "pin this", "save this as a pin".
databases
Use when looking up pins by type, tags, or source. Drives lib/pins/query.ts to retrieve matching pin entries from the knowledge graph. Supersedes the legacy manual ls+frontmatter procedure.