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

org.cristalise.dsl.lifecycle.instance.BlockDelegate.groovy Maven / Gradle / Ivy

Go to download

CRISTAL-iSE Domain Specific Language module for bootstrapping and funtional testing

There is a newer version: 6.0.0
Show newest version
/**
 * This file is part of the CRISTAL-iSE kernel.
 * Copyright (c) 2001-2015 The CRISTAL Consortium. All rights reserved.
 *
 * This library 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.
 *
 * This library is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; with out 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 this library; if not, write to the Free Software Foundation,
 * Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
 *
 * http://www.fsf.org/licensing/licenses/lgpl.html
 */
package org.cristalise.dsl.lifecycle.instance

import static org.cristalise.kernel.graph.model.BuiltInVertexProperties.ROUTING_EXPR
import static org.cristalise.kernel.graph.model.BuiltInVertexProperties.ROUTING_SCRIPT_NAME
import static org.cristalise.kernel.graph.model.BuiltInVertexProperties.ROUTING_SCRIPT_VERSION

import org.cristalise.kernel.common.InvalidDataException
import org.cristalise.kernel.lifecycle.instance.Activity
import org.cristalise.kernel.lifecycle.instance.AndSplit
import org.cristalise.kernel.lifecycle.instance.CompositeActivity
import org.cristalise.kernel.lifecycle.instance.Join
import org.cristalise.kernel.lifecycle.instance.Loop
import org.cristalise.kernel.lifecycle.instance.OrSplit
import org.cristalise.kernel.lifecycle.instance.Split
import org.cristalise.kernel.lifecycle.instance.WfVertex
import org.cristalise.kernel.lifecycle.instance.XOrSplit
import org.cristalise.kernel.lifecycle.instance.WfVertex.Types

import groovy.transform.CompileStatic
import groovy.util.logging.Slf4j


/**
 * Block is a group of WfVertices, it should only be used within Splits
 */
@CompileStatic @Slf4j
public class BlockDelegate {
    String name = ""

    CompActDelegate parentCABlock = null

    WfVertex firstVertex = null
    WfVertex lastVertex  = null

    Map vertexCache = null

    Map  properties = [:]

    int index = -1

    public BlockDelegate() {}

    public BlockDelegate(CompActDelegate caBlock, Map cache) {
        assert caBlock
        parentCABlock = caBlock
        vertexCache = cache
    }

    
    public WfVertex getVertex(String vName, Types vType = null) throws InvalidDataException {
        def v = vertexCache[vName]
        if(v) {
            if(vType) {
                switch(vType) {
                    case Types.Composite: if(v instanceof CompositeActivity) { return v }; break;
                    case Types.Atomic:    if(v instanceof Activity)          { return v }; break;
                    case Types.OrSplit:   if(v instanceof OrSplit)           { return v }; break;
                    case Types.XOrSplit:  if(v instanceof XOrSplit)          { return v }; break;
                    case Types.AndSplit:  if(v instanceof AndSplit)          { return v }; break;
                    case Types.LoopSplit: if(v instanceof Loop)              { return v }; break;
                    case Types.Join:      if(v instanceof Join)              { return v }; break;
                }
            }
            else return v
        }
        else throw new InvalidDataException("Vertex name:'$vName' is not found in the cache")

        throw new InvalidDataException("Vertex name/type:'$vName/$vType' is in the cache but with incompatible type: ${v.class}")
    }


    protected void setVertexProperties(WfVertex vertex) {
        properties.each { key, value -> 
            vertex.properties.put(key, (value instanceof String) ? (String)value : value, false)
        }
    }

    /**
     * 
     * @param aSplit
     */
    protected void setSplitProperties(WfVertex aSplit) {
        if(properties.containsKey(ROUTING_SCRIPT_NAME.name) || properties.containsKey(ROUTING_EXPR.name)) return

        if(properties.javascript) {
            setRoutingScript(aSplit, (String)"javascript:\"${properties.javascript}\";", null);
            properties.remove('javascript')
        }
        else if(properties.groovy) {
            setRoutingScript(aSplit, (String)"groovy:\"${properties.groovy}\";", null);
            properties.remove('groovy')
        }
        else {
            setRoutingScript(aSplit, "javascript:\"true\";", null);
        }
    }

    /**
     * Convenience method to set RoutingScript quickly
     * 
     * @param aSplit the vertex of type Split
     * @param name value of the RoutingScriptName property
     * @param version value of the RoutingScriptVersion property
     */
    protected static void setRoutingScript(WfVertex aSplit, String name, Integer version) {
        assert aSplit instanceof Split, "BlockDelegate.setRoutingScript() - Vertex '$aSplit.name' must be instance of Split"

        log.debug "setRoutingScript() - splitName: $aSplit.name, name: '$name' version: '$version'"

        aSplit.getProperties().setBuiltInProperty(ROUTING_SCRIPT_NAME,    name,    false);
        aSplit.getProperties().setBuiltInProperty(ROUTING_SCRIPT_VERSION, version, false)
    }

    /**
     * 
     * @param t
     * @return
     */
    public static String getNamePrefix(Types t) {
        switch(t) {
            case Types.Composite: return 'CA'
            case Types.Atomic:    return 'EA'
            default:              return t.toString()
        }
    }

    /**
     * 
     * @param n
     * @param t
     * @param i
     * @return
     */
    public static String getAutoName(String n, Types t, int i) {
        log.debug "getAutoName() - name:'$n', type: $t, index: $i"

        String namePrefix = getNamePrefix(t)

        if (n) {
            if (t.name().contains("Split")) assert n.startsWith(namePrefix), "Name shall start with prefix '$namePrefix' for Splits"

            return n
        }
        else {
            return "${namePrefix}" + ((i == 0) ? "" : "$i")
        }
    }

    public void updateVertexCache(Types t, String n, WfVertex v) {
        if (n) {
            if(vertexCache.containsKey(n)) throw new RuntimeException("Vertex name '$n' must be unique")
            else vertexCache[n] = v
        }
    }

    public WfVertex addVertex(Types t, String name) {
        WfVertex v = parentCABlock.createVertex(t, name)

        log.debug "addVertex() - type: '$t'; id: '$v.ID'; name: '$name;' path: '$v.path'"

        if(!firstVertex) firstVertex = v
        if(lastVertex) lastVertex.addNext(v)
        lastVertex = v

        return v
    }

    /**
     * 
     * @param parentBlock
     * @param cl
     */
    public void processClosure(Closure cl) {
        assert cl, "Block only works with a valid Closure"

        cl.delegate = this
        cl.resolveStrategy = Closure.DELEGATE_FIRST
        cl()
    }

    /**
     * Links the current Block with its child Block
     * 
     * @param childBlock the child Block to be linked with
     */
    protected void linkWithChild(BlockDelegate childBlock) {
        log.debug "linkWithChild() - class:'${childBlock.getClass()}', name: '$childBlock.name'"

        if(!firstVertex) firstVertex = childBlock.firstVertex
        else             lastVertex.addNext(childBlock.firstVertex)

        lastVertex = childBlock.lastVertex
    }

    /**
     * DSL method to be used to set RoutingScript quickly
     * 
     * @param splitName name of the Split
     * @param name value of the RoutingScriptName property
     * @param version value of the RoutingScriptVersion property
     */
    public void setRoutingScript(String splitName, String name, Integer version) {
        setRoutingScript(getVertex(splitName), name, version)
    }

    /**
     * 
     * @param splitName
     * @param scriptText
     */
    public void setRoutingScript(String splitName, String scriptText) {
        setRoutingScript(getVertex(splitName), scriptText, null)
    }

    /**
     * DSL method to add properties
     * 
     * @param props Map containing properties
     */
    public void Property(Map props) {
        log.debug "Property() - adding props: $props"
        properties << props
    }

    /**
     * Alias of the DSL method Block()
     * 
     * @param cl the closure to be executed to build the Block
     */
    public void B(Closure cl) {
        Block(cl)
    }

    /**
     * DSL method to add a Block to the Workflow
     * 
     * @param cl the closure to be executed to build the Block
     */
    public void Block(Closure cl) {
        def b = new BlockDelegate(parentCABlock, vertexCache)
        b.processClosure(cl)
        linkWithChild(b)
    }

    /**
     * DSL method to add a Composite Activity to the Workflow
     *
     * @param name the name of the Composite Activity, can be omitted
     * @param cl the closure to be executed to build the Composite Activity
     */
    public void CompAct(String n = "", Closure cl) {
        def b = new CompActDelegate(n, vertexCache)
        b.processClosure(this,cl)
    }

    /**
     * 
     * @param props
     * @param cl
     */
    public void Split(Map props, Types t, Closure cl) {
        def b = new SplitDelegate(props, t, parentCABlock, vertexCache)
        b.processClosure(cl)
        linkWithChild(b)
    }

    /**
     * DSL method to add a typed Split to the Workflow
     * 
     * @param name the name of the Split, can be omitted
     * @param type the type (And,Or,XOr, Loop) of the Split
     * @param cl the closure to be executed to build the Split
     */
    public void Split(String n = "", Types t, Closure cl) {
        Split(name: n, t, cl)
    }

    /**
     * DSL method to add an Elementary Activity to the Workflow
     *
     * @param name
     * @param cl the closure to be executed to build the Elementary Activity. Can be omitted.
     */
    public void ElemAct(String n = "", Closure cl = null) {
        new ElemActDelegate(n).processClosure(this, cl)
    }


    /**
     * Alias of the DSL method ElemAct()
     * 
     * @param name
     * @param cl the closure to be executed to build the Elementary Activity. Can be omitted.
     */
    public void EA(String n = "", Closure cl = null) {
        ElemAct(n, cl)
    }

    /**
     * Alias of DSL method CompAct()
     * 
     * @param name
     * @param cl
     */
    public void CA(String n = "", Closure cl) {
        CompAct(n, cl)
    }

    /**
     * 
     * @param props
     * @param cl
     */
    public void AndSplit(Map props, Closure cl) {
        Split(props, Types.AndSplit, cl)
    }

    /**
     * 
     * 
     * @param name
     * @param cl
     */
    public void AndSplit(String n = "", Closure cl) {
        Split(n, Types.AndSplit, cl)
    }

    /**
     * 
     * @param props
     * @param cl
     */
    public void OrSplit(Map props, Closure cl) {
        Split(props, Types.OrSplit, cl)
    }

    /**
     * 
     * @param name
     * @param cl
     */
    public void OrSplit(String n = "", Closure cl) {
        Split(name: n, Types.OrSplit, cl)
    }

    /**
     * 
     * @param name
     * @param cl
     */
    public void XOrSplit(Map props, Closure cl) {
        Split(props, Types.XOrSplit, cl)
    }

    /**
     * 
     * @param name
     * @param cl
     */
    public void XOrSplit(String n = "", Closure cl) {
        Split(n, Types.XOrSplit, cl)
    }

    /**
     * 
     * @param name
     * @param cl
     */
    public void Loop(Map props, Closure cl) {
        def b = new LoopDelegate(props, parentCABlock, vertexCache)
        b.processClosure(cl)
        linkWithChild(b)
    }

    /**
     * 
     * @param name
     * @param cl
     */
    public void Loop(String n = "", Closure cl) {
        Loop(name: n, cl)
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy