Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.backendless.FootprintsManager Maven / Gradle / Ivy
/*
* ********************************************************************************************************************
*
* BACKENDLESS.COM CONFIDENTIAL
*
* ********************************************************************************************************************
*
* Copyright 2012 BACKENDLESS.COM. All Rights Reserved.
*
* NOTICE: All information contained herein is, and remains the property of Backendless.com and its suppliers,
* if any. The intellectual and technical concepts contained herein are proprietary to Backendless.com and its
* suppliers and may be covered by U.S. and Foreign Patents, patents in process, and are protected by trade secret
* or copyright law. Dissemination of this information or reproduction of this material is strictly forbidden
* unless prior written permission is obtained from Backendless.com.
*
* ********************************************************************************************************************
*/
package com.backendless;
import com.backendless.utils.ReflectionUtil;
import weborb.reader.AnonymousObject;
import weborb.reader.ArrayType;
import weborb.reader.NamedObject;
import java.util.ArrayList;
import java.util.Arrays;
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.Set;
import java.util.WeakHashMap;
/**
* If server sends object object
with fields a
, b
, c
* and the client's class declares only fields a
, b
, then this class saves
* field c
in memory in order to send it with the next user's request.
*/
public class FootprintsManager
{
private static final FootprintsManager instance = new FootprintsManager();
private static final Set marked = new HashSet<>(); //for cyclic entities
public final Inner Inner = new Inner();
private final Map persistenceCache = new WeakHashMap<>();
private FootprintsManager()
{
}
public static FootprintsManager getInstance()
{
return instance;
}
public Footprint getEntityFootprint( Object entity )
{
return persistenceCache.get( entity );
}
public String getObjectId( Object entity )
{
if( entity instanceof BackendlessUser )
return ((BackendlessUser) entity).getObjectId();
Footprint footprint = getEntityFootprint( entity );
if( footprint != null )
return footprint.getObjectId();
return null;
}
public String getMeta( Object entity )
{
Object obj = persistenceCache.get( entity );
if( obj instanceof Footprint )
return ((Footprint) obj).get__meta();
if( obj instanceof BackendlessUser )
return (String) ((BackendlessUser) obj).getProperty( "__meta" );
return null;
}
public Date getCreated( Object entity )
{
if( persistenceCache.containsKey( entity ) )
{
return getEntityFootprint( entity ).getCreated();
}
return null;
}
public Date getUpdated( Object entity )
{
if( persistenceCache.containsKey( entity ) )
{
return getEntityFootprint( entity ).getUpdated();
}
return null;
}
public class Inner
{
/**
* Puts missing properties like objectId, ___meta etc. into entity map.
*
* @param entity entity object
* @param entityMap entity map
*/
public void putMissingPropsToEntityMap( Object entity, Map entityMap )
{
//put objectId if exists in cache
if( !entityMap.containsKey( Persistence.DEFAULT_OBJECT_ID_FIELD ) )
{
String objectId = getObjectId( entity );
if( objectId != null )
{
entityMap.put( Persistence.DEFAULT_OBJECT_ID_FIELD, objectId );
}
}
//put __meta if exists in cache
if( !entityMap.containsKey( Persistence.DEFAULT_META_FIELD ) )
{
String meta = getMeta( entity );
if( meta != null )
{
entityMap.put( Persistence.DEFAULT_META_FIELD, meta );
}
}
}
/**
* When the object is created on server, client gets new instance of it. In order to remember the system fields
* (objectId, __meta etc.) it is required to duplicate the old instance in cache.
*
* @param serializedEntity entity's map used to iterate through fields and duplicate footprints recursively
* @param persistedEntity entity from server
* @param initialEntity entity on which a method was called (.save(), .create() etc.)
*/
void duplicateFootprintForObject( Map serializedEntity, Object persistedEntity,
Object initialEntity )
{
//to avoid endless recursion
if( marked.contains( persistedEntity ) )
{
return;
}
else
{
marked.add( persistedEntity );
}
try
{
//iterate through fields in the object and duplicate entities recursively
Set> entries = serializedEntity.entrySet();
for( Map.Entry entry : entries )
{
if( entry.getValue() instanceof Map )
{
// retrieve persisted entity's field value
Object persistedEntityFieldValue = getFieldValue( persistedEntity, entry.getKey() );
// retrieve initial entity's field value
Object initialEntityFieldValue = getFieldValue( initialEntity, entry.getKey() );
// duplicate footprint recursively
duplicateFootprintForObject((Map) entry.getValue(), persistedEntityFieldValue, initialEntityFieldValue);
}
else if( entry.getValue() instanceof Collection )
{
Collection collection = (Collection) entry.getValue();
// retrieve persisted entity's field value (which is collection)
Collection persistedEntityFieldValue = getFieldValueAsCollection(persistedEntity, entry.getKey());
if (persistedEntityFieldValue.isEmpty())
continue;
// retrieve initial entity's field value (which is collection)
Collection initialEntityFieldValue = getFieldValueAsCollection(initialEntity, entry.getKey());
Collection mapCollection = (Collection) entry.getValue();
// recursively duplicate footprint for each object in collection
Iterator persistedEntityFieldValueIterator = persistedEntityFieldValue.iterator();
Iterator initialEntityFieldValueIterator = initialEntityFieldValue.iterator();
Iterator mapCollectionIterator = mapCollection.iterator();
while( initialEntityFieldValueIterator.hasNext() )
{
duplicateFootprintForObject( (Map) mapCollectionIterator.next(), persistedEntityFieldValueIterator.next(), initialEntityFieldValueIterator.next() );
}
}
}
Footprint footprint = persistenceCache.get( persistedEntity );
if( footprint != null )
{
persistenceCache.put( initialEntity, footprint );
}
}
finally
{
marked.remove( persistedEntity );
}
}
/**
* Recursively sets Footprint objects from OLD entities to NEW ones.
*
* @param serialized entity's map used to iterate through fields and update footprints recursively
* @param newEntity entity from server
* @param oldEntity entity on which a method was called (.save(), .create() etc.)
*/
void updateFootprintForObject( Map serialized, Object newEntity, Object oldEntity )
{
//to avoid endless recursion
if( marked.contains( newEntity ) )
{
return;
}
else
{
marked.add( newEntity );
}
try
{
//update footprints recursively
Set> entries = serialized.entrySet();
for( Map.Entry entry : entries )
{
String key = entry.getKey();
if( entry.getValue() instanceof Map )
{
Object newEntityField = getFieldValue( newEntity, key );
Object oldEntityField = getFieldValue( oldEntity, key );
updateFootprintForObject((Map) entry.getValue(), newEntityField, oldEntityField);
}
else if( entry.getValue() instanceof Collection )
{
Collection valueCollection = (Collection) entry.getValue();
Iterator valueIterator = valueCollection.iterator();
Collection newObjectCollection= getFieldValueAsCollection( newEntity, key );
Collection oldObjectCollection = getFieldValueAsCollection( oldEntity, key );
Collection mapCollection = (Collection) entry.getValue();
Iterator mapCollectionIterator = mapCollection.iterator();
Iterator oldObjectCollectionIterator = oldObjectCollection.iterator();
Iterator newObjectCollectionIterator = newObjectCollection.iterator();
while( oldObjectCollectionIterator.hasNext() )
updateFootprintForObject((Map) mapCollectionIterator.next(), newObjectCollectionIterator.next(), oldObjectCollectionIterator.next());
}
}
Footprint footprint = persistenceCache.get( newEntity );
persistenceCache.put(oldEntity, footprint);
}
finally
{
marked.remove( newEntity );
}
}
/**
* Removes entity's footprint from cache.
*
* @param serializedEntity entity's map used to iterate through fields and remove footprints recursively
* @param entity entity to be removed
*/
void removeFootprintForObject( Map serializedEntity, Object entity )
{
//to avoid endless recursion
if( marked.contains( entity ) )
{
return;
}
else
{
marked.add( entity );
}
try
{
//iterate through object's properties and remove footprints recursively
Set> entries = serializedEntity.entrySet();
for( Map.Entry entry : entries )
{
String key = entry.getKey();
if( entry.getValue() instanceof Map )
{
Object entityFieldValue = getFieldValue( entity, key );
// remove footprints recursively
removeFootprintForObject( (Map) entry.getValue(), entityFieldValue );
}
else if( entry.getValue() instanceof Collection )
{
Collection objectCollection = getFieldValueAsCollection( entity, key );
Collection mapCollection = (Collection) entry.getValue();
// remove footprints recursively for each object in collection
Iterator objectCollectionIterator = objectCollection.iterator();
Iterator mapCollectionIterator = mapCollection.iterator();
while( objectCollectionIterator.hasNext() )
removeFootprintForObject( (Map) mapCollectionIterator.next(), objectCollectionIterator.next() );
}
}
persistenceCache.remove( entity );
}
finally
{
marked.remove( entity );
}
}
/**
* Puts saved entity's footprint to cache.
* Key - instance
, value - footprint, initialized from instance
.
* Footprint is just an object, containing only fields
* objectId
, created
, updated
, meta
.
*
* @param instance saved object received from server
* @param entity IAdaptingType
*/
public void putEntityFootprintToCache( Object instance, Object entity )
{
//to avoid endless recursion
if( marked.contains( entity ) )
{
return;
}
else
{
marked.add( entity );
}
try
{
if( instance instanceof Collection )
{
ArrayType dataArray = null;
if( entity instanceof ArrayType )
{
dataArray = (ArrayType) entity;
}
else if( entity instanceof NamedObject )
{
AnonymousObject typedObject = (AnonymousObject) ((NamedObject) entity).getTypedObject();
dataArray = (ArrayType) typedObject.getProperties().get( "data" );
}
Object[] instances = ((Collection) instance).toArray();
putEntityFootprintToCache( instances, dataArray );
}
else if( entity instanceof NamedObject )
{
putEntityFootprintToCache( instance, ((NamedObject) entity).getTypedObject() );
}
else if( entity instanceof AnonymousObject )
{
putEntityFootprintToCache( instance, ((AnonymousObject) entity).getProperties() );
}
else if( entity instanceof ArrayType )
{
Object[] entities = (Object[]) ((ArrayType) entity).getArray();
Object[] arrayInstance = instance instanceof List ? ((List) instance).toArray() : (Object[]) instance;
for( int i = 0; i < arrayInstance.length; i++ )
{
putEntityFootprintToCache( arrayInstance[ i ], entities[ i ] );
}
}
else
{
Map e = (Map) entity;
for( Map.Entry entityEntry : e.entrySet() )
{
Object entityEntryValue = entityEntry.getValue();
if( entityEntryValue instanceof NamedObject || entityEntryValue instanceof ArrayType )
{
Object innerInstance = getFieldValue( instance, entityEntry.getKey() );
if( innerInstance != null )
putEntityFootprintToCache( innerInstance, entityEntry.getValue() );
}
}
Map cachedEntity = (Map) entity;
persistenceCache.put( instance, Footprint.initFromEntity( cachedEntity ) );
}
}
catch( Exception e )
{/*Error in caching process should not fail application*/
//e.printStackTrace();
}
marked.remove( entity );
}
/**
* Retrieves value for 'fieldName' field from the 'entity' object in a form of Collection.
*
* @param entity object to retrieve 'key' field from
* @param fieldName field name to retrieve from 'entity' object
*/
private Collection getFieldValueAsCollection( Object entity, String fieldName )
{
Object rawEntity = getFieldValue( entity, fieldName );
if( rawEntity == null )
return new ArrayList();
else if( rawEntity instanceof Collection )
return (Collection) rawEntity;
else if( rawEntity.getClass().isArray() )
return Arrays.asList( (Object[]) rawEntity );
else
throw new RuntimeException( "unknown data type - " + rawEntity.getClass() );
}
/**
* Retrieves value for 'fieldName' field from the 'entity' object without casting it to any type.
* Use #getFieldValueAsCollection to get result in a form of Collection.
*
* @param entity object to retrieve 'fieldName' field from
* @param fieldName field name to retrieve from 'entity' object
*/
private Object getFieldValue( Object entity, String fieldName )
{
Object entityFieldValue;
String firstLetter = fieldName.substring( 0, 1 );
String remainder = fieldName.substring( 1 );
String lowerKey = firstLetter.toLowerCase().concat( remainder );
String upperKey = firstLetter.toUpperCase().concat( remainder );
if( entity instanceof BackendlessUser )
{
entityFieldValue = ((BackendlessUser) entity).getProperty( lowerKey );
if( entityFieldValue == null )
entityFieldValue = ((BackendlessUser) entity).getProperty( upperKey );
}
else if( entity instanceof HashMap )
{
entityFieldValue = ((HashMap) entity).get( lowerKey );
if( entityFieldValue == null )
entityFieldValue = ((HashMap) entity).get( upperKey );
}
else
{
// retrieve entity field value
entityFieldValue = ReflectionUtil.getFieldValue( entity, lowerKey, upperKey );
}
return entityFieldValue;
}
}
}