我如何:
- 用 python 代码连接我的邮件收件箱以自动获取未读电子邮件的加密内容;
- 解码 S/MIME 加密电子邮件(我有密钥);
- 检索电子邮件正文纯文本;
- 检查正文(或主题)是否与某个关键字(现在为“test”)匹配,并在匹配时打印一些内容;
- 在树莓派上使用此代码,无需手动执行任何操作。
我已经检查过 类似的帖子 ,但我无法设法将其实际连接到我的邮箱并使其全部工作,而无需手动执行任何操作。
我正在使用 iOS' /iCloud 的 IMAP (imap.mail.me.com),我现在正在使用
mail.login()
,但它似乎实际上没有从未读电子邮件中获取任何内容。
我现在拥有的代码:|| |目前最好的代码,在 Windows 上的 Visual Studio Code 中编程,而不是 Linux
import imaplib
import email
from email.header import decode_header
from OpenSSL import crypto
import time, datetime
# Email receiving configuration
imap_server = "imap.mail.me.com" # <-- iCloud's mail IMAP
imap_user = "example@example.com" # <- Email
imap_password = "example_password" # <-- Email App-Specific password
# Paths to your certificate and private key
cert_path = 'path\to\my\certificate.pem' #<-- Path to cert.pem
key_path = 'path\to\my\privatekey.pem' # <-- Path to privkey.pem
# Function to decrypt S/MIME email
def decrypt_smime(encrypted_message):
try:
# Load the certificate and private key
with open(cert_path, 'rb') as f:
cert = crypto.load_certificate(crypto.FILETYPE_PEM, f.read())
with open(key_path, 'rb') as f:
key = crypto.load_privatekey(crypto.FILETYPE_PEM, f.read())
# Create a Crypto object and decrypt the email
smime = crypto.SMIME()
smime.load_cert(cert)
smime.load_key(key)
decrypted_message = smime.decrypt(encrypted_message)
return decrypted_message.decode('utf-8')
except Exception as e:
print(f"Error decrypting S/MIME email: {e}")
return None
# Function to process email payload
def process_email_payload(msg):
if msg.is_multipart():
for part in msg.walk():
content_type = part.get_content_type()
if content_type == "application/pkcs7-mime":
payload = part.get_payload(decode=True)
payload = decrypt_smime(payload)
if payload:
return payload.strip()
elif content_type in ["text/plain", "text/html"]:
payload = part.get_payload(decode=True).strip()
return payload.decode('utf-8')
else:
payload = msg.get_payload(decode=True)
if msg.get_content_type() == "application/pkcs7-mime":
payload = decrypt_smime(payload)
return payload.decode('utf-8')
return None
# Function to check for new emails
def check_email():
try:
mail = imaplib.IMAP4_SSL(imap_server)
mail.login(imap_user, imap_password)
mail.select("inbox")
result, data = mail.search(None, "UNSEEN")
# Debugging information
print(f"Search result {datetime.datetime.now():%H.%M.%S}: {result}") # <-- Also prints current time so I can differentiate the different outputs
print(f"Search data: {data}")
if result != "OK":
print(f"Error searching Inbox: {result}")
return
if data[0] is None:
print("No new emails.")
return
email_ids = data[0].split()
if not email_ids:
print("No new emails.")
return
print(f"Email IDs: {email_ids}") # Debug email IDs
for email_id in email_ids:
result, msg_data = mail.fetch(email_id, "(RFC822)")
print(f"Fetch result: {result}") # Debug fetch result
print(f"Fetch msg_data: {msg_data}") # Debug msg_data
if result != "OK":
print(f"Error fetching email ID {email_id}: {result}")
continue
for response_part in msg_data:
if isinstance(response_part, tuple):
raw_email = response_part[1]
print(f"Raw email: {raw_email}") # Debug raw email data
try:
msg = email.message_from_bytes(raw_email)
subject, encoding = decode_header(msg["Subject"])[0]
if isinstance(subject, bytes):
subject = subject.decode(encoding if encoding else 'utf-8')
sender = email.utils.parseaddr(msg["From"])[1]
print(f"Subject: {subject}")
print(f"From: {sender}")
payload = process_email_payload(msg)
if payload:
print(f"Payload: {payload}") # Debug payload content
else:
print(f"Could not decode payload for email ID {email_id}")
continue
if "test" in payload:
print("Test prompt received")
except Exception as e:
print(f"Error decoding email ID {email_id}: {e}")
continue
mail.close()
mail.logout()
except Exception as e:
print(f"Failed to check email: {e}")
# Actually try to check and decode emails (loop)
while True:
check_email()
time.sleep(10)
它确实会主动检查电子邮件,但不会获取任何内容:
终端输出
Fetch result: OK
Fetch msg_data: [b'0000 ()']
并且不打印主题,等等,因为它不是
我假设msg_data实际上是空的(基本上)。
isinstance(response_part, tuple)
我已经尝试了数十种其他可能的解决方案,但这似乎是目前最好的。
所以我实际上如何主动检查未读电子邮件并解码其正文?
请尽力描述解决方案,因为我真的有兴趣了解更多有关此的信息,并且我想提高我的整体编码技能。我也对我做错了什么感兴趣。
Please try your best with describing the solution, since I'm really interested in learning more information about this and I would like to improve my over-all coding skills. I'm also interested in what I did wrong.
代码的问题在于,没有正确检索电子邮件内容。
mail.fetch
方法返回电子邮件数据(包括其内容)的封装在元组中的列表。正在访问元组的错误部分,从而导致空数据。
以下是改进代码的步骤:
- 更正电子邮件内容检索:
-
需要正确解析
mail.fetch
返回的数据。它返回一个包含两个元素的元组列表:- 第一个元素是状态代码。
-
第二个元素是包含电子邮件数据的元组,形式为
(flags, data)
。
-
需要提取
data
部分,然后使用email.message_from_bytes
解析它。 -
优化电子邮件处理:
-
在遍历电子邮件部分时,如果找到匹配的
content_type
(application/pkcs7-mime
或text/plain
/text/html
),可以使用break
语句退出循环,因为已经找到了所需的有效负载。
以下是更新后的代码:
import imaplib
import email
from email.header import decode_header
from OpenSSL import crypto
import time, datetime
# ...(电子邮件配置、证书路径与之前相同)...
# ...(decrypt_smime 函数与之前相同)...
# 函数来处理电子邮件有效负载
def process_email_payload(msg):
if msg.is_multipart():
for part in msg.walk():
content_type = part.get_content_type()
if content_type == "application/pkcs7-mime":
payload = part.get_payload(decode=True)
payload = decrypt_smime(payload)
if payload:
return payload.strip()
break # 找到 S/MIME 部分,退出循环
elif content_type in ["text/plain", "text/html"]:
payload = part.get_payload(decode=True).strip()
return payload.decode('utf-8')
break # 找到文本或 HTML 部分,退出循环
else:
payload = msg.get_payload(decode=True)
if msg.get_content_type() == "application/pkcs7-mime":
payload = decrypt_smime(payload)
return payload.decode('utf-8')
return None
# 函数来检查新电子邮件
def check_email():
try:
mail = imaplib.IMAP4_SSL(imap_server)
mail.login(imap_user, imap_password)
mail.select("inbox")
result, data = mail.search(None, "UNSEEN")
# ...(调试信息与之前相同)...
if result != "OK":
print(f"搜索收件箱时出错:{result}")
return
if data[0] is None:
print("没有新邮件。")
return
email_ids = data[0].split()
if not email_ids:
print("没有新邮件。")
return
print(f"电子邮件 ID:{email_ids}") # 调试电子邮件 ID
for email_id in email_ids:
result, msg_data = mail.fetch(email_id, "(RFC822)")
# ...(调试信息与之前相同)...
if result != "OK":
print(f"提取电子邮件 ID {email_id} 时出错:{result}")
continue
for response_part in msg_data:
if isinstance(response_part, tuple):
# 从元组中获取电子邮件数据
_, raw_email = response_part # 忽略第一个元素(flags)
print(f"原始电子邮件:{raw_email}") # 调试原始电子邮件数据
try:
msg = email.message_from_bytes(raw_email)
subject, encoding = decode_header(msg["Subject"])[0]
if isinstance(subject, bytes):
subject = subject.decode(encoding if encoding else 'utf-8')
sender = email.utils.parseaddr(msg["From"])[1]
print(f"主题:{subject}")
print(f"发件人:{sender}")
payload = process_email_payload(msg)
if payload:
print(f"有效负载:{payload}") # 调试有效负载内容
else:
print(f"无法解码电子邮件 ID {email_id} 的有效负载")
continue
if "test" in payload:
print("收到测试提示")
except Exception as e:
print(f"解码电子邮件 ID {email_id} 时出错:{e}")
continue
mail.close()
mail.logout()
except Exception as e:
print(f"检查电子邮件失败:{e}")
# 实际尝试检查和解码电子邮件(循环)
while True:
check_email()
time.sleep(10)
这些更正应该可以解决电子邮件内容检索问题,并允许正确解码和处理 S/MIME 加密的电子邮件。
标签:python,email,encryption,openssl,smime From: 78809096