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

org.datanucleus.store.mongodb.MongoDBUtils Maven / Gradle / Ivy

/**********************************************************************
Copyright (c) 2011 Andy Jefferson and others. All rights reserved.
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.

Contributors:
2012 Chris Rued - patch for MapReduceCount and collection.count()
   ...
**********************************************************************/
package org.datanucleus.store.mongodb;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Collection;
import java.util.Date;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Set;

import org.bson.types.ObjectId;
import org.datanucleus.ClassLoaderResolver;
import org.datanucleus.ExecutionContext;
import org.datanucleus.FetchPlan;
import org.datanucleus.exceptions.NucleusDataStoreException;
import org.datanucleus.exceptions.NucleusException;
import org.datanucleus.exceptions.NucleusUserException;
import org.datanucleus.identity.IdentityUtils;
import org.datanucleus.identity.SCOID;
import org.datanucleus.metadata.AbstractClassMetaData;
import org.datanucleus.metadata.AbstractMemberMetaData;
import org.datanucleus.metadata.ClassMetaData;
import org.datanucleus.metadata.ColumnMetaData;
import org.datanucleus.metadata.EmbeddedMetaData;
import org.datanucleus.metadata.FieldPersistenceModifier;
import org.datanucleus.metadata.FieldRole;
import org.datanucleus.metadata.IdentityType;
import org.datanucleus.metadata.MetaDataUtils;
import org.datanucleus.metadata.RelationType;
import org.datanucleus.metadata.VersionMetaData;
import org.datanucleus.state.ObjectProvider;
import org.datanucleus.store.FieldValues;
import org.datanucleus.store.StoreManager;
import org.datanucleus.store.connection.ManagedConnection;
import org.datanucleus.store.mongodb.fieldmanager.FetchFieldManager;
import org.datanucleus.store.mongodb.query.LazyLoadQueryResult;
import org.datanucleus.store.query.Query;
import org.datanucleus.store.schema.naming.ColumnType;
import org.datanucleus.store.schema.table.Column;
import org.datanucleus.store.schema.table.MemberColumnMapping;
import org.datanucleus.store.schema.table.Table;
import org.datanucleus.store.types.SCO;
import org.datanucleus.store.types.SCOUtils;
import org.datanucleus.store.types.converters.TypeConverter;
import org.datanucleus.util.NucleusLogger;
import org.datanucleus.util.TypeConversionHelper;

import com.mongodb.BasicDBObject;
import com.mongodb.DB;
import com.mongodb.DBCollection;
import com.mongodb.DBCursor;
import com.mongodb.DBObject;
import com.mongodb.MongoException;
import com.mongodb.ReadPreference;

import java.util.LinkedList;

/**
 * Utilities for MongoDB.
 */
public class MongoDBUtils
{
    private MongoDBUtils() {}

    public static List performMongoCount(DB db, BasicDBObject filterObject, Class candidateClass, boolean subclasses, ExecutionContext ec)
    throws MongoException 
    {
        StoreManager storeMgr = ec.getStoreManager();
        long count = 0;
        for (AbstractClassMetaData cmd : MetaDataUtils.getMetaDataForCandidates(candidateClass, subclasses, ec)) 
        {
            Table table = storeMgr.getStoreDataForClass(cmd.getFullClassName()).getTable();
            String collectionName = table.getName();
            count += db.getCollection(collectionName).count(filterObject);
        }

        List results = new LinkedList<>();
        results.add(count);
        if (ec.getStatistics() != null)
        {
            // Add to statistics
            ec.getStatistics().incrementNumReads();
        }
        return results;
    }

    public static boolean isMemberNested(AbstractMemberMetaData mmd)
    {
        boolean nested = true;
        String nestedStr = mmd.getValueForExtension("nested");
        if (nestedStr != null && nestedStr.equalsIgnoreCase("false"))
        {
            nested = false;
        }

        return nested;
    }

