Skip to content
Advertisement

Jackson JSON custom class instantiator for third-party class

I am using a third-party POJO class RetryOptions that can only be created using a builder. The builder can only be instantiated using a static method RetryOptions.newBuilder(), or by calling options.toBuilder() on an existing instance.

I would like to create custom de/serializers for the third-party POJO (RetryOptions). My first approach was to write the object as a builder, then read the object as a builder and return the built result:

JavaScript

But the problem is that Jackson doesn’t know how to create an instance of RetryOptions.Builder in order to populate it’s fields.

Is there a way I can instruct Jackson in how to create the builder instance, but let Jackson handle the parsing, reflection, and assignment of the fields?

Perhaps something like:

JavaScript

Or perhaps there is a way to tell the object mapper how to create an instance of RetryOptions.Builder:

JavaScript

Or is there another way to slice this problem without resorting to my own reflection logic or a brute-force duplication of the third-party class?

Note: my solution must use the Jackson JSON library (no Guava, etc.)
Note: there are several classes in this third party library that run into this same issue, so a generic solution is helpful

Advertisement

Answer

Update

Jackson can deserialize private fields as long as they have a getter (see https://www.baeldung.com/jackson-field-serializable-deserializable-or-not).

So, it turns out, in my scenario, that I don’t need to deserialize RetryOptions through the builder, I just need to be able to construct an instance of RetryOptions that Jackson can use to populate the fields.

As I had multiple classes with this same constraint (no public constructors on a third-party class), I wrote the following method to generate ValueInstantiators from a Supplier lambda:

JavaScript

Then I registered ValueInstantiators for each of my classes, e.g:

JavaScript

No custom de/serializers are needed.

Original response

I found a way.

First, define a ValueInstantiator for the class. The Jackson documentation strongly encourages you to extend StdValueInstantiator.

In my scenario, I only needed the “default” (parameter-less) instantiator, so I overrode the canCreateUsingDefault and createUsingDefault methods.

There are other methods for creating from arguments if needed.

JavaScript

Then I register my ValueInstantiator with the ObjectMapper:

JavaScript

Now I can deserialize an instance of RetryOptions like so:

JavaScript

Note: my solution makes use of the de/serializers defined in the question – i.e. that first convert the RetryOptions instance to a builder before serializing, then deserializing back to a builder and building to restore the instance.

End of original response

User contributions licensed under: CC BY-SA
4 People found this is helpful
Advertisement