Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Conditional mocking: Call original function if condition does match

Tags:

python

mocking

How can I conditionally call the original method in a mock?

In this example I only want to fake a return value if bar=='x'. Otherwise I want to call the original method.

def mocked_some_method(bar):     if bar=='x':         return 'fake'     return some_how_call_original_method(bar)  with mock.patch('mylib.foo.some_method', mocked_some_method):     do_some_stuff()      

I know that it is a bit strange. If I want to fake mylib.foo.some_method in side do_some_stuff() it should be condition-less. All (not some) calls to some_method should be mocked.

In my case it is an integration test, not a s tiny unittest and mylib.foo.some_method is a kind of dispatcher which gets used very often. And in one case I need to fake the result.

Update

I wrote this question four years ago. Today, it feels very strange to do conditional mocking. Mocks should only get used in tests. Tests (and production code) should be simple and small. Tests should be conditionless. As I wrote this question, we still used huge production methods and long test. Today, I follow these rules (simple methods, conditionless tests ...). I wrote my findings down: my programming guidelines

like image 717
guettli Avatar asked Apr 10 '15 13:04

guettli


Video Answer


1 Answers

If you need just replace behavior without care of mock's calls assert function you can use new argument; otherwise you can use side_effect that take a callable.

I guess that some_method is a object method (instead of a staticmethod) so you need a reference its object to call it. Your wrapper should declare as first argument the object and your patch use autospec=True to use the correct signature for side_effect case.

Final trick is save the original method reference and use it to make the call.

orig = mylib.foo.some_method def mocked_some_method(self, bar):     if bar=='x':         return 'fake'     return orig(self, bar)  #Just replace: with mock.patch('mylib.foo.some_method', new=mocked_some_method):     do_some_stuff()  #Replace by mock with mock.patch('mylib.foo.some_method', side_effect=mocked_some_method, autospec=True) as mock_some_method:     do_some_stuff()     assert mock_some_method.called 
like image 198
Michele d'Amico Avatar answered Sep 20 '22 06:09

Michele d'Amico