/*
 * #%L
 * Alfresco Search Services
 * %%
 * Copyright (C) 2005 - 2020 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 <http://www.gnu.org/licenses/>.
 * #L%
 */

package org.alfresco.solr;

import static java.util.Collections.unmodifiableList;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.stream.Collectors;

import org.alfresco.error.AlfrescoRuntimeException;
import org.alfresco.model.ContentModel;
import org.alfresco.opencmis.dictionary.CMISAbstractDictionaryService;
import org.alfresco.opencmis.dictionary.CMISDictionaryService;
import org.alfresco.opencmis.dictionary.CMISStrictDictionaryService;
import org.alfresco.opencmis.dictionary.QNameFilter;
import org.alfresco.opencmis.search.CMISQueryOptions;
import org.alfresco.opencmis.search.CMISQueryOptions.CMISQueryMode;
import org.alfresco.opencmis.search.CMISQueryParser;
import org.alfresco.opencmis.search.CmisFunctionEvaluationContext;
import org.alfresco.repo.dictionary.CompiledModelsCache;
import org.alfresco.repo.dictionary.DictionaryComponent;
import org.alfresco.repo.dictionary.DictionaryDAOImpl;
import org.alfresco.repo.dictionary.Facetable;
import org.alfresco.repo.dictionary.IndexTokenisationMode;
import org.alfresco.repo.dictionary.M2Model;
import org.alfresco.repo.dictionary.M2ModelDiff;
import org.alfresco.repo.dictionary.NamespaceDAO;
import org.alfresco.repo.i18n.StaticMessageLookup;
import org.alfresco.repo.search.MLAnalysisMode;
import org.alfresco.repo.search.adaptor.lucene.QueryConstants;
import org.alfresco.repo.search.impl.QueryParserUtils;
import org.alfresco.repo.search.impl.parsers.AlfrescoFunctionEvaluationContext;
import org.alfresco.repo.search.impl.parsers.FTSParser;
import org.alfresco.repo.search.impl.parsers.FTSQueryParser;
import org.alfresco.repo.search.impl.querymodel.Constraint;
import org.alfresco.repo.search.impl.querymodel.QueryModelFactory;
import org.alfresco.repo.search.impl.querymodel.QueryOptions.Connective;
import org.alfresco.repo.search.impl.querymodel.impl.lucene.LuceneQueryBuilder;
import org.alfresco.repo.search.impl.querymodel.impl.lucene.LuceneQueryBuilderContext;
import org.alfresco.repo.search.impl.querymodel.impl.lucene.LuceneQueryModelFactory;
import org.alfresco.repo.tenant.SingleTServiceImpl;
import org.alfresco.repo.tenant.TenantService;
import org.alfresco.service.cmr.dictionary.DataTypeDefinition;
import org.alfresco.service.cmr.dictionary.DictionaryException;
import org.alfresco.service.cmr.dictionary.ModelDefinition;
import org.alfresco.service.cmr.dictionary.PropertyDefinition;
import org.alfresco.service.cmr.repository.StoreRef;
import org.alfresco.service.cmr.search.SearchParameters;
import org.alfresco.service.namespace.NamespaceException;
import org.alfresco.service.namespace.NamespaceService;
import org.alfresco.service.namespace.QName;
import org.alfresco.solr.AlfrescoClientDataModelServicesFactory.DictionaryKey;
import org.alfresco.solr.client.AlfrescoModel;
import org.alfresco.solr.query.Lucene4QueryBuilderContextSolrImpl;
import org.alfresco.solr.query.Solr4QueryParser;
import org.alfresco.solr.tracker.pool.DefaultTrackerPoolFactory;
import org.alfresco.solr.tracker.pool.TrackerPoolFactory;
import org.alfresco.util.NumericEncoder;
import org.alfresco.util.Pair;
import org.alfresco.util.cache.DefaultAsynchronouslyRefreshedCacheRegistry;
import org.apache.chemistry.opencmis.commons.enums.BaseTypeId;
import org.apache.chemistry.opencmis.commons.enums.CapabilityJoin;
import org.apache.chemistry.opencmis.commons.enums.CmisVersion;
import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.queryparser.classic.ParseException;
import org.apache.lucene.search.Query;
import org.apache.lucene.search.Sort;
import org.apache.lucene.util.Version;
import org.apache.solr.core.CoreDescriptorDecorator;
import org.apache.solr.core.SolrResourceLoader;
import org.apache.solr.request.SolrQueryRequest;
import org.apache.solr.search.SyntaxError;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.context.support.FileSystemXmlApplicationContext;

import static java.util.Optional.ofNullable;
import static org.alfresco.solr.SolrInformationServer.UNIT_OF_TIME_DAY_FIELD_SUFFIX;
import static org.alfresco.solr.SolrInformationServer.UNIT_OF_TIME_HOUR_FIELD_SUFFIX;
import static org.alfresco.solr.SolrInformationServer.UNIT_OF_TIME_MINUTE_FIELD_SUFFIX;
import static org.alfresco.solr.SolrInformationServer.UNIT_OF_TIME_MONTH_FIELD_SUFFIX;
import static org.alfresco.solr.SolrInformationServer.UNIT_OF_TIME_SECOND_FIELD_SUFFIX;
import static org.alfresco.solr.SolrInformationServer.UNIT_OF_TIME_QUARTER_FIELD_SUFFIX;
import static org.alfresco.solr.SolrInformationServer.UNIT_OF_TIME_YEAR_FIELD_SUFFIX;

/**
 * @author Andy
 *
 */
public class AlfrescoSolrDataModel implements QueryConstants
{
    public static class TenantDbId
    {
        public String tenant;
        public Long dbId;

        public Map<String, Object> optionalBag = new HashMap<>();

        public void setProperty(String name, Object value)
        {
            optionalBag.put(name, value);
        }
    }

    public enum FieldUse
    {
        FTS,          // Term/Phrase/Range/Fuzzy/Prefix/Proximity
        ID,           // ExactRange - Comparison, In, Upper, Lower, Wildcard
        EXACT,        // Exact
        FACET,        // Field, Range, Query
        MULTI_FACET,  // Text fields will require cross language support to avoid tokenisation for facets
        STATS,        // Stats
        SORT,         // Locale
        SUGGESTION,
        COMPLETION,
        HIGHLIGHT
    }

    public enum SpecializedFieldType
    {
        CONTENT_DOCID,
        CONTENT_SIZE,
        CONTENT_LOCALE,
        CONTENT_MIMETYPE,
        CONTENT_ENCODING,
        TRANSFORMATION_STATUS,
        TRANSFORMATION_TIME,
        TRANSFORMATION_EXCEPTION,
        UNIT_OF_TIME_SECOND,
        UNIT_OF_TIME_MINUTE,
        UNIT_OF_TIME_HOUR,
        UNIT_OF_TIME_DAY,
        UNIT_OF_TIME_MONTH,
        UNIT_OF_TIME_QUARTER,
        UNIT_OF_TIME_YEAR
    }

    public static final String CONTENT_S_LOCALE_PREFIX = "[email protected][email protected]";

    static final String PART_FIELDNAME_PREFIX = "[email protected]@";
    static final Set<String> DATE_PART_SUFFIXES =
            Set.of(UNIT_OF_TIME_YEAR_FIELD_SUFFIX,
                UNIT_OF_TIME_QUARTER_FIELD_SUFFIX,
                UNIT_OF_TIME_MONTH_FIELD_SUFFIX,
                UNIT_OF_TIME_DAY_FIELD_SUFFIX,
                UNIT_OF_TIME_HOUR_FIELD_SUFFIX,
                UNIT_OF_TIME_MINUTE_FIELD_SUFFIX,
                UNIT_OF_TIME_SECOND_FIELD_SUFFIX);
    /**
     * Infix used for denoting a primitive single valued field with no doc values enabled
     */
    private static final String SINGLE_VALUE_WITHOUT_DOC_VALUES_MARKER = "@[email protected]";

    /**
     * Infix used for denoting a primitive single valued field with no doc values enabled
     */
    private static final String SINGLE_VALUE_WITH_DOC_VALUES_MARKER = "@[email protected]";

    static final String SHARED_PROPERTIES = "shared.properties";

