mirror of
				https://github.com/Alfresco/alfresco-community-repo.git
				synced 2025-10-15 15:02:20 +00:00 
			
		
		
		
	Performance improvement: prevent unnecessary 304 revalidation requests for thumbnails in detailed view of My-Documents and Recently Modified Documents dashlets Fixed bean config problem (caused by r34662) Fix build break Refactored revalidation code to remove previously added WebScripts that are now surplus to requirements Performance improvement: prevent unnecessary 304 revalidation for avatars on site colleagues dashlet Performance improvement: prevent unnecessary 304 revalidation for avatars on following/follwers pages Performance improvement: prevent unnecessary 304 revalidation for avatars in activity feeds Performance improvement: prevent unecessary 304 revalidation for user avatar thumbnails in header WebScript Prevent 304 revalidations for unchanged thumbnails in document library, web preview and search git-svn-id: https://svn.alfresco.com/repos/alfresco-enterprise/alfresco/HEAD/root@34698 c4b6b30b-aa2e-2d43-bbcb-ca4b014f7261
		
			
				
	
	
		
			691 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			691 lines
		
	
	
		
			31 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2005-2011 Alfresco Software Limited.
 | |
|  *
 | |
|  * This file is part of Alfresco
 | |
|  *
 | |
|  * Alfresco is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU Lesser General Public License as published by
 | |
|  * the Free Software Foundation, either version 3 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * Alfresco is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU Lesser General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public License
 | |
|  * along with Alfresco. If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| package org.alfresco.repo.thumbnail;
 | |
| 
 | |
| import java.io.Serializable;
 | |
| import java.util.ArrayList;
 | |
| import java.util.Collections;
 | |
| import java.util.Date;
 | |
| import java.util.HashMap;
 | |
| import java.util.HashSet;
 | |
| import java.util.List;
 | |
| import java.util.Map;
 | |
| import java.util.Set;
 | |
| 
 | |
| import org.alfresco.model.ContentModel;
 | |
| import org.alfresco.repo.content.transform.RuntimeExecutableContentTransformerOptions;
 | |
| import org.alfresco.repo.content.transform.magick.ImageTransformationOptions;
 | |
| import org.alfresco.repo.content.transform.swf.SWFTransformationOptions;
 | |
| import org.alfresco.repo.node.NodeServicePolicies;
 | |
| import org.alfresco.repo.policy.Behaviour;
 | |
| import org.alfresco.repo.policy.JavaBehaviour;
 | |
| import org.alfresco.repo.policy.PolicyComponent;
 | |
| import org.alfresco.repo.rendition.executer.AbstractRenderingEngine;
 | |
| import org.alfresco.repo.rendition.executer.ImageRenderingEngine;
 | |
| import org.alfresco.repo.rendition.executer.ReformatRenderingEngine;
 | |
| import org.alfresco.repo.security.authentication.AuthenticationUtil;
 | |
| import org.alfresco.service.cmr.rendition.RenditionDefinition;
 | |
| import org.alfresco.service.cmr.rendition.RenditionService;
 | |
| import org.alfresco.service.cmr.rendition.RenditionServiceException;
 | |
| import org.alfresco.service.cmr.repository.ChildAssociationRef;
 | |
| import org.alfresco.service.cmr.repository.ContentData;
 | |
| import org.alfresco.service.cmr.repository.NodeRef;
 | |
| import org.alfresco.service.cmr.repository.NodeService;
 | |
| import org.alfresco.service.cmr.repository.TransformationOptions;
 | |
| import org.alfresco.service.cmr.thumbnail.FailedThumbnailInfo;
 | |
| import org.alfresco.service.cmr.thumbnail.ThumbnailException;
 | |
| import org.alfresco.service.cmr.thumbnail.ThumbnailParentAssociationDetails;
 | |
| import org.alfresco.service.cmr.thumbnail.ThumbnailService;
 | |
| import org.alfresco.service.namespace.NamespaceService;
 | |
| import org.alfresco.service.namespace.QName;
 | |
| import org.alfresco.service.namespace.RegexQNamePattern;
 | |
| import org.alfresco.util.GUID;
 | |
| import org.apache.commons.logging.Log;
 | |
| import org.apache.commons.logging.LogFactory;
 | |
| import org.springframework.extensions.surf.util.ParameterCheck;
 | |
| 
 | |
| /**
 | |
|  * @author Roy Wetherall
 | |
|  * @author Neil McErlean
 | |
|  */
 | |
