This commit is contained in:
RainSun 2020-04-15 17:21:54 +08:00
parent b9d85df2c2
commit 4306439f5d
7 changed files with 570 additions and 288 deletions

View File

@ -15,6 +15,8 @@ pip install bs4
pip install gunicorn
pip install pymongo
pip install flask_cors
// 根据依赖文件安装环境
pip install -r requirements.txt -i https://pypi.tuna.tsinghua.edu.cn/simple
// 设置全局变量
export FLASK_APP=coc.py
export FLASK_ENV=development

0
lib/__init__.py Normal file
View File

View File

@ -1,5 +1,5 @@
# coding=utf-8
from crawler import connection, getGrade, getSchedule
from crawler import Crawler
import json
from hashlib import md5
from urllib.parse import urlencode, unquote_plus
@ -10,20 +10,22 @@ from db import addRank, getRank
# 这里三个接口公用一个session所以合并成一个接口一个session走到底一次性返回所有数据
def manageLogin(request):
# json化应该能当dict用
data_cache = json.loads(request.form['data'])
data = json.loads(request.form['data'])
# MD5校验
checked = checkData(data_cache)
data_cache.pop('sign')
checked = checkData(data)
if checked:
# 创建会话
res = connect(data_cache)
if res['errcode'] == 200:
# 登录成功并进行查询
grade = getGrade(res['ip'], res['s'])
schedule = getSchedule(res['ip'], res['s'])
return {'errcode': '200', 'errmsg': 'ok', 'student_name':res['student_name'], 'student_id':res['student_id'], 'grade': grade, 'schedule': schedule}
phone = ''
if data.get('phone'):
phone = data['phone']
c = Crawler(data['cid'], data['pwd'], phone)
res = c.connection()
if res[-1] == 200:
c.getGrade()
c.getSchedule()
return c.getData()
else:
return res
return {'errcode': res[-1], 'errmsg': res[0]}
else:
return {'errcode': 100, 'errmsg':'数据校验失败'}
@ -70,19 +72,3 @@ def checkData(data):
md.update(d.encode('utf-8'))
r = md.hexdigest().upper()
return r == data['sign']
# 创建会话
def connect(data):
cid = data['cid']
pwd = data['pwd']
# 进行登录
phone = ''
if data.get('phone'):
phone = data['phone']
try:
# 这里教务没问题账户没问题就是200密码错了就是101
res = connection(cid,pwd,phone)
return res
except:
# 这了就是教务挂了
return {'errcode': 102, 'errmsg':'教务挂了'}

View File