    protected final static Logger log = LoggerFactory.getLogger(AlfrescoSolrDataModel.class);

    private static final String CHANGE_SET = "CHANGE_SET";

    private static final String  TX = "TX";

    public static final String DEFAULT_TENANT = "_DEFAULT_";

    private final static AlfrescoSolrDataModel INSTANCE = new AlfrescoSolrDataModel();

    private final TenantService tenantService;

    private final NamespaceDAO namespaceDAO;

    private final DictionaryDAOImpl dictionaryDAO;

    private final Map<String,DictionaryComponent> dictionaryServices;

    private final Map<DictionaryKey,CMISAbstractDictionaryService> cmisDictionaryServices;

    private final Map<String, Set<String>> modelErrors = new HashMap<>();
    private final Set<QName> suggestableProperties = new HashSet<>();
    private final Set<QName> crossLocaleSearchDataTypes = new HashSet<>();
    private final Set<QName> crossLocaleSearchProperties = new HashSet<>();
    private final Set<QName> identifierProperties = new HashSet<>();
    private final ThreadPoolExecutor threadPool;

    public void close() {
        threadPool.shutdown();
    }

    public AlfrescoSolrDataModel()
    {
        tenantService = new SingleTServiceImpl();

        dictionaryDAO = new DictionaryDAOImpl();
        dictionaryDAO.setTenantService(tenantService);

        try
        {
            CompiledModelsCache compiledModelsCache = new CompiledModelsCache();
            compiledModelsCache.setDictionaryDAO(dictionaryDAO);
            compiledModelsCache.setTenantService(tenantService);
            compiledModelsCache.setRegistry(new DefaultAsynchronouslyRefreshedCacheRegistry());
            TrackerPoolFactory trackerPoolFactory = new DefaultTrackerPoolFactory(new Properties(), "_dictionary_", "_internal_");
            threadPool = trackerPoolFactory.create();
            compiledModelsCache.setThreadPoolExecutor(threadPool);

            dictionaryDAO.setDictionaryRegistryCache(compiledModelsCache);
            dictionaryDAO.setResourceClassLoader(getResourceClassLoader());
            dictionaryDAO.init();
        }
        catch (Exception e)
        {
            throw new AlfrescoRuntimeException("Failed to create dictionaryDAO ", e);
        }

        namespaceDAO = dictionaryDAO;

        QNameFilter qnameFilter = getQNameFilter();
        dictionaryServices = AlfrescoClientDataModelServicesFactory.constructDictionaryServices(qnameFilter, dictionaryDAO);
        DictionaryComponent dictionaryComponent = getDictionaryService(CMISStrictDictionaryService.DEFAULT);
        dictionaryComponent.setMessageLookup(new StaticMessageLookup());

        cmisDictionaryServices = AlfrescoClientDataModelServicesFactory.constructDictionaries(qnameFilter, namespaceDAO, dictionaryComponent, dictionaryDAO);

        Properties props = getCommonConfig();
        for (Object key : props.keySet())
        {
            String stringKey = (String) key;
            if (stringKey.startsWith("alfresco.suggestable.property."))
            {
                QName qName = QName.createQName(props.getProperty(stringKey));
                suggestableProperties.add(qName);
            }
            else if (stringKey.startsWith("alfresco.cross.locale.property."))
            {
                QName qName = QName.createQName(props.getProperty(stringKey));
                crossLocaleSearchProperties.add(qName);
            }
            else if (stringKey.startsWith("alfresco.cross.locale.datatype."))
            {
                QName qName = QName.createQName(props.getProperty(stringKey));
                crossLocaleSearchDataTypes.add(qName);
            }
            else if (stringKey.startsWith("alfresco.identifier.property."))
            {
                QName qName = QName.createQName(props.getProperty(stringKey));
                identifierProperties.add(qName);
            }
        }

        if(props.isEmpty())
        {
            suggestableProperties.add(ContentModel.PROP_NAME);
            suggestableProperties.add(ContentModel.PROP_TITLE);
            suggestableProperties.add(ContentModel.PROP_DESCRIPTION);
            suggestableProperties.add(ContentModel.PROP_CONTENT);

            crossLocaleSearchDataTypes.add(DataTypeDefinition.TEXT);
            crossLocaleSearchDataTypes.add(DataTypeDefinition.CONTENT);
            crossLocaleSearchDataTypes.add(DataTypeDefinition.MLTEXT);

            identifierProperties.add(ContentModel.PROP_CREATOR);
            identifierProperties.add(ContentModel.PROP_MODIFIER);
            identifierProperties.add(ContentModel.PROP_USERNAME);
            identifierProperties.add(ContentModel.PROP_AUTHORITY_NAME);
        }
    }

    public static String getTenantId(String tenant)
    {
        if((tenant == null) || tenant.equals(TenantService.DEFAULT_DOMAIN))
        {
            return DEFAULT_TENANT;
        }
        else
        {
            return tenant.replaceAll("!", "_-._");
        }
    }

    /**
     * Returns the ACL Document identifier in the following format &lt;TENANT>!&lt;ACLID>!ACL
     *
     * @param tenant the tenant code.
     * @param aclId the acl identifer
     * @return the ACL Document identifier (&lt;TENANT>!&lt;ACLID>!ACL)
     */
    public static String getAclDocumentId(String tenant, Long aclId)
    {
        return getTenantId(tenant) + "!" + NumericEncoder.encode(aclId) + "!ACL";
    }

    /**
     * Returns the ACL ChangeSet Document identifier in the following format TRACKER!CHANGESET!&lt;ACL_CHANGESET_ID>
     *
     * @param aclChangeSetId the ACL changeset identifier.
     * @return the ACL ChangeSet Document identifier (TRACKER!CHANGESET!&lt;ACL_CHANGESET_ID>)
     */
    public static String getAclChangeSetDocumentId(Long aclChangeSetId)
    {
        return "TRACKER" + "!" + CHANGE_SET + "!" + NumericEncoder.encode(aclChangeSetId);
    }


    /**
     * Extracts the transaction identifier from the input document identifier.
     * The "input document identifier" is the alphanumeric identifier used for Solr documents.
     *
     * @param documentId the document identifier.
     * @return the numeric ACL changeset identifier.
     */
    public static Long parseTransactionId(String documentId)
    {
        return parseIdFromDocumentId(documentId);
    }

    private static Long parseIdFromDocumentId(String documentId)
    {
        String[] split = documentId.split("!");
        if (split.length > 0)
            return NumericEncoder.decodeLong(split[split.length - 1]);
        else
            return null;
    }

    /**
     * Returns the Solr document identifier in the following format &lt;TENANT>!&lt;!&lt;DBID>
     *
     * @param tenant the tenant code.
     * @param dbid the DB identifier.
     * @return the Solr document identifier in the following format &lt;TENANT>!&lt;!&lt;DBID>
     */
    public static String getNodeDocumentId(String tenant, Long dbid)
    {
        return getTenantId(tenant) + "!" + NumericEncoder.encode(dbid);
    }

    /**
     * Destructures a document identifier in the three compounding parts (tenant and dbid).
     *
     * @param id the document identifier.
     * @return a {@link TenantDbId} instance containing the destructured identifier.
     */
    public static TenantDbId decodeNodeDocumentId(String id)
    {
        TenantDbId ids = new TenantDbId();
        String[] split = id.split("!");
        if (split.length > 0)
            ids.tenant = split[0];
        if (split.length > 1)
            ids.dbId = NumericEncoder.decodeLong(split[1]);
        return ids;
    }

    /**
     * Returns the Transaction Document identifier in the following format TRACKER!TX!&lt;TX_ID>
     *
     * @param txId the transaction identifier.
     * @return the Transaction Document identifier in the following format TRACKER!TX!&lt;TX_ID>
     */
    public static String getTransactionDocumentId(Long txId)
    {
        // TODO - check and encode for ":"
        return "TRACKER" +
                "!" +
                TX +
                "!" +
                NumericEncoder.encode(txId);
    }

