/*
 * Copyright (C) 2005-2010 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 .
 */
package org.alfresco.cmis.renditions;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.Map.Entry;
import org.alfresco.cmis.CMISFilterNotValidException;
import org.alfresco.cmis.CMISRendition;
import org.alfresco.cmis.CMISRenditionService;
import org.alfresco.error.AlfrescoRuntimeException;
import org.alfresco.model.ContentModel;
import org.alfresco.model.RenditionModel;
import org.alfresco.repo.content.transform.magick.ImageResizeOptions;
import org.alfresco.repo.content.transform.magick.ImageTransformationOptions;
import org.alfresco.repo.thumbnail.ThumbnailDefinition;
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.thumbnail.ThumbnailService;
import org.alfresco.service.namespace.RegexQNamePattern;
/**
 * Rendition Service Implementation
 * 
 * @author Stas Sokolovsky
 */
public class CMISRenditionServiceImpl implements CMISRenditionService
{
    /** Rendition filter constants */
    private static final String FILTER_WILDCARD = "*";
    private static final String FILTER_NONE = "cmis:none";
    private static final String FILTER_DELIMITER = ",";
    private static final String SUBTYPES_POSTFIX = "/*";
    private static final String SUBTYPES_DELIMITER = "/";
    /** Service dependencies */
    private ThumbnailService thumbnailService;
    private NodeService nodeService;
    /** Kind to thumbnail mapping */
    private Map> kindToThumbnailNames = new HashMap>();
    private Map thumbnailNamesToKind = new HashMap();
    /** Custom renditions */
    private CustomRenditionsCache customRenditionsCache;
    /**
     * @throws CMISFilterNotValidException 
     * @see org.alfresco.cmis.CMISRenditionService#getRenditions(org.alfresco.service.cmr.repository.NodeRef, java.lang.String)
     */
    public List getRenditions(NodeRef node, String renditionFilter) throws CMISFilterNotValidException
    {
        Collection result = null;
        ThumbnailFilter thumbnailFilter = getThumbnailFilter(renditionFilter);
        if (thumbnailFilter != null)
        {
            result = getRenditions(node, thumbnailFilter);
        }
        return result != null ? new ArrayList(result) : null;
    }
    /**
     * Get renditions for a document.
     * 
     * @param node node reference of document
     * @param thumbnailFilter thumbnail filter
     * @return set of renditions
     */
    private Set getRenditions(NodeRef node, ThumbnailFilter thumbnailFilter)
    {
        Set result = new HashSet();
        if (thumbnailFilter.isAny())
        {
            result.addAll(getAllRenditions(node));
        }
        else
        {
            for (String thumbnailName : thumbnailFilter.getThumbnailNames())
            {
                CMISRendition rendition = getRenditionByThumbnailName(node, thumbnailName);
                if (rendition != null)
                {
                    result.add(rendition);
                }
            }
            for (String mimetype : thumbnailFilter.getMimetypes())
            {
                result.addAll(getRenditionByMimetype(node, mimetype));
            }
        }
        result.addAll(getCustomRenditions(thumbnailFilter));
        return result;
    }
    /**
     * Get rendition by thumbnail name.
     * 
     * @param node node reference of document
     * @param thumbnailName thumbnail name
     * @return rendition
     */
    private CMISRendition getRenditionByThumbnailName(NodeRef node, String thumbnailName)
    {
        CMISRendition result = null;
        NodeRef thumbnailNode = thumbnailService.getThumbnailByName(node, ContentModel.PROP_CONTENT, thumbnailName);
        
        if (thumbnailNode != null)
        {
            result = getRendition(thumbnailNode, node);
        }
        return result;
    }
    /**
     * Get rendition by mimetype.
     * 
     * @param node node reference of document
     * @param mimetype rendition mimetype
     * @return list of renditions
     */
    private List getRenditionByMimetype(NodeRef node, String mimetype)
    {
        List result = new ArrayList();
        List thumbnails = thumbnailService.getThumbnails(node, ContentModel.PROP_CONTENT, mimetype, null);
        if (thumbnails != null)
        {
            for (NodeRef thumbnailNode : thumbnails)
            {
                CMISRendition rendition = getRendition(thumbnailNode, node);
                if (rendition != null)
                {
                    result.add(rendition);
                }
            }
        }
        return result;
    }
    /**
     * Convert the rendition filter to thumbnail filter.
     * 
     * @param renditionFilter rendition filter
     * @return thumbnail filter
     * @throws CMISFilterNotValidException 
     */
    private ThumbnailFilter getThumbnailFilter(String renditionFilter) throws CMISFilterNotValidException
    {
        ThumbnailFilter result = null;
        if (renditionFilter != null && !renditionFilter.equals(FILTER_NONE))
        {
            // Scan the filter for whitespace, which is disallowed by the spec
            for (int i=0; i < renditionFilter.length(); i = renditionFilter.offsetByCodePoints(i, 1))
            {
                if (Character.isWhitespace(renditionFilter.codePointAt(i)))
                {
                    throw new CMISFilterNotValidException(renditionFilter);
                }
            }
            result = new ThumbnailFilter();
            if (renditionFilter.equals(FILTER_WILDCARD))
            {
                result.setAny(true);
            }
            else
            {
                String[] filterElements = renditionFilter.split(FILTER_DELIMITER);
                if (filterElements == null || filterElements.length < 1)
                {
                    throw new AlfrescoRuntimeException("Invalid rendition filter");
                }
                for (String filterElement : filterElements)
                {
                    filterElement = filterElement.trim();
                    if (filterElement.indexOf('/') == -1)
                    {
                        result.getKinds().add(filterElement);
                        List thumbnails = kindToThumbnailNames.get(filterElement);
                        if (thumbnails != null)
                        {
                            result.getThumbnailNames().addAll(thumbnails);
                        }
                    }
                    else
                    {
                        result.getMimetypes().add(filterElement);
                    }
                }
            }
        }
        return result;
    }
    /**
     * Get the thumbnail definition.
     * 
     * @param thumbnailName thumbnail name
     * @return thumbnail definition
     */
    private ThumbnailDefinition getThumbnailDefinition(String thumbnailName)
    {
        return thumbnailService.getThumbnailRegistry().getThumbnailDefinition(thumbnailName);
    }
    /**
     * Get the image attributes of thumbnail.
     * 
     * @param thumbnailName thumbnail name
     * @return image attributes
     */
    private ImageResizeOptions getImageAttributes(String thumbnailName)
    {
        ThumbnailDefinition thumbnailDefinition = getThumbnailDefinition(thumbnailName);
        if (thumbnailDefinition != null && thumbnailDefinition.getTransformationOptions() != null
                && thumbnailDefinition.getTransformationOptions() instanceof ImageTransformationOptions)
        {
            return ((ImageTransformationOptions) thumbnailDefinition.getTransformationOptions()).getResizeOptions();
        }
        return null;
    }
    /**
     * Create CMISRendition by thumbnailNode and documentNode.
     * 
     * @param thumbnailNode thumbnail node reference
     * @param documentNode document node reference
     * @return CMISRendition
     */
    private CMISRendition getRendition(NodeRef thumbnailNode, NodeRef documentNode)
    {
        CMISRenditionImpl rendition = null;
        String thumbnailName = getThumbnailName(thumbnailNode);
        String kind = thumbnailNamesToKind.get(thumbnailName);
        kind = (kind == null) ? thumbnailName : kind;
        
        rendition = new CMISRenditionImpl();
        ContentData contentData = (ContentData) nodeService.getProperty(thumbnailNode, ContentModel.PROP_CONTENT);
        rendition.setNodeRef(thumbnailNode);
        rendition.setStreamId(thumbnailNode.toString());
        rendition.setRenditionDocumentId(documentNode.toString());
        rendition.setTitle(thumbnailName);
        rendition.setKind(kind);
        rendition.setMimeType(contentData.getMimetype());
        rendition.setLength((int) contentData.getSize());
        ImageResizeOptions imageAttributes = getImageAttributes(thumbnailName);
        if (imageAttributes != null)
        {
            rendition.setWidth(imageAttributes.getWidth());
            rendition.setHeight(imageAttributes.getHeight());
        }
        return rendition;
    }
    private String getThumbnailName(NodeRef thumbnailNode)
    {
        String thumbnailName = null;
        List parentAssocs = nodeService.getParentAssocs(thumbnailNode,
                RenditionModel.ASSOC_RENDITION, RegexQNamePattern.MATCH_ALL);
        if (parentAssocs.size() == 1) {
            ChildAssociationRef parentAssoc = parentAssocs.get(0);
            thumbnailName = parentAssoc.getQName().getLocalName();
        }
        return thumbnailName;
    }
    /**
     * Get custom renditions.
     * 
     * @param thumbnailFilter thumbnail filter
     * @return list of renditions
     */
    private List getCustomRenditions(ThumbnailFilter filter)
    {
        List result = new ArrayList();
        if (customRenditionsCache != null)
        {
            if (filter.isAny())
            {
                result.addAll(customRenditionsCache.getAllRenditions());
            }
            else
            {
                for (String kind : filter.getKinds())
                {
                    List renditions = customRenditionsCache.getRenditionsByKind(kind);
                    if (renditions != null)
                    {
                        result.addAll(renditions);
                    }
                }
                for (String mimetype : filter.getMimetypes())
                {
                    List renditions = customRenditionsCache.getRenditionsByMimeType(mimetype);
                    if (renditions != null)
                    {
                        result.addAll(renditions);
                    }
                }
            }
        }
        return result;
    }
    /**
     * Get all renditions for a document.
     * 
     * @param node document node ref
     * @return list of renditions
     */
    private List getAllRenditions(NodeRef node)
    {
        return getRenditionByMimetype(node, null);
    }
    /**
     * Set rendition kind mapping.
     * 
     * @param RenditionKind to Thumbnail Definition mapping
     */
    public void setRenditionKindMapping(Map> renditionKinds)
    {
        this.kindToThumbnailNames = renditionKinds;
        for (Entry> entry : renditionKinds.entrySet())
        {
            for (String thumbnailName : entry.getValue())
            {
                thumbnailNamesToKind.put(thumbnailName, entry.getKey());
            }
        }
    }
    /**
     * Set custom renditions.
     * 
     * @param renditions list of renditions
     */
    public void setCustomRenditions(List renditions)
    {
        this.customRenditionsCache = new CustomRenditionsCache(renditions);
    }
    /**
     * Set the NodeService.
     * 
     * @param nodeService NodeService
     */
    public void setNodeService(NodeService nodeService)
    {
        this.nodeService = nodeService;
    }
    /**
     * Set the ThumbnailService.
     * 
     * @param thumbnailService thumbnailService
     */
    public void setThumbnailService(ThumbnailService thumbnailService)
    {
        this.thumbnailService = thumbnailService;
    }
    /**
     * Cache that aggregates renditions informaition to allow fast searching by kind and mimetype    
     */
    private class CustomRenditionsCache
    {
        private Map> renditionsByKind;
        private Map> renditionsByMimeType;
        private Map> renditionsByBaseMimeType;
        private List allRenditions;
        public CustomRenditionsCache(List renditions)
        {
            allRenditions = renditions;
            renditionsByKind = new HashMap>(renditions.size());
            renditionsByMimeType = new HashMap>(renditions.size());
            renditionsByBaseMimeType = new HashMap>(renditions.size());
            for (CMISRendition rendition : renditions)
            {
                String baseType = getBaseType(rendition.getMimeType());
                if (!renditionsByKind.containsKey(rendition.getKind()))
                {
                    renditionsByKind.put(rendition.getKind(), new ArrayList(1));
                }
                if (!renditionsByMimeType.containsKey(rendition.getMimeType()))
                {
                    renditionsByMimeType.put(rendition.getMimeType(), new ArrayList(1));
                }
                if (!renditionsByBaseMimeType.containsKey(baseType))
                {
                    renditionsByBaseMimeType.put(baseType, new ArrayList(1));
                }
                renditionsByKind.get(rendition.getKind()).add(rendition);
                renditionsByMimeType.get(rendition.getMimeType()).add(rendition);
                renditionsByBaseMimeType.get(baseType).add(rendition);
            }
        }
        public List getRenditionsByKind(String kind)
        {
            return renditionsByKind.get(kind);
        }
        public List getRenditionsByMimeType(String mimetype)
        {
            if (mimetype.endsWith(SUBTYPES_POSTFIX))
            {
                String baseMimetype = mimetype.substring(0, mimetype.length() - SUBTYPES_POSTFIX.length());
                return renditionsByBaseMimeType.get(baseMimetype);
            }
            else
            {
                return renditionsByMimeType.get(mimetype);
            }
        }
        public Collection getAllRenditions()
        {
            return allRenditions;
        }
        private String getBaseType(String mimetype)
        {
            String baseMymetype = mimetype;
            int subTypeIndex = mimetype.indexOf(SUBTYPES_DELIMITER);
            if (subTypeIndex > 0 || subTypeIndex < mimetype.length())
            {
                baseMymetype = mimetype.substring(0, subTypeIndex);
            }
            return baseMymetype;
        }
    }
    /**
     * Parsed RenditionFilter     
     */
    private class ThumbnailFilter
    {
        private List kinds = new ArrayList();
        private List thumbnailNames = new ArrayList();
        private List mimetypes = new ArrayList();
        private boolean any = false;
        public List getThumbnailNames()
        {
            return thumbnailNames;
        }
        public List getMimetypes()
        {
            return mimetypes;
        }
        public List getKinds()
        {
            return kinds;
        }
        public boolean isAny()
        {
            return any;
        }
        public void setAny(boolean any)
        {
            this.any = any;
        }
    }
}