Чтение электронных писем из Outlook с помощью Python через MAPI
Я пытаюсь написать короткую программу, которая будет читать содержимое электронных писем в папке в моем профиле Exchange/Outlook, чтобы я мог манипулировать данными. Однако у меня возникла проблема с поиском большой информации об интеграции с python и Exchange/Outlook. Многие вещи либо очень старые/не имеют документов/не объяснены. Я пробовал несколько фрагментов, но, похоже, получаю те же ошибки. Я пробовал код Тима Голда:
import win32com.client
session = win32com.client.gencache.EnsureDispatch ("MAPI.Session")
#
# Leave blank to be prompted for a session, or use
# your own profile name if not "Outlook". It is also
# possible to pull the default profile from the registry.
#
session.Logon ("Outlook")
messages = session.Inbox.Messages
#
# Although the inbox_messages collection can be accessed
# via getitem-style calls (inbox_messages[1] etc.) this
# is the recommended approach from Microsoft since the
# Inbox can mutate while you're iterating.
#
message = messages.GetFirst ()
while message:
print message.Subject
message = messages.GetNext ()
Однако я получаю сообщение об ошибке:
pywintypes.com_error: (-2147221005, 'Invalid class string', None, None)
Не знаю, как меня зовут профиль, поэтому я пробовал:
session.Logon()
но это не сработало (такая же ошибка). Также попробовал оба с открытым и закрытым Outlook и ничего не изменил.
Ответы
Ответ 1
У меня была та же проблема, что и вы, - не нашел много того, что сработало. Следующий код, однако, работает как прелесть.
import win32com.client
outlook = win32com.client.Dispatch("Outlook.Application").GetNamespace("MAPI")
inbox = outlook.GetDefaultFolder(6) # "6" refers to the index of a folder - in this case,
# the inbox. You can change that number to reference
# any other folder
messages = inbox.Items
message = messages.GetLast()
body_content = message.body
print body_content
Ответ 2
Я создал свой собственный итератор для перебора объектов Outlook через python. Проблема в том, что python пытается итерации, начиная с Index [0], но outlook ожидает первого элемента Index [1]... Чтобы сделать его более простым в Ruby, ниже вспомогательный класс Oli со следующими
методы:
.items() - дает кортеж (индекс, элемент)...
.prop() - помощь в интроспекции объекта outlook, показывающего доступные свойства (методы и атрибуты)
from win32com.client import constants
from win32com.client.gencache import EnsureDispatch as Dispatch
outlook = Dispatch("Outlook.Application")
mapi = outlook.GetNamespace("MAPI")
class Oli():
def __init__(self, outlook_object):
self._obj = outlook_object
def items(self):
array_size = self._obj.Count
for item_index in xrange(1,array_size+1):
yield (item_index, self._obj[item_index])
def prop(self):
return sorted( self._obj._prop_map_get_.keys() )
for inx, folder in Oli(mapi.Folders).items():
# iterate all Outlook folders (top level)
print "-"*70
print folder.Name
for inx,subfolder in Oli(folder.Folders).items():
print "(%i)" % inx, subfolder.Name,"=> ", subfolder
Ответ 3
У меня была такая же проблема. Объединение различных подходов из Интернета (и выше) приводит к следующему подходу (checkEmails.py)
class CheckMailer:
def __init__(self, filename="LOG1.txt", mailbox="Mailbox - Another User Mailbox", folderindex=3):
self.f = FileWriter(filename)
self.outlook = win32com.client.Dispatch("Outlook.Application").GetNamespace("MAPI").Folders(mailbox)
self.inbox = self.outlook.Folders(folderindex)
def check(self):
#===============================================================================
# for i in xrange(1,100): #Uncomment this section if index 3 does not work for you
# try:
# self.inbox = self.outlook.Folders(i) # "6" refers to the index of inbox for Default User Mailbox
# print "%i %s" % (i,self.inbox) # "3" refers to the index of inbox for Another user mailbox
# except:
# print "%i does not work"%i
#===============================================================================
self.f.pl(time.strftime("%H:%M:%S"))
tot = 0
messages = self.inbox.Items
message = messages.GetFirst()
while message:
self.f.pl (message.Subject)
message = messages.GetNext()
tot += 1
self.f.pl("Total Messages found: %i" % tot)
self.f.pl("-" * 80)
self.f.flush()
if __name__ == "__main__":
mail = CheckMailer()
for i in xrange(320): # this is 10.6 hours approximately
mail.check()
time.sleep(120.00)
Для согласованности я включаю также код для класса FileWriter (найденный в FileWrapper.py). Мне это нужно, потому что
попытка пропустить UTF8 в файл в окнах не работала.
class FileWriter(object):
'''
convenient file wrapper for writing to files
'''
def __init__(self, filename):
'''
Constructor
'''
self.file = open(filename, "w")
def pl(self, a_string):
str_uni = a_string.encode('utf-8')
self.file.write(str_uni)
self.file.write("\n")
def flush(self):
self.file.flush()
Enjoy.
Ответ 4
Извините за мой плохой английский.
Проверка писем с помощью Python с MAPI проще,
outlook =win32com.client.Dispatch("Outlook.Application").GetNamespace("MAPI")
folder = outlook.Folders[5]
Subfldr = folder.Folders[5]
messages_REACH = Subfldr.Items
message = messages_REACH.GetFirst()
Здесь мы можем получить самую первую почту в почтовом ящике или в любую подпапку. Фактически, нам нужно проверить номер и ориентацию почтового ящика. С помощью этого анализа мы можем проверить каждый почтовый ящик и его папки подпапок.
Аналогично, пожалуйста, найдите приведенный ниже код, где мы можем видеть, последние/более ранние письма. Как нам нужно проверить.
`outlook =win32com.client.Dispatch("Outlook.Application").GetNamespace("MAPI")
folder = outlook.Folders[5]
Subfldr = folder.Folders[5]
messages_REACH = Subfldr.Items
message = messages_REACH.GetLast()`
С этим мы можем получить самую последнюю электронную почту в почтовый ящик.
В соответствии с вышеупомянутым кодом мы можем проверить все наши почтовые ящики и его подпапки.