Python:定时获取通知内容并发送到邮箱

由于需要看通知内容,但是每次从手机上看需要输入账号密码之后才能看,于是便萌生了用 Python 获取通知内容并定时发送到自己邮箱的想法。

实现并不算复杂,用 BeautifulSoup 抓取内容,Redis 记录文章是否阅读过,Jinja2 是邮件内容的模板引擎。用和风天气的API 在邮件正文前加了个天气预报。

只是有一个点要注意,启动程序前要先留意 locale (Linux 命令)输出的内容是否为zh_CN.UTF-8。最后我是写了个 shell 脚本启动并在运行前 export LC_ALL=zh_CN.UTF-8

Redis
Welcome to Jinja2 — Jinja2 Documentation (2.9)
yagmail 0.10.190 : Python Package Index
API说明文档 | 和风天气
Beautiful Soup 4.4.0 文档 — beautifulsoup 4.4.0 文档

程序在启动的时候加-t的参数只会给自己的邮箱发邮件,用作测试(当然需要提前配置好)
实现如下:

主文件

#!/usr/bin/python3
# -*- coding:utf-8 -*-

'''
【留意!!】
启动程序前要先留意 locale (Linux 命令)输出的内容是否为zh_CN.UTF-8
建议写 shell 脚本启动并在运行前 export LC_ALL=zh_CN.UTF-8 
'''

from conf import *
from sys import argv
from urllib.parse import unquote
from bs4 import BeautifulSoup
from jinja2 import  Environment,FileSystemLoader,select_autoescape
import re,os,json,time,redis,yagmail,requests

session = requests.Session()
session.headers.update({'UserAgent':'Mozilla/5.0 (X11; Linux x86_64; rv:57.0) Gecko/20100101 Firefox/57.0'})

jinja2_env = Environment(
    loader = FileSystemLoader(os.getcwd()+'/template'),
    autoescape = select_autoescape(['html'])
)


def printf(string):
    if string:
        print(time.strftime("%Y-%m-%d %H:%M:%S : ", time.localtime()) + string)


def article_id_exist(id):
    if not id:
        printf('empty article id')
        return False

    if not id.isdigit():
        printf('need number instead of other value type')
        return False

    r = redis.StrictRedis(host='localhost', port=6379, db=0)

    if not r.get(id):
        r.set(id,"True")
        return False
    else:
        return True


def get_weather_data():
    weather_request_url = WEATHER_API_URL + 'city=' + WEATHER_API_CITY + '&&key=' + WEATHER_API_KEY
    weather_request = requests.get(weather_request_url)
    weather_data = []
    if weather_request.status_code == 200:
        weather_data = json.loads(weather_request.content.decode(encoding='utf-8'))
    return weather_data


def get_index():
    session.get(url=ROOT_URL+'/UserLogin.aspx')
    result = session.post(ROOT_URL+'/UserLogin.aspx', data=LOGIN_DATA)
    if result.status_code == 200 and result.url == ROOT_URL+'/':
        category = session.get(ROOT_URL + '/ArticleList.aspx?category=4')
        if category.status_code == 200:
            return category.content
    else:
        printf('login failed '+str(result.status_code) )


def parse_html(html):
    if not html:
        printf('empty html')
        return

    html_soup = BeautifulSoup(html, 'lxml')
    articles = html_soup.find('div', attrs={'class': 'articles'})

    if not articles:
        printf('article not found')
        return

    article_result = []

    for val in articles.find_all('p'):
        article_id = val.find('a')['href'][-6:]
        article_url = ROOT_URL + val.find('a')['href'][1:]
        article_date = val.find_all('span')[1].getText()[:-1]
        article_title = val.find('a')['title']
        article_author = val.span['title']
        article_attachment = []
        article_excerpt = ''

        if article_id_exist(article_id):
            printf('article exist in database %s' % (article_title))
            continue

        article_detail = session.get(url=article_url)

        if article_detail.status_code != 200:
            printf('get article detail error %s' % (article_id) )
            continue

        article_soup = BeautifulSoup(article_detail.content,'lxml')
        article_content = article_soup.find('div', attrs={'id': 'articleBody'})

        article_link = article_content.find_all('a')
        attachment_url_pattern = re.compile(r'http://news.gdut.edu.cn/DepartmentUploadFiles/(.+)/files/(.+)')
        for link in article_link:
            if 'http://news.gdut.edu.cn/DepartmentUploadFiles' not in link['href']:
                printf('%s do not have file attachment' % (link['href']))
                continue

            match = attachment_url_pattern.match(link['href'])
            if not match:
                printf('%s do not have file attachment' % (link['href']))
                continue

            attachment_name = match.group(2)
            attachment_url = link['href']

            if '%' in attachment_name:
                attachment_name = unquote(attachment_name)

            article_attachment.append({'attach_name':attachment_name,'attach_url':attachment_url})

        info = ''.join(article_content.getText().split())
        info = info.replace(article_title, '')
        info = info.replace('单位:'+article_author,'')
        article_excerpt = article_excerpt.join(info[:150])

        article_result.append(
            {
                'url':article_url,
                'date':article_date,
                'title':article_title,
                'author':article_author,
                'excerpt':article_excerpt,
                'attachment':article_attachment
            }
        )
    return article_result


