Skip to content

Commit f9cd444

Browse files
committed
doc update -- includes wiki articles
1 parent 1a2ff2a commit f9cd444

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

44 files changed

+2867
-594
lines changed

Makefile

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
TEST_FILE := tests/run_all.py
2+
ALL_TESTS = $(patsubst %.py,%,$(wildcard tests/test_*.py)) # e.g. tests/test_content tests/...
3+
4+
# Run the test of a single file
5+
test: export PYTHONWARNINGS=ignore
6+
test: $(TEST_FILE)
7+
-python $^
8+
9+
# Run submake for testing each test file
10+
test_parallel: $(ALL_TESTS)
11+
12+
$(ALL_TESTS)::
13+
@$(MAKE) test TEST_FILE="$@.py"

docs/environment.yml

+2
Original file line numberDiff line numberDiff line change
@@ -3,3 +3,5 @@ dependencies:
33
- python=3.5.1=0
44
- sphinx>1.4.0
55
- sphinx_rtd_theme>=0.1.9
6+
- pip:
7+
- recommonmark>=0.4.0

docs/source/conf.py

+9
Original file line numberDiff line numberDiff line change
@@ -299,6 +299,15 @@
299299

300300
add_module_names = False
301301

302+
from recommonmark.transform import AutoStructify
303+
def setup(app):
304+
github_doc_root = 'https://github.com/datacamp/pythonwhat/blob/master/docs/source/'
305+
app.add_config_value('recommonmark_config', {
306+
'url_resolver': lambda url: github_doc_root + url,
307+
'auto_toc_tree_section': 'Contents',
308+
}, True)
309+
app.add_transform(AutoStructify)
310+
302311
on_rtd = os.environ.get('READTHEDOCS', None) == 'True'
303312
if not on_rtd: # only import and set the theme if we're building docs locally
304313
import sphinx_rtd_theme
File renamed without changes.

docs/source/index.rst

+14-6
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,21 @@ Contents:
88
:maxdepth: 2
99

1010
Home
11-
quickstart_guide.md
12-
simple_tests.md
13-
parts_cheatsheet.rst
14-
Processes
15-
logic
11+
quickstart_guide
12+
simple_tests/index.rst
13+
part_checks
14+
expression_tests
15+
logic_tests/index.rst
1616
spec2_summary
17-
test_functions
17+
18+
19+
Pythonwhat V1
20+
-------------
21+
22+
.. toctree::
23+
24+
pythonwhat.wiki/index.rst
25+
1826

1927
Indices and tables
2028
==================

docs/source/logic.md

-135
This file was deleted.

docs/source/logic_tests/index.rst

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
Logic Tests
2+
===========
3+
4+
.. toctree::
5+
:maxdepth: 2
6+
7+
test_correct
8+
test_or
9+
test_not
10+
misc

docs/source/logic_tests/misc.md

+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
Misc
2+
----
3+
4+
[TODO: this is a previous version, will update with spec2 functions]
5+
6+
Several functions, such as `test_correct()`, `test_or()`, `test_not()`, among others, also have arguments that expect another set of tests. This article will explain the different ways of specifying these 'sub-tests'.
7+
8+
Let's take the example of `test_correct()`; this function takes two sets of tests. The first set is executed, and if it's fine, the second set is left alone. If the first set of tests results in an error, the second set is executed and the feedback is logged.
9+
10+
### Example 1
11+
12+
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:
13+
14+
*** =solution
15+
```{python}
16+
# Import numpy and create array
17+
import numpy as np
18+
arr = np.array([1, 2, 3, 4, 5, 6])
19+
20+
# Calculate result
21+
result = np.mean(arr)
22+
```
23+
24+
The first part of the tests here would be to check `result`. If `result` is not correct, you want to check whether `np.mean()` has been called.
25+
26+
The most concise way to do this, is with lambda functions; you specify two sets of tests, that in this case consist of one test each:
27+
28+
*** =sct
29+
```{python}
30+
test_correct(check_object('result').has_equal_value(),
31+
test_function('numpy.mean'))
32+
success_msg("You own numpy!")
33+
```
34+
35+
### Example 2
36+
37+
When writing SCTs for more complicated exercises, you'll probably want to pass along several tests to an argument.
38+
39+
Suppose that you expect the student to create an object `result` once more, but this time it's the sum of calling the `np.mean()` function twice; once on `arr1`, once on `arr2`:
40+
41+
*** =solution
42+
```{python}
43+
# Import numpy and create array
44+
import numpy as np
45+
arr1 = np.array([1, 2, 3, 4, 5, 6])
46+
arr2 = np.array([6, 5, 4, 3, 2, 1])
47+
48+
# Calculate result
49+
result = np.mean(arr) + np.mean(arr)
50+
```
51+
52+
Now, in the 'digging deeper' part of `test_correct()`, you want to check the `np.mean()` function twice.
53+
To do this, you'll want to use a function definition; lambda functions are not practical anymore:
54+
55+
*** =sct
56+
```{python}
57+
diagnose = [test_function('numpy.mean', index = i) for i in [1,2]]
58+
59+
test_correct(test_object('result'), diagnose)
60+
success_msg("You own numpy!")
61+
```
+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
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`.

docs/source/logic_tests/test_not.md

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
test_not
2+
--------

docs/source/logic_tests/test_or.md

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
test_or
2+
-------
3+
4+
```eval_rst
5+
.. automodule:: pythonwhat.test_funcs.test_or
6+
:members:
7+
```
8+
9+
This function simply tests whether one of the SCTs you specify inside it passes. You should pass your SCTs within lambda expressions or custom defined functions (like you do for functions like `test_correct()`, `test_for_loop()`, or `test_function_definition`).
10+
11+
Suppose you want to check whether people correctly printed out any integer between 3 and 7. A solution could be:
12+
13+
*** =solution
14+
```{python}
15+
print(4)
16+
```
17+
18+
To test this in a robust way, you could use `test_output_contains()` with a suitable regular expression that covers everything, or you can use `test_or()` with three separate `test_output_contains()` functions.
19+
20+
*** =sct
21+
```{python}
22+
test_or(lambda: test_output_contains('4'),
23+
lambda: test_output_contains('5'),
24+
lambda: test_output_contains('6'))
25+
success_msg("Nice job!")
26+
```
27+
28+
You can consider `test_or()` a logic-inducing function. The different calls to `pythonwhat` functions that are in your SCT are actually all tests that _have_ to pass: they are `AND` tests. With `test_or()` you can add chunks of `OR` tests in there.

0 commit comments

Comments
 (0)