index.html 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446
  1. <!DOCTYPE html>
  2. <head>
  3. <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
  4. <title>moe</title>
  5. <script type="text/javascript" src="/js/jquery.min.js"></script>
  6. <link rel="stylesheet" href="/css/bootstrap.min.css">
  7. <script type="text/javascript" src="/js/bootstrap.min.js"></script>
  8. <link rel="preconnect" href="https://fonts.googleapis.com">
  9. <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
  10. <link href="/css/css2.css" rel="stylesheet">
  11. <link rel="stylesheet" href="/css/style.css">
  12. <script>
  13. var status = 0, oid, user, pwd, paused = []
  14. const rot = '43.143.233.184:81', root = 'https://43.143.233.184:81/'
  15. function getCookie(name) {
  16. let arr = document.cookie.match(new RegExp("(^| )" + name + "=([^;]*)(;|$)"));
  17. if (arr != null) return arr[2];
  18. return null;
  19. }
  20. function delCookie(name, path = 'u') {
  21. var exp = new Date();
  22. exp.setTime(exp.getTime() - 1);
  23. var cval = getCookie(name);
  24. if (cval != null)
  25. document.cookie = name + "=" + cval + ";expires=" + exp.toGMTString() + ((path == 'u') ? '' : ';path=' + path)
  26. }
  27. function getUrl(e) {
  28. return e.replace('dd.sdsz.com.cn', rot).replace(/^http:/, 'https:').replace(rot.split(':')[0] + '/', rot + '/').replace('service=http%3A%2F%2F' + root, 'service=http%3A%2F%2Fdd.sdsz.com.cn')
  29. }
  30. function setStatus(e, fast) {
  31. console.log('STATUS', e)
  32. status = e
  33. if (e == 1) {
  34. paused = []
  35. $('#loginSt').text('登录成功')
  36. $('.login').hide(200)
  37. $('#loginSt').addClass('label-success')
  38. $('#loginSt').removeClass('label-danger')
  39. $('.reqLogin').show(300)
  40. } else if (e == 2) {
  41. $('#loginSt').text('登录失败')
  42. $('#loginSt').removeClass('label-success')
  43. $('#loginSt').addClass('label-danger')
  44. } else {
  45. $('.login').show(200)
  46. $('#loginSt').text('未登录')
  47. $('#loginSt').removeClass('label-success')
  48. $('#loginSt').removeClass('label-danger')
  49. $('.reqLogin').hide(300 * !fast)
  50. $('#user').text('-')
  51. }
  52. }
  53. function delCookies(force = 0) {
  54. if (force) {
  55. delCookie('CASTGC', '/')
  56. delCookie('moe', '/')
  57. delCookie('gosh', '/score/')
  58. if (force <= 2) {
  59. delCookie('gosh', '/sso/')
  60. delCookie('gowd', '/sso/')
  61. }
  62. }
  63. if (force >= 2) {
  64. delCookie('JSESSIONID', '/bxn-portal/')
  65. delCookie('JSESSIONID', '/bxn-library/')
  66. delCookie('JSESSIONID', '/bxn-core-uic/')
  67. }
  68. delCookie('JSESSIONID', '/')
  69. delCookie('SSOID', '/')
  70. delCookie('JSESSIONID', '/sso/')
  71. }
  72. function gget(url, call) {
  73. fetch(url, {
  74. "headers": {
  75. "content-type": "application/x-www-form-urlencoded; charset=UTF-8",
  76. },
  77. "method": "GET",
  78. "credentials": 'include'
  79. }).then(e => e.text()).then(call)
  80. }
  81. function getStatus(ds, dt) {
  82. let s = new Date(ds), t = new Date(dt), e = new Date();
  83. console.log(s, t, e)
  84. if (e < s) return '未开始'
  85. if (e < t) return '进行中'
  86. else return '已结束'
  87. }
  88. function adminStuff() {
  89. $('.admin').show()
  90. // $('.box-contest').append('<input type="date" id="start" min="2022-01-01">')
  91. }
  92. function contestTouch() {
  93. fetch('contest/touch', {
  94. headers: { 'Content-type': 'application/json' },
  95. method: "POST",
  96. credentials: 'include',
  97. body: JSON.stringify({
  98. no: $('#tNo').val(),
  99. title: $('#tTitle').val(),
  100. description: $('#tDescription').val(),
  101. dates: new Date($('#tStart')[0].valueAsNumber).toUTCString(),
  102. datet: new Date($('#tEnd')[0].valueAsNumber).toUTCString()
  103. }),
  104. }).then(e => e.json()).then(e => {
  105. if (e.no) alert('您无权进行此操作')
  106. else {
  107. $('.contestNew').toggle(300)
  108. debug('TOUCH', 'affected ' + e.affectedRows + ' rows.')
  109. postLogin();
  110. }
  111. })
  112. }
  113. function contestRm(id, name) {
  114. console.log(name)
  115. if (prompt('确认删除?输入“' + name + '”以确认。') != name) return
  116. fetch('contest/rm', {
  117. headers: { 'Content-type': 'application/json' },
  118. method: "POST",
  119. credentials: 'include',
  120. body: `{"id":"${id}"}`
  121. }).then(e => e.json()).then(e => {
  122. console.log(e)
  123. if (e.no) alert('您无权进行此操作')
  124. else {
  125. debug('RM', 'affected ' + e.affectedRows + ' rows.')
  126. postLogin();
  127. }
  128. })
  129. }
  130. function postLogin() {
  131. debug('LOGIN DONE.', 'initiate fetching.')
  132. if (status != 1) return;
  133. if (!user) user = getCookie('gosh'), pwd = getCookie('gowd')
  134. document.cookie = "gosh=" + user + ";path=/score/;expires=Fri, 05 Feb 2077 12:34:56 GMT";
  135. document.cookie = "gosh=" + user + ";path=/sso/;expires=Fri, 05 Feb 2077 12:34:56 GMT";
  136. document.cookie = "gowd=" + pwd + ";path=/sso/;expires=Fri, 05 Feb 2077 12:34:56 GMT";
  137. if (getCookie('moe')[0] == '!') $('#user').text('admin')
  138. else get('bxn-portal/portal/api/proxy?username=true&userId=true&needvalidate=true&t=http%3A%2F%2F202.94.10.118%2Fbxn-library%2Fapi%2Flibrary%2FgetLatestBorrowBook%3Fcategory%3D1&resourceId=103519&cache=nocache&systime=0', e => {
  139. e = JSON.parse(e)
  140. $('#user').text(e.requestParams[0].ownerName)
  141. oid = e.requestParams[0].ownerId
  142. })
  143. gget('contest', e => {
  144. e = JSON.parse(e)
  145. $('#contest').empty()
  146. for (let i = 0; i < e.length; i++) {
  147. $('#contest').prepend(`\
  148. <li id="contest-${e[i].id}">\
  149. <span onclick="$(this).toggleClass('active').siblings('div').toggle(300)" class="title">第 ${e[i].no} 届${e[i].title}\
  150. <span class="pull-right" style="color:#999"><span class="admin" style="display:none"><a href="javascript:contestRm(${e[i].id},'第${e[i].no}届${e[i].title}')">删除</a> ${e[i].id} - </span>${getStatus(e[i].dates, e[i].datet)}</span>\
  151. </span>
  152. <div style="display:none">${e[i].description}</div>\
  153. </li>`)
  154. }
  155. if (getCookie('moe')[0] == '!') adminStuff()
  156. })
  157. }
  158. function debug(e, str = '') {
  159. $('.debugList').append(`<li class="auto-warp"><b>${e}</b> ${str} <span class="pull-right" style="color:#999">${status}</span></li>`)
  160. }
  161. function login() {
  162. debug('LOGIN.')
  163. if (status == 1) return;
  164. delCookies()
  165. user = $('#loginId').val(), pwd = $('#loginPwd').val()
  166. return fetch(root + "sso/login", {
  167. headers: {
  168. "content-type": "application/x-www-form-urlencoded; charset=UTF-8",
  169. // "user-agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36"
  170. },
  171. method: "GET",
  172. mode: "cors",
  173. credentials: 'include'
  174. }).then(() => {
  175. return fetch('https://43.143.233.184/moelogin', {
  176. method: 'POST',
  177. headers: { 'Content-type': 'application/json' },
  178. body: JSON.stringify({
  179. user: user,
  180. pwd: pwd,
  181. cook: getCookie('JSESSIONID')
  182. }),
  183. credentials: 'include',
  184. })
  185. }).then(e => e.text()).then(e => { setStatus((e != 'success') + 1), Promise.resolve() })
  186. }
  187. function matches(e) {
  188. return e.match(/(^\[\]$|param":"|<tr>|DOCTYPE|in\?username=|requestParams|\"newmessage\"|Sorry, Page Not Found|北师大实验中学--登录)/s)
  189. }
  190. // 异步递归嘿嘿嘿,再多 302 也不怕;再加上回调函数,真是妙
  191. async function next(e, call, ac, login) {
  192. if (e.length < 3000 && !matches(e) && (!e.match('\/login\\?') || login)) {
  193. return fetch(getUrl(e), {
  194. method: 'GET',
  195. credentials: 'include',
  196. headers: {
  197. Accept: ac
  198. }
  199. }).then(e => e.text()).then(res => { next(res, call, ac, login) })
  200. }
  201. else return call(e), new Promise((resolve, reject) => { })
  202. }
  203. function pre(e, url, call) {
  204. if (e.match('北师大实验中学--登录')) {
  205. debug('OUTDATED CAS! NEED RELOGIN')
  206. delCookie('CASTGC', '/')
  207. tryLogin()
  208. }
  209. if (e.length < 3000 && e.match('\/login\\?')) {
  210. var name = url.split('/')[0], f = paused.filter(e => e.name == name)
  211. debug('FAILED', url)
  212. if (url.match('\/login\\?')) return
  213. if (f.length) f[0].arr.push(url), f[0].carr.push(call)
  214. else {
  215. paused.push({ name: name, arr: [url], carr: [call] })
  216. debug('GET COOKIE...')
  217. get(name + '/casfailed.jsp', () => {
  218. let f = paused.filter(e => e.name == name)[0]
  219. for (let i = 0; i < f.arr.length; i++)get(f.arr[i], f.carr[i])
  220. paused.splice(paused.indexOf(f), 1)
  221. }, '*/*', 1)
  222. }
  223. } else {
  224. $('#bxn' + url.split('bxn')[1].split(/(\/|%2F)/)[0]).addClass('bg-success').addClass('text-success')
  225. if (status == 1) call(e)
  226. }
  227. }
  228. function get(url, call = () => { }, ac = '*/*', login = 0) {
  229. if (getCookie('moe')[0] == '!') return
  230. console.log("GET", url)
  231. debug('GET', url)
  232. if (status != 1) return
  233. next(root + url, e => { pre(e, url, call); }, ac, login)
  234. }
  235. let trying = 0
  236. function tryLogin() {
  237. if (trying) return debug('LOGIN', 'already in progress.')
  238. if (getCookie('moe') && getCookie('moe')[0] == '!') {
  239. debug('LOGIN', 'admin!')
  240. setStatus(1), postLogin()
  241. } else if (getCookie('CASTGC')) {
  242. debug('LOGIN', 'using CAS.')
  243. setStatus(1), postLogin()
  244. } else if (getCookie('gosh') && getCookie('gosh') != 'null' && getCookie('gowd') != 'null') {
  245. debug('LOGIN', 'using stored cookie.')
  246. setStatus(0)
  247. trying = 1
  248. $('#loginId').val(getCookie('gosh'))
  249. $('#loginPwd').val(getCookie('gowd'))
  250. login().then(() => { status == 2 ? setStatus(0, 1) : (trying = 0, postLogin()); })
  251. } else debug('LOGIN', 'no available method.'), setStatus(0, 1)
  252. }
  253. $().ready(() => {
  254. $('.e').addClass('reqLogin')
  255. $('.e').removeClass('e')
  256. tryLogin()
  257. $("[data-toggle='tooltip']").tooltip();
  258. $('.banner').text('庆祝 moe 建站 ' + Math.ceil((new Date().getTime() - 1676000714191) / 86400000) + ' 天')
  259. })
  260. </script>
  261. <style>
  262. .pass {
  263. font-weight: bold;
  264. font-family: 'Courier New', Courier, monospace;
  265. }
  266. li {
  267. border-bottom: 3px solid #eee;
  268. word-wrap: break-word;
  269. }
  270. li>.title {
  271. transition-duration: 300ms;
  272. display: inline-block;
  273. width: 100%;
  274. cursor: pointer;
  275. font-size: 18px
  276. }
  277. li>.active.title {
  278. font-size: 22px;
  279. font-weight: bold;
  280. }
  281. li>.active+div {
  282. background-image: linear-gradient(to right, #fffaf0, #f4f2ef);
  283. border-left: 3px solid #ccc;
  284. padding-left: 5px;
  285. }
  286. li:hover {
  287. background-color: #eee;
  288. }
  289. .box {
  290. transition-duration: 300ms;
  291. min-height: 210px;
  292. }
  293. .box::before {
  294. content: '';
  295. display: block;
  296. width: 100%;
  297. height: 3px;
  298. margin-top: 10px;
  299. border-bottom: 1px solid;
  300. border-image: linear-gradient(to right, transparent, #6aaad8, #75ddb6, transparent) 1;
  301. }
  302. .box:hover::before {
  303. border-bottom: 3px solid;
  304. border-image: linear-gradient(to right, transparent, #6aaad8, #75ddb6, transparent) 1;
  305. }
  306. .box:hover {
  307. background-image: linear-gradient(#faffef, transparent);
  308. }
  309. .page-header.nohr {
  310. margin-bottom: 0;
  311. border-bottom: none;
  312. }
  313. span.full {
  314. width: 100%;
  315. display: inline-block;
  316. cursor: pointer;
  317. }
  318. .breadcrumb {
  319. padding: 0;
  320. margin-bottom: 5px;
  321. }
  322. .auto-warp {
  323. word-wrap: break-word;
  324. word-break: break-all;
  325. overflow: hidden;
  326. transition-duration: 500ms;
  327. max-height: 22.4px;
  328. }
  329. .auto-warp:hover {
  330. max-height: 100px;
  331. }
  332. </style>
  333. </head>
  334. <body>
  335. <div class="page-header nohr">
  336. <h2>
  337. <!-- <img src="RC.png" width="32px" style="position:relative;top:-4px"> -->
  338. 实萌
  339. </h2>
  340. <span id="loginSt" class="label label-default">未登录</span>
  341. <span id="user" class="label label-default">-</span>
  342. <button class="btn btn-danger btn-xs e" onclick="postLogin()" title="刷新数据">
  343. <span class="glyphicon glyphicon-refresh"></span>
  344. </button>
  345. <button class="btn btn-danger btn-xs e" onclick="delCookies(1);setStatus(0)" onmouseenter="$('#double-off').css('scale','100%')" onmouseleave="$('#double-off').css('scale','0')" title="退出登录">
  346. <span class="glyphicon glyphicon-log-out"></span>
  347. </button>
  348. <button id="double-off" class="btn btn-danger btn-xs e" onclick="delCookies(2);location.reload()" onmouseenter="$('#double-off').css('scale','100%')" onmouseleave="$('#double-off').css('scale','0')" style="scale:0;transition-duration: 1000ms;" data-toggle="tooltip" data-placement="right" title="全部重置">
  349. <span class="glyphicon glyphicon-off"></span>
  350. </button>
  351. <button class="btn btn-danger btn-xs login" onclick="delCookies(2);location.reload()" data-toggle="tooltip" data-placement="right" title="全部重置">
  352. <span class="glyphicon glyphicon-off"></span>
  353. </button>
  354. </div>
  355. <div class="col-sm-12 e box box-contest">
  356. <div class="page-header">
  357. <h3><span class="glyphicon glyphicon-fire"></span> 比赛列表
  358. </h3>
  359. </div>
  360. <ul id="contest" class="list-unstyled"></ul>
  361. <button class="btn btn-default" onclick="$('.contestNew').toggle(300)">新建比赛 <span class="glyphicon glyphicon-plus"></span></button>
  362. <div class="contestNew" style="display:none">
  363. <div class="col-md-4">
  364. 标题
  365. <div class="input-group">
  366. <div class="input-group-addon">第</div>
  367. <input id="tNo" class="form-control" placeholder="0" autocomplete="off">
  368. <div class="input-group-addon">届</div>
  369. <input id="tTitle" class="form-control" placeholder="测试赛" autocomplete="off">
  370. </div>
  371. </div>
  372. <div class="col-md-4">
  373. 开始时间
  374. <input class="form-control" type="date" id="tStart" min="2022-01-01" value="2023-02-10">
  375. </div>
  376. <div class="col-md-4">
  377. 结束时间
  378. <input class="form-control" type="date" id="tEnd" min="2022-01-01" value="2023-02-10">
  379. </div>
  380. <div class="col-sm-12" style="margin-top: 10px;">
  381. <textarea id="tDescription" class="form-control" placeholder="介绍" style="resize: vertical"></textarea>
  382. <button class="btn btn-default" onclick="contestTouch()" style="margin-top: 10px;">提交 <span class="glyphicon glyphicon-send tSend"></span></button>
  383. </div>
  384. </div>
  385. </div>
  386. <div class="col-md-4 col-sm-6 box">
  387. <div class="page-header">
  388. <h3><span class="glyphicon glyphicon-user"></span> 账号
  389. <ul class="btn-group breadcrumb debug" style="display: none;font-size:10px;top:5px">
  390. <li id="bxn-portal" onclick="get('bxn-portal/portal/osforstudent/index')">bxn-portal</li>
  391. <li id="bxn-core-uic" onclick="get('bxn-core-uic/uic/index')">bxn-core-uic</li>
  392. <li id="bxn-library" onclick="get('bxn-library/library/jumpExamreport?jumpUrl=')">bxn-library</li>
  393. <li id="bxn-office" onclick="get('bxn-office/mail/index')">bxn-office</li>
  394. </ul>
  395. </h3>
  396. </div>
  397. <button class="btn btn-default e" onclick="$('.debug').toggle(200)">debug</button>
  398. <div class="login">
  399. <br>
  400. <div class="input-group">
  401. <span class="input-group-addon">账号</span>
  402. <input id="loginId" type="id" class="pass form-control" placeholder="20222222">
  403. </div>
  404. <br>
  405. <div class="input-group">
  406. <span class="input-group-addon">密码</span>
  407. <input id="loginPwd" type="password" class="pass form-control" placeholder="password">
  408. <span class="input-group-btn">
  409. <button class="btn btn-default" type="button" onclick="login().then(postLogin)">
  410. <span class="glyphicon glyphicon-log-in"></span> 登录
  411. </button>
  412. </span>
  413. </div>
  414. <br>
  415. <button class="btn btn-default"><span class="glyphicon glyphicon-user"></span> 游客</button><small>(可能不计入投票结果)</small>
  416. </div>
  417. <div class="debug" style="display: none;">
  418. <button class="btn btn-default btn-xs" onclick="$('.debugList').empty()"><span class="glyphicon glyphicon-trash"></span> 清除</button>
  419. <ul class="debugList list-unstyled" style="font-family: 'Courier New', Courier, monospace;"></ul>
  420. </div>
  421. </div>
  422. <div style="position:fixed;bottom:10px;left:10px;color:#ccc">
  423. 北师大实验中学
  424. </div>
  425. <div style="position:fixed;bottom:10px;right:10px;color:#ccc" class="banner"></div>
  426. </body>