/******************************************************************************* * 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.library.executor; import java.util.Collection; import java.util.Collections; import java.util.Map; import java.util.Set; import java.util.regex.Pattern; import org.eclipse.emf.common.util.Diagnostic; import org.eclipse.jdt.annotation.NonNull; import org.eclipse.jdt.annotation.Nullable; import org.eclipse.ocl.pivot.CompleteEnvironment; import org.eclipse.ocl.pivot.NamedElement; import org.eclipse.ocl.pivot.OCLExpression; import org.eclipse.ocl.pivot.StandardLibrary; import org.eclipse.ocl.pivot.Type; import org.eclipse.ocl.pivot.TypedElement; import org.eclipse.ocl.pivot.evaluation.EvaluationEnvironment; import org.eclipse.ocl.pivot.evaluation.EvaluationLogger; import org.eclipse.ocl.pivot.evaluation.Evaluator; import org.eclipse.ocl.pivot.evaluation.Executor; import org.eclipse.ocl.pivot.evaluation.ModelManager; import org.eclipse.ocl.pivot.messages.StatusCodes; import org.eclipse.ocl.pivot.utilities.EnvironmentFactory; import org.eclipse.ocl.pivot.utilities.MetamodelManager; import org.eclipse.ocl.pivot.utilities.Option; import org.eclipse.ocl.pivot.values.CollectionValue; import org.eclipse.ocl.pivot.values.InvalidValueException; import org.eclipse.ocl.pivot.values.Value; public abstract class ExecutorManager implements Executor { // This is the same as HashMap's default initial capacity private static final int DEFAULT_REGEX_CACHE_LIMIT = 16; // this is the same as HashMap's default load factor private static final float DEFAULT_REGEX_CACHE_LOAD_FACTOR = 0.75f; private static final @NonNull EvaluationEnvironment nullEvaluationEnvironment = new EvaluationEnvironment() { @Override public void add(@NonNull TypedElement referredVariable, @Nullable Object value) {} @Override public void clear() {} @SuppressWarnings("null") @Override public @NonNull Map<Option<?>, Object> clearOptions() { return Collections.emptyMap(); } @Override public void dispose() {} @Override public <T> @Nullable T getAdapter(Class<T> adapterType) { return null; } @Override public @NonNull EnvironmentFactory getEnvironmentFactory() { throw new UnsupportedOperationException(); } @Override public @NonNull NamedElement getExecutableObject() { throw new UnsupportedOperationException(); } // @Override // public @NonNull ExecutorInternal getExecutor() { // throw new UnsupportedOperationException(); // } @Override public @NonNull ModelManager getModelManager() { throw new UnsupportedOperationException(); } @Override public Map<Option<?>, Object> getOptions() { return null; } // @Override // public EvaluationEnvironment.@Nullable EvaluationEnvironmentExtension getParent() { // return null; // } @Override public <@Nullable T> T getValue(@NonNull Option<T> option) { return null; } @Override public @Nullable Object getValueOf(@NonNull TypedElement referredVariable) { return null; } @Override public @NonNull Set<TypedElement> getVariables() { return Collections.emptySet(); } @Override public boolean isEnabled(@NonNull Option<@Nullable Boolean> option) { return false; } @Override public <@Nullable T> void putOptions(@NonNull Map<? extends Option<T>, ? extends T> options) { } @Override public @Nullable Object remove(@NonNull TypedElement referredVariable) { return null; } @Override public <@Nullable T> @Nullable T removeOption(@NonNull Option<T> option) { return null; } @SuppressWarnings("null") @Override public <@Nullable T> @NonNull Map<Option<T>, T> removeOptions(@NonNull Collection<Option<T>> options) { return Collections.emptyMap(); } @Override public void replace(@NonNull TypedElement referredVariable, @Nullable Object value) {} @Override public <T> void setOption(@NonNull Option<T> option, @Nullable T value) { } }; protected final @NonNull CompleteEnvironment environment; protected final @NonNull StandardLibrary standardLibrary; /** * Set true by {@link #setCanceled} to terminate execution at next call to {@link #getValuefactory()}. */ private boolean isCanceled = false; /** * Lazily-created cache of reusable regex patterns to avoid * repeatedly parsing the same regexes. */ private /*@LazyNonNull*/ Map<String, Pattern> regexPatterns = null; public ExecutorManager(@NonNull CompleteEnvironment environment) { this.environment = environment; this.standardLibrary = environment.getOwnedStandardLibrary(); } /** * @since 1.1 */ @Override public void add(@NonNull TypedElement referredVariable, @Nullable Object value) {} /** @deprecated Evaluator no longer nests * @since 1.1*/ @Deprecated @Override public @NonNull Evaluator createNestedEvaluator() { return this; } /** * Creates (on demand) the regular-expression matcher cache. The default * implementation creates an access-ordered LRU cache with a limit of 16 * entries. Subclasses may override to create a map with whatever different * performance characteristics may be required. * * @return the new regular-expression matcher cache * * @see #getRegexPattern(String) */ protected @NonNull Map<String, Pattern> createRegexCache() { return new java.util.LinkedHashMap<String, Pattern>( DEFAULT_REGEX_CACHE_LIMIT, DEFAULT_REGEX_CACHE_LOAD_FACTOR, true) { private static final long serialVersionUID = 1L; @Override protected boolean removeEldestEntry( Map.Entry<String, Pattern> eldest) { return size() > DEFAULT_REGEX_CACHE_LIMIT; } }; } @Override public void dispose() {} @Override public @NonNull Value evaluate(@NonNull OCLExpression body) { throw new UnsupportedOperationException(); // try { // return ((LibraryUnaryOperation)body).evaluate(this, null, null); // WIP // } catch (InvalidValueException e) { // return throwInvalidEvaluation(e); // } } public Value evaluateIteration(Type returnType, CollectionValue sourceVal, TypedElement accumulator, OCLExpression body, TypedElement[] iterators) { // TODO Auto-generated method stub throw new UnsupportedOperationException(); } @Override public @NonNull CompleteEnvironment getCompleteEnvironment() { return environment; } @Override public int getDiagnosticSeverity(int severityPreference, @Nullable Object resultValue) { if (resultValue == null) { return Diagnostic.ERROR; } else if (resultValue instanceof InvalidValueException) { return Diagnostic.CANCEL; } else { return severityPreference; } } public @NonNull Type getDynamicTypeOf(@Nullable Object value) { return getIdResolver().getDynamicTypeOf(value); } /** * @since 1.1 */ @Override public @NonNull EnvironmentFactory getEnvironmentFactory() { throw new UnsupportedOperationException(); } @Override public @NonNull EvaluationEnvironment getEvaluationEnvironment() { throw new UnsupportedOperationException(); } // @Override // public @NonNull Executor getExecutor() { // return this; // } // public @NonNull IdResolver getIdResolver() { // return standardLibrary.getIdResolver(); // } @Override public @Nullable EvaluationLogger getLogger() { return null; } /** * @since 1.1 */ @Override public @NonNull MetamodelManager getMetamodelManager() { throw new UnsupportedOperationException(); } /** * Return a cached matcher for a give regular expression. */ @Override public @NonNull Pattern getRegexPattern(@NonNull String regex) { if (regexPatterns == null) { synchronized (this) { if (regexPatterns == null) { regexPatterns = createRegexCache(); } } } synchronized (regexPatterns) { Pattern pattern = regexPatterns.get(regex); if (pattern == null) { // System.out.println("Compile " + regex); pattern = Pattern.compile(regex); assert pattern != null; regexPatterns.put(regex, pattern); } // else { // System.out.println("Re-use " + regex); // } return pattern; } } // @Override // public @NonNull EvaluationEnvironment getRootEvaluationEnvironment() { // return nullEvaluationEnvironment; // } @Override public int getSeverity(@Nullable Object validationKey) { return StatusCodes.WARNING; } @Override public @NonNull StandardLibrary getStandardLibrary() { return standardLibrary; } @Override public org.eclipse.ocl.pivot.@NonNull Class getStaticTypeOf(@Nullable Object value) { return getIdResolver().getStaticTypeOf(value); } @Override public org.eclipse.ocl.pivot.@NonNull Class getStaticTypeOf(@Nullable Object value, @NonNull Object... values) { return getIdResolver().getStaticTypeOf(value, values); } @Override public org.eclipse.ocl.pivot.@NonNull Class getStaticTypeOf(@Nullable Object value, @NonNull Iterable<?> values) { return getIdResolver().getStaticTypeOf(value, values); } // public @NonNull ValueFactory getValueFactory() { // if (isCanceled) { // throw new EvaluationHaltedException("Canceled"); //$NON-NLS-1$ // } // return valueFactory; // } // @Override // public @Nullable Object getValueOf(@NonNull TypedElement referredVariable) { // return null; // } @Override public boolean isCanceled() { return isCanceled; } @Override public void setCanceled(boolean isCanceled) { this.isCanceled = isCanceled; } @Override public void setLogger(@Nullable EvaluationLogger logger) { /* ignored */; } /** * @since 1.1 */ @Override public void popEvaluationEnvironment() {} /** * @since 1.1 */ @Override public @NonNull EvaluationEnvironment pushEvaluationEnvironment(@NonNull NamedElement executableObject, @Nullable OCLExpression callingObject) { return nullEvaluationEnvironment; } /** * @since 1.1 */ @Override public void replace(@NonNull TypedElement referredVariable, @Nullable Object value) {} /* public DomainType typeOf(Value value, Value... values) { DomainStandardLibrary standardLibrary = valueFactory.getStandardLibrary(); DomainType type = value.getType(standardLibrary); for (Value anotherValue : values) { DomainType anotherType = anotherValue.getType(standardLibrary); type = type.getCommonType(standardLibrary, anotherType); } return type; } */ // public @NonNull NullValue throwInvalidEvaluation(InvalidValueException e) { // throw new InvalidEvaluationException(null, e); // } // public @NonNull NullValue throwInvalidEvaluation(Throwable e, DomainExpression expression, Object context, // String message, Object... bindings) { // throw new InvalidEvaluationException(null, NLS.bind(message, bindings), e, expression, context); // } }