
來源:早起Python
作者:陳熹
大家好,又來到Python辦公自動化專題。
在之前的系列文章中,我們已經(jīng)講解了如何利用Python讀取、收發(fā)、管理郵件。本文將進一步分享如何用Python制作一個郵件自動回復(fù)機器人。
比如當(dāng)發(fā)送標(biāo)題為“來句詩”時,能夠自動返回一句詩;當(dāng)發(fā)送郵件標(biāo)題為“xx(城市)天氣”如“廣州天氣”時,能夠返回所需城市的天氣情況等等,更多功能可以自己定義,主要將涉及
“imbox 讀取及解析附件yagmail 發(fā)送郵件郵件與爬蟲的結(jié)合”
和之前的文章類似,我們首先整理下思路,然后逐個解決,簡單來說這個需求可以分為下面的步驟:
“定時讀取未讀郵件,如有則獲取標(biāo)題及發(fā)件人如果標(biāo)題為“來句詩”,則從“今日詩詞”的網(wǎng)站上獲取一句詩;如果標(biāo)題為“xx(城市)天氣”則從在線天氣預(yù)報網(wǎng)站中獲取相應(yīng)城市的天氣情況和溫度將獲取的信息組合成新郵件發(fā)送會指定收件人將未讀郵件標(biāo)為已讀”
基本邏輯很簡單,需要用到的知識點我們之前的文章中都有提過,可以直接嘗試完成這個案例。兩個子需求爬取的網(wǎng)站分別是 今日詩詞:https://www.jinrishici.com 和 中國天氣網(wǎng):http://wthrcdn.etouch.cn/weather_mini?city={城市}
郵箱方面,之前我們講過qq郵箱、網(wǎng)易郵箱、這次再換個郵箱(88郵箱),首先通過 imbox 庫解析郵件,可以通過 kering 庫獲取預(yù)先存在本地的系統(tǒng)密鑰(本文以 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
根據(jù)需求自然而然可以想到是反復(fù)獲取未讀郵件,解析其標(biāo)題觀察是否符合條件,符合相應(yīng)條件則執(zhí)行相應(yīng)的函數(shù),并將函數(shù)返回的內(nèi)容組裝成新的郵件。最后無論是否符合要求都將其標(biāo)記為已讀。
當(dāng)然,如果要持續(xù)運行就還需要將核心代碼包裝成函數(shù),并放在循環(huán)體內(nèi)部。循環(huán)可以間隔10分鐘。代碼如下所示:
import keyring from imbox import Imboximport time
password = keyring.get_password('88mail', 'test@88.com')def get_verse():
passdef get_weather(): passdef send_mail(email, results): passdef 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)
發(fā)送郵件可以利用之前介紹的 yagmail 庫,核心代碼 mail.send 接收收件人郵箱、郵件標(biāo)題、郵件內(nèi)容三個參數(shù):
import yagmail# 用服務(wù)器、用戶名、密碼實例化郵件mail = yagmail.SMTP(user='xxx@88.com', password =
password, host='smtp.88.com') # 待發(fā)送的內(nèi)容contents = ['第一段內(nèi)容', '第二段內(nèi)容']#
發(fā)送郵件mail.send('收件人郵箱', '郵件標(biāo)題', contents)
由于 send_mail 函數(shù)接受爬蟲返回的 results 作為內(nèi)容,也獲取了 imbox 解析后得到的特定發(fā)件人郵箱,因此可以寫成如下形式:
import yagmaildef send_mail(email, results):
mail = yagmail.SMTP(user='test@88.com', password=password, host='smtp.88.com')
contents = [results]
mail.send(email, '【自動回復(fù)】您要的信息見正文', contents)
問題只剩下如何獲取每日一句以及如何獲取指定城市天氣了,首先看一下每日一句的網(wǎng)站特點(實際上這個網(wǎng)站有 API 接口,讀者可以自行嘗試):
先試試直接返回網(wǎng)站內(nèi)容:
import requests
url = 'https://www.jinrishici.com/'response = requests.get(url).textprint(response)
可以返回內(nèi)容,沒有特別的反爬措施,但返回的正文是亂碼,同時我們也注意到 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)
有趣的是,并沒有按意愿返回詩句,原因是網(wǎng)頁中的詩句是以Ajax動態(tài)加載的,而非靜態(tài)出現(xiàn)在網(wǎng)頁中。
重新分析網(wǎng)頁 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'])
返回的詩句直接就可以作為函數(shù)結(jié)果返回,因此代碼又可以寫成:
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())
根據(jù)返回的 json 數(shù)據(jù)很容易獲取今日的天氣情況和最高最低氣溫,組合成函數(shù)效果如下:
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]}度'
至此,代碼部分就寫完了。我們的郵箱自動回復(fù)機器人也就擁有了兩個簡單的功能,當(dāng)然你可以結(jié)合自己的需求實現(xiàn)更多有意思的功能!最后附上完整代碼供大家學(xué)習(xí)與交流
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, '【自動回復(fù)】您要的信息見正文', 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)
數(shù)據(jù)分析咨詢請掃描二維碼
若不方便掃碼,搜微信號:CDAshujufenxi
CDA 數(shù)據(jù)分析師:表結(jié)構(gòu)數(shù)據(jù) “獲取 - 加工 - 使用” 全流程的賦能者 表結(jié)構(gòu)數(shù)據(jù)(如數(shù)據(jù)庫表、Excel 表、CSV 文件)是企業(yè)數(shù)字 ...
2025-09-18DSGE 模型中的 Et:理性預(yù)期算子的內(nèi)涵、作用與應(yīng)用解析 動態(tài)隨機一般均衡(Dynamic Stochastic General Equilibrium, DSGE)模 ...
2025-09-17Python 提取 TIF 中地名的完整指南 一、先明確:TIF 中的地名有哪兩種存在形式? 在開始提取前,需先判斷 TIF 文件的類型 —— ...
2025-09-17CDA 數(shù)據(jù)分析師:解鎖表結(jié)構(gòu)數(shù)據(jù)特征價值的專業(yè)核心 表結(jié)構(gòu)數(shù)據(jù)(以 “行 - 列” 規(guī)范存儲的結(jié)構(gòu)化數(shù)據(jù),如數(shù)據(jù)庫表、Excel 表、 ...
2025-09-17Excel 導(dǎo)入數(shù)據(jù)含缺失值?詳解 dropna 函數(shù)的功能與實戰(zhàn)應(yīng)用 在用 Python(如 pandas 庫)處理 Excel 數(shù)據(jù)時,“缺失值” 是高頻 ...
2025-09-16深入解析卡方檢驗與 t 檢驗:差異、適用場景與實踐應(yīng)用 在數(shù)據(jù)分析與統(tǒng)計學(xué)領(lǐng)域,假設(shè)檢驗是驗證研究假設(shè)、判斷數(shù)據(jù)差異是否 “ ...
2025-09-16CDA 數(shù)據(jù)分析師:掌控表格結(jié)構(gòu)數(shù)據(jù)全功能周期的專業(yè)操盤手 表格結(jié)構(gòu)數(shù)據(jù)(以 “行 - 列” 存儲的結(jié)構(gòu)化數(shù)據(jù),如 Excel 表、數(shù)據(jù) ...
2025-09-16MySQL 執(zhí)行計劃中 rows 數(shù)量的準(zhǔn)確性解析:原理、影響因素與優(yōu)化 在 MySQL SQL 調(diào)優(yōu)中,EXPLAIN執(zhí)行計劃是核心工具,而其中的row ...
2025-09-15解析 Python 中 Response 對象的 text 與 content:區(qū)別、場景與實踐指南 在 Python 進行 HTTP 網(wǎng)絡(luò)請求開發(fā)時(如使用requests ...
2025-09-15CDA 數(shù)據(jù)分析師:激活表格結(jié)構(gòu)數(shù)據(jù)價值的核心操盤手 表格結(jié)構(gòu)數(shù)據(jù)(如 Excel 表格、數(shù)據(jù)庫表)是企業(yè)最基礎(chǔ)、最核心的數(shù)據(jù)形態(tài) ...
2025-09-15Python HTTP 請求工具對比:urllib.request 與 requests 的核心差異與選擇指南 在 Python 處理 HTTP 請求(如接口調(diào)用、數(shù)據(jù)爬取 ...
2025-09-12解決 pd.read_csv 讀取長浮點數(shù)據(jù)的科學(xué)計數(shù)法問題 為幫助 Python 數(shù)據(jù)從業(yè)者解決pd.read_csv讀取長浮點數(shù)據(jù)時的科學(xué)計數(shù)法問題 ...
2025-09-12CDA 數(shù)據(jù)分析師:業(yè)務(wù)數(shù)據(jù)分析步驟的落地者與價值優(yōu)化者 業(yè)務(wù)數(shù)據(jù)分析是企業(yè)解決日常運營問題、提升執(zhí)行效率的核心手段,其價值 ...
2025-09-12用 SQL 驗證業(yè)務(wù)邏輯:從規(guī)則拆解到數(shù)據(jù)把關(guān)的實戰(zhàn)指南 在業(yè)務(wù)系統(tǒng)落地過程中,“業(yè)務(wù)邏輯” 是連接 “需求設(shè)計” 與 “用戶體驗 ...
2025-09-11塔吉特百貨孕婦營銷案例:數(shù)據(jù)驅(qū)動下的精準(zhǔn)零售革命與啟示 在零售行業(yè) “流量紅利見頂” 的當(dāng)下,精準(zhǔn)營銷成為企業(yè)突圍的核心方 ...
2025-09-11CDA 數(shù)據(jù)分析師與戰(zhàn)略 / 業(yè)務(wù)數(shù)據(jù)分析:概念辨析與協(xié)同價值 在數(shù)據(jù)驅(qū)動決策的體系中,“戰(zhàn)略數(shù)據(jù)分析”“業(yè)務(wù)數(shù)據(jù)分析” 是企業(yè) ...
2025-09-11Excel 數(shù)據(jù)聚類分析:從操作實踐到業(yè)務(wù)價值挖掘 在數(shù)據(jù)分析場景中,聚類分析作為 “無監(jiān)督分組” 的核心工具,能從雜亂數(shù)據(jù)中挖 ...
2025-09-10統(tǒng)計模型的核心目的:從數(shù)據(jù)解讀到?jīng)Q策支撐的價值導(dǎo)向 統(tǒng)計模型作為數(shù)據(jù)分析的核心工具,并非簡單的 “公式堆砌”,而是圍繞特定 ...
2025-09-10CDA 數(shù)據(jù)分析師:商業(yè)數(shù)據(jù)分析實踐的落地者與價值創(chuàng)造者 商業(yè)數(shù)據(jù)分析的價值,最終要在 “實踐” 中體現(xiàn) —— 脫離業(yè)務(wù)場景的分 ...
2025-09-10機器學(xué)習(xí)解決實際問題的核心關(guān)鍵:從業(yè)務(wù)到落地的全流程解析 在人工智能技術(shù)落地的浪潮中,機器學(xué)習(xí)作為核心工具,已廣泛應(yīng)用于 ...
2025-09-09