All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.gs.dmn.signavio.SignavioDMNModelRepository Maven / Gradle / Ivy

/*
 * Copyright 2016 Goldman Sachs.
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License.
 *
 * You may obtain a copy of the License at
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations under the License.
 */
package com.gs.dmn.signavio;

import com.gs.dmn.DMNModelRepository;
import com.gs.dmn.DRGElementReference;
import com.gs.dmn.ImportPath;
import com.gs.dmn.ast.*;
import com.gs.dmn.runtime.DMNRuntimeException;
import com.gs.dmn.signavio.extension.MultiInstanceDecisionLogic;
import com.gs.dmn.signavio.extension.SignavioExtension;
import org.apache.commons.lang3.StringUtils;

import javax.xml.XMLConstants;
import javax.xml.namespace.QName;
import java.util.*;
import java.util.stream.Collectors;

import static com.gs.dmn.serialization.DMNVersion.LATEST;
import static com.gs.dmn.signavio.extension.SignavioExtension.SIG_EXT_NAMESPACE;

public class SignavioDMNModelRepository extends DMNModelRepository {
    private String schemaNamespace = SIG_EXT_NAMESPACE;
    private final String[] schemaPrefixes = new String[] {
        "signavio", "sigExt", XMLConstants.DEFAULT_NS_PREFIX
    };

    private QName diagramIdQName = new QName(this.schemaNamespace, "diagramId");
    private QName shapeIdQName = new QName(this.schemaNamespace, "shapeId");

    public final SignavioExtension extension = new SignavioExtension(this);

    private final Map cache = new LinkedHashMap<>();
    private final Set iterators = new LinkedHashSet<>();

    public SignavioDMNModelRepository() {
        this(OBJECT_FACTORY.createTDefinitions());
    }

    public SignavioDMNModelRepository(TDefinitions definitions) {
        this(Collections.singletonList(definitions));
    }

    public SignavioDMNModelRepository(List definitionsList) {
        super(definitionsList);
        populateSignavioCaches(definitionsList);
    }

    // Testing only
    public SignavioDMNModelRepository(TDefinitions definitions, String schemaNamespace) {
        this(Collections.singletonList(definitions), schemaNamespace);
    }

    // Testing only
    public SignavioDMNModelRepository(List definitionsList, String schemaNamespace) {
        super(definitionsList);
        this.schemaNamespace = schemaNamespace;
        this.diagramIdQName = new QName(schemaNamespace, "diagramId");
        this.shapeIdQName = new QName(schemaNamespace, "shapeId");
        populateSignavioCaches(definitionsList);
    }

    private void populateSignavioCaches(List definitionsList) {
        for (TDefinitions definitions: definitionsList) {
            // Add DSs
            List elementList = this.extension.findExtensions(definitions.getExtensionElements(), LATEST.getNamespace(), "decisionService");
            for (Object element: elementList) {
                if (element instanceof TDecisionService) {
                    this.addElementMap((TDecisionService) element, definitions);
                }
            }
            // Populate MID iterators
            for (TDecision element: findDecisions(definitions)) {
                if (isMultiInstanceDecision(element)) {
                    MultiInstanceDecisionLogic midLogic = this.extension.multiInstanceDecisionLogic(element);
                    this.iterators.add((TInputData) midLogic.getIterator());
                }
            }
        }
    }

    @Override
    protected void addModelCoordinates(TDefinitions definitions, TDMNElement element, List locationParts) {
        String diagramId = getDiagramId(element);
        if (!StringUtils.isBlank(diagramId)) {
            locationParts.add(String.format("model='%s'", diagramId));
        } else {
            String modelName = definitions.getName();
            if (!StringUtils.isBlank(modelName)) {
                locationParts.add(String.format("model='%s'", modelName));
            }
        }
    }

    @Override
    public SignavioDMNModelRepository copy() {
        return new SignavioDMNModelRepository(this.definitionsList, this.schemaNamespace);
    }

    public String getSchemaNamespace() {
        return this.schemaNamespace;
    }

    public String[] getSchemaPrefixes() {
        return this.schemaPrefixes;
    }

    private QName getDiagramIdQName() {
        return this.diagramIdQName;
    }

    private QName getShapeIdQName() {
        return this.shapeIdQName;
    }

    public String getDiagramId(TDMNElement element) {
        if (element == null) {
            return null;
        }
        Map otherAttributes = element.getOtherAttributes();
        javax.xml.namespace.QName diagramIdQName = getDiagramIdQName();
        return otherAttributes == null ? null : otherAttributes.get(diagramIdQName);
    }

    public String getShapeId(TDRGElement element) {
        if (element == null) {
            return null;
        }
        Map otherAttributes = element.getOtherAttributes();
        javax.xml.namespace.QName shapeIdQName = getShapeIdQName();
        return otherAttributes == null ? null : otherAttributes.get(shapeIdQName);
    }

    public SignavioExtension getExtension() {
        return this.extension;
    }

    public boolean isBKMLinkedToDecision(TNamedElement element) {
        return element instanceof TBusinessKnowledgeModel
                && getOutputDecision((TBusinessKnowledgeModel) element) != null;
    }

    public TDecision getOutputDecision(TBusinessKnowledgeModel element) {
        TDecisionService decisionService = getExtension().referencedService(element);
        return decisionService == null ? null : this.getOutputDecision(decisionService);
    }

    public boolean isMultiInstanceDecision(TDRGElement decision) {
        return this.extension.isMultiInstanceDecision(decision);
    }

    public boolean isIterator(TInputData inputData) {
        return this.iterators.contains(inputData);
    }

