Slide 1

Slide 1 text

T o
 
 or not to
 
 that is the question MO CK MO CK

Slide 2

Slide 2 text

T o
 
 or not to
 
 that is the question MO CK MO CK

Slide 3

Slide 3 text

@anabalica a treatise narrated by
 
 from Potato of Londontowne. ANA BALICA

Slide 4

Slide 4 text

Thou 
 shalt 
 write 
 tests. “ ” William Shakespeare

Slide 5

Slide 5 text

Chapter one

Slide 6

Slide 6 text

Mocks simulate
 the looks and behaviour 
 of real objects

Slide 7

Slide 7 text

REAL fig.1 fig.2 MOCK

Slide 8

Slide 8 text

mocks != stubs

Slide 9

Slide 9 text

✓ Setup ✓ Test ✓ Verify state ✓ Teardown ✓ Setup ✓ Setup expectations ✓ Test ✓ Verify expectations ✓ Verify state ✓ Teardown Stubs Mocks

Slide 10

Slide 10 text

unittest.mock # Python 3.3 mock # Python 2.x

Slide 11

Slide 11 text

Mock()

Slide 12

Slide 12 text

Mock >>> from mock import Mock >>> m = Mock() >>> m.foo = 1 >>> m.foo 1 >>> m.bar

Slide 13

Slide 13 text

Mock() MagicMock()

Slide 14

Slide 14 text

__lt__ __gt__ __len__ __iter__ __bool__ __str__ __int__ __hash__ __exit__ __sizeof__

Slide 15

Slide 15 text

Mock() MagicMock() patch()

Slide 16

Slide 16 text

from mock import patch with patch('rainbow.Pony') as MockPony: MockPony.return_value = 42 Patching

Slide 17

Slide 17 text

Patching 'rainbow.Pony' # creatures.py class Pony: pass # rainbow.py from creatures import Pony pony = Pony()

Slide 18

Slide 18 text

Mock the object where it’s used, 
 not where it came from

Slide 19

Slide 19 text

Chapter two

Slide 20

Slide 20 text

Good mocks

Slide 21

Slide 21 text

with patch.dict('os.environ', {'ANDROID_ARGUMENT': ''}): pf = Platform() self.assertTrue(pf == ‘android') os.environ System calls

Slide 22

Slide 22 text

@mock.patch('sys.stdout', new_callable=six.StringIO) def test_print_live_refs_empty(self, stdout): trackref.print_live_refs() self.assertEqual(stdout.getvalue(), 'Live References\n\n\n') sys.stdout Streams

Slide 23

Slide 23 text

request.urlopen @patch('django.utils.six.moves.urllib.request.urlopen') def test_oembed_photo_request(self, urlopen): urlopen.return_value = self.dummy_response result = wagtail_oembed("http://www.youtube.com/watch/") self.assertEqual(result['type'], 'photo') Networking

Slide 24

Slide 24 text

@patch.object(DataLoader, '_get_file_contents') def test_parse_json_from_file(self, mock_def): mock_def.return_value = ('{"a": 1, "b": 2, "c": 3}', True) output = self._loader.load_from_file('dummy_json.txt') self.assertEqual(output, dict(a=1, b=2, c=3)) json.loads IO operations

Slide 25

Slide 25 text

@mock.patch('time.sleep') def test_500_retry(self, sleep_mock): self.set_http_response(status_code=500) # Create a bucket, a key and a file with self.assertRaises(BotoServerError): k.send_file(fail_file) Clocks, time, timezones time.sleep

Slide 26

Slide 26 text

with mock.patch('random.random', return_value=0.0): with self.assertChanges(get_timeline_size, before=10, after=5): backend.add(timeline, next(self.records)) random.random Unpredictable results

Slide 27

Slide 27 text

✓ System calls ✓ Streams ✓ Networking ✓ IO operations ✓ Clocks, time, timezones ✓ Unpredictable results

Slide 28

Slide 28 text

✓ Save time ✓ Make impossible possible ✓ Exclude external dependencies Why we like them

Slide 29

Slide 29 text

Chapter three

Slide 30

Slide 30 text

mocks Bad

Slide 31

Slide 31 text

with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertTrue(saved_pony.age, 3) mock_save.assert_called_once() Problems?

Slide 32

Slide 32 text

with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertTrue(saved_pony.age, 3) mock_save.assert_called_once() Problems?

Slide 33

Slide 33 text

with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertEqual(saved_pony.age, 3) mock_save.assert_called_once() Problems?

Slide 34

Slide 34 text

with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertEqual(saved_pony.age, 3) mock_save.assert_called_once() Problems?

Slide 35

Slide 35 text

with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertEqual(saved_pony.age, 3) mock_save.assert_called_twice() Problems?

Slide 36

Slide 36 text

with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertEqual(saved_pony.age, 3) mock_save.make_me_sandwich() Problems?

Slide 37

Slide 37 text

¯\_(ϑ)_/¯

Slide 38

Slide 38 text

Solution 1 with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertEqual(saved_pony.age, 3) mock_save.assert_called_once_with()

Slide 39

Slide 39 text

Solution 2 with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertEqual(saved_pony.age, 3) self.assertEqual(mock_save.call_count, 1)

Slide 40

Slide 40 text

Failure with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertEqual(saved_pony.age, 3) self.assertEqual(mock_save.sandwich_count, 1)

Slide 41

Slide 41 text

Solution 3 Test Driven Development

Slide 42

Slide 42 text

Problems? with patch.object(Pony, 'save_base') as mock_save: form = PonyForm(instance=pony, data=data) self.assertTrue(form.is_valid()) saved_pony = form.save() self.assertEqual(saved_pony.age, 3) self.assertEqual(mock_save.call_count, 1)

Slide 43

Slide 43 text

Tests pass? SHIP IT!

Slide 44

Slide 44 text

Maybe 
 it’s incomplete?

Slide 45

Slide 45 text

No content

Slide 46

Slide 46 text

Integration tests c = Client() response = c.post('/pony/', {'age': 1}) self.assertEqual(response.status_code, 201)

Slide 47

Slide 47 text

Unit tests Integration tests Mocks

Slide 48

Slide 48 text

Only mock types 
 that you own

Slide 49

Slide 49 text

Building on
 Third-Party
 Code

Slide 50

Slide 50 text

Adapter
 layer 3rd party API Application
 objects

Slide 51

Slide 51 text

Adapter
 layer 3rd party API Application
 objects Test this cluster Mock this

Slide 52

Slide 52 text

Conclusions

Slide 53

Slide 53 text

Mocks
 can be
 dangerous

Slide 54

Slide 54 text

Passing faulty tests give a 
 false sense of security

Slide 55

Slide 55 text

The end