装饰 win32com COM 对象的每个方法

Decorate each method of win32com COM object

Python 3
我有一个通过 win32com.client.Dispatch 的 COM 对象,其中包含许多用于自动化应用程序的方法。我希望每次调用 COM 对象的任何方法时,python 都会根据返回值执行某些操作(实际上是记录)。
我的自动化伪代码是这样的:

obj = win32com.client.DispatchEx("3rdParty.Application")
obj.methodA(Parameter)
obj.methodB()
obj.non_method_call
obj.methodN()
...

当对 obj 进行任何方法调用时,我希望此伪代码的意图发生:

x = obj.any_method(*args)
if x:
    logger.debug(any_method.__name__ + ' was called at ' + datetime.now().strftime("%H:%M") + ' with parameters ' + str(*args)
else:
    logger.error(obj.cerrmsg)
    obj.logout

请注意,any_method.__name__ 部分会很好,但并不重要。 python 可以做到这一点,尤其是对于 COM 对象,并且无需为一组有限的方法编写逻辑吗?
装饰器听起来不错,因为它们修改了 functions/classes,但我看过的帖子在这种情况下不起作用(比如通过对象的方法字典),然后我听说它们只适用于我自己的代码中定义的方法。有人建议@属性,但我不知道如何在这种情况下应用它。
欢迎使用高级技巧(getattr、元类、functools、wraps 等),但请演示。

我针对 win32com.client 测试了我在评论中提到的方法并遇到了一个基本问题:

self.__getattribute__ 的递归性质意味着尝试存储对象然后使用 self.stored_obj 失败,因为它永远递归地调用 __getattribute__

如果你原谅全局变量的糟糕使用是一种绕过这个问题的骇人听闻的方式,并且意味着你一次只能使用一次,那么它基本上是有效的。

即debugging/tracing 您的 COM 对象发生的事情可能没问题,但这不是一个好的或可靠的解决方案:

global _wrapped_object

class LogWrap:
    def __init__(self, object):
        global _wrapped_object
        _wrapped_object = object

    def __getattribute__(self, name):
        global _wrapped_object

        next_hop_attr = _wrapped_object.__getattr__(name)

        if callable(next_hop_attr):

            def logging_attr(*args, **kwargs):
                retval = next_hop_attr(*args, **kwargs)
                print("logging a call to {0} with args -{1}-".format(name, ','.join(args)))
                print("Returned: {0}".format(retval))
                return retval

            return logging_attr

        else:
            return next_hop_attr

    def __setattr__(self, name, value):
        global _wrapped_object
        _wrapped_object.__setattr__(name, value)


from win32com.client import Dispatch
ie = Dispatch('InternetExplorer.Application')
ie = LogWrap(ie)
ie.Visible = True

ie.Navigate2('google.com')

示例输出:

>>> logging a call to Navigate2 with args -google.com-
Returned: None
>>>