    @Override
    protected List> collectTransitiveInputDatas(DRGElementReference parentReference) {
        TDRGElement parent = parentReference.getElement();
        ImportPath parentImportPath = parentReference.getImportPath();
        List> result = new ArrayList<>();
        result.addAll(directInputDatas(parent));
        // Add inputs used in iteration body / topLevelDecision
        if (isMultiInstanceDecision(parent)) {
            MultiInstanceDecisionLogic multiInstanceDecisionLogic = this.extension.multiInstanceDecisionLogic(parent);
            TDecision topLevelDecision = multiInstanceDecisionLogic.getTopLevelDecision();
            DRGElementReference topLevelReference = makeDRGElementReference(topLevelDecision);

            List> inputDataList = collectTransitiveInputDatas(topLevelReference);
            inputDataList.removeIf(tInputDataDMNReference -> tInputDataDMNReference.getElement() == multiInstanceDecisionLogic.getIterator());
            result.addAll(inputDataList);
        }

        // Process direct children
        List childReferences = requiredDecisionReferences(parent);
        for (TDMNElementReference reference: childReferences) {
            TDecision child = findDecisionByRef(parent, reference.getHref());
            if (child != null) {
                String importName = findImportName(parent, reference);
                DRGElementReference childReference = makeDRGElementReference(new ImportPath(parentImportPath, importName), child);
                List> inputReferences = collectTransitiveInputDatas(childReference);
                result.addAll(inputReferences);
            } else {
                throw new DMNRuntimeException(String.format("Cannot find Decision for '%s' in parent '%s'", reference.getHref(), parent.getName()));
            }
        }
        return result;
    }

    public void addItemDefinition(TDefinitions definitions, TItemDefinition itemDefinition) {
        definitions.getItemDefinition().add(itemDefinition);
        this.itemDefinitions.add(itemDefinition);
    }

    public TDRGElement findDRGElementById(String id) {
        String key = makeKey(id);
        TDRGElement result = cache.get(key);
        if (result == null) {
            TDefinitions definitions = getRootDefinitions();
            List drgElements = findDRGElements(definitions);
            for (TDRGElement element: drgElements) {
                if (sameId(element, id)) {
                    result = element;
                    cache.put(key, result);
                    break;
                }
            }
        }
        return result;
    }

    public TDRGElement findDRGElementByDiagramAndShapeIds(String diagramId, String shapeId) {
        String key = makeKey(diagramId, shapeId);
        TDRGElement result = cache.get(key);
        if (result == null) {
            TDefinitions definitions = getRootDefinitions();
            List drgElements = findDRGElements(definitions);
            for (TDRGElement element: drgElements) {
                if (idEndsWith(element, shapeId) || (sameDiagramId(element, diagramId) && sameShapeId(element, shapeId))) {
                    result = element;
                    cache.put(key, result);
                    break;
                }
            }
        }
        return result;
    }

    public TDRGElement findDRGElementByLabel(String label, String diagramId, String shapeId) {
        String key = makeKey(label, diagramId, shapeId);
        TDRGElement result = cache.get(key);
        if (result == null) {
            TDefinitions definitions = getRootDefinitions();
            List drgElements = findDRGElements(definitions);
            List elements = drgElements.stream().filter(element -> sameLabel(element, label)).collect(Collectors.toList());
            if (elements.size() == 0) {
                result = null;
            } else if (elements.size() == 1) {
                result = elements.get(0);
            } else {
                List sameShapeIdElements = elements.stream().filter(e -> sameShapeId(e, shapeId)).collect(Collectors.toList());
                QName diagramQName = getDiagramIdQName();
                String newDiagramID = elements.stream().filter(e -> sameShapeId(e, shapeId)).map(e -> e.getOtherAttributes().get(diagramQName)).collect(Collectors.joining(", "));
                if (sameShapeIdElements.size() == 1) {
                    LOGGER.warn(String.format("Incorrect diagramId for test input with label '%s' diagramId='%s' shapeId='%s'. DiagramId should be '%s'", label, diagramId, shapeId, newDiagramID));
                    result = sameShapeIdElements.get(0);
                } else {
                    throw new DMNRuntimeException(String.format("Multiple DRGElements for label '%s' with diagramId='%s' shapeId='%s'. Diagram ID should be one of '%s'", label, diagramId, shapeId, newDiagramID));
                }
            }
            if (result != null) {
                cache.put(key, result);
            }
        }
        return result;
    }

    public boolean idEndsWith(TNamedElement element, String id) {
        return id != null && element.getId().endsWith(id);
    }

    private boolean sameId(TNamedElement element, String id) {
        return id != null && id.equals(element.getId());
    }

    private boolean sameDiagramId(TDRGElement element, String id) {
        if (id == null) {
            return false;
        }
        Map otherAttributes = element.getOtherAttributes();
        QName diagramIdQName = getDiagramIdQName();
        String diagramId = otherAttributes.get(diagramIdQName);
        return id.equals(diagramId);
    }

    private boolean sameShapeId(TDRGElement element, String id) {
        if (id == null) {
            return false;
        }
        Map otherAttributes = element.getOtherAttributes();
        QName shapeIdQName = this.getShapeIdQName();
        String shapeId = otherAttributes.get(shapeIdQName);
        return id.equals(shapeId);
    }

    public boolean sameLabel(TNamedElement element, String label) {
        return label != null && label.equals(element.getLabel());
    }

    private String makeKey(String id) {
        return String.format("%s::", id);
    }

    private String makeKey(String diagramId, String shapeId) {
        return String.format("%s:%s:", diagramId, shapeId);
    }

    private String makeKey(String label, String diagramId, String shapeId) {
        return String.format("%s:%s:%s", label, diagramId, shapeId);
    }
}