if __name__ == '__main__':

    welcome_string = [
        '周日:今天是周末的最后一天,好好珍惜时间\n',
        '周一:你从周末的作息里调整过来了吗?把上周的通知邮件都删了吧\n',
        '周二:吾日三省吾身\n',
        '周三:生活仍将继续\n',
        '周四:未来近在咫尺\n',
        '周五:明天就是周末了,加油!\n',
        '周六:你今天打算做什么?别浪费时间\n',
    ]

    welcome_content = welcome_string[ int( time.strftime('%w',time.localtime(time.time())) ) ]
    update_content = '最近更新:'+VERSION+':'+ANNOUNCEMENT+'\n'

    weather_data = get_weather_data()
    printf('get weather data finish')

    weather_render = jinja2_env.get_template('weather.html')

    now = weather_data['HeWeather5'][0]['now']
    forecast =  weather_data['HeWeather5'][0]['hourly_forecast'];
    weather_content = weather_render.render(now=now,forecast=forecast)

    index = get_index()
    article_data = parse_html(index)

    article_render = jinja2_env.get_template('article.html')
    if article_data:
        article_content = article_render.render(articles=article_data)
    else:
        article_content = article_render.render()

    mail_client = yagmail.SMTP(user=SEND_MAIL_USER, password=SEND_MAIL_PWD, host=SEND_MAIL_HOST, port=SEND_MAIL_PORT)
    mail_content = welcome_content + weather_content + update_content + article_content
    if len(argv) == 2 and '-t' in argv:
        for addr in SEND_TO_LIST_TEST:
            printf('sending[test user]: ' + addr)
            mail_client.send(addr, subject=SEND_MAIL_SUBJECT, contents=mail_content)
            time.sleep(1)
    else:
        for addr in SEND_TO_LIST:
            printf('sending : '+addr)
            mail_client.send(addr,subject=SEND_MAIL_SUBJECT,contents =mail_content)
            time.sleep(1)

同级目录下的 conf.py 的配置文件

#!/usr/bin/python3
# -*- coding:utf-8 -*-

import time

#通知网站的地址
ROOT_URL = 'http://test.com'

LOGIN_DATA = {}
LOGIN_DATA['__VIEWSTATE'] = '/wEPDwUKLTQwOTA4NzE2NmQYAQUeX19Db250cm9sc1JlcXVpcmVQb3N0QmFja0tleV9fFgEFI2N0bDAwJENvbnRlbnRQbGFjZUhvbGRlcjEkQ2hlY2tCb3gxBufpEJuDDaf6eTj0A4Cn2Erf8u98KcGrQqATTB3mEaQ='
LOGIN_DATA['__EVENTVALIDATION'] = '/wEWBQKb37HjDwLgvLy9BQKi4MPwCQL+zqO2BAKA4sljg4IvzC7ksG01o7aN0RZUOKEC4lV0bTeXI4zrbaQsj0c='

# 联系校内人员获取账号密码,此处的账号密码无效
LOGIN_DATA['ctl00$ContentPlaceHolder1$userEmail'] = 'test'
LOGIN_DATA['ctl00$ContentPlaceHolder1$userPassWord'] = 'test'

LOGIN_DATA['ctl00$ContentPlaceHolder1$CheckBox1'] = 'on'
LOGIN_DATA['ctl00$ContentPlaceHolder1$Button1'] = '%E7%99%BB%E5%BD%95'

#发送者邮箱
SEND_MAIL_USER = 'account'
#发送者邮箱对应的密码
SEND_MAIL_PWD = 'password'
#腾讯企业邮箱
SEND_MAIL_HOST = 'smtp.exmail.qq.com'
#发送端口
SEND_MAIL_PORT = 465
#邮件正文标题
SEND_MAIL_SUBJECT = time.strftime("%Y-%m-%d",time.localtime()) + '@今日校内通知'
#接收邮件的人
SEND_TO_LIST = [
   'mail@mail.com',
]
#用来测试接收邮件的用户,加上-t选项即可
SEND_TO_LIST_TEST = ['mail@mail.com']
#和风天气API地址
WEATHER_API_URL = 'https://free-api.heweather.com/v5/weather?'
#天气API城市,拼音汉字均可
WEATHER_API_CITY = 'guangzhou'
#免费版key,一天4000次调用,注册后可用
WEATHER_API_KEY = 'key'

ANNOUNCEMENT = '重构,使用模板引擎取代字符串拼接生成邮件内容(https://github.com/ypingcn/)'
VERSION = '2017.09.26'

template文件夹的内容是邮件正文的模板

  • article.html
{%- if articles %}
<p> 今日的新闻通知如下 </p>
    <ul>
    {%- for article in articles %}
    <li>
        <a href='{{article.url}}'>
            <font color="red"> {{ article.title }} </font>
        </a>
        {{ article.author }} - {{ article.date }}
        {{ article.excerpt }}
        {%- for link in article.attachment %}
        <a href='{{link.attach_url}}'>{{ link.attach_name }}</a>
        {%- endfor %}
    </li>
    {%- endfor %}
</ul>
{%- else %}
<p> 暂无未读的新闻通知 </p>
{%- endif %}
  • weather.html
<p>天气:{{ now.cond.txt }},气温:{{ now.tmp }}℃,体感温度:{{ now.fl }}摄氏度</p>
<br>未来几个小时内的天气预报为:
{%- for hour in forecast %}
<br>{{ hour.date }} : {{ hour.cond.txt }}
{%- endfor %}

写的不是太好,还是有很多需要改正的地方。以后再作修改。

来自个人 Python 文集

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,377评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,390评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,967评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,344评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,441评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,492评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,497评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,274评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,732评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,008评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,184评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,837评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,520评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,156评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,407评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,056评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,074评论 2 352

推荐阅读更多精彩内容