[JAVA, JAVA-SPRING, KOTLIN-CLIENT, KOTLIN-SPRING] - feature - add x-jackson-default-impl / typeInfoDefaultImpls support to allow specifying of fallback schema when deserializing#23955
Conversation
…foDefaultImpls support
Emit `@JsonTypeInfo(defaultImpl = ...)` for both deduction-based and
discriminator-based oneOf interfaces.
Two configuration sources (config option takes precedence):
- Schema extension: `x-jackson-default-impl: ClassName`
- Config option: `typeInfoDefaultImpls: {SchemaName: ClassName}`
The resolved class name passes through `toModelName()` so schemaMapping,
importMapping, and DTO prefix/suffix are all honoured. A LOGGER.warn is
emitted when the config option shadows a schema-level annotation.
There was a problem hiding this comment.
1 issue found across 9 files
Reply with feedback, questions, or to request a fix.
Re-trigger cubic
| public static final String USE_SEALED_RESPONSE_INTERFACES = "useSealedResponseInterfaces"; | ||
| public static final String COMPANION_OBJECT = "companionObject"; | ||
| public static final String SUSPEND_FUNCTIONS = "suspendFunctions"; | ||
| public static final String TYPE_INFO_DEFAULT_IMPLS = "typeInfoDefaultImpls"; |
There was a problem hiding this comment.
@Picazsoo Could you also add the feature to the java generators?
See src/main/resources/Java/deductionAnnotation.mustache and the different typeInfoAnnotation.mustache under java/libraries
There was a problem hiding this comment.
I will take a look at it. And this is early implementation without any polish. I marked temporarily as "ready for review" just to gather early feedback from Cubic
There was a problem hiding this comment.
But thanks for the review! I will properly tag you once I think it is ready for wasting human review time (-:
There was a problem hiding this comment.
Hi @jpfinne, I added it to the java generators as well.
There was a problem hiding this comment.
This is now ready for review
# Conflicts: # modules/openapi-generator/src/test/java/org/openapitools/codegen/kotlin/spring/KotlinSpringServerCodegenTest.java
…Java generators - Add TYPE_INFO_DEFAULT_IMPLS/DESC constants to CodegenConstants (addresses reviewer request to avoid duplication across SpringCodegen and KotlinSpringServerCodegen) - Move typeInfoDefaultImpls field, option registration, processOpts parsing, and x-jackson-resolved-default-impl resolution loop from SpringCodegen into AbstractJavaCodegen so all Java generators inherit the feature - Remove duplicated code from SpringCodegen; update KotlinSpringServerCodegen to reference CodegenConstants.TYPE_INFO_DEFAULT_IMPLS and apply the same safe instanceof cast fix already present in SpringCodegen - Update Java/deductionAnnotation.mustache: emit defaultImpl = X.class on @JsonTypeInfo(use = DEDUCTION) when x-jackson-resolved-default-impl is set - Update Java/typeInfoAnnotation.mustache: emit defaultImpl = X.class on both @JsonTypeInfo(use = NAME) variants (discriminator present/absent) via discriminator.vendorExtensions.x-jackson-resolved-default-impl - Add JavaClientCodegenTest tests covering deduction, discriminator, typeInfoDefaultImpls config option, and absence cases for Java client Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
There was a problem hiding this comment.
1 issue found across 17 files
Prompt for AI agents (unresolved issues)
Check if these issues are valid — if so, understand the root cause of each and fix them. If appropriate, use sub-agents to investigate and fix each issue separately.
<file name="modules/openapi-generator/src/main/resources/kotlin-spring/typeInfoAnnotation.mustache">
<violation number="1" location="modules/openapi-generator/src/main/resources/kotlin-spring/typeInfoAnnotation.mustache:6">
P1: Missing validation that the resolved defaultImpl is a subtype of the polymorphic base; misconfigured `x-jackson-default-impl` or `typeInfoDefaultImpls` values will compile but fail at runtime during deserialization.</violation>
</file>
Reply with feedback, questions, or to request a fix.
Re-trigger cubic
| allowSetters = true // allows the {{{discriminator.propertyBaseName}}} to be set during deserialization | ||
| ) | ||
| @JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.PROPERTY, property = "{{{discriminator.propertyBaseName}}}", visible = true) | ||
| @JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.PROPERTY, property = "{{{discriminator.propertyBaseName}}}", visible = true{{#vendorExtensions.x-jackson-resolved-default-impl}}, defaultImpl = {{vendorExtensions.x-jackson-resolved-default-impl}}::class{{/vendorExtensions.x-jackson-resolved-default-impl}}) |
There was a problem hiding this comment.
P1: Missing validation that the resolved defaultImpl is a subtype of the polymorphic base; misconfigured x-jackson-default-impl or typeInfoDefaultImpls values will compile but fail at runtime during deserialization.
Prompt for AI agents
Check if this issue is valid — if so, understand the root cause and fix it. At modules/openapi-generator/src/main/resources/kotlin-spring/typeInfoAnnotation.mustache, line 6:
<comment>Missing validation that the resolved defaultImpl is a subtype of the polymorphic base; misconfigured `x-jackson-default-impl` or `typeInfoDefaultImpls` values will compile but fail at runtime during deserialization.</comment>
<file context>
@@ -3,7 +3,7 @@
allowSetters = true // allows the {{{discriminator.propertyBaseName}}} to be set during deserialization
)
-@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.PROPERTY, property = "{{{discriminator.propertyBaseName}}}", visible = true)
+@JsonTypeInfo(use = JsonTypeInfo.Id.NAME, include = JsonTypeInfo.As.PROPERTY, property = "{{{discriminator.propertyBaseName}}}", visible = true{{#vendorExtensions.x-jackson-resolved-default-impl}}, defaultImpl = {{vendorExtensions.x-jackson-resolved-default-impl}}::class{{/vendorExtensions.x-jackson-resolved-default-impl}})
@JsonSubTypes(
{{#discriminator.mappedModels}}
</file context>
There was a problem hiding this comment.
I think this is not a valid issue - defaultImpl does not need to be a @JsonSubTypes subtype.
Jackson's @JsonTypeInfo(defaultImpl = ...) intentionally accepts any class assignable to the field's declared type at the injection site, not just registered subtypes. So one can use e.g. a no-op catch-all class (e.g. class UnknownAnimal : Animal) to absorb unrecognized payloads.
The PR already handles the "unknown model" case correctly by emitting a LOGGER.warn(...) rather than failing.
I want to warn on potential typos, but to not block intentional use of external/no-op fallback classes.
There was a problem hiding this comment.
9 issues found across 27 files (changes from recent commits).
Prompt for AI agents (unresolved issues)
Check if these issues are valid — if so, understand the root cause of each and fix them. If appropriate, use sub-agents to investigate and fix each issue separately.
<file name="modules/openapi-generator/src/main/resources/kotlin-spring/typeInfoAnnotation.mustache">
<violation number="1" location="modules/openapi-generator/src/main/resources/kotlin-spring/typeInfoAnnotation.mustache:6">
P1: Missing validation that the resolved defaultImpl is a subtype of the polymorphic base; misconfigured `x-jackson-default-impl` or `typeInfoDefaultImpls` values will compile but fail at runtime during deserialization.</violation>
</file>
Tip: Review your code locally with the cubic CLI to iterate faster.
Re-trigger cubic
# Conflicts: # modules/openapi-generator/src/test/java/org/openapitools/codegen/java/spring/SpringCodegenTest.java
Purpose of this PR is to complement the current oneOf jackson polymorphism - to make it possible to supply default fallback value. this is useful e.g. when one wants to specify a default for deduction based polymorphism (FasterXML/jackson-databind#3055) or when one wants to fallback to some no-op class.
PR checklist
Commit all changed files.
This is important, as CI jobs will verify all generator outputs of your HEAD commit as it would merge with master.
These must match the expectations made by your contribution.
You may regenerate an individual generator by passing the relevant config(s) as an argument to the script, for example
./bin/generate-samples.sh bin/configs/java*.IMPORTANT: Do NOT purge/delete any folders/files (e.g. tests) when regenerating the samples as manually written tests may be removed.
Summary by cubic
Adds Jackson default deserialization fallback for polymorphic oneOf models across all Java generators,
kotlin-spring, and thekotlinclient. Configure@JsonTypeInfo(defaultImpl = ...)viax-jackson-default-implor thetypeInfoDefaultImplsoption.typeInfoDefaultImpls: {SchemaName: ClassName}; overridesx-jackson-default-implper schema.AbstractJavaCodegen; Java and Spring emitdefaultImplfor bothId.NAMEandId.DEDUCTION.kotlin-springdoes both;kotlinclient supports discriminator-based (Id.NAME) with Jackson.toModelName, honoring mappings and DTO prefix/suffix; warns on overrides and unknown targets.x-jackson-default-impl; docs updated forjava,spring,kotlin-spring,kotlin, andjava-camel.java-microprofiledisables this (its models don’t use oneOf interfaces).kotlin-spring, andkotlinclient covering deduction/discriminator paths, overrides, suffix handling, and absence cases.Written for commit e005ea4. Summary will update on new commits.