| public class ThumbnailServiceImpl implements ThumbnailService,
 | |
|                                              NodeServicePolicies.BeforeCreateNodePolicy,
 | |
|                                              NodeServicePolicies.OnCreateNodePolicy
 | |
| {
 | |
|     /** Logger */
 | |
|     private static Log logger = LogFactory.getLog(ThumbnailServiceImpl.class);
 | |
|     
 | |
|     /** Error messages */
 | |
| //    private static final String ERR_NO_CREATE = "Thumbnail could not be created as required transformation is not supported from {0} to {1}";
 | |
|     private static final String ERR_DUPLICATE_NAME = "Thumbnail could not be created because of a duplicate name";
 | |
|     private static final String ERR_NO_PARENT = "Thumbnail has no parent so update cannot take place.";
 | |
|     
 | |
|     /** Mimetype wildcard postfix */
 | |
|     private static final String SUBTYPES_POSTFIX = "/*";
 | |
|     
 | |
|     /** Node service */
 | |
|     private NodeService nodeService;
 | |
|     
 | |
|     /** Thumbnail registry */
 | |
|     private ThumbnailRegistry thumbnailRegistry;
 | |
|     
 | |
|     /** Rendition service */
 | |
|     private RenditionService renditionService;
 | |
|     
 | |
|     /**
 | |
|      * The policy component.
 | |
|      * @since 3.5.0
 | |
|      */
 | |
|     private PolicyComponent policyComponent;
 | |
| 
 | |
|     
 | |
|     /**
 | |
|      * Set the rendition service.
 | |
|      * 
 | |
|      * @param renditionService
 | |
|      */
 | |
|     public void setRenditionService(RenditionService renditionService)
 | |
|     {
 | |
|         this.renditionService = renditionService;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Set the node service
 | |
|      * 
 | |
|      * @param nodeService   node service
 | |
|      */
 | |
|     public void setNodeService(NodeService nodeService)
 | |
|     {
 | |
|         this.nodeService = nodeService;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Set thumbnail registry
 | |
|      * 
 | |
|      * @param thumbnailRegistry     thumbnail registry
 | |
|      */
 | |
|     public void setThumbnailRegistry(ThumbnailRegistry thumbnailRegistry)
 | |
|     {
 | |
|         this.thumbnailRegistry = thumbnailRegistry;
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * Set the policy component to listen for various events
 | |
|      * @since 3.5.0
 | |
|      */
 | |
|     public void setPolicyComponent(PolicyComponent policyComponent)
 | |
|     {
 | |
|         this.policyComponent = policyComponent;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Registers to listen for events of interest.
 | |
|      * @since 3.5.0
 | |
|      */
 | |
|     public void init()
 | |
|     {
 | |
|         policyComponent.bindClassBehaviour(
 | |
|                 NodeServicePolicies.OnCreateNodePolicy.QNAME,
 | |
|                 ContentModel.TYPE_THUMBNAIL,
 | |
|                 new JavaBehaviour(this, "onCreateNode", Behaviour.NotificationFrequency.EVERY_EVENT));
 | |
|         policyComponent.bindClassBehaviour(
 | |
|                 NodeServicePolicies.BeforeCreateNodePolicy.QNAME,
 | |
|                 ContentModel.TYPE_FAILED_THUMBNAIL,
 | |
|                 new JavaBehaviour(this, "beforeCreateNode", Behaviour.NotificationFrequency.EVERY_EVENT));
 | |
|     }
 | |
|     
 | |
|     public void beforeCreateNode(
 | |
|             NodeRef parentRef,
 | |
|             QName assocTypeQName,
 | |
|             QName assocQName,
 | |
|             QName nodeTypeQName)
 | |
|     {
 | |
|         // When a thumbnail has failed, we must delete any existing (successful) thumbnails of that thumbnailDefinition.
 | |
|         if (ContentModel.TYPE_FAILED_THUMBNAIL.equals(nodeTypeQName))
 | |
|         {
 | |
|             // In fact there should only be zero or one such thumbnails
 | |
|             Set<QName> childNodeTypes = new HashSet<QName>();
 | |
|             childNodeTypes.add(ContentModel.TYPE_THUMBNAIL);
 | |
|             List<ChildAssociationRef> existingThumbnails = nodeService.getChildAssocs(parentRef, childNodeTypes);
 | |
|             
 | |
|             for (ChildAssociationRef chAssRef : existingThumbnails)
 | |
|             {
 | |
|                 if (chAssRef.getQName().equals(assocQName))
 | |
|                 {
 | |
|                     if (logger.isDebugEnabled())
 | |
|                     {
 | |
|                         StringBuilder msg = new StringBuilder();
 | |
|                         msg.append("Deleting thumbnail node ").append(chAssRef.getChildRef());
 | |
|                         logger.debug(msg.toString());
 | |
|                     }
 | |
|                     nodeService.deleteNode(chAssRef.getChildRef());
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         // We can't respond to the creation of a cm:thumbnail node at this point as they are created with
 | |
|         // temporary assoc qnames and so cannot be matched to the relevant thumbnail definition.
 | |
|         // Instead we must do it "onCreateNode()"
 | |
|     }
 | |
|     
 | |
|     public void onCreateNode(ChildAssociationRef childAssoc)
 | |
|     {
 | |
|         // When a thumbnail succeeds, we must delete any existing thumbnail failure nodes.
 | |
|         String thumbnailName = (String) nodeService.getProperty(childAssoc.getChildRef(), ContentModel.PROP_NAME);
 | |
|         
 | |
|         // Update the parent node with the thumbnail update...
 | |
|         addThumbnailModificationData(childAssoc.getChildRef(), thumbnailName);
 | |
|         
 | |
|         // In fact there should only be zero or one such failedThumbnails
 | |
|         Map<String, FailedThumbnailInfo> failures = getFailedThumbnails(childAssoc.getParentRef());
 | |
|         FailedThumbnailInfo existingFailedThumbnail = failures.get(thumbnailName);
 | |
|         
 | |
|         if (existingFailedThumbnail != null)
 | |
|         {
 | |
|             if (logger.isDebugEnabled())
 | |
|             {
 | |
|                 StringBuilder msg = new StringBuilder();
 | |
|                 msg.append("Deleting failedThumbnail node ").append(existingFailedThumbnail.getFailedThumbnailNode());
 | |
|                 logger.debug(msg.toString());
 | |
|             }
 | |
|             nodeService.deleteNode(existingFailedThumbnail.getFailedThumbnailNode());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     
 | |
|     /**
 | |
|      * @see org.alfresco.service.cmr.thumbnail.ThumbnailService#getThumbnailRegistry()
 | |
|      */
 | |
|     public ThumbnailRegistry getThumbnailRegistry()
 | |
|     {
 | |
|        return this.thumbnailRegistry;
 | |
|     }    
 | |
|     
 | |
|     /**
 | |
|      * @see org.alfresco.service.cmr.thumbnail.ThumbnailService#createThumbnail(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.namespace.QName, java.lang.String, org.alfresco.service.cmr.repository.TransformationOptions, java.lang.String)
 | |
|      */
 | |
|     public NodeRef createThumbnail(NodeRef node, QName contentProperty, String mimetype, TransformationOptions transformationOptions, String thumbnailName)
 | |
|     {
 | |
|         return createThumbnail(node, contentProperty, mimetype, transformationOptions, thumbnailName, null);
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * @see org.alfresco.service.cmr.thumbnail.ThumbnailService#createThumbnail(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.namespace.QName, java.lang.String, org.alfresco.service.cmr.repository.TransformationOptions, java.lang.String, org.alfresco.service.cmr.thumbnail.ThumbnailParentAssociationDetails)
 | |
|      */
 | |
|     public NodeRef createThumbnail(final NodeRef node, final QName contentProperty, final String mimetype,
 | |
|             final TransformationOptions transformationOptions, final String thumbnailName, final ThumbnailParentAssociationDetails assocDetails)
 | |
|     {
 | |
|         // Parameter check
 | |
|         ParameterCheck.mandatory("node", node);
 | |
|         ParameterCheck.mandatory("contentProperty", contentProperty);
 | |
|         ParameterCheck.mandatoryString("mimetype", mimetype);
 | |
|         ParameterCheck.mandatory("transformationOptions", transformationOptions);
 | |
| 
 | |
|         if (logger.isDebugEnabled() == true)
 | |
|         {
 | |
|             logger.debug("Creating thumbnail (node=" + node.toString() + "; contentProperty="
 | |
|                         + contentProperty.toString() + "; mimetype=" + mimetype);
 | |
|         }
 | |
|         
 | |
|         if (thumbnailName != null)
 | |
|         {
 | |
|             NodeRef existingThumbnail = getThumbnailByName(node, contentProperty, thumbnailName);
 | |
|             if (existingThumbnail != null)
 | |
|             {
 | |
|                 if (logger.isDebugEnabled() == true)
 | |
|                 {
 | |
|                     logger.debug("Creating thumbnail: There is already a thumbnail with the name '" + thumbnailName + "' (node=" + node.toString() + "; contentProperty=" + contentProperty.toString() + "; mimetype=" + mimetype);
 | |
|                 }
 | |
|                 
 | |
|                 // Return the thumbnail that has already been created
 | |
|                 return existingThumbnail;
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         return AuthenticationUtil.runAs(new AuthenticationUtil.RunAsWork<NodeRef>()
 | |
|         {
 | |
|             public NodeRef doWork() throws Exception
 | |
|             {
 | |
|                 return createThumbnailNode( node, 
 | |
|                                             contentProperty,
 | |
|                                             mimetype, 
 | |
|                                             transformationOptions, 
 | |
|                                             thumbnailName, 
 | |
|                                             assocDetails);
 | |
|             }
 | |
|         }, AuthenticationUtil.getSystemUserName());
 | |
|     }
 | |
|     
 | |
|     private QName getThumbnailQName(String localThumbnailName)
 | |
|     {
 | |
|         if (localThumbnailName == null || localThumbnailName.length() == 0)
 | |
|         {
 | |
|             localThumbnailName = GUID.generate();
 | |
|         }
 | |
|         
 | |
|         // We're prepending the cm namespace here.
 | |
|         QName thumbnailQName = QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, localThumbnailName);
 | |
|         return thumbnailQName;
 | |
|     }
 | |
| 
 | |
|     private String getRenderingEngineNameFor(TransformationOptions options)
 | |
|     {
 | |
|         if (options instanceof ImageTransformationOptions)
 | |
|         {
 | |
|             return ImageRenderingEngine.NAME;
 | |
|         }
 | |
|         else if (options instanceof SWFTransformationOptions)
 | |
|         {
 | |
|             return ReformatRenderingEngine.NAME;
 | |
|         }
 | |
|         else if (options instanceof RuntimeExecutableContentTransformerOptions)
 | |
|         {
 | |
|             return ReformatRenderingEngine.NAME;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             // TODO What can we do here? Can we treat this as an error?
 | |
|             // Isn't this a 'standard' TransformationOptions?
 | |
|             return "";
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * This method returns the NodeRef for the thumbnail, using the ChildAssociationRef
 | |
|      * that links the sourceNode to its associated Thumbnail node.
 | |
|      * 
 | |
|      * @param thumbnailRef the ChildAssociationRef containing the child NodeRef.
 | |
|      * @return the NodeRef of the thumbnail itself.
 | |
|      */
 | |
|     public NodeRef getThumbnailNode(ChildAssociationRef thumbnailRef)
 | |
|     {
 | |
|         return thumbnailRef.getChildRef();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @see org.alfresco.service.cmr.thumbnail.ThumbnailService#updateThumbnail(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.cmr.repository.TransformationOptions)
 | |
|      */
 | |
|     public void updateThumbnail(final NodeRef thumbnail, final TransformationOptions transformationOptions)
 | |
|     {
 | |
|         if (logger.isDebugEnabled() == true)
 | |
|         {
 | |
|             logger.debug("Updating thumbnail (thumbnail=" + thumbnail.toString() + ")");
 | |
|         }
 | |
|         
 | |
|         // First check that we are dealing with a rendition object
 | |
|         if (renditionService.isRendition(thumbnail))
 | |
|         {
 | |
|             // Get the node that is the source of the thumbnail
 | |
|             ChildAssociationRef parentAssoc = renditionService.getSourceNode(thumbnail);
 | |
|             
 | |
|             if (parentAssoc == null)
 | |
|             {
 | |
|                 if (logger.isDebugEnabled() == true)
 | |
|                 {
 | |
|                     logger.debug("Updating thumbnail: The thumbnails parent cannot be found (thumbnail=" + thumbnail.toString() + ")");
 | |
|                 }
 | |
|                 throw new ThumbnailException(ERR_NO_PARENT);
 | |
|             }
 | |
|             
 | |
|             final QName renditionAssociationName = parentAssoc.getQName();
 | |
|             NodeRef sourceNode = parentAssoc.getParentRef();
 | |
| 
 | |
|             // Get the content property
 | |
|             QName contentProperty = (QName)nodeService.getProperty(thumbnail, ContentModel.PROP_CONTENT_PROPERTY_NAME);
 | |
| 
 | |
|             // Set the basic detail of the transformation options
 | |
|             transformationOptions.setSourceNodeRef(sourceNode);
 | |
|             transformationOptions.setSourceContentProperty(contentProperty);
 | |
|             transformationOptions.setTargetContentProperty(ContentModel.PROP_CONTENT);
 | |
| 
 | |
|             // Do the thumbnail transformation. Rendition Definitions are persisted underneath the Data Dictionary for which Group ALL
 | |
|             // has Consumer access by default. However, we cannot assume that that access level applies for all deployments. See ALF-7334.
 | |
|             RenditionDefinition rendDefn = AuthenticationUtil.runAs(new AuthenticationUtil.RunAsWork<RenditionDefinition>()
 | |
|                 {
 | |
|                     @Override
 | |
|                     public RenditionDefinition doWork() throws Exception
 | |
|                     {
 | |
|                         return renditionService.loadRenditionDefinition(renditionAssociationName);
 | |
|                     }
 | |
|                 }, AuthenticationUtil.getSystemUserName());
 | |
|             
 | |
|             if (rendDefn == null)
 | |
|             {
 | |
|                 String renderingEngineName = getRenderingEngineNameFor(transformationOptions);
 | |
| 
 | |
|                 rendDefn = renditionService.createRenditionDefinition(parentAssoc.getQName(), renderingEngineName);
 | |
|             }
 | |
|             Map<String, Serializable> params = thumbnailRegistry.getThumbnailRenditionConvertor().convert(transformationOptions, null);
 | |
|             for (String key : params.keySet())
 | |
|             {
 | |
|                 rendDefn.setParameterValue(key, params.get(key));
 | |
|             }
 | |
|             
 | |
|             renditionService.render(sourceNode, rendDefn);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             if (logger.isDebugEnabled() == true)
 | |
|             {
 | |
|                 logger.debug("Updating thumbnail: cannot update a thumbnail node that isn't the correct thumbnail type (thumbnail=" + thumbnail.toString() + ")");
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @see org.alfresco.service.cmr.thumbnail.ThumbnailService#getThumbnailByName(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.namespace.QName, java.lang.String)
 | |
|      */
 | |
|     public NodeRef getThumbnailByName(NodeRef node, QName contentProperty, String thumbnailName)
 | |
|     {
 | |
|         //
 | |
|         // NOTE:
 | |
|         //
 | |
|         // Since there is not an easy alternative and for clarity the node service is being used to retrieve the thumbnails.
 | |
|         // If retrieval performance becomes an issue then this code can be replaced
 | |
|         //
 | |
|         
 | |
|         if (logger.isDebugEnabled() == true)
 | |
|         {
 | |
|             logger.debug("Getting thumbnail by name (nodeRef=" + node.toString() + "; contentProperty=" + contentProperty.toString() + "; thumbnailName=" + thumbnailName + ")");
 | |
|         }
 | |
|         
 | |
|         // Thumbnails have a cm: prefix.
 | |
|         QName namespacedThumbnailName = QName.createQName(NamespaceService.CONTENT_MODEL_1_0_URI, thumbnailName);
 | |
|         
 | |
|         ChildAssociationRef existingRendition = renditionService.getRenditionByName(node, namespacedThumbnailName);
 | |
|         NodeRef thumbnail = null;
 | |
|         
 | |
|         // Check the child to see if it matches the content property we are concerned about.
 | |
|         // We can assume there will only ever be one per content property since createThumbnail enforces this.
 | |
|         if (existingRendition != null)
 | |
|         {
 | |
|             NodeRef child = existingRendition.getChildRef();
 | |
|             Serializable contentPropertyName = this.nodeService.getProperty(child, ContentModel.PROP_CONTENT_PROPERTY_NAME);
 | |
|             if (contentProperty.equals(contentPropertyName) == true)
 | |
|             {
 | |
|                 thumbnail = child;
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         return thumbnail;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @see org.alfresco.service.cmr.thumbnail.ThumbnailService#getThumbnails(org.alfresco.service.cmr.repository.NodeRef, org.alfresco.service.namespace.QName, java.lang.String, org.alfresco.service.cmr.repository.TransformationOptions)
 | |
|      */
 | |
|     public List<NodeRef> getThumbnails(NodeRef node, QName contentProperty, String mimetype, TransformationOptions options)
 | |
|     {
 | |
|         List<NodeRef> thumbnails = new ArrayList<NodeRef>(5);
 | |
|         
 | |
|         //
 | |
|         // NOTE:
 | |
|         //
 | |
|         // Since there is not an easy alternative and for clarity the node service is being used to retrieve the thumbnails.
 | |
|         // If retrieval performance becomes an issue then this code can be replaced
 | |
|         //
 | |
|         
 | |
|         if (logger.isDebugEnabled() == true)
 | |
|         {
 | |
|             logger.debug("Getting thumbnails (nodeRef=" + node.toString() + "; contentProperty=" + contentProperty.toString() + "; mimetype=" + mimetype + ")");
 | |
|         }
 | |
|         
 | |
|         List<ChildAssociationRef> renditions = this.renditionService.getRenditions(node);
 | |
|         
 | |
|         for (ChildAssociationRef assoc : renditions)
 | |
|         {
 | |
|             // Check the child to see if it matches the content property we are concerned about.
 | |
|             // We can assume there will only ever be one per content property since createThumbnail enforces this.
 | |
|             NodeRef child = assoc.getChildRef();
 | |
|             if (contentProperty.equals(this.nodeService.getProperty(child, ContentModel.PROP_CONTENT_PROPERTY_NAME)) == true &&
 | |
|                 matchMimetypeOptions(child, mimetype, options) == true)
 | |
|             {
 | |
|                 thumbnails.add(child);
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         //TODO Ensure this doesn't return non-thumbnail renditions.
 | |
|         return thumbnails;
 | |
|     }
 | |
|     
 | |
|     @Override
 | |
|     public Map<String, FailedThumbnailInfo> getFailedThumbnails(NodeRef sourceNode)
 | |
|     {
 | |
|         Map<String, FailedThumbnailInfo> result = Collections.emptyMap();
 | |
|         
 | |
|         if (nodeService.hasAspect(sourceNode, ContentModel.ASPECT_FAILED_THUMBNAIL_SOURCE))
 | |
|         {
 | |
|             List<ChildAssociationRef> failedThumbnailChildren = nodeService.getChildAssocs(sourceNode,
 | |
|                                                  ContentModel.ASSOC_FAILED_THUMBNAIL, RegexQNamePattern.MATCH_ALL);
 | |
|             result = new HashMap<String, FailedThumbnailInfo>();
 | |
|             for (ChildAssociationRef chAssRef : failedThumbnailChildren)
 | |
|             {
 | |
|                 final QName failedThumbnailName = chAssRef.getQName();
 | |
|                 NodeRef failedThumbnailNode = chAssRef.getChildRef();
 | |
|                 Map<QName, Serializable> props = nodeService.getProperties(failedThumbnailNode);
 | |
|                 Date failureDateTime = (Date)props.get(ContentModel.PROP_FAILED_THUMBNAIL_TIME);
 | |
|                 int failureCount = (Integer)props.get(ContentModel.PROP_FAILURE_COUNT);
 | |
|                 
 | |
|                 final FailedThumbnailInfo failedThumbnailInfo = new FailedThumbnailInfo(failedThumbnailName.getLocalName(),
 | |
|                                                                                 failureDateTime, failureCount,
 | |
|                                                                                 failedThumbnailNode);
 | |
|                 result.put(failedThumbnailName.getLocalName(), failedThumbnailInfo);
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     
 | |
|     /**
 | |
|      * Determine whether the thumbnail meta-data matches the given mimetype and options
 | |
|      * 
 | |
|      * If mimetype and transformation options are null then match is guarenteed
 | |
|      * 
 | |
|      * @param  thumbnail     thumbnail node reference
 | |
|      * @param  mimetype      mimetype
 | |
|      * @param  options       transformation options
 | |
|      * @return boolean       true if the mimetype and options match the thumbnail metadata, false otherwise
 | |
|      */
 | |
|     private boolean matchMimetypeOptions(NodeRef thumbnail, String mimetype, TransformationOptions options)
 | |
|     {
 | |
|         boolean result = true;
 | |
|         
 | |
|         if (mimetype != null)
 | |
|         {
 | |
|             // Check the mimetype
 | |
|             String thumbnailMimetype = ((ContentData) this.nodeService.getProperty(thumbnail, ContentModel.PROP_CONTENT)).getMimetype();
 | |
| 
 | |
|             if (mimetype.endsWith(SUBTYPES_POSTFIX))
 | |
|             {
 | |
|                 String baseMimetype = mimetype.substring(0, mimetype.length() - SUBTYPES_POSTFIX.length());
 | |
|                 if (thumbnailMimetype == null || thumbnailMimetype.startsWith(baseMimetype) == false)
 | |
|                 {
 | |
|                     result = false;
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 if (mimetype.equals(thumbnailMimetype) == false)
 | |
|                 {
 | |
|                     result = false;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         if (result != false && options != null)
 | |
|         {
 | |
|             // TODO .. check for matching options here ...
 | |
|         }
 | |
|         
 | |
|         return result;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Creates a {@link RenditionDefinition} with no parameters set.
 | |
|      * @param thumbnailName
 | |
|      * @param transformationOptions
 | |
|      * @return
 | |
|      */
 | |
|     private RenditionDefinition createRawRenditionDefinition(QName thumbnailQName,
 | |
|                 final TransformationOptions transformationOptions)
 | |
|     {
 | |
|         // Create the renditionDefinition
 | |
|         String renderingEngineName = getRenderingEngineNameFor(transformationOptions);
 | |
|         RenditionDefinition definition = renditionService.createRenditionDefinition(thumbnailQName, renderingEngineName);
 | |
|         return definition;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Creates a fully parameterized {@link RenditionDefinition}.
 | |
|      * @param contentProperty
 | |
|      * @param mimetype
 | |
|      * @param transformationOptions
 | |
|      * @param thumbnailName
 | |
|      * @param assocDetails
 | |
|      * @return
 | |
|      */
 | |
|     private RenditionDefinition createRenditionDefinition(final QName contentProperty, final String mimetype,
 | |
|                 final TransformationOptions transformationOptions, final QName thumbnailQName,
 | |
|                 final ThumbnailParentAssociationDetails assocDetails)
 | |
|     {
 | |
|         RenditionDefinition definition = createRawRenditionDefinition(thumbnailQName, transformationOptions);
 | |
| 
 | |
|         // Convert the TransformationOptions and ThumbnailParentAssocDetails to
 | |
|         // rendition-style parameters
 | |
|         Map<String, Serializable> params = thumbnailRegistry.getThumbnailRenditionConvertor().convert(transformationOptions, assocDetails);
 | |
|         // Add the other parameters given in this method signature.
 | |
|         params.put(AbstractRenderingEngine.PARAM_SOURCE_CONTENT_PROPERTY, contentProperty);
 | |
|         params.put(AbstractRenderingEngine.PARAM_MIME_TYPE, mimetype);
 | |
| 
 | |
|         // Set the parameters on the rendition definition.
 | |
|         definition.addParameterValues(params);
 | |
|         return definition;
 | |
|     }
 | |
| 
 | |
|     private NodeRef createThumbnailNode(final NodeRef node, final QName contentProperty,
 | |
|                 final String mimetype, final TransformationOptions transformationOptions, final String thumbnailName,
 | |
|                 final ThumbnailParentAssociationDetails assocDetails)
 | |
|     {
 | |
|         // Get the name of the thumbnail and add to properties map
 | |
|         QName thumbnailQName = getThumbnailQName(thumbnailName);
 | |
|         RenditionDefinition definition = createRenditionDefinition(contentProperty, mimetype,
 | |
|                     transformationOptions, thumbnailQName, assocDetails);
 | |
|         try
 | |
|         {
 | |
|             ChildAssociationRef thumbnailAssoc = renditionService.render(node, definition);
 | |
|             NodeRef thumbnail = getThumbnailNode(thumbnailAssoc);
 | |
|             setThumbnailNameProperty(thumbnail, thumbnailName);
 | |
|             return thumbnail;
 | |
|         } catch (RenditionServiceException rsx)
 | |
|         {
 | |
|             throw new ThumbnailException(rsx.getMessage(), rsx);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sets the thumbnail name if the rendition is of type cm:thumbnail.
 | |
|      * @param thumbnailAssoc
 | |
|      * @param thumbnailName
 | |
|      */
 | |
|     private void setThumbnailNameProperty(NodeRef thumbnail, String thumbnailName)
 | |
|     {
 | |
|         if (thumbnailName != null && thumbnailName.length() > 0)
 | |
|         {
 | |
|             if (ContentModel.TYPE_THUMBNAIL.equals(nodeService.getType(thumbnail)))
 | |
|             {
 | |
|                 nodeService.setProperty(thumbnail, ContentModel.PROP_THUMBNAIL_NAME, thumbnailName);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     
 | |
|     /**
 | |
|      * <p>Updates the parent of the supplied {@link NodeRef} to ensure that it has the "cm:thumbnailModification" aspect
 | |
|      * and sets the last modification data for it.</p>
 | |
|      * @param nodeRef A {@link NodeRef} representing a thumbnail to provide last modification data for.
 | |
|      */
 | |
|     @SuppressWarnings("unchecked")
 | |
|     public void addThumbnailModificationData(NodeRef nodeRef, String thumbnailName)
 | |
|     {
 | |
|         if (nodeService.exists(nodeRef))
 | |
|         {
 | |
|            if (thumbnailName != null && !nodeRef.toString().endsWith(thumbnailName))
 | |
|            {
 | |
|                Date modified = (Date)nodeService.getProperty(nodeRef, ContentModel.PROP_MODIFIED);
 | |
|                if (modified != null)
 | |
|                {
 | |
|                    // Get the last modified value as a timestamp...
 | |
|                    Long timestamp = modified.getTime();
 | |
|                    
 | |
|                    // Create the value we want to set...
 | |
|                    String lastModifiedValue = thumbnailName + ":" + timestamp;
 | |
|                    
 | |
|                    // Get the parent node (there should be only one) and apply the aspect and
 | |
|                    // set the property to indicate which thumbnail the checksum refers to...
 | |
|                    for (ChildAssociationRef parent: nodeService.getParentAssocs(nodeRef))
 | |
|                    {
 | |
|                       List<String> thumbnailMods = null;
 | |
|                        
 | |
|                       NodeRef parentNode = parent.getParentRef();
 | |
|                       if (nodeService.hasAspect(parentNode, ContentModel.ASPECT_THUMBNAIL_MODIFICATION))
 | |
|                       {
 | |
|                           // The node already has the aspect, check to see if the current thumbnail modification exists...
 | |
|                           thumbnailMods = (List<String>) nodeService.getProperty(parentNode, ContentModel.PROP_LAST_THUMBNAIL_MODIFICATION_DATA);
 | |
|                           
 | |
|                           // If we have previously set last modified thumbnail data then it will exist as part of the multi-value
 | |
|                           // property. The value will consist of the "cm:thumbnailName" value delimited with a ":" and then the
 | |
|                           // timestamp. We need to find the appropriate entry in the multivalue property and then update it
 | |
|                           String target = null;
 | |
|                           for (String currThumbnailMod: thumbnailMods)
 | |
|                           {
 | |
|                               if (currThumbnailMod.startsWith(thumbnailName))
 | |
|                               {
 | |
|                                   target = currThumbnailMod;
 | |
|                               }
 | |
|                           }
 | |
|                           
 | |
|                           // Remove the previous value
 | |
|                           if (target != null)
 | |
|                           {
 | |
|                               thumbnailMods.remove(target);
 | |
|                           }
 | |
|                           
 | |
|                           // Add the timestamp...
 | |
|                           thumbnailMods.add(lastModifiedValue);
 | |
|                           
 | |
|                           // Set the property...
 | |
|                           nodeService.setProperty(parentNode, ContentModel.PROP_LAST_THUMBNAIL_MODIFICATION_DATA, (Serializable) thumbnailMods);
 | |
|                       }
 | |
|                       else
 | |
|                       {
 | |
|                           // If the aspect has not previously been added then we'll need to set it now...
 | |
|                           thumbnailMods = new ArrayList<String>();
 | |
|                           thumbnailMods.add(lastModifiedValue);
 | |
|                           
 | |
|                           // Add the aspect with the new property...
 | |
|                           Map<QName, Serializable> properties = new HashMap<QName, Serializable>();
 | |
|                           properties.put(ContentModel.PROP_LAST_THUMBNAIL_MODIFICATION_DATA, (Serializable) thumbnailMods);
 | |
|                           nodeService.addAspect(parentNode, ContentModel.ASPECT_THUMBNAIL_MODIFICATION, properties);
 | |
|                       }
 | |
|                    }
 | |
|                }
 | |
|            }
 | |
|         }
 | |
|     }
 | |
| }
 |