/*
 * Copyright (C) 2005-2009 Alfresco Software Limited.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 * This program 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 General Public License for more details.
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
 * As a special exception to the terms and conditions of version 2.0 of 
 * the GPL, you may redistribute this Program in connection with Free/Libre 
 * and Open Source Software ("FLOSS") applications as described in Alfresco's 
 * FLOSS exception.  You should have recieved a copy of the text describing 
 * the FLOSS exception, and it is also available here: 
 * http://www.alfresco.com/legal/licensing"
 */
package org.alfresco.repo.domain.encoding;
import java.io.Serializable;
import org.alfresco.error.AlfrescoRuntimeException;
import org.alfresco.repo.cache.SimpleCache;
import org.alfresco.util.Pair;
import org.alfresco.util.ParameterCheck;
/**
 * Abstract implementation for Encoding DAO.
 * 
 * This provides basic services such as caching, but defers to the underlying implementation
 * for CRUD operations. 
 * 
 * @author Derek Hulley
 * @since 3.2
 */
public abstract class AbstractEncodingDAOImpl implements EncodingDAO
{
    private static final Long CACHE_NULL_LONG = Long.MIN_VALUE;
    private SimpleCache encodingEntityCache;
    /**
     * 
     * @param encodingEntityCache           the cache of IDs to mimetypes
     */
    public void setEncodingEntityCache(SimpleCache encodingEntityCache)
    {
        this.encodingEntityCache = encodingEntityCache;
    }
    public Pair getEncoding(Long id)
    {
        // Check the cache
        String encoding = (String) encodingEntityCache.get(id);
        if (encoding != null)
        {
            return new Pair(id, encoding);
        }
        // Get it from the DB
        EncodingEntity mimetypeEntity = getEncodingEntity(id);
        if (mimetypeEntity == null)
        {
            throw new AlfrescoRuntimeException("The MimetypeEntity ID " + id + " doesn't exist.");
        }
        encoding = mimetypeEntity.getEncoding();
        // Cache it
        encodingEntityCache.put(encoding, id);
        encodingEntityCache.put(id, encoding);
        // Done
        return new Pair(id, encoding);
    }
    public Pair getEncoding(String encoding)
    {
        ParameterCheck.mandatory("encoding", encoding);
        
        // Check the cache
        Long id = (Long) encodingEntityCache.get(encoding);
        if (id != null)
        {
            if (id.equals(CACHE_NULL_LONG))
            {
                return null;
            }
            else
            {
                return new Pair(id, encoding);
            }
        }
        // It's not in the cache, so query
        EncodingEntity result = getEncodingEntity(encoding);
        if (result == null)
        {
            // Cache it
            encodingEntityCache.put(encoding, CACHE_NULL_LONG);
            // Done
            return null;
        }
        else
        {
            id = result.getId();
            // Cache it
            encodingEntityCache.put(id, encoding);
            encodingEntityCache.put(encoding, id);
            // Done
            return new Pair(id, encoding);
        }
    }
    public Pair getOrCreateEncoding(String encoding)
    {
        ParameterCheck.mandatory("encoding", encoding);
        
        Pair result = getEncoding(encoding);
        if (result == null)
        {
            EncodingEntity encodingEntity = createEncodingEntity(encoding);
            Long id = encodingEntity.getId();
            result = new Pair(id, encoding);
            // Cache it
            encodingEntityCache.put(id, encoding);
            encodingEntityCache.put(encoding, id);
        }
        return result;
    }
    
    /**
     * @param id            the ID of the encoding entity
     * @return              Return the entity or null if it doesn't exist
     */
    protected abstract EncodingEntity getEncodingEntity(Long id);
    protected abstract EncodingEntity getEncodingEntity(String encoding);
    protected abstract EncodingEntity createEncodingEntity(String encoding);
}