/*
 * This file is part of ***  M y C o R e  ***
 * See http://www.mycore.de/ for details.
 *
 * MyCoRe 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 3 of the License, or
 * (at your option) any later version.
 *
 * MyCoRe 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 MyCoRe.  If not, see <http://www.gnu.org/licenses/>.
 */

package org.mycore.mods;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.jdom2.Element;
import org.mycore.access.MCRAccessException;
import org.mycore.common.MCRConstants;
import org.mycore.common.MCRException;
import org.mycore.common.MCRPersistenceException;
import org.mycore.common.events.MCREvent;
import org.mycore.common.events.MCREventHandlerBase;
import org.mycore.datamodel.metadata.MCRMetaLinkID;
import org.mycore.datamodel.metadata.MCRMetadataManager;
import org.mycore.datamodel.metadata.MCRObject;
import org.mycore.datamodel.metadata.MCRObjectID;

/**
 * Extracts occurences of mods:relatedItem and stores them as separate MCRObjects. For mods:relatedItem/@type='host',
 * sets the extracted object as parent. Always, sets @xlink:href of mods:relatedItem to the extracted object's ID.
 *
 * @author Frank L\u00FCtzenkirchen
 */
public class MCRExtractRelatedItemsEventHandler extends MCREventHandlerBase {

    private static final Logger LOGGER = LogManager.getLogger(MCRExtractRelatedItemsEventHandler.class);

    /* (non-Javadoc)
     * @see org.mycore.common.events.MCREventHandlerBase
     * #handleObjectCreated(org.mycore.common.events.MCREvent, org.mycore.datamodel.metadata.MCRObject)
     */
    @Override
    protected void handleObjectCreated(final MCREvent evt, final MCRObject obj) {
        extractRelatedItems(obj);
    }

    /* (non-Javadoc)
     * @see org.mycore.common.events.MCREventHandlerBase
     * #handleObjectUpdated(org.mycore.common.events.MCREvent, org.mycore.datamodel.metadata.MCRObject)
     */
    @Override
    protected void handleObjectUpdated(final MCREvent evt, final MCRObject obj) {
        extractRelatedItems(obj);
    }

    /* (non-Javadoc)
     * @see org.mycore.common.events.MCREventHandlerBase
     * #handleObjectRepaired(org.mycore.common.events.MCREvent, org.mycore.datamodel.metadata.MCRObject)
     */
    @Override
    protected void handleObjectRepaired(final MCREvent evt, final MCRObject obj) {
        extractRelatedItems(obj);
    }

    private void extractRelatedItems(final MCRObject object) {
        if (!MCRMODSWrapper.isSupported(object)) {
            return;
        }

        Element mods = new MCRMODSWrapper(object).getMODS();
        MCRObjectID oid = object.getId();
        for (Element relatedItem : mods.getChildren("relatedItem", MCRConstants.MODS_NAMESPACE)) {
            String href = relatedItem.getAttributeValue("href", MCRConstants.XLINK_NAMESPACE);
            LOGGER.info("Found related item in {}, href={}", object.getId(), href);
            //MCR-957: only create releated object if mycoreId
            MCRObjectID mcrIdCheck;
            try {
                mcrIdCheck = MCRObjectID.getInstance(href);
            } catch (Exception e) {
                //not a valid MCRObjectID -> don't create anyway
                continue;
            }
            //create if integer value == 0
            if (mcrIdCheck.getNumberAsInteger() == 0) {
                //MCR-931: check for type='host' and present parent document
                if (!isHost(relatedItem) || object.getStructure().getParentID() == null) {
                    MCRObjectID relatedID;
                    try {
                        relatedID = createRelatedObject(relatedItem, oid);
                    } catch (MCRAccessException e) {
                        throw new MCRException(e);
                    }

                    href = relatedID.toString();
                    LOGGER.info("Setting href of related item to {}", href);
                    relatedItem.setAttribute("href", href, MCRConstants.XLINK_NAMESPACE);

                    if (isHost(relatedItem)) {
                        LOGGER.info("Setting {} as parent of {}", href, oid);
                        object.getStructure().setParent(relatedID);
                    }
                }
            } else if (isParentExists(relatedItem)) {
                MCRObjectID relatedID = MCRObjectID.getInstance(href);
                if (object.getStructure().getParentID() == null) {
                    LOGGER.info("Setting {} as parent of {}", href, oid);
                    object.getStructure().setParent(relatedID);
                } else if (!object.getStructure().getParentID().equals(relatedID)) {
                    LOGGER.info("Setting {} as parent of {}", href, oid);
                    object.getStructure().setParent(relatedID);
                }
            }
        }
    }

    private boolean isHost(Element relatedItem) {
        return "host".equals(relatedItem.getAttributeValue("type"));
    }

    private MCRObjectID createRelatedObject(Element relatedItem, MCRObjectID childID)
        throws MCRPersistenceException, MCRAccessException {
        MCRMODSWrapper wrapper = new MCRMODSWrapper();
        MCRObject object = wrapper.getMCRObject();
        MCRObjectID oid = MCRObjectID.getNextFreeId(childID.getBase());
        if (oid.equals(childID)) {
            oid = MCRObjectID.getNextFreeId(childID.getBase());
        }
        object.setId(oid);

        if (isHost(relatedItem)) {
            object.getStructure().addChild(new MCRMetaLinkID("child", childID, childID.toString(), childID.toString()));
        }

        Element mods = cloneRelatedItem(relatedItem);
        wrapper.setMODS(mods);

        LOGGER.info("create object {}", oid);
        MCRMetadataManager.create(object);
        return oid;
    }

    private Element cloneRelatedItem(Element relatedItem) {
        Element mods = relatedItem.clone();
        mods.setName("mods");
        mods.removeAttribute("type");
        mods.removeAttribute("href", MCRConstants.XLINK_NAMESPACE);
        mods.removeAttribute("type", MCRConstants.XLINK_NAMESPACE);
        mods.removeChildren("part", MCRConstants.MODS_NAMESPACE);
        return mods;
    }

    /**
     * Checks if the given related item is if type host and contains a valid MCRObjectID from an existing object.
     *
     * @param relatedItem
     * @return true if @type='host' and MCRObjectID in @href contains is valid and this MCRObject exists
     */
    private boolean isParentExists(Element relatedItem) {
        String href = relatedItem.getAttributeValue("href", MCRConstants.XLINK_NAMESPACE);
        if (isHost(relatedItem) && href != null && !href.isEmpty()) {
            MCRObjectID relatedID = MCRObjectID.getInstance(href);
            return relatedID != null;
        }
        return false;
    }
}