kotlin-spring-proxy-compatibility
Kotlin Spring Proxy Compatibility
Source mapping: Tier 1 critical skill derived from Kotlin_Spring_Developer_Pipeline.md (SK-03).
Mission
Explain exactly why Spring interception does or does not happen for Kotlin code, then propose the safest fix. Focus on runtime behavior, not on code that merely looks annotated.
Read These Inputs
- The class and method carrying
@Transactional,@Cacheable,@Async,@Retryable, security annotations, or other proxy-triggering annotations. - The call site. Determine whether the method is called through another bean or through
this. - Build files. Verify
kotlin("plugin.spring"),kotlin("plugin.jpa"), or any customallOpenandnoArgconfiguration. - Whether the bean is proxied through an interface or through CGLIB class proxies.
- For persistence problems, read the entity class shape and JPA plugin setup.
Diagnose In This Order
- Verify whether a proxy should exist at all.
- Verify whether the call crosses the proxy boundary.
- Verify whether the class or method is proxyable.
- Verify whether the annotation is placed on the method Spring actually intercepts.
- Verify whether build-time plugins opened the relevant classes.
- Verify whether the bug is actually transactional semantics, not proxy creation.
Core Checks
- Check for self-invocation. A method calling another annotated method in the same class bypasses the proxy.
- Check whether the target class or method is effectively final for the chosen proxy strategy.
- Check whether a JDK proxy is used. If so, interception works through interface methods, not arbitrary class methods.
- Check whether the annotated method is private or otherwise non-interceptable.
- Check whether
@Configurationclasses,@Serviceclasses, and JPA entities rely on the correct Kotlin compiler plugins. - Check whether the symptom comes from coroutines or async boundaries rather than proxy absence.
Preferred Fixes
Prefer fixes in this order:
- Enable the correct Kotlin compiler plugin in Gradle.
- Move the intercepted method behind a real Spring bean boundary.
- Adjust interface or proxy strategy only if the current strategy is incompatible.
- Refactor responsibilities so that external callers cross the proxy naturally.
- Only as a last resort, inject the proxied bean into itself or use
AopContext, and explain why this is a compromise.
Advanced Interception Traps
@PostConstruct, constructors, and init blocks run on the target object before ordinary proxy interception can help. Advice that depends on transactions, caching, security, or retries will not rescue initialization logic.@Asyncchanges threads. Transaction, security, MDC, and request context do not automatically flow the same way they do in synchronous code.@Retryable,@Transactional,@Cacheable, and method security can stack. Advisor ordering changes behavior, especially when retries and transactions combine.@TransactionalEventListenerdepends on transaction phase. If no transaction exists, the listener may never fire or may fire immediately depending on fallback behavior.@Configuration(proxyBeanMethods = false)disables configuration-class method interception. Treat this as a performance and semantics choice, not a cosmetic flag.- Interface default methods, non-public methods, and methods reached only from internal helper paths may compile cleanly while still missing interception at runtime.
- In coroutine and reactive flows, transaction and security propagation depend on the underlying stack and context propagation model. A proxy can exist while context still does not flow the way the author expects.
Expert Heuristics
- If the symptom is "annotation present but behavior absent," first ask whether the relevant call crossed a bean boundary. This catches more real bugs than checking
openalone. - If the team wants a local helper method to stay private, do not force proxy semantics onto it. Move the transactional or cache boundary outward instead.
- For caching and security, verify key computation and authorization point as separate concerns from proxy presence.
- For JPA entities and configuration classes, distinguish opening for framework mechanics from opening for general extensibility. The former is usually acceptable; the latter may not be.
Kotlin-Specific Rules
- Prefer compiler plugins over manually marking entire class hierarchies as
open. - Do not generate JPA entities as
data class. - Do not assume annotations on private methods will work because they compile.
- Explain the difference between Kotlin source semantics and Spring runtime semantics.
- If migrating Java to Kotlin, re-check every proxy-reliant class after translation.
Output Contract
Return these sections:
Broken mechanism: what Spring feature is expected to intercept the call.Why it failed: one concrete runtime reason with code evidence.Minimal fix: the smallest change that restores interception.Safer design option: only if the current structure invites future proxy bugs.Verification: how to prove the annotation now takes effect.
Guardrails
- Do not suggest making every class
openmanually without first checking compiler plugins. - Do not confuse proxy absence with wrong propagation, wrong cache key, or other business logic issues.
- Do not recommend
data classentities or blanketallOpenfor unrelated code. - Do not leave self-invocation unexplained. It is one of the most common hidden causes.
Quality Bar
A good run of this skill makes the proxy model explicit and the fix easy to verify.
A bad run tells the user to add annotations or open keywords without proving why interception failed.
More from jetbrains/skills
spring-kotlin-code-review
Review Kotlin + Spring changes for behavioral regressions, transaction and proxy bugs, API and serialization mistakes, persistence risks, security issues, configuration drift, and missing tests. Use when reviewing a PR, diff, patch, or design change where generic style-focused review would miss Spring-specific correctness and operational risks.
4dependency-conflict-resolver
Diagnose and resolve Gradle and Spring classpath conflicts, version drift, and binary incompatibilities in Kotlin applications. Use when `NoSuchMethodError`, `ClassNotFoundException`, linkage errors, duplicate logging bindings, Jackson or Hibernate mismatches, or BOM-versus-explicit-version conflicts appear, and the fix must respect the repository's real version authorities.
3doc
Use when the task involves reading, creating, or editing `.docx` documents, especially when formatting or layout fidelity matters; prefer `python-docx` plus the bundled `scripts/render_docx.py` for visual checks.
3ci-cd-containerization-advisor
Design reproducible build, image, and deployment pipelines for Kotlin plus Spring applications, including CI verification, layered containers, rollout safety, and deployment-time migration coordination. Use when creating or improving Dockerfiles, CI workflows, image hardening, Kubernetes manifests, release gates, or deployment strategies for Spring Boot services, especially where build reproducibility and operational safety matter.
3kotlin-idiomatic-refactorer-spring-aware
Refactor Kotlin code toward clearer, more idiomatic design without breaking Spring behavior, serialization, persistence, or public contracts. Use when Java-flavored Kotlin needs cleanup, domain modeling should become more expressive, or boilerplate should be reduced, but the refactoring must remain safe for proxies, Jackson, JPA, configuration binding, and existing tests.
3configuration-properties-profiles-kotlin-safe
Design and diagnose Spring configuration, profiles, and `@ConfigurationProperties` binding for Kotlin applications. Use when property binding fails, environment-specific overrides behave unexpectedly, profile layering is confusing, secrets or defaults are modeled unsafely, or Kotlin nullability and constructor binding semantics make configuration errors hard to detect.
3