|
| 1 | +test_correct |
| 2 | +------------ |
| 3 | + |
| 4 | +```eval_rst |
| 5 | +.. automodule:: pythonwhat.test_funcs.test_correct |
| 6 | + :members: |
| 7 | +``` |
| 8 | + |
| 9 | +A wrapper function around `test_or()`, `test_correct()` allows you to add logic to your SCT. Normally, your SCT is simply a script with subsequent `pythonwhat` function calls, all of which have to pass. `test_correct()` allows you to bypass this: you can specify a "sub-SCT" in the `check` part, that should pass. If these tests pass, the "sub-SCT" in `diagnose` is not executed. If the tests don't pass, the "sub-SCT" in `diagnose` is run, typically to dive deeper into what the error might be and give more specific feedback. |
| 10 | + |
| 11 | +To accomplish this, the lambda function in `check` is executed silently, so that failure will not cause the SCT to stop and generate a feedback message. If the execution passes, all is good and `test_correct()` is abandoned. If it fails, `diagnose` is executed, not silently. If the `diagnose` part fails, the feedback message that it generates is presented to the student. If it passes, the `check` part is executed again, this time not silently, to make sure that a `test_correct()` that contains a failing `check` part leads to a failing SCT. |
| 12 | + |
| 13 | +### Example 1 |
| 14 | + |
| 15 | +As an example, suppose you want the student to calculate the mean of a Numpy array `arr` and store it in `res`. A possible solution could be: |
| 16 | + |
| 17 | + *** =solution |
| 18 | + ```{python} |
| 19 | + # Import numpy and create array |
| 20 | + import numpy as np |
| 21 | + arr = np.array([1, 2, 3, 4, 5, 6]) |
| 22 | + |
| 23 | + # Calculate result |
| 24 | + result = np.mean(arr) |
| 25 | + ``` |
| 26 | + |
| 27 | +You want the SCT to pass when the student manages to store the correct value in the object `result`. How `result` was calculated, does not matter to you: as long as `result` is correct, the SCT should accept the submission. If something about `result` is not correct, you want to dig a little deeper and see if the student used the `np.mean()` function correctly. The following SCT will do just that: |
| 28 | + |
| 29 | + *** =sct |
| 30 | + ```{python} |
| 31 | + test_correct(lambda: test_object('result'), |
| 32 | + lambda: test_function('numpy.mean')) |
| 33 | + success_msg("You own numpy!") |
| 34 | + ``` |
| 35 | + |
| 36 | +Notice that you have to use lambda functions to use Python as a functional programming language. |
| 37 | +Let's go over what happens when the student submits different pieces of code: |
| 38 | + |
| 39 | +- The student submits `result = np.mean(arr)`, exactly the same as the solution. `test_correct()` executes the first lambda function, `test_object('result')`. This test passes, so `test_correct()` is exited and `test_function()` is not executed. The SCT passes. |
| 40 | +- The student submits `result = np.sum(arr) / arr.size`, which also leads to the correct value in `result`. `test_correct()` executes the first lambda function, `test_object('result')`. This test passes, so `test_correct()` is exited and `test_function()` is not executed. So the entire SCT passes even though `np.mean()` was not used. |
| 41 | +- The student submits `result = np.mean(arr + 1)`. `test_correct()` executes the first lambda function, `test_object('result')`. This test fails, so `test_correct()` heads over to second, 'diagnose' lambda function and executes `test_function('numpy.mean')`. This function will fail, because the argument passed to `numpy.mean()` in the student submission does not correspond to the argument passed in the solution. A meaningful, specific feedback message is presented to the student: you did not correctly specify the arguments inside `np.mean()`. |
| 42 | +- The student submits `result = np.mean(arr) + 1`. `test_correct()` executes the first lambda function, `test_object('result')`. This test fails, so `test_correct()` heads over to the second, 'diagnose' lambda function and executes `test_function('numpy.mean'). This function passes, because `np.mean()` is called in exactly the same way in the student code as in the solution. Because there is something wrong - `result` is not correct - the 'check' lambda function, `test_object('result')` is executed again, and this time its feedback on failure is presented to the student. The student gets the message that `result` does not contain the correct value. |
| 43 | + |
| 44 | +### Multiple functions in `diagnose` and `check` |
| 45 | + |
| 46 | +You can also use `test_correct()` with entire 'sub-SCTs' that are composed of several SCT calls. In this case, you have to define an additional function that executes the tests you want to perform in this sub-SCT, and pass this function to `test_correct()`. |
| 47 | + |
| 48 | +### Why to use `test_correct()` |
| 49 | + |
| 50 | +You will find that `test_correct()` is an extremely powerful function to allow for different ways of solving the same problem. You can use `test_correct()` to check the end result of a calculation. If the end result is correct, you can go ahead and accept the entire exercise. If the end result is incorrect, you can use the `diagnose` part of `test_correct()` to dig a little deeper. |
| 51 | + |
| 52 | +It is also perfectly possible to use `test_correct()` inside another `test_correct()`, although things can get funky with the lambda functions in this case. |
| 53 | + |
| 54 | +### Wrapper around `test_or()` |
| 55 | + |
| 56 | +`test_correct()` is a wrapper around `test_or()`. `test_correct(diagnose, check)` is equivalent with: |
| 57 | + |
| 58 | + def diagnose_and_check() |
| 59 | + diagnose() |
| 60 | + check() |
| 61 | + |
| 62 | + test_or(diagnose_and_check, check) |
| 63 | + |
| 64 | +Note that in each of the `test_or` cases here, the submission has to pass the SCTs specified in `check`. |
0 commit comments