最近使用了不少通訊工具的介面, 比如企業微信機器人,釘釘,微信公眾號的介面(未認證的訂閱公眾號),相對於郵件來說,它們的表現形式太弱。比如沒有更豐富的版本方式。當然了,並不是說表現形式越棒就是越好的通知手段,這個依個人情況而定,而我恰恰需要比較豐富的表現形式,最終還是回到了郵件,郵件真香!
而個人微訊號的介面我沒有合適的微訊號可以登入,如果網頁版微信沒有被封的話,我想這個是表現形式與訊息時效性結合的最好的方式。
環境
雖說就發郵件這麼個小事,很容易相容Python2, Python3, 但是大家還是擁抱Python3吧, 我這裡沒有做python2的相容寫法,所以需要python3以上。
郵件的格式
郵件的格式主要就兩種: plain和html
plain就像一個普通的文字, 沒有格式。
html就如其名, 是html的格式,相當於一個郵件就是一個
靜態
的網頁,這樣的話可玩性就很高了,你可以透過css控制表現形式。
注意: 這裡的css雖然語法一樣,但,是否與瀏覽器渲染結果完全一致, 是不一定的。
那麼可能有人要問了,我要發一個動態的網頁怎麼辦? 發個連結呀
郵箱賬號
無論是QQ郵箱抑或網易郵箱都是沒有問題的,重要的是有一個可以透過smtp伺服器傳送郵件的賬戶名及密碼,這裡大家百度吧。
傳送郵件的程式碼
因為傳送郵件的程式碼在下面每個步驟都是一樣的所以先貼出來
def send_email(msg, mail_to, smtp_host, smtp_username, smtp_password, subject, from_): msg[“Subject”] = Header(subject, “utf-8”) msg[“From”] = Header(from_, “utf-8”) if not isinstance(mail_to, list): mail_to = [mail_to] msg[“To”] = COMMASPACE。join(mail_to) try: print(“準備連線smtp郵件伺服器: %s” % smtp_host) client = smtplib。SMTP(smtp_host) print(“連線成功”) # client = smtplib。SMTP(“localhost”) # client。set_debuglevel(1) # print(self。mail_user, self。mail_pass) client。login(smtp_username, smtp_password) print(“登入成功”) # print(“=====>”, self。mail_from, mail_to) print(“透過郵箱[%s]傳送郵件給 %s” % (smtp_username, COMMASPACE。join(mail_to))) client。sendmail(smtp_username, mail_to, msg。as_string()) print(“傳送成功。。。”) return True except Exception: print(“傳送郵件失敗”) finally: client。quit()
如果遇到郵件傳送的問題可以將client。set_debuglevel(1)的註釋取消,這樣會顯示足夠多的debug資訊用於排查問題。
傳送本地圖片
這裡傳送圖片的意思是指, 圖片內嵌在郵件中而不是以附件的形式出現。
效果如下:
程式碼如下:
EMAIL_IMAGE_TEMPLATE = “”“
這是一張圖片
”“”def create_image_eamil_contant(fp): tpl = Template(EMAIL_IMAGE_TEMPLATE) if not path。exists(fp): sys。exit(“要傳送的本地圖片不存在”) msg = MIMEMultipart(“related”) image_name = “demo” with open(fp, “rb”) as rf: mime_image = MIMEImage(rf。read()) # 注意: 一定需要<>括號 mime_image。add_header(“Content-ID”, “<%s>” % image_name) msg。attach(mime_image) # 渲染郵件文字內容 text = tpl。render(image_name=image_name) msg_alternative = MIMEMultipart(“alternative”) msg_alternative。attach(MIMEText(text, “html”, “utf-8”)) msg。attach(msg_alternative) return msg如果你使用過python的web框架,你對文字的渲染一定不陌生,因為大多數web框架都支援文字渲染,這裡使用的jinja2。
傳送程式生成的照片
其實這裡跟上面沒什麼區別的,唯一的區別就是是否儲存在本地,既然能傳送本地圖片,我就先儲存到本地然後再按照上面的方式不就可以了麼? 首先這個方法是沒有問題的,不過多了一次IO, 能在記憶體中解決的事為什麼要放到本地呢?
這種情況主要是應對回去圖片的方式是從其他介面獲取到的,或者實時生成的時候。雖然很簡單,但覺得說說也挺有意思的。
這裡的模擬方式是假設在網上獲取到了多張base64編碼的圖片,需要將其組合在一起,然後在不儲存在本地情況下直接傳送這張照片。
這個base64編碼的圖片已經儲存在本地了,名字是demo_base64。txt
效果如下:
程式碼如下:
EMAIL_ONLINE_IMAGE_TEMPLATE = “”“
這是一張圖片
”“”def create_online_image_content(): from PIL import Image tpl = Template(EMAIL_ONLINE_IMAGE_TEMPLATE) fp = “demo_base64。txt” if not path。exists(fp): sys。exit(“要傳送的base64編碼的圖片不存在”) msg = MIMEMultipart(“related”) image_name = “demo” with open(fp, “rb”) as rf: base64_data = rf。read() img_data = base64。b64decode(base64_data) # 因為open方法需要一個file-like檔案物件,而我們解碼後的物件型別是bytes型別 # bytes型別沒有檔案物件的read, close方法,所以我們需要透過BytesIO物件包裝一下,它會返回一個file-like檔案物件 img = Image。open(BytesIO(img_data)) img_width, img_height = img。size repeat_times = 5 # compose images ret_img = Image。new(img。mode, (img_width, img_height * repeat_times)) for index in range(repeat_times): ret_img。paste(img, box=(0, index * img_height)) # 因為MIMEImage需要一個bytes物件,所以們需要獲取圖片編碼後的二進位制資料而不是圖片的array資料 img_bytes = BytesIO() # 如果不指定圖片格式,會因為沒有檔名而報錯 ret_img。save(img_bytes, “png”) mime_image = MIMEImage(img_bytes。getvalue()) # 注意: 一定需要<>括號 mime_image。add_header(“Content-ID”, “<%s>” % image_name) msg。attach(mime_image) # 渲染郵件文字內容 text = tpl。render(image_name=image_name) msg_alternative = MIMEMultipart(“alternative”) msg_alternative。attach(MIMEText(text, “html”, “utf-8”)) msg。attach(msg_alternative) return msg這裡很有意思一點是用BytesIO模擬file-like物件。這裡需要安裝PIL哦
傳送一個帶樣式的靜態網頁
前面的程式碼已經足夠說明圖片怎麼發了,這裡透過一個寫了css樣式的表格進行演示
效果如下:
程式碼如下:
EMAIL_TEMPLATE = “”“
一共有以下{{record_size}}條資料
{{label}} | {% endfor %}
---|
{{value}} | {% endfor %}
原始碼地址
https://github。com/youerning/blog/tree/master/sendmail
如果期待後續文章可以關注我的微信公眾號(又耳筆記),頭條號(又耳筆記),github。
後記
其實發送一個附件也是不錯的方式,比如傳送一個生成的PDF, PDF是一個很棒的檔案格式。但是PDF暫時沒用到,以後有機會再說吧。最後要注意的是,手機端的顯示效果跟電腦網頁版的顯示效果是不一樣的。
參考連結
https://www。runoob。com/python/python-email。html