Slide 1

Slide 1 text

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

Slide 2

Slide 2 text

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

Slide 3

Slide 3 text

Thou 
 shalt 
 write 
 tests. “ ” William Shakespeare

Slide 4

Slide 4 text

Chapter one

Slide 5

Slide 5 text

Mocks simulate
 the looks and behaviour 
 of real objects

Slide 6

Slide 6 text

REAL fig.1 fig.2 MOCK

Slide 7

Slide 7 text

mocks != stubs

Slide 8

Slide 8 text

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

Slide 9

Slide 9 text

Mock()

Slide 10

Slide 10 text

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

Slide 11

Slide 11 text

Mock() MagicMock()

Slide 12

Slide 12 text

__lt__ __gt__ __len__ __iter__ __bool__ __str__ __int__ __hash__ __exit__ __sizeof__

Slide 13

Slide 13 text

Mock() MagicMock() patch()

Slide 14

Slide 14 text

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

Slide 15

Slide 15 text

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

Slide 16

Slide 16 text

Chapter two

Slide 17

Slide 17 text

Good mocks

Slide 18

Slide 18 text

System calls with mock.patch.dict("foo.os.environ", {"LOCAL_ENV": "production"}): conf = Configuration() self.assertEqual(conf.env, "production") os.environ

Slide 19

Slide 19 text

Streams @mock.patch('sys.stdout', new=StringIO) def test_silly_logger(self, mock_stdout): logger.print_refs() self.assertEqual(mock_stdout.getvalue(), 'Some refs') sys.stdout

Slide 20

Slide 20 text

HTTP requests /responses requests.get with mock.patch("foo.requests.get") as mock_get: mock_get.return_value = mock.Mock(status_code=200) status = is_it_down(“http://djangocon.eu") self.assertFalse(status)

Slide 21

Slide 21 text

IO operations m = mock.mock_open(read_data="DjangoCon \o/“) with mock.patch("foo.open", m, create=True): with open("bar.txt") as h: result = h.read() self.assertEqual(result, "DjangoCon \o/") open

Slide 22

Slide 22 text

Clocks, time, timezones from django.utils import timezone tz = pytz.timezone('Asia/Tokyo') dt = datetime.datetime(2016, 1, 1, tzinfo=tz) with mock.patch.object(timezone, 'now', return_value=dt): print timezone.now() timezone.now

Slide 23

Slide 23 text

Unpredictable results with mock.patch('foo.random.random', return_value=0.42): result = throw_dice() self.assertEqual(result, 0.42) random.random

Slide 24

Slide 24 text

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

Slide 25

Slide 25 text

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

Slide 26

Slide 26 text

Chapter three

Slide 27

Slide 27 text

Bad mocks

Slide 28

Slide 28 text

with mock.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 29

Slide 29 text

with mock.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 30

Slide 30 text

Problems? with mock.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()

Slide 31

Slide 31 text

Problems? with mock.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()

Slide 32

Slide 32 text

Problems? with mock.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()

Slide 33

Slide 33 text

Problems? with mock.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_a_sandwich()

Slide 34

Slide 34 text

¯\_(ツ)_/¯

Slide 35

Slide 35 text

Solution 1 with mock.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 36

Slide 36 text

Solution 2 with mock.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 37

Slide 37 text

Failure with mock.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 38

Slide 38 text

Solution 3 Test Driven Development

Slide 39

Slide 39 text

Problems? with mock.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

Tests pass? SHIP IT!

Slide 41

Slide 41 text

Maybe 
 it’s incomplete?

Slide 42

Slide 42 text

No content

Slide 43

Slide 43 text

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

Slide 44

Slide 44 text

Unit tests Functional tests Mocks

Slide 45

Slide 45 text

Conclusions

Slide 46

Slide 46 text

Mocks
 can be
 dangerous

Slide 47

Slide 47 text

Passing faulty tests give a false sense of security

Slide 48

Slide 48 text

The end