001/*
002 *  Copyright 2014 Anyware Services
003 *
004 *  Licensed under the Apache License, Version 2.0 (the "License");
005 *  you may not use this file except in compliance with the License.
006 *  You may obtain a copy of the License at
007 *
008 *      http://www.apache.org/licenses/LICENSE-2.0
009 *
010 *  Unless required by applicable law or agreed to in writing, software
011 *  distributed under the License is distributed on an "AS IS" BASIS,
012 *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 *  See the License for the specific language governing permissions and
014 *  limitations under the License.
015 */
016
017package org.ametys.cms.content.references;
018
019import java.util.HashMap;
020import java.util.Map;
021
022import org.apache.avalon.framework.component.Component;
023import org.apache.commons.lang3.StringUtils;
024
025import org.ametys.cms.repository.Content;
026import org.ametys.plugins.repository.data.holder.ModelAwareDataHolder;
027import org.ametys.plugins.repository.data.holder.group.impl.ModelAwareRepeater;
028import org.ametys.plugins.repository.data.holder.group.impl.ModelAwareRepeaterEntry;
029import org.ametys.plugins.repository.data.type.ModelItemTypeConstants;
030import org.ametys.runtime.model.ModelItem;
031import org.ametys.runtime.model.type.ModelItemType;
032
033/**
034 * The outgoing references extractor is a class that analyzes a content to extract the list of outgoing references in the metadata.
035 * Theses references should be stored on the content to allow research or post-processing later (example: consistency analyzer).
036 */
037public class OutgoingReferencesExtractor implements Component
038{
039    /** Avalon role */
040    public static final String ROLE = OutgoingReferencesExtractor.class.getName();
041
042    /**
043     * This method analyzes the content to return the list of outgoing references grouped by data path.
044     * @param content The content to analyze
045     * @return A Map where keys are data path and values are outgoing references for this data. Can not be null.
046     */
047    public Map<String, OutgoingReferences> getOutgoingReferences(Content content)
048    {
049        return _getOutgoingReferences(content, StringUtils.EMPTY);
050    }
051
052    /**
053     * Analyze a data holder to extract the outgoing references
054     * @param dataHolder the data holder to analyze
055     * @param dataPathPrefix the data path of the current data holder
056     * @return the outgoing references, indexed by data path
057     */
058    protected Map<String, OutgoingReferences> _getOutgoingReferences(ModelAwareDataHolder dataHolder, String dataPathPrefix)
059    {
060        Map<String, OutgoingReferences> outgoingReferencesByPath = new HashMap<>();
061        
062        for (String dataName : dataHolder.getDataNames())
063        {
064            ModelItem definition = dataHolder.getDefinition(dataName);
065            ModelItemType type = definition.getType();
066
067            String dataPath = StringUtils.isNotEmpty(dataPathPrefix) ? dataPathPrefix + ModelItem.ITEM_PATH_SEPARATOR + dataName : dataName;
068
069            if (type instanceof Referencer)
070            {
071                Object value = dataHolder.getValue(dataName);
072                OutgoingReferences outgoingReferences = ((Referencer) type).getOutgoingReferences(value);
073                outgoingReferencesByPath.put(dataPath, outgoingReferences);
074            }
075            else if (ModelItemTypeConstants.REPEATER_TYPE_ID.equals(type.getId()))
076            {
077                ModelAwareRepeater repeater = dataHolder.getRepeater(dataName);
078                for (ModelAwareRepeaterEntry entry : repeater.getEntries())
079                {
080                    Map<String, OutgoingReferences> entryOutgoingReferences = _getOutgoingReferences(entry, dataPath + '[' + entry.getPosition() + ']');
081                    outgoingReferencesByPath.putAll(entryOutgoingReferences);
082                }
083            }
084            else if (ModelItemTypeConstants.COMPOSITE_TYPE_ID.equals(type.getId()))
085            {
086                Map<String, OutgoingReferences> compOutgoingReferences = _getOutgoingReferences(dataHolder.getComposite(dataName), dataPath);
087                outgoingReferencesByPath.putAll(compOutgoingReferences);
088            }
089        }
090
091        return outgoingReferencesByPath;
092    }
093}