    public String getAlfrescoPropertyFromSchemaField(String schemaField)
    {
        int index = schemaField.lastIndexOf("@{");
        if(index == -1)
        {
            return schemaField;
        }

        String alfrescoQueryField = schemaField.substring(index+1);
        QName qName = QName.createQName(alfrescoQueryField);
        alfrescoQueryField = qName.toPrefixString(namespaceDAO);

        PropertyDefinition propertyDefinition = getPropertyDefinition(qName);
        if((propertyDefinition == null))
        {
            return alfrescoQueryField;
        }
        if(!propertyDefinition.isIndexed() && !propertyDefinition.isStoredInIndex())
        {
            return alfrescoQueryField;
        }

        DataTypeDefinition dataTypeDefinition = propertyDefinition.getDataType();
        if(dataTypeDefinition.getName().equals(DataTypeDefinition.CONTENT))
        {
            if(schemaField.contains("[email protected]"))
            {
                return alfrescoQueryField + ".size";
            }
            else if(schemaField.contains("[email protected]"))
            {
                return alfrescoQueryField + ".locale";
            }
            else if(schemaField.contains("[email protected]"))
            {
                return alfrescoQueryField + ".mimetype";
            }
            else if(schemaField.contains("[email protected]"))
            {
                return alfrescoQueryField + ".encoding";
            }
            else if(schemaField.contains("[email protected]"))
            {
                return alfrescoQueryField + ".docid";
            }
            else if(schemaField.contains("[email protected]"))
            {
                return alfrescoQueryField + ".tr_ex";
            }
            else if(schemaField.contains("[email protected]"))
            {
                return alfrescoQueryField + ".tr_time";
            }
            else if(schemaField.contains("[email protected]"))
            {
                return alfrescoQueryField + ".tr_status";
            }
            else
            {
                return alfrescoQueryField;
            }
        }
        else
        {
            return alfrescoQueryField;
        }
    }

    public static AlfrescoSolrDataModel getInstance()
    {
        return INSTANCE;
    }

    public NamespaceDAO getNamespaceDAO()
    {
        return namespaceDAO;
    }

    /**
     * TODO: Fix to load type filter/exclusions from somewhere sensible
     * @return QNameFilter
     */
    private QNameFilter getQNameFilter()
    {
        QNameFilter qnameFilter = null;
        FileSystemXmlApplicationContext ctx = null;

        File resourceDirectory = getResourceDirectory();
        // If we ever need to filter out models in the future, then we must put a filter somewhere.
        // Currently, we do not need to filter any models, so this filter does not exist.
        File filterContext = new File(resourceDirectory, "opencmis-qnamefilter-context.xml");

        if(!filterContext.exists())
        {
            log.info("No type filter context found at " + filterContext.getAbsolutePath() + ", no type filtering");
            return qnameFilter;
        }

        try
        {
            ctx = new FileSystemXmlApplicationContext(new String[] { "file:" + filterContext.getAbsolutePath() }, false);
            ctx.setClassLoader(this.getClass().getClassLoader());
            ctx.refresh();
            qnameFilter = (QNameFilter)ctx.getBean("cmisTypeExclusions");
            if(qnameFilter == null)
            {
                log.warn("Unable to find type filter at " + filterContext.getAbsolutePath() + ", no type filtering");
            }
        }
        catch(BeansException e)
        {
            log.warn("Unable to parse type filter at " + filterContext.getAbsolutePath() + ", no type filtering");
        }
        finally
        {
            if(ctx != null && ctx.getBeanFactory() != null && ctx.isActive())
            {
                ctx.close();
            }
        }

        return qnameFilter;
    }

    public static Properties getCommonConfig()
    {
        File resourceDirectory = getResourceDirectory();
        // If we ever need to filter out models in the future, then we must put a filter somewhere.
        // Currently, we do not need to filter any models, so this filter does not exist.
        File propertiesFile = new File(resourceDirectory, SHARED_PROPERTIES);

        Properties props = new Properties();
        if(!propertiesFile.exists())
        {
            log.info("No shared properties found at  " + propertiesFile.getAbsolutePath());
            return props;
        }

        try(InputStream is = new FileInputStream(propertiesFile))
        {
            props.load(is);
        }
        catch (IOException e)
        {
            log.info("Failed to read shared properties at  " + propertiesFile.getAbsolutePath());
        }

        return props;
    }

    /**
     * @return ClassLoader
     */
    public ClassLoader getResourceClassLoader()
    {
        File f = getResourceDirectory();
        if (f.canRead() && f.isDirectory())
        {
            URL[] urls = new URL[1];

            try
            {
                URL url = f.toURI().normalize().toURL();
                urls[0] = url;
            }
            catch (MalformedURLException e)
            {
                throw new AlfrescoRuntimeException("Failed to add resources to classpath ", e);
            }

            return URLClassLoader.newInstance(urls, this.getClass().getClassLoader());
        }
        else
        {
            return this.getClass().getClassLoader();
        }
    }

    public static File getResourceDirectory()
    {
        return new File(SolrResourceLoader.locateSolrHome().toFile(), "conf");
    }

    /**
     * Gets a DictionaryService, if an Alternative dictionary is specified it tries to get that.
     * It will attempt to get the DEFAULT dictionary service if null is specified or it can't find
     * a dictionary with the name of "alternativeDictionary"
     * @param alternativeDictionary - can be null;
     * @return DictionaryService
     */
    public DictionaryComponent getDictionaryService(String alternativeDictionary)
    {
        DictionaryComponent dictionaryComponent = null;

        if (alternativeDictionary != null && !alternativeDictionary.trim().isEmpty())
        {
            dictionaryComponent = dictionaryServices.get(alternativeDictionary);
        }

        if (dictionaryComponent == null)
        {
            dictionaryComponent = dictionaryServices.get(CMISStrictDictionaryService.DEFAULT);
        }
        return dictionaryComponent;
    }

    public IndexedField getIndexedFieldForSpecializedPropertyMetadata(QName propertyQName, SpecializedFieldType type)
    {
        IndexedField indexedField = new IndexedField();
        PropertyDefinition propertyDefinition = getPropertyDefinition(propertyQName);
        if((propertyDefinition == null))
        {
            return indexedField;
        }
        if(!propertyDefinition.isIndexed() && !propertyDefinition.isStoredInIndex())
        {
            return indexedField;
        }

        DataTypeDefinition dataTypeDefinition = propertyDefinition.getDataType();
        if(dataTypeDefinition.getName().equals(DataTypeDefinition.CONTENT))
        {
            StringBuilder builder = new StringBuilder();
            builder.append(dataTypeDefinition.getName().getLocalName());
            builder.append('@');
            // TODO wher we support multi value propertis correctly .... builder.append(propertyDefinition.isMultiValued() ? "m" : "s");
            builder.append('s');
            builder.append("_");
            builder.append('_');
            switch (type)
            {
                case CONTENT_DOCID:
                    builder.append("docid");
                    break;
                case CONTENT_ENCODING:
                    builder.append("encoding");
                    break;
                case CONTENT_LOCALE:
                    builder.append("locale");
                    break;
                case CONTENT_MIMETYPE:
                    builder.append("mimetype");
                    break;
                case CONTENT_SIZE:
                    builder.append("size");
                    break;
                case TRANSFORMATION_EXCEPTION:
                    builder.append("tr_ex");
                    break;
                case TRANSFORMATION_STATUS:
                    builder.append("tr_status");
                    break;
                case TRANSFORMATION_TIME:
                    builder.append("tr_time");
                    break;
                default:
                    break;
            }

            builder.append('@');
            builder.append(propertyQName);
            indexedField.addField(builder.toString(), false, false);
        }
        else if (isDateOrDatetime(dataTypeDefinition))
        {
            String dateDerivedSuffix = getDateDerivedSuffix(type);
            if (dateDerivedSuffix != null)
            {
                indexedField.addField(getDateDerivedField(propertyQName, dateDerivedSuffix), false, true);
            }
        }
        return indexedField;

    }


    public IndexedField getQueryableFields(QName propertyQName, SpecializedFieldType type, FieldUse fieldUse)
    {
        if(type != null)
        {
            return getIndexedFieldForSpecializedPropertyMetadata(propertyQName, type);
        }

        IndexedField indexedField = new IndexedField();
        PropertyDefinition propertyDefinition = getPropertyDefinition(propertyQName);
        if((propertyDefinition == null))
        {
            indexedField.addField("_dummy_", false, false);
            return indexedField;
        }
        if(!propertyDefinition.isIndexed() && !propertyDefinition.isStoredInIndex())
        {
            indexedField.addField("_dummy_", false, false);
            return indexedField;
        }

        if(isTextField(propertyDefinition))
        {
            switch(fieldUse)
            {
                case COMPLETION:
                    addCompletionFields(propertyDefinition, indexedField);
                    break;
                case FACET:
                    addFacetSearchFields(propertyDefinition, indexedField);
                    break;
                case FTS:
                    addFullTextSearchFields(propertyDefinition, indexedField);
                    break;
                case ID:
                    addIdentifierSearchFields(propertyDefinition, indexedField);
                    break;
                case EXACT:
                    addExactSearchFields(propertyDefinition, indexedField);
                    break;
                case MULTI_FACET:
                    addMultiSearchFields(propertyDefinition, indexedField);
                    break;
                case SORT:
                    addSortSearchFields(propertyDefinition, indexedField);
                    break;
                case STATS:
                    addStatsSearchFields(propertyDefinition, indexedField);
                    break;
                case SUGGESTION:
                    if(isSuggestable(propertyQName))
                    {
                        indexedField.addField("suggest", false, false);
                    }
                    addCompletionFields(propertyDefinition, indexedField);
                    break;
                case HIGHLIGHT:
                    addHighlightSearchFields(propertyDefinition, indexedField);
                    break;
            }
        }
        else
        {
            indexedField.addField(getFieldForNonText(propertyDefinition), false, false);
        }
        return indexedField;
    }

