Skip to content

Commit b274ce7

Browse files
authored
Merge pull request #1 from gregmalcolm/master
merge the head
2 parents abc65a0 + 53df61e commit b274ce7

Some content is hidden

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

57 files changed

+671
-380
lines changed

.travis.yml

+2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,8 @@ python:
44
- 2.7
55
- 3.2
66
- 3.3
7+
- 3.4
8+
- 3.5
79

810
script:
911
- PYTHON_VER=`python -c 'import sys; print(sys.version_info[0])'`

README.rst

+32-1
Original file line numberDiff line numberDiff line change
@@ -63,7 +63,7 @@ You can download Python from here:
6363
http://www.python.org/download
6464

6565
After installing Python make sure the folder containing the python executable
66-
is in the system path. In other words, you need to be able to be able to run
66+
is in the system path. In other words, you need to be able to run
6767
Python from a command console. With Python 2 it will be called `python`
6868
or `python.exe` depending on the operating system. For Python 3 it will either
6969
be `python3` or for windows it will be `python.exe`.
@@ -124,6 +124,37 @@ fire up the command line, recreate the scenario and run queries:
124124

125125
.. image:: http://i442.photobucket.com/albums/qq150/gregmalcolm/DebuggingPython.png
126126

127+
Sniffer Support
128+
---------------
129+
130+
Sniffer allows you to run the tests continuously. If you modify any files files
131+
in the koans directory, it will rerun the tests.
132+
133+
To set this up, you need to install sniffer::
134+
135+
$ pip install sniffer
136+
137+
You should also run one of these libraries depending on your system. This will
138+
automatically trigger sniffer when a file changes, otherwise sniffer will have
139+
to poll to see if the files have changed.
140+
141+
On Linux::
142+
143+
$ pip install pyinotify
144+
145+
On Windows::
146+
147+
$ pip install pywin32
148+
149+
On Mac OS X::
150+
151+
$ pip install MacFSEvents
152+
153+
Once it is set up, you just run::
154+
155+
$ sniffer
156+
157+
Just modify one of the koans files and you'll see that the tests are triggered automatically. Sniffer is controlled by `scent.py`
127158

128159
Getting the Most From the Koans
129160
-------------------------------

python2/_runner_tests.py

+6
Original file line numberDiff line numberDiff line change
@@ -7,14 +7,20 @@
77
from runner.runner_tests.test_mountain import TestMountain
88
from runner.runner_tests.test_sensei import TestSensei
99
from runner.runner_tests.test_helper import TestHelper
10+
from runner.runner_tests.test_path_to_enlightenment import TestFilterKoanNames
11+
from runner.runner_tests.test_path_to_enlightenment import TestKoansSuite
12+
1013

1114
def suite():
1215
suite = unittest.TestSuite()
1316
suite.addTests(unittest.TestLoader().loadTestsFromTestCase(TestMountain))
1417
suite.addTests(unittest.TestLoader().loadTestsFromTestCase(TestSensei))
1518
suite.addTests(unittest.TestLoader().loadTestsFromTestCase(TestHelper))
19+
suite.addTests(unittest.TestLoader().loadTestsFromTestCase(TestFilterKoanNames))
20+
suite.addTests(unittest.TestLoader().loadTestsFromTestCase(TestKoansSuite))
1621
return suite
1722

23+
1824
if __name__ == '__main__':
1925
res = unittest.TextTestRunner(verbosity=2).run(suite())
2026
sys.exit(not res.wasSuccessful())

python2/koans.txt

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
# Lines starting with # are ignored.
2+
koans.about_asserts.AboutAsserts
3+
koans.about_strings.AboutStrings
4+
koans.about_none.AboutNone
5+
koans.about_lists.AboutLists
6+
koans.about_list_assignments.AboutListAssignments
7+
koans.about_dictionaries.AboutDictionaries
8+
koans.about_string_manipulation.AboutStringManipulation
9+
koans.about_tuples.AboutTuples
10+
koans.about_methods.AboutMethods
11+
koans.about_control_statements.AboutControlStatements
12+
koans.about_true_and_false.AboutTrueAndFalse
13+
koans.about_sets.AboutSets
14+
koans.about_triangle_project.AboutTriangleProject
15+
koans.about_exceptions.AboutExceptions
16+
koans.about_triangle_project2.AboutTriangleProject2
17+
koans.about_iteration.AboutIteration
18+
koans.about_comprehension.AboutComprehension
19+
koans.about_generators.AboutGenerators
20+
koans.about_lambdas.AboutLambdas
21+
koans.about_scoring_project.AboutScoringProject
22+
koans.about_classes.AboutClasses
23+
koans.about_new_style_classes.AboutNewStyleClasses
24+
koans.about_with_statements.AboutWithStatements
25+
koans.about_monkey_patching.AboutMonkeyPatching
26+
koans.about_dice_project.AboutDiceProject
27+
koans.about_method_bindings.AboutMethodBindings
28+
koans.about_decorating_with_functions.AboutDecoratingWithFunctions
29+
koans.about_decorating_with_classes.AboutDecoratingWithClasses
30+
koans.about_inheritance.AboutInheritance
31+
koans.about_multiple_inheritance.AboutMultipleInheritance
32+
koans.about_scope.AboutScope
33+
koans.about_modules.AboutModules
34+
koans.about_packages.AboutPackages
35+
koans.about_class_attributes.AboutClassAttributes
36+
koans.about_attribute_access.AboutAttributeAccess
37+
koans.about_deleting_objects.AboutDeletingObjects
38+
koans.about_proxy_object_project.AboutProxyObjectProject
39+
koans.about_proxy_object_project.TelevisionTest
40+
koans.about_extra_credit.AboutExtraCredit
41+
koans.about_regex.AboutRegex

python2/koans/about_asserts.py

+5-5
Original file line numberDiff line numberDiff line change
@@ -15,13 +15,13 @@ def test_assert_truth(self):
1515
#
1616
# http://bit.ly/about_asserts
1717

18-
self.assertTrue(False) # This should be true
18+
self.assertTrue(False) # This should be True
1919

2020
def test_assert_with_message(self):
2121
"""
2222
Enlightenment may be more easily achieved with appropriate messages.
2323
"""
24-
self.assertTrue(False, "This should be true -- Please fix this")
24+
self.assertTrue(False, "This should be True -- Please fix this")
2525

