All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.ehcache.impl.internal.spi.serialization.DefaultSerializationProvider Maven / Gradle / Ivy

There is a newer version: 3.3.1
Show newest version
/*
 * 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>>emptyMap());
      persistentProvider = new PersistentProvider(Collections., Class>>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>> serializers, Class clazz, Class> serializerClass) {
    if (!serializers.containsKey(clazz)) {
      serializers.put(clazz, serializerClass);
    }
  }

  static class TransientProvider extends AbstractProvider {

    public TransientProvider(Map, Class>> serializers) {
      super(serializers);
    }

    @Override
    protected  Serializer createSerializer(String suffix, Class clazz, ClassLoader classLoader, DefaultSerializerConfiguration config, ServiceConfiguration... configs) throws UnsupportedTypeException {
      try {
        Class> 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>> serializers) {
      super(serializers);
    }

    @Override
    protected  Serializer createSerializer(String suffix, Class clazz, ClassLoader classLoader, DefaultSerializerConfiguration config, ServiceConfiguration... configs) throws UnsupportedTypeException {
      Class> klazz = getClassFor(clazz, config, classLoader);
      try {
        Constructor> 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>> serializers;

    private AbstractProvider(Map, Class>> serializers) {
      this.serializers = new LinkedHashMap, Class>>(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> getClassFor(Class clazz, DefaultSerializerConfiguration config, ClassLoader classLoader) throws UnsupportedTypeException {
      if (config != null) {
        Class> configured = config.getClazz();
        if (configured != null) {
          return configured;
        }
      }

      Class> direct = (Class>) serializers.get(clazz);
      if (direct != null) {
        return direct;
      }
      for (Map.Entry, Class>> entry : serializers.entrySet()) {
        if (entry.getKey().isAssignableFrom(clazz)) {
          return (Class>) entry.getValue();
        }
      }
      throw new UnsupportedTypeException("No serializer found for type '" + clazz.getName() + "'");
    }

    protected  Serializer constructSerializer(Class clazz, Constructor> 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