Logo

dev-resources.site

for different kinds of informations.

A tale of tests, greenlets and patches

Published at
4/16/2021
Categories
python
testdev
gevent
patch
Author
rucamzu
Categories
4 categories in total
python
open
testdev
open
gevent
open
patch
open
Author
7 person written this
rucamzu
open
A tale of tests, greenlets and patches

Once upon a working day there was some Python code I needed to update. The changes required were unremarkably clear, however some unit tests managed to hit me good enough with surprise.

Green(let) pastures

I needed to verify that the system under test called its dependencies on a certain order, and the way I decided to do this was by setting assertions up as side effects on the mocked dependencies:

def test_execution_order():
    dependency1 = mock.Mock()
    dependency2 = mock.Mock()

    # dependency1.foo() must be called before dependency2.foo()
    dependency1.foo.side_effect = dependency2.foo.assert_not_called
    dependency2.foo.side_effect = dependency1.foo.assert_called_once

    sut(dependency1, dependency2)
Enter fullscreen mode Exit fullscreen mode

This test would have passed/failed just fine on a plain, synchronous implementation of the system under test:

# right order, the test passes
def sut(dependency1, dependency2):
    dependency1.foo()
    dependency2.foo()
Enter fullscreen mode Exit fullscreen mode
# wrong order, the test fails
def sut(dependency1, dependency2):
    dependency2.foo()
    dependency1.foo()
Enter fullscreen mode Exit fullscreen mode

However, the actual implementation made fairly heavy use of gevent to execute the dependencies in separate greenlets, then wait for them to complete:

# right order, the test passes
def sut(dependency1, dependency2):
    gevent.joinall([gevent.spawn(dependency1.foo)])
    gevent.joinall([gevent.spawn(dependency2.foo)])
Enter fullscreen mode Exit fullscreen mode
# wrong order, the test... passes!?
def sut(dependency1, dependency2):
    gevent.joinall([gevent.spawn(dependency2.foo)])
    gevent.joinall([gevent.spawn(dependency1.foo)])
Enter fullscreen mode Exit fullscreen mode

Here came the puzzling bit as the test passed in both cases.

All the cats join in

The test output showed assertion exceptions when run against the bad implementation, which at least reassured me on the approach taken. Still, the test was passing nonetheless.

I went back to the gevent documentation and soon the reason behind revealed itself:

joinall(greenlets, timeout=None, raise_error=False, count=None)
Enter fullscreen mode Exit fullscreen mode

Notice the raise_error=False bit? gevent.joinall waits for a set of greenlets to complete, but by default does not bubble up any exceptions raised from within the awaited greenlets - including the failed assertions.

Unfortunately, changing the production code to call gevent.joinall with raise_error=True was not an option.

Patchwork

When one door closes, another opens, and the one opening here would allow me to patch the behavior of gevent.joinall in the test to force the raise_error=True parameter on it.

The first attempt naively fell into a RecursionError as the patched function ended up calling itself ad infinitum:

def gevent_joinall_raising_errors(*args, **kwargs):
    kwargs['raise_error'] = True
    return gevent.joinall(*args, **kwargs)  # calls the patched function (itself)
Enter fullscreen mode Exit fullscreen mode
@patch('gevent.joinall', gevent_joinall_raising_errors)
def test_execution_order():
    ...
Enter fullscreen mode Exit fullscreen mode

Second time was the charm, though. The final solution involved capturing the original, non-patched gevent.joinall function inside a nested scope:

def gevent_joinall_with(**kwargs_):
    gevent_joinall = gevent.joinall

    def gevent_joinall_wrapper(*args, **kwargs):
        kwargs.update(kwargs_)
        return gevent_joinall(*args, **kwargs)  # calls the non-patched function

    return gevent_joinall_wrapper
Enter fullscreen mode Exit fullscreen mode
@patch('gevent.joinall', gevent_joinall_with(raise_error=True))
def test_execution_order():
    ...
Enter fullscreen mode Exit fullscreen mode

In the end

This is one of those cases where I could have easily written a test, seen it pass and moved on. I am very glad that over the years many authors have reinforced on me the idea of always asking a test to fail before accepting it passing.

A remark on unittest.mock.patch. By definition, patching the system under test changes its behavior, which is inherently dangerous as we might not end up testing the logic we mean to anymore. In this scenario, I was sufficiently assured that the patch would not affect the behavior of my production code beyond allowing assertion exceptions up.

patch Article's
30 articles in total
Favicon
Stop Worrying About EC2 Patching – Automate It Like a Pro!
Favicon
Oracle Patch Update: Securing Your Database and Applications
Favicon
How to Manage the Oracle Critical Patch Update
Favicon
Critical Insights About Oracle Critical Patch Update: Enhancing System Security and Functionality
Favicon
A Guide to Oracle Critical Patch Update
Favicon
Key Considerations And Benefits Of Choosing A Testing Solution For Businesses
Favicon
All About Oracle Critical Patch Update
Favicon
How to Apply a Magento 2 Patch
Favicon
How to use LLM for efficient text outputs longer than 4k tokens?
Favicon
How to Leverage Flawless Integration of Oracle Patch Updates in Your System?
Favicon
A Comprehensive Guide to Oracle's Latest Critical Patch Updates
Favicon
Ensure Security in Financial Institutions After Oracle Patch Update Release
Favicon
Oracle Critical Patch Update: Ensuring Software Security
Favicon
Achieving Harmony: Balancing Oracle Patch Testing’s Cost, Time, and Risk
Favicon
Zero-day: Server Message Block (SMB) Server in Linux Kernel 5.15 Has a Critical Vulnerability!
Favicon
How Critical Are Oracle Critical Patch Updates?
Favicon
Maintain High-Quality Oracle Patch Update With Automation Testing
Favicon
Ensuring Security Using Oracle Critical Patch Update and Cloud Testing
Favicon
Oracle Patch Updates: Navigating the Future with Confidence and Innovation
Favicon
Unveiling The Power Of Oracle’s Critical Patch Update
Favicon
Seamless Oracle Patch Testing: Empowering Efficiency
Favicon
Oracle Patch Update: Enhancing Performance and Security
Favicon
Open class
Favicon
Oracle’s Critical Patch Update April 2023 Readiness
Favicon
Patch me up.
Favicon
A tale of tests, greenlets and patches
Favicon
HTTP Request Methods (aka HTTP Verbs)
Favicon
Sharing your git patches
Favicon
Fetch() in JavaScript pt. II, Revenge of the options
Favicon
Run Windows update now!

Featured ones: