实战|用Python制作邮箱自动回复机器人

Python大本营 昨天

以下文章来源于早起Python ,作者陈熹

大家好,又来到Python办公自动化专题。

在之前的系列文章中,我们已经讲解了如何利用Python读取、收发、管理邮件。本文将进一步分享如何用Python制作一个邮件自动回复机器人

比如当发送标题为“来句诗”时,能够自动返回一句诗;当发送邮件标题为“xx(城市)天气”如“广州天气”时,能够返回所需城市的天气情况等等,更多功能可以自己定义,主要将涉及

  1. imbox 读取及解析附件
  2. yagmail 发送邮件
  3. 邮件与爬虫的结合

思路分析

和之前的文章类似,我们首先整理下思路,然后逐个解决,简单来说这个需求可以分为下面的步骤:

  1. 定时读取未读邮件,如有则获取标题及发件人
  2. 如果标题为“来句诗”,则从“今日诗词”的网站上获取一句诗;如果标题为“xx(城市)天气”则从在线天气预报网站中获取相应城市的天气情况和温度
  3. 将获取的信息组合成新邮件发送会指定收件人
  4. 将未读邮件标为已读

基本逻辑很简单,需要用到的知识点我们之前的文章中都有提过,可以直接尝试完成这个案例。两个子需求爬取的网站分别是 今日诗词https://www.jinrishici.com 和 中国天气网http://wthrcdn.etouch.cn/weather_mini?city={城市}

代码实现

邮箱方面,之前我们讲过qq邮箱、网易邮箱、这次再换个邮箱(88邮箱),首先通过 imbox 库解析邮件,可以通过 kering 库获取预先存在本地的系统密钥(本文以 88 邮箱为例):

import keyring 
from imbox import Imbox
password = keyring.get_password('88mail', 'test@88.com')

with Imbox('imap.88.com', 'test@88.com', password, ssl=True) as imbox: 
    unread_inbox_messages = imbox.messages(unread = True) # 获取未读邮件
    pass

根据需求自然而然可以想到是反复获取未读邮件,解析其标题观察是否符合条件,符合相应条件则执行相应的函数,并将函数返回的内容组装成新的邮件。最后无论是否符合要求都将其标记为已读。

当然,如果要持续运行就还需要将核心代码包装成函数,并放在循环体内部。循环可以间隔10分钟。代码如下所示:

import keyring 
from imbox import Imbox
import time
password = keyring.get_password('88mail', 'test@88.com')

def get_verse():
    pass

def get_weather():
    pass

def send_mail(email, results):
    pass

def main():
    with Imbox('imap.88.com', 'test@88.com', password, ssl=True) as imbox: 
        unread_inbox_messages = imbox.messages(unread = True) # 获取未读邮件
        for uid, message in unread_inbox_messages :
            title = message.subject
            email = message.sent_from[0]['email']
            results = ''
            if title == '来句诗':
                results = get_verse()
            if title[-2:] == '天气':
                results = get_weather(title[:-2])
            if results:
                send_mail(email, results)
            imbox.mark_seen(uid)
            
while True:
    main()
    time.sleep(600)

发送邮件可以利用之前介绍的 yagmail 库,核心代码 mail.send 接收收件人邮箱、邮件标题、邮件内容三个参数:

import yagmail

# 用服务器、用户名、密码实例化邮件
mail = yagmail.SMTP(user='xxx@88.com', password = password, host='smtp.88.com') 
# 待发送的内容
contents = ['第一段内容', '第二段内容']
# 发送邮件
mail.send('收件人邮箱', '邮件标题', contents) 

由于 send_mail 函数接受爬虫返回的 results 作为内容,也获取了 imbox 解析后得到的特定发件人邮箱,因此可以写成如下形式:

import yagmail

def send_mail(email, results):
    mail = yagmail.SMTP(user='test@88.com', password=password, host='smtp.88.com')
    contents = [results]
    mail.send(email, '【自动回复】您要的信息见正文', contents)

问题只剩下如何获取每日一句以及如何获取指定城市天气了,首先看一下每日一句的网站特点(实际上这个网站有 API 接口,读者可以自行尝试):

实战|用Python制作邮箱自动回复机器人

先试试直接返回网站内容:

import requests

url = 'https://www.jinrishici.com/'
response = requests.get(url).text
print(response)
实战|用Python制作邮箱自动回复机器人

可以返回内容,没有特别的反爬措施,但返回的正文是乱码,同时我们也注意到 utf-8 编码,因此直接修改编码即可:

import requests

response = requests.get(url)
response.encoding = "UTF-8"
print(response.text)

编码问题解决以后就可以利用 xpath 解析获取诗句了:

import requests
from lxml import html

url = 'https://www.jinrishici.com/'
response = requests.get(url)
response.encoding = "UTF-8"
selector = html.fromstring(response.text)
verse = selector.xpath('//*[@id="sentence"]/text()')
print(verse)

有趣的是,并没有按意愿返回诗句,原因是网页中的诗句是以Ajax动态加载的,而非静态出现在网页中。

重新分析网页 XHR 即可获取真正的访问连接 https://v2.jinrishici.com/one.json?client=browser-sdk/1.2&X-User-Token=xxxxxx,Token见下图:

分析好原因后代码反而更加简单了:

