eventlet 是否为线程模块做 monkey_patch?

Does eventlet do monkey_patch for threading module?

http://eventlet.net/doc/patching.htm 中的文档说 "If no arguments are specified, everything is patched." 和 "thread, which patches thread, threading, and Queue"。

但是通过简单的测试:

#!/bin/env python

import threading

import eventlet
eventlet.monkey_patch()

if __name__ == '__main__':
    patched = eventlet.patcher.is_monkey_patched(threading)
    print('patched : %s' % patched)

结果是:

patched : False

线程似乎根本没有打补丁。 文档有误?

我发现文档是正确的。问题是关于 is_monkey_patched(),它无法检测到某些情况,例如 'threading, Queue' 模块。看看这个函数的src,行为很容易理解。

def _green_thread_modules():
    from eventlet.green import Queue
    from eventlet.green import thread
    from eventlet.green import threading
    if six.PY2:
        return [('Queue', Queue), ('thread', thread), ('threading', threading)]
    if six.PY3:
        return [('queue', Queue), ('_thread', thread), ('threading', threading)]

    if on['thread'] and not already_patched.get('thread'):
        modules_to_patch += _green_thread_modules()
        already_patched['thread'] = True

def is_monkey_patched(module):
    """Returns True if the given module is monkeypatched currently, False if
    not.  *module* can be either the module itself or its name.

    Based entirely off the name of the module, so if you import a
    module some other way than with the import keyword (including
    import_patched), this might not be correct about that particular
    module."""
    return module in already_patched or \
        getattr(module, '__name__', None) in already_patched

又因为补丁操作是这样实现的:

    for name, mod in modules_to_patch:
        orig_mod = sys.modules.get(name)
        if orig_mod is None:
            orig_mod = __import__(name)
        for attr_name in mod.__patched__:
            patched_attr = getattr(mod, attr_name, None)
            if patched_attr is not None:
                setattr(orig_mod, attr_name, patched_attr)

我们可以检查 threading/Queue 这样的模块是否打了补丁:

 >>>import threading
 >>>eventlet.monkey_patch()
 >>>threading.current_thread.__module__
 >>>'eventlet.green.threading'