- 
          
- 
                Notifications
    You must be signed in to change notification settings 
- Fork 1.4k
feat(#1381): Add a way to specify "inject-only" with @JacksonInject #5175
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: 2.x
Are you sure you want to change the base?
Conversation
…lization discard the injected value in favor of input (if any)
| @@ -1,4 +1,4 @@ | |||
| package com.fasterxml.jackson.databind.tofix; | |||
| package com.fasterxml.jackson.databind.deser.inject; | |||
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
moved since this should not fail anymore
| } | ||
|  | ||
| @Test | ||
| @DisplayName("input YES, injectable YES, useInput DEFAULT|FALSE => injected") | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
According to useInput's javadocs:
Default is
OptBoolean.DEFAULT, which translates toOptBoolean.TRUE: this is
for backwards compatibility (2.8 and earlier always allow binding input value).
This combination: input YES, injectable YES, useInput DEFAULT should actually behave as if we had useInput = TRUE, that is dropping the injected value and returning the input.
Nevertheless, since useInput never worked that way, this would mean introducing breaking changes. My suggestion is to keep things as per this test and change the javadoc accordingly:
Default is
OptBoolean.DEFAULT, which translates toOptBoolean.FALSE
| import static org.junit.jupiter.api.Assertions.assertEquals; | ||
| import static org.junit.jupiter.api.Assertions.assertThrows; | ||
|  | ||
| class JacksonInject1381WithOptionalTest extends DatabindTestUtil { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This class shows how the behavior changes for all the combinations of the other test class when all injected fields are optional. Do you think we still need a required property in the annotation?
I feel like combining useInput and optional would cover all cases, also because optional should drive the same logic (but reversed) of required. Given this, adding a required property would make things unclear for the users, and the internal logic would be quite messy. Does this make sense?
        
          
                src/main/java/com/fasterxml/jackson/databind/deser/BeanDeserializerFactory.java
              
                Outdated
          
            Show resolved
            Hide resolved
        
              
          
                src/test/java/com/fasterxml/jackson/databind/deser/inject/JacksonInject1381Test.java
              
                Outdated
          
            Show resolved
            Hide resolved
        
      There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
LGTM.
|  | ||
| // 04-Jun-2025, tatu: [databind#1381]: default for "useInput" is false | ||
| if (!Boolean.TRUE.equals(useInput)) { | ||
| builder.addInjectable(PropertyName.construct(m.getName()), | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is part I don't fully understand: if injectable is only added if no input is to be used, how do things work with useInput = false, injectable value being present.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
As per this test, with no input but injected value present, the injected value is always returned regardless of useInput:
@Test
@DisplayName("input NO, injectable YES, useInput DEFAULT|TRUE|FALSE => injected")
void test2() throws Exception {
    assertEquals("injected", injectedMapper.readValue(empty, InputDefault.class).getField());
    assertEquals("injected", injectedMapper.readValue(empty, InputTrue.class).getField());
    assertEquals("injected", injectedMapper.readValue(empty, InputFalse.class).getField());
}I read it like "we have no input but we have a value to inject, so we don't care about anything else: let's use the injected value". Is this the expected behavior? Or you meant something else?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Logic-wise, yes, you are correct (expected behavior I agree with).
But I meant code; meaning...  what  does addInjectable() do -- to me, it's the thing that injects non-input (injectable) value. So why is that contingent on "useInput"? And I think check here seems wrong in that sense: injectable value may be needed even input is used.
It is not only injected if useInput is false (or missing); can also be injected if useInput is true but input has no value for property.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It's a way to eagerly discard the injectable if we're going to use the input anyways. I think it's better in terms of performance since any further evaluation is useless, and there are many methods involved down that path.
But I also get this:
injectable value may be needed even input is used
Not now maybe, but in general I get that if a field is marked as injected, we should add it to the internal injectables for whatever future logic.
I can try to change this, which means passing useInput to the builder (last param in the snippet), as we made for optional:
builder.addInjectable(PropertyName.construct(m.getName()),
                            m.getType(),
                            beanDesc.getClassAnnotations(), m, entry.getKey(), optional, useInput);Maybe we could also think of passing the whole injectableValue to the builder, instead of passing its properties one by one.
But I think this is not going to be easy. What do you think, is it worth it?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, figured it was meant as optimization.
But the problem with " if we're going to use the input anyways" is that there might not be any input.
So it's not "use input if any matching, otherwise null" but rather "use input if present; if not, use injected value". Otherwise why even mark it with @JacksonInject at all, actually.
So we cannot know statically which one to use.
And yes, I think this is necessary to handle properly.
But I understand things are messy, tricky, complicated, esp. when passing things via Constructors (CreatorProperty properties) vs Field/MethodProperty ones.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Actually, no: if I change things to put @JacksonInject into Constructor parameter (instead of Field) -- common alternative (although adding on Field is allowed too), a few tests fail.
So this doesn't quite work fully yet.
| Ok. I don't think this fix is complete: I changed test set up slightly (in a way I think users are more likely to use it -- although original way is legit too... but both should work) and now a few tests fail. So would need to figure a way to make things fully work with combination of input and/or injectable values. | 
| Fair point, I missed that constructor properties take another path. On it | 
…e useInput, optional, and/or the DeserializationFeature
| _creatorParameters[_anyParamSetter.getParameterIndex()] = _createAndSetAnySetterValue(); | ||
| } | ||
|  | ||
| for (int ix = 0; ix < props.length; ++ix) { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is the moment when creator properties are resolved, so I think it's ok to have the logic here, as per the method's javadoc as well:
/**
* Method called to do necessary post-processing such as injection of values
* and verification of values for required properties,
* after either {@link #assignParameter(SettableBeanProperty, Object)}
* returnstrue(to indicate all creator properties are found), or when
* then whole JSON Object has been processed,
*/
public Object[] getParameters(SettableBeanProperty[] props)
I placed the new part right before the code that checks for null creator properties in case we have an injected value to use, so to not fail. Should I move it after?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It makes sense to me to do it after, so injection can prevent failure.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
However: this does another pass, as earlier calls to _findMissing() -- only in case of no input, tho -- has already done lookups. So we'd rather avoid those calls.
I'll add a high-level note on this, I think we can avoid redundancy.
| We now have the tests to cover all the scenarios (I guess), meaning combinations of injected creator properties vs fields, with or without  | 
| 
 I will try to have a good look tomorrow, to answer with proper understanding. Thank you for going through this again. | 
| I reworked the logic to always add the properties annotated with  Few things to note, for completeness: 
 I hope the overall implementation is good enough to be merged, despite the bullets above. What do you think? | 
Comment probably obsolete, removing blocker.
| Ok: I merged plumbing changes (to pipe through  Will go over remaining PR again, will see if there's a chance this could make it in 2.20.0 or not. | 
        
          
                src/main/java/com/fasterxml/jackson/databind/deser/SettableBeanProperty.java
              
                Outdated
          
            Show resolved
            Hide resolved
        
      | Ok, made some minor changes but I really need to read through the changes, with focus and thought. This is quite tricky. | 
| || (optional == null && !isEnabled(DeserializationFeature.FAIL_ON_UNKNOWN_INJECT_VALUE))) { | ||
| if (Boolean.TRUE.equals(useInput) | ||
| || Boolean.TRUE.equals(optional) | ||
| || (useInput == null || optional == null) | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This looks odd; should second || actually be &&? Or maybe it's just odd grouping of clauses with parenthesis.
Should probably update comment to describe logic fully.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We're returning empty() (we want to discard the injected value) in one of these cases:
- either useInputoroptionalistrue
- either useInputoroptionalisnullandFAIL_ON_UNKNOWN_INJECT_VALUEis disabled
maybe it's just odd grouping of clauses with parenthesis
Yup, due to formatting, I think the precedence in the last two lines is not immediately clear
if (Boolean.TRUE.equals(useInput)
        || Boolean.TRUE.equals(optional)
        || (useInput == null || optional == null)
        && !isEnabled(DeserializationFeature.FAIL_ON_UNKNOWN_INJECT_VALUE)) {I'm updating the comment
        
          
                src/main/java/com/fasterxml/jackson/databind/deser/impl/ValueInjector.java
              
                Outdated
          
            Show resolved
            Hide resolved
        
      | if (!JacksonInject.Value.empty().equals(value)) { | ||
|  | ||
| if (value == JacksonInject.Value.empty()) { | ||
| if (Boolean.FALSE.equals(_optional)) { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this is slightly wrong as it won't match null; and as per @JacksonInject.optional Javadocs, default value, expressed as null here, is same as false.
So I think this should be:
if (!Boolean.TRUE.equals(_optional)) {
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, but I think it's more complicated than just flipping that condition, since now optional must work together with useInput. As an example, if I just apply your suggestion, this test fails:
@DisplayName("input YES, injectable NO, useInput TRUE => input")
void test4() throws Exception {
    assertEquals("input", plainMapper.readValue(input, InputTrue.class).getField());In this case, I think that even if optional is null, useInput should take precedence: the user should not be forced to specify also optional = trueimho, since they're already saying "use the input whatever the injection is". Forcing users to also specify optional = true seems a bit redundant to me. What do you think?
On the other side, there are a couple test that are currently passing but should not, such as this (the @DisplayName is correct while the assertion is wrong):
@DisplayName("FAIL_ON_UNKNOWN_INJECT_VALUE NO, input YES, injectable NO, useInput DEFAULT|FALSE => exception")
    void test3() throws Exception {
        assertEquals("input", plainMapper.readValue(input, InputDefault.class).getField());Here we have:
- an input
- no injection
- both optionalanduseInputarenull
So I think the user here should explicitly state that optional = true or expect an exception. Am I right?
If we agree on the behavior I can start fixing the current implementation.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, I am not sure. The problem wrt "input" is simply this: existence of value from input is unknown at processing time (the way things are; not a theoretical limit but practical implementation limit).
So whereas we can easily see if injectable value exists, we can't say -- at time we need it -- whether there is input value. At least when injecting a Field (for Constructor injection we do actually keep track).
Having said that: on
In this case, I think that even if optional is null, useInput should take precedence: the user should not be forced to specify also optional = true
imho, since they're already saying "use the input whatever the injection is". Forcing users to also specify optional = true seems a bit redundant to me.
What do you think?
I think I agree: if "useInput" is effectively true (OptBoolean.TRUE or default, i.e. not specified), then only explicit optional = OptionalBoolean.FALSE) should lead to failure.
My only concern there is slight deviation from existing handling but... that may be more theoretical than actual.
So, yes: I think you are right, after all -- esp. when considering both "optional" and "useInput" together.
| Ok, so this getting closer! But I think duplicate setting and traversal could and should be avoided. So the existing problem is that before this PR injection only occurs in cases where input does not have property value. PR fixes this by iterating over all creator properties and checks each one, injecting as appropriate, possibly re-setting already injected value. I think we should be able to do better and handle injection case when input is being bound, instead; and blocking input from being used if configured with  | 
… when no value with the provided id is found
| 
 I fixed it with this commit. My first idea was to keep track of already injected properties leveraging a flag in  
 Then, I tried leveraging the existing parameters such as  So, I added  | 
| } | ||
| } | ||
|  | ||
| private void _inject(final SettableBeanProperty prop) throws JsonMappingException { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Needs comment/javadoc: I think this is only called for cases where we post-process "missing" Creator properties: ones for which no Input value was provided.
| } | ||
| } | ||
|  | ||
| if (_paramsInjectedBig == null) { | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok so now my remaining implementation problem is this: we should not
- Allocate _paramsInjectedBigif no injectable values are defined (to avoid overhead for common enough case) OR
- Do post-processing similarly (scanning over all properties for "small" case).
because I think it's not good to impose measurable overhead for non-injection cases.
To avoid these, checking for "are there any Injectables" would need to be done by pre-processing SettableBeanProperty set, I think, and passing a flag or something in PropertyValueBuffer constructor, because it cannot be done efficiently in getParameters().
Based on this flag, could eagerly construct _paramsInjectedBig.
I realize this is not trivial to do... but I feel it is important for performance.
As an extension, could create new helper class for holding Injectable status (BufferInjectables inner class?), and its existence could be used instead of flag.
useInput=TRUE in JacksonInject now make deserialization discard the injected value in favor of input (if any)