
org.ehcache.impl.internal.spi.serialization.DefaultSerializationProvider Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of ehcache-impl Show documentation
Show all versions of ehcache-impl Show documentation
The implementation module of Ehcache 3
/*
* Copyright Terracotta, Inc.
*
* 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 org.ehcache.impl.internal.spi.serialization;
import org.ehcache.impl.config.serializer.DefaultSerializationProviderConfiguration;
import org.ehcache.impl.config.serializer.DefaultSerializerConfiguration;
import org.ehcache.CachePersistenceException;
import org.ehcache.impl.serialization.ByteArraySerializer;
import org.ehcache.impl.serialization.CharSerializer;
import org.ehcache.impl.serialization.CompactJavaSerializer;
import org.ehcache.impl.serialization.CompactPersistentJavaSerializer;
import org.ehcache.core.internal.service.ServiceLocator;
import org.ehcache.impl.serialization.DoubleSerializer;
import org.ehcache.impl.serialization.FloatSerializer;
import org.ehcache.impl.serialization.IntegerSerializer;
import org.ehcache.impl.serialization.LongSerializer;
import org.ehcache.impl.serialization.StringSerializer;
import org.ehcache.spi.service.ServiceProvider;
import org.ehcache.spi.serialization.SerializationProvider;
import org.ehcache.spi.serialization.Serializer;
import org.ehcache.spi.serialization.UnsupportedTypeException;
import org.ehcache.core.spi.service.FileBasedPersistenceContext;
import org.ehcache.core.spi.service.LocalPersistenceService;
import org.ehcache.core.spi.service.LocalPersistenceService.PersistenceSpaceIdentifier;
import org.ehcache.spi.service.Service;
import org.ehcache.spi.service.ServiceConfiguration;
import org.ehcache.core.internal.util.ConcurrentWeakIdentityHashMap;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.Closeable;
import java.io.IOException;
import java.io.Serializable;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger;
import static org.ehcache.core.internal.service.ServiceLocator.findSingletonAmongst;
/**
* @author Ludovic Orban
*/
public class DefaultSerializationProvider implements SerializationProvider {
private static final Logger LOG = LoggerFactory.getLogger(DefaultSerializationProvider.class);
private final TransientProvider transientProvider;
private final PersistentProvider persistentProvider;
protected final ConcurrentWeakIdentityHashMap, AtomicInteger> providedVsCount = new ConcurrentWeakIdentityHashMap, AtomicInteger>();
protected final Set> instantiated = Collections.newSetFromMap(new ConcurrentWeakIdentityHashMap, Boolean>());
public DefaultSerializationProvider(DefaultSerializationProviderConfiguration configuration) {
if (configuration != null) {
transientProvider = new TransientProvider(configuration.getTransientSerializers());
persistentProvider = new PersistentProvider(configuration.getPersistentSerializers());
} else {
transientProvider = new TransientProvider(Collections., Class extends Serializer>>>emptyMap());
persistentProvider = new PersistentProvider(Collections., Class extends Serializer>>>emptyMap());
}
}
@Override
public Serializer createKeySerializer(Class clazz, ClassLoader classLoader, ServiceConfiguration>... configs) throws UnsupportedTypeException {
Serializer serializer = (Serializer)getUserProvidedSerializer(find(DefaultSerializerConfiguration.Type.KEY, configs));
if (serializer == null) {
if (findSingletonAmongst(PersistenceSpaceIdentifier.class, (Object[])configs) == null) {
serializer = transientProvider.createKeySerializer(clazz, classLoader, configs);
} else {
serializer = persistentProvider.createKeySerializer(clazz, classLoader, configs);
}
instantiated.add(serializer);
}
updateProvidedInstanceCounts(serializer);
return serializer;
}
@Override
public Serializer createValueSerializer(Class clazz, ClassLoader classLoader, ServiceConfiguration>... configs) throws UnsupportedTypeException {
Serializer serializer = (Serializer)getUserProvidedSerializer(find(DefaultSerializerConfiguration.Type.VALUE, configs));
if (serializer == null) {
if (findSingletonAmongst(PersistenceSpaceIdentifier.class, (Object[])configs) == null) {
serializer = transientProvider.createValueSerializer(clazz, classLoader, configs);
} else {
serializer = persistentProvider.createValueSerializer(clazz, classLoader, configs);
}
instantiated.add(serializer);
}
updateProvidedInstanceCounts(serializer);
return serializer;
}
private void updateProvidedInstanceCounts(Serializer> serializer) {
AtomicInteger currentCount = providedVsCount.putIfAbsent(serializer, new AtomicInteger(1));
if(currentCount != null) {
currentCount.incrementAndGet();
}
}
@Override
public void releaseSerializer(final Serializer> serializer) throws IOException {
AtomicInteger currentCount = providedVsCount.get(serializer);
if(currentCount != null) {
if(currentCount.decrementAndGet() < 0) {
currentCount.incrementAndGet();
throw new IllegalArgumentException("Given serializer:" + serializer.getClass().getName() + " is not managed by this provider");
}
} else {
throw new IllegalArgumentException("Given serializer:" + serializer.getClass().getName() + " is not managed by this provider");
}
if(instantiated.remove(serializer) && serializer instanceof Closeable) {
((Closeable)serializer).close();
}
}
@Override
public void start(ServiceProvider serviceProvider) {
transientProvider.start(serviceProvider);
persistentProvider.start(serviceProvider);
}
@Override
public void stop() {
transientProvider.stop();
persistentProvider.stop();
}
private static void addDefaultSerializerIfNoneRegistered(Map, Class extends Serializer>>> serializers, Class clazz, Class extends Serializer> serializerClass) {
if (!serializers.containsKey(clazz)) {
serializers.put(clazz, serializerClass);
}
}
static class TransientProvider extends AbstractProvider {
public TransientProvider(Map, Class extends Serializer>>> serializers) {
super(serializers);
}
@Override
protected Serializer createSerializer(String suffix, Class clazz, ClassLoader classLoader, DefaultSerializerConfiguration config, ServiceConfiguration>... configs) throws UnsupportedTypeException {
try {
Class extends Serializer> klazz = getClassFor(clazz, config, classLoader);
return constructSerializer(clazz, klazz.getConstructor(ClassLoader.class), classLoader);
} catch (NoSuchMethodException e) {
throw new RuntimeException(e);
}
}
@Override
public void start(ServiceProvider serviceProvider) {
addDefaultSerializerIfNoneRegistered(serializers, Serializable.class, (Class) CompactJavaSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Long.class, LongSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Integer.class, IntegerSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Float.class, FloatSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Double.class, DoubleSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Character.class, CharSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, String.class, StringSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, byte[].class, ByteArraySerializer.class);
}
}
static class PersistentProvider extends AbstractProvider {
private volatile LocalPersistenceService persistence;
private PersistentProvider(Map, Class extends Serializer>>> serializers) {
super(serializers);
}
@Override
protected Serializer createSerializer(String suffix, Class clazz, ClassLoader classLoader, DefaultSerializerConfiguration config, ServiceConfiguration>... configs) throws UnsupportedTypeException {
Class extends Serializer> klazz = getClassFor(clazz, config, classLoader);
try {
Constructor extends Serializer> constructor = klazz.getConstructor(ClassLoader.class, FileBasedPersistenceContext.class);
PersistenceSpaceIdentifier space = findSingletonAmongst(PersistenceSpaceIdentifier.class, (Object[]) configs);
FileBasedPersistenceContext context = persistence.createPersistenceContextWithin(space, DefaultSerializationProvider.class.getSimpleName() + suffix);
return constructSerializer(clazz, constructor, classLoader, context);
} catch (NoSuchMethodException e) {
throw new RuntimeException(e);
} catch (CachePersistenceException e) {
throw new RuntimeException(e);
}
}
@Override
public void start(ServiceProvider serviceProvider) {
persistence = serviceProvider.getService(LocalPersistenceService.class);
addDefaultSerializerIfNoneRegistered(serializers, Serializable.class, (Class) CompactPersistentJavaSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Long.class, LongSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Integer.class, IntegerSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Float.class, FloatSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Double.class, DoubleSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, Character.class, CharSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, String.class, StringSerializer.class);
addDefaultSerializerIfNoneRegistered(serializers, byte[].class, ByteArraySerializer.class);
}
}
static abstract class AbstractProvider implements SerializationProvider {
protected final Map, Class extends Serializer>>> serializers;
private AbstractProvider(Map, Class extends Serializer>>> serializers) {
this.serializers = new LinkedHashMap, Class extends Serializer>>>(serializers);
}
@Override
public Serializer createKeySerializer(Class clazz, ClassLoader classLoader, ServiceConfiguration>... configs) throws UnsupportedTypeException {
DefaultSerializerConfiguration conf = find(DefaultSerializerConfiguration.Type.KEY, configs);
return createSerializer("-Key", clazz, classLoader, conf, configs);
}
@Override
public Serializer createValueSerializer(Class clazz, ClassLoader classLoader, ServiceConfiguration>... configs) throws UnsupportedTypeException {
DefaultSerializerConfiguration conf = find(DefaultSerializerConfiguration.Type.VALUE, configs);
return createSerializer("-Value", clazz, classLoader, conf, configs);
}
protected abstract Serializer createSerializer(String suffix, Class clazz, ClassLoader classLoader, DefaultSerializerConfiguration config, ServiceConfiguration>... configs) throws UnsupportedTypeException;
protected Class extends Serializer> getClassFor(Class clazz, DefaultSerializerConfiguration config, ClassLoader classLoader) throws UnsupportedTypeException {
if (config != null) {
Class extends Serializer> configured = config.getClazz();
if (configured != null) {
return configured;
}
}
Class extends Serializer> direct = (Class extends Serializer>) serializers.get(clazz);
if (direct != null) {
return direct;
}
for (Map.Entry, Class extends Serializer>>> entry : serializers.entrySet()) {
if (entry.getKey().isAssignableFrom(clazz)) {
return (Class extends Serializer>) entry.getValue();
}
}
throw new UnsupportedTypeException("No serializer found for type '" + clazz.getName() + "'");
}
protected Serializer constructSerializer(Class clazz, Constructor extends Serializer> constructor, Object ... args) {
try {
Serializer serializer = constructor.newInstance(args);
LOG.debug("Serializer for <{}> : {}", clazz.getName(), serializer);
return serializer;
} catch (InstantiationException e) {
throw new RuntimeException(e);
} catch (IllegalAccessException e) {
throw new RuntimeException(e);
} catch (IllegalArgumentException e) {
throw new AssertionError(e);
} catch (InvocationTargetException e) {
throw new RuntimeException(e);
}
}
@Override
public void stop() {
// no-op
}
@Override
public void releaseSerializer(final Serializer> serializer) {
// no-op
}
}
private static Serializer getUserProvidedSerializer(DefaultSerializerConfiguration conf) {
if(conf != null) {
Serializer instance = conf.getInstance();
if(instance != null) {
return instance;
}
}
return null;
}
@SuppressWarnings("unchecked")
private static DefaultSerializerConfiguration find(DefaultSerializerConfiguration.Type type, ServiceConfiguration>... serviceConfigurations) {
DefaultSerializerConfiguration result = null;
Collection serializationProviderConfigurations = ServiceLocator.findAmongst(DefaultSerializerConfiguration.class, (Object[]) serviceConfigurations);
for (DefaultSerializerConfiguration serializationProviderConfiguration : serializationProviderConfigurations) {
if (serializationProviderConfiguration.getType() == type) {
if (result != null) {
throw new IllegalArgumentException("Duplicate " + type + " serialization provider : " + serializationProviderConfiguration);
}
result = serializationProviderConfiguration;
}
}
return result;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy