/*
 * #%L
 * Alfresco Repository
 * %%
 * Copyright (C) 2005 - 2016 Alfresco Software Limited
 * %%
 * This file is part of the Alfresco software. 
 * If the software was purchased under a paid Alfresco license, the terms of 
 * the paid license agreement will prevail.  Otherwise, the software is 
 * provided under the following open source license terms:
 * 
 * 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 .
 * #L%
 */
package org.alfresco.opencmis.search;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;
import org.alfresco.opencmis.dictionary.CMISDictionaryService;
import org.alfresco.opencmis.search.CMISQueryOptions.CMISQueryMode;
import org.alfresco.repo.search.impl.lucene.PagingLuceneResultSet;
import org.alfresco.repo.search.impl.querymodel.Query;
import org.alfresco.repo.search.impl.querymodel.QueryEngine;
import org.alfresco.repo.search.impl.querymodel.QueryEngineResults;
import org.alfresco.repo.search.impl.querymodel.QueryModelException;
import org.alfresco.repo.security.permissions.impl.acegi.FilteringResultSet;
import org.alfresco.service.cmr.dictionary.DictionaryService;
import org.alfresco.service.cmr.repository.NodeRef;
import org.alfresco.service.cmr.repository.NodeService;
import org.alfresco.service.cmr.repository.StoreRef;
import org.alfresco.service.cmr.search.LimitBy;
import org.alfresco.service.cmr.search.QueryConsistency;
import org.alfresco.service.cmr.search.ResultSet;
import org.alfresco.util.Pair;
import org.apache.chemistry.opencmis.commons.enums.BaseTypeId;
import org.apache.chemistry.opencmis.commons.enums.CapabilityJoin;
import org.apache.chemistry.opencmis.commons.enums.CapabilityQuery;
/**
 * @author andyh
 */
public class CMISQueryServiceImpl implements CMISQueryService
{
    private CMISDictionaryService cmisDictionaryService;
    private QueryEngine luceneQueryEngine;
    private QueryEngine dbQueryEngine;
    private NodeService nodeService;
    private DictionaryService alfrescoDictionaryService;
    public void setOpenCMISDictionaryService(CMISDictionaryService cmisDictionaryService)
    {
        this.cmisDictionaryService = cmisDictionaryService;
    }
    /**
     * @param queryEngine
     *            the luceneQueryEngine to set
     */
    public void setLuceneQueryEngine(QueryEngine queryEngine)
    {
        this.luceneQueryEngine = queryEngine;
    }
    /**
     * @param queryEngine
     *            the dbQueryEngine to set
     */
    public void setDbQueryEngine(QueryEngine queryEngine)
    {
        this.dbQueryEngine = queryEngine;
    }
    /**
     * @param nodeService
     *            the nodeService to set
     */
    public void setNodeService(NodeService nodeService)
    {
        this.nodeService = nodeService;
    }
    /**
     * @param alfrescoDictionaryService
     *            the Alfresco Dictionary Service to set
     */
    public void setAlfrescoDictionaryService(DictionaryService alfrescoDictionaryService)
    {
        this.alfrescoDictionaryService = alfrescoDictionaryService;
    }
    public CMISResultSet query(CMISQueryOptions options)
    {
        Pair resultPair = executeQuerySwitchingImpl(options);
        
        Query query = resultPair.getFirst();
        QueryEngineResults results = resultPair.getSecond();
        
        Map wrapped = new HashMap();
        Map, ResultSet> map = results.getResults();
        for (Set group : map.keySet())
        {
            ResultSet current = map.get(group);
            for (String selector : group)
            {
                wrapped.put(selector, filterNotExistingNodes(current));
            }
        }
        LimitBy limitBy = null;
        if ((null != results.getResults()) && !results.getResults().isEmpty()
                && (null != results.getResults().values()) && !results.getResults().values().isEmpty())
        {
            limitBy = results.getResults().values().iterator().next().getResultSetMetaData().getLimitedBy();
        }
        CMISResultSet cmis = new CMISResultSet(wrapped, options, limitBy, nodeService, query, cmisDictionaryService,
                alfrescoDictionaryService);
        return cmis;
    }
    private Pair executeQuerySwitchingImpl(CMISQueryOptions options)
    {
        switch (options.getQueryConsistency())
        {
            case TRANSACTIONAL_IF_POSSIBLE :
            {
                try
                {
                    return executeQueryUsingEngine(dbQueryEngine, options);
                }
                catch(QueryModelException qme)
                {
                    return executeQueryUsingEngine(luceneQueryEngine, options);
                }
            }
            case TRANSACTIONAL :
            {
                return executeQueryUsingEngine(dbQueryEngine, options);
            }
            case EVENTUAL :
            case DEFAULT :
            default :
            {
                return executeQueryUsingEngine(luceneQueryEngine, options);
            }
        }
    }
    
    private Pair executeQueryUsingEngine(QueryEngine queryEngine, CMISQueryOptions options)
    {
        CapabilityJoin joinSupport = getJoinSupport();
        if (options.getQueryMode() == CMISQueryOptions.CMISQueryMode.CMS_WITH_ALFRESCO_EXTENSIONS)
        {
            joinSupport = CapabilityJoin.INNERONLY;
        }
        // TODO: Refactor to avoid duplication of valid scopes here and in
        // CMISQueryParser
        BaseTypeId[] validScopes = (options.getQueryMode() == CMISQueryMode.CMS_STRICT) ? CmisFunctionEvaluationContext.STRICT_SCOPES
                : CmisFunctionEvaluationContext.ALFRESCO_SCOPES;
        CmisFunctionEvaluationContext functionContext = new CmisFunctionEvaluationContext();
        functionContext.setCmisDictionaryService(cmisDictionaryService);
        functionContext.setNodeService(nodeService);
        functionContext.setValidScopes(validScopes);
        CMISQueryParser parser = new CMISQueryParser(options, cmisDictionaryService, joinSupport);
        QueryConsistency queryConsistency = options.getQueryConsistency();
        if (queryConsistency == QueryConsistency.DEFAULT)
        {
        	options.setQueryConsistency(QueryConsistency.EVENTUAL);
        }
        
        Query query = parser.parse(queryEngine.getQueryModelFactory(), functionContext);
        QueryEngineResults queryEngineResults = queryEngine.executeQuery(query, options, functionContext);
        
        return new Pair(query, queryEngineResults);
    }
    /* MNT-8804 filter ResultSet for nodes with corrupted indexes */
    private ResultSet filterNotExistingNodes(ResultSet resultSet)
    {
        if (resultSet instanceof PagingLuceneResultSet)
        {
            ResultSet wrapped = ((PagingLuceneResultSet)resultSet).getWrapped();
            
            if (wrapped instanceof FilteringResultSet)
            {
                FilteringResultSet filteringResultSet = (FilteringResultSet)wrapped;
                
                for (int i = 0; i < filteringResultSet.length(); i++)
                {
                    NodeRef nodeRef = filteringResultSet.getNodeRef(i);
                    /* filter node if it does not exist */
                    if (!nodeService.exists(nodeRef))
                    {
                        filteringResultSet.setIncluded(i, false);
                    }
                }
            }
        }
        
        return resultSet;
    }
    public CMISResultSet query(String query, StoreRef storeRef)
    {
        CMISQueryOptions options = new CMISQueryOptions(query, storeRef);
        return query(options);
    }
    public boolean getPwcSearchable()
    {
        return true;
    }
    public boolean getAllVersionsSearchable()
    {
        return false;
    }
    public CapabilityQuery getQuerySupport()
    {
        return CapabilityQuery.BOTHCOMBINED;
    }
    public CapabilityJoin getJoinSupport()
    {
        return CapabilityJoin.NONE;
    }
}