import requests

url = 'https://v2.jinrishici.com/one.json?client=browser-sdk/1.2&X-User-Token=xxxxxx'
response = requests.get(url)
print(response.json()['data']['content'])

返回的诗句直接就可以作为函数结果返回,因此代码又可以写成:

import requests

def get_verse():
    url = 'https://v2.jinrishici.com/one.json?client=browser-sdk/1.2&X-User-Token=xxxxxx'
    response = requests.get(url)
    return f'您要的每日诗句为:{response.json()["data"]["content"]}'

获取天气可以使用官方提供的 API 了,以广州为例:

import requests

url = 'http://wthrcdn.etouch.cn/weather_mini?city=广州'
response = requests.get(url)
print(response.json())

根据返回的 json 数据很容易获取今日的天气情况和最高最低气温,组合成函数效果如下:

def get_weather(city):
    url = f'http://wthrcdn.etouch.cn/weather_mini?city={city}'
    response = requests.get(url).json()
    results = response['data']['forecast'][0]
    return f'{city}今天的天气情况为{results["type"]},{results["high"][:-1]}度,{results["low"][:-1]}度'

至此,代码部分就写完了。我们的邮箱自动回复机器人也就拥有了两个简单的功能,当然你可以结合自己的需求实现更多有意思的功能!最后附上完整代码供大家学习与交流👇

import keyring
import yagmail
from imbox import Imbox
import requests
import time
password = keyring.get_password('88mail', 'test@88.com')

def get_verse():
    url = 'https://v2.jinrishici.com/one.json?client=browser-sdk/1.2&X-User-Token=xxxxxx'
    response = requests.get(url)
    return f'您要的每日诗句为:{response.json()["data"]["content"]}'

def get_weather(city):
    url = f'http://wthrcdn.etouch.cn/weather_mini?city={city}'
    response = requests.get(url).json()
    results = response['data']['forecast'][0]
    return f'{city}今天的天气情况为{results["type"]},{results["high"][:-1]}度,{results["low"][:-1]}度'

def send_mail(email, results):
    mail = yagmail.SMTP(user='test@88.com', password=password, host='smtp.88.com')
    contents = [results]
    mail.send(email, '【自动回复】您要的信息见正文', contents)

def main():
    with Imbox('imap.88.com', 'test@88.com', password, ssl=True) as imbox:
        unread_inbox_messages = imbox.messages(unread=True)  # 获取未读邮件
        for uid, message in unread_inbox_messages:
            title = message.subject
            email = message.sent_from[0]['email']
            results = ''
            if title == '来句诗':
                results = get_verse()
            if title[-2:] == '天气':
                results = get_weather(title[:-2])
            if results:
                send_mail(email, results)
            imbox.mark_seen(uid)

while True:
    main()
    time.sleep(600)
2021-03-31

实战|用Python制作邮箱自动回复机器人的相关文章

Python项目实战练习:制作小型图书管理系统

本文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理 以下文章来源于IT丛林 ,作者: IT丛林 说明:此程序适合入门基本练习. 私信回 ...

Python制作一款简易音乐播放器

大家好,欢迎来到 Crossin的编程教室 ! 今天给大家分享一个迷你项目案例:利用Python制作一款简易音乐播放器.这个程序具有一定的实用性,用到的技术也不算复杂,可以作为完成基础学习后的练手项目 ...

用 Python 制作一个迷宫游戏

相信大家都玩过迷宫的游戏,对于简单的迷宫,我们可以一眼就看出通路,但是对于复杂的迷宫,可能要仔细寻找好久,甚至耗费数天,然后可能还要分别从入口和出口两头寻找才能找的到通路,甚至也可能找不到通路. 虽然 ...

PQ-综合实战:工资条制作一键刷新之M解

小勤:上次你跟我说的用Power Query做工资条的操作方法--<一个工资条引发的议案>,如果要增加空行的话,步骤就很多了啊,如果用函数做的话,会不会步骤少些? 大海:嗯.用函数解的话步 ...

Django实战: channels+celery+websocket打造聊天机器人(附源码)

原创 大江狗 Python Web与Django开发 Channels是Django团队研发的一个给Django提供websocket支持的框架,使用它我们可以轻松开发需要长链接的实时通讯应用.本文在 ...

假如波士顿机器人投入实战,盘点影视中的机器人技术

假如波士顿机器人投入实战,盘点影视中的机器人技术

Python制作一个数据预处理小工具,非常实用(拿走不谢)

我们平常使用Python进行数据处理与分析时,在import完一大堆库之后 ,就是对数据进行预览,查看数据是否出现了缺失值.重复值等异常情况,并进行处理. 本文将结合GUI工具PySimpleGUI, ...

如何用 Python 制作地球仪?

来源:Python 技术「ID: pythonall」 Python 功能真的很强,强大到让人吃惊,它能做的事囊括爬虫.数据分析.数据可视化.游戏等等各方面,这些功能在实际的使用中应用广泛,开发程序讲 ...

实战!Python 30 行代码画各种 3D 图形

来源:Python 技术「ID: pythonall」 在之前的文章有讲解过 Matplotlib 的用法,可能有的小伙伴们已经略有忘记,如果有不熟悉的读者朋友们请回顾Matplotlib学习进阶 , ...