/******************************************************************************* * Copyright (c) 2011, 2015 Willink Transformations 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: * E.D.Willink - initial API and implementation *******************************************************************************/ package org.eclipse.ocl.pivot.internal.ecore; import java.util.List; import org.eclipse.emf.ecore.EOperation; import org.eclipse.jdt.annotation.NonNull; import org.eclipse.jdt.annotation.Nullable; import org.eclipse.ocl.pivot.ExpressionInOCL; import org.eclipse.ocl.pivot.OCLExpression; import org.eclipse.ocl.pivot.Operation; import org.eclipse.ocl.pivot.OperationCallExp; import org.eclipse.ocl.pivot.Variable; import org.eclipse.ocl.pivot.evaluation.EvaluationEnvironment; import org.eclipse.ocl.pivot.evaluation.Executor; import org.eclipse.ocl.pivot.library.AbstractOperation; import org.eclipse.ocl.pivot.utilities.ClassUtil; import org.eclipse.ocl.pivot.utilities.MetamodelManager; import org.eclipse.ocl.pivot.utilities.ParserException; import org.eclipse.ocl.pivot.values.InvalidValueException; /** * An EObjectOperation provides the standard LibraryOperation to implement an EOperation defined by an OCL specification. * The specification is resolved lazily and so an invalid specification may throw an InvalidValueException at run-time. */ public class EObjectOperation extends AbstractOperation { protected final @NonNull Operation operation; protected final @NonNull EOperation eFeature; protected final @NonNull ExpressionInOCL specification; public EObjectOperation(@NonNull Operation operation, @NonNull EOperation eFeature, @NonNull ExpressionInOCL specification) { this.operation = operation; this.eFeature = eFeature; this.specification = specification; } /** * @since 1.1 */ @Override public @Nullable Object dispatch(@NonNull Executor executor, @NonNull OperationCallExp callExp, @Nullable Object sourceValue) { if (specification.getOwnedBody() == null) { try { MetamodelManager metamodelManager = executor.getMetamodelManager(); metamodelManager.parseSpecification(specification); } catch (ParserException e) { throw new InvalidValueException(e, "parse failure", executor.getEvaluationEnvironment(), sourceValue, callExp); } } ExpressionInOCL query = specification; List<? extends OCLExpression> arguments = callExp.getOwnedArguments(); Object[] argumentValues = new Object[arguments.size()]; for (int i = 0; i < arguments.size(); i++) { OCLExpression argument = arguments.get(i); assert argument != null; argumentValues[i] = executor.evaluate(argument); } EvaluationEnvironment nestedEvaluationEnvironment = executor.pushEvaluationEnvironment(query, callExp); nestedEvaluationEnvironment.add(ClassUtil.nonNullModel(query.getOwnedContext()), sourceValue); List<Variable> parameterVariables = query.getOwnedParameters(); int iMax = Math.min(parameterVariables.size(), argumentValues.length); for (int i = 0; i < iMax; i++) { nestedEvaluationEnvironment.add(ClassUtil.nonNullModel(parameterVariables.get(i)), argumentValues[i]); } try { return executor.evaluate(ClassUtil.nonNullPivot(query.getOwnedBody())); } finally { executor.popEvaluationEnvironment(); } } }