    /**
     * Accessor for the default value specified for the provided member.
     * If no defaultValue is provided on the column then returns null.
     * @param mmd Metadata for the member
     * @return The default value
     */
    public static String getDefaultValueForMember(AbstractMemberMetaData mmd)
    {
        ColumnMetaData[] colmds = mmd.getColumnMetaData();
        if (colmds == null || colmds.length < 1)
        {
            return null;
        }
        return colmds[0].getDefaultValue();
    }

    /**
     * Accessor for the MongoDB field for the field of this embedded field.
     * Uses the column name from the embedded definition (if present), otherwise falls back to
     * the column name of the field in its own class definition, otherwise uses its field name.
     * @param mmd Metadata for the owning member
     * @param fieldNumber Member number of the embedded object
     * @return The field name to use
     */
    public static String getFieldName(AbstractMemberMetaData mmd, int fieldNumber)
    {
        String columnName = null;

        // Try the first column if specified
        EmbeddedMetaData embmd = mmd.getEmbeddedMetaData();
        AbstractMemberMetaData embMmd = null;
        if (embmd != null)
        {
            AbstractMemberMetaData[] embmmds = embmd.getMemberMetaData();
            embMmd = embmmds[fieldNumber];
        }

        if (embMmd != null)
        {
            ColumnMetaData[] colmds = embMmd.getColumnMetaData();
            if (colmds != null && colmds.length > 0)
            {
                columnName = colmds[0].getName();
            }
            if (columnName == null)
            {
                // Fallback to the field/property name
                columnName = embMmd.getName();
            }
            if (columnName == null)
            {
                columnName = embMmd.getName();
            }
        }
        return columnName;
    }

