![JAR search and dependency download from the Maven repository](/logo.png)
com.avanza.astrix.versioning.jackson2.VersionedJsonObjectMapper Maven / Gradle / Ivy
/*
* Copyright 2014 Avanza Bank AB
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.avanza.astrix.versioning.jackson2;
import java.io.IOException;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import com.avanza.astrix.versioning.jackson2.JsonMessageMigrator.Builder;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.core.Version;
import com.fasterxml.jackson.databind.*;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.node.ObjectNode;
public class VersionedJsonObjectMapper implements JsonObjectMapper.Impl {
private ObjectMapper migratingMapper;
private ThreadLocal versionHolder;
public VersionedJsonObjectMapper(ThreadLocal versionHolder,
ObjectMapper migratingMapper) {
this.versionHolder = versionHolder;
this.migratingMapper = migratingMapper;
}
@Override
public String serialize(Object object, int toVersion) throws Exception {
versionHolder.set(toVersion);
try {
return migratingMapper.writeValueAsString(object);
} finally {
versionHolder.remove();
}
}
@Override
public T deserialize(String json, Type target, int fromVersion) throws Exception {
versionHolder.set(fromVersion);
try {
JavaType javaType = migratingMapper.getTypeFactory().constructType(target);
return migratingMapper.readValue(json, javaType);
} finally {
versionHolder.remove();
}
}
// TODO: document whats going on in this class (the migrating object mapper)
static class JsonSerializerHolder {
private Class type;
private JsonSerializer serializer;
public JsonSerializerHolder(Class type, JsonSerializer serializer) {
this.type = type;
this.serializer = serializer;
}
public void register(SimpleModule module) {
module.addSerializer(type, serializer);
}
}
static class JsonDeserializerHolder {
private Class type;
private JsonDeserializer deserializer;
public JsonDeserializerHolder(Class type, JsonDeserializer deserializer) {
this.type = type;
this.deserializer = deserializer;
}
public void register(SimpleModule module) {
module.addDeserializer(type, deserializer);
}
}
static class MigratingJsonSerializer extends JsonSerializer {
private ObjectMapper rawMapper;
private JsonMessageMigrator migrator;
private ThreadLocal versionHolder;
public MigratingJsonSerializer(ObjectMapper rawMapper,
JsonMessageMigrator migrator,
ThreadLocal versionHolder) {
this.rawMapper = rawMapper;
this.migrator = migrator;
this.versionHolder = versionHolder;
}
public static MigratingJsonSerializer create(ObjectMapper rawMapper, JsonMessageMigrator migrator, ThreadLocal versionHolder) {
return new MigratingJsonSerializer<>(rawMapper, migrator, versionHolder);
}
@Override
public void serialize(T value, JsonGenerator jgen,
SerializerProvider provider) throws IOException,
JsonProcessingException {
ObjectNode objectNode = rawMapper.convertValue(value, ObjectNode.class);
migrator.downgrade(objectNode, getVersion());
jgen.writeObject(objectNode);
}
int getVersion() {
return versionHolder.get();
}
}
static class MigratingJsonDeserializer extends JsonDeserializer {
private ObjectMapper rawMapper;
private JsonMessageMigrator migrator;
private ThreadLocal versionHolder;
public MigratingJsonDeserializer(ObjectMapper rawMapper,
JsonMessageMigrator migrator,
ThreadLocal versionHolder) {
this.rawMapper = rawMapper;
this.migrator = migrator;
this.versionHolder = versionHolder;
}
public static MigratingJsonDeserializer create(ObjectMapper rawMapper, JsonMessageMigrator migrator, ThreadLocal versionHolder) {
return new MigratingJsonDeserializer<>(rawMapper, migrator, versionHolder);
}
@Override
public T deserialize(JsonParser jp, DeserializationContext ctxt)
throws IOException, JsonProcessingException {
ObjectNode objectNode = jp.readValueAs(ObjectNode.class);
migrator.upgrade(objectNode, getVersion());
return rawMapper.convertValue(objectNode, migrator.getJavaType());
}
int getVersion() {
return versionHolder.get();
}
}
static class MessageMigratorsBuilder {
Map, JsonMessageMigrator.Builder>> buildersByType = new HashMap<>();
MessageMigratorsBuilder registerAll(List extends AstrixJsonApiMigration> migrations) {
for (AstrixJsonApiMigration apiMigration : migrations) {
int version = apiMigration.fromVersion();
for (AstrixJsonMessageMigration> messageMigration : apiMigration.getMigrations()) {
register(version, messageMigration);
}
}
return this;
}
void register(int version, AstrixJsonMessageMigration messageMigration) {
@SuppressWarnings("unchecked")
Builder builder = (Builder) buildersByType.get(messageMigration.getJavaType());
if (builder == null) {
builder = new Builder<>(messageMigration.getJavaType());
buildersByType.put(messageMigration.getJavaType(), builder);
}
builder.addMigration(messageMigration, version);
}
ConcurrentMap, JsonMessageMigrator>> build() {
ConcurrentMap, JsonMessageMigrator>> migratorsByType = new ConcurrentHashMap<>();
for (JsonMessageMigrator.Builder> builder : this.buildersByType.values()) {
JsonMessageMigrator> jsonMessageMigrator = builder.build();
migratorsByType.put(jsonMessageMigrator.getJavaType(), jsonMessageMigrator);
}
return migratorsByType;
}
}
public static class VersionedObjectMapperBuilder implements JacksonObjectMapperBuilder {
private List> serializers = new ArrayList<>();
private List> deserializers = new ArrayList<>();
private ConcurrentMap, JsonMessageMigrator>> migratorsByType;
public VersionedObjectMapperBuilder(List extends AstrixJsonApiMigration> migrations) {
this.migratorsByType = new MessageMigratorsBuilder().registerAll(migrations).build();
}
@Override
public void addSerializer(Class type, JsonSerializer serializer) {
this.serializers.add(new JsonSerializerHolder<>(type, serializer));
}
@Override
public void addDeserializer(Class type, JsonDeserializer deserializer) {
this.deserializers.add(new JsonDeserializerHolder<>(type, deserializer));
}
public VersionedJsonObjectMapper build() {
ThreadLocal versionHolder = new ThreadLocal<>();
ObjectMapper rawMapper = buildRaw();
ObjectMapper migratingMapper = buildMigratingMapper(rawMapper, versionHolder);
return new VersionedJsonObjectMapper(versionHolder, migratingMapper);
}
private ObjectMapper buildMigratingMapper(ObjectMapper rawMapper, ThreadLocal versionHolder) {
SimpleModule module = new SimpleModule("Astrix-migratingModule", new Version(1, 0, 0, "", null, null));
for (JsonMessageMigrator> migrator : this.migratorsByType.values()) {
registerSerializerAndDeserializer(rawMapper, versionHolder, module, migrator);
}
// register custom serializers/deserializers for all custom types without migrator since those won't be intercepted by migratingObjectMapper
for (JsonDeserializerHolder> deserializer : this.deserializers) {
if (!this.migratorsByType.containsKey(deserializer.type)) {
deserializer.register(module);
}
}
for (JsonSerializerHolder> serializer : this.serializers) {
if (!this.migratorsByType.containsKey(serializer.type)) {
serializer.register(module);
}
}
ObjectMapper result = new ObjectMapper();
result.registerModule(module);
return result;
}
private void registerSerializerAndDeserializer(ObjectMapper rawMapper,
ThreadLocal versionHolder,
SimpleModule module,
JsonMessageMigrator migrator) {
module.addSerializer(migrator.getJavaType(), MigratingJsonSerializer.create(rawMapper, migrator, versionHolder));
module.addDeserializer(migrator.getJavaType(), MigratingJsonDeserializer.create(rawMapper, migrator, versionHolder));
}
private ObjectMapper buildRaw() {
SimpleModule rawModule = new SimpleModule("Astrix-rawModule", new Version(1,0,0, "", null, null));
for (JsonDeserializerHolder> deserializer : this.deserializers) {
deserializer.register(rawModule);
}
for (JsonSerializerHolder> serializer : this.serializers) {
serializer.register(rawModule);
}
ObjectMapper rawMapper = new ObjectMapper();
rawMapper.registerModule(rawModule);
return rawMapper;
}
}
// public static JsonObjectMapper create(AstrixRemotingServerApi jsonObjectMapperFactory) {
// VersionedObjectMapperBuilder objectMapperBuilder = new VersionedObjectMapperBuilder(jsonObjectMapperFactory.getMigrations());
// jsonObjectMapperFactory.configure(objectMapperBuilder);
// return JsonObjectMapper.create(objectMapperBuilder.build());
// }
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy