/* * Copyright (c) 2009-2015 * IT-Consulting Stephan Schloepke (http://www.schloepke.de/) * klemm software consulting Mirko Klemm (http://www.klemm-scs.com/) * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ package org.jbasics.math.expression.simple; import org.jbasics.checker.ContractCheck; import org.jbasics.math.MathFunction; import org.jbasics.math.expression.simple.impl.SimpleFunctionCallExpression; import org.jbasics.math.expression.simple.impl.SimpleSymbolExpression; import org.jbasics.pattern.builder.Builder; import org.jbasics.pattern.instance.NamedInstance; import org.jbasics.pattern.strategy.ContextualExecuteStrategy; import org.jbasics.pattern.strategy.ContextualResolveStrategy; import org.jbasics.types.builders.MapBuilder; import org.jbasics.utilities.DataUtilities; import java.math.BigDecimal; import java.math.MathContext; import java.util.Map; public class SimpleExpressionContext implements ContextualResolveStrategy<BigDecimal, SimpleSymbolExpression, SimpleExpressionContext>, ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext> { private final MathContext mathContext; private final Map<String, BigDecimal> symbols; private final Map<String, ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext>> functions; private final ContextualResolveStrategy<BigDecimal, SimpleSymbolExpression, SimpleExpressionContext> additionalSymbolResolver; private final ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext> additionalFunctionExecutor; private final boolean missingSymbolOrFunctionResolvesToNull; private SimpleExpressionContext(final MathContext mathContext, final boolean missingSymbolOrFunctionResolvesToNull, final Map<String, BigDecimal> symbols, final Map<String, MathFunction> mathFunctions, final Map<String, ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext>> functions, final ContextualResolveStrategy<BigDecimal, SimpleSymbolExpression, SimpleExpressionContext> additionalSymbolResolver, final ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext> additionalFunctionExecutor) { this.mathContext = ContractCheck.mustNotBeNull(mathContext, "mathContext"); this.missingSymbolOrFunctionResolvesToNull = missingSymbolOrFunctionResolvesToNull; this.symbols = ContractCheck.mustNotBeNull(symbols, "symbols"); this.functions = ContractCheck.mustNotBeNull(functions, "functions"); this.additionalSymbolResolver = additionalSymbolResolver; this.additionalFunctionExecutor = additionalFunctionExecutor; } public static ContextBuilder newBuilder() { return new ContextBuilder(null); } public MathContext getMathContext() { return this.mathContext; } public BigDecimal execute(final SimpleFunctionCallExpression functionCallExpression) { return execute(functionCallExpression, this); } @Override public BigDecimal execute(final SimpleFunctionCallExpression functionCallExpression, final SimpleExpressionContext context) { final ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext> function = this.functions .get(functionCallExpression.getFunctionName()); if (function != null) { return function.execute(functionCallExpression, context); } else if (this.additionalFunctionExecutor != null) { return this.additionalFunctionExecutor.execute(functionCallExpression, context); } else if (!isMissingSymbolOrFunctionResolvesToNull()) { throw new FunctionNotCalledException("Function could not be executed " + functionCallExpression, functionCallExpression); } else { return null; } } public boolean isMissingSymbolOrFunctionResolvesToNull() { return this.missingSymbolOrFunctionResolvesToNull; } public BigDecimal resolve(final SimpleSymbolExpression symbolExpression) { return resolve(symbolExpression, this); } @Override public BigDecimal resolve(final SimpleSymbolExpression symbolExpression, final SimpleExpressionContext context) { BigDecimal result = this.symbols.get(symbolExpression.getSymbol()); if (result == null && this.additionalSymbolResolver != null) { result = this.additionalSymbolResolver.resolve(symbolExpression, context); } if (result == null && !isMissingSymbolOrFunctionResolvesToNull()) { throw new SymbolNotFoundException("Cannot find symbol " + symbolExpression.getSymbol(), symbolExpression); } return result; } public static class ContextBuilder implements Builder<SimpleExpressionContext> { private final boolean subContextBuilder; private final MapBuilder<String, BigDecimal> symbolsBuilder = new MapBuilder<String, BigDecimal>().immutable(); private final MapBuilder<String, ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext>> functionsBuilder = new MapBuilder<String, ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext>>() .immutable(); private final MapBuilder<String, MathFunction> mathFunctionBuilder = new MapBuilder<String, MathFunction>().immutable(); private MathContext mathContext; private boolean missingSymbolOrFunctionResolvesToNull; private ContextualResolveStrategy<BigDecimal, SimpleSymbolExpression, SimpleExpressionContext> additionalSymbolResolver; private ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext> additionalFunctionExecutor; private ContextBuilder(final SimpleExpressionContext parent) { this.subContextBuilder = parent != null; if (parent != null) { this.mathContext = parent.mathContext; this.additionalFunctionExecutor = parent; this.additionalSymbolResolver = parent; } } public ContextBuilder withMathContext(final MathContext mc) { this.mathContext = mc; return this; } public ContextBuilder withMissingSymbolOrFunctionResolvesToNull() { this.missingSymbolOrFunctionResolvesToNull = true; return this; } public ContextBuilder withMissingSymbolOrFunctionThrowsException() { this.missingSymbolOrFunctionResolvesToNull = false; return this; } public ContextBuilder withAdditionalSymbolResolver( final ContextualResolveStrategy<BigDecimal, SimpleSymbolExpression, SimpleExpressionContext> additionalSymbolResolver) { if (this.subContextBuilder) { throw new IllegalStateException("Sub context builder dosn't allow to set additional symbol resolver"); } this.additionalSymbolResolver = additionalSymbolResolver; return this; } public ContextBuilder withAdditionalFunctionExecutor( final ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext> additionalFunctionExecutor) { if (this.subContextBuilder) { throw new IllegalStateException("Sub context builder dosn't allow to set additional function executor"); } this.additionalFunctionExecutor = additionalFunctionExecutor; return this; } public ContextBuilder withSymbol(final String name, final BigDecimal value) { this.symbolsBuilder.put(name, value); return this; } public ContextBuilder withMathFunction(final MathFunction... functions) { for (final MathFunction function : functions) { withMathFunction(function); } return this; } public ContextBuilder withMathFunction(final MathFunction function) { if (ContractCheck.mustNotBeNull(function, "function") instanceof NamedInstance) { return withMathFunction(((NamedInstance) function).name(), function); } else { return withMathFunction(function.getClass().getSimpleName(), function); } } public ContextBuilder withMathFunction(final String name, final MathFunction function) { this.mathFunctionBuilder.put(name, function); return this; } public ContextBuilder withFunction(final String name, final ContextualExecuteStrategy<BigDecimal, SimpleFunctionCallExpression, SimpleExpressionContext> strategy) { this.functionsBuilder.put(name, strategy); return this; } @Override public void reset() { this.symbolsBuilder.reset(); this.functionsBuilder.reset(); this.mathFunctionBuilder.reset(); } @Override public SimpleExpressionContext build() { return new SimpleExpressionContext(DataUtilities.coalesce(this.mathContext, MathContext.DECIMAL64), // The math context should never be null here this.missingSymbolOrFunctionResolvesToNull, // Either throw exception or return null for expressions with missing functions or symbols this.symbolsBuilder.build(), // The map will be immutable and never null this.mathFunctionBuilder.build(), // The map will be immutable and never null this.functionsBuilder.build(), // The map will be immutable and never null this.additionalSymbolResolver, // additional resolver for symbols on root context (sub context uses parent here) this.additionalFunctionExecutor // additional function executor for functions on root context (sub context uses parent here) ); } } }