来自不同线程的Pythoncom PumpMessages

时间:2019-02-20 13:35:24

标签: python multithreading outlook com win32com

我想做与要求here类似的事情,但是要像here那样使用threading。还使用here的答案,我的代码可以正常工作,只是无法识别ItemAdd事件(实际上,我认为是,但是在另一个线程中,这就是为什么没有输出的原因)。

"""Handler class that watches for incoming mails"""
import ctypes # for the WM_QUIT to stop PumpMessage()
import logging
import win32com.client
import sys
import threading
import time
import pythoncom

# outlook config
CENTRAL_MAILBOX = "My Mailbox"

# get the outlook instance and inbox folders
outlook = win32com.client.Dispatch("Outlook.Application")
marshalled_otlk = pythoncom.CoMarshalInterThreadInterfaceInStream(
    pythoncom.IID_IDispatch, outlook)


class HandlerClass(object):

    def OnItemAdd(self, item):
        logger.info("New item added in central mailbox")
        if item.Class == 43:
            logger.info("The item is an email!")


class OtlkThread(threading.Thread):

    def __init__(self, marshalled_otlk, *args, **kwargs):
        super().__init__(*args, **kwargs)
        self.marshalled_otlk = marshalled_otlk
        self.logger = logging.getLogger("OLThread")

    def run(self):
        self.logger.info("Starting up Outlook watcher\n"
                         "To terminate the program, press 'Ctrl + C'")
        pythoncom.CoInitialize()
        outlook = win32com.client.Dispatch(
            pythoncom.CoGetInterfaceAndReleaseStream(
                self.marshalled_otlk,
                pythoncom.IID_IDispatch
            )
        )
        user = outlook.Session.CreateRecipient(CENTRAL_MAILBOX)
        central_inbox = outlook.Session.GetSharedDefaultFolder(user, 6).Items
        self.logger.info(f"{central_inbox.Count} messages in central inbox")

        win32com.client.DispatchWithEvents(central_inbox, HandlerClass)
        pythoncom.PumpMessages()
        pythoncom.CoUninitialize()  # this is prbly unnecessary as it will never be reached


def main():
    # pythoncom.CoInitialize()
    OtlkThread(marshalled_otlk, daemon=True).start()


if __name__ == "__main__":
    status = main()
    while True:
        try:
            # pythoncom.PumpWaitingMessages()
            time.sleep(1)
        except KeyboardInterrupt:
            logger.info("Terminating program..")
            ctypes.windll.user32.PostQuitMessage(0)
            sys.exit(status)

我尝试了各种方法,例如将sys.coinit_flags=0放在顶部,如建议的here),在主线程中调用PumpWaitingMessages(),并在侧面线程中获取Outlook应用程序本身,而不是传递编组的对象。这些都不起作用。

当我仅将PumpMessages放入主线程(相同的HandlerClass但没有单独的线程)时,它可以工作,并且在到达时可以识别电子邮件,但是很明显,该线程已被阻塞,甚至无法捕获KeyboardInterrupt异常。 / p>

那么,如何使Outlook Watcher在单独的线程中运行以将消息发送到主线程以在那里输出?

2 个答案:

答案 0 :(得分:2)

格式精美的问题,带有参考文献。谢谢。

要回答。在这种情况下,我使用“ threading.Thread(target = ....`”。但是您也可以使用继承:

import logging
import threading
import time


def task(name):
    log = logging.getLogger('task-' + name)
    log.info("Starting up Outlook watcher\n"
                     "To terminate the program, press 'Ctrl + C'")
    while True:
        log.info("Doing work that takes time")
        time.sleep(1)


class OtlkThread(threading.Thread):

    def __init__(self, *args, **kwargs):
        self.log = logging.getLogger('task-class')
        super().__init__(*args, **kwargs)

    def run(self):
        self.log.info("Starting up Outlook watcher\n"
                 "To terminate the program, press 'Ctrl + C'")
        while True:
            self.log.info("Doing work that takes time")
            time.sleep(1)

def main():
    t1 = threading.Thread(target=task, args=('daemon',), daemon=True)
    t1.start()
    t2 = OtlkThread()
    t2.start()


if __name__ == "__main__":
    logging.basicConfig(level=logging.DEBUG,)

    main()
    while True:
        time.sleep(1)

答案 1 :(得分:0)

再次感谢您的回答,迈克尔,这使我想到了this的答案,其中还包含指向excellent example的链接。从答案和示例中得出的主要结论是,与其将Outlook作为一个编组对象传递,不如将它作为客户端传递给处理程序。另外,使用WithEvents代替DispatchWithEvents,并在导入sys.coinit_flags = 0之前将pythoncom设置为"""Handler class that watches for incoming mails""" import ctypes # for the WM_QUIT to stop PumpMessage() import logging import sys import time from threading import Thread sys.coinit_flags = 0 # pythoncom.COINIT_MULTITHREADED == 0 from pythoncom import (CoInitializeEx, CoUninitialize, COINIT_MULTITHREADED, PumpWaitingMessages) from win32com.client import Dispatch, WithEvents # outlook config CENTRAL_MAILBOX = "My Mailbox" # COM event handler class IncomingMailHandler: def OnItemAdd(self, item): logger.info("New item added in central mailbox") if item.Class == 43: logger.info(f"The item is an email with subject {item.Subject}") # main thread def main(): # get the outlook instance and inbox folders outlook = Dispatch("Outlook.Application") user = outlook.Session.CreateRecipient(CENTRAL_MAILBOX) central_inbox = outlook.Session.GetSharedDefaultFolder(user, 6).Items logger.info(f"{central_inbox.Count} messages in central inbox") # launch the second thread thread = Thread(target=watcher, args=(central_inbox,), daemon=True) thread.start() # other thread worker function def watcher(client): logger = logging.getLogger("watcher") CoInitializeEx(COINIT_MULTITHREADED) WithEvents(client, IncomingMailHandler) # event loop 2 _loop = 0 while True: PumpWaitingMessages() _loop += 1 if _loop % 20 == 0: logger.info("Watcher is running..") time.sleep(0.5) CoUninitialize() if __name__ == "__main__": logger.info("Starting up Outlook watcher\n" "To terminate the program, press 'Ctrl + C'") status = main() while True: try: time.sleep(0.5) except KeyboardInterrupt: logger.info("Terminating program..") ctypes.windll.user32.PostQuitMessage(0) sys.exit(status)

最终结果如下:

var elm = $('#my_div');    /* The element to expand or collapse */
var t = 589;               /* Top position of element */
var h = 200;               /* Height of element */

// Set initial position and height ...

elm.css({ 'top': t + 'px' });
elm.css({ 'height': h + 'px' });