@ -4,275 +4,342 @@ from urllib.parse import quote
import base64
from bs4 import BeautifulSoup
import random
import sys
def connection(username,password,phone):
s = requests.Session()
# 获取统一身份系统的网页
r = s.get(url='http://mysso-cust-edu-cn-s.webvpn.cust.edu.cn:8118/cas/login?service=https%3A%2F%2Fwebvpn.cust.edu.cn%2Fauth%2Fcas_validate%3Fentry_id%3D1')
# soup = BeautifulSoup(r.text,'lxml')
soup = BeautifulSoup(r.text,'html.parser')
execution=soup.find_all(name='input')[6]['value']
formdata={
'username':username,
'password':password,
'execution':execution,
'_eventId':'submit',
'geolocation':''
}
r = s.post(url='http://mysso-cust-edu-cn-s.webvpn.cust.edu.cn:8118/cas/login?service=https%3A%2F%2Fwebvpn.cust.edu.cn%2Fauth%2Fcas_validate%3Fentry_id%3D1',data=formdata)
soup=BeautifulSoup(r.text,'html.parser')
flag = soup.find(name='title')
if(flag.text=="手机号设置"):
execution=soup.find_all(name='input')[1]['value']
formdata = {
'phone': phone,
'execution': execution,
'_eventId': 'submit',
'submit': '提交'
class Crawler(object):
def __init__(self, username, password, phone):
self.__username = username
self.__password = password
self.__phone = phone
self.__session = None
self.__ip = None
self.__student_id = None
self.__student_name = None
self.__grade_data = None
self.__schedule_data = None
# 链接教务 -----------------------------------------------------------------------------
def connection(self):
try:
self.__session = requests.Session()
# 获取统一身份系统的网页
r = self.__session.get(
url='http://mysso-cust-edu-cn-s.webvpn.cust.edu.cn:8118/cas/login?service=https%3A%2F%2Fwebvpn.cust.edu.cn%2Fauth%2Fcas_validate%3Fentry_id%3D1')
soup = BeautifulSoup(r.text, 'html.parser')
execution = soup.find_all(name='input')[6]['value']
formdata = {
'username': self.__username,
'password': self.__password,
'execution': execution,
'_eventId': 'submit',
'geolocation': ''
}
r = self.__session.post(
url='http://mysso-cust-edu-cn-s.webvpn.cust.edu.cn:8118/cas/login?service=https%3A%2F%2Fwebvpn.cust.edu.cn%2Fauth%2Fcas_validate%3Fentry_id%3D1', data=formdata)
soup = BeautifulSoup(r.text, 'html.parser')
flag = soup.find(name='title')
if(flag.text == "手机号设置"):
if self.__phone == '':
return ('账号或者密码错误', 401)
execution = soup.find_all(name='input')[1]['value']
formdata = {
'phone': self.__phone,
'execution': execution,
'_eventId': 'submit',
'submit': '提交'
}
r = self.__session.post(
url="http://mysso-cust-edu-cn-s.webvpn.cust.edu.cn:8118/cas/login?service=https%3A%2F%2Fwebvpn.cust.edu.cn%2Fauth%2Fcas_validate%3Fentry_id%3D1", data=formdata)
r = self.__session.get(
url='http://portal-cust-edu-cn-s.webvpn.cust.edu.cn:8118/custp/index')
soup = BeautifulSoup(r.text, 'html.parser')
try:
self.__ip = soup.findAll(name='a')[7]['href'][7:].split("-")
except:
return ('账号或者密码错误', 401)
r = self.__session.get(url='http://mysso-cust-edu-cn-s.webvpn.cust.edu.cn:8118/cas/login?service=http://' +
self.__ip[0] + '.' + self.__ip[1] + '.' + self.__ip[2] + '.' + self.__ip[3] + ':8080/welcome', allow_redirects=False)
ticket = r.headers['Location'][68:]
asp_net_sessionid_param = {'Ticket': ticket, 'Url': 'http://' +
self.__ip[0] + '.' + self.__ip[1] + '.' + self.__ip[2] + '.' + self.__ip[3] + ':8080/welcome'}
asp_net_sessionid_param = base64.b64encode(
quote(json.dumps(asp_net_sessionid_param)).encode('utf-8')).decode('utf-8')
asp_net_sessionid_param = {'param': asp_net_sessionid_param}
headers = {'Content-Type': 'application/json'}
r = self.__session.post(url='http://' + self.__ip[0] + '-' + self.__ip[1] + '-' + self.__ip[2] + '-' + self.__ip[3] +
'-8080-p.webvpn.cust.edu.cn:8118/api/LoginApi/LGSSOLocalLogin?sf_request_type=ajax', data=json.dumps(asp_net_sessionid_param), headers=headers)
data = json.loads(r.content.decode('utf-8'))
self.__student_name = data['data']['StudentDto']['XM']
self.__student_id = data['data']['StudentDto']['XH']
return ('ok', 200)
except Exception as e:
print(e)
return ('教务挂了', 502)
# 获取课表 -----------------------------------------------------------------------------
def getGrade(self):
headers = {'Content-Type': 'application/json'}
r = self.__session.post(
url='http://' + self.__ip[0] + '-' + self.__ip[1] + '-' + self.__ip[2] + '-' + self.__ip[3] +
'-8080-p.webvpn.cust.edu.cn:8118/api/ClientStudent/QueryService/GradeQueryApi/GetDataByStudent?sf_request_type=ajax',
data=json.dumps({"param": "JTdCJTIyU2hvd0dyYWRlVHlwZSUyMiUzQTElN0Q=", "__permission": {"MenuID": "4443798E-EB6E-4D88-BFBD-BB0A76FF6BD5",
"Operation": 0}, "__log": {"MenuID": "4443798E-EB6E-4D88-BFBD-BB0A76FF6BD5", "Logtype": 6, "Context": "查询"}}),
headers=headers
)
data = json.loads(r.content.decode('utf-8'))
if data['state'] != 0:
return ('教务挂了', 502)
# 分解数据并重命名
total = data['data']['GradeStatistics']
split = data['data']['GradeList']
# 成绩总览
total_grade = {
'total_GPA': total['PJJD'],
'total_credit': total['SDXF'],
'total_kill': total['TGMS'],
'total_dead': total['WTGMS']
}
r = s.post(url="http://mysso-cust-edu-cn-s.webvpn.cust.edu.cn:8118/cas/login?service=https%3A%2F%2Fwebvpn.cust.edu.cn%2Fauth%2Fcas_validate%3Fentry_id%3D1",data=formdata)
r = s.get(url='http://portal-cust-edu-cn-s.webvpn.cust.edu.cn:8118/custp/index')
soup=BeautifulSoup(r.text,'html.parser')
try:
ip = soup.findAll(name='a')[7]['href'][7:].split("-")
except:
return {'errcode': 101, 'errmsg':'账号或者密码错误'}
r = s.get(url='http://mysso-cust-edu-cn-s.webvpn.cust.edu.cn:8118/cas/login?service=http://'+ip[0]+'.'+ip[1]+'.'+ip[2]+'.'+ip[3]+':8080/welcome',allow_redirects=False)
ticket = r.headers['Location'][68:]
asp_net_sessionid_param = {'Ticket':ticket,'Url':'http://'+ip[0]+'.'+ip[1]+'.'+ip[2]+'.'+ip[3]+':8080/welcome'}
asp_net_sessionid_param = base64.b64encode(quote(json.dumps(asp_net_sessionid_param)).encode('utf-8')).decode('utf-8')
asp_net_sessionid_param = {'param':asp_net_sessionid_param}
headers = {'Content-Type': 'application/json'}
r = s.post(url='http://'+ip[0]+'-'+ip[1]+'-'+ip[2]+'-'+ip[3]+'-8080-p.webvpn.cust.edu.cn:8118/api/LoginApi/LGSSOLocalLogin?sf_request_type=ajax',data=json.dumps(asp_net_sessionid_param),headers=headers)
data = json.loads(r.content.decode('utf-8'))
student_name = data['data']['StudentDto']['XM']
student_id = data['data']['StudentDto']['XH']
return {'errcode': 200, 'errmsg': 'ok', 'ip': ip, 's': s,'student_name':student_name,'student_id':student_id}
def getGrade(Ip, S):
headers = {'Content-Type': 'application/json'}
r = S.post(
url='http://'+Ip[0]+'-'+Ip[1]+'-'+Ip[2]+'-'+Ip[3]+'-8080-p.webvpn.cust.edu.cn:8118/api/ClientStudent/QueryService/GradeQueryApi/GetDataByStudent?sf_request_type=ajax',
data=json.dumps({"param":"JTdCJTIyU2hvd0dyYWRlVHlwZSUyMiUzQTElN0Q=","__permission":{"MenuID":"4443798E-EB6E-4D88-BFBD-BB0A76FF6BD5","Operation":0},"__log":{"MenuID":"4443798E-EB6E-4D88-BFBD-BB0A76FF6BD5","Logtype":6,"Context":"查询"}}),
headers=headers
)
data = json.loads(r.content.decode('utf-8'))
if data['state'] != 0:
return {'errcode': 102, 'errmsg': '教务挂了'}
#分解数据并重命名
total = data['data']['GradeStatistics']
split = data['data']['GradeList']
#成绩总览
GradeStatistics = {
'total_GPA': total['PJJD'],
'total_credit': total['SDXF'],
'total_kill': total['TGMS'],
'total_dead': total['WTGMS']
}
#提取第一和最后一学期
first_term = split[0]['KSXNXQ']
last_term = split[len(split)-1]['KSXNXQ']
#转换成int元组
first_term = (int(first_term[0:4]),int(first_term[4:5]))
last_term = (int(last_term[0:4]),int(last_term[4:5]))
print(first_term, last_term)
#生成中间学期
total_term = [];
for i in range(last_term[0],first_term[0] + 1):
for j in range(1,3):
total_term.append(str(i) + str(j))
if i == first_term[0] and j == first_term[1]:
break
total_term.reverse()
grade_list = []
#当前学期索引/上学期总通过/上学期总挂科/上学期总学分/上学期总学分绩点/上一课程通过次数
this_term, last_term_kill, last_term_dead, last_term_credit, last_term_c_x_g, last_lesson_kill = 0, 0, 0, 0, 0, 0
#上学期课程列表
last_term_grade_list = []
#上一课程名称
last_lesson_name = ''
#flag,将是否通过延后一个循环
flag = True
#总必修学分
total_bixiu_credit = 0;
#总必修学分绩点
total_bixiu_c_x_g = 0;
#遍历课程
for item in split:
if not item['YXCJ']:
continue
#如果和上一个课程重名
if item['LessonInfo']['KCMC'] == last_lesson_name:
#判断是否通过
if item['YXCJ'] >= 60:
#如果通过贡献1通过
last_lesson_kill += 1
#贡献总学分绩点
last_term_c_x_g += item['XF'] * (item['YXCJ'] - 50) / 10
if item['KCXZ'] == '必修':
total_bixiu_c_x_g += item['XF'] * (item['YXCJ'] - 50) / 10
total_bixiu_credit += item['XF']
else:#如果不重名
if flag:#将else中的判断延后一个循环
flag = False
else:
#如果上一课程通过
if last_lesson_kill > 0:
#贡献上学期通过数
last_term_kill += 1
else:
#贡献上学期挂科数
last_term_dead += 1
last_lesson_kill = 0
#更新上一课程名称
last_lesson_name = item['LessonInfo']['KCMC']
#如果不是当前学期
if item['KSXNXQ'] != total_term[this_term]:
#成绩列表添加上学期数据
grade_list.append({
'term_time': total_term[this_term],
'term_GPA': last_term_c_x_g / last_term_credit,
'term_kill': last_term_kill,
'term_dead': last_term_dead,
'term_credit': last_term_credit,
'term_grade': last_term_grade_list
})
#当前学期索引+1
while item['KSXNXQ'] != total_term[this_term]:
this_term += 1
#初始化所有值
last_term_kill, last_term_dead, last_term_credit = 0, 0, item['XF']
last_term_grade_list = []
#如果通过
# 提取第一和最后一学期
first_term = split[0]['KSXNXQ']
last_term = split[len(split)-1]['KSXNXQ']
# 转换成int元组
first_term = (int(first_term[0:4]), int(first_term[4:5]))
last_term = (int(last_term[0:4]), int(last_term[4:5]))
# 生成中间学期
total_term = []
for i in range(last_term[0], first_term[0] + 1):
for j in range(1, 3):
total_term.append(str(i) + str(j))
if i == first_term[0] and j == first_term[1]:
break
total_term.reverse()
grade_list = []
# 当前学期索引/上学期总通过/上学期总挂科/上学期总学分/上学期总学分绩点/上一课程通过次数
this_term, last_term_kill, last_term_dead, last_term_credit, last_term_c_x_g, last_lesson_kill = 0, 0, 0, 0, 0, 0
# 上学期课程列表
last_term_grade_list = []
# 上一课程名称
last_lesson_name = ''
# flag,将是否通过延后一个循环
flag = True
# 总必修学分
total_bixiu_credit = 0
# 总必修学分绩点
total_bixiu_c_x_g = 0
# 遍历课程
for item in split:
if not item['YXCJ']:
continue
# 如果和上一个课程重名
if item['LessonInfo']['KCMC'] == last_lesson_name:
# 判断是否通过
if item['YXCJ'] >= 60:
#贡献总学分绩点
last_term_c_x_g = item['XF'] * (item['YXCJ'] - 50) / 10
#贡献通过次数
# 如果通过贡献1通过
last_lesson_kill += 1
if item['KCXZ'] == '必修':
total_bixiu_c_x_g += item['XF'] * (item['YXCJ'] - 50) / 10
total_bixiu_credit += item['XF']
else:
last_term_c_x_g = 0
else:#如果是当前学期
#贡献总学分
last_term_credit += item['XF']
#如果通过
if item['YXCJ'] >= 60:
#贡献通过数
last_lesson_kill += 1
#贡献学分绩点
# 贡献总学分绩点
last_term_c_x_g += item['XF'] * (item['YXCJ'] - 50) / 10
if item['KCXZ'] == '必修':
total_bixiu_c_x_g += item['XF'] * (item['YXCJ'] - 50) / 10
total_bixiu_c_x_g += item['XF'] * \
(item['YXCJ'] - 50) / 10
total_bixiu_credit += item['XF']
#加入学期成绩列表
last_term_grade_list.append({
'title': item['LessonInfo']['KCMC'],
'credit': item['XF'],
'grade': item['ShowYXCJ'],
'kill': 'yes' if (item['YXCJ'] >= 60) else 'no',
'class': item['KSXZ']
else: # 如果不重名
if flag: # 将else中的判断延后一个循环
flag = False
else:
# 如果上一课程通过
if last_lesson_kill > 0:
# 贡献上学期通过数
last_term_kill += 1
else:
# 贡献上学期挂科数
last_term_dead += 1
last_lesson_kill = 0
# 更新上一课程名称
last_lesson_name = item['LessonInfo']['KCMC']
# 如果不是当前学期
if item['KSXNXQ'] != total_term[this_term]:
# 成绩列表添加上学期数据
grade_list.append({
'term_time': total_term[this_term],
'term_GPA': last_term_c_x_g / last_term_credit,
'term_kill': last_term_kill,
'term_dead': last_term_dead,
'term_credit': last_term_credit,
'term_grade': last_term_grade_list
})
# 当前学期索引+1
while item['KSXNXQ'] != total_term[this_term]:
this_term += 1
# 初始化所有值
last_term_kill, last_term_dead, last_term_credit = 0, 0, item['XF']
last_term_grade_list = []
# 如果通过
if item['YXCJ'] >= 60:
# 贡献总学分绩点
last_term_c_x_g = item['XF'] * (item['YXCJ'] - 50) / 10
# 贡献通过次数
last_lesson_kill += 1
if item['KCXZ'] == '必修':
total_bixiu_c_x_g += item['XF'] * \
(item['YXCJ'] - 50) / 10
total_bixiu_credit += item['XF']
else:
last_term_c_x_g = 0
else: # 如果是当前学期
# 贡献总学分
last_term_credit += item['XF']
# 如果通过
if item['YXCJ'] >= 60:
# 贡献通过数
last_lesson_kill += 1
# 贡献学分绩点
last_term_c_x_g += item['XF'] * \
(item['YXCJ'] - 50) / 10
if item['KCXZ'] == '必修':
total_bixiu_c_x_g += item['XF'] * \
(item['YXCJ'] - 50) / 10
total_bixiu_credit += item['XF']
# 加入学期成绩列表
last_term_grade_list.append({
'title': item['LessonInfo']['KCMC'],
'credit': item['XF'],
'grade': item['ShowYXCJ'],
'kill': 'yes' if (item['YXCJ'] >= 60) else 'no',
'class': item['KSXZ']
})
# 补充最后一次遍历的数据
if last_lesson_kill > 0:
last_term_kill += 1
else:
last_term_dead += 1
grade_list.append({
'term_time': total_term[this_term],
'term_GPA': last_term_c_x_g / last_term_credit,
'term_kill': last_term_kill,
'term_dead': last_term_dead,
'term_credit': last_term_credit,
'term_grade': last_term_grade_list
})
#补充最后一次遍历的数据
if last_lesson_kill > 0:
last_term_kill += 1
else:
last_term_dead += 1
grade_list.append({
'term_time': total_term[this_term],
'term_GPA': last_term_c_x_g / last_term_credit,
'term_kill': last_term_kill,
'term_dead': last_term_dead,
'term_credit': last_term_credit,
'term_grade': last_term_grade_list
})
GradeStatistics['total_bixiu_GPA'] = total_bixiu_c_x_g / total_bixiu_credit;
#合并数据
data_cache = {
'total': GradeStatistics,
'split': grade_list
}
return {'errcode': 200, 'errmsg': 'ok', 'data': data_cache}
total_grade['total_bixiu_GPA'] = total_bixiu_c_x_g / \
total_bixiu_credit
# 合并数据
self.__grade_data = {
'total': total_grade,
'split': grade_list
}
return ('ok', 200)
def getSchedule(Ip, S):
headers = {'Content-Type': 'application/json'}
r = S.post(
url='http://'+Ip[0]+'-'+Ip[1]+'-'+Ip[2]+'-'+Ip[3]+'-8080-p.webvpn.cust.edu.cn:8118/api/ClientStudent/Home/StudentHomeApi/GetHomeCurWeekTime?sf_request_type=ajax',
data=json.dumps({"param":"JTdCJTdE","__permission":{"MenuID":"F71C97D5-D3E2-4FDA-9209-D7FA8626390E","Operation":0},"__log":{"MenuID":"F71C97D5-D3E2-4FDA-9209-D7FA8626390E","Logtype":6,"Context":"查询"}}),
headers=headers
)
CurWeek = json.loads(r.content.decode('utf-8'))['data']['CurWeek']
r = S.post(
url='http://'+Ip[0]+'-'+Ip[1]+'-'+Ip[2]+'-'+Ip[3]+'-8080-p.webvpn.cust.edu.cn:8118/api/ClientStudent/Home/StudentHomeApi/QueryStudentScheduleData?sf_request_type=ajax',
data=json.dumps({"param":"JTdCJTdE","__permission":{"MenuID":"F71C97D5-D3E2-4FDA-9209-D7FA8626390E","Operation":0},"__log":{"MenuID":"F71C97D5-D3E2-4FDA-9209-D7FA8626390E","Logtype":6,"Context":"查询"}}),
headers=headers
)
data = json.loads(r.content.decode('utf-8'))
if data['state'] != 0:
return {'errcode': 102, 'errmsg':'教务挂了'}
time = ['AM__TimePieces','PM__TimePieces','EV__TimePieces']
data = data['data']['AdjustDays']
days_per_week = [0] * 23
lesson = [[0] * 6 for _ in range(7)]
lesson_set = {}
color_set = [0] * 9
color_used = 9
for i in range(7):
for j in range(3):
for k in range(2):
if( data[i][time[j]][k]['Dtos'] ):
lesson[i][j*2+k] = []
for l in data[i][time[j]][k]['Dtos']:
temp_lesson = {}
Time = [0] * 23
mod = ''
for m in l['Content']:
if temp_lesson.get(m['Key']):
temp_lesson[m['Key']] += ','+m['Name']
# 获取成绩 -----------------------------------------------------------------------------
def getSchedule(self):
headers = {'Content-Type': 'application/json'}
r = self.__session.post(
url='http://'+self.__ip[0]+'-'+self.__ip[1]+'-'+self.__ip[2]+'-'+self.__ip[3] +
'-8080-p.webvpn.cust.edu.cn:8118/api/ClientStudent/Home/StudentHomeApi/GetHomeCurWeekTime?sf_request_type=ajax',
data=json.dumps({"param": "JTdCJTdE", "__permission": {"MenuID": "F71C97D5-D3E2-4FDA-9209-D7FA8626390E",
"Operation": 0}, "__log": {"MenuID": "F71C97D5-D3E2-4FDA-9209-D7FA8626390E", "Logtype": 6, "Context": "查询"}}),
headers=headers
)
CurWeek = json.loads(r.content.decode('utf-8'))['data']['CurWeek']
r = self.__session.post(
url='http://'+self.__ip[0]+'-'+self.__ip[1]+'-'+self.__ip[2]+'-'+self.__ip[3] +
'-8080-p.webvpn.cust.edu.cn:8118/api/ClientStudent/Home/StudentHomeApi/QueryStudentScheduleData?sf_request_type=ajax',
data=json.dumps({"param": "JTdCJTdE", "__permission": {"MenuID": "F71C97D5-D3E2-4FDA-9209-D7FA8626390E",
"Operation": 0}, "__log": {"MenuID": "F71C97D5-D3E2-4FDA-9209-D7FA8626390E", "Logtype": 6, "Context": "查询"}}),
headers=headers
)
data = json.loads(r.content.decode('utf-8'))
if data['state'] != 0:
return ('教务挂了', 502)
time = ['AM__TimePieces', 'PM__TimePieces', 'EV__TimePieces']
data = data['data']['AdjustDays']
days_per_week = [0] * 23
lesson = [[0] * 6 for _ in range(7)]
lesson_set = {}
color_set = [0] * 9
color_used = 9
for i in range(7):
for j in range(3):
for k in range(2):
if(data[i][time[j]][k]['Dtos']):
lesson[i][j*2+k] = []
for l in data[i][time[j]][k]['Dtos']:
temp_lesson = {}
Time = [0] * 23
mod = ''
for m in l['Content']:
if temp_lesson.get(m['Key']):
temp_lesson[m['Key']] += ','+m['Name']
else:
temp_lesson[m['Key']] = m['Name']
if lesson_set.get(l['Content'][0]['Name']):
temp_lesson['color'] = lesson_set[l['Content'][0]['Name']]
else:
temp_lesson[m['Key']] = m['Name']
if lesson_set.get(l['Content'][0]['Name']):
temp_lesson['color'] = lesson_set[l['Content'][0]['Name']]
else:
color = random.randint(0, 8)
while color_set[color]:
if color_used <= 0:
break;
color = random.randint(0, 8)
temp_lesson['color'] = color
lesson_set[l['Content'][0]['Name']] = color
color_used -= 1
color_set[color] = 1
temp_Time = temp_lesson['Time']
if '单周' in temp_Time:
mod = 'single'
temp_Time = temp_Time[0:len(temp_Time)-5]
elif '双周' in temp_Time:
mod = 'double'
temp_Time = temp_Time[0:len(temp_Time)-5]
else:
mod = 'all'
temp_Time = temp_Time[0:-1]
temp_Time = temp_Time.split(',')
index = 0
for n in temp_Time:
temp_Time[index] = n.split('-')
index += 1
index = 0
for n in temp_Time:
if len(n) > 1 :
for o in range( int(n[0]) , int(n[1]) + 1):
if ( o%2==0 and mod is 'double' ) or ( o%2==1 and mod is 'single' ) or ( mod is 'all' ):
days_per_week[o] = max(days_per_week[o] , i+1)
Time[o] = 1
else:
Time[o] = 0
while color_set[color]:
if color_used <= 0:
break
color = random.randint(0, 8)
temp_lesson['color'] = color
lesson_set[l['Content'][0]['Name']] = color
color_used -= 1
color_set[color] = 1
temp_Time = temp_lesson['Time']
temp_lesson['Time'] = temp_Time[0:int(
temp_Time.find('') + 1)]
if '单周' in temp_Time:
mod = 'single'
# temp_Time = temp_Time[0:len(temp_Time)-5]
elif '双周' in temp_Time:
mod = 'double'
# temp_Time = temp_Time[0:len(temp_Time)-5]
else:
days_per_week[int(n[0])] = max(days_per_week[int(n[0])] , i+1)
Time[int(n[0])] = 1
index += 1
temp_lesson['Time_split'] = Time
lesson[i][j*2+k].append(temp_lesson)
data_cache = {'lesson':lesson,'days_per_week':days_per_week,'cur_week':CurWeek}
return {'errcode': 200, 'errmsg': 'ok', 'data': data_cache}
mod = 'all'
# temp_Time = temp_Time[0:-1]
zhou_pos = temp_Time.find('')
temp_Time = temp_Time[0:zhou_pos]
temp_Time = temp_Time.split(',')
index = 0
for n in temp_Time:
temp_Time[index] = n.split('-')
index += 1
index = 0
for n in temp_Time:
if len(n) > 1:
for o in range(int(n[0]), int(n[1]) + 1):
if (o % 2 == 0 and mod == 'double') or (o % 2 == 1 and mod == 'single') or (mod == 'all'):
days_per_week[o] = max(
days_per_week[o], i+1)
Time[o] = 1
else:
Time[o] = 0
else:
days_per_week[int(n[0])] = max(
days_per_week[int(n[0])], i+1)
Time[int(n[0])] = 1
index += 1
temp_lesson['Time_split'] = Time
lesson[i][j*2+k].append(temp_lesson)
self.__schedule_data = {'lesson': lesson,
'days_per_week': days_per_week, 'cur_week': CurWeek}
return ('ok', 200)
# 获取信息 -----------------------------------------------------------------------------
def getData(self):
# return (
# {
# 'student_id': self.__student_id,
# 'student_name': self.__student_name,
# 'grade': self.__grade_data,
# 'schedule': self.__schedule_data
# },
# 200
# )
return {
'errcode': '200',
'errmsg': 'ok',
'student_name': self.__student_name,
'student_id': self.__student_id,
'grade': self.__grade_data,
'schedule': self.__schedule_data
}

View File

@ -1 +1,203 @@
[2020-04-10 10:47:35 +0000] [9] [INFO] Starting gunicorn 20.0.4
[2020-04-10 10:47:35 +0000] [9] [INFO] Listening at: http://0.0.0.0:80 (9)
[2020-04-10 10:47:35 +0000] [9] [INFO] Using worker: sync
[2020-04-10 10:47:35 +0000] [11] [INFO] Booting worker with pid: 11
[2020-04-10 10:47:35 +0000] [12] [INFO] Booting worker with pid: 12
[2020-04-10 10:47:35 +0000] [13] [INFO] Booting worker with pid: 13
[2020-04-10 10:47:35 +0000] [14] [INFO] Booting worker with pid: 14
[2020-04-10 11:58:56 +0000] [10] [INFO] Starting gunicorn 20.0.4
[2020-04-10 11:58:56 +0000] [10] [INFO] Listening at: http://0.0.0.0:80 (10)
[2020-04-10 11:58:56 +0000] [10] [INFO] Using worker: sync
[2020-04-10 11:58:56 +0000] [12] [INFO] Booting worker with pid: 12
[2020-04-10 11:58:56 +0000] [13] [INFO] Booting worker with pid: 13
[2020-04-10 11:58:56 +0000] [14] [INFO] Booting worker with pid: 14
[2020-04-10 11:58:56 +0000] [15] [INFO] Booting worker with pid: 15
[2020-04-10 11:58:56 +0000] [12] [ERROR] Exception in worker process
Traceback (most recent call last):
File "/usr/local/lib/python3.7/site-packages/gunicorn/arbiter.py", line 583, in spawn_worker
worker.init_process()
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/base.py", line 119, in init_process
self.load_wsgi()
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/base.py", line 144, in load_wsgi
self.wsgi = self.app.wsgi()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/base.py", line 67, in wsgi
self.callable = self.load()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 49, in load
return self.load_wsgiapp()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 39, in load_wsgiapp
return util.import_app(self.app_uri)
File "/usr/local/lib/python3.7/site-packages/gunicorn/util.py", line 358, in import_app
mod = importlib.import_module(module)
File "/usr/local/lib/python3.7/importlib/__init__.py", line 127, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
File "<frozen importlib._bootstrap>", line 983, in _find_and_load
File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 728, in exec_module
File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
File "/app/cherry.py", line 5, in <module>
from allFunction import manageLogin, managePhoto, manageAdd, manageDel, manageGet, manageDetail, manageComment, manageScheduleUpload, manageScheduleGet
ImportError: cannot import name 'managePhoto' from 'allFunction' (./lib/allFunction.py)
[2020-04-10 11:58:56 +0000] [12] [INFO] Worker exiting (pid: 12)
[2020-04-10 11:58:56 +0000] [13] [ERROR] Exception in worker process
Traceback (most recent call last):
File "/usr/local/lib/python3.7/site-packages/gunicorn/arbiter.py", line 583, in spawn_worker
worker.init_process()
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/base.py", line 119, in init_process
self.load_wsgi()
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/base.py", line 144, in load_wsgi
self.wsgi = self.app.wsgi()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/base.py", line 67, in wsgi
self.callable = self.load()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 49, in load
return self.load_wsgiapp()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 39, in load_wsgiapp
return util.import_app(self.app_uri)
File "/usr/local/lib/python3.7/site-packages/gunicorn/util.py", line 358, in import_app
mod = importlib.import_module(module)
File "/usr/local/lib/python3.7/importlib/__init__.py", line 127, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
File "<frozen importlib._bootstrap>", line 983, in _find_and_load
File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 728, in exec_module
File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
File "/app/cherry.py", line 5, in <module>
from allFunction import manageLogin, managePhoto, manageAdd, manageDel, manageGet, manageDetail, manageComment, manageScheduleUpload, manageScheduleGet
ImportError: cannot import name 'managePhoto' from 'allFunction' (./lib/allFunction.py)
[2020-04-10 11:58:56 +0000] [13] [INFO] Worker exiting (pid: 13)
[2020-04-10 11:58:56 +0000] [14] [ERROR] Exception in worker process
Traceback (most recent call last):
File "/usr/local/lib/python3.7/site-packages/gunicorn/arbiter.py", line 583, in spawn_worker
worker.init_process()
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/base.py", line 119, in init_process
self.load_wsgi()
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/base.py", line 144, in load_wsgi
self.wsgi = self.app.wsgi()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/base.py", line 67, in wsgi
self.callable = self.load()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 49, in load
return self.load_wsgiapp()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 39, in load_wsgiapp
return util.import_app(self.app_uri)
File "/usr/local/lib/python3.7/site-packages/gunicorn/util.py", line 358, in import_app
mod = importlib.import_module(module)
File "/usr/local/lib/python3.7/importlib/__init__.py", line 127, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
File "<frozen importlib._bootstrap>", line 983, in _find_and_load
File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 728, in exec_module
File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
File "/app/cherry.py", line 5, in <module>
from allFunction import manageLogin, managePhoto, manageAdd, manageDel, manageGet, manageDetail, manageComment, manageScheduleUpload, manageScheduleGet
ImportError: cannot import name 'managePhoto' from 'allFunction' (./lib/allFunction.py)
[2020-04-10 11:58:56 +0000] [14] [INFO] Worker exiting (pid: 14)
[2020-04-10 11:58:56 +0000] [15] [ERROR] Exception in worker process
Traceback (most recent call last):
File "/usr/local/lib/python3.7/site-packages/gunicorn/arbiter.py", line 583, in spawn_worker
worker.init_process()
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/base.py", line 119, in init_process
self.load_wsgi()
File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/base.py", line 144, in load_wsgi
self.wsgi = self.app.wsgi()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/base.py", line 67, in wsgi
self.callable = self.load()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 49, in load
return self.load_wsgiapp()
File "/usr/local/lib/python3.7/site-packages/gunicorn/app/wsgiapp.py", line 39, in load_wsgiapp
return util.import_app(self.app_uri)
File "/usr/local/lib/python3.7/site-packages/gunicorn/util.py", line 358, in import_app
mod = importlib.import_module(module)
File "/usr/local/lib/python3.7/importlib/__init__.py", line 127, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
File "<frozen importlib._bootstrap>", line 983, in _find_and_load
File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 728, in exec_module
File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
File "/app/cherry.py", line 5, in <module>
from allFunction import manageLogin, managePhoto, manageAdd, manageDel, manageGet, manageDetail, manageComment, manageScheduleUpload, manageScheduleGet
ImportError: cannot import name 'managePhoto' from 'allFunction' (./lib/allFunction.py)
[2020-04-10 11:58:56 +0000] [15] [INFO] Worker exiting (pid: 15)
[2020-04-10 12:01:52 +0000] [9] [INFO] Starting gunicorn 20.0.4
[2020-04-10 12:01:52 +0000] [9] [INFO] Listening at: http://0.0.0.0:80 (9)
[2020-04-10 12:01:52 +0000] [9] [INFO] Using worker: sync
[2020-04-10 12:01:52 +0000] [11] [INFO] Booting worker with pid: 11
[2020-04-10 12:01:52 +0000] [12] [INFO] Booting worker with pid: 12
[2020-04-10 12:01:52 +0000] [13] [INFO] Booting worker with pid: 13
[2020-04-10 12:01:53 +0000] [14] [INFO] Booting worker with pid: 14
[2020-04-15 08:34:13,823] ERROR in app: Exception on /api/login [POST]
Traceback (most recent call last):
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 2446, in wsgi_app
response = self.full_dispatch_request()
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1951, in full_dispatch_request
rv = self.handle_user_exception(e)
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1820, in handle_user_exception
reraise(exc_type, exc_value, tb)
File "/usr/local/lib/python3.7/site-packages/flask/_compat.py", line 39, in reraise
raise value
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1949, in full_dispatch_request
rv = self.dispatch_request()
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1935, in dispatch_request
return self.view_functions[rule.endpoint](**req.view_args)
File "/app/cherry.py", line 24, in login
res = manageLogin(request)
File "./lib/allFunction.py", line 23, in manageLogin
schedule = getSchedule(res['ip'], res['s'])
File "./lib/crawler.py", line 265, in getSchedule
for o in range( int(n[0]) , int(n[1]) + 1):
ValueError: invalid literal for int() with base 10: '16周[3'
[2020-04-15 08:39:41 +0000] [9] [INFO] Starting gunicorn 20.0.4
[2020-04-15 08:39:41 +0000] [9] [INFO] Listening at: http://0.0.0.0:80 (9)
[2020-04-15 08:39:41 +0000] [9] [INFO] Using worker: sync
[2020-04-15 08:39:41 +0000] [11] [INFO] Booting worker with pid: 11
[2020-04-15 08:39:41 +0000] [12] [INFO] Booting worker with pid: 12
[2020-04-15 08:39:41 +0000] [13] [INFO] Booting worker with pid: 13
[2020-04-15 08:39:41 +0000] [14] [INFO] Booting worker with pid: 14
[2020-04-15 08:42:08,965] ERROR in app: Exception on /api/login [POST]
Traceback (most recent call last):
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 2446, in wsgi_app
response = self.full_dispatch_request()
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1951, in full_dispatch_request
rv = self.handle_user_exception(e)
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1820, in handle_user_exception
reraise(exc_type, exc_value, tb)
File "/usr/local/lib/python3.7/site-packages/flask/_compat.py", line 39, in reraise
raise value
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1949, in full_dispatch_request
rv = self.dispatch_request()
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1935, in dispatch_request
return self.view_functions[rule.endpoint](**req.view_args)
File "/app/cherry.py", line 24, in login
res = manageLogin(request)
File "./lib/allFunction.py", line 23, in manageLogin
schedule = getSchedule(res['ip'], res['s'])
File "./lib/crawler.py", line 265, in getSchedule
for o in range( int(n[0]) , int(n[1]) + 1):
ValueError: invalid literal for int() with base 10: '4周[3'
[2020-04-15 08:42:29,909] ERROR in app: Exception on /api/login [POST]
Traceback (most recent call last):
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 2446, in wsgi_app
response = self.full_dispatch_request()
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1951, in full_dispatch_request
rv = self.handle_user_exception(e)
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1820, in handle_user_exception
reraise(exc_type, exc_value, tb)
File "/usr/local/lib/python3.7/site-packages/flask/_compat.py", line 39, in reraise
raise value
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1949, in full_dispatch_request
rv = self.dispatch_request()
File "/usr/local/lib/python3.7/site-packages/flask/app.py", line 1935, in dispatch_request
return self.view_functions[rule.endpoint](**req.view_args)
File "/app/cherry.py", line 24, in login
res = manageLogin(request)
File "./lib/allFunction.py", line 23, in manageLogin
schedule = getSchedule(res['ip'], res['s'])
File "./lib/crawler.py", line 265, in getSchedule
for o in range( int(n[0]) , int(n[1]) + 1):
ValueError: invalid literal for int() with base 10: '4周[3'

0
test/__init__.py Normal file
View File

25
test/crawler_test.py Normal file
View File

@ -0,0 +1,25 @@
import unittest
from ..lib.crawler import Crawler
c = Crawler('2017002372', '623910ert&', '15143211127')
class TestCrawler(unittest.TestCase):
# 测试链接
def test_connection(self):
self.assertEqual(c.connection(), ('ok', 200))
# 测试获取成绩
def test_grade(self):
self.assertEqual(c.getGrade(), ('ok', 200))
# 测试获取课表
def test_schedule(self):
self.assertEqual(c.getSchedule(), ('ok', 200))
# 测试返回信息
def test_getData(self):
get_res = c.getData()
self.assertEqual(get_res['errcode'], '200')
if __name__ == '__main__':
unittest.main()