Code Duplication    Length = 64-82 lines in 2 locations

app.js 2 locations

@@ 66-147 (lines=82) @@
63
var app = express();
64
65
// 查成绩API,通过GET传入用户名和密码
66
app.get('/grades', function (req, res, next) {
67
    if (!req.query.id || !req.query.pwd || (req.query.sem && !(/^20\d{2}-20\d{2}-[1-2]$/).test(req.query.sem))) {
68
        res.send({ error: "参数不正确" });
69
        return;
70
    }
71
    if (fullLog) {
72
        var start = new Date();
73
        console.log((timeStamp() + 'Started to query the grades: ').cyan + req.query.id.yellow);
74
    }
75
    access.login(req.query.id, req.query.pwd, res, function (headers, ires) {
76
        fullLog && console.log((timeStamp() + 'Successfully logged in.').green);
77
78
        var ret = {};
79
        var $ = cheerio.load(ires.text);
80
81
        ret.name = escaper.unescape($('.block1text').html()).match(/姓名:.+</)[0].substring(3).replace(/</, '');
82
        ret.id = req.query.id;
83
84
        // 实际上xnxq01id为空的时候和GET这个URL的效果是一样的,都是查询所有学期
85
        superagent
86
            .post('http://csujwc.its.csu.edu.cn/jsxsd/kscj/yscjcx_list')
87
            .set(headers)
88
            .type('form')
89
            .send({
90
                xnxq01id: req.query.sem
91
            })
92
            .end(function (err, iires) {
93
                if (err) {
94
                    console.log((timeStamp() + 'Failed to get grades page\n' + err.stack).red);
95
                    res.send({ error: '无法进入成绩页面' });
96
                    return next(err);
97
                }
98
                fullLog && console.log((timeStamp() + 'Successfully entered grades page.').green);
99
100
                $ = cheerio.load(iires.text);
101
                
102
                ret.grades = {};
103
                ret.failed = {};
104
105
                // 获取成绩列表,如果有补考记录,则以最高分的为准
106
                $('#dataList tr').each(function (index) {
107
                    if (index === 0) {
108
                        return;
109
                    }
110
                    let element = $(this).find('td');
111
                    let title = escaper.unescape(element.eq(3).text().match(/].+$/)[0].substring(1));
112
113
                    let item = {};
114
                    item.sem = escaper.unescape(element.eq(2).text());
115
                    item.regular = escaper.unescape(element.eq(4).text());
116
                    item.exam = escaper.unescape(element.eq(5).text());
117
                    item.overall = escaper.unescape(element.eq(6).text());
118
119
                    if (title in ret.grades) {
120
                        if (item.overall < ret.grades[title].overall) {
121
                            return;
122
                        }
123
                    }
124
125
                    ret.grades[title] = item;
126
127
                    // 暂不考虑NaN,(字符串 < 60)为false
128
                    if (parseInt(item.overall) < 60) {
129
                        ret.failed[title] = item;
130
                    } else {
131
                        delete ret.failed[title];
132
                    }
133
                });
134
135
                ret['subject-count'] = Object.keys(ret.grades).length;
136
                ret['failed-count'] = Object.keys(ret.failed).length;
137
138
                access.logout(headers, res, function() {
139
                    // 返回JSON
140
                    res.send(JSON.stringify(ret));
141
                    fullLog && console.log((timeStamp() + 'Successfully logged out: ').green + req.query.id.yellow + (' (processed in ' + (new Date() - start) + 'ms)').green);
142
                });
143
            });
144
145
146
    });
147
});
148
149
// 查考试API,通过GET传入用户名和密码
150
app.get('/exams', function (req, res, next) {
@@ 150-213 (lines=64) @@
147
});
148
149
// 查考试API,通过GET传入用户名和密码
150
app.get('/exams', function (req, res, next) {
151
    if (!req.query.id || !req.query.pwd || (req.query.sem && !(/^20\d{2}-20\d{2}-[1-2]$/).test(req.query.sem))) {
152
        res.send({ error: "参数不正确" });
153
        return;
154
    }
155
    if (fullLog) {
156
        var start = new Date();
157
        console.log((timeStamp() + 'Started to query the exams: ').cyan + req.query.id.yellow);
158
    }
159
    access.login(req.query.id, req.query.pwd, res, function (headers, ires) {
160
        fullLog && console.log((timeStamp() + 'Successfully logged in.').green);
161
162
        var ret = {};
163
        var $ = cheerio.load(ires.text);
164
165
        ret.name = escaper.unescape($('.block1text').html()).match(/姓名:.+</)[0].replace('<', '').substring(3);
166
        ret.id = req.query.id;
167
        ret.sem = req.query.sem || getSem();
168
169
        superagent
170
            .post('http://csujwc.its.csu.edu.cn/jsxsd/xsks/xsksap_list')
171
            .set(headers)
172
            .type('form')
173
            .send({
174
                xqlbmc: '',
175
                xnxqid: ret.sem,
176
                xqlb: ''
177
            })
178
            .end(function (err, iires) {
179
                if (err) {
180
                    console.log((timeStamp() + 'Failed to reach exams page\n' + err.stack).red);
181
                    res.send({ error: '获取成绩失败' });
182
                    return next(err);
183
                }
184
                fullLog && console.log((timeStamp() + 'Successfully entered exams page.').green);
185
186
                $ = cheerio.load(iires.text);
187
188
                ret.exams = {};
189
                ret['exams-count'] = 0;
190
191
                $('#dataList tr').each(function (index) {
192
                    if (index === 0) {
193
                        return;
194
                    }
195
                    let element = $(this).find('td');
196
                    let title = escaper.unescape(element.eq(3).text());
197
198
                    let item = {};
199
                    item.time = escaper.unescape(element.eq(4).text());
200
                    item.location = escaper.unescape(element.eq(5).text());
201
                    item.seat = escaper.unescape(element.eq(6).text());
202
203
                    ret.exams[title] = item;
204
                    ret['exams-count']++;
205
                });
206
207
                access.logout(headers, res, function() {
208
                    res.send(JSON.stringify(ret));
209
                    fullLog && console.log((timeStamp() + 'Successfully logged out: ').green + req.query.id.yellow + (' (processed in ' + (new Date() - start) + 'ms)').green);
210
                });
211
            });
212
    });
213
});
214
215
app.listen(port);
216
console.log((timeStamp() + 'The server is now running on port ' + port + '.').green);