742 lines
37 KiB
Java
742 lines
37 KiB
Java
package jmri.jmrit.logixng.expressions;
|
|
|
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
|
import static org.junit.jupiter.api.Assertions.assertFalse;
|
|
import static org.junit.jupiter.api.Assertions.assertInstanceOf;
|
|
import static org.junit.jupiter.api.Assertions.assertNotNull;
|
|
import static org.junit.jupiter.api.Assertions.assertNull;
|
|
import static org.junit.jupiter.api.Assertions.assertSame;
|
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
|
import static org.junit.jupiter.api.Assertions.assertTrue;
|
|
|
|
import java.beans.PropertyChangeEvent;
|
|
import java.lang.reflect.InvocationTargetException;
|
|
import java.util.*;
|
|
import java.util.concurrent.atomic.AtomicBoolean;
|
|
|
|
import jmri.*;
|
|
import jmri.jmrit.logixng.*;
|
|
import jmri.jmrit.logixng.actions.AnalogActionMemory;
|
|
import jmri.jmrit.logixng.actions.DoAnalogAction;
|
|
import jmri.jmrit.logixng.implementation.DefaultConditionalNGScaffold;
|
|
import jmri.jmrit.logixng.util.parser.ParserException;
|
|
import jmri.util.JUnitUtil;
|
|
|
|
import org.junit.jupiter.api.AfterEach;
|
|
import org.junit.jupiter.api.BeforeEach;
|
|
import org.junit.jupiter.api.Test;
|
|
|
|
/**
|
|
* Test AnalogFormula
|
|
*
|
|
* @author Daniel Bergqvist 2018
|
|
*/
|
|
public class AnalogFormulaTest extends AbstractAnalogExpressionTestBase {
|
|
|
|
private LogixNG logixNG;
|
|
private ConditionalNG conditionalNG;
|
|
private AnalogFormula expressionFormula;
|
|
private AnalogActionMemory analogActionMemory;
|
|
private Memory analogMemory;
|
|
|
|
|
|
@Override
|
|
public ConditionalNG getConditionalNG() {
|
|
return conditionalNG;
|
|
}
|
|
|
|
@Override
|
|
public LogixNG getLogixNG() {
|
|
return logixNG;
|
|
}
|
|
|
|
private int beanID = 901;
|
|
|
|
@Override
|
|
public MaleSocket getConnectableChild() {
|
|
AnalogExpressionBean childExpression = new AnalogExpressionConstant("IQAE"+Integer.toString(beanID), null);
|
|
beanID++;
|
|
MaleSocket maleSocketChild =
|
|
InstanceManager.getDefault(AnalogExpressionManager.class).registerExpression(childExpression);
|
|
return maleSocketChild;
|
|
}
|
|
|
|
@Override
|
|
public String getExpectedPrintedTree() {
|
|
return String.format(
|
|
"Analog Formula: E1 ::: Use default%n" +
|
|
" ?* E1%n" +
|
|
" Get analog constant 1 ::: Use default%n" +
|
|
" ?* E2%n" +
|
|
" Socket not connected%n");
|
|
}
|
|
|
|
@Override
|
|
public String getExpectedPrintedTreeFromRoot() {
|
|
return String.format(
|
|
"LogixNG: A new logix for test%n" +
|
|
" ConditionalNG: A conditionalNG%n" +
|
|
" ! A%n" +
|
|
" Read analog E and set analog A ::: Use default%n" +
|
|
" ?~ E%n" +
|
|
" Analog Formula: E1 ::: Use default%n" +
|
|
" ?* E1%n" +
|
|
" Get analog constant 1 ::: Use default%n" +
|
|
" ?* E2%n" +
|
|
" Socket not connected%n" +
|
|
" !~ A%n" +
|
|
" Set memory IM2 ::: Use default%n");
|
|
}
|
|
|
|
@Override
|
|
public NamedBean createNewBean(String systemName) {
|
|
AnalogFormula a = new AnalogFormula(systemName, null);
|
|
// a.setFormula("R1");
|
|
return a;
|
|
}
|
|
|
|
@Override
|
|
public boolean addNewSocket() throws SocketAlreadyConnectedException {
|
|
int count = _base.getChildCount();
|
|
for (int i=0; i < count; i++) {
|
|
if (!_base.getChild(i).isConnected()) {
|
|
_base.getChild(i).connect(getConnectableChild());
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@Test
|
|
public void testCtor() {
|
|
AnalogFormula expression2;
|
|
|
|
expression2 = new AnalogFormula("IQAE321", null);
|
|
// expression2.setFormula("R1");
|
|
assertNotNull( expression2, "object exists");
|
|
assertNull( expression2.getUserName(), "Username matches");
|
|
// Assert.assertEquals("String matches", "Formula: R1", expression2.getLongDescription());
|
|
|
|
expression2 = new AnalogFormula("IQAE321", "My expression");
|
|
// expression2.setFormula("R1");
|
|
assertNotNull( expression2, "object exists");
|
|
assertEquals( "My expression", expression2.getUserName(), "Username matches");
|
|
// Assert.assertEquals("String matches", "Formula: R1", expression2.getLongDescription());
|
|
|
|
expression2 = new AnalogFormula("IQAE321", null);
|
|
// expression2.setFormula("R1 and R2");
|
|
assertNotNull( expression2, "object exists");
|
|
assertNull( expression2.getUserName(), "Username matches");
|
|
// Assert.assertEquals("String matches", "Formula: R1 and R2", expression2.getLongDescription());
|
|
|
|
expression2 = new AnalogFormula("IQAE321", "My expression");
|
|
// expression2.setFormula("R1 or R2");
|
|
assertNotNull( expression2, "object exists");
|
|
assertEquals( "My expression", expression2.getUserName(), "Username matches");
|
|
// Assert.assertEquals("String matches", "Formula: R1 or R2", expression2.getLongDescription());
|
|
|
|
IllegalArgumentException ex = assertThrows( IllegalArgumentException.class,
|
|
() -> {
|
|
AnalogFormula af = new AnalogFormula("IQE55:12:XY11", null);
|
|
assertNull(af, "should not have got here");
|
|
}, "Expected Illegal system name exception thrown");
|
|
assertNotNull(ex);
|
|
|
|
|
|
ex = assertThrows( IllegalArgumentException.class,
|
|
() -> {
|
|
AnalogFormula af = new AnalogFormula("IQE55:12:XY11", "A name");
|
|
assertNull(af, "should not have got here");
|
|
}, "Expected Illegal system name exception thrown");
|
|
assertNotNull(ex);
|
|
|
|
}
|
|
|
|
// Test action when at least one child socket is not connected
|
|
@Test
|
|
public void testCtorAndSetup1() {
|
|
AnalogExpressionManager m = InstanceManager.getDefault(AnalogExpressionManager.class);
|
|
String managerName = m.getClass().getName();
|
|
|
|
List<MaleSocket> maleSockets = new ArrayList<>();
|
|
maleSockets.add(m.registerExpression(new AnalogExpressionMemory("IQAE52", null)));
|
|
maleSockets.add(null); // This is null by purpose
|
|
maleSockets.add(m.registerExpression(new AnalogExpressionMemory("IQAE554", null)));
|
|
maleSockets.add(null); // This is null by purpose
|
|
maleSockets.add(m.registerExpression(new AnalogExpressionMemory("IQAE3", null)));
|
|
|
|
List<AnalogFormula.SocketData> actionSystemNames = new ArrayList<>();
|
|
actionSystemNames.add(new AnalogFormula.SocketData("XYZ123", "IQAE52", managerName));
|
|
actionSystemNames.add(new AnalogFormula.SocketData("ZH12", null, managerName)); // This is null by purpose
|
|
actionSystemNames.add(new AnalogFormula.SocketData("Hello", "IQAE554", managerName));
|
|
// IQAE61232 doesn't exist by purpose
|
|
actionSystemNames.add(new AnalogFormula.SocketData("SomethingElse", "IQAE61232", managerName));
|
|
actionSystemNames.add(new AnalogFormula.SocketData("Yes123", "IQAE3", managerName));
|
|
|
|
AnalogFormula expression = new AnalogFormula("IQAE321", null, actionSystemNames);
|
|
assertNotNull( expression, "exists");
|
|
assertEquals( 5, expression.getChildCount(), "expression has 5 female sockets");
|
|
|
|
for (int i=0; i < 5; i++) {
|
|
AnalogFormula.SocketData socketData = actionSystemNames.get(i);
|
|
assertEquals( socketData._socketName, expression.getChild(i).getName(),
|
|
() -> "expression female socket name is "+socketData._socketName);
|
|
assertEquals( "jmri.jmrit.logixng.implementation.DefaultFemaleGenericExpressionSocket",
|
|
expression.getChild(i).getClass().getName(),
|
|
"expression female socket is of correct class");
|
|
assertFalse( expression.getChild(i).isConnected(),
|
|
"expression female socket is not connected");
|
|
}
|
|
|
|
// Setup action. This connects the child actions to this action
|
|
expression.setup();
|
|
|
|
jmri.util.JUnitAppender.assertMessage("cannot load analog expression IQAE61232");
|
|
|
|
for (int i=0; i < 5; i++) {
|
|
AnalogFormula.SocketData socketName = actionSystemNames.get(i);
|
|
assertEquals( socketName._socketName, expression.getChild(i).getName(),
|
|
"expression female socket name is "+socketName._socketName);
|
|
|
|
if (maleSockets.get(i) != null) {
|
|
assertTrue( expression.getChild(i).isConnected(),
|
|
"expression female socket is connected");
|
|
// assertEquals(
|
|
// maleSockets.get(i),
|
|
// expression.getChild(i).getConnectedSocket(), "child is correct bean");
|
|
} else {
|
|
assertFalse( expression.getChild(i).isConnected(),
|
|
"expression female socket is not connected");
|
|
}
|
|
}
|
|
|
|
assertEquals( 5, expression.getChildCount(), "expression has 5 female sockets");
|
|
}
|
|
|
|
@Test
|
|
public void testCtorAndSetup2() {
|
|
AnalogExpressionManager m = InstanceManager.getDefault(AnalogExpressionManager.class);
|
|
String managerName = m.getClass().getName();
|
|
|
|
List<MaleSocket> maleSockets = new ArrayList<>();
|
|
maleSockets.add(m.registerExpression(new AnalogExpressionMemory("IQAE52", null)));
|
|
maleSockets.add(m.registerExpression(new AnalogExpressionMemory("IQAE99", null)));
|
|
maleSockets.add(m.registerExpression(new AnalogExpressionMemory("IQAE554", null)));
|
|
maleSockets.add(m.registerExpression(new AnalogExpressionMemory("IQAE61232", null)));
|
|
maleSockets.add(m.registerExpression(new AnalogExpressionMemory("IQAE3", null)));
|
|
|
|
List<AnalogFormula.SocketData> actionSystemNames = new ArrayList<>();
|
|
actionSystemNames.add(new AnalogFormula.SocketData("XYZ123", "IQAE52", managerName));
|
|
actionSystemNames.add(new AnalogFormula.SocketData("ZH12", "IQAE99", managerName));
|
|
actionSystemNames.add(new AnalogFormula.SocketData("Hello", "IQAE554", managerName));
|
|
actionSystemNames.add(new AnalogFormula.SocketData("SomethingElse", "IQAE61232", managerName));
|
|
actionSystemNames.add(new AnalogFormula.SocketData("Yes123", "IQAE3", managerName));
|
|
|
|
AnalogFormula expression = new AnalogFormula("IQAE321", null, actionSystemNames);
|
|
assertNotNull( expression, "exists");
|
|
assertEquals( 5, expression.getChildCount(), "expression has 5 female sockets");
|
|
|
|
for (int i=0; i < 5; i++) {
|
|
AnalogFormula.SocketData socketData = actionSystemNames.get(i);
|
|
assertEquals( socketData._socketName, expression.getChild(i).getName(),
|
|
"expression female socket name is "+socketData._socketName);
|
|
assertEquals( "jmri.jmrit.logixng.implementation.DefaultFemaleGenericExpressionSocket",
|
|
expression.getChild(i).getClass().getName(),
|
|
"expression female socket is of correct class");
|
|
assertFalse( expression.getChild(i).isConnected(),
|
|
"expression female socket is not connected");
|
|
}
|
|
|
|
// Setup action. This connects the child actions to this action
|
|
expression.setup();
|
|
|
|
for (int i=0; i < 5; i++) {
|
|
AnalogFormula.SocketData socketData = actionSystemNames.get(i);
|
|
assertEquals( socketData._socketName, expression.getChild(i).getName(),
|
|
() -> "expression female socket name is "+socketData._socketName);
|
|
|
|
if (maleSockets.get(i) != null) {
|
|
assertTrue( expression.getChild(i).isConnected(),
|
|
"expression female socket is connected");
|
|
// assertEquals(
|
|
// maleSockets.get(i),
|
|
// expression.getChild(i).getConnectedSocket(), "child is correct bean");
|
|
} else {
|
|
assertFalse( expression.getChild(i).isConnected(),
|
|
"expression female socket is not connected");
|
|
}
|
|
}
|
|
|
|
// Since all the sockets are connected, a new socket must have been created.
|
|
assertEquals( 6, expression.getChildCount(), "expression has 6 female sockets");
|
|
|
|
// Try run setup() again. That should not cause any problems.
|
|
expression.setup();
|
|
|
|
assertEquals( 6, expression.getChildCount(), "expression has 6 female sockets");
|
|
}
|
|
|
|
// Test calling setActionSystemNames() twice
|
|
@Test
|
|
public void testCtorAndSetup3() throws NoSuchMethodException, IllegalAccessException, IllegalArgumentException {
|
|
List<AnalogFormula.SocketData> actionSystemNames = new ArrayList<>();
|
|
actionSystemNames.add(new AnalogFormula.SocketData("XYZ123", "IQAE52", ""));
|
|
|
|
AnalogFormula expression = new AnalogFormula("IQAE321", null, actionSystemNames);
|
|
|
|
java.lang.reflect.Method method =
|
|
expression.getClass().getDeclaredMethod("setExpressionSystemNames", new Class<?>[]{List.class});
|
|
method.setAccessible(true);
|
|
|
|
InvocationTargetException e = assertThrows( InvocationTargetException.class,
|
|
() -> method.invoke(expression, new Object[]{null}),
|
|
"Exception thrown");
|
|
RuntimeException ex = assertInstanceOf( RuntimeException.class, e.getCause());
|
|
assertEquals( "expression system names cannot be set more than once",
|
|
ex.getMessage(), "Exception message is correct");
|
|
}
|
|
|
|
@Test
|
|
public void testSetChildCount() throws SocketAlreadyConnectedException {
|
|
AnalogFormula a = (AnalogFormula)_base;
|
|
AtomicBoolean ab = new AtomicBoolean(false);
|
|
|
|
_base.addPropertyChangeListener((PropertyChangeEvent evt) -> {
|
|
ab.set(true);
|
|
});
|
|
|
|
a.setChildCount(1);
|
|
assertEquals( 1, a.getChildCount(), "numChilds are correct");
|
|
|
|
// Test increase num children
|
|
ab.set(false);
|
|
a.setChildCount(a.getChildCount()+1);
|
|
assertEquals( 2, a.getChildCount(), "numChilds are correct");
|
|
assertTrue( ab.get(), "PropertyChangeEvent fired");
|
|
|
|
// Test decrease num children
|
|
ab.set(false);
|
|
assertTrue( a.getChildCount() > 1, "We have least two children");
|
|
a.setChildCount(1);
|
|
assertEquals( 1, a.getChildCount(), "numChilds are correct");
|
|
assertTrue( ab.get(), "PropertyChangeEvent fired");
|
|
|
|
// Test decrease num children when all children are connected
|
|
ab.set(false);
|
|
a.getChild(0).disconnect();
|
|
a.getChild(0).connect(getConnectableChild());
|
|
a.getChild(1).disconnect();
|
|
a.getChild(1).connect(getConnectableChild());
|
|
a.getChild(2).disconnect();
|
|
a.getChild(2).connect(getConnectableChild());
|
|
assertEquals( 4, a.getChildCount(), "numChilds are correct");
|
|
a.setChildCount(2);
|
|
assertEquals( 2, a.getChildCount(), "numChilds are correct");
|
|
assertTrue( ab.get(), "PropertyChangeEvent fired");
|
|
|
|
// Test setChildCount to same number of children as before
|
|
ab.set(false);
|
|
a.setChildCount(2);
|
|
assertEquals( 2, a.getChildCount(), "numChilds are correct");
|
|
assertFalse( ab.get(), "PropertyChangeEvent not fired");
|
|
}
|
|
|
|
@Test
|
|
public void testFormula() throws ParserException, SocketAlreadyConnectedException {
|
|
AnalogFormula a = (AnalogFormula)_base;
|
|
|
|
a.getChild(0).disconnect();
|
|
a.getChild(0).connect(getConnectableChild());
|
|
a.getChild(0).setName("Abc");
|
|
a.getChild(1).disconnect();
|
|
a.getChild(1).connect(getConnectableChild());
|
|
a.getChild(1).setName("Xyz");
|
|
|
|
a.setFormula("Xyz + Abc");
|
|
assertEquals( "Xyz + Abc", a.getFormula(), "Formula is correct");
|
|
|
|
a.setFormula("Abc - Xyz");
|
|
assertEquals( "Abc - Xyz", a.getFormula(), "Formula is correct");
|
|
}
|
|
|
|
@Test
|
|
public void testGetChild() throws JmriException {
|
|
AnalogFormula expression2 = new AnalogFormula("IQAE321", null);
|
|
// expression2.setFormula("R1");
|
|
|
|
for (int i=0; i < 3; i++) {
|
|
assertEquals( i+1, expression2.getChildCount(),
|
|
"getChildCount() returns "+(i+1));
|
|
|
|
assertNotNull( expression2.getChild(0),
|
|
"getChild(0) returns a non null value");
|
|
|
|
assertIndexOutOfBoundsException(expression2::getChild, i+1, i+1);
|
|
|
|
// Connect a new child expression
|
|
AnalogExpressionConstant expr = new AnalogExpressionConstant("IQAE"+i, null);
|
|
MaleSocket maleSocket =
|
|
InstanceManager.getDefault(AnalogExpressionManager.class).registerExpression(expr);
|
|
expression2.getChild(i).connect(maleSocket);
|
|
}
|
|
}
|
|
|
|
@Test
|
|
public void testCategory() {
|
|
assertSame( LogixNG_Category.COMMON, _base.getCategory(), "Category matches");
|
|
}
|
|
|
|
// Test the methods connected(FemaleSocket) and getExpressionSystemName(int)
|
|
@Test
|
|
public void testConnected_getExpressionSystemName() throws SocketAlreadyConnectedException {
|
|
AnalogFormula expression = new AnalogFormula("IQAE121", null);
|
|
|
|
AnalogExpressionMemory analogExpressionMemory = new AnalogExpressionMemory("IQAE122", null);
|
|
MaleSocket maleSAMSocket =
|
|
InstanceManager.getDefault(AnalogExpressionManager.class).registerExpression(analogExpressionMemory);
|
|
|
|
assertEquals( 1, expression.getChildCount(), "Num children is correct");
|
|
|
|
// Test connect and disconnect
|
|
expression.getChild(0).connect(maleSAMSocket);
|
|
assertEquals( 2, expression.getChildCount(), "Num children is correct");
|
|
assertEquals( "IQAE122", expression.getExpressionSystemName(0), "getExpressionSystemName(0) is correct");
|
|
assertNull( expression.getExpressionSystemName(1), "getExpressionSystemName(1) is null");
|
|
expression.getChild(0).disconnect();
|
|
assertEquals( 2, expression.getChildCount(), "Num children is correct");
|
|
assertNull( expression.getExpressionSystemName(0), "getExpressionSystemName(0) is null");
|
|
assertNull( expression.getExpressionSystemName(1), "getExpressionSystemName(1) is null");
|
|
|
|
expression.getChild(1).connect(maleSAMSocket);
|
|
assertEquals( 2, expression.getChildCount(), "Num children is correct");
|
|
assertNull( expression.getExpressionSystemName(0), "getExpressionSystemName(0) is null");
|
|
assertEquals( expression.getExpressionSystemName(1), "IQAE122", "getExpressionSystemName(1) is correct");
|
|
expression.getChild(0).disconnect(); // Test removing child with the wrong index.
|
|
assertEquals( 2, expression.getChildCount(), "Num children is correct");
|
|
assertNull( expression.getExpressionSystemName(0), "getExpressionSystemName(0) is null");
|
|
assertEquals( "IQAE122", expression.getExpressionSystemName(1), "getExpressionSystemName(1) is correct");
|
|
expression.getChild(1).disconnect();
|
|
assertEquals( 2, expression.getChildCount(), "Num children is correct");
|
|
assertNull( expression.getExpressionSystemName(0), "getExpressionSystemName(0) is null");
|
|
assertNull( expression.getExpressionSystemName(1), "getExpressionSystemName(1) is null");
|
|
}
|
|
|
|
@Test
|
|
public void testDescription() {
|
|
AnalogFormula expression = new AnalogFormula("IQAE321", null);
|
|
assertEquals( "Analog Formula", expression.getShortDescription(), "strings matches");
|
|
assertEquals( "Analog Formula: empty", expression.getLongDescription(), "strings matches");
|
|
}
|
|
|
|
@Test
|
|
public void testEvaluateEmptyFormula() throws ParserException, JmriException {
|
|
AnalogFormula expression = new AnalogFormula("IQAE321", null);
|
|
expression.setFormula("");
|
|
assertEquals( 0.0, expression.evaluate(), "Empty formula returns 0.0");
|
|
}
|
|
/*
|
|
private void testValidate(boolean expectedResult, String formula, List<AnalogExpressionBean> conditionalVariablesList) throws Exception {
|
|
Formula ix1 = new Formula("IQDE321", "IXIC 1");
|
|
ix1.setFormula("R1");
|
|
|
|
int count = 0;
|
|
List<ExpressionEntry> expressionEntryList = new ArrayList<>();
|
|
for (AnalogExpressionBean expressionFormula : conditionalVariablesList) {
|
|
String socketName = "E"+Integer.toString(count++);
|
|
FemaleAnalogExpressionSocket socket =
|
|
InstanceManager.getDefault(AnalogExpressionManager.class)
|
|
.createFemaleSocket(null, this, socketName);
|
|
socket.connect((MaleSocket) expressionFormula);
|
|
expressionEntryList.add(new ExpressionEntry(socket, socketName));
|
|
}
|
|
|
|
if (expectedResult) {
|
|
Assert.assertTrue("validateFormula() returns null for '"+formula+"'",
|
|
ix1.validateFormula(formula, expressionEntryList) == null);
|
|
} else {
|
|
Assert.assertTrue("validateFormula() returns error message for '"+formula+"'",
|
|
ix1.validateFormula(formula, expressionEntryList) != null);
|
|
}
|
|
}
|
|
|
|
private void testCalculate(int expectedResult, String formula,
|
|
List<AnalogExpressionBean> conditionalVariablesList, String errorMessage)
|
|
throws Exception {
|
|
|
|
Formula ix1 = new Formula("IQDE321", "IXIC 1");
|
|
ix1.setFormula(formula);
|
|
|
|
// for (int i=0; i < ix1.getChildCount(); i++) {
|
|
// ix1.getChild(i).disconnect();
|
|
// }
|
|
|
|
ix1.setChildCount(conditionalVariablesList.size());
|
|
|
|
for (int i=0; i < conditionalVariablesList.size(); i++) {
|
|
ix1.getChild(i).connect((MaleSocket)conditionalVariablesList.get(i));
|
|
}
|
|
|
|
switch (expectedResult) {
|
|
case Formula.FALSE:
|
|
Assert.assertFalse("validateFormula() returns FALSE for '"+formula+"'",
|
|
ix1.evaluate());
|
|
break;
|
|
|
|
case Formula.TRUE:
|
|
// System.err.format("formula: %s%n", formula);
|
|
// System.err.format("variable: %b%n", conditionalVariablesList.get(0).evaluate(isCompleted));
|
|
Assert.assertTrue("validateFormula() returns TRUE for '"+formula+"'",
|
|
ix1.evaluate());
|
|
break;
|
|
|
|
default:
|
|
throw new RuntimeException(String.format("Unknown expected result: %d", expectedResult));
|
|
}
|
|
|
|
if (! errorMessage.isEmpty()) {
|
|
jmri.util.JUnitAppender.assertErrorMessageStartsWith(errorMessage);
|
|
}
|
|
}
|
|
|
|
@Test
|
|
public void testValidate() throws Exception {
|
|
AnalogExpressionBean[] conditionalVariables_Empty = { };
|
|
List<AnalogExpressionBean> conditionalVariablesList_Empty = Arrays.asList(conditionalVariables_Empty);
|
|
|
|
AnalogExpressionBean trueExpression =
|
|
InstanceManager.getDefault(
|
|
AnalogExpressionManager.class).registerExpression(
|
|
new True(InstanceManager.getDefault(AnalogExpressionManager.class).getAutoSystemName(), null));
|
|
// AnalogExpressionBean falseExpression = InstanceManager.getDefault(AnalogExpressionManager.class).registerExpression(new False(conditionalNG));
|
|
|
|
AnalogExpressionBean[] conditionalVariables_True
|
|
= { trueExpression };
|
|
List<AnalogExpressionBean> conditionalVariablesList_True = Arrays.asList(conditionalVariables_True);
|
|
|
|
AnalogExpressionBean[] conditionalVariables_TrueTrueTrue
|
|
= { trueExpression
|
|
, trueExpression
|
|
, trueExpression };
|
|
List<AnalogExpressionBean> conditionalVariablesList_TrueTrueTrue = Arrays.asList(conditionalVariables_TrueTrueTrue);
|
|
|
|
// Test empty formula string
|
|
testValidate(EXPECT_FAILURE, "", conditionalVariablesList_Empty);
|
|
|
|
testValidate(EXPECT_SUCCESS, "R1", conditionalVariablesList_True);
|
|
testValidate(EXPECT_FAILURE, "R2", conditionalVariablesList_True);
|
|
|
|
// Test parentheses
|
|
testValidate(EXPECT_SUCCESS, "([{R1)}]", conditionalVariablesList_True);
|
|
testValidate(EXPECT_FAILURE, "(R2", conditionalVariablesList_True);
|
|
testValidate(EXPECT_FAILURE, "R2)", conditionalVariablesList_True);
|
|
|
|
// Test several items
|
|
testValidate(EXPECT_FAILURE, "R1 and R2 and R3", conditionalVariablesList_True);
|
|
testValidate(EXPECT_FAILURE, "R1", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_SUCCESS, "R1 and R2 and R3", conditionalVariablesList_TrueTrueTrue);
|
|
|
|
// Test uppercase and lowercase
|
|
testValidate(EXPECT_SUCCESS, "R2 AND R1 or R3", conditionalVariablesList_TrueTrueTrue);
|
|
|
|
// Test several items and parenthese
|
|
testValidate(EXPECT_SUCCESS, "(R1 and R3) and not R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "(R1 and) R3 and not R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "R1( and R3) and not R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "R1 (and R3 and) not R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "(R1 and R3) and not R2)", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_SUCCESS, "(R1 and (R3) and not R2)", conditionalVariablesList_TrueTrueTrue);
|
|
|
|
// Test invalid combinations
|
|
testValidate(EXPECT_FAILURE, "R1 and or R3 and R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "R1 or or R3 and R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "R1 or and R3 and R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "R1 not R3 and R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "and R1 not R3 and R2", conditionalVariablesList_TrueTrueTrue);
|
|
testValidate(EXPECT_FAILURE, "R1 or R3 and R2 or", conditionalVariablesList_TrueTrueTrue);
|
|
}
|
|
|
|
@Test
|
|
public void testCalculate() throws Exception {
|
|
AnalogExpressionBean[] conditionalVariables_Empty = { };
|
|
List<AnalogExpressionBean> conditionalVariablesList_Empty = Arrays.asList(conditionalVariables_Empty);
|
|
|
|
AnalogExpressionBean trueExpression =
|
|
InstanceManager.getDefault(
|
|
AnalogExpressionManager.class).registerExpression(
|
|
new True(InstanceManager.getDefault(AnalogExpressionManager.class).getAutoSystemName(), null));
|
|
AnalogExpressionBean falseExpression =
|
|
InstanceManager.getDefault(
|
|
AnalogExpressionManager.class).registerExpression(
|
|
new False(InstanceManager.getDefault(AnalogExpressionManager.class).getAutoSystemName(), null));
|
|
|
|
AnalogExpressionBean[] conditionalVariables_True
|
|
= { trueExpression };
|
|
List<AnalogExpressionBean> conditionalVariablesList_True = Arrays.asList(conditionalVariables_True);
|
|
|
|
AnalogExpressionBean[] conditionalVariables_False
|
|
= { falseExpression };
|
|
List<AnalogExpressionBean> conditionalVariablesList_False = Arrays.asList(conditionalVariables_False);
|
|
|
|
AnalogExpressionBean[] conditionalVariables_TrueTrueTrue
|
|
= { trueExpression
|
|
, trueExpression
|
|
, trueExpression };
|
|
List<AnalogExpressionBean> conditionalVariablesList_TrueTrueTrue = Arrays.asList(conditionalVariables_TrueTrueTrue);
|
|
|
|
// Test with two digit variable numbers
|
|
AnalogExpressionBean[] conditionalVariables_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse
|
|
= {trueExpression
|
|
, trueExpression
|
|
, falseExpression
|
|
, trueExpression
|
|
, trueExpression
|
|
, falseExpression
|
|
, trueExpression
|
|
, trueExpression
|
|
, falseExpression
|
|
, trueExpression
|
|
, trueExpression
|
|
, falseExpression };
|
|
List<AnalogExpressionBean> conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse =
|
|
Arrays.asList(conditionalVariables_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse);
|
|
|
|
|
|
// Test empty formula string
|
|
testCalculate(Formula.FALSE, "", conditionalVariablesList_Empty, "");
|
|
// testCalculate(Formula.FALSE, "", conditionalVariablesList_True,
|
|
// "IXIC 1 parseCalculation error formula= , ex= java.lang.StringIndexOutOfBoundsException");
|
|
testCalculate(Formula.FALSE, "", conditionalVariablesList_True, "");
|
|
|
|
// Test illegal number
|
|
testCalculate(Formula.FALSE, "R#", conditionalVariablesList_True,
|
|
"IXIC 1 parseCalculation error formula= R#, ex= java.lang.NumberFormatException");
|
|
testCalculate(Formula.FALSE, "R-", conditionalVariablesList_True,
|
|
"IXIC 1 parseCalculation error formula= R-, ex= java.lang.NumberFormatException");
|
|
testCalculate(Formula.FALSE, "Ra", conditionalVariablesList_True,
|
|
"IXIC 1 parseCalculation error formula= Ra, ex= java.lang.NumberFormatException");
|
|
|
|
// Test single condition
|
|
testCalculate(Formula.TRUE, "R1", conditionalVariablesList_True, "");
|
|
testCalculate(Formula.FALSE, "R1", conditionalVariablesList_False, "");
|
|
testCalculate(Formula.FALSE, "not R1", conditionalVariablesList_True, "");
|
|
testCalculate(Formula.TRUE, "not R1", conditionalVariablesList_False, "");
|
|
|
|
// Test single item but wrong item (R2 instead of R1)
|
|
// testCalculate(Formula.FALSE, "R2)", conditionalVariablesList_True,
|
|
// "IXIC 1 parseCalculation error formula= R2), ex= java.lang.ArrayIndexOutOfBoundsException");
|
|
|
|
// Test two digit variable numbers
|
|
testCalculate(Formula.TRUE, "R3 and R12 or R5 and R10",
|
|
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
|
|
testCalculate(Formula.FALSE, "R3 and (R12 or R5) and R10",
|
|
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
|
|
testCalculate(Formula.FALSE, "R12 and R10",
|
|
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
|
|
testCalculate(Formula.TRUE, "R12 or R10",
|
|
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
|
|
testCalculate(Formula.FALSE, "not (R12 or R10)",
|
|
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
|
|
|
|
// Test parentheses
|
|
testCalculate(Formula.TRUE, "([{R1)}]", conditionalVariablesList_True, "");
|
|
// testCalculate(Formula.FALSE, "(R2", conditionalVariablesList_True,
|
|
// "IXIC 1 parseCalculation error formula= (R2, ex= java.lang.ArrayIndexOutOfBoundsException");
|
|
|
|
// Test several items
|
|
testCalculate(Formula.FALSE, "R1 and R2 and R3", conditionalVariablesList_True,
|
|
"IXIC 1 parseCalculation error formula= R1 and R2 and R3, ex= java.lang.IndexOutOfBoundsException");
|
|
testCalculate(Formula.TRUE, "R1", conditionalVariablesList_TrueTrueTrue, "");
|
|
testCalculate(Formula.TRUE, "R2", conditionalVariablesList_TrueTrueTrue, "");
|
|
testCalculate(Formula.TRUE, "R3", conditionalVariablesList_TrueTrueTrue, "");
|
|
testCalculate(Formula.TRUE, "R1 and R2 and R3", conditionalVariablesList_TrueTrueTrue, "");
|
|
testCalculate(Formula.TRUE, "R2 AND R1 or R3", conditionalVariablesList_TrueTrueTrue, "");
|
|
|
|
// Test invalid combinations of and, or, not
|
|
testCalculate(Formula.FALSE, "R1 and or R3 and R2", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= R1 and or R3 and R2, ex= jmri.JmriException: Unexpected operator or characters < ORR3ANDR2 >");
|
|
testCalculate(Formula.FALSE, "R1 or or R3 and R2", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= R1 or or R3 and R2, ex= jmri.JmriException: Unexpected operator or characters < ORR3ANDR2 >");
|
|
testCalculate(Formula.FALSE, "R1 or and R3 and R2", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= R1 or and R3 and R2, ex= jmri.JmriException: Unexpected operator or characters < ANDR3ANDR2 >");
|
|
testCalculate(Formula.FALSE, "R1 not R3 and R2", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= R1 not R3 and R2, ex= jmri.JmriException: Could not find expected operator < NOTR3ANDR2 >");
|
|
testCalculate(Formula.FALSE, "and R1 not R3 and R2", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= and R1 not R3 and R2, ex= jmri.JmriException: Unexpected operator or characters < ANDR1NOTR3ANDR2 >");
|
|
testCalculate(Formula.FALSE, "R1 or R3 and R2 or", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= R1 or R3 and R2 or, ex= java.lang.StringIndexOutOfBoundsException");
|
|
|
|
// Test several items and parenthese
|
|
testCalculate(Formula.TRUE, "(R1 and R3) and R2", conditionalVariablesList_TrueTrueTrue, "");
|
|
testCalculate(Formula.FALSE, "(R1 and R3) and not R2", conditionalVariablesList_TrueTrueTrue, "");
|
|
testCalculate(Formula.FALSE, "(R1 and) R3 and not R2", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= (R1 and) R3 and not R2, ex= jmri.JmriException: Unexpected operator or characters < )R3ANDNOTR2 >");
|
|
testCalculate(Formula.FALSE, "R1( and R3) and not R2", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= R1( and R3) and not R2, ex= jmri.JmriException: Could not find expected operator < (ANDR3)ANDNOTR2 >");
|
|
testCalculate(Formula.FALSE, "R1 (and R3 and) not R2", conditionalVariablesList_TrueTrueTrue,
|
|
"IXIC 1 parseCalculation error formula= R1 (and R3 and) not R2, ex= jmri.JmriException: Could not find expected operator < (ANDR3AND)NOTR2 >");
|
|
testCalculate(Formula.FALSE, "(R1 and R3) and not R2)", conditionalVariablesList_TrueTrueTrue, "");
|
|
testCalculate(Formula.TRUE, "(R1 and (R3) and R2)", conditionalVariablesList_TrueTrueTrue, "");
|
|
testCalculate(Formula.FALSE, "(R1 and (R3) and not R2)", conditionalVariablesList_TrueTrueTrue, "");
|
|
}
|
|
*/
|
|
|
|
@BeforeEach
|
|
public void setUp() throws JmriException {
|
|
JUnitUtil.setUp();
|
|
JUnitUtil.resetInstanceManager();
|
|
JUnitUtil.resetProfileManager();
|
|
JUnitUtil.initConfigureManager();
|
|
JUnitUtil.initInternalSensorManager();
|
|
JUnitUtil.initInternalTurnoutManager();
|
|
JUnitUtil.initLogixNGManager();
|
|
|
|
_category = LogixNG_Category.COMMON;
|
|
_isExternal = false;
|
|
|
|
logixNG = InstanceManager.getDefault(LogixNG_Manager.class).createLogixNG("A new logix for test"); // NOI18N
|
|
conditionalNG = new DefaultConditionalNGScaffold("IQC1", "A conditionalNG"); // NOI18N;
|
|
InstanceManager.getDefault(ConditionalNG_Manager.class).register(conditionalNG);
|
|
conditionalNG.setRunDelayed(false);
|
|
conditionalNG.setEnabled(true);
|
|
logixNG.addConditionalNG(conditionalNG);
|
|
DoAnalogAction doAnalogAction = new DoAnalogAction("IQDA321", null);
|
|
MaleSocket maleSocket =
|
|
InstanceManager.getDefault(DigitalActionManager.class).registerAction(doAnalogAction);
|
|
conditionalNG.getChild(0).connect(maleSocket);
|
|
|
|
expressionFormula = new AnalogFormula("IQAE321", null);
|
|
// expressionFormula.setFormula("1");
|
|
// expressionFormula.setFormula("true");
|
|
expressionFormula.setFormula("E1");
|
|
MaleSocket maleSocketExpressionFormula =
|
|
InstanceManager.getDefault(AnalogExpressionManager.class).registerExpression(expressionFormula);
|
|
doAnalogAction.getChild(0).connect(maleSocketExpressionFormula);
|
|
|
|
AnalogExpressionConstant childExpression = new AnalogExpressionConstant("IQAE322", null);
|
|
childExpression.setValue(1.0);
|
|
MaleSocket maleSocketChild =
|
|
InstanceManager.getDefault(AnalogExpressionManager.class).registerExpression(childExpression);
|
|
maleSocketExpressionFormula.getChild(0).connect(maleSocketChild);
|
|
|
|
_base = expressionFormula;
|
|
_baseMaleSocket = maleSocketExpressionFormula;
|
|
|
|
analogMemory = InstanceManager.getDefault(MemoryManager.class).provide("IM2");
|
|
analogActionMemory = new AnalogActionMemory("IQAA1", null);
|
|
analogActionMemory.getSelectNamedBean().setNamedBean(analogMemory);
|
|
MaleSocket socketAnalogActionMemory = InstanceManager.getDefault(AnalogActionManager.class).registerAction(analogActionMemory);
|
|
doAnalogAction.getChild(1).connect(socketAnalogActionMemory);
|
|
|
|
assertTrue( logixNG.setParentForAllChildren(new ArrayList<>()));
|
|
logixNG.activate();
|
|
logixNG.setEnabled(true);
|
|
}
|
|
|
|
@AfterEach
|
|
public void tearDown() {
|
|
jmri.jmrit.logixng.util.LogixNG_Thread.stopAllLogixNGThreads();
|
|
JUnitUtil.deregisterBlockManagerShutdownTask();
|
|
JUnitUtil.tearDown();
|
|
}
|
|
|
|
}
|