2626
def test_fill_in_values(self):
2727
"""
@@ -62,17 +62,17 @@ def test_that_sometimes_we_need_to_know_the_class_type(self):
6262

6363
# Sometimes we will ask you what the class type of an object is.
6464
#
65-
# For example, contemplate the text string "naval". What is its class type?
65+
# For example, contemplate the text string "navel". What is its class type?
6666
# The koans runner will include this feedback for this koan:
6767
#
6868
# AssertionError: '-=> FILL ME IN! <=-' != <type 'str'>
6969
#
70-
# So "naval".__class__ is equal to <type 'str'>? No not quite. This
70+
# So "navel".__class__ is equal to <type 'str'>? No not quite. This
7171
# is just what it displays. The answer is simply str.
7272
#
7373
# See for yourself:
7474

75-
self.assertEqual(__, "naval".__class__) # It's str, not <type 'str'>
75+
self.assertEqual(__, "navel".__class__) # It's str, not <type 'str'>
7676

7777
# Need an illustration? More reading can be found here:
7878
#

python2/koans/about_attribute_access.py

-4
Original file line numberDiff line numberDiff line change
@@ -105,10 +105,6 @@ def __init__(self):
105105
self.no_of_getattribute_calls = 0
106106

107107
def __getattribute__(self, attr_name):
108-
#Uncomment for debugging info:
109-
#print 'Debug __getattribute__(' + type(self).__name__ + \
110-
# "." + attr_name + ") dict=" + str(self.__dict__)
111-
112108
# We need something that is outside the scope of this class:
113109
global stack_depth
114110
stack_depth += 1

python2/koans/about_comprehension.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@ def test_creating_a_set_with_set_comprehension(self):
5050

5151
def test_creating_a_dictionary_with_dictionary_comprehension(self):
5252
dict_of_weapons = {'first': 'fear', 'second': 'surprise',
53-
'third':'ruthless efficiency', 'forth':'fanatical devotion',
53+
'third':'ruthless efficiency', 'fourth':'fanatical devotion',
5454
'fifth': None}
5555

5656
dict_comprehension = { k.upper(): weapon for k, weapon in dict_of_weapons.iteritems() if weapon}

python2/koans/about_control_statements.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,11 @@ def test_if_then_statements(self):
1818
if True:
1919
result = 'true value'
2020
self.assertEqual(__, result)
21-
21+
2222
def test_if_then_elif_else_statements(self):
2323
if False:
2424
result = 'first value'
25-
elif True:
25+
elif True:
2626
result = 'true value'
2727
else:
2828
result = 'default value'
@@ -66,7 +66,7 @@ def test_for_statement_with_tuples(self):
6666
("Lancelot", "Blue"),
6767
("Galahad", "I don't know!"),
6868
("Robin", "Blue! I mean Green!"),
69-
("Arthur", "Is that an African Swallow or Amazonian Swallow?")
69+
("Arthur", "Is that an African Swallow or European Swallow?")
7070
]
7171
result = []
7272
for knight, answer in round_table:

python2/koans/about_generators.py

+7-7
Original file line numberDiff line numberDiff line change
@@ -43,8 +43,8 @@ def test_generator_expressions_are_a_one_shot_deal(self):
4343
attempt1 = list(dynamite)
4444
attempt2 = list(dynamite)
4545

46-
self.assertEqual(__, list(attempt1))
47-
self.assertEqual(__, list(attempt2))
46+
self.assertEqual(__, attempt1)
47+
self.assertEqual(__, attempt2)
4848

4949
# ------------------------------------------------------------------
5050

@@ -60,7 +60,7 @@ def test_generator_method_will_yield_values_during_iteration(self):
6060
result.append(item)
6161
self.assertEqual(__, result)
6262

63-
def test_coroutines_can_take_arguments(self):
63+
def test_generators_can_be_manually_iterated_and_closed(self):
6464
result = self.simple_generator_method()
6565
self.assertEqual(__, next(result))
6666
self.assertEqual(__, next(result))
@@ -91,12 +91,12 @@ def test_generator_keeps_track_of_local_variables(self):
9191

9292
# ------------------------------------------------------------------
9393

94-
def generator_with_coroutine(self):
94+
def coroutine(self):
9595
result = yield
9696
yield result
9797

98-
def test_generators_can_take_coroutines(self):
99-
generator = self.generator_with_coroutine()
98+
def test_generators_can_act_as_coroutines(self):
99+
generator = self.coroutine()
100100

101101
# THINK ABOUT IT:
102102
# Why is this line necessary?
@@ -108,7 +108,7 @@ def test_generators_can_take_coroutines(self):
108108
self.assertEqual(__, generator.send(1 + 2))
109109

110110
def test_before_sending_a_value_to_a_generator_next_must_be_called(self):
111-
generator = self.generator_with_coroutine()
111+
generator = self.coroutine()
112112

113113
try:
114114
generator.send(1 + 2)

python2/koans/about_lists.py

+1
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,7 @@ def test_lists_and_ranges(self):
5858
self.assertEqual(__, range(5, 9))
5959

6060
def test_ranges_with_steps(self):
61+
self.assertEqual(__, range(5, 3, -1))
6162
self.assertEqual(__, range(0, 8, 2))
6263
self.assertEqual(__, range(1, 8, 3))
6364
self.assertEqual(__, range(5, -7, -4))

python2/koans/about_modules.py

+5-2
Original file line numberDiff line numberDiff line change
@@ -60,8 +60,11 @@ def test_private_attributes_are_still_accessible_in_modules(self):
6060
# module level attribute hiding doesn't affect class attributes
6161
# (unless the class itself is hidden).
6262

63-
def test_a_modules_XallX_statement_limits_what_wildcards_will_match(self):
64-
"""Examine results of from local_module_with_all_defined import *"""
63+
def test_a_module_can_limit_wildcard_imports(self):
64+
"""
65+
Examine results of:
66+
from local_module_with_all_defined import *
67+
"""
6568

6669
# 'Goat' is on the __all__ list
6770
goat = Goat()

python2/koans/about_new_style_classes.py

+8-4
Original file line numberDiff line numberDiff line change
@@ -7,14 +7,18 @@
77
class AboutNewStyleClasses(Koan):
88
class OldStyleClass:
99
"An old style class"
10-
# Original class style have been phased out in Python 3.
10+
# Original style classes from Python 2.1 and earlier, also known as
11+
# "classic classes", have been phased out in Python 3.
1112

1213
class NewStyleClass(object):
1314
"A new style class"
14-
# Introduced in Python 2.2
15+
# Introduced in Python 2.2 to unify types and classes.
16+
#
17+
# If you want to learn more, see:
18+
# https://www.python.org/download/releases/2.2.3/descrintro/
1519
#
1620
# Aside from this set of tests, Python Koans sticks exclusively to this
17-
# kind of class
21+
# kind of class.
1822
pass
1923

2024
def test_new_style_classes_inherit_from_object_base_class(self):
@@ -46,7 +50,7 @@ def test_old_style_classes_have_type_but_no_class_attribute(self):
4650

4751
def test_new_style_classes_have_same_class_as_type(self):
4852
new_style = self.NewStyleClass()
49-
self.assertEqual(__, self.NewStyleClass.__class__)
53+
self.assertEqual(__, self.NewStyleClass.__class__.__name__)
5054
self.assertEqual(
5155
__,
5256
type(self.NewStyleClass) == self.NewStyleClass.__class__)

python2/koans/about_packages.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ def test_use_absolute_imports_to_import_upper_level_modules(self):
5353

5454
self.assertEqual(__, contemplate_koans.__name__)
5555

56-
# contemplate_koans.py is the root module in this package because its
56+
# contemplate_koans.py is the root module in this package because it's
5757
# the first python module called in koans.
5858
#
5959
# If contemplate_koans.py was based in a_package_folder that would be

python2/koans/about_triangle_project2.py

+8-2
Original file line numberDiff line numberDiff line change
@@ -11,9 +11,15 @@ class AboutTriangleProject2(Koan):
1111
# The first assignment did not talk about how to handle errors.
1212
# Let's handle that part now.
1313
def test_illegal_triangles_throw_exceptions(self):
14-
# Calls triangle(0, 0, 0)
14+
# In the code below, each line calls the specfied method with the arguments passed to it.
15+
# E.g. this line:
16+
# self.assertRaises(TriangleError, triangle, 0, 0, 0)
17+
# calls triangle(0, 0, 0)
18+
19+
# All sides should be greater than 0
1520
self.assertRaises(TriangleError, triangle, 0, 0, 0)
16-
1721
self.assertRaises(TriangleError, triangle, 3, 4, -5)
22+
23+
# The sum of any two sides should be greater than the third one
1824
self.assertRaises(TriangleError, triangle, 1, 1, 3)
1925
self.assertRaises(TriangleError, triangle, 2, 5, 2)

python2/koans/about_true_and_false.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,8 @@ def test_blank_strings_are_treated_as_false(self):
3434

3535
def test_everything_else_is_treated_as_true(self):
3636
self.assertEqual(__, self.truth_value(1))
37-
self.assertEqual(__, self.truth_value(1,))
37+
self.assertEqual(__, self.truth_value([0]))
38+
self.assertEqual(__, self.truth_value((0,)))
3839
self.assertEqual(
3940
__,
4041
self.truth_value("Python is named after Monty Python"))

python2/run.bat

+10
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,9 @@ REM Set this to your python folder:
88
SET PYTHON_PATH=C:\Python27
99

1010
set SCRIPT=
11+
12+
:loop
13+
1114
REM Hunt around for python
1215
IF EXIST "python.exe" (
1316
SET SCRIPT=%RUN_KOANS%
@@ -32,4 +35,11 @@ IF NOT "" == "%SCRIPT%" (
3235
echo python.exe contemplate_koans.py
3336
pause
3437
)
38+
39+
Set /p keepgoing="Test again? Y or N - "
40+
if "%keepgoing%" == "y" (
41+
goto loop
42+
)
43+
44+
3545
:end

python2/runner/mountain.py

+3-2
Original file line numberDiff line numberDiff line change
@@ -8,16 +8,17 @@
88
from sensei import Sensei
99
from writeln_decorator import WritelnDecorator
1010

11+
1112
class Mountain:
1213
def __init__(self):
1314
self.stream = WritelnDecorator(sys.stdout)
1415
self.tests = path_to_enlightenment.koans()
1516
self.lesson = Sensei(self.stream)
1617

1718
def walk_the_path(self, args=None):
18-
"Run the koans tests with a custom runner output."
19+
"""Run the koans tests with a custom runner output."""
1920

20-
if args and len(args) >=2:
21+
if args and len(args) >= 2:
2122
args.pop(0)
2223
test_names = ["koans." + test_name for test_name in args]
2324
self.tests = unittest.TestLoader().loadTestsFromNames(test_names)

0 commit comments

Comments
 (0)