加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
该仓库未声明开源许可证文件(LICENSE),使用请关注具体项目描述及其代码上游依赖。
克隆/下载
server.js 24.43 KB
一键复制 编辑 原始数据 按行查看 历史
aouyyy 提交于 2021-05-10 20:20 . 京东扫码
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803
/*
* @Author: Jerrykuku https://github.com/jerrykuku
* @Date: 2021-1-8
* @Version: v0.0.2
* @thanks: FanchangWang https://github.com/FanchangWang
*/
var express = require('express');
var session = require('express-session');
var bodyParser = require('body-parser');
var got = require('got');
var path = require('path');
var fs = require('fs');
var { execSync, exec } = require('child_process');
var rootPath = path.resolve(__dirname, '..')
// config.sh 文件所在目录
var confFile = path.join(rootPath, 'config/config.sh');
// config.sh.sample 文件所在目录
var sampleFile = path.join(rootPath, 'sample/config.sh.sample');
// crontab.list 文件所在目录
var crontabFile = path.join(rootPath, 'config/crontab.list');
// config.sh 文件备份目录
var confBakDir = path.join(rootPath, 'config/bak/');
// auth.json 文件目录
var authConfigFile = path.join(rootPath, 'config/auth.json');
// Share Code 文件目录
var shareCodeDir = path.join(rootPath, 'log/jd_get_share_code/');
// diy.sh 文件目录
var diyFile = path.join(rootPath, 'config/diy.sh');
// 日志目录
var logPath = path.join(rootPath, 'log/');
// 脚本目录
var ScriptsPath = path.join(rootPath, 'scripts/');
var authError = "错误的用户名密码,请重试";
var loginFaild = "请先登录!";
var configString = "config sample crontab shareCode diy";
var s_token, cookies, guid, lsid, lstoken, okl_token, token, userCookie = ""
function praseSetCookies(response) {
s_token = response.body.s_token
guid = response.headers['set-cookie'][0]
guid = guid.substring(guid.indexOf("=") + 1, guid.indexOf(";"))
lsid = response.headers['set-cookie'][2]
lsid = lsid.substring(lsid.indexOf("=") + 1, lsid.indexOf(";"))
lstoken = response.headers['set-cookie'][3]
lstoken = lstoken.substring(lstoken.indexOf("=") + 1, lstoken.indexOf(";"))
cookies = "guid=" + guid + "; lang=chs; lsid=" + lsid + "; lstoken=" + lstoken + "; "
}
function getCookie(response) {
var TrackerID = response.headers['set-cookie'][0]
TrackerID = TrackerID.substring(TrackerID.indexOf("=") + 1, TrackerID.indexOf(";"))
var pt_key = response.headers['set-cookie'][1]
pt_key = pt_key.substring(pt_key.indexOf("=") + 1, pt_key.indexOf(";"))
var pt_pin = response.headers['set-cookie'][2]
pt_pin = pt_pin.substring(pt_pin.indexOf("=") + 1, pt_pin.indexOf(";"))
var pt_token = response.headers['set-cookie'][3]
pt_token = pt_token.substring(pt_token.indexOf("=") + 1, pt_token.indexOf(";"))
var pwdt_id = response.headers['set-cookie'][4]
pwdt_id = pwdt_id.substring(pwdt_id.indexOf("=") + 1, pwdt_id.indexOf(";"))
var s_key = response.headers['set-cookie'][5]
s_key = s_key.substring(s_key.indexOf("=") + 1, s_key.indexOf(";"))
var s_pin = response.headers['set-cookie'][6]
s_pin = s_pin.substring(s_pin.indexOf("=") + 1, s_pin.indexOf(";"))
cookies = "TrackerID=" + TrackerID + "; pt_key=" + pt_key + "; pt_pin=" + pt_pin + "; pt_token=" + pt_token + "; pwdt_id=" + pwdt_id + "; s_key=" + s_key + "; s_pin=" + s_pin + "; wq_skey="
var userCookie = "pt_key=" + pt_key + ";pt_pin=" + pt_pin + ";";
console.log("\n############ 登录成功,获取到 Cookie #############\n\n");
console.log('Cookie1="' + userCookie + '"\n');
console.log("\n####################################################\n\n");
return userCookie;
}
async function step1() {
try {
s_token, cookies, guid, lsid, lstoken, okl_token, token = ""
let timeStamp = (new Date()).getTime()
let url = 'https://plogin.m.jd.com/cgi-bin/mm/new_login_entrance?lang=chs&appid=300&returnurl=https://wq.jd.com/passport/LoginRedirect?state=' + timeStamp + '&returnurl=https://home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action&source=wq_passport'
const response = await got(url, {
responseType: 'json',
headers: {
'Connection': 'Keep-Alive',
'Content-Type': 'application/x-www-form-urlencoded',
'Accept': 'application/json, text/plain, */*',
'Accept-Language': 'zh-cn',
'Referer': 'https://plogin.m.jd.com/login/login?appid=300&returnurl=https://wq.jd.com/passport/LoginRedirect?state=' + timeStamp + '&returnurl=https://home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action&source=wq_passport',
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/86.0.4240.111 Safari/537.36',
'Host': 'plogin.m.jd.com'
}
});
praseSetCookies(response)
} catch (error) {
cookies = "";
console.log(error.response.body);
}
};
async function step2() {
try {
if (cookies == "") {
return 0
}
let timeStamp = (new Date()).getTime()
let url = 'https://plogin.m.jd.com/cgi-bin/m/tmauthreflogurl?s_token=' + s_token + '&v=' + timeStamp + '&remember=true'
const response = await got.post(url, {
responseType: 'json',
json: {
'lang': 'chs',
'appid': 300,
'returnurl': 'https://wqlogin2.jd.com/passport/LoginRedirect?state=' + timeStamp + '&returnurl=//home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action',
'source': 'wq_passport'
},
headers: {
'Connection': 'Keep-Alive',
'Content-Type': 'application/x-www-form-urlencoded; Charset=UTF-8',
'Accept': 'application/json, text/plain, */*',
'Cookie': cookies,
'Referer': 'https://plogin.m.jd.com/login/login?appid=300&returnurl=https://wqlogin2.jd.com/passport/LoginRedirect?state=' + timeStamp + '&returnurl=//home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action&source=wq_passport',
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/86.0.4240.111 Safari/537.36',
'Host': 'plogin.m.jd.com',
}
});
token = response.body.token
okl_token = response.headers['set-cookie'][0]
okl_token = okl_token.substring(okl_token.indexOf("=") + 1, okl_token.indexOf(";"))
var qrUrl = 'https://plogin.m.jd.com/cgi-bin/m/tmauth?appid=300&client_type=m&token=' + token;
return qrUrl;
} catch (error) {
console.log(error.response.body);
return 0
}
}
var i = 0;
async function checkLogin() {
try {
if (cookies == "") {
return 0
}
let timeStamp = (new Date()).getTime()
let url = 'https://plogin.m.jd.com/cgi-bin/m/tmauthchecktoken?&token=' + token + '&ou_state=0&okl_token=' + okl_token;
const response = await got.post(url, {
responseType: 'json',
form: {
lang: 'chs',
appid: 300,
returnurl: 'https://wqlogin2.jd.com/passport/LoginRedirect?state=1100399130787&returnurl=//home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action',
source: 'wq_passport'
},
headers: {
'Referer': 'https://plogin.m.jd.com/login/login?appid=300&returnurl=https://wqlogin2.jd.com/passport/LoginRedirect?state=' + timeStamp + '&returnurl=//home.m.jd.com/myJd/newhome.action?sceneval=2&ufc=&/myJd/home.action&source=wq_passport',
'Cookie': cookies,
'Connection': 'Keep-Alive',
'Content-Type': 'application/x-www-form-urlencoded; Charset=UTF-8',
'Accept': 'application/json, text/plain, */*',
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/86.0.4240.111 Safari/537.36',
}
});
return response;
} catch (error) {
console.log(error.response.body);
let res = {}
res.body = { check_ip: 0, errcode: 222, message: '出错' }
res.headers = {}
return res;
}
}
/**
* 检查 config.sh 以及 config.sh.sample 文件是否存在
*/
function checkConfigFile() {
if (!fs.existsSync(confFile)) {
console.error('脚本启动失败,config.sh 文件不存在!');
process.exit(1);
}
if (!fs.existsSync(sampleFile)) {
console.error('脚本启动失败,config.sh.sample 文件不存在!');
process.exit(1);
}
}
/**
* 检查 config/bak/ 备份目录是否存在,不存在则创建
*/
function mkdirConfigBakDir() {
if (!fs.existsSync(confBakDir)) {
fs.mkdirSync(confBakDir);
}
}
/**
* 备份 config.sh 文件
*/
function bakConfFile(file) {
mkdirConfigBakDir();
let date = new Date();
let bakConfFile = confBakDir + file + '_' + date.getFullYear() + '-' + date.getMonth() + '-' + date.getDay() + '-' + date.getHours() + '-' + date.getMinutes() + '-' + date.getMilliseconds();
let oldConfContent = "";
switch (file) {
case "config.sh":
oldConfContent = getFileContentByName(confFile);
fs.writeFileSync(bakConfFile, oldConfContent);
break;
case "crontab.list":
oldConfContent = getFileContentByName(crontabFile);
fs.writeFileSync(bakConfFile, oldConfContent);
break;
case "diy.sh":
oldConfContent = getFileContentByName(diyFile);
fs.writeFileSync(bakConfFile, oldConfContent);
break;
default:
break;
}
}
/**
* 将 post 提交内容写入 config.sh 文件(同时备份旧的 config.sh 文件到 bak 目录)
* @param content
*/
function saveNewConf(file, content) {
bakConfFile(file);
switch (file) {
case "config.sh":
fs.writeFileSync(confFile, content);
break;
case "crontab.list":
fs.writeFileSync(crontabFile, content);
execSync('crontab ' + crontabFile);
break;
case "diy.sh":
fs.writeFileSync(diyFile, content);
break;
default:
break;
}
}
/**
* 获取文件内容
* @param fileName 文件路径
* @returns {string}
*/
function getFileContentByName(fileName) {
if (fs.existsSync(fileName)) {
return fs.readFileSync(fileName, 'utf8');
}
return '';
}
/**
* 获取目录中最后修改的文件的路径
* @param dir 目录路径
* @returns {string} 最新文件路径
*/
function getLastModifyFilePath(dir) {
var filePath = '';
if (fs.existsSync(dir)) {
var lastmtime = 0;
var arr = fs.readdirSync(dir);
arr.forEach(function (item) {
var fullpath = path.join(dir, item);
var stats = fs.statSync(fullpath);
if (stats.isFile()) {
if (stats.mtimeMs >= lastmtime) {
filePath = fullpath;
}
}
});
}
return filePath;
}
var app = express();
app.use(session({
secret: 'secret',
name: `connect.${Math.random()}`,
resave: true,
saveUninitialized: true
}));
app.use(bodyParser.json({ limit: '50mb' }));
app.use(bodyParser.urlencoded({ limit: '50mb', extended: true }));
app.use(express.static(path.join(__dirname, 'public')));
/**
* 登录页面
*/
app.get('/', function (request, response) {
if (request.session.loggedin) {
response.redirect('./home');
} else {
response.sendFile(path.join(__dirname + '/public/auth.html'));
}
});
/**
* 用户名密码
*/
app.get('/changepwd', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/pwd.html'));
} else {
response.redirect('/');
}
});
/**
* 获取二维码链接
*/
app.get('/qrcode', function (request, response) {
if (request.session.loggedin) {
(async () => {
try {
await step1();
const qrurl = await step2();
if (qrurl != 0) {
response.send({ err: 0, qrcode: qrurl });
} else {
response.send({ err: 1, msg: "错误" });
}
} catch (err) {
response.send({ err: 1, msg: err });
}
})();
} else {
response.send({ err: 1, msg: loginFaild });
}
})
/**
* 获取返回的cookie信息
*/
app.get('/cookie', function (request, response) {
if (request.session.loggedin && cookies != "") {
(async () => {
try {
const cookie = await checkLogin();
if (cookie.body.errcode == 0) {
let ucookie = getCookie(cookie);
response.send({ err: 0, cookie: ucookie });
} else {
response.send({ err: cookie.body.errcode, msg: cookie.body.message });
}
} catch (err) {
response.send({ err: 1, msg: err });
}
})();
} else {
response.send({ err: 1, msg: loginFaild });
}
})
/**
* 获取各种配置文件api
*/
app.get('/api/config/:key', function (request, response) {
if (request.session.loggedin) {
if (configString.indexOf(request.params.key) > -1) {
switch (request.params.key) {
case 'config':
content = getFileContentByName(confFile);
break;
case 'sample':
content = getFileContentByName(sampleFile);
break;
case 'crontab':
content = getFileContentByName(crontabFile);
break;
case 'shareCode':
let shareCodeFile = getLastModifyFilePath(shareCodeDir);
content = getFileContentByName(shareCodeFile);
break;
case 'diy':
content = getFileContentByName(diyFile);
break;
default:
break;
}
response.setHeader("Content-Type", "text/plain");
response.send(content);
} else {
response.send("no config");
}
} else {
response.send(loginFaild);
}
})
/**
* 首页 配置页面
*/
app.get('/home', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/home.html'));
} else {
response.redirect('/');
}
});
/**
* 对比 配置页面
*/
app.get('/diff', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/diff.html'));
} else {
response.redirect('/');
}
});
/**
* Share Code 页面
*/
app.get('/shareCode', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/shareCode.html'));
} else {
response.redirect('/');
}
});
/**
* crontab 配置页面
*/
app.get('/crontab', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/crontab.html'));
} else {
response.redirect('/');
}
});
/**
* 自定义脚本 页面
*/
app.get('/diy', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/diy.html'));
} else {
response.redirect('/');
}
});
/**
* 手动执行脚本 页面
*/
app.get('/run', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/run.html'));
} else {
response.redirect('/');
}
});
app.post('/runCmd', function(request, response) {
if (request.session.loggedin) {
const cmd = request.body.cmd;
const delay = request.body.delay || 0;
// console.log('before exec');
exec(cmd, (error, stdout, stderr) => {
// console.log(error, stdout, stderr);
// 根据传入延时返回数据,有时太快会出问题
setTimeout(() => {
if (error) {
console.error(`执行的错误: ${error}`);
response.send({ err: 1, msg: '执行出错!' });
} else if (stdout) {
// console.log(`stdout: ${stdout}`)
response.send({ err: 0, msg: `${stdout}` });
} else if (stderr) {
console.error(`stderr: ${stderr}`);
response.send({ err: 1, msg: `${stderr}` });
}
}, delay);
});
} else {
response.redirect('/');
}
});
/**
* 使用jsName获取最新的日志
*/
app.get('/runLog/:jsName', function (request, response) {
if (request.session.loggedin) {
let shareCodeFile = getLastModifyFilePath(path.join(rootPath, `log/${request.params.jsName}/`));
if (shareCodeFile) {
content = getFileContentByName(shareCodeFile);
response.setHeader("Content-Type", "text/plain");
response.send(content);
} else {
response.send("no logs");
}
} else {
response.send(loginFaild);
}
})
/**
* auth
*/
app.post('/auth', function (request, response) {
let username = request.body.username;
let password = request.body.password;
fs.readFile(authConfigFile, 'utf8', function (err, data) {
if (err) console.log(err);
var con = JSON.parse(data);
if (username && password) {
if (username == con.user && password == con.password) {
request.session.loggedin = true;
request.session.username = username;
response.send({ err: 0 });
} else {
response.send({ err: 1, msg: authError });
}
} else {
response.send({ err: 1, msg: "请输入用户名密码!" });
}
});
});
/**
* change pwd
*/
app.post('/changepass', function (request, response) {
if (request.session.loggedin) {
let username = request.body.username;
let password = request.body.password;
let config = {
user: username,
password: password
}
if (username && password) {
fs.writeFile(authConfigFile, JSON.stringify(config), function (err) {
if (err) {
response.send({ err: 1, msg: "写入错误请重试!" });
} else {
response.send({ err: 0, msg: "更新成功!" });
}
});
} else {
response.send({ err: 1, msg: "请输入用户名密码!" });
}
} else {
response.send(loginFaild);
}
});
/**
* change pwd
*/
app.get('/logout', function (request, response) {
request.session.destroy()
response.redirect('/');
});
/**
* save config
*/
app.post('/api/save', function (request, response) {
if (request.session.loggedin) {
let postContent = request.body.content;
let postfile = request.body.name;
saveNewConf(postfile, postContent);
response.send({ err: 0, title: "保存成功! ", msg: "将自动刷新页面查看修改后的 " + postfile + " 文件" });
} else {
response.send({ err: 1, title: "保存失败! ", msg: loginFaild });
}
});
/**
* 日志查询 页面
*/
app.get('/log', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/tasklog.html'));
} else {
response.redirect('/');
}
});
/**
* 日志列表
*/
app.get('/api/logs', function (request, response) {
if (request.session.loggedin) {
var fileList = fs.readdirSync(logPath, 'utf-8');
var dirs = [];
var rootFiles = [];
for (var i = 0; i < fileList.length; i++) {
var stat = fs.lstatSync(logPath + fileList[i]);
// 是目录,需要继续
if (stat.isDirectory()) {
var fileListTmp = fs.readdirSync(logPath + '/' + fileList[i], 'utf-8');
fileListTmp.reverse();
var dirMap = {
dirName: fileList[i],
files: fileListTmp
}
dirs.push(dirMap);
} else {
rootFiles.push(fileList[i]);
}
}
dirs.push({
dirName: '@',
files: rootFiles
});
var result = { dirs };
response.send(result);
} else {
response.redirect('/');
}
});
/**
* 日志文件
*/
app.get('/api/logs/:dir/:file', function (request, response) {
if (request.session.loggedin) {
let filePath;
if (request.params.dir === '@') {
filePath = logPath + request.params.file;
} else {
filePath = logPath + request.params.dir + '/' + request.params.file;
}
var content = getFileContentByName(filePath);
response.setHeader("Content-Type", "text/plain");
response.send(content);
} else {
response.redirect('/');
}
});
/**
* 查看脚本 页面
*/
app.get('/viewScripts', function (request, response) {
if (request.session.loggedin) {
response.sendFile(path.join(__dirname + '/public/viewScripts.html'));
} else {
response.redirect('/');
}
});
/**
* 脚本列表
*/
app.get('/api/scripts', function (request, response) {
if (request.session.loggedin) {
var fileList = fs.readdirSync(ScriptsPath, 'utf-8');
var dirs = [];
var rootFiles = [];
var excludeRegExp = /(git)|(node_modules)|(icon)/;
for (var i = 0; i < fileList.length; i++) {
var stat = fs.lstatSync(ScriptsPath + fileList[i]);
// 是目录,需要继续
if (stat.isDirectory()) {
var fileListTmp = fs.readdirSync(ScriptsPath + '/' + fileList[i], 'utf-8');
fileListTmp.reverse();
if (excludeRegExp.test(fileList[i])) {
continue;
}
var dirMap = {
dirName: fileList[i],
files: fileListTmp
}
dirs.push(dirMap);
} else {
if (excludeRegExp.test(fileList[i])) {
continue;
}
rootFiles.push(fileList[i]);
}
}
dirs.push({
dirName: '@',
files: rootFiles
});
var result = { dirs };
response.send(result);
} else {
response.redirect('/');
}
});
/**
* 脚本文件
*/
app.get('/api/scripts/:dir/:file', function (request, response) {
if (request.session.loggedin) {
let filePath;
if (request.params.dir === '@') {
filePath = ScriptsPath + request.params.file;
} else {
filePath = ScriptsPath + request.params.dir + '/' + request.params.file;
}
var content = getFileContentByName(filePath);
response.setHeader("Content-Type", "text/plain");
response.send(content);
} else {
response.redirect('/');
}
});
/**
* 更新已经存在的人的cookie & 自动添加新用户
* */
app.post('/updateCookie', function (request, response) {
if (request.body.cookie) {
const cookie = request.body.cookie;
const content = getFileContentByName(confFile);
const lines = content.split('\n');
const pt_pin = cookie.match(/pt_pin=.+?;/)[0];
let updateFlag = false;
let lastIndex = 0;
let maxCookieCount = 0;
for (var i = 0; i < lines.length; i++) {
var line = lines[i];
if (line.startsWith('Cookie')) {
maxCookieCount = line.split('=')[0].split('Cookie')[1];
lastIndex = i;
if (
line.match(/pt_pin=.+?;/) &&
line.match(/pt_pin=.+?;/)[0] == pt_pin
) {
const head = line.split('=')[0];
const newLine = [head, '=', '"', cookie, '"'].join('');
lines[i] = newLine;
updateFlag = true;
}
}
}
if (!updateFlag) {
const newLine = [
'Cookie',
Number(maxCookieCount) + 1,
'=',
'"',
cookie,
'"',
].join('');
lines.splice(lastIndex + 1, 0, newLine);
}
saveNewConf('config.sh', lines.join('\n'));
response.send({
err: 0,
msg: updateFlag ? '更新成功' : '没有此CK,已自动添加',
});
} else {
response.send({ msg: '参数错误', err: -1 });
}
});
checkConfigFile()
app.listen(5678, () => {
console.log('应用正在监听 5678 端口!');
});
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化