package org.jtwig.render.expression.calculator;
import com.google.common.base.Optional;
import com.google.common.base.Supplier;
import org.jtwig.exceptions.CalculationException;
import org.jtwig.functions.FunctionArguments;
import org.jtwig.model.expression.Expression;
import org.jtwig.model.expression.FunctionExpression;
import org.jtwig.model.position.Position;
import org.jtwig.render.RenderRequest;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
import java.util.List;
import static java.util.Arrays.asList;
import static org.hamcrest.Matchers.containsString;
import static org.junit.Assert.assertSame;
import static org.mockito.Matchers.eq;
import static org.mockito.Mockito.*;
public class FunctionExpressionCalculatorTest {
private final FunctionArgumentsFactory argumentsFactory = mock(FunctionArgumentsFactory.class);
private FunctionExpressionCalculator underTest = new FunctionExpressionCalculator(argumentsFactory);
@Rule
public ExpectedException expectedException = ExpectedException.none();
@Test
public void calculateNotFound() throws Exception {
String function = "function";
RenderRequest request = mock(RenderRequest.class, RETURNS_DEEP_STUBS);
FunctionExpression expression = mock(FunctionExpression.class);
Expression argument1 = mock(Expression.class);
Expression argument2 = mock(Expression.class);
Position position = mock(Position.class);
FunctionArguments functionArguments = mock(FunctionArguments.class);
Object argument1Value = new Object();
Object argument2Value = new Object();
List<Expression> expressions = asList(argument1, argument2);
when(argumentsFactory.create(eq(request), eq(expressions))).thenReturn(functionArguments);
when(expression.getPosition()).thenReturn(position);
when(expression.getFunctionIdentifier()).thenReturn(function);
when(expression.getArguments()).thenReturn(expressions);
when(functionArguments.getValues()).thenReturn(asList(argument1Value, argument2Value));
when(request.getEnvironment().getRenderEnvironment().getCalculateExpressionService().calculate(request, argument1)).thenReturn(argument1Value);
when(request.getEnvironment().getRenderEnvironment().getCalculateExpressionService().calculate(request, argument2)).thenReturn(argument2Value);
when(request.getEnvironment().getFunctionResolver().resolve(eq(request), eq(position), eq(function), eq(functionArguments)))
.thenReturn(Optional.<Supplier<Object>>absent());
expectedException.expect(CalculationException.class);
expectedException.expectMessage(containsString(String.format("Unable to resolve function 'function' with arguments [%s, %s]", argument1Value, argument2Value)));
underTest.calculate(request, expression);
}
@Test
public void calculateFound() throws Exception {
String function = "function";
RenderRequest request = mock(RenderRequest.class, RETURNS_DEEP_STUBS);
FunctionExpression expression = mock(FunctionExpression.class);
Expression argument1 = mock(Expression.class);
Expression argument2 = mock(Expression.class);
Supplier<Object> supplier = mock(Supplier.class);
FunctionArguments functionArguments = mock(FunctionArguments.class);
Position position = mock(Position.class);
Object argument1Value = new Object();
Object argument2Value = new Object();
Object expected = new Object();
List<Expression> expressions = asList(argument1, argument2);
when(argumentsFactory.create(eq(request), eq(expressions))).thenReturn(functionArguments);
when(expression.getPosition()).thenReturn(position);
when(expression.getFunctionIdentifier()).thenReturn(function);
when(expression.getArguments()).thenReturn(expressions);
when(functionArguments.getValues()).thenReturn(asList(argument1Value, argument2Value));
when(request.getEnvironment().getRenderEnvironment().getCalculateExpressionService().calculate(request, argument1)).thenReturn(argument1Value);
when(request.getEnvironment().getRenderEnvironment().getCalculateExpressionService().calculate(request, argument2)).thenReturn(argument2Value);
when(request.getEnvironment().getFunctionResolver().resolve(eq(request), eq(position), eq(function), eq(functionArguments)))
.thenReturn(Optional.of(supplier));
when(supplier.get()).thenReturn(expected);
Object result = underTest.calculate(request, expression);
assertSame(expected, result);
}
}