| /******************************************************************************* |
| * Copyright (c) 2006 Oracle Corporation and others. |
| * All rights reserved. This program and the accompanying materials |
| * are made available under the terms of the Eclipse Public License v1.0 |
| * which accompanies this distribution, and is available at |
| * http://www.eclipse.org/legal/epl-v10.html |
| * |
| * Contributors: |
| * Oracle Corporation - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.bpel.validator.rules; |
| |
| |
| import javax.xml.namespace.QName; |
| |
| import org.eclipse.bpel.validator.model.ARule; |
| import org.eclipse.bpel.validator.model.IFilter; |
| import org.eclipse.bpel.validator.model.IModelQueryLookups; |
| import org.eclipse.bpel.validator.model.INode; |
| import org.eclipse.bpel.validator.model.IProblem; |
| import org.eclipse.bpel.validator.model.NodeNameFilter; |
| |
| |
| /** |
| * Validates variable related rules. |
| * <p> |
| * When thinking |
| * @author Michal Chmielewski (michal.chmielewski@oracle.com) |
| * @date Sep 14, 2006 |
| * |
| */ |
| |
| |
| @SuppressWarnings("nls") |
| public class VariableValidator extends CValidator { |
| |
| /** Parent nodes */ |
| static public IFilter<INode> PARENTS = new NodeNameFilter( ND_VARIABLES ); |
| |
| String ncName ; |
| |
| protected INode fMessageTypeNode; |
| |
| protected INode fElementNode; |
| |
| private INode fTypeNode; |
| |
| protected INode fFromNode ; |
| |
| /** |
| * Start performing variable checks. |
| */ |
| |
| @Override |
| protected void start () { |
| super.start(); |
| ncName = mNode.getAttribute( AT_NAME ); |
| |
| fFromNode = mNode.getNode(ND_FROM); |
| } |
| |
| |
| /** |
| * @see org.eclipse.bpel.validator.rules.CValidator#checkChildren() |
| */ |
| @Override |
| public void checkChildren() { |
| super.checkChildren(); |
| checkChild(ND_FROM, 0, 1); |
| } |
| |
| |
| /** |
| * Rule to check the name of the variable. |
| */ |
| @ARule( |
| author = "michal.chmielewski@oracle.com", |
| date = "9/14/2006", |
| desc = "Checks that variable NCName further does not contain a period (.) in the name.", |
| sa = 24, |
| errors="BPELC__UNSET_ATTRIBUTE,General.NCName_Bad,BPELC_VARIABLE__NO_DOT" |
| ) |
| public void rule_CheckName_1 () { |
| |
| // Must be a valid NCName ... |
| if (checkNCName(mNode, ncName, AT_NAME ) == false) { |
| return ; |
| } |
| |
| IProblem problem ; |
| // ... and not contain a . |
| if (ncName.indexOf('.') >= 0) { |
| problem = createError(); |
| problem.setAttribute(IProblem.CONTEXT, AT_NAME); |
| problem.fill("BPELC_VARIABLE__NO_DOT", //$NON-NLS-1$ |
| ncName); |
| } |
| } |
| |
| |
| |
| /** |
| * Rule to check the type of the variable. |
| * |
| * It can be either: |
| * 1) MessageType |
| * 2) Element |
| * 3) Type |
| * |
| * Only one must be defined, more then one cannot be defined. |
| * |
| */ |
| @ARule( |
| author = "michal.chmielewski@oracle.com", |
| date = "9/14/2006", |
| desc = "Variable type specification (either element, messaageType, or type).", |
| sa = 25 , |
| errors="BPELC_VARIABLE__NO_TYPE" |
| ) |
| |
| public void rule_CheckType_2 () { |
| |
| int typeCount = 0; |
| IProblem problem; |
| |
| // Check messageType |
| QName messageType = mNode.getAttributeAsQName(AT_MESSAGE_TYPE); |
| if (messageType != null) { |
| typeCount += 1; |
| fMessageTypeNode = mModelQuery.lookup(mNode, |
| IModelQueryLookups.LOOKUP_NODE_MESSAGE_TYPE, |
| messageType); |
| } |
| |
| |
| // Check element |
| QName element = mNode.getAttributeAsQName(AT_ELEMENT); |
| if (element != null) { |
| typeCount += 1; |
| fElementNode = mModelQuery.lookup(mNode, |
| IModelQueryLookups.LOOKUP_NODE_XSD_ELEMENT, |
| element ); |
| } |
| |
| // Check Type (XMLType) |
| QName type = mNode.getAttributeAsQName( AT_TYPE ); |
| if (type != null) { |
| typeCount += 1; |
| fTypeNode = mModelQuery.lookup(mNode, |
| IModelQueryLookups.LOOKUP_NODE_XSD_TYPE, |
| type ); |
| } |
| |
| |
| |
| // Missing and too many types |
| if (typeCount == 0) { |
| problem = createError( ); |
| problem.setAttribute(IProblem.CONTEXT, AT_TYPE); |
| problem.fill( "BPELC_VARIABLE__NO_TYPE", ncName); //$NON-NLS-1$ |
| |
| } else if (typeCount > 1) { |
| |
| problem = createError( ); |
| problem.setAttribute(IProblem.CONTEXT, AT_TYPE); |
| problem.fill( "BPELC_VARIABLE__NO_TYPE", ncName); //$NON-NLS-1$ |
| } |
| } |
| |
| |
| /** |
| * Check message type node |
| */ |
| @ARule( |
| sa = 10, |
| desc = "Make sure that Message Type is visible from the import(s)", |
| author = "michal.chmielewski@oracle.com", |
| date = "01/25/2007", |
| errors="BPELC__UNSET_ATTRIBUTE,BPELC__UNRESOLVED_ATTRIBUTE" |
| ) |
| |
| public void rule_CheckMessageTypeNode_4 () { |
| if (fMessageTypeNode == null) { |
| return ; |
| } |
| |
| checkAttributeNode (mNode, fMessageTypeNode, AT_MESSAGE_TYPE, KIND_NODE ); |
| setValue("type",fMessageTypeNode); |
| } |
| |
| /** |
| * Check message type node |
| */ |
| @ARule( |
| sa = 10, |
| desc = "Make sure that Element is visible from the import(s)", |
| author = "michal.chmielewski@oracle.com", |
| date = "01/25/2007", |
| errors="BPELC__UNSET_ATTRIBUTE,BPELC__UNRESOLVED_ATTRIBUTE" |
| ) |
| public void rule_CheckElementNode_4 () { |
| if (fElementNode == null) { |
| return ; |
| } |
| checkAttributeNode (mNode, fElementNode, AT_ELEMENT , KIND_NODE) ; |
| setValue("type",fElementNode); |
| } |
| |
| /** |
| * Check message type node |
| */ |
| @ARule( |
| sa = 10, |
| desc = "Make sure that Type is visible from the import(s)", |
| author = "michal.chmielewski@oracle.com", |
| date = "01/25/2007", |
| errors="BPELC__UNSET_ATTRIBUTE" |
| ) |
| public void rule_CheckTypeNode_4 () { |
| if (fTypeNode == null) { |
| return ; |
| } |
| checkAttributeNode (mNode, fTypeNode, AT_TYPE, KIND_NODE ); |
| setValue("type",fTypeNode); |
| } |
| |
| |
| /** |
| * |
| * Check if variable is hiding other variables (in higher scopes,process). |
| */ |
| |
| @ARule( |
| sa = 0, |
| desc = "Check to see if variable hides super variables", |
| author = "michal.chmielewski@oracle.com", |
| date = "01/25/2007", |
| infos = "BPELC__LEXICAL_HIDE" |
| ) |
| public void rule_CheckIfVariableHidesSuperVariables_15 () { |
| |
| // check if our variable hides other global variables ? |
| INode refNode = mNode.parentNode(); |
| // refNode == "variables" |
| // |
| if (refNode == null) { |
| return ; |
| } |
| refNode = refNode.parentNode(); |
| // refNode == "scope" or "process" |
| if (refNode == null) { |
| return ; |
| } |
| refNode = refNode.parentNode(); |
| |
| // if scope, then this is non-empty, if process then this is empty |
| if (refNode == null) { |
| return ; |
| } |
| |
| INode node = mModelQuery.lookup(refNode, |
| IModelQueryLookups.LOOKUP_NODE_VARIABLE, |
| ncName); |
| |
| // Does not exist or is the same thing. |
| if (node == null || node.isResolved() == false |
| || |
| mModelQuery.check(IModelQueryLookups.TEST_EQUAL,node, mNode) ) { |
| return ; |
| } |
| |
| IProblem problem = createInfo(); |
| problem.setAttribute(IProblem.CONTEXT, AT_NAME); |
| problem.fill( "BPELC__LEXICAL_HIDE", |
| ND_VARIABLE, |
| ncName ); |
| } |
| |
| |
| /** |
| * Register the types to check for double or conflicting definitions ... |
| */ |
| |
| @ARule( |
| sa = 14, |
| desc = "Mark the types/elements/messages used to be checked for duplicates", |
| author = "michal.chmielewski@oracle.com", |
| date = "02/25/2007", |
| errors="BPELC_XSD__CONFLICTING_DEFINITION" |
| ) |
| public void rule_RegisterTypesToCheck_50 () { |
| |
| registerTypeToCheck ( fElementNode ); |
| registerTypeToCheck ( fTypeNode ); |
| registerTypeToCheck ( fMessageTypeNode ); |
| } |
| |
| |
| |
| /** |
| * Check that if we are a scope that belongs to forEach, we don't declare |
| * a variable that matches the counter name. |
| * |
| */ |
| @ARule( |
| sa = 76, |
| desc = "For <forEach> the enclosed scope MUST NOT " |
| + " declare a variable with the same name as specified in the counterName " |
| + " attribute of <forEach>.", |
| author = "michal.chmielewski@oracle.com", |
| date = "01/20/2007", |
| errors="BPELC_VARIABLE__COUNTER" |
| ) |
| |
| public void rule_CheckForEachCounter_30 () { |
| // scope/variables/variable |
| INode scope = fParentNode.parentNode(); |
| |
| if (ND_SCOPE.equals(scope.nodeName()) == false) { |
| return ; |
| } |
| // forEach/scope/variables/variable |
| INode forEachNode = scope.parentNode(); |
| if (ND_FOR_EACH.equals(forEachNode.nodeName()) == false) { |
| return; |
| } |
| |
| // counter name from forEach. |
| String counterName = forEachNode.getAttribute(AT_COUNTER_NAME); |
| if (isEmptyOrWhitespace(counterName)) { |
| return ; |
| } |
| |
| |
| IProblem problem; |
| |
| if (counterName.equals(ncName)) { |
| problem = createError(); |
| problem.fill("BPELC_VARIABLE__COUNTER", |
| toString(mNode.nodeName()), |
| ncName, |
| forEachNode.nodeName(), |
| forEachNode.getAttribute(AT_NAME)); |
| |
| } |
| |
| |
| } |
| |
| |
| /** |
| * Check compatible implicit copy on this from initializer. |
| * This rule is run at the end (pass2) |
| */ |
| |
| @ARule( |
| sa = 43, |
| desc = "From and to spec must have compatible types", |
| author = "michal.chmielewski@oracle.com", |
| date = "02/16/2007", |
| tag = "pass2", |
| errors="BPELC_COPY__INCOMPATIBLE", |
| warnings="BPELC_COPY__INCOMPATIBLE_SIMPLE", |
| infos = "BPELC_COPY__NOT_CHECKED" |
| ) |
| public void rule_CheckCompatibleCopy_40 () { |
| |
| if (isUndefined(fFromNode)) { |
| return ; |
| } |
| |
| compatibleCopyCheck(fFromNode,mNode); |
| } |
| |
| /** |
| * End of public rule methods. |
| * |
| * Other methods are support methods for this class to perform its |
| * validation function. |
| * |
| */ |
| |
| } |