com.datastax.driver.mapping.MappingManager Maven / Gradle / Ivy
Show all versions of cassandra-driver-mapping Show documentation
/*
* Copyright (C) 2012-2015 DataStax 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 com.datastax.driver.mapping;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import com.datastax.driver.core.*;
import com.datastax.driver.mapping.annotations.Accessor;
import com.datastax.driver.mapping.annotations.Table;
/**
* Mapping manager from which to obtain entity mappers.
*/
public class MappingManager {
private final Session session;
final boolean isCassandraV1;
private volatile Map, Mapper>> mappers = Collections., Mapper>>emptyMap();
private volatile Map, UDTMapper>> udtMappers = Collections., UDTMapper>>emptyMap();
private volatile Map, Object> accessors = Collections., Object>emptyMap();
/**
* Creates a new {@code MappingManager} using the provided {@code Session}.
*
* Note that this constructor forces the initialization of the session (see
* {@link #MappingManager(Session, ProtocolVersion)} if that is a problem for you).
*
* @param session the {@code Session} to use.
*/
public MappingManager(Session session) {
this(session, getProtocolVersion(session));
}
private static ProtocolVersion getProtocolVersion(Session session) {
session.init();
return session.getCluster().getConfiguration().getProtocolOptions().getProtocolVersionEnum();
}
/**
* Creates a new {@code MappingManager} using the provided {@code Session}.
*
* This constructor is only provided for backward compatibility: before 2.1.7, {@code MappingManager} could be
* built from an uninitialized session; since 2.1.7, the mapper needs to know the active protocol version to
* adapt its internal requests, so {@link #MappingManager(Session)} will now initialize the session if needed.
* If you rely on the session not being initialized, use this constructor and provide the version manually.
*
* @param session the {@code Session} to use.
* @param protocolVersion the protocol version that will be used with this session.
*
* @since 2.1.7
*/
public MappingManager(Session session, ProtocolVersion protocolVersion) {
this.session = session;
// This is not strictly correct because we could connect to C* 2.0 with the v1 protocol.
// But mappers need to make a decision early so that generated queries are compatible, and we don't know in advance
// which nodes might join the cluster later.
// At least if protocol >=2 we know there won't be any 1.2 nodes ever.
this.isCassandraV1 = (protocolVersion == ProtocolVersion.V1);
}
/**
* The underlying {@code Session} used by this manager.
*
* Note that you can get obtain the {@code Cluster} object corresponding
* to that session using {@code getSession().getCluster()}.
*
* It is inadvisable to close the returned Session while this manager and
* its mappers are in use.
*
* @return the underlying session used by this manager.
*/
public Session getSession() {
return session;
}
/**
* Creates a {@code Mapper} for the provided class (that must be annotated by a
* {@link Table} annotation).
*
* The {@code MappingManager} only ever keeps one Mapper for each class, and so calling this
* method multiple times on the same class will always return the same object.
*
* @param the type of the class to map.
* @param klass the (annotated) class for which to return the mapper.
* @return the {@code Mapper} object for class {@code klass}.
*/
public Mapper mapper(Class klass) {
return getMapper(klass);
}
/**
* Creates a {@code UDTMapper} for the provided class (that must be
* annotated by a {@link com.datastax.driver.mapping.annotations.UDT} annotation).
*
*
* The {@code MappingManager} only ever keeps one {@code UDTMapper} for each
* class, and so calling this method multiple times on the same class will
* always return the same object.
*
*
* @param the type of the class to map.
* @param klass the (annotated) class for which to return the mapper.
* @return the {@code UDTMapper} object for class {@code klass}.
*/
public UDTMapper udtMapper(Class klass) {
return getUDTMapper(klass);
}
/**
* Creates an accessor object based on the provided interface (that must be annotated by
* a {@link Accessor} annotation).
*
* The {@code MappingManager} only ever keep one Accessor for each class, and so calling this
* method multiple time on the same class will always return the same object.
*
* @param the type of the accessor class.
* @param klass the (annotated) class for which to create an accessor object.
* @return the accessor object for class {@code klass}.
*/
public T createAccessor(Class klass) {
return getAccessor(klass);
}
@SuppressWarnings("unchecked")
private Mapper getMapper(Class klass) {
Mapper mapper = (Mapper)mappers.get(klass);
if (mapper == null) {
synchronized (mappers) {
mapper = (Mapper)mappers.get(klass);
if (mapper == null) {
EntityMapper entityMapper = AnnotationParser.parseEntity(klass, ReflectionMapper.factory(), this);
mapper = new Mapper(this, klass, entityMapper);
Map, Mapper>> newMappers = new HashMap, Mapper>>(mappers);
newMappers.put(klass, mapper);
mappers = newMappers;
}
}
}
return mapper;
}
@SuppressWarnings("unchecked")
UDTMapper getUDTMapper(Class klass) {
UDTMapper mapper = (UDTMapper)udtMappers.get(klass);
if (mapper == null) {
synchronized (udtMappers) {
mapper = (UDTMapper)udtMappers.get(klass);
if (mapper == null) {
EntityMapper entityMapper = AnnotationParser.parseUDT(klass, ReflectionMapper.factory(), this);
mapper = new UDTMapper(entityMapper, session);
Map, UDTMapper>> newMappers = new HashMap, UDTMapper>>(udtMappers);
newMappers.put(klass, mapper);
udtMappers = newMappers;
}
}
}
return mapper;
}
@SuppressWarnings("unchecked")
private T getAccessor(Class klass) {
T accessor = (T)accessors.get(klass);
if (accessor == null) {
synchronized (accessors) {
accessor = (T)accessors.get(klass);
if (accessor == null) {
AccessorMapper mapper = AnnotationParser.parseAccessor(klass, AccessorReflectionMapper.factory(), this);
mapper.prepare(this);
accessor = mapper.createProxy();
Map, Object> newAccessors = new HashMap, Object>(accessors);
newAccessors.put(klass, accessor);
accessors = newAccessors;
}
}
}
return accessor;
}
}