    /*
     * Adds best completion fields in order of preference
     */
    private void addCompletionFields(PropertyDefinition propertyDefinition, IndexedField indexedField)
    {
        if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
        {
            if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
            {
                indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
            }
            else
            {
                indexedField.addField(getFieldForText(true, true, false, propertyDefinition), false, false);
            }
        }
        else if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.TRUE))
        {
            if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
            {
                indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
            }
            else
            {
                indexedField.addField(getFieldForText(true, true, false, propertyDefinition), false, false);
            }
        }
        else if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE))
        {
            indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
        }
    }

    /*
     * Adds best fts fields in order of preference
     */

    private void addFullTextSearchFields( PropertyDefinition propertyDefinition , IndexedField indexedField)
    {
        if (((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.TRUE)
                || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
                && !isIdentifierTextProperty(propertyDefinition.getName()))
        {
            indexedField.addField(getFieldForText(true, true, false, propertyDefinition), true, false);
            if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
            {
                indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
            }
        }
        else
        {
            indexedField.addField(getFieldForText(true, false, false, propertyDefinition), true, false);
            indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
        }
    }

    private void addHighlightSearchFields( PropertyDefinition propertyDefinition , IndexedField indexedField)
    {
        QName propertyName = propertyDefinition.getName();
        QName propertyDataTypeQName = propertyDefinition.getDataType().getName();
        String fieldName;

        if(propertyDataTypeQName.equals(DataTypeDefinition.MLTEXT))
        {
            fieldName = getStoredMLTextField(propertyName);
        }
        else if(propertyDataTypeQName.equals(DataTypeDefinition.CONTENT))
        {
            fieldName = getStoredContentField(propertyName);
        }
        else
        {
            fieldName = getStoredTextField(propertyName);
        }

        indexedField.addField(fieldName, false, false);
    }

    /*
     * If the untokenised version is available, go for it.
     * If not it takes the tokenised versions available.
     */
    private void addIdentifierSearchFields( PropertyDefinition propertyDefinition , IndexedField indexedField)
    {
        if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
        {

            indexedField.addField(getFieldForText(true, false, false, propertyDefinition), true, false);
            indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
        }
        else
        {
            indexedField.addField(getFieldForText(true, true, false, propertyDefinition), true, false);
            if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
            {
                indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
            }
        }
    }

    /*
     * If only the untokenised version is available, go for it.
     * If not it takes the tokenised cross locale version.
     */
    private void addExactSearchFields(PropertyDefinition propertyDefinition, IndexedField indexedField)
    {
        if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                || !(propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
        {

            indexedField.addField(getFieldForText(true, false, false, propertyDefinition), true, false);
            indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
        }
        else
        {
            if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
            {
                indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
            } else{
                throw new UnsupportedOperationException("Exact Term search is not supported unless you configure the field <"+propertyDefinition.getName()+"> for cross locale search");
            }
        }
    }

    /*
     * Adds best identifier fields in order of preference
     */
    private void addFacetSearchFields(PropertyDefinition propertyDefinition, IndexedField indexedField)
    {
        if(propertyDefinition.getDataType().getName().equals(DataTypeDefinition.TEXT))
        {
            if (!isIdentifierTextProperty(propertyDefinition.getName()))
            {
                if(propertyDefinition.getFacetable() == Facetable.TRUE)
                {
                    indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
                }
            }
        }


        if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH)
                || isIdentifierTextProperty(propertyDefinition.getName()))
        {

            indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
        }
        else
        {
            if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
            {
                indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
            }
            else
            {
                indexedField.addField(getFieldForText(true, true, false, propertyDefinition), false, false);
            }
        }
    }

    private void addMultiSearchFields( PropertyDefinition propertyDefinition , IndexedField indexedField)
    {
        if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH)
                || isIdentifierTextProperty(propertyDefinition.getName()))
        {

            indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
        }
        else
        {
            if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
            {
                indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
            }
            else
            {
                indexedField.addField(getFieldForText(true, true, false, propertyDefinition), false, false);
            }
        }
    }

    private void addStatsSearchFields( PropertyDefinition propertyDefinition , IndexedField indexedField)
    {
        addFacetSearchFields(propertyDefinition, indexedField);
    }

    private void addSortSearchFields( PropertyDefinition propertyDefinition , IndexedField indexedField)
    {
        // Can only order on single valued fields
        DataTypeDefinition dataTypeDefinition = propertyDefinition.getDataType();
        if(dataTypeDefinition.getName().equals(DataTypeDefinition.TEXT))
        {
            if(!propertyDefinition.isMultiValued())
            {
                if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                        || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
                {
                    indexedField.addField(getFieldForText(false, false, true, propertyDefinition), false, true);
                }
                else if (isIdentifierTextProperty(propertyDefinition.getName()))
                {
                    indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
                }
                else
                {
                    if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
                    {
                        indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
                    }
                    else
                    {
                        indexedField.addField(getFieldForText(true, true, false, propertyDefinition), false, false);
                    }
                }
            }
        }

        if(dataTypeDefinition.getName().equals(DataTypeDefinition.MLTEXT))
        {
            if(!propertyDefinition.isMultiValued())
            {
                if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                        || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
                {
                    indexedField.addField(getFieldForText(false, false, true, propertyDefinition), false, true);
                }
                else
                {
                    if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
                    {
                        indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
                    }
                    else
                    {
                        indexedField.addField(getFieldForText(true, true, false, propertyDefinition), false, false);
                    }
                }
            }
        }
    }


    public String getStoredTextField(QName propertyQName)
    {
        return getStoredTextField(propertyQName, null);
    }

    public String getStoredTextField(QName propertyQName, String suffix)
    {
        PropertyDefinition propertyDefinition = getPropertyDefinition(propertyQName);

        StringBuilder sb = new StringBuilder();
        sb.append("[email protected]" + (propertyDefinition.isMultiValued()? "m" : "s") + "_stored_");

        sb.append((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.TRUE ||
            propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH)? "t" : "_");

        sb.append((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE ||
            propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH ||
            isIdentifierTextProperty(propertyDefinition.getName()))? "s" : "_");

        sb.append((crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) ||
            crossLocaleSearchProperties.contains(propertyDefinition.getName())) ? "c" : "_");

        sb.append((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE ||
            propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH ||
            isIdentifierTextProperty(propertyDefinition.getName())) && !propertyDefinition.isMultiValued()? "s" : "_");

        sb.append(isSuggestable(propertyQName)? "s": "_");

        sb.append("@");
        sb.append(propertyDefinition.getName().toString());

        if (suffix != null)
        {
            sb.append(suffix);
        }

        return sb.toString();

    }

    public String getStoredMLTextField(QName propertyQName)
    {
        return getStoredMLTextField(propertyQName, null);
    }

    public String getStoredMLTextField(QName propertyQName, String suffix)
    {
        PropertyDefinition propertyDefinition = getPropertyDefinition(propertyQName);

        StringBuilder sb = new StringBuilder();
        sb.append("[email protected]_stored_");

        sb.append((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.TRUE ||
            propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH)? "t" : "_");

        sb.append((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE ||
            propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH ||
            isIdentifierTextProperty(propertyDefinition.getName()))? "s" : "_");

        sb.append((crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) ||
            crossLocaleSearchProperties.contains(propertyDefinition.getName())) ? "c" : "_");

        sb.append("_");

        sb.append(isSuggestable(propertyQName)? "s": "_");

        sb.append("@");
        sb.append(propertyDefinition.getName().toString());

        if (suffix != null)
        {
            sb.append(suffix);
        }

        return sb.toString();

    }

    public String getStoredContentField(QName propertyQName)
    {
        return getStoredContentField(propertyQName, null);
    }

    public String getStoredContentField(QName propertyQName, String suffix)
    {
        PropertyDefinition propertyDefinition = getPropertyDefinition(propertyQName);

        StringBuilder sb = new StringBuilder();
        sb.append("[email protected]_stored_");

        sb.append((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.TRUE ||
            propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH)? "t" : "_");

        sb.append((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE ||
            propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH ||
            isIdentifierTextProperty(propertyDefinition.getName()))? "s" : "_");

        sb.append((crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) ||
            crossLocaleSearchProperties.contains(propertyDefinition.getName())) ? "c" : "_");


        sb.append("_");
        sb.append(isSuggestable(propertyQName)? "s": "_");

        sb.append("@");
        sb.append(propertyDefinition.getName().toString());

        if (suffix != null)
        {
            sb.append(suffix);
        }

        return sb.toString();

    }

    public String getDateDerivedField(QName propertyQName, String suffix)
    {
        PropertyDefinition propertyDefinition = getPropertyDefinition(propertyQName);
        return "[email protected]@" + propertyDefinition.getName().toString() + suffix;
    }

    public String getDateDerivedSuffix(SpecializedFieldType type)
    {
        switch (type)
        {
            case UNIT_OF_TIME_SECOND:
                return UNIT_OF_TIME_SECOND_FIELD_SUFFIX;
            case UNIT_OF_TIME_MINUTE:
                return UNIT_OF_TIME_MINUTE_FIELD_SUFFIX;
            case UNIT_OF_TIME_HOUR:
                return UNIT_OF_TIME_HOUR_FIELD_SUFFIX;
            case UNIT_OF_TIME_DAY:
                return UNIT_OF_TIME_DAY_FIELD_SUFFIX;
            case UNIT_OF_TIME_MONTH:
                return UNIT_OF_TIME_MONTH_FIELD_SUFFIX;
            case UNIT_OF_TIME_QUARTER:
                return UNIT_OF_TIME_QUARTER_FIELD_SUFFIX;
            case UNIT_OF_TIME_YEAR:
                return UNIT_OF_TIME_YEAR_FIELD_SUFFIX;
            default:
                return null;
        }
    }


    /**
     * Get all the field names into which we must copy the source data
     *
     * @param propertyQName QName
     * @return IndexedField
     */
    public IndexedField getIndexedFieldNamesForProperty(QName propertyQName)
    {
        // TODO: Cache and throw on model refresh

        IndexedField indexedField = new IndexedField();
        PropertyDefinition propertyDefinition = getPropertyDefinition(propertyQName);
        if((propertyDefinition == null))
        {
            return indexedField;
        }
        if(!propertyDefinition.isIndexed() && !propertyDefinition.isStoredInIndex())
        {
            return indexedField;
        }

        DataTypeDefinition dataTypeDefinition = propertyDefinition.getDataType();
        if(isTextField(propertyDefinition))
        {
            if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.TRUE)
                    || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
            {
                indexedField.addField(getFieldForText(true, true, false, propertyDefinition), true, false);
                if(crossLocaleSearchDataTypes.contains(propertyDefinition.getDataType().getName()) || crossLocaleSearchProperties.contains(propertyDefinition.getName()))
                {
                    indexedField.addField(getFieldForText(false, true, false, propertyDefinition), false, false);
                }
            }

            if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                    || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH
                    || isIdentifierTextProperty(propertyDefinition.getName())))
            {
                indexedField.addField(getFieldForText(true, false, false, propertyDefinition), true, false);
                indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
            }

            if(dataTypeDefinition.getName().equals(DataTypeDefinition.TEXT))
            {
                if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                        || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
                {
                    if(!propertyDefinition.isMultiValued())
                    {
                        indexedField.addField(getFieldForText(false, false, true, propertyDefinition), false, true);
                    }
                }
                else if (!isIdentifierTextProperty(propertyDefinition.getName()))
                {
                    if(propertyDefinition.getFacetable() == Facetable.TRUE)
                    {
                        indexedField.addField(getFieldForText(false, false, false, propertyDefinition), false, false);
                    }
                }
            }

            if(dataTypeDefinition.getName().equals(DataTypeDefinition.MLTEXT))
            {
                if ((propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.FALSE)
                        || (propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.BOTH))
                {
                    if(!propertyDefinition.isMultiValued())
                    {
                        indexedField.addField(getFieldForText(true, false, true, propertyDefinition), true, true);
                    }
                }
            }

            if(isSuggestable(propertyQName))
            {
                indexedField.addField("[email protected]" + propertyDefinition.getName().toString(), false, false);
            }
        }
        else
        {
            indexedField.addField(getFieldForNonText(propertyDefinition), false, false);
        }

        return indexedField;
    }

    private boolean isIdentifierTextProperty(QName propertyQName)
    {
        return identifierProperties.contains(propertyQName);
    }

    public boolean isTextField(PropertyDefinition propertyDefinition)
    {
        return ofNullable(propertyDefinition)
                .map(PropertyDefinition::getDataType)
                .map(DataTypeDefinition::getName)
                .map(name ->
                        name.equals(DataTypeDefinition.MLTEXT)
                        ||
                        name.equals(DataTypeDefinition.CONTENT)
                        ||
                        name.equals(DataTypeDefinition.TEXT))
                .orElse(false);
    }

    private boolean isSuggestable(QName propertyQName)
    {
        if(propertyQName == null)
        {
            return false;
        }
        return suggestableProperties.contains(propertyQName);
    }

    private boolean hasDocValues(PropertyDefinition propertyDefinition)
    {

        if(isTextField(propertyDefinition))
        {
            // We only call this if text is untokenised and localised
            return propertyDefinition.getFacetable() != Facetable.FALSE;
        }
        else
        {
            if(propertyDefinition.getFacetable() == Facetable.FALSE)
            {
                return false;
            }
            else if(propertyDefinition.getFacetable() == Facetable.TRUE)
            {
                return true;
            }
            else
            {
                if(propertyDefinition.getIndexTokenisationMode() == IndexTokenisationMode.TRUE)
                {
                    return false;
                }
                else
                {
                    return isPrimitive(propertyDefinition.getDataType());
                }
            }
        }
    }

    public String getFieldForNonText(PropertyDefinition propertyDefinition)
    {
        StringBuilder builder = new StringBuilder();
        QName qName = propertyDefinition.getDataType().getName();
        builder.append(qName.getLocalName());
        builder.append("@");
        builder.append(propertyDefinition.isMultiValued() ? "m" : "s");
        builder.append(hasDocValues(propertyDefinition) ? "d" : "_");
        builder.append("@");
        builder.append(propertyDefinition.getName().toString());
        return builder.toString();
    }

    private String getFieldForText(boolean localised, boolean tokenised, boolean sort, PropertyDefinition propertyDefinition)
    {
        StringBuilder builder = new StringBuilder();
        QName qName = propertyDefinition.getDataType().getName();
        builder.append(qName.getLocalName());
        builder.append("@");
        QName propertyDataTypeQName = propertyDefinition.getDataType().getName();
        if(propertyDataTypeQName.equals(DataTypeDefinition.MLTEXT))
        {
            builder.append('m');
        }
        else  if(propertyDataTypeQName.equals(DataTypeDefinition.CONTENT))
        {
            builder.append('s');
        }
        else
        {
            builder.append(propertyDefinition.isMultiValued() ? "m" : "s");
        }
        if(sort || localised || tokenised || propertyDataTypeQName.equals(DataTypeDefinition.CONTENT) ||  propertyDataTypeQName.equals(DataTypeDefinition.MLTEXT))
        {
            builder.append('_');
        }
        else
        {
            builder.append(hasDocValues(propertyDefinition) ? "d" : "_");
        }
        builder.append('_');
        if(!sort)
        {
            builder.append(localised ? "l" : "_");
            builder.append(tokenised ? "t" : "_");
        }
        else
        {
            builder.append("sort");
        }
        builder.append("@");
        builder.append(propertyDefinition.getName().toString());
        return builder.toString();
    }

    public PropertyDefinition getPropertyDefinition(QName propertyQName)
    {
        return getDictionaryService(CMISStrictDictionaryService.DEFAULT).getProperty(propertyQName);
    }

    public boolean putModel(M2Model model)
    {
        Set<String> errors = validateModel(model);
        if(errors.isEmpty())
        {
            modelErrors.remove(model.getName());
            dictionaryDAO.putModelIgnoringConstraints(model);
            return true;
        }
        else
        {
            if(!modelErrors.containsKey(model.getName()))
            {
                modelErrors.put(model.getName(), errors);
                log.warn(errors.iterator().next());
            }
            return false;
        }
    }

    public void removeModel(QName modelQName)
    {
        modelErrors.remove(getM2Model(modelQName).getName());
        dictionaryDAO.removeModel(modelQName);
    }

    M2Model getM2Model(QName modelQName)
    {
        return dictionaryDAO.getCompiledModel(modelQName).getM2Model();
    }

    public void afterInitModels()
    {
        for (CMISAbstractDictionaryService cds : cmisDictionaryServices.values())
        {
            cds.afterDictionaryInit();
        }
    }

    public org.alfresco.repo.search.impl.querymodel.Query parseCMISQueryToAlfrescoAbstractQuery(CMISQueryMode mode, SearchParameters searchParameters,
                                                                                                SolrQueryRequest req, String alternativeDictionary, CmisVersion cmisVersion)
    {
        // convert search parameters to cmis query options
        // TODO: how to handle store ref
        CMISQueryOptions options = new CMISQueryOptions(searchParameters.getQuery(), StoreRef.STORE_REF_WORKSPACE_SPACESSTORE);
        options.setQueryMode(CMISQueryMode.CMS_WITH_ALFRESCO_EXTENSIONS);
        options.setDefaultFieldName(searchParameters.getDefaultFieldName());
        // TODO: options.setDefaultFTSConnective()
        // TODO: options.setDefaultFTSFieldConnective()
        options.setIncludeInTransactionData(!searchParameters.excludeDataInTheCurrentTransaction());
        options.setLocales(searchParameters.getLocales());
        options.setMlAnalaysisMode(searchParameters.getMlAnalaysisMode());
        options.setQueryParameterDefinitions(searchParameters.getQueryParameterDefinitions());
        for(String name : searchParameters.getQueryTemplates().keySet())
        {
            String template = searchParameters.getQueryTemplates().get(name);
            options.addQueryTemplate(name, template);
        }

        // parse cmis syntax
        CapabilityJoin joinSupport = (mode == CMISQueryMode.CMS_STRICT) ? CapabilityJoin.NONE : CapabilityJoin.INNERANDOUTER;
        CmisFunctionEvaluationContext functionContext = getCMISFunctionEvaluationContext(mode, cmisVersion, alternativeDictionary);

        CMISDictionaryService cmisDictionary = getCMISDictionary(alternativeDictionary, cmisVersion);

        CMISQueryParser parser = new CMISQueryParser(options, cmisDictionary, joinSupport);
        org.alfresco.repo.search.impl.querymodel.Query queryModelQuery = parser.parse(new LuceneQueryModelFactory<Query, Sort, SyntaxError>(), functionContext);

        if (queryModelQuery.getSource() != null)
        {
            List<Set<String>> selectorGroups = queryModelQuery.getSource().getSelectorGroups(functionContext);
            if (selectorGroups.size() == 0)
            {
                throw new UnsupportedOperationException("No selectors");
            }
        }
        return queryModelQuery;
    }

    public CmisFunctionEvaluationContext getCMISFunctionEvaluationContext(CMISQueryMode mode, CmisVersion cmisVersion, String alternativeDictionary)
    {
        BaseTypeId[] validScopes = (mode == CMISQueryMode.CMS_STRICT) ? CmisFunctionEvaluationContext.STRICT_SCOPES : CmisFunctionEvaluationContext.ALFRESCO_SCOPES;
        CmisFunctionEvaluationContext functionContext = new CmisFunctionEvaluationContext();
        functionContext.setCmisDictionaryService(getCMISDictionary(alternativeDictionary, cmisVersion));
        functionContext.setValidScopes(validScopes);
        return functionContext;
    }

    /**
     * Gets the CMISDictionaryService, if an Alternative dictionary is specified it tries to get that.
     * It will attempt to get the DEFAULT dictionary service if null is specified or it can't find
     * a dictionary with the name of "alternativeDictionary"
     *
     * @param alternativeDictionary - can be null;
     * @return CMISDictionaryService
     */
    public CMISDictionaryService getCMISDictionary(String alternativeDictionary, CmisVersion cmisVersion)
    {
        CMISDictionaryService cmisDictionary = null;

        if (alternativeDictionary != null && !alternativeDictionary.trim().isEmpty())
        {
            DictionaryKey key = new DictionaryKey(cmisVersion, alternativeDictionary);
            cmisDictionary = cmisDictionaryServices.get(key);
        }

        if (cmisDictionary == null)
        {
            DictionaryKey key = new DictionaryKey(cmisVersion, CMISStrictDictionaryService.DEFAULT);
            cmisDictionary = cmisDictionaryServices.get(key);
        }
        return cmisDictionary;
    }

    /**
     * Returns the Alfresco models associated with the current dictionary.
     *
     * @return the Alfresco models associated with the current dictionary.
     */
    public List<AlfrescoModel> getAlfrescoModels()
    {
        return dictionaryDAO.getModels().stream()
                .map(qname -> {
                    M2Model m2Model = dictionaryDAO.getCompiledModel(qname).getM2Model();
                    return new AlfrescoModel(
                            m2Model,
                            getDictionaryService(
                                    CMISStrictDictionaryService.DEFAULT).getModel(qname).getChecksum(ModelDefinition.XMLBindingType.DEFAULT));})
                .collect(Collectors.toList());
    }

    Map<String, Set<String>> getModelErrors()
    {
        return modelErrors;
    }

    public static class IndexedField
    {
        private final List<FieldInstance> fields = new LinkedList<>();

        public List<FieldInstance> getFields()
        {
            return unmodifiableList(fields);
        }

        public void addField(String prefix, boolean localised, boolean sort)
        {
            fields.add(new FieldInstance(prefix, localised, sort));
        }
    }

    public static class FieldInstance
    {
        final String field;
        final boolean localised;
        final boolean sort;

        public FieldInstance(String field, boolean localised, boolean sort)
        {
            this.field = field;
            this.localised = localised;
            this.sort = sort;
        }

        public String getField()
        {
            return field;
        }

        public boolean isLocalised()
        {
            return localised;
        }

        public boolean isSort()
        {
            return sort;
        }
    }

    public Query getCMISQuery(CMISQueryMode mode, Pair<SearchParameters, Boolean> searchParametersAndFilter, SolrQueryRequest req, org.alfresco.repo.search.impl.querymodel.Query queryModelQuery, CmisVersion cmisVersion, String alternativeDictionary) throws ParseException
    {
        SearchParameters searchParameters = searchParametersAndFilter.getFirst();
        Boolean isFilter = searchParametersAndFilter.getSecond();

        CmisFunctionEvaluationContext functionContext = getCMISFunctionEvaluationContext(mode, cmisVersion, alternativeDictionary);

        Set<String> selectorGroup = queryModelQuery.getSource().getSelectorGroups(functionContext).get(0);

        LuceneQueryBuilderContext<Query, Sort, ParseException> luceneContext = getLuceneQueryBuilderContext(searchParameters, req, alternativeDictionary, FTSQueryParser.RerankPhase.SINGLE_PASS);
        @SuppressWarnings("unchecked")
        LuceneQueryBuilder<Query, Sort, ParseException> builder = (LuceneQueryBuilder<Query, Sort, ParseException>) queryModelQuery;
        org.apache.lucene.search.Query luceneQuery = builder.buildQuery(selectorGroup, luceneContext, functionContext);

        return new ContextAwareQuery(luceneQuery, Boolean.TRUE.equals(isFilter) ? null : searchParameters);
    }

    public LuceneQueryBuilderContext<Query, Sort, ParseException> getLuceneQueryBuilderContext(SearchParameters searchParameters, SolrQueryRequest req, String alternativeDictionary, FTSQueryParser.RerankPhase rerankPhase)
    {
        return new Lucene4QueryBuilderContextSolrImpl(
                getDictionaryService(alternativeDictionary),
                namespaceDAO,
                tenantService,
                searchParameters,
                MLAnalysisMode.EXACT_LANGUAGE,
                req,
                this,
                rerankPhase);
    }

    public Solr4QueryParser getLuceneQueryParser(SearchParameters searchParameters, SolrQueryRequest req, FTSQueryParser.RerankPhase rerankPhase)
    {
        Analyzer analyzer =  req.getSchema().getQueryAnalyzer();
        Solr4QueryParser parser = new Solr4QueryParser(req, Version.LATEST, searchParameters.getDefaultFieldName(), analyzer, rerankPhase);
        parser.setNamespacePrefixResolver(namespaceDAO);
        parser.setDictionaryService(getDictionaryService(CMISStrictDictionaryService.DEFAULT));
        parser.setTenantService(tenantService);
        parser.setSearchParameters(searchParameters);
        parser.setAllowLeadingWildcard(true);

        Properties props = new CoreDescriptorDecorator(req.getCore().getCoreDescriptor()).getProperties();
        int topTermSpanRewriteLimit = Integer.parseInt(props.getProperty("alfresco.topTermSpanRewriteLimit", "1000"));
        parser.setTopTermSpanRewriteLimit(topTermSpanRewriteLimit);

        return parser;
    }

    public Query getFTSQuery(Pair<SearchParameters, Boolean> searchParametersAndFilter, SolrQueryRequest req, FTSQueryParser.RerankPhase rerankPhase) throws ParseException
    {

        SearchParameters searchParameters = searchParametersAndFilter.getFirst();
        Boolean isFilter = searchParametersAndFilter.getSecond();

        QueryModelFactory factory = new LuceneQueryModelFactory<Query, Sort, SyntaxError>();
        AlfrescoFunctionEvaluationContext functionContext = new AlfrescoSolr4FunctionEvaluationContext(namespaceDAO, getDictionaryService(CMISStrictDictionaryService.DEFAULT), NamespaceService.CONTENT_MODEL_1_0_URI, req.getSchema());

        FTSParser.Mode mode;

        if (searchParameters.getDefaultFTSOperator() == org.alfresco.service.cmr.search.SearchParameters.Operator.AND)
        {
            mode = FTSParser.Mode.DEFAULT_CONJUNCTION;
        }
        else
        {
            mode = FTSParser.Mode.DEFAULT_DISJUNCTION;
        }

        Constraint constraint = FTSQueryParser.buildFTS(searchParameters.getQuery(), factory, functionContext, null, null, mode,
                searchParameters.getDefaultFTSOperator() == org.alfresco.service.cmr.search.SearchParameters.Operator.OR ? Connective.OR : Connective.AND,
                searchParameters.getQueryTemplates(), searchParameters.getDefaultFieldName(), rerankPhase);
        org.alfresco.repo.search.impl.querymodel.Query queryModelQuery = factory.createQuery(null, null, constraint, new ArrayList<>());

        @SuppressWarnings("unchecked")
        LuceneQueryBuilder<Query, Sort, ParseException> builder = (LuceneQueryBuilder<Query, Sort, ParseException>) queryModelQuery;

        LuceneQueryBuilderContext<Query, Sort, ParseException> luceneContext = getLuceneQueryBuilderContext(searchParameters, req, CMISStrictDictionaryService.DEFAULT, rerankPhase);

        Set<String> selectorGroup = null;
        if (queryModelQuery.getSource() != null)
        {
            List<Set<String>> selectorGroups = queryModelQuery.getSource().getSelectorGroups(functionContext);

            if (selectorGroups.size() == 0)
            {
                throw new UnsupportedOperationException("No selectors");
            }

            if (selectorGroups.size() > 1)
            {
                throw new UnsupportedOperationException("Advanced join is not supported");
            }

            selectorGroup = selectorGroups.get(0);
        }
        Query luceneQuery = builder.buildQuery(selectorGroup, luceneContext, functionContext);
        // query needs some search parameters fro correct caching ....

        return new ContextAwareQuery(luceneQuery, Boolean.TRUE.equals(isFilter) ? null : searchParameters);
    }

    private PropertyDefinition getPropertyDefinition(String identifier)
    {
        return QueryParserUtils.matchPropertyDefinition(NamespaceService.CONTENT_MODEL_1_0_URI,
                getNamespaceDAO(),
                getDictionaryService(CMISStrictDictionaryService.DEFAULT),
                identifier);
    }

    public String  mapProperty(String  potentialProperty,  FieldUse fieldUse, SolrQueryRequest req)
    {
        return mapProperty(potentialProperty, fieldUse, req, 0);
    }

    /**
     *
     * return the stored field associated to potentialProperty parameter
     */
    public String mapStoredProperty(String potentialProperty, SolrQueryRequest req)
    {
        if(potentialProperty.equals("asc") || potentialProperty.equals("desc") || potentialProperty.equals("_docid_"))
        {
            return potentialProperty;
        }

        if(potentialProperty.equalsIgnoreCase("score") || potentialProperty.equalsIgnoreCase("SEARCH_SCORE"))
        {
            return "score";
        }

        AlfrescoFunctionEvaluationContext functionContext =
            new AlfrescoSolr4FunctionEvaluationContext(
                getNamespaceDAO(),
                getDictionaryService(CMISStrictDictionaryService.DEFAULT),
                NamespaceService.CONTENT_MODEL_1_0_URI,
                req.getSchema());


        Pair<String, String> fieldNameAndEnding = QueryParserUtils.extractFieldNameAndEnding(potentialProperty);
        String luceneField =  functionContext.getLuceneFieldName(fieldNameAndEnding.getFirst());

        PropertyDefinition propertyDef = getPropertyDefinition(fieldNameAndEnding.getFirst());
        //Retry scan using luceneField.
        if(propertyDef == null)
        {
            if(luceneField.contains("@"))
            {
                int index = luceneField.lastIndexOf("@");
                propertyDef = getPropertyDefinition(luceneField.substring(index +1));
            }
        }

        if (propertyDef == null || propertyDef.getName() == null)
        {
            return mapNonPropertyFields(luceneField);
        }

        if (isDateOrDatetime(propertyDef.getDataType()) && isDerivedDateField(fieldNameAndEnding.getSecond()))
        {
            return getDateDerivedField(propertyDef.getName(), fieldNameAndEnding.getSecond());
        }
        else if (propertyDef.getDataType().getName().equals(DataTypeDefinition.TEXT))
        {
            return getStoredTextField(propertyDef.getName(), fieldNameAndEnding.getSecond());
        }
        else if (propertyDef.getDataType().getName().equals(DataTypeDefinition.MLTEXT))
        {
            return getStoredMLTextField(propertyDef.getName(), fieldNameAndEnding.getSecond());
        }
        else if (propertyDef.getDataType().getName().equals(DataTypeDefinition.CONTENT))
        {
            return getStoredContentField(propertyDef.getName(), fieldNameAndEnding.getSecond());
        }
        else
        {
            return mapAlfrescoField(FieldUse.FTS, 0, fieldNameAndEnding, luceneField, propertyDef)
                    + fieldNameAndEnding.getSecond();
        }
    }

    public boolean isDerivedDateField(String suffix)
    {
        return DATE_PART_SUFFIXES.contains(suffix);
    }

    public String  mapProperty(String  potentialProperty,  FieldUse fieldUse, SolrQueryRequest req, int position)
    {
        if(potentialProperty.equals("asc") || potentialProperty.equals("desc") || potentialProperty.equals("_docid_"))
        {
            return potentialProperty;
        }

        if(potentialProperty.equalsIgnoreCase("score") || potentialProperty.equalsIgnoreCase("SEARCH_SCORE"))
        {
            return "score";
        }

        if(req.getSchema().getFieldOrNull(potentialProperty) != null)
        {
            return mapNonPropertyFields(potentialProperty);
        }

        AlfrescoFunctionEvaluationContext functionContext =
                new AlfrescoSolr4FunctionEvaluationContext(
                        getNamespaceDAO(),
                        getDictionaryService(CMISStrictDictionaryService.DEFAULT),
                        NamespaceService.CONTENT_MODEL_1_0_URI,
                        req.getSchema());

        Pair<String, String> fieldNameAndEnding = QueryParserUtils.extractFieldNameAndEnding(potentialProperty);
        String luceneField =  functionContext.getLuceneFieldName(fieldNameAndEnding.getFirst());

        PropertyDefinition propertyDef = getPropertyDefinition(fieldNameAndEnding.getFirst());
        //Retry scan using luceneField.
        if(propertyDef == null)
        {
            if(luceneField.contains("@"))
            {
                int index = luceneField.lastIndexOf("@");
                propertyDef = getPropertyDefinition(luceneField.substring(index +1));
            }
        }
        String solrSortField;
        solrSortField = mapAlfrescoField(fieldUse, position, fieldNameAndEnding, luceneField, propertyDef);
        return solrSortField;
    }

    public String mapNonPropertyFields(String queryField)
    {
        switch(queryField)
        {
            case "ID":
                return "LID";
            case "EXACTTYPE":
                return "TYPE";
            default:
                return queryField;

        }
    }

    /**
     * @param ending String
     * @return SpecializedFieldType
     */
    public SpecializedFieldType getTextField(String ending)
    {
        switch(ending)
        {
            case FIELD_MIMETYPE_SUFFIX:
                return SpecializedFieldType.CONTENT_MIMETYPE;
            case FIELD_SIZE_SUFFIX:
                return SpecializedFieldType.CONTENT_SIZE;
            case FIELD_LOCALE_SUFFIX:
                return SpecializedFieldType.CONTENT_LOCALE;
            case FIELD_ENCODING_SUFFIX:
                return SpecializedFieldType.CONTENT_ENCODING;
            case UNIT_OF_TIME_SECOND_FIELD_SUFFIX:
                return SpecializedFieldType.UNIT_OF_TIME_SECOND;
            case UNIT_OF_TIME_MINUTE_FIELD_SUFFIX:
                return SpecializedFieldType.UNIT_OF_TIME_MINUTE;
            case UNIT_OF_TIME_HOUR_FIELD_SUFFIX:
                return SpecializedFieldType.UNIT_OF_TIME_HOUR;
            case UNIT_OF_TIME_DAY_FIELD_SUFFIX:
                return SpecializedFieldType.UNIT_OF_TIME_DAY;
            case UNIT_OF_TIME_MONTH_FIELD_SUFFIX:
                return SpecializedFieldType.UNIT_OF_TIME_MONTH;
            case UNIT_OF_TIME_QUARTER_FIELD_SUFFIX:
                return SpecializedFieldType.UNIT_OF_TIME_QUARTER;
            case UNIT_OF_TIME_YEAR_FIELD_SUFFIX:
                return SpecializedFieldType.UNIT_OF_TIME_YEAR;
            case FIELD_TRANSFORMATION_STATUS_SUFFIX:
            case FIELD_TRANSFORMATION_TIME_SUFFIX:
            case FIELD_TRANSFORMATION_EXCEPTION_SUFFIX:
            default:
                return null;
        }
    }

    public void setCMDefaultUri()
    {
        if(getNamespaceDAO().getURIs().contains(NamespaceService.CONTENT_MODEL_1_0_URI))
        {
            getNamespaceDAO().addPrefix("", NamespaceService.CONTENT_MODEL_1_0_URI);
        }
    }

    /**
     * Returns the prefix used for denoting a field which is meant to represent a constituent part of a
     * date or datetime (e.g. year, month, second, minute).
     *
     * @param sourceFieldName the date/datetime source field name.
     * @return the prefix that can be used for denoting a date or datetime part
     */
    public String destructuredDateTimePartFieldNamePrefix(String sourceFieldName) {
        // source field name example: [email protected]@{http://www.alfresco.org/model/content/1.0}created
        // prefix: datetime
        String prefix = sourceFieldName.substring(0, sourceFieldName.indexOf("@"));

        // prefix, docValues disabled option: [email protected][email protected]
        String sourceFieldNamePrefixWithoutDocValues = prefix + SINGLE_VALUE_WITHOUT_DOC_VALUES_MARKER;

        // prefix, docValues enabled option: [email protected]@
        String sourceFieldNamePrefixWithDocValues = prefix + SINGLE_VALUE_WITH_DOC_VALUES_MARKER;

        return sourceFieldName.replace(sourceFieldNamePrefixWithoutDocValues, PART_FIELDNAME_PREFIX)
                .replace(sourceFieldNamePrefixWithDocValues, PART_FIELDNAME_PREFIX);
    }

    private boolean isDateOrDatetime(DataTypeDefinition dataType)
    {
        return dataType.getName().equals(DataTypeDefinition.DATE) ||
                dataType.getName().equals(DataTypeDefinition.DATETIME);
    }

    private boolean isPrimitive(DataTypeDefinition dataType)
    {
        QName name = dataType.getName();
        return name.equals(DataTypeDefinition.INT)
                || name.equals(DataTypeDefinition.LONG)
                || name.equals(DataTypeDefinition.FLOAT)
                || name.equals(DataTypeDefinition.DOUBLE)
                || isDateOrDatetime(dataType)
                || name.equals(DataTypeDefinition.BOOLEAN)
                || name.equals(DataTypeDefinition.CATEGORY)
                || name.equals(DataTypeDefinition.NODE_REF);
    }

    private String mapAlfrescoField(FieldUse fieldUse, int position, Pair<String, String> fieldNameAndEnding, String luceneField, PropertyDefinition propertyDef)
    {
        String solrSortField;
        if(propertyDef != null)
        {

            IndexedField fields = AlfrescoSolrDataModel.getInstance().getQueryableFields(propertyDef.getName(), getTextField(fieldNameAndEnding.getSecond()), fieldUse);
            if(fields.getFields().size() > 0)
            {
                if(fields.getFields().size() > position)
                {
                    solrSortField = fields.getFields().get(position).getField();
                }
                else
                {
                    solrSortField = fields.getFields().get(0).getField();
                }
            }
            else
            {
                solrSortField = mapNonPropertyFields(luceneField);
            }
        }
        else
        {
            solrSortField = mapNonPropertyFields(luceneField);
        }
        return solrSortField;
    }

    private Set<String> validateModel(M2Model model)
    {
        try
        {
            dictionaryDAO.getCompiledModel(QName.createQName(model.getName(), namespaceDAO));
        }
        catch (DictionaryException | NamespaceException exception)
        {
            // No model to diff
            return Collections.emptySet();
        }

        // namespace unknown - no model
        List<M2ModelDiff> modelDiffs = dictionaryDAO.diffModelIgnoringConstraints(model);
        return modelDiffs.stream()
                .filter(diff -> diff.getDiffType().equals(M2ModelDiff.DIFF_UPDATED))
                .map(diff ->
                        String.format("Model not updated: %s Failed to validate model update - found non-incrementally updated %s '%s'",
                                model.getName(),
                                diff.getElementType(),
                                diff.getElementName()))
                .collect(Collectors.toSet());
    }
}