Benutzer-Werkzeuge

Webseiten-Werkzeuge


outlook_exchangelib

Dies ist eine alte Version des Dokuments!


Exchange Server Abfrage - Mail und Kalender

Auf Smartphone und Computer wird man immer informiert, ob neue mail eingetroffen ist. Damit aber die Heim-Automation auch die aktuellsten Informationen hat, muss man sie sich über z.B. den Exchange-Server besorgen. Mit der exchangelib kann ich vom Raspberry Pi aus den Arbeits-Exchange-Server abfragen.

Installation und Probleme

sudo apt-get install python3-pip
python3 -m pip install exchangelib --user

So wenige Zeilen und so ein langer Weg dahin. Erst kam der Fehler

Traceback (most recent call last):
  File "/usr/local/bin/pip", line 7, in <module>
    from pip import main
ImportError: cannot import name main

Neu installieren hat nicht geholfen, kann man aber versuchen

sudo python3 -m pip uninstall pip && sudo apt install python3-pip --reinstall

Weitere Fehlermeldungen

Fehlermeldung:
Could not install packages due to an EnvironmentError: [Errno 13] Keine Berechtigung: '/usr/local/lib/python3.6/site-packages/dnspython-2.1.0.dist-info'
Consider using the `--user` option or check the permissions.

Mit sudo davor geht es zwar, aber das ist ja nicht so sicher, daher dann das '–user' hintendrin, was erfolgreich war.

Um festzustellen, mit welcher Verschlüsselung der Exchange Server zurecht kommt, kann man einen CURL-Aufruf auf den Webmailer machen.

curl -v -I https://webmail.adresse/EWS/Exchange.asmx

Nach der Fehlermeldung

ssl.SSLError: [SSL: SSLV3_ALERT_HANDSHAKE_FAILURE] sslv3 alert handshake failure (_ssl.c:720)

habe ich erst lange viele versuche gemacht, das zu beheben. Eine Idee war, die TLS-Validierung auszuhebeln mit

BaseProtocol.HTTP_ADAPTER_CLS = NoVerifyHTTPAdapter

hat aber nicht geklappt. Auch mittels pyopenssl und dem Import in urllib3 hat es nicht geklappt. Hier der Artikel und dann noch die Code-Zeilen

sudo python3 -m pip  install pyopenssl
import OpenSSL
import urllib3
import urllib3.contrib.pyopenssl as pyopenssl
pyopenssl.inject_into_urllib3()

Erst mit dem DEBUG-Log und dem automatischen Discover unseres Exchange-Servers bin ich den richtigen Einstellungen auf die Schliche gekommen.

import logging
from exchangelib.util import PrettyXmlHandler
from exchangelib.protocol import BaseProtocol, NoVerifyHTTPAdapter
logging.basicConfig(level=logging.DEBUG, handlers=[PrettyXmlHandler()])

Bei 'unserem' Server gibt es ein Konfiguration-Problem.

DEBUG:exchangelib.version:API version "Exchange2013_SP1" worked but server reports version "V2_23". Using "Exchange2013_SP1"
DEBUG:exchangelib.services.common:Found new version (Build=None, API=Exchange2019, Fullname=Microsoft Exchange Server 2019 -> Build=15.0.1497.26, API=Exchange2013_SP1, Fullname=Microsoft Exchange Server 2013 SP1)

Abfrage der letzten 5 mails in der Inbox

Hier der funktionierende Aufruf:

import paho.mqtt.client as mqtt
import json
import time
import pytz
import requests.auth
import logging

from datetime    import datetime, timedelta, time
from exchangelib import DELEGATE, IMPERSONATION, Account, Credentials, EWSDateTime
from exchangelib import EWSDate, EWSTimeZone, Configuration, NTLM, GSSAPI
from exchangelib import CalendarItem, Message, Mailbox, Attendee, Q, ExtendedProperty
from exchangelib import FileAttachment, ItemAttachment, HTMLBody, Build, Version, FolderCollection
from exchangelib import transport

from exchangelib.util     import PrettyXmlHandler
from exchangelib.protocol import BaseProtocol, NoVerifyHTTPAdapter

#logging.basicConfig(level=logging.DEBUG, handlers=[PrettyXmlHandler()])


creds = Credentials(
    username="domain\\user",
    password="geheim"
)

version = Version(build=Build(15,0,1497,26))

config = Configuration(service_endpoint='https://mapi.adresse.des.servers/EWS/Exchange.asmx', 
                       credentials=creds, version=version)

account = Account(
    primary_smtp_address="mail@adresse",
    credentials=creds,
    autodiscover=False,
    config=config,
    access_type=DELEGATE
)


for item in account.inbox.all().order_by('-datetime_received')[:5]:
    print(item.subject)

Danach habe ich das ganze noch an den MQTT-Server angebunden, damit nur bei Anwesenheit die ungelesenen Mails der letzten paar Stunden vorgelesen werden, sowie ein crontab Aufruf, der das alle 5 Minuten startet.

Kalender-Einträge

Die Anmeldung an den Exchange-Server ist die gleiche wie oben. Um Kalendereinträge abzufragen braucht es diesen Code. Ein wenig Probleme hat die Zeitzone gemacht.

now=datetime.now()
current = EWSDateTime(now.year,now.month,now.day,now.hour,00,tzinfo = EWSTimeZone.localzone())
future = current + timedelta(days=1)
erg=list(account.calendar.view(start=current, end=future))

mytext=""
if not erg:
    print('No meetings')
    mytext="keine termine heute"
else:
    mytext="Deine nächsten Termine:"
    for item in erg:
        d=item.start.astimezone(EWSTimeZone.localzone())
        print(d.strftime("%d-%m-%Y %H:%M"),' - ', item.subject)
        mytext=mytext+" Um "+d.strftime("%H:%M")+" Uhr hast Du den Termin: "+item.subject+". "

Nachdem der String erzeugt wurde, kann ich ihn dann von meinem Mac bzw. SNIPS, vorlesen lassen. Dann noch 'schnell' die Sprachsteuerung, damit ich mit „SNIPS: termine“ meine aktuellen Termine vorlesen lassen kann.

SNIPS und falsche Intents

Es macht leider einen Unterschied, ob ich sage „termine“ oder „was sind meine aktuellen termine“. Es werden zwei verschiedene hermes-intents aufgerufen. Auch da hat es wieder einige Zeit gedauert, bis ich es gelöst habe. In dem Intentionen, der nicht genutzt werden soll, muss ich eine Weiterleitung machen. Die schnellste Lösung steht hier. Man könnte aber auch aus dem ersten Intention die 'alternative' auslesen, und diesen Intentionen dann Publishen. Das mache ich dann beim nächsten mal. Jetzt steht einfach fester der Name drin.

Das ganze ist hier für node.js geschrieben
client.publish('hermes/dialogueManager/endSession', 
  JSON.stringify({'sessionId': intent.sessionId}));
client.publish('hermes/intent/varlet:erinnerung', 
  JSON.stringify({'input':'termine',
  'intent':intent.intent, 'slots':intent.slots, 'id':intent.id, 
  'sessionId':intent.sessionId,   
  'customData':intent.customData, 'asrTokens':intent.asrTokens}));

Stand Januar 2022

outlook_exchangelib.1641067407.txt.gz · Zuletzt geändert: 2022/01/01 20:03 von varnholt