    /**
     * Convenience method that tries to find the object with the specified identity from all DBCollection objects
     * from the rootCmd and subclasses. Returns the class name of the object with this identity (or null if not found).
     * @param id The identity
     * @param rootCmd ClassMetaData for the root class in the inheritance tree
     * @param ec ExecutionContext
     * @param clr ClassLoader resolver
     * @return The class name of the object with this id
     */
    public static String getClassNameForIdentity(Object id, AbstractClassMetaData rootCmd, ExecutionContext ec, ClassLoaderResolver clr)
    {
        Map> classNamesByDbCollectionName = new HashMap<>();
        StoreManager storeMgr = ec.getStoreManager();
        Set rootClassNames = new HashSet();
        rootClassNames.add(rootCmd.getFullClassName());
        Table rootTable = storeMgr.getStoreDataForClass(rootCmd.getFullClassName()).getTable();
        classNamesByDbCollectionName.put(rootTable.getName(), rootClassNames);
        Collection subclassNames = storeMgr.getSubClassesForClass(rootCmd.getFullClassName(), true, clr);
        if (subclassNames != null && !subclassNames.isEmpty())
        {
            for (String subclassName : subclassNames)
            {
                AbstractClassMetaData cmd = ec.getMetaDataManager().getMetaDataForClass(subclassName, clr);
                Table subTable = storeMgr.getStoreDataForClass(cmd.getFullClassName()).getTable();
                String subTableName = subTable.getName();
                Set classNames = classNamesByDbCollectionName.get(subTableName);
                if (classNames == null)
                {
                    classNames = new HashSet();
                    classNamesByDbCollectionName.put(subTableName, classNames);
                }
                classNames.add(cmd.getFullClassName());
            }
        }

        ManagedConnection mconn = storeMgr.getConnection(ec);
        try
        {
            DB db = (DB)mconn.getConnection();

            for (Map.Entry> dbCollEntry : classNamesByDbCollectionName.entrySet())
            {
                // Check each DBCollection for the id PK field(s)
                String dbCollName = dbCollEntry.getKey();
                Set classNames = dbCollEntry.getValue();
                DBCollection dbColl = db.getCollection(dbCollName);
                BasicDBObject query = new BasicDBObject();
                if (rootCmd.getIdentityType() == IdentityType.DATASTORE)
                {
                    Object key = IdentityUtils.getTargetKeyForDatastoreIdentity(id);
                    if (storeMgr.isStrategyDatastoreAttributed(rootCmd, -1))
                    {
                        query.put("_id", new ObjectId((String)key));
                    }
                    else
                    {
                        query.put(rootTable.getDatastoreIdColumn().getName(), key);
                    }
                }
                else if (rootCmd.getIdentityType() == IdentityType.APPLICATION)
                {
                    if (IdentityUtils.isSingleFieldIdentity(id))
                    {
                        Object key = IdentityUtils.getTargetKeyForSingleFieldIdentity(id);
                        int[] pkNums = rootCmd.getPKMemberPositions();
                        AbstractMemberMetaData pkMmd = rootCmd.getMetaDataForManagedMemberAtAbsolutePosition(pkNums[0]);
                        String pkPropName = rootTable.getMemberColumnMappingForMember(pkMmd).getColumn(0).getName();
                        query.put(pkPropName, key);
                    }
                    else
                    {
                        int[] pkNums = rootCmd.getPKMemberPositions();
                        for (int i=0;i embMmds = new ArrayList();
                    embMmds.add(pkMmd);

                    ObjectProvider embOP = ec.findObjectProvider(fieldVal);
                    AbstractClassMetaData embCmd = embOP.getClassMetaData();
                    int[] memberPositions = embCmd.getAllMemberPositions();

                    String embOwnerCol = null;
                    if (MongoDBUtils.isMemberNested(pkMmd))
                    {
                        MemberColumnMapping mapping = table.getMemberColumnMappingForMember(pkMmd);
                        embOwnerCol = mapping.getColumn(0).getName();
                    }

                    for (int j=0;j options)
    {
        return getObjectsOfCandidateType(q, db, filterObject, null, options, null, null);
    }

    /**
     * Convenience method to return all objects of the candidate type (optionally allowing subclasses).
     * @param q Query
     * @param db Mongo DB
     * @param filterObject Optional filter object
     * @param orderingObject Optional ordering object
     * @param options Set of options for controlling this query
     * @param skip Number of records to skip
     * @param limit Max number of records to return
     * @return List of all candidate objects (implements QueryResult)
     */
    public static List getObjectsOfCandidateType(Query q, DB db, BasicDBObject filterObject, BasicDBObject orderingObject, Map options, Integer skip, Integer limit)
    {
        LazyLoadQueryResult qr = new LazyLoadQueryResult(q);

        // Find the DBCollections we need to query
        ExecutionContext ec = q.getExecutionContext();
        StoreManager storeMgr = ec.getStoreManager();
        ClassLoaderResolver clr = ec.getClassLoaderResolver();
        List cmds = MetaDataUtils.getMetaDataForCandidates(q.getCandidateClass(), q.isSubclasses(), ec);

        Map> classesByCollectionName = new HashMap<>();
        for (AbstractClassMetaData cmd : cmds)
        {
            if (cmd instanceof ClassMetaData && ((ClassMetaData)cmd).isAbstract())
            {
                // Omit any classes that are not instantiable (e.g abstract)
            }
            else
            {
                if (!storeMgr.managesClass(cmd.getFullClassName()))
                {
                    // Make sure schema exists, using this connection
                    ((MongoDBStoreManager)storeMgr).manageClasses(new String[] {cmd.getFullClassName()}, ec.getClassLoaderResolver(), db);
                }
                Table table = storeMgr.getStoreDataForClass(cmd.getFullClassName()).getTable();
                String collectionName = table.getName();
                List cmdsForCollection = classesByCollectionName.get(collectionName);
                if (cmdsForCollection == null)
                {
                    cmdsForCollection = new ArrayList<>();
                    classesByCollectionName.put(collectionName, cmdsForCollection);
                }
                cmdsForCollection.add(cmd);
            }
        }

        // Add a query for each DBCollection we need
        Iterator>> iter = classesByCollectionName.entrySet().iterator();
        while (iter.hasNext())
        {
            Map.Entry> entry = iter.next();
            String collectionName = entry.getKey();
            List cmdsForCollection = entry.getValue();

            AbstractClassMetaData rootCmd = cmdsForCollection.get(0);
            Table rootTable = storeMgr.getStoreDataForClass(rootCmd.getFullClassName()).getTable();
            int[] fpMembers = q.getFetchPlan().getFetchPlanForClass(rootCmd).getMemberNumbers();
            BasicDBObject fieldsSelection = new BasicDBObject();
            if (fpMembers != null && fpMembers.length > 0)
            {
                fieldsSelection = new BasicDBObject();
                for (int i=0;i> filterEntryIter = filterObject.entrySet().iterator();
                while (filterEntryIter.hasNext())
                {
                    Map.Entry filterEntry = filterEntryIter.next();
                    query.put(filterEntry.getKey(), filterEntry.getValue());
                }
            }

            if (rootCmd.hasDiscriminatorStrategy() && cmdsForCollection.size() == 1)
            {
                // TODO Add this restriction on *all* possible cmds for this DBCollection
                // Discriminator present : Add restriction on the discriminator value for this class
                query.put(rootTable.getDiscriminatorColumn().getName(), rootCmd.getDiscriminatorValue());
            }

            if (ec.getNucleusContext().isClassMultiTenant(rootCmd))
            {
                // Multitenancy discriminator present : Add restriction for this tenant
                String fieldName = rootTable.getMultitenancyColumn().getName();
                String value = ec.getNucleusContext().getMultiTenancyId(ec, rootCmd);
                query.put(fieldName, value);
            }

            DBCollection dbColl = db.getCollection(collectionName);
            Object val = (options != null ? options.get("slave-ok") : Boolean.FALSE);
            if (val == Boolean.TRUE)
            {
                dbColl.setReadPreference(ReadPreference.secondaryPreferred());
            }

            if (NucleusLogger.DATASTORE_NATIVE.isDebugEnabled())
            {
                NucleusLogger.DATASTORE_NATIVE.debug("Performing find() using query on collection " + collectionName +
                    " for fields=" + fieldsSelection + " with filter=" + query + " and ordering=" + orderingObject);
            }
            DBCursor curs = dbColl.find(query, fieldsSelection);
            if (ec.getStatistics() != null)
            {
                // Add to statistics
                ec.getStatistics().incrementNumReads();
            }

            if (classesByCollectionName.size() == 1)
            {
                if (orderingObject != null)
                {
                    curs = curs.sort(orderingObject);
                    qr.setOrderProcessed(true);
                }

                // We have a single DBCursor so apply the range specification directly to this DBCursor
                if (skip != null && skip > 0)
                {
                    curs = curs.skip(skip);
                    qr.setRangeProcessed(true);
                }
                if (limit != null && limit > 0)
                {
                    curs = curs.limit(limit);
                    qr.setRangeProcessed(true);
                }
            }

            if (curs.hasNext())
            {
                // Contains result(s) so add it to our QueryResult
                qr.addCandidateResult(rootCmd, curs, fpMembers);
            }
        }

        return qr;
    }

    /**
     * Convenience method that takes the provided DBObject and the details of the candidate that it is an instance of, and converts it into the associated POJO.
     * @param dbObject The DBObject
     * @param ec ExecutionContext
     * @param cmd Metadata for the candidate class
     * @param fpMembers FetchPlan members for the class that are provided
     * @param ignoreCache Whether to ignore the cache
     * @return The Pojo object
     */
    public static Object getPojoForDBObjectForCandidate(DBObject dbObject, ExecutionContext ec, AbstractClassMetaData cmd, int[] fpMembers, boolean ignoreCache)
    {
        Table table = ec.getStoreManager().getStoreDataForClass(cmd.getFullClassName()).getTable();
        if (cmd.hasDiscriminatorStrategy())
        {
            // Determine the class from the discriminator property
            String disPropName = table.getDiscriminatorColumn().getName();
            String discValue = (String)dbObject.get(disPropName);
            String clsName = ec.getMetaDataManager().getClassNameFromDiscriminatorValue(discValue, cmd.getDiscriminatorMetaData());
            if (!cmd.getFullClassName().equals(clsName) && clsName != null)
            {
                cmd = ec.getMetaDataManager().getMetaDataForClass(clsName, ec.getClassLoaderResolver());
            }
        }

        Object pojo = null;
        if (cmd.getIdentityType() == IdentityType.APPLICATION)
        {
            pojo = MongoDBUtils.getObjectUsingApplicationIdForDBObject(dbObject, cmd, ec, ignoreCache, fpMembers);
        }
        else if (cmd.getIdentityType() == IdentityType.DATASTORE)
        {
            pojo = MongoDBUtils.getObjectUsingDatastoreIdForDBObject(dbObject, cmd, ec, ignoreCache, fpMembers);
        }
        else
        {
            pojo = MongoDBUtils.getObjectUsingNondurableIdForDBObject(dbObject, cmd, ec, ignoreCache, fpMembers);
        }
        return pojo;
    }

    protected static void selectAllFieldsOfEmbeddedObject(AbstractMemberMetaData mmd, BasicDBObject fieldsSelection, ExecutionContext ec, ClassLoaderResolver clr)
    {
        EmbeddedMetaData embmd = mmd.getEmbeddedMetaData();
        AbstractMemberMetaData[] embmmds = embmd.getMemberMetaData();
        for (int i=0;i entryIter = rawMap.entrySet().iterator();
                    while (entryIter.hasNext())
                    {
                        Map.Entry entry = entryIter.next();

                        BasicDBObject entryObj = new BasicDBObject();
                        Object storeKey = getStoredValueForField(ec, mmd, mapping, entry.getKey(), FieldRole.ROLE_MAP_KEY);
                        entryObj.put("key", storeKey);
                        Object storeValue = getStoredValueForField(ec, mmd, mapping, entry.getValue(), FieldRole.ROLE_MAP_VALUE);
                        entryObj.put("value", storeValue);

                        coll.add(entryObj);
                    }
                    return coll;
                }
                else if (fieldRole == FieldRole.ROLE_MAP_KEY)
                {
                    TypeConverter keyConv = mapping.getTypeConverterForComponent(fieldRole);
                    if (keyConv != null)
                    {
                        return keyConv.toDatastoreType(value);
                    }
                }
                else if (fieldRole == FieldRole.ROLE_MAP_VALUE)
                {
                    TypeConverter valConv = mapping.getTypeConverterForComponent(fieldRole);
                    if (valConv != null)
                    {
                        return valConv.toDatastoreType(value);
                    }
                }
            }
        }

        Class type = value.getClass();
        if (mmd != null)
        {
            if (optional)
            {
                type = ec.getClassLoaderResolver().classForName(mmd.getCollection().getElementType());
            }
            else
            {
                if (mmd.hasCollection() && fieldRole == FieldRole.ROLE_COLLECTION_ELEMENT)
                {
                    type = ec.getClassLoaderResolver().classForName(mmd.getCollection().getElementType());
                }
                else if (mmd.hasArray() && fieldRole == FieldRole.ROLE_ARRAY_ELEMENT)
                {
                    type = ec.getClassLoaderResolver().classForName(mmd.getArray().getElementType());
                }
                else if (mmd.hasMap() && fieldRole == FieldRole.ROLE_MAP_KEY)
                {
                    type = ec.getClassLoaderResolver().classForName(mmd.getMap().getKeyType());
                }
                else if (mmd.hasMap() && fieldRole == FieldRole.ROLE_MAP_VALUE)
                {
                    type = ec.getClassLoaderResolver().classForName(mmd.getMap().getValueType());
                }
            }
        }

        if (Long.class.isAssignableFrom(type) ||
            Integer.class.isAssignableFrom(type) ||
            Short.class.isAssignableFrom(type) ||
            String.class.isAssignableFrom(type) ||
            Byte.class.isAssignableFrom(type) ||
            Boolean.class.isAssignableFrom(type))
        {
            return value;
        }
        else if (Enum.class.isAssignableFrom(type))
        {
            return TypeConversionHelper.getStoredValueFromEnum(mmd, fieldRole, (Enum)value);
        }
        else if (type == Date.class)
        {
            return value;
        }
        else if (Date.class.isAssignableFrom(type))
        {
            // Convert to java.util.Date since MongoDB doesn't support java.sql
            return new java.util.Date(((Date)value).getTime());
        }
        else if (Calendar.class.isAssignableFrom(type))
        {
            ColumnMetaData colmd = null;
            if (mmd != null && mmd.getColumnMetaData() != null && mmd.getColumnMetaData().length > 0)
            {
                colmd = mmd.getColumnMetaData()[0];
            }
            if (!MetaDataUtils.persistColumnAsString(colmd))
            {
                // Persisted as Date
                return ((Calendar)value).getTime();
            }
        }
        else if (Character.class.isAssignableFrom(type) || char.class.isAssignableFrom(type))
        {
            // store as String
            return "" + value;
        }

        // Fallback to built-in type converters
        // TODO Make use of default TypeConverter for a type before falling back to String/Long
        TypeConverter strConv = ec.getTypeManager().getTypeConverterForType(type, String.class);
        TypeConverter longConv = ec.getTypeManager().getTypeConverterForType(type, Long.class);
        if (strConv != null)
        {
            // store as a String
            return strConv.toDatastoreType(value);
        }
        else if (longConv != null)
        {
            // store as a Long
            return longConv.toDatastoreType(value);
        }

        // store as the raw value
        return value;
    }

    /**
     * Convenience method to convert the stored value for an object field into the value that will be held
     * in the object. Note that this does not cater for relation fields, just basic fields.
     * @param ec ExecutionContext
     * @param mmd Metadata for the field holding this value (if available)
     * @param mapping Member column mapping (if available)
     * @param value The stored value for the field
     * @param fieldRole The role of this value for the field
     * @return The value to put in the field
     */
    public static Object getFieldValueFromStored(ExecutionContext ec, AbstractMemberMetaData mmd, MemberColumnMapping mapping, Object value, FieldRole fieldRole)
    {
        if (value == null)
        {
            return null;
        }

        boolean optional = (mmd != null ? Optional.class.isAssignableFrom(mmd.getType()) : false);

        if (mmd != null)
        {
            if (mmd.hasCollection() && !optional)
            {
                if (fieldRole == FieldRole.ROLE_FIELD)
                {
                    Collection coll;
                    try
                    {
                        Class instanceType = SCOUtils.getContainerInstanceType(mmd.getType(), mmd.getOrderMetaData() != null);
                        coll = (Collection) instanceType.newInstance();
                    }
                    catch (Exception e)
                    {
                        throw new NucleusDataStoreException(e.getMessage(), e);
                    }

                    Collection rawColl = (Collection)value;
                    for (Object elem : rawColl)
                    {
                        Object storeElem = getFieldValueFromStored(ec, mmd, mapping, elem, FieldRole.ROLE_COLLECTION_ELEMENT);
                        coll.add(storeElem);
                    }
                    return coll;
                }
                else if (fieldRole == FieldRole.ROLE_COLLECTION_ELEMENT)
                {
                    TypeConverter elemConv = mapping.getTypeConverterForComponent(fieldRole);
                    if (elemConv != null)
                    {
                        return elemConv.toMemberType(value);
                    }
                }
            }
            else if (mmd.hasArray())
            {
                if (fieldRole == FieldRole.ROLE_FIELD)
                {
                    Collection rawColl = (Collection)value;
                    Object array = Array.newInstance(mmd.getType().getComponentType(), rawColl.size());
                    int i=0;
                    for (Object elem : rawColl)
                    {
                        Object storeElem = getFieldValueFromStored(ec, mmd, mapping, elem, FieldRole.ROLE_ARRAY_ELEMENT);
                        storeElem = TypeConversionHelper.convertTo(storeElem, mmd.getType().getComponentType());
                        Array.set(array, i++, storeElem);
                    }
                    return array;
                }
            }
            else if (mmd.hasMap())
            {
                if (fieldRole == FieldRole.ROLE_FIELD)
                {
                    Map map;
                    try
                    {
                        Class instanceType = SCOUtils.getContainerInstanceType(mmd.getType(), mmd.getOrderMetaData() != null);
                        map = (Map) instanceType.newInstance();
                    }
                    catch (Exception e)
                    {
                        throw new NucleusDataStoreException(e.getMessage(), e);
                    }

                    Collection rawColl = (Collection)value;
                    for (DBObject mapEntryObj : rawColl)
                    {
                        Object dbKey = mapEntryObj.get("key");
                        Object dbVal = mapEntryObj.get("value");
                        Object key = getFieldValueFromStored(ec, mmd, mapping, dbKey, FieldRole.ROLE_MAP_KEY);
                        Object val = getFieldValueFromStored(ec, mmd, mapping, dbVal, FieldRole.ROLE_MAP_VALUE);
                        map.put(key, val);
                    }
                    return map;
                }
                else if (fieldRole == FieldRole.ROLE_MAP_KEY)
                {
                    TypeConverter keyConv = mapping.getTypeConverterForComponent(fieldRole);
                    if (keyConv != null)
                    {
                        return keyConv.toMemberType(value);
                    }
                }
                else if (fieldRole == FieldRole.ROLE_MAP_VALUE)
                {
                    TypeConverter valConv = mapping.getTypeConverterForComponent(fieldRole);
                    if (valConv != null)
                    {
                        return valConv.toMemberType(value);
                    }
                }
            }
        }

        Class type = value.getClass();
        if (mmd != null)
        {
            if (optional)
            {
                type = ec.getClassLoaderResolver().classForName(mmd.getCollection().getElementType());
            }
            else
            {
                if (fieldRole == FieldRole.ROLE_COLLECTION_ELEMENT)
                {
                    type = ec.getClassLoaderResolver().classForName(mmd.getCollection().getElementType());
                }
                else if (fieldRole == FieldRole.ROLE_ARRAY_ELEMENT)
                {
                    type = ec.getClassLoaderResolver().classForName(mmd.getArray().getElementType());
                }
                else if (fieldRole == FieldRole.ROLE_MAP_KEY)
                {
                    type = ec.getClassLoaderResolver().classForName(mmd.getMap().getKeyType());
                }
                else if (fieldRole == FieldRole.ROLE_MAP_VALUE)
                {
                    type = ec.getClassLoaderResolver().classForName(mmd.getMap().getValueType());
                }
                else
                {
                    type = mmd.getType();
                }
            }
        }

        if (Character.class.isAssignableFrom(type))
        {
            if (value instanceof Character)
            {
                return value;
            }
            return ((String)value).charAt(0);
        }
        else if (Short.class.isAssignableFrom(type))
        {
            if (value instanceof Short)
            {
                return value;
            }
            return ((Number)value).shortValue();
        }
        else if (Integer.class.isAssignableFrom(type))
        {
            if (value instanceof Integer)
            {
                return value;
            }
            return ((Number)value).intValue();
        }
        else if (Long.class.isAssignableFrom(type))
        {
            if (value instanceof Long)
            {
                return value;
            }
            return ((Number)value).longValue();
        }
        else if (Float.class.isAssignableFrom(type))
        {
            if (value instanceof Float)
            {
                return value;
            }
            return ((Number)value).floatValue();
        }
        else if (Double.class.isAssignableFrom(type))
        {
            if (value instanceof Double)
            {
                return value;
            }
            return ((Number)value).doubleValue();
        }
        else if (Byte.class.isAssignableFrom(type))
        {
            if (value instanceof Byte)
            {
                return value;
            }
            return ((Number)value).byteValue();
        }
        else if (Enum.class.isAssignableFrom(type))
        {
            return TypeConversionHelper.getEnumForStoredValue(mmd, fieldRole, value, ec.getClassLoaderResolver());
        }
        else if (java.sql.Date.class.isAssignableFrom(type) && value instanceof Date)
        {
            java.sql.Date sqlDate = null;
            if (value instanceof java.sql.Date)
            {
                sqlDate = (java.sql.Date) value;
            }
            else
            {
                sqlDate = new java.sql.Date(((Date)value).getTime());
            }
            return sqlDate;
        }
        else if (java.sql.Time.class.isAssignableFrom(type) && value instanceof Date)
        {
            java.sql.Time sqlTime = null;
            if (value instanceof java.sql.Time)
            {
                sqlTime = (java.sql.Time) value;
            }
            else
            {
                sqlTime = new java.sql.Time(((Date)value).getTime());
            }
            return sqlTime;
        }
        else if (java.sql.Timestamp.class.isAssignableFrom(type) && value instanceof Date)
        {
            java.sql.Timestamp sqlTs = null;
            if (value instanceof java.sql.Timestamp)
            {
                sqlTs = (java.sql.Timestamp) value;
            }
            else
            {
                sqlTs = new java.sql.Timestamp(((Date)value).getTime());
            }
            return sqlTs;
        }
        else if (Date.class.isAssignableFrom(type) && value instanceof Date)
        {
            // Date could have been persisted as String (in 3.0m4) so this just handles the Date case
            return value;
        }
        else if (Calendar.class.isAssignableFrom(type))
        {
            ColumnMetaData colmd = null;
            if (mmd != null && mmd.getColumnMetaData() != null && mmd.getColumnMetaData().length > 0)
            {
                colmd = mmd.getColumnMetaData()[0];
            }
            if (!MetaDataUtils.persistColumnAsString(colmd))
            {
                // Persisted as Date
                Calendar cal = Calendar.getInstance();
                cal.setTime((Date)value);
                return cal;
            }
        }

        // TODO Make use of default TypeConverter for a type before falling back to String/Long
        TypeConverter strConv = ec.getTypeManager().getTypeConverterForType(type, String.class);
        TypeConverter longConv = ec.getTypeManager().getTypeConverterForType(type, Long.class);
        if (strConv != null)
        {
            // Persisted as a String, so convert back
            String strValue = (String)value;
            return strConv.toMemberType(strValue);
        }
        else if (longConv != null)
        {
            // Persisted as a Long, so convert back
            Long longValue = (Long)value;
            return longConv.toMemberType(longValue);
        }

        return value;
    }

    public static Object getAcceptableDatastoreValue(Object value)
    {
        if (value == null)
        {
            return null;
        }

        Object returnValue = value;
        if (returnValue instanceof SCO)
        {
            returnValue = ((SCO)returnValue).getValue();
        }

        if (returnValue instanceof java.sql.Timestamp)
        {
            return new java.util.Date(((java.sql.Timestamp)returnValue).getTime());
        }
        else if (returnValue instanceof java.sql.Time)
        {
            return new java.util.Date(((java.sql.Time)returnValue).getTime());
        }
        else if (returnValue instanceof java.sql.Date)
        {
            return new java.util.Date(((java.sql.Date)returnValue).getTime());
        }
        return returnValue;
    }

    /**
     * Method to return the value to store when using java serialisation.
     * @param mmd Metadata of the member
     * @param value The raw value of the member
     * @return The value to store
     */
    public static byte[] getStoredValueForJavaSerialisedField(AbstractMemberMetaData mmd, Object value)
    {
        byte[] storeValue = null;
        try
        {
            ByteArrayOutputStream bos = new ByteArrayOutputStream();
            ObjectOutputStream oos = new ObjectOutputStream(bos);
            oos.writeObject(value);
            storeValue = bos.toByteArray();
            oos.close();
            bos.close();
        }
        catch (IOException e)
        {
            throw new NucleusException("Exception thrown serialising value for field " + mmd.getFullFieldName(), e);
        }
        return storeValue;
    }

    /**
     * Method to return the raw value when stored java serialised.
     * @param mmd Metadata for the member
     * @param value The serialised value
     * @return The raw value
     */
    public static Object getFieldValueForJavaSerialisedField(AbstractMemberMetaData mmd, Object value)
    {
        Object returnValue = null;
        try
        {
            if (value != null)
            {
                byte[] bytes = (byte[])value;
                ByteArrayInputStream bis = new ByteArrayInputStream(bytes);
                ObjectInputStream ois = new ObjectInputStream(bis);
                returnValue = ois.readObject();
                ois.close();
                bis.close();
            }
        }
        catch (Exception e)
        {
            throw new NucleusUserException("Exception thrown deserialising field at " + mmd.getFullFieldName(), e);
        }
        return returnValue;
    }
}