| /* |
| * Copyright (c) 2005, 2009 Sven Efftinge 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: |
| * Sven Efftinge - Initial API and implementation |
| * Artem Tikhomirov (Borland) - Migration to OCL expressions |
| */ |
| package org.eclipse.gmf.internal.xpand.ast; |
| |
| import java.util.Arrays; |
| import java.util.Collection; |
| import java.util.Iterator; |
| import java.util.Set; |
| |
| import org.eclipse.emf.ecore.EClassifier; |
| import org.eclipse.gmf.internal.xpand.XpandFacade; |
| import org.eclipse.gmf.internal.xpand.model.AmbiguousDefinitionException; |
| import org.eclipse.gmf.internal.xpand.model.AnalysationIssue; |
| import org.eclipse.gmf.internal.xpand.model.EvaluationException; |
| import org.eclipse.gmf.internal.xpand.model.ExecutionContext; |
| import org.eclipse.gmf.internal.xpand.model.Variable; |
| import org.eclipse.gmf.internal.xpand.model.XpandDefinition; |
| import org.eclipse.gmf.internal.xpand.ocl.ExpressionHelper; |
| import org.eclipse.gmf.internal.xpand.ocl.TypeHelper; |
| import org.eclipse.ocl.cst.OCLExpressionCS; |
| import org.eclipse.ocl.cst.PathNameCS; |
| import org.eclipse.ocl.ecore.CollectionType; |
| |
| /** |
| * @author Sven Efftinge |
| */ |
| public class ExpandStatement extends Statement { |
| |
| private final boolean isForeach; |
| |
| private final ExpressionHelper[] parameters; |
| |
| private final ExpressionHelper separator; |
| |
| private final ExpressionHelper target; |
| |
| private final String definition; |
| |
| public ExpandStatement(final int start, final int end, final int line, final PathNameCS definition, |
| final OCLExpressionCS target, final OCLExpressionCS separator, final OCLExpressionCS[] parameters, final boolean foreach) { |
| super(start, end, line); |
| this.definition = TypeHelper.toString(definition); |
| this.target = target == null ? null : new ExpressionHelper(target, this); |
| this.separator = separator == null ? null : new ExpressionHelper(separator, this); |
| if (parameters == null) { |
| this.parameters = new ExpressionHelper[0]; |
| } else { |
| this.parameters = new ExpressionHelper[parameters.length]; |
| for (int i = 0; i < parameters.length; i++) { |
| this.parameters[i] = new ExpressionHelper(parameters[i], this); |
| } |
| } |
| this.isForeach = foreach; |
| } |
| |
| public void analyze(final ExecutionContext ctx, final Set<AnalysationIssue> issues) { |
| final EClassifier[] paramTypes = new EClassifier[parameters.length]; |
| for (int i = 0; i < parameters.length; i++) { |
| paramTypes[i] = parameters[i].analyze(ctx, issues); |
| |
| } |
| EClassifier targetType = null; |
| if (isForeach) { |
| targetType = target.analyze(ctx, issues); |
| if (targetType instanceof CollectionType) { |
| targetType = ((CollectionType) targetType).getElementType(); |
| } else { |
| issues.add(new AnalysationIssue(AnalysationIssue.Type.INCOMPATIBLE_TYPES, "Collection type expected!", target)); |
| return; |
| } |
| } else { |
| final Variable var = ctx.getImplicitVariable(); |
| if (var == null) { |
| issues.add(new AnalysationIssue(AnalysationIssue.Type.INTERNAL_ERROR, "No implicite variable 'this/self' could be found!", target)); |
| return; |
| } |
| targetType = var.getType(); |
| if (target != null) { |
| targetType = target.analyze(ctx, issues); |
| } |
| } |
| if ((targetType == null) || Arrays.asList(paramTypes).contains(null)) { |
| return; |
| } |
| try { |
| final XpandDefinition def = ctx.findDefinition(definition, targetType, paramTypes); |
| if (def == null) { |
| issues.add(new AnalysationIssue(AnalysationIssue.Type.DEFINITION_NOT_FOUND, |
| "Couldn't find definition " + definition + getParamTypeString(paramTypes) |
| + " for type " + targetType.getName(), this)); |
| } |
| } catch (AmbiguousDefinitionException e) { |
| issues.add(new AnalysationIssue(AnalysationIssue.Type.DEFINITION_NOT_FOUND, e.getMessage(), this)); |
| } |
| } |
| |
| @Override |
| public void evaluateInternal(final ExecutionContext ctx) { |
| final Object[] params = new Object[parameters.length]; |
| for (int i = 0; i < parameters.length; i++) { |
| params[i] = parameters[i].evaluate(ctx); |
| } |
| final String sep = (String) (separator != null ? separator.evaluate(ctx) : null); |
| Object targetObject = null; |
| XpandFacade xpandFacade = new XpandFacade(ctx); |
| try { |
| if (isForeach) { |
| targetObject = target.evaluate(ctx); |
| if (!(targetObject instanceof Collection)) { |
| throw new EvaluationException("Collection expected (was: " + targetObject.getClass().getName() + ")!", target); |
| } |
| |
| final Collection<?> col = (Collection<?>) targetObject; |
| for (final Iterator<?> iter = col.iterator(); iter.hasNext();) { |
| |
| xpandFacade.evaluate(definition, iter.next(), params); |
| if ((sep != null) && iter.hasNext()) { |
| ctx.getScope().getOutput().write(sep); |
| } |
| } |
| |
| } else { |
| if (target != null) { |
| targetObject = target.evaluate(ctx); |
| } else { |
| final Variable var = ctx.getImplicitVariable(); |
| targetObject = var.getValue(); |
| } |
| if (targetObject != null) { |
| xpandFacade.evaluate(definition, targetObject, params); |
| } else { |
| // XXX logInfo that feature value is null or conditionally fail? |
| // perhaps, could check if target is feature and multiplicity of the feature is at least 1 and fail then? |
| // though all these checks are not template's tasks |
| } |
| } |
| } catch (AmbiguousDefinitionException e) { |
| throw new EvaluationException(e.getMessage(), this); |
| } |
| |
| } |
| |
| private String getParamTypeString(final EClassifier[] paramTypes) { |
| if (paramTypes.length == 0) { |
| return ""; |
| } |
| final StringBuffer buff = new StringBuffer("("); |
| for (int i = 0; i < paramTypes.length; i++) { |
| final EClassifier type = paramTypes[i]; |
| buff.append(type.getName()); |
| if (i + 1 < paramTypes.length) { |
| buff.append(", "); |
| } |
| } |
| return buff.append(")").toString(); |
| } |
| |
| private String getParamString() { |
| if (parameters.length == 0) { |
| return ""; |
| } |
| final StringBuffer buff = new StringBuffer("("); |
| for (int i = 0; i < parameters.length; i++) { |
| buff.append(parameters[i]); |
| if (i + 1 < parameters.length) { |
| buff.append(", "); |
| } |
| } |
| return buff.append(")").toString(); |
| } |
| |
| @Override |
| public String toString() { |
| return "EXPAND " + definition + getParamString() |
| + (target != null ? (isForeach ? " FOREACH " : " FOR ") + target : "") |
| + (separator != null ? " SEPARATOR " + separator : ""); |
| } |
| |
| ExpressionHelper getTarget() { |
| return target; |
| } |
| |
| ExpressionHelper getSeparator() { |
| return separator; |
| } |
| |
| ExpressionHelper[] getParameters() { |
| return parameters; |
| } |
| |
| } |