I finally wrote the first bunch of serious tests with respect to the way the simulator processes circuits. I grouped together a few tests that test the jump labels in a circuit under a class. The code can be found in my bitbucket and sourceforge repos:
https://bitbucket.org/shivkiyer/ppe_simulator/src/testing/
https://sourceforge.net/p/pythonpowerelec/code/ci/testing/tree/
Also, to get a full length course on power electronics simulations, check out my online course:
https://www.udemy.com/course/simulating-power-electronic-circuits-using-python/
The class in the command line interface is:
The functionalities that I am testing for specifically are:
https://bitbucket.org/shivkiyer/ppe_simulator/src/testing/
https://sourceforge.net/p/pythonpowerelec/code/ci/testing/tree/
Also, to get a full length course on power electronics simulations, check out my online course:
https://www.udemy.com/course/simulating-power-electronic-circuits-using-python/
The class in the command line interface is:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class TestJumpHandling: | |
""" | |
This class groups together tests related to checking and handling jump | |
labels in the circuit schematic. | |
""" | |
def test_jump_sanity(self): | |
""" | |
jump_sanity checks for whether an element in the complete system matrix | |
is a jump, a component or no connection. | |
param - 3 dimensional array (3 level nested list) of elements. | |
param - the sheet, row, column position. | |
param - an indicator dictionary about whether the element is a jump, a component or null. | |
result - a modification to the indicator dictionary object. | |
""" | |
from network_reader import jump_sanity, scrub_elements | |
print() | |
print('*'*80) | |
print('Testing whether a jump can be identitifed on a circuit schematic element') | |
test_matrix = [ | |
[ | |
['wire', '', 'dyxcsdc', 'jump1', 'Resistor'], | |
[' ', '', ' jump2', ' \n ', ' wire '], | |
], | |
[ | |
['wire', '', 'dyxcsdc', 'jump1', 'Resistor'], | |
[' ', '', ' jump2', ' \n ', ' wire '], | |
], | |
] | |
def repeat_test(sheet, row, column, expected_val): | |
""" | |
A quick test method that takes the co-ordinates and | |
checks the expected value. | |
""" | |
test_element = {"exist":0, "jump":1} | |
jump_sanity(test_matrix, test_element, sheet, row, column) | |
assert test_element == expected_val | |
return | |
repeat_test(0, 0, 0, {"exist":0}) | |
repeat_test(0, 0, 1, {}) | |
repeat_test(0, 0, 2, {"exist":0}) | |
repeat_test(0, 0, 3, {"jump":1}) | |
repeat_test(0, 0, 4, {"exist":0}) | |
# empty space cannot be handled and is treated as an element | |
repeat_test(0, 1, 0, {"exist":0}) | |
repeat_test(0, 1, 1, {}) | |
# trailing empty space turns jump into an element | |
repeat_test(0, 1, 2, {"exist":0}) | |
# newline character is seen as an element | |
repeat_test(0, 1, 3, {"exist":0}) | |
repeat_test(0, 1, 4, {"exist":0}) | |
# Repitition to check for multiple sheets. | |
repeat_test(1, 0, 0, {"exist":0}) | |
repeat_test(1, 0, 1, {}) | |
repeat_test(1, 0, 2, {"exist":0}) | |
repeat_test(1, 0, 3, {"jump":1}) | |
repeat_test(1, 0, 4, {"exist":0}) | |
repeat_test(1, 1, 0, {"exist":0}) | |
repeat_test(1, 1, 1, {}) | |
repeat_test(1, 1, 2, {"exist":0}) | |
repeat_test(1, 1, 3, {"exist":0}) | |
repeat_test(1, 1, 4, {"exist":0}) | |
print() | |
return | |
def test_jump_checking(self): | |
""" | |
jump_checking - checks if a jump label is legal | |
jumps have to be extreme elements in a branch | |
jumps cannot be next to each other | |
params - complete circuit matrix, and sheet, row, col co-ordinates | |
params - rows, columns in that particular sheet | |
params - list of circuit spreadsheet names | |
returns/modifies - jump list adding a legal jump to it | |
""" | |
from network_reader import jump_checking | |
print() | |
print('*'*80) | |
print('Testing the jump_checking function') | |
print() | |
test_matrix = [ | |
[ | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
['dyxcsdc', 'wire', 'wire', 'Resistor', 'jump1'], | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
], | |
] | |
test_jump = [] | |
test_nw_name = ['test1',] | |
test_no_of_rows = len(test_matrix[0]) | |
test_no_of_cols = len(test_matrix[0][0]) | |
# jump1 is a legal jump | |
jump_checking(test_matrix, test_jump, 0, 2, 4, test_no_of_rows, test_no_of_cols, test_nw_name) | |
assert len(test_jump) == 1 | |
assert test_jump[0] == [0, 2, 4, 'jump1', 'left'] | |
# jump2 is at a node. Exception is not specially stating that it cannot be a node. | |
test_matrix[0][2][2] = 'jump2' | |
with pytest.raises(SystemExit): | |
jump_checking(test_matrix, test_jump, 0, 2, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
test_matrix[0][2][2] = 'wire' | |
# A legal jump at the extreme bottom | |
test_matrix[0][5][2] = 'jump2' | |
jump_checking(test_matrix, test_jump, 0, 5, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
assert len(test_jump) == 2 | |
assert test_jump[1] == [0, 5, 2, 'jump2', 'up'] | |
# Illegal - jump not extreme element on branch. | |
test_matrix[0][4][2] = 'jump3' | |
with pytest.raises(SystemExit): | |
jump_checking(test_matrix, test_jump, 0, 4, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
test_matrix[0][4][2] = 'wire' | |
# jump2 is not extreme in a branch. Exception is not specially stating that it cannot be next to a node. | |
test_matrix[0][1][2] = 'jump3' | |
with pytest.raises(SystemExit): | |
jump_checking(test_matrix, test_jump, 0, 1, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
test_matrix[0][0][2] = '' | |
# This should raise an exception because a jump should not be next to a node. | |
# But this test fails | |
# with pytest.raises(SystemExit): | |
# jump_checking(test_matrix, test_jump, 0, 1, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
# Illegal - two jumps next to each other | |
test_matrix[0][0][2] = 'jump3' | |
with pytest.raises(SystemExit): | |
jump_checking(test_matrix, test_jump, 0, 0, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
# Resetting the element below makes it a legal jump | |
test_matrix[0][1][2] = 'xyz' | |
jump_checking(test_matrix, test_jump, 0, 0, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
assert len(test_jump) == 3 | |
assert test_jump[2] == [0, 0, 2, 'jump3', 'down'] | |
# Illegal - not extreme element in a branch | |
test_matrix[0][2][1] = 'jump4' | |
with pytest.raises(SystemExit): | |
jump_checking(test_matrix, test_jump, 0, 2, 1, test_no_of_rows, test_no_of_cols, test_nw_name) | |
# Illegal - jumps next to each other | |
test_matrix[0][2][0] = 'jump5' | |
with pytest.raises(SystemExit): | |
jump_checking(test_matrix, test_jump, 0, 2, 0, test_no_of_rows, test_no_of_cols, test_nw_name) | |
# Restting the jump to the right to an element makes it a legal jump | |
test_matrix[0][2][1] = 'xyz' | |
jump_checking(test_matrix, test_jump, 0, 2, 0, test_no_of_rows, test_no_of_cols, test_nw_name) | |
assert len(test_jump) == 4 | |
assert test_jump[3] == [0, 2, 0, 'jump5', 'right'] | |
# Illegal - jump labels next to each other | |
test_matrix[0][0][1] = 'jump6' | |
with pytest.raises(SystemExit): | |
jump_checking(test_matrix, test_jump, 0, 0, 1, test_no_of_rows, test_no_of_cols, test_nw_name) | |
# Legal jump when element to the right is reset to a component. | |
test_matrix[0][0][2] = 'compo' | |
jump_checking(test_matrix, test_jump, 0, 0, 1, test_no_of_rows, test_no_of_cols, test_nw_name) | |
assert len(test_jump) == 5 | |
assert test_jump[4] == [0, 0, 1, 'jump6', 'right'] | |
return | |
def test_jump_node_check(self): | |
""" | |
jump_node_check - checks whether a jump label is next to a node. | |
param - entire circuit matrix | |
param - list of nodes | |
param - a direction in which a particular node has an element | |
param - index of that particular node | |
param - list of circuit spreadsheet names | |
result - throws an error if in the direction specified for the node, | |
there is a jump label in the adjacent cell. Adjacent cell is measured | |
with respect to the node. | |
""" | |
from network_reader import jump_node_check | |
print() | |
print('*'*80) | |
print('Testing the jump_node_check function') | |
print() | |
test_matrix = [ | |
[ | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
['dyxcsdc', 'wire', 'wire', 'Resistor', 'jump1'], | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
], | |
] | |
test_node_list = [ | |
[0, 2, 2] | |
] | |
test_nw_name = ['test1',] | |
try: | |
jump_node_check(test_matrix, test_node_list, 'left', 0, test_nw_name) | |
except: | |
pytest.fail('This should haved worked') | |
test_matrix[0][2][1] = 'jump2' | |
with pytest.raises(SystemExit): | |
jump_node_check(test_matrix, test_node_list, 'left', 0, test_nw_name) | |
test_matrix[0][2][1] = 'xyz' | |
test_matrix[0][2][0] = 'jump2' | |
try: | |
jump_node_check(test_matrix, test_node_list, 'left', 0, test_nw_name) | |
except: | |
pytest.fail('This should haved worked') | |
test_matrix[0][1][2] = 'jump3' | |
with pytest.raises(SystemExit): | |
jump_node_check(test_matrix, test_node_list, 'up', 0, test_nw_name) | |
test_matrix[0][1][2] = 'xyz' | |
test_matrix[0][0][2] = 'jump3' | |
try: | |
jump_node_check(test_matrix, test_node_list, 'up', 0, test_nw_name) | |
except: | |
pytest.fail('This should haved worked') | |
test_matrix[0][3][2] = 'jump4' | |
with pytest.raises(SystemExit): | |
jump_node_check(test_matrix, test_node_list, 'down', 0, test_nw_name) | |
test_matrix[0][3][2] = 'xyz' | |
test_matrix[0][4][2] = 'jump4' | |
try: | |
jump_node_check(test_matrix, test_node_list, 'down', 0, test_nw_name) | |
except: | |
pytest.fail('This should haved worked') | |
print() | |
return |
The functionalities that I am testing for specifically are:
- Check whether an element is a component (could also be a wire), a jump label or no connection at all.
- A jump must be the extreme element in a branch segment. If it has components on more than one side, that is a violation that throws an exception.
- Two jump labels cannot be adjacent to each other - that is a violation that throws an exception.
- A jump label cannot be next to a node - a violation that throws an exception.
Most of the tests could be accomplished by a simple assert statement. However, to check if a method throws an exception, I found this block to be most useful:
with pytest.raises(SystemExit): ...method...
So, the method or the block of code must throw an error or else the test fails. SystemExit could be replaced by other errors, but in this case, all I do is exit with an error code of 1 and so a SystemExit is all I can check for.
Conversely, if a method or block of code should not throw and error, the test can be written as:
try: ...method... except: pytest.fail('This should haved worked')
The class for the web app is a bit different:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class TestJumpHandling(TestCase): | |
""" | |
This class groups together tests related to checking and handling jump | |
labels in the circuit schematic. | |
""" | |
def test_jump_sanity(self): | |
""" | |
jump_sanity checks for whether an element in the complete system matrix | |
is a jump, a component or no connection. | |
param - 3 dimensional array (3 level nested list) of elements. | |
param - the sheet, row, column position. | |
param - an indicator dictionary about whether the element is a jump, a component or null. | |
result - a modification to the indicator dictionary object. | |
""" | |
from simulations.network_reader import jump_sanity, scrub_elements | |
print() | |
print('*'*80) | |
print('Testing whether a jump can be identitifed on a circuit schematic element') | |
test_matrix = [ | |
[ | |
['wire', '', 'dyxcsdc', 'jump1', 'Resistor'], | |
[' ', '', ' jump2', ' \n ', ' wire '], | |
], | |
[ | |
['wire', '', 'dyxcsdc', 'jump1', 'Resistor'], | |
[' ', '', ' jump2', ' \n ', ' wire '], | |
], | |
] | |
def repeat_test(sheet, row, column, expected_val): | |
""" | |
A quick test method that takes the co-ordinates and | |
checks the expected value. | |
""" | |
test_element = {"exist":0, "jump":1} | |
jump_sanity(test_matrix, test_element, sheet, row, column) | |
self.assertEqual(test_element, expected_val) | |
return | |
repeat_test(0, 0, 0, {"exist":0}) | |
repeat_test(0, 0, 1, {}) | |
repeat_test(0, 0, 2, {"exist":0}) | |
repeat_test(0, 0, 3, {"jump":1}) | |
repeat_test(0, 0, 4, {"exist":0}) | |
# empty space cannot be handled and is treated as an element | |
repeat_test(0, 1, 0, {"exist":0}) | |
repeat_test(0, 1, 1, {}) | |
# trailing empty space turns jump into an element | |
repeat_test(0, 1, 2, {"exist":0}) | |
# newline character is seen as an element | |
repeat_test(0, 1, 3, {"exist":0}) | |
repeat_test(0, 1, 4, {"exist":0}) | |
# Repitition to check for multiple sheets. | |
repeat_test(1, 0, 0, {"exist":0}) | |
repeat_test(1, 0, 1, {}) | |
repeat_test(1, 0, 2, {"exist":0}) | |
repeat_test(1, 0, 3, {"jump":1}) | |
repeat_test(1, 0, 4, {"exist":0}) | |
repeat_test(1, 1, 0, {"exist":0}) | |
repeat_test(1, 1, 1, {}) | |
repeat_test(1, 1, 2, {"exist":0}) | |
repeat_test(1, 1, 3, {"exist":0}) | |
repeat_test(1, 1, 4, {"exist":0}) | |
print() | |
return | |
def test_jump_checking(self): | |
""" | |
jump_checking - checks if a jump label is legal | |
jumps have to be extreme elements in a branch | |
jumps cannot be next to each other | |
params - complete circuit matrix, and sheet, row, col co-ordinates | |
params - rows, columns in that particular sheet | |
params - list of circuit spreadsheet names | |
returns/modifies - jump list adding a legal jump to it | |
""" | |
from simulations.network_reader import jump_checking | |
import simulations.circuit_exceptions as CktEx | |
print() | |
print('*'*80) | |
print('Testing the jump_checking function') | |
print() | |
test_matrix = [ | |
[ | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
['dyxcsdc', 'wire', 'wire', 'Resistor', 'jump1'], | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
], | |
] | |
test_jump = [] | |
test_nw_name = ['test1',] | |
test_no_of_rows = len(test_matrix[0]) | |
test_no_of_cols = len(test_matrix[0][0]) | |
# jump1 is a legal jump | |
jump_checking(test_matrix, test_jump, 0, 2, 4, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertEqual(test_jump[-1], [0, 2, 4, 'jump1', 'left']) | |
# jump2 is at a node. Exception is not specially stating that it cannot be a node. | |
test_matrix[0][2][2] = 'jump2' | |
result = jump_checking(test_matrix, test_jump, 0, 2, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
test_matrix[0][2][2] = 'wire' | |
# A legal jump at the extreme bottom | |
test_matrix[0][5][2] = 'jump2' | |
jump_checking(test_matrix, test_jump, 0, 5, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertEqual(test_jump[-1], [0, 5, 2, 'jump2', 'up']) | |
# Illegal - jump not extreme element on branch. | |
test_matrix[0][4][2] = 'jump3' | |
result = jump_checking(test_matrix, test_jump, 0, 4, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
test_matrix[0][4][2] = 'wire' | |
# jump2 is not extreme in a branch. Exception is not specially stating that it cannot be next to a node. | |
test_matrix[0][1][2] = 'jump3' | |
result = jump_checking(test_matrix, test_jump, 0, 1, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
test_matrix[0][0][2] = '' | |
# This should raise an exception because a jump should not be next to a node. | |
# But this test fails | |
# result = jump_checking(test_matrix, test_jump, 0, 1, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
# self.assertTrue(len(result) > 0) | |
# Illegal - two jumps next to each other | |
test_matrix[0][0][2] = 'jump3' | |
result = jump_checking(test_matrix, test_jump, 0, 0, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
# Resetting the element below makes it a legal jump | |
test_matrix[0][1][2] = 'xyz' | |
jump_checking(test_matrix, test_jump, 0, 0, 2, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertEqual(test_jump[-1], [0, 0, 2, 'jump3', 'down']) | |
# Illegal - not extreme element in a branch | |
test_matrix[0][2][1] = 'jump4' | |
result = jump_checking(test_matrix, test_jump, 0, 2, 1, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
# Illegal - jumps next to each other | |
test_matrix[0][2][0] = 'jump5' | |
result = jump_checking(test_matrix, test_jump, 0, 2, 0, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
# Restting the jump to the right to an element makes it a legal jump | |
test_matrix[0][2][1] = 'xyz' | |
jump_checking(test_matrix, test_jump, 0, 2, 0, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertEqual(test_jump[-1], [0, 2, 0, 'jump5', 'right']) | |
# Illegal - jump labels next to each other | |
test_matrix[0][0][1] = 'jump6' | |
result = jump_checking(test_matrix, test_jump, 0, 0, 1, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
# Legal jump when element to the right is reset to a component. | |
test_matrix[0][0][2] = 'compo' | |
jump_checking(test_matrix, test_jump, 0, 0, 1, test_no_of_rows, test_no_of_cols, test_nw_name) | |
self.assertEqual(test_jump[-1], [0, 0, 1, 'jump6', 'right']) | |
return | |
def test_jump_node_check(self): | |
""" | |
jump_node_check - checks whether a jump label is next to a node. | |
param - entire circuit matrix | |
param - list of nodes | |
param - a direction in which a particular node has an element | |
param - index of that particular node | |
param - list of circuit spreadsheet names | |
result - throws an error if in the direction specified for the node, | |
there is a jump label in the adjacent cell. Adjacent cell is measured | |
with respect to the node. | |
""" | |
from simulations.network_reader import jump_node_check | |
print() | |
print('*'*80) | |
print('Testing the jump_node_check function') | |
print() | |
test_matrix = [ | |
[ | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
['dyxcsdc', 'wire', 'wire', 'Resistor', 'jump1'], | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
['', '', 'wire', '', ''], | |
], | |
] | |
test_node_list = [ | |
[0, 2, 2] | |
] | |
test_nw_name = ['test1',] | |
result = jump_node_check(test_matrix, test_node_list, 'left', 0, test_nw_name) | |
self.assertTrue(len(result) == 0) | |
test_matrix[0][2][1] = 'jump2' | |
result = jump_node_check(test_matrix, test_node_list, 'left', 0, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
test_matrix[0][2][1] = 'xyz' | |
test_matrix[0][2][0] = 'jump2' | |
result = jump_node_check(test_matrix, test_node_list, 'left', 0, test_nw_name) | |
self.assertTrue(len(result) == 0) | |
test_matrix[0][1][2] = 'jump3' | |
result = jump_node_check(test_matrix, test_node_list, 'up', 0, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
test_matrix[0][1][2] = 'xyz' | |
test_matrix[0][0][2] = 'jump3' | |
result = jump_node_check(test_matrix, test_node_list, 'up', 0, test_nw_name) | |
self.assertTrue(len(result) == 0) | |
test_matrix[0][3][2] = 'jump4' | |
result = jump_node_check(test_matrix, test_node_list, 'down', 0, test_nw_name) | |
self.assertTrue(len(result) > 0) | |
test_matrix[0][3][2] = 'xyz' | |
test_matrix[0][4][2] = 'jump4' | |
result = jump_node_check(test_matrix, test_node_list, 'down', 0, test_nw_name) | |
self.assertTrue(len(result) == 0) | |
print() | |
return |
The primary reason is that the web application does not exit with an error code when a violation takes place but merely displays the errors to the user on the web browser. So the check is for the result of the methods to be a non null list that would be an error message.
One major advantage of testing was that I found myself investigating code and thinking of possible ways that it could fail. It is a completely different form of coding as opposed to regular development where you are just trying to get the code to work. In testing, you are thinking of ways to break the code. This is a totally different aspect to development that can be fun if you look at it the right way.