java-kotlin-migration-assistant
This skill contains shell command directives (!`command`) that may execute system commands. Review carefully before installing.
Java Kotlin Migration Assistant
Source mapping: Tier 3 specialized skill derived from Kotlin_Spring_Developer_Pipeline.md (SK-19).
Mission
Translate Java into Kotlin in a way that improves long-term maintainability without introducing semantic drift. Prioritize behavioral equivalence first, idiomatic cleanup second, and broad redesign only when explicitly requested.
Read First
- The Java class and its callers.
- Existing tests and any contract documentation.
- Spring annotations, JPA annotations, Jackson annotations, validation annotations, and logging patterns.
- Build plugins and Kotlin compiler configuration already present in the repository.
- Whether the module is internal-only or has external consumers that care about binary compatibility.
Migration Order
Use this default order unless the repository strongly suggests otherwise:
- Tests and support utilities.
- Stateless helpers and internal-only classes.
- Services and controllers.
- Domain models and DTOs.
- Entities and framework-sensitive classes last.
The most fragile classes are usually entities, heavily proxied services, reflection-heavy config classes, and public library APIs.
Translation Rules
- Preserve public behavior before pursuing idiomatic compression.
- Convert fields plus getters plus setters into properties only when that does not change framework expectations.
- Replace Lombok consciously:
@Datais not a universal reason to create a Kotlindata class@Buildermay map to named/default parameters or may need an explicit builder for Java callers@Slf4jusually becomes an explicit logger field or companion object
- Replace checked-exception-heavy flows carefully. Kotlin does not enforce checked exceptions, but Java callers may still depend on them semantically.
- Keep annotations on the correct target after translation. Kotlin use-site targets matter.
Nullability And Platform-Type Rules
- Treat every Java type crossing into Kotlin as suspect until nullability is proven.
- Add explicit Kotlin nullability based on contracts, not on optimistic guesses.
- If upstream Java code lacks annotations, prefer defensive boundary checks over sprinkling
!!. - Distinguish internal invariants from external compatibility. Sometimes a Kotlin non-null model is correct internally even when the Java boundary remains nullable.
- Re-check collections, optionals, and maps separately. Java usage patterns often hide null values inside otherwise non-null containers.
Spring And JPA Nuances
- Verify proxy-sensitive classes still work after translation:
@Transactional@Cacheable@Async- method security
- Do not convert JPA entities into
data class. - Preserve JPA constructor, field access, and equality semantics deliberately.
- Re-check
@ConfigurationProperties, Jackson creator behavior, and validation annotations after constructor translation. - If the project relies on interface-based proxies, do not accidentally move critical behavior into non-interface methods that callers bypass.
Advanced Migration Traps
- Default parameters can change Java call sites, overload generation, and reflective constructor expectations.
- Kotlin properties may alter method signatures, bean-introspection shape, and serialization behavior compared with explicit Java accessors.
- Inline classes, sealed hierarchies, and scope-function-heavy refactors are not first-pass migration tools. Use them after compatibility is proven.
- Replacing mutable Java collections with read-only Kotlin interfaces does not magically make the backing data immutable.
equals,hashCode, andtoStringgeneration can silently change semantics for entities, cache keys, and logging.- Library modules may need
@JvmStatic,@JvmOverloads,@JvmField, or explicit overload preservation for Java consumers. - Mixed Java/Kotlin modules can expose incremental compilation and annotation-processing surprises. Keep build verification tight during migration.
Bytecode And Interop Nuances
internalvisibility is a module-level bytecode concept, not a source-level privacy guarantee. Do not use it casually in previously public or Java-consumed APIs.- Interface default methods, companion objects, and top-level functions alter Java call ergonomics and sometimes reflective lookup paths.
- SAM conversion, raw types, and wildcard variance can change overload resolution at Java call sites even when Kotlin looks cleaner.
Optional<T>, primitive wrappers, arrays, and checked exceptions each deserve explicit migration treatment; do not lump them under generic nullability cleanup.- Parameter-name retention, generated constructors, and method overloading affect frameworks that use reflection or bytecode-generated proxies.
- Annotation processors, generated metamodels, and kapt or KSP transitions can change what "successful migration" means for the build beyond source compilation.
Expert Heuristics
- If a class is consumed by Java, optimize for interop predictability before Kotlin elegance.
- Migrate behaviorally boring classes first to establish project conventions for logging, nullability, builders, and testing.
- Delay introduction of coroutines, sealed redesigns, and stronger domain modeling until the mixed Java/Kotlin seam is stable.
- When uncertain, preserve explicit methods over collapsing everything into properties or generated helpers. Debuggability and binary stability matter during migration.
Output Contract
Return these sections:
Migration scope: what is being converted now and what stays in Java for the moment.Behavioral invariants: what must remain unchanged.Kotlin translation plan: the exact language-level changes to make.Framework constraints: Spring, Jackson, JPA, validation, and binary-compatibility concerns.Verification: tests and runtime checks that prove no semantic drift.
Guardrails
- Do not change public API shape casually.
- Do not hide uncertain nullability behind
!!. - Do not mix migration with broad architectural redesign unless explicitly asked.
- Do not make entity, proxy, or serialization classes "more idiomatic" at the cost of framework correctness.
Quality Bar
A good run of this skill yields Kotlin that is safer and cleaner while still behaving like the original Java where it matters. A bad run produces "Java with Kotlin syntax" or, worse, a pretty rewrite that silently breaks callers, proxies, or persistence.
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.
3kotlin-spring-proxy-compatibility
Diagnose and prevent Kotlin plus Spring proxy failures around `@Transactional`, `@Cacheable`, `@Async`, method security, retry, configuration proxies, and JPA entity requirements. Use when AOP annotations appear to do nothing, transactional or cache behavior is inconsistent, compiler plugins may be missing, self-invocation is suspected, or Kotlin final-by-default semantics may break Spring behavior.
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.
3jpa-spring-data-kotlin-mapper
Model Kotlin persistence code correctly for Spring Data JPA and Hibernate, including entity design, relationships, fetch plans, repository queries, lazy loading, and common performance traps. Use when creating or reviewing entities and repositories, diagnosing N+1 or `LazyInitializationException`, placing indexes and uniqueness rules, or preventing Kotlin-specific ORM bugs such as `data class` entities and broken `equals` or `hashCode`.
3