Inhaltsverzeichnis
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 bzw. Weiterleitung von 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