| /******************************************************************************* |
| * Copyright (c) 2013 Obeo. |
| * 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: |
| * Obeo - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.acceleo.ui.interpreter.ocl; |
| |
| import java.util.LinkedList; |
| import java.util.List; |
| import java.util.concurrent.Callable; |
| |
| import org.eclipse.acceleo.ui.interpreter.language.CompilationResult; |
| import org.eclipse.acceleo.ui.interpreter.language.EvaluationContext; |
| import org.eclipse.acceleo.ui.interpreter.language.SplitExpression; |
| import org.eclipse.acceleo.ui.interpreter.language.SubExpression; |
| import org.eclipse.core.runtime.IStatus; |
| import org.eclipse.ocl.pivot.CallExp; |
| import org.eclipse.ocl.pivot.ExpressionInOCL; |
| import org.eclipse.ocl.pivot.IfExp; |
| import org.eclipse.ocl.pivot.LetExp; |
| import org.eclipse.ocl.pivot.LiteralExp; |
| import org.eclipse.ocl.pivot.OCLExpression; |
| import org.eclipse.ocl.pivot.OperationCallExp; |
| import org.eclipse.ocl.pivot.Variable; |
| import org.eclipse.ocl.pivot.VariableExp; |
| import org.eclipse.ocl.pivot.util.AbstractExtendingVisitor; |
| import org.eclipse.ocl.pivot.util.Visitable; |
| |
| /** |
| * This class aims at providing the necessary API to split an OCL query that was compiled from a simple |
| * String. |
| * |
| * @author <a href="mailto:marwa.rostren@obeo.fr">Marwa Rostren</a> |
| */ |
| public class OCLExpressionSplittingTask implements Callable<SplitExpression> { |
| /** The current context. */ |
| private EvaluationContext context; |
| |
| /** |
| * Instantiates the splitting task for the given evaluation context. |
| * |
| * @param context |
| * The current context. |
| */ |
| public OCLExpressionSplittingTask(EvaluationContext context) { |
| this.context = context; |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see java.util.concurrent.Callable#call() |
| */ |
| public SplitExpression call() throws Exception { |
| CompilationResult compilationResult = context.getCompilationResult(); |
| if (compilationResult == null |
| || (compilationResult.getStatus() != null && compilationResult.getStatus().getSeverity() != IStatus.OK)) { |
| return null; |
| } |
| |
| final SplitExpression result; |
| Object expression = compilationResult.getCompiledExpression(); |
| if (expression instanceof ExpressionInOCL) { |
| SplittingVisitor visitor = new SplittingVisitor((ExpressionInOCL)expression); |
| visitor.visitExpressionInOCL((ExpressionInOCL)expression); |
| |
| result = new SplitExpression(expression, visitor.getSubExpressions()); |
| } else { |
| // Cannot split an expression that is not an ExpressionInOCL |
| result = null; |
| } |
| |
| return result; |
| } |
| |
| /** |
| * The only purpose of this visitor is to split the given expression into its individual components. |
| * |
| * @author <a href="mailto:marwa.rostren@obeo.fr">Marwa Rostren</a> |
| */ |
| private static class SplittingVisitor extends AbstractExtendingVisitor<SubExpression, ExpressionInOCL> { |
| /** |
| * Some sub-expressions can have sub-steps of theirs own (for example, an operation call contains all |
| * of its arguments as sub-steps). We'll push each such "parent" sub-expression on top of this stack, |
| * whereas the "leaf" sub-expressions will be added as a sub-step of the parent currently on the tip |
| * of this stack. |
| */ |
| private LinkedList<SubExpression> expressionStack = new LinkedList<SubExpression>(); |
| |
| /** |
| * This splits the OCL Expression to several sub-steps. |
| * |
| * @param expressionInOCL |
| * The OCL expression to evaluate. |
| */ |
| public SplittingVisitor(ExpressionInOCL expressionInOCL) { |
| super(expressionInOCL); |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visitExpressionInOCL(ExpressionInOCL) |
| */ |
| @Override |
| public SubExpression visitExpressionInOCL(ExpressionInOCL object) { |
| expressionStack.addFirst(new SubExpression(object.getOwnedBody())); |
| return object.getOwnedBody().accept(this); |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visiting(Visitable) |
| */ |
| public SubExpression visiting(Visitable visitable) { |
| return null; |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visitOperationCallExp(OperationCallExp) |
| */ |
| @Override |
| public SubExpression visitOperationCallExp(OperationCallExp object) { |
| for (OCLExpression expression : object.getOwnedArguments()) { |
| addAndVisitSubStep(expression); |
| } |
| SubExpression result = super.visitOperationCallExp(object); |
| return result; |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visitLetExp(LetExp) |
| */ |
| @Override |
| public SubExpression visitLetExp(LetExp object) { |
| addChild(new SubExpression(object)); |
| |
| // variable definition should not be displayed |
| object.getOwnedVariable().accept(this); |
| |
| return addAndVisitSubStep(object.getOwnedIn()); |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visitVariable(Variable) |
| */ |
| @Override |
| public SubExpression visitVariable(Variable object) { |
| return addAndVisitSubStep(object.getOwnedInit()); |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visitIfExp(IfExp) |
| */ |
| @Override |
| public SubExpression visitIfExp(IfExp object) { |
| addChild(new SubExpression(object)); |
| SubExpression result = super.visitIfExp(object); |
| |
| addAndVisitSubStep(object.getOwnedCondition()); |
| addAndVisitSubStep(object.getOwnedThen()); |
| addAndVisitSubStep(object.getOwnedElse()); |
| |
| return result; |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visitVariableExp(VariableExp) |
| */ |
| @Override |
| public SubExpression visitVariableExp(VariableExp object) { |
| addChild(new SubExpression(object)); |
| return super.visitVariableExp(object); |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visitLiteralExp(LiteralExp) |
| */ |
| @Override |
| public SubExpression visitLiteralExp(LiteralExp object) { |
| addChild(new SubExpression(object)); |
| return super.visitLiteralExp(object); |
| } |
| |
| /** |
| * {@inheritDoc} |
| * |
| * @see org.eclipse.ocl.pivot.util.Visitor#visitCallExp(CallExp) |
| */ |
| @Override |
| public SubExpression visitCallExp(CallExp object) { |
| addChild(new SubExpression(object)); |
| return addAndVisitSubStep(object.getOwnedSource()); |
| } |
| |
| /** |
| * Returns all sub-expressions registered by this visitor. |
| * |
| * @return All sub-expressions registered by this visitor. |
| */ |
| public List<SubExpression> getSubExpressions() { |
| return expressionStack.pop().getSubSteps(); |
| } |
| |
| /** |
| * This will, in order : |
| * <ul> |
| * <li>Add the given expression as a sub-step of the expression currently at the tip of the expression |
| * stack,</li> |
| * <li>Push the given expression on top of the expression stack so that all of its own sub-steps can |
| * be registered,</li> |
| * <li>Visit the given expression to register its sub-steps,</li> |
| * <li>Pop the given expression out of the expression stack.</li> |
| * </ul> |
| * |
| * @param subStep |
| * The expression which sub-steps to register. |
| * @return The SubExpression instance corresponding to the given expression. |
| */ |
| private SubExpression addAndVisitSubStep(OCLExpression subStep) { |
| final SubExpression subExpression = new SubExpression(subStep); |
| addChild(subExpression); |
| expressionStack.addFirst(subExpression); |
| subStep.accept(this); |
| expressionStack.pop(); |
| return subExpression; |
| } |
| |
| /** |
| * Adds the given expression to the children of the expression currently at the tip of the expression |
| * stack. |
| * |
| * @param subExpression |
| * The sub-expression to add. |
| */ |
| private void addChild(SubExpression subExpression) { |
| SubExpression parent = expressionStack.peek(); |
| if (!parent.getExpression().equals(subExpression.getExpression())) { |
| parent.addSubStep(subExpression); |
| } |
| } |
| } |
| } |