Code Duplication    Length = 71-95 lines in 2 locations

app.js 2 locations

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