SMTP через Exchange с использованием встроенной проверки подлинности Windows (NTLM) с помощью Python
Я хочу использовать учетные данные вошедшего в систему пользователя Windows для аутентификации SMTP-соединения с сервером Exchange с помощью NTLM.
Я в курсе в Python-NTLM в и два патчи которые включают аутентификацию NTLM для SMTP, однако я хочу использовать токен безопасности текущего пользователя и не должен указывать имя пользователя и пароль.
очень похожая проблема на аутентификация Windows с помощью Python и urllib2.
3 ответов
хотя решение ниже использует только расширения Win32 Python (пример кода sspi, включенный в расширения Win32 Python, был очень полезен), патчи Python-ntlm IMAP & SMTP, упомянутые в вопросе, также служили полезными руководствами.
from smtplib import SMTPException, SMTPAuthenticationError
import string
import base64
import sspi
# NTLM Guide -- http://curl.haxx.se/rfc/ntlm.html
SMTP_EHLO_OKAY = 250
SMTP_AUTH_CHALLENGE = 334
SMTP_AUTH_OKAY = 235
def asbase64(msg):
return string.replace(base64.encodestring(msg), '\n', '')
def connect_to_exchange_as_current_user(smtp):
"""Example:
>>> import smtplib
>>> smtp = smtplib.SMTP("my.smtp.server")
>>> connect_to_exchange_as_current_user(smtp)
"""
# Send the SMTP EHLO command
code, response = smtp.ehlo()
if code != SMTP_EHLO_OKAY:
raise SMTPException("Server did not respond as expected to EHLO command")
sspiclient = sspi.ClientAuth('NTLM')
# Generate the NTLM Type 1 message
sec_buffer=None
err, sec_buffer = sspiclient.authorize(sec_buffer)
ntlm_message = asbase64(sec_buffer[0].Buffer)
# Send the NTLM Type 1 message -- Authentication Request
code, response = smtp.docmd("AUTH", "NTLM " + ntlm_message)
# Verify the NTLM Type 2 response -- Challenge Message
if code != SMTP_AUTH_CHALLENGE:
raise SMTPException("Server did not respond as expected to NTLM negotiate message")
# Generate the NTLM Type 3 message
err, sec_buffer = sspiclient.authorize(base64.decodestring(response))
ntlm_message = asbase64(sec_buffer[0].Buffer)
# Send the NTLM Type 3 message -- Response Message
code, response = smtp.docmd("", ntlm_message)
if code != SMTP_AUTH_OKAY:
raise SMTPAuthenticationError(code, response)
отличный ответ, но как обновление для Python 3.0+
def asbase64(msg):
# encoding the message then convert to string
return((base64.b64encode(msg)).decode("utf-8"))
Python 2.7.x завершится ошибкой при отправке сообщения NTLM Type 3 из-за пустого cmd:
code, response = smtp.docmd("", ntlm_message)
это заканчивается отправкой правильного ответа обратно на сервер, однако он предварительно заполняет пространство из-за природы docmd (), вызывающего putcmd().
smtplib.py:
def putcmd(self, cmd, args=""):
"""Send a command to the server."""
if args == "":
str = '%s%s' % (cmd, CRLF)
else:
str = '%s %s%s' % (cmd, args, CRLF)
self.send(str)
# ...
def docmd(self, cmd, args=""):
"""Send a command, and return its response code."""
self.putcmd(cmd, args)
return self.getreply()
который в результате принимает путь условия else, тем самым отправляя str(' ' + ntlm_message + CRLF)
что приводит к (501, 'Syntax error in parameters or arguments')
.
как таковое исправление просто отправить сообщение NTLM в качестве cmd.
code, response = smtp.docmd(ntlm_message)
исправление к вышеуказанному ответу было представлено, хотя кто знает, когда он будет рассмотрен / принят.