spring-data-rest icon indicating copy to clipboard operation
spring-data-rest copied to clipboard

DATAREST-1383 causes top level properties with @JsonSetter to be missed. [DATAREST-1573]

Open spring-projects-issues opened this issue 4 years ago • 3 comments

Joseph Valerio opened DATAREST-1573 and commented

We just upgraded to SpringBoot 2.3.2.

We are using spring datarest 3.3.2 with JPA against a RDBMS

We use a PATCH to update our user's passwords.  We have a setter on our User Object setPassword() annotated with @JsonSetter, but there is no corresponding persisted property.  This method is used to set the persisted encryptedPassword property on the user.  The following commit on DomainObjectReader causes this to no longer work:

https://github.com/spring-projects/spring-data-rest/commit/c616c539052ad30c5dcc7a7acf8076f2f993d513

229:       if (!mappedProperties.hasPersistentPropertyForField(fieldName)) {				 
230:           i.remove();
231:           continue;
232:       }

The i.remove on line 230 removes the password property from consideration since it is not a PersistentProperty  

If this property is not removed, it will cause the setter to be invoked at line 285:

 

return mapper.readerForUpdating(target).readValue(root);

 

If we change the Method to a PUT this works correctly, but it should also work for patch.

If you are not following my explaination, I can create a test project to expose the issue on request


1 votes, 2 watchers

spring-projects-issues avatar Oct 22 '20 19:10 spring-projects-issues

Joseph Valerio commented

I think you need something like this:

      if (!mappedProperties.isWritableProperty(fieldName)) {

        if(!isPropertyJsonWritable(target, fieldName))
          i.remove();

        continue;
      }

...

  private boolean isPropertyJsonWritable(Object bean, String fieldName) {

    BeanWrapper wrapper = new BeanWrapperImpl();
    if(wrapper.isWritableProperty(fieldName)) {
      if(wrapper.getWrappedClass().getAnnotation(JsonSetter.class) != null)
        return true;
      if(wrapper.getWrappedClass().getAnnotation(JsonProperty.class).access() == Access.READ_ONLY)
        return false;

      // default to writable if nothing else
      return true;
    }

    return false;

  }

I know this is brute force-ish, but it conveys the idea that I am proposing.  We can not just rely on the persistent property information for determining if this property is writable

spring-projects-issues avatar Oct 23 '20 15:10 spring-projects-issues

bschoenmaeckers commented

duplicate of  DATAREST-1524

spring-projects-issues avatar Oct 25 '20 12:10 spring-projects-issues

I believe this is essentially the same issue as https://github.com/spring-projects/spring-data-rest/issues/1928 edit: and also https://github.com/spring-projects/spring-data-rest/issues/1881.

darioseidl avatar Jan 18 '22 16:01 darioseidl