forked from yuban01652/ucas-covid19
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsub.py
277 lines (232 loc) · 10.9 KB
/
sub.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
"""
author: Les1ie
mail: [email protected]
license: CC BY-NC-SA 3.0
"""
import os
import json
import pytz
import hashlib
import smtplib
import requests
from time import sleep
from pathlib import Path
from random import randint
from datetime import datetime
from email.utils import formataddr
from email.mime.text import MIMEText
# 开启debug将会输出打卡填报的数据,关闭debug只会输出打卡成功或者失败,如果使用github actions,请务必设置该选项为False
debug = False
# 忽略网站的证书错误,这很不安全 :(
verify_cert = True
# 全局变量,如果使用自己的服务器运行请根据需要修改 ->以下变量<-
user = "USERNAME" # sep 账号
passwd = r"PASSWORD" # sep 密码
api_key = "" # 可选, server 酱的通知 api key
# 可选,如果需要邮件通知,那么修改下面五行 :)
smtp_port = 465
smtp_server = "smtp.sina.com"
sender_email = "[email protected]"
sender_email_passwd = r""
receiver_email = "[email protected]"
# 可选,如果需要Telegram通知,修改下面
tg_chat_id = "" # 和bot的chat_id
tg_bot_token = r"" # bot的token
# 全局变量,使用自己的服务器运行请根据需要修改 ->以上变量<-
# 如果检测到程序在 github actions 内运行,那么读取环境变量中的登录信息
if os.environ.get('GITHUB_RUN_ID', None):
user = os.environ.get('SEP_USER_NAME', '') # sep账号
passwd = os.environ.get('SEP_PASSWD', '') # sep密码
api_key = os.environ.get('API_KEY', '') # server酱的api,填了可以微信通知打卡结果,不填没影响
# smtp_port = os.environ.get('SMTP_PORT', '465') # 邮件服务器端口,默认为qq smtp服务器端口
# smtp_server = os.environ.get('SMTP_SERVER', 'smtp.qq.com') # 邮件服务器,默认为qq smtp服务器
# sender_email = os.environ.get('SENDER_EMAIL', '') # 发送通知打卡通知邮件的邮箱
sender_email_passwd = os.environ.get('SENDER_EMAIL_PASSWD', "") # 发送通知打卡通知邮件的邮箱密码
# receiver_email = os.environ.get('RECEIVER_EMAIL', '') # 接收打卡通知邮件的邮箱
tg_chat_id = os.environ.get('TG_CHAT_ID', '') # 和bot的chat_id
tg_bot_token = os.environ.get('TG_BOT_TOKEN', '') # bot的token
def login(s: requests.Session, username, password, cookie_file: Path):
# r = s.get(
# "https://app.ucas.ac.cn/uc/wap/login?redirect=https%3A%2F%2Fapp.ucas.ac.cn%2Fsite%2FapplicationSquare%2Findex%3Fsid%3D2")
# print(r.text)
if cookie_file.exists():
cookie = json.loads(cookie_file.read_text(encoding='utf-8'))
s.cookies = requests.utils.cookiejar_from_dict(cookie)
# 测试cookie是否有效
if get_daily(s) == False:
print("cookie失效,进入登录流程")
else:
print("cookie有效,跳过登录环节")
return
payload = {
"username": username,
"password": password
}
r = s.post("https://app.ucas.ac.cn/uc/wap/login/check", data=payload)
# print(r.text)
if r.json().get('m') != "操作成功":
print("登录失败")
message(api_key, sender_email, sender_email_passwd, receiver_email,
tg_bot_token, tg_chat_id, "健康打卡登录失败", "登录失败")
else:
cookie_file.write_text(json.dumps(requests.utils.dict_from_cookiejar(r.cookies), indent=2), encoding='utf-8', )
print("登录成功,cookies 保存在文件 {},下次登录将优先使用cookies".format(cookie_file))
def get_daily(s: requests.Session):
daily = s.get("https://app.ucas.ac.cn/ncov/api/default/daily?xgh=0&app_id=ucas")
# info = s.get("https://app.ucas.ac.cn/ncov/api/default/index?xgh=0&app_id=ucas")
if '操作成功' not in daily.text:
# 会话无效,跳转到了登录页面
print("会话无效")
return False
j = daily.json()
return j.get('d') if j.get('d', False) else False
def submit(s: requests.Session, old: dict):
new_daily = {
'realname': old['realname'],
'number': old['number'],
'szgj_api_info': old['szgj_api_info'],
# 'szgj': old['szgj'],# 2021.8.1 del
# 'old_sfzx': old['sfzx'],# 2021.8.1 del
'sfzx': old['sfzx'],
'szdd': old['szdd'],
'ismoved': 0, # 如果前一天位置变化这个值会为1,第二天仍然获取到昨天的1,而事实上位置是没变化的,所以置0
# 'ismoved': old['ismoved'],
'tw': old['tw'],
# 'bztcyy': old['bztcyy'], # 2021.8.1 del
# 'sftjwh': old['sfsfbh'], # 2020.9.16 del
# 'sftjhb': old['sftjhb'], # 2020.9.16 del
'sfcxtz': old['sfcxtz'],
# 'sfyyjc': old['sfyyjc'],# 2021.8.1 del
# 'jcjgqr': old['jcjgqr'],# 2021.8.1 del
# 'sfjcwhry': old['sfjcwhry'], # 2020.9.16 del
# 'sfjchbry': old['sfjchbry'], # 2020.9.16 del
'sfjcbh': old['sfjcbh'], # 是否接触病患
# 'jcbhlx': old['jcbhlx'], # 2021.1.29 del 接触病患类型
'sfcyglq': old['sfcyglq'], # 是否处于隔离期
# 'gllx': old['gllx'], # 2021.1.29 del 隔离类型
'sfcxzysx': old['sfcxzysx'],
# 'old_szdd': old['szdd'],# 2021.8.1 del
'geo_api_info': old['old_city'], # 保持昨天的结果
'old_city': old['old_city'],
'geo_api_infot': old['geo_api_infot'],
'date': datetime.now(tz=pytz.timezone("Asia/Shanghai")).strftime("%Y-%m-%d"),
# 'fjsj': old['fjsj'], # 返京时间# 2021.8.1 del
# 'ljrq': old['ljrq'], # 离京日期 [email protected]# 2021.8.1 del
# 'qwhd': old['qwhd'], # 去往何地 [email protected]# 2021.8.1 del
# 'chdfj': old['chdfj'], # 从何地返京 [email protected]# 2021.8.1 del
# 'jcbhrq': old['jcbhrq'], # del 2021.1.29 接触病患日期
# 'glksrq': old['glksrq'], # del 2021.1.29 隔离开始日期
# 'fxyy': old['fxyy'],# 2021.8.1 del
# 'jcjg': old['jcjg'],# 2021.8.1 del
# 'jcjgt': old['jcjgt'],# 2021.8.1 del
# 'qksm': old['qksm'],# 2021.8.1 del
# 'remark': old['remark'],
'jcjgqk': old['jcjgqk'],
# 'jcwhryfs': old['jcwhryfs'],# 2021.8.1 del
# 'jchbryfs': old['jchbryfs'],# 2021.8.1 del
'gtshcyjkzt': old['gtshcyjkzt'], # add @2020.9.16
'jrsfdgzgfxdq': old['jrsfdgzgfxdq'], # add @2020.9.16
'jrsflj': old['jrsflj'], # add @2020.9.16
'app_id': 'ucas'
}
check_data_msg = check_submit_data(new_daily) # 检查上报结果
if check_data_msg is not None:
message(api_key, sender_email, sender_email_passwd, receiver_email, tg_bot_token,
tg_chat_id, "每日健康打卡-{}".format(check_data_msg), "{}".format(new_daily))
print("提交数据存在问题,请手动打卡,问题原因: {}".format(check_data_msg))
return
r = s.post("https://app.ucas.ac.cn/ncov/api/default/save", data=new_daily)
if debug:
from urllib.parse import parse_qs, unquote
print("昨日信息:", json.dumps(old, ensure_ascii=False, indent=2))
print("提交信息:",
json.dumps(parse_qs(unquote(r.request.body), keep_blank_values=True), indent=2, ensure_ascii=False))
result = r.json()
if result.get('m') == "操作成功":
print("打卡成功")
else:
print("打卡失败,错误信息: ", r.json().get("m"))
message(api_key, sender_email, sender_email_passwd, receiver_email,
tg_bot_token, tg_chat_id, result.get('m'), new_daily)
def check_submit_data(data: dict):
"""
检查提交数据是否正常
"""
msg = []
# 所在地点
if data['szdd'] != "国内":
msg.append("所在地点不是国内,请手动填报")
# 体温
if int(data['tw']) > 4:
msg.append("体温大于 37.3 度 ,请手动填报")
if data['jrsflj'] == '是':
msg.append("近日有离京经历,请手动填报")
return ";".join(msg) if msg else None
def message(key, sender, mail_passwd, receiver, bot_token, chat_id, subject, msg):
"""
再封装一下 :) 减少调用通知写的代码
"""
if api_key != "":
server_chan_message(key, subject, msg)
if sender_email != "" and receiver_email != "":
send_email(sender, mail_passwd, receiver, subject, msg)
if tg_bot_token != "" and tg_chat_id != "":
send_telegram_message(bot_token, chat_id, "{}\n{}".format(subject, msg))
def server_chan_message(key, title, body):
"""
微信通知打卡结果
"""
# 错误的key也可以发送消息,无需处理 :)
msg_url = "https://sctapi.ftqq.com/{}.send?text={}&desp={}".format(key, title, body)
requests.get(msg_url)
def send_email(sender, mail_passwd, receiver, subject, msg):
"""
邮件通知打卡结果
"""
try:
body = MIMEText(str(msg), 'plain', 'utf-8')
body['From'] = formataddr(["notifier", sender])
body['To'] = formataddr(["me", receiver])
body['Subject'] = "UCAS疫情填报助手通知-" + subject
global smtp_port, smtp_server
if smtp_server == "" or smtp_port == "":
smtp_port = 465
smtp_server = "smtp.qq.com"
smtp = smtplib.SMTP_SSL(smtp_server, smtp_port)
smtp.login(sender, mail_passwd)
smtp.sendmail(sender, receiver, body.as_string())
smtp.quit()
print("邮件发送成功")
except Exception as ex:
print("邮件发送失败")
if debug:
print(ex)
def send_telegram_message(bot_token, chat_id, msg):
"""
Telegram通知打卡结果
python-telegram-bot 只支持 python 3.6或更高的版本
此处使用时再导入以保证向后兼容 python 3.5;
如果要使用 tg 消息通知,请使用 python 3.6或更高的版本
"""
import telegram
bot = telegram.Bot(token=bot_token)
bot.send_message(chat_id=chat_id, text=msg)
def report(username, password):
s = requests.Session()
s.verify = verify_cert # 不验证证书
header = {
"User-Agent": "Mozilla/5.0 (Linux; Android 10; AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 \
Chrome/78.0.3904.62 XWEB/2693 MMWEBSDK/201201 Mobile Safari/537.36 MMWEBID/1300 \
MicroMessenger/7.0.22.1820 WeChat/arm64 Weixin NetType/WIFI Language/zh_CN ABI/arm64"
}
s.headers.update(header)
print(datetime.now(tz=pytz.timezone("Asia/Shanghai")).strftime("%Y-%m-%d %H:%M:%S %Z"))
for i in range(randint(10, 600), 0, -1):
print("\r等待{}秒后填报".format(i), end='')
sleep(1)
cookie_file_name = Path("{}.json".format(hashlib.sha512(username.encode()).hexdigest()[:8]))
login(s, username, password, cookie_file_name)
yesterday = get_daily(s)
submit(s, yesterday)
if __name__ == "__main__":
report(username=user, password=passwd)