Files
2026-06-17 14:00:51 +02:00

732 lines
36 KiB
Java

package jmri.jmrit.logixng.expressions;
import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;
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 static org.junit.jupiter.api.Assertions.fail;
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.ActionAtomicBoolean;
import jmri.jmrit.logixng.actions.IfThenElse;
import jmri.jmrit.logixng.expressions.Antecedent.ExpressionEntry;
import jmri.jmrit.logixng.implementation.DefaultConditionalNGScaffold;
import jmri.util.JUnitUtil;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
/**
* Test Antecedent
*
* @author Daniel Bergqvist 2018
*/
public class AntecedentTest extends AbstractDigitalExpressionTestBase implements FemaleSocketListener {
private static final boolean EXPECT_SUCCESS = true;
private static final boolean EXPECT_FAILURE = false;
private LogixNG logixNG;
private ConditionalNG conditionalNG;
private Antecedent expressionAntecedent;
private ActionAtomicBoolean actionAtomicBoolean;
private AtomicBoolean atomicBoolean;
@Override
public ConditionalNG getConditionalNG() {
return conditionalNG;
}
@Override
public LogixNG getLogixNG() {
return logixNG;
}
private int beanID = 901;
@Override
public MaleSocket getConnectableChild() {
DigitalExpressionBean childExpression = new True("IQDE"+Integer.toString(beanID), null);
beanID++;
MaleSocket maleSocketChild =
InstanceManager.getDefault(DigitalExpressionManager.class).registerExpression(childExpression);
return maleSocketChild;
}
@Override
public String getExpectedPrintedTree() {
return String.format(
"Antecedent: R1 ::: Use default%n" +
" ? E1%n" +
" Always true ::: 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" +
" If Then Else. Execute on change ::: Use default%n" +
" ? If%n" +
" Antecedent: R1 ::: Use default%n" +
" ? E1%n" +
" Always true ::: Use default%n" +
" ? E2%n" +
" Socket not connected%n" +
" ! Then%n" +
" Set the atomic boolean to true ::: Use default%n" +
" ! Else%n" +
" Socket not connected%n");
}
@Override
public NamedBean createNewBean(String systemName) {
Antecedent a = new Antecedent(systemName, null);
assertDoesNotThrow( () -> a.setAntecedent("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() throws JmriException {
Antecedent expression2;
expression2 = new Antecedent("IQDE321", null);
expression2.setAntecedent("R1");
assertNotNull( expression2, "object exists");
assertNull( expression2.getUserName(), "Username matches");
assertEquals( "Antecedent: R1", expression2.getLongDescription(), "String matches");
expression2 = new Antecedent("IQDE321", "My expression");
expression2.setAntecedent("R1");
assertNotNull( expression2, "object exists");
assertEquals( "My expression", expression2.getUserName(), "Username matches");
assertEquals( "Antecedent: R1", expression2.getLongDescription(), "String matches");
expression2 = new Antecedent("IQDE321", null);
expression2.setAntecedent("R1 and R2");
assertNotNull( expression2, "object exists");
assertNull( expression2.getUserName(), "Username matches");
assertEquals( "Antecedent: R1 and R2", expression2.getLongDescription(), "String matches");
expression2 = new Antecedent("IQDE321", "My expression");
expression2.setAntecedent("R1 or R2");
assertNotNull( expression2, "object exists");
assertEquals( "My expression", expression2.getUserName(), "Username matches");
assertEquals( "Antecedent: R1 or R2", expression2.getLongDescription(), "String matches");
IllegalArgumentException ex = assertThrows( IllegalArgumentException.class,
() -> {
Antecedent ant = new Antecedent("IQE55:12:XY11", null);
assertNotNull(ant, "Should not reach here");
}, "Illegal system name Expected exception thrown");
assertNotNull(ex);
ex = assertThrows( IllegalArgumentException.class,
() -> {
Antecedent ant = new Antecedent("IQE55:12:XY11", "A name");
assertNotNull(ant, "Should not reach here");
}, "Illegal system name Expected exception thrown");
assertNotNull(ex);
}
// Test action when at least one child socket is not connected
@Test
public void testCtorAndSetup1() {
DigitalExpressionManager m = InstanceManager.getDefault(DigitalExpressionManager.class);
List<MaleSocket> maleSockets = new ArrayList<>();
maleSockets.add(m.registerExpression(new ExpressionMemory("IQDE52", null)));
maleSockets.add(null); // This is null by purpose
maleSockets.add(m.registerExpression(new ExpressionMemory("IQDE554", null)));
maleSockets.add(null); // This is null by purpose
maleSockets.add(m.registerExpression(new ExpressionMemory("IQDE3", null)));
List<Map.Entry<String, String>> actionSystemNames = new ArrayList<>();
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("XYZ123", "IQDE52"));
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("ZH12", null)); // This is null by purpose
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("Hello", "IQDE554"));
// IQDE61232 doesn't exist by purpose
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("SomethingElse", "IQDE61232"));
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("Yes123", "IQDE3"));
Antecedent expression = new Antecedent("IQDE321", null, actionSystemNames);
assertNotNull( expression, "exists");
assertEquals( 5, expression.getChildCount(), "expression has 5 female sockets");
for (int i=0; i < 5; i++) {
Map.Entry<String,String> entry = actionSystemNames.get(i);
assertEquals( entry.getKey(), expression.getChild(i).getName(),
() -> "expression female socket name is "+entry.getKey());
assertEquals(
// "jmri.jmrit.logixng.implementation.DefaultFemaleGenericExpressionSocket$DigitalSocket",
"jmri.jmrit.logixng.implementation.DefaultFemaleDigitalExpressionSocket",
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 digital expression IQDE61232");
for (int i=0; i < 5; i++) {
Map.Entry<String,String> entry = actionSystemNames.get(i);
assertEquals( entry.getKey(), expression.getChild(i).getName(),
() -> "expression female socket name is "+entry.getKey());
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 action when at least one child socket is not connected.
// This should never happen, but test it anyway.
@Test
public void testCtorAndSetup2() {
DigitalExpressionManager m = InstanceManager.getDefault(DigitalExpressionManager.class);
List<MaleSocket> maleSockets = new ArrayList<>();
maleSockets.add(m.registerExpression(new ExpressionMemory("IQDE52", null)));
maleSockets.add(m.registerExpression(new ExpressionMemory("IQDE99", null)));
maleSockets.add(m.registerExpression(new ExpressionMemory("IQDE554", null)));
maleSockets.add(m.registerExpression(new ExpressionMemory("IQDE61232", null)));
maleSockets.add(m.registerExpression(new ExpressionMemory("IQDE3", null)));
List<Map.Entry<String, String>> actionSystemNames = new ArrayList<>();
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("XYZ123", "IQDE52"));
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("ZH12", "IQDE99"));
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("Hello", "IQDE554"));
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("SomethingElse", "IQDE61232"));
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("Yes123", "IQDE3"));
Antecedent expression = new Antecedent("IQDE321", null, actionSystemNames);
assertNotNull( expression, "exists");
assertEquals( 5, expression.getChildCount(), "expression has 5 female sockets");
for (int i=0; i < 5; i++) {
Map.Entry<String,String> entry = actionSystemNames.get(i);
assertEquals( entry.getKey(), expression.getChild(i).getName(),
() -> "expression female socket name is "+entry.getKey());
assertEquals(
// "jmri.jmrit.logixng.implementation.DefaultFemaleGenericExpressionSocket$DigitalSocket",
"jmri.jmrit.logixng.implementation.DefaultFemaleDigitalExpressionSocket",
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++) {
Map.Entry<String,String> entry = actionSystemNames.get(i);
assertEquals( entry.getKey(), expression.getChild(i).getName(),
() -> "expression female socket name is "+entry.getKey());
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<Map.Entry<String, String>> actionSystemNames = new ArrayList<>();
actionSystemNames.add(new java.util.HashMap.SimpleEntry<>("XYZ123", "IQDE52"));
Antecedent expression = new Antecedent("IQDE321", 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 {
_baseMaleSocket.setEnabled(false);
Antecedent a = (Antecedent)_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
public void testGetChild() throws JmriException {
Antecedent expression2 = new Antecedent("IQDE321", null);
expression2.setAntecedent("R1");
for (int i=0; i < 3; i++) {
assertTrue( i+1 == expression2.getChildCount(),
"getChildCount() returns "+i);
assertNotNull( expression2.getChild(0),
"getChild(0) returns a non null value");
assertIndexOutOfBoundsException(expression2::getChild, i+1, i+1);
// Connect a new child expression
True expr = new True("IQDE"+i, null);
MaleSocket maleSocket =
InstanceManager.getDefault(DigitalExpressionManager.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 {
Antecedent expression = new Antecedent("IQDE121", null);
ExpressionMemory stringExpressionMemory = new ExpressionMemory("IQDE122", null);
MaleSocket maleSAMSocket =
InstanceManager.getDefault(DigitalExpressionManager.class).registerExpression(stringExpressionMemory);
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( "IQDE122", 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( "IQDE122", expression.getExpressionSystemName(1), "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( "IQDE122", 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() {
Antecedent e1 = new Antecedent("IQDE321", null);
assertEquals( "Antecedent", e1.getShortDescription(), "string matches");
assertEquals( "Antecedent: empty", e1.getLongDescription(), "string matches 2");
}
private void testValidate(boolean expectedResult, String antecedent,
List<DigitalExpressionBean> conditionalVariablesList) throws JmriException {
Antecedent ix1 = new Antecedent("IQDE321", "IXIC 1");
ix1.setAntecedent("R1");
int count = 0;
List<ExpressionEntry> expressionEntryList = new ArrayList<>();
for (DigitalExpressionBean digExpressionAntecedent : conditionalVariablesList) {
count++;
String socketName = "E"+Integer.toString(count);
FemaleDigitalExpressionSocket socket =
InstanceManager.getDefault(DigitalExpressionManager.class)
.createFemaleSocket(conditionalNG, this, socketName);
socket.connect((MaleSocket) digExpressionAntecedent);
expressionEntryList.add(new ExpressionEntry(socket, socketName));
}
if (expectedResult) {
assertNull( ix1.validateAntecedent(antecedent, expressionEntryList),
() -> "validateAntecedent() returns null for '"+antecedent+"'");
} else {
assertNotNull( ix1.validateAntecedent(antecedent, expressionEntryList),
() -> "validateAntecedent() returns error message for '"+antecedent+"'");
}
}
private void testCalculate(int expectedResult, String antecedent,
List<DigitalExpressionBean> conditionalVariablesList, String errorMessage)
throws JmriException {
Antecedent ix1 = new Antecedent("IQDE321", "IXIC 1");
ix1.setParent(conditionalNG);
ix1.setAntecedent(antecedent);
// 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 Antecedent.FALSE:
assertFalse( ix1.evaluate(),
() -> "validateAntecedent() returns FALSE for '"+antecedent+"'");
break;
case Antecedent.TRUE:
// System.err.format("antecedent: %s%n", antecedent);
// System.err.format("variable: %b%n", conditionalVariablesList.get(0).evaluate(isCompleted));
assertTrue( ix1.evaluate(),
() -> "validateAntecedent() returns TRUE for '"+antecedent+"'");
break;
default:
fail( () -> String.format("Unknown expected result: %d", expectedResult));
}
if (! errorMessage.isEmpty()) {
jmri.util.JUnitAppender.assertErrorMessageStartsWith(errorMessage);
}
}
@Test
public void testValidate() throws JmriException {
DigitalExpressionBean[] conditionalVariables_Empty = { };
List<DigitalExpressionBean> conditionalVariablesList_Empty = Arrays.asList(conditionalVariables_Empty);
DigitalExpressionBean trueExpression =
InstanceManager.getDefault(
DigitalExpressionManager.class).registerExpression(
new True(InstanceManager.getDefault(DigitalExpressionManager.class).getAutoSystemName(), null));
// DigitalExpressionBean falseExpression = InstanceManager.getDefault(DigitalExpressionManager.class).registerExpression(new False(conditionalNG));
DigitalExpressionBean[] conditionalVariables_True
= { trueExpression };
List<DigitalExpressionBean> conditionalVariablesList_True = Arrays.asList(conditionalVariables_True);
DigitalExpressionBean[] conditionalVariables_TrueTrueTrue
= { trueExpression
, trueExpression
, trueExpression };
List<DigitalExpressionBean> conditionalVariablesList_TrueTrueTrue = Arrays.asList(conditionalVariables_TrueTrueTrue);
// Test empty antecedent 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 JmriException {
DigitalExpressionBean[] conditionalVariables_Empty = { };
List<DigitalExpressionBean> conditionalVariablesList_Empty = Arrays.asList(conditionalVariables_Empty);
DigitalExpressionBean trueExpression =
InstanceManager.getDefault(
DigitalExpressionManager.class).registerExpression(
new True(InstanceManager.getDefault(DigitalExpressionManager.class).getAutoSystemName(), null));
DigitalExpressionBean falseExpression =
InstanceManager.getDefault(
DigitalExpressionManager.class).registerExpression(
new False(InstanceManager.getDefault(DigitalExpressionManager.class).getAutoSystemName(), null));
DigitalExpressionBean[] conditionalVariables_True
= { trueExpression };
List<DigitalExpressionBean> conditionalVariablesList_True = Arrays.asList(conditionalVariables_True);
DigitalExpressionBean[] conditionalVariables_False
= { falseExpression };
List<DigitalExpressionBean> conditionalVariablesList_False = Arrays.asList(conditionalVariables_False);
DigitalExpressionBean[] conditionalVariables_TrueTrueTrue
= { trueExpression
, trueExpression
, trueExpression };
List<DigitalExpressionBean> conditionalVariablesList_TrueTrueTrue = Arrays.asList(conditionalVariables_TrueTrueTrue);
// Test with two digit variable numbers
DigitalExpressionBean[] conditionalVariables_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse
= {trueExpression
, trueExpression
, falseExpression
, trueExpression
, trueExpression
, falseExpression
, trueExpression
, trueExpression
, falseExpression
, trueExpression
, trueExpression
, falseExpression };
List<DigitalExpressionBean> conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse =
Arrays.asList(conditionalVariables_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse);
// Test empty antecedent string
testCalculate(Antecedent.FALSE, "", conditionalVariablesList_Empty, "");
// testCalculate(Antecedent.FALSE, "", conditionalVariablesList_True,
// "IXIC 1 parseCalculation error antecedent= , ex= java.lang.StringIndexOutOfBoundsException");
testCalculate(Antecedent.FALSE, "", conditionalVariablesList_True, "");
// Test illegal number
testCalculate(Antecedent.FALSE, "R#", conditionalVariablesList_True,
"IXIC 1 parseCalculation error antecedent= R#, ex= java.lang.NumberFormatException: For input string: \"#\"");
testCalculate(Antecedent.FALSE, "R-", conditionalVariablesList_True,
"IXIC 1 parseCalculation error antecedent= R-, ex= java.lang.NumberFormatException: For input string: \"-\"");
testCalculate(Antecedent.FALSE, "Ra", conditionalVariablesList_True,
"IXIC 1 parseCalculation error antecedent= Ra, ex= java.lang.NumberFormatException: For input string: \"A\"");
// Test single condition
testCalculate(Antecedent.TRUE, "R1", conditionalVariablesList_True, "");
testCalculate(Antecedent.FALSE, "R1", conditionalVariablesList_False, "");
testCalculate(Antecedent.FALSE, "not R1", conditionalVariablesList_True, "");
testCalculate(Antecedent.TRUE, "not R1", conditionalVariablesList_False, "");
// Test single item but wrong item (R2 instead of R1)
// testCalculate(Antecedent.FALSE, "R2)", conditionalVariablesList_True,
// "IXIC 1 parseCalculation error antecedent= R2), ex= java.lang.ArrayIndexOutOfBoundsException");
// Test two digit variable numbers
testCalculate(Antecedent.TRUE, "R3 and R12 or R5 and R10",
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
testCalculate(Antecedent.FALSE, "R3 and (R12 or R5) and R10",
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
testCalculate(Antecedent.FALSE, "R12 and R10",
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
testCalculate(Antecedent.TRUE, "R12 or R10",
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
testCalculate(Antecedent.FALSE, "not (R12 or R10)",
conditionalVariablesList_TrueTrueFalseTrueTrueFalseTrueTrueFalseTrueTrueFalse, "");
// Test parentheses
testCalculate(Antecedent.TRUE, "([{R1)}]", conditionalVariablesList_True, "");
// testCalculate(Antecedent.FALSE, "(R2", conditionalVariablesList_True,
// "IXIC 1 parseCalculation error antecedent= (R2, ex= java.lang.ArrayIndexOutOfBoundsException");
// Test several items
testCalculate(Antecedent.FALSE, "R1 and R2 and R3", conditionalVariablesList_True,
"IXIC 1 parseCalculation error antecedent= R1 and R2 and R3, ex= java.lang.IndexOutOfBoundsException");
testCalculate(Antecedent.TRUE, "R1", conditionalVariablesList_TrueTrueTrue, "");
testCalculate(Antecedent.TRUE, "R2", conditionalVariablesList_TrueTrueTrue, "");
testCalculate(Antecedent.TRUE, "R3", conditionalVariablesList_TrueTrueTrue, "");
testCalculate(Antecedent.TRUE, "R1 and R2 and R3", conditionalVariablesList_TrueTrueTrue, "");
testCalculate(Antecedent.TRUE, "R2 AND R1 or R3", conditionalVariablesList_TrueTrueTrue, "");
// Test invalid combinations of and, or, not
testCalculate(Antecedent.FALSE, "R1 and or R3 and R2", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= R1 and or R3 and R2, ex= jmri.JmriException: Unexpected operator or characters < ORR3ANDR2 >");
testCalculate(Antecedent.FALSE, "R1 or or R3 and R2", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= R1 or or R3 and R2, ex= jmri.JmriException: Unexpected operator or characters < ORR3ANDR2 >");
testCalculate(Antecedent.FALSE, "R1 or and R3 and R2", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= R1 or and R3 and R2, ex= jmri.JmriException: Unexpected operator or characters < ANDR3ANDR2 >");
testCalculate(Antecedent.FALSE, "R1 not R3 and R2", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= R1 not R3 and R2, ex= jmri.JmriException: Could not find expected operator < NOTR3ANDR2 >");
testCalculate(Antecedent.FALSE, "and R1 not R3 and R2", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= and R1 not R3 and R2, ex= jmri.JmriException: Unexpected operator or characters < ANDR1NOTR3ANDR2 >");
testCalculate(Antecedent.FALSE, "R1 or R3 and R2 or", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= R1 or R3 and R2 or, ex= java.lang.StringIndexOutOfBoundsException");
// Test several items and parenthese
testCalculate(Antecedent.TRUE, "(R1 and R3) and R2", conditionalVariablesList_TrueTrueTrue, "");
testCalculate(Antecedent.FALSE, "(R1 and R3) and not R2", conditionalVariablesList_TrueTrueTrue, "");
testCalculate(Antecedent.FALSE, "(R1 and) R3 and not R2", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= (R1 and) R3 and not R2, ex= jmri.JmriException: Unexpected operator or characters < )R3ANDNOTR2 >");
testCalculate(Antecedent.FALSE, "R1( and R3) and not R2", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= R1( and R3) and not R2, ex= jmri.JmriException: Could not find expected operator < (ANDR3)ANDNOTR2 >");
testCalculate(Antecedent.FALSE, "R1 (and R3 and) not R2", conditionalVariablesList_TrueTrueTrue,
"IXIC 1 parseCalculation error antecedent= R1 (and R3 and) not R2, ex= jmri.JmriException: Could not find expected operator < (ANDR3AND)NOTR2 >");
testCalculate(Antecedent.FALSE, "(R1 and R3) and not R2)", conditionalVariablesList_TrueTrueTrue, "");
testCalculate(Antecedent.TRUE, "(R1 and (R3) and R2)", conditionalVariablesList_TrueTrueTrue, "");
testCalculate(Antecedent.FALSE, "(R1 and (R3) and not R2)", conditionalVariablesList_TrueTrueTrue, "");
}
@Override
public void connected(FemaleSocket socket) {
// Do nothing
}
@Override
public void disconnected(FemaleSocket socket) {
// Do nothing
}
@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);
IfThenElse ifThenElse = new IfThenElse("IQDA321", null);
MaleSocket maleSocket =
InstanceManager.getDefault(DigitalActionManager.class).registerAction(ifThenElse);
conditionalNG.getChild(0).connect(maleSocket);
expressionAntecedent = new Antecedent("IQDE321", null);
expressionAntecedent.setAntecedent("R1");
MaleSocket maleSocket2 =
InstanceManager.getDefault(DigitalExpressionManager.class).registerExpression(expressionAntecedent);
ifThenElse.getChild(0).connect(maleSocket2);
DigitalExpressionBean childExpression = new True("IQDE322", null);
MaleSocket maleSocketChild =
InstanceManager.getDefault(DigitalExpressionManager.class).registerExpression(childExpression);
maleSocket2.getChild(0).connect(maleSocketChild);
_base = expressionAntecedent;
_baseMaleSocket = maleSocket2;
atomicBoolean = new AtomicBoolean(false);
actionAtomicBoolean = new ActionAtomicBoolean(atomicBoolean, true);
MaleSocket socketAtomicBoolean = InstanceManager.getDefault(DigitalActionManager.class).registerAction(actionAtomicBoolean);
ifThenElse.getChild(1).connect(socketAtomicBoolean);
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();
}
}