main.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593
  1. let argv = require('minimist')(process.argv.slice(2));
  2. const fs = require('fs');
  3. const express = require('express');
  4. const app = express();
  5. const mysql = require('mysql2')
  6. const SendSeekable = require('send-seekable');
  7. const pool = mysql.createPool({
  8. keepAliveInitialDelay: 0, enableKeepAlive: true,
  9. })
  10. require('cnchar');
  11. require('sort-array');
  12. const Kind = {
  13. 0: "去和声伴奏",
  14. 1: "和声伴奏",
  15. 2: "人声",
  16. 3: "贝斯",
  17. 4: "鼓",
  18. 5: "其他",
  19. }
  20. const db = mysql.createConnection({
  21. host: 'mc.andyxie.cn',
  22. user: 'instrunet',
  23. password: 'Moyingren2015',
  24. database: "instrunet_data",
  25. pool: pool,
  26. enableKeepAlive: true,
  27. keepAliveInitialDelay: 0,
  28. })
  29. const nodemailer = require('nodemailer')
  30. let Queue = require('js-queue');
  31. const webp = require('webp-converter')
  32. webp.grant_permission()
  33. let queue = new Queue();
  34. queue.autoRun = false;
  35. const bodyParser = require("body-parser");
  36. const nrc = require('node-run-cmd')
  37. const https = require("node:https");
  38. const {OpenCC} = require("opencc");
  39. const {HttpsProxyAgent} = require("https-proxy-agent");
  40. const sortArray = require("sort-array");
  41. const converters2t = new OpenCC('s2t.json')
  42. const convertert2s = new OpenCC('t2s.json')
  43. app.use(bodyParser.json({"limit": "200mb"}));
  44. app.use(express.json());
  45. app.use(SendSeekable)
  46. const transporter = nodemailer.createTransport({
  47. host: 'smtp.qq.com', port: 465, secure: true, auth: {
  48. user: '3095864740@qq.com', pass: 'caemyuagapsadfff',
  49. }
  50. })
  51. let ncmAPIUrl = "http://localhost:5999";
  52. let currentTask = [];
  53. async function Submit(req) {
  54. let uuid = crypto.randomUUID()
  55. db.connect(function (err) {
  56. if (err) {
  57. console.log(err)
  58. }
  59. })
  60. let albumCover = null;
  61. try {
  62. let resource = await fetch(req.body.file)
  63. if (!resource.ok) {
  64. GC()
  65. currentTask.shift();
  66. console.log(queue.contents)
  67. console.log(currentTask)
  68. queue.next()
  69. return
  70. }
  71. let ab = await resource.arrayBuffer()
  72. fs.writeFileSync("./" + uuid, Buffer.from(ab))
  73. if (URL.canParse(req.body.albumCover)) {
  74. console.log("can parse ")
  75. fetch(req.body.albumCover).then(res => {
  76. res.arrayBuffer().then(async r => {
  77. if(r.byteLength === 0){
  78. GC()
  79. currentTask.shift();
  80. console.log(queue.contents)
  81. console.log(currentTask)
  82. queue.next()
  83. return
  84. }
  85. if (!fs.existsSync("tmp")) {
  86. fs.mkdirSync("tmp")
  87. }
  88. fs.writeFileSync("tmp/" + uuid, new Buffer.from(r));
  89. await webp.cwebp("tmp/" + uuid, "tmp/" + uuid + ".webp", "-q 50 -mt ", "-v")
  90. try {
  91. albumCover = fs.readFileSync("tmp/" + uuid + ".webp")
  92. } catch (e) {
  93. console.log(e)
  94. fs.rmSync("tmp/" + uuid);
  95. albumCover = null;
  96. }
  97. })
  98. })
  99. }
  100. } catch (err) {
  101. GC()
  102. currentTask.shift();
  103. console.log(queue.contents)
  104. console.log(currentTask)
  105. queue.next()
  106. return
  107. }
  108. const callback = function (d) {
  109. console.log(d.toString());
  110. }
  111. const errcb = function (d) {
  112. console.log(d.toString());
  113. }
  114. let kind_of = [];
  115. switch (req.body.kind) {
  116. case 0:
  117. kind_of[0] = `audio-separator ./${uuid} --model_filename UVR-MDX-NET-Inst_HQ_4.onnx --mdx_enable_denoise --mdx_segment_size 4000 --mdx_overlap 0.85 --mdx_batch_size 300 --output_format mp3 --output_dir output`
  118. kind_of[1] = `./output/${uuid}_(Instrumental)_UVR-MDX-NET-Inst_HQ_4.mp3`
  119. kind_of[2] = `${uuid}_(Instrumental)_UVR-MDX-NET-Inst_HQ_4.mp3`
  120. kind_of[3] = `./output/${uuid}_(Vocals)_UVR-MDX-NET-Inst_HQ_4.mp3`
  121. break;
  122. case 1:
  123. kind_of[0] = `audio-separator ./${uuid} --model_filename UVR_MDXNET_KARA_2.onnx --mdx_enable_denoise --mdx_segment_size 4000 --mdx_overlap 0.85 --mdx_batch_size 300 --output_format mp3 --output_dir output`
  124. kind_of[1] = `./output/${uuid}_(Instrumental)_UVR_MDXNET_KARA_2.mp3`
  125. kind_of[2] = `${uuid}_(Instrumental)_UVR_MDXNET_KARA_2.mp3`
  126. kind_of[3] = `./output/${uuid}_(Vocals)_UVR_MDXNET_KARA_2.mp3`
  127. break;
  128. case 3:
  129. kind_of[0] = `audio-separator ./${uuid} --model_filename kuielab_a_bass.onnx --mdx_enable_denoise --mdx_segment_size 4000 --mdx_overlap 0.75 --sample_rate=48000 --output_format mp3 --mdx_batch_size 300 --output_dir output`
  130. kind_of[1] = `./output/${uuid}_(Bass)_kuielab_a_bass.mp3`
  131. kind_of[2] = `${uuid}_(Bass)_kuielab_a_bass.mp3`
  132. kind_of[3] = `./output/${uuid}_(No Bass)_kuielab_a_bass.mp3`
  133. break;
  134. case 4:
  135. kind_of[0] = `audio-separator ./${uuid} --model_filename kuielab_a_drums.onnx --mdx_enable_denoise --mdx_segment_size 4000 --mdx_overlap 0.75 --sample_rate=48000 --output_format mp3 --mdx_batch_size 300 --output_dir output`
  136. kind_of[1] = `./output/${uuid}_(Drums)_kuielab_a_drums.mp3`
  137. kind_of[2] = `${uuid}_(Drums)_kuielab_a_drums.mp3`
  138. kind_of[3] = `./output/${uuid}_(No Drums)_kuielab_a_drums.mp3`
  139. break;
  140. }
  141. function GC() {
  142. try {
  143. fs.rm(kind_of[1], (err) => {
  144. if (err) {
  145. console.log(err);
  146. }
  147. })
  148. fs.rm(kind_of[3], (err) => {
  149. if (err) {
  150. console.log(err);
  151. }
  152. })
  153. fs.rm(uuid, (err) => {
  154. if (err) {
  155. console.log(err);
  156. }
  157. })
  158. fs.rmSync("tmp/" + uuid)
  159. fs.rmSync("tmp/" + uuid + ".webp")
  160. } catch (err) {
  161. }
  162. }
  163. try {
  164. nrc.run([kind_of[0]], {
  165. onData: callback, onError: errcb
  166. }).then(() => {
  167. // db.execute(("INSERT INTO instrunet_entry (uuid, song_name, album_name, link_to, databinary, artist,kind, albumcover, email) VALUES (?,?,?,?,?,?,?,?,?)"), [uuid, req.body.name, req.body.albumName, req.body.link, fs.readFileSync(kind_of[1]), req.body.artist, req.body.kind, albumCover === null ? new Buffer.from([]) : albumCover, (req.body.email === null || req.body.email === undefined || req.body.email === "") ? "" : req.body.email])
  168. GC();
  169. if (req.body.email !== undefined && req.body.email !== "") {
  170. console.log(req.body.email)
  171. transporter.sendMail({
  172. from: '"xiey0" <xiey0@qq.com>',
  173. to: req.body.email,
  174. subject: "你的音频已处理完成。",
  175. text: "你的音频已处理完成。",
  176. html: fs.readFileSync("./Template.html", "utf8").toString().replace("{song_name}", req.body.name).replace("{href_link}", "https://andyxie.cn:4000/player?play=" + uuid).replace("{kind}", Kind[req.body.kind]).replace("{album}", req.body.albumName)
  177. }).then((result) => {
  178. console.log("Message sent: %s", result.messageId)
  179. })
  180. }
  181. currentTask.shift();
  182. console.log(queue.contents)
  183. console.log(currentTask)
  184. queue.next()
  185. })
  186. } catch (e) {
  187. GC()
  188. console.error(e)
  189. return
  190. }
  191. }
  192. app.get('/queue', (req, res) => {
  193. res.header("Access-Control-Allow-Origin", "*");
  194. res.end(JSON.stringify(currentTask));
  195. })
  196. app.post('/submit', SubmitWrapper)
  197. async function SubmitWrapper(req, res) {
  198. // TODO
  199. // May extract logic for dupe check in the future.
  200. if (req.body.file === undefined || req.body.file === null || req.body.file === "" || req.body.kind === undefined || req.body.kind === null || req.body.kind === "") {
  201. res.status(500).header("Access-Control-Allow-Origin", "*").end("想都别想");
  202. return
  203. }
  204. if (req.body.artist === undefined || req.body.albumName === undefined || req.body.artist === null || req.body.albumName === null || req.body.artist === "" || req.body.albumName === "") {
  205. req.body.artist = "未知艺术家"
  206. req.body.albumName = "未知专辑"
  207. }
  208. if (req.body.name === undefined || req.body.name === null || req.body.name === "") {
  209. res.status(500).header("Access-Control-Allow-Origin", "*").end("想都别想");
  210. return
  211. }
  212. if (!URL.canParse(req.body.file)) {
  213. res.status(500).header("Access-Control-Allow-Origin", "*").end("想都别想");
  214. return
  215. }
  216. db.execute(("SELECT uuid, song_name, album_name, artist, kind FROM instrunet_entry WHERE song_name = ? and artist = ? and kind = ?"), [await converters2t.convertPromise(req.body.name), await converters2t.convertPromise(req.body.artist), req.body.kind], async (err, rowsT) => {
  217. db.execute("SELECT uuid, song_name, album_name, artist, kind FROM instrunet_entry WHERE song_name = ? and artist = ? and kind = ?", [await convertert2s.convertPromise(req.body.name), await convertert2s.convertPromise(req.body.artist), req.body.kind], (err, rowsS) => {
  218. let dedupe = [];
  219. rowsT.forEach(row => {
  220. dedupe = dedupe.concat(row);
  221. })
  222. rowsS.forEach(row => {
  223. if (JSON.stringify(dedupe).indexOf(JSON.stringify(row)) === -1) {
  224. dedupe = dedupe.concat(row);
  225. }
  226. })
  227. if (dedupe.length === 0) {
  228. // Verify
  229. if (req.body.file.substring(0, 5) !== "data:") {
  230. res.status(500).header("Access-Control-Allow-Origin", "*").end("想都别想");
  231. return
  232. }
  233. for (const item of currentTask) {
  234. if (req.body.name === item.name) {
  235. if (req.body.albumName === item.albumName) {
  236. if (req.body.kind === item.kind) {
  237. res.header("Access-Control-Allow-Origin", "*");
  238. res.statusCode = 500
  239. res.end("傻逼,重复了。请在盲目上传之前看看库里有没有好么傻逼?")
  240. return
  241. }
  242. }
  243. }
  244. }
  245. for (let prop of [req.body.name, req.body.albumName, req.body.artist]) {
  246. if (prop === undefined || prop === "" || prop === null) {
  247. res.status(500).header("Access-Control-Allow-Origin", "*").end("想都别想");
  248. return
  249. }
  250. }
  251. if (req.body.link === undefined) {
  252. req.body.link = ""
  253. }
  254. queue.add(() => {
  255. Submit(req)
  256. })
  257. currentTask.push({
  258. name: req.body.name,
  259. albumName: req.body.albumName,
  260. kind: req.body.kind,
  261. artist: req.body.artist,
  262. email: req.body.email !== undefined ? req.body.email : null,
  263. })
  264. console.log(queue.contents)
  265. console.log(currentTask)
  266. if (currentTask.length === 1) {
  267. queue.next()
  268. }
  269. res.header("Access-Control-Allow-Origin", "*");
  270. res.end("api_success")
  271. } else {
  272. res.header("Access-Control-Allow-Origin", "*");
  273. res.statusCode = 500
  274. res.end("傻逼,重复了。请在盲目上传之前看看库里有没有好么傻逼?")
  275. }
  276. })
  277. })
  278. }
  279. app.post('/lyric', async (req, res) => {
  280. let name = req.body.name
  281. let artist = req.body.artist;
  282. let album = req.body.albumName;
  283. let lrc = null;
  284. await fetchThatShit()
  285. async function fetchThatShit() {
  286. try {
  287. let fetched = (await (await fetch(`http://192.168.1.166:28883/jsonapi?title=${convertert2s.convertSync(name)}&artist=${convertert2s.convertSync(artist)}&album=${convertert2s.convertSync(album)}`)).text())
  288. lrc = fetched
  289. } catch (err) {
  290. await fetchThatShit()
  291. }
  292. }
  293. res.header("Access-Control-Allow-Origin", "*");
  294. res.end(lrc)
  295. })
  296. app.options('/lyric', async (req, res) => {
  297. res.header("Access-Control-Allow-Origin", "*");
  298. res.header("Access-Control-Allow-Headers", "Content-Type");
  299. res.end()
  300. })
  301. app.options('/submit', function (req, res) {
  302. res.header("Access-Control-Allow-Origin", "*");
  303. res.header("Access-Control-Allow-Headers", "Content-Type");
  304. res.end()
  305. })
  306. app.post('/search_api', async function (req, res) {
  307. db.execute("SELECT uuid, song_name, album_name, artist, kind FROM instrunet_entry WHERE song_name like ? or album_name like ? or artist like ?", [`%${await converters2t.convertPromise(req.body.searchStr)}%`, `%${await converters2t.convertPromise(req.body.searchStr)}%`, `%${await converters2t.convertPromise(req.body.searchStr)}%`], async (err, rowsT) => {
  308. db.execute("SELECT uuid, song_name, album_name, artist, kind FROM instrunet_entry WHERE song_name like ? or album_name like ? or artist like ?", [`%${await convertert2s.convertPromise(req.body.searchStr)}%`, `%${await convertert2s.convertPromise(req.body.searchStr)}%`, `%${await convertert2s.convertPromise(req.body.searchStr)}%`], (err, rowsS) => {
  309. try {
  310. let prepare = [];
  311. rowsT.forEach(row => {
  312. prepare = prepare.concat(row);
  313. })
  314. rowsS.forEach(row => {
  315. if (JSON.stringify(prepare).indexOf(JSON.stringify(row)) === -1) {
  316. prepare = prepare.concat(row);
  317. }
  318. })
  319. for (let obj of prepare) {
  320. obj.stroke = obj.song_name[0].stroke() // Problematic as shit. Do not use.
  321. obj.spell = obj.song_name.spell()
  322. }
  323. sortArray(prepare, {
  324. by: 'spell'
  325. })
  326. res.header("Access-Control-Allow-Origin", "*");
  327. res.end(JSON.stringify(prepare));
  328. } catch (e) {
  329. console.log(e)
  330. }
  331. });
  332. })
  333. })
  334. app.options('/search_api', function (req, res) {
  335. res.header("Access-Control-Allow-Origin", "*");
  336. res.header("Access-Control-Allow-Headers", "Content-Type");
  337. res.end()
  338. })
  339. app.get("/getSingle", function (req, res) {
  340. let uuid = crypto.randomUUID()
  341. if (req.query.id) {
  342. if (req.query.albumcover === "true") {
  343. db.execute("SELECT albumcover FROM instrunet_entry WHERE uuid = ?", [req.query.id], async function (err, rows) {
  344. if (err) {
  345. console.log(err);
  346. }
  347. res.contentType("application/json");
  348. res.header("Access-Control-Allow-Origin", "*");
  349. res.header("Access-Control-Allow-Headers", "Content-Type");
  350. if (rows[0] !== undefined) {
  351. if (rows[0].albumcover !== null) {
  352. if (!fs.existsSync("tmp")) {
  353. fs.mkdirSync("tmp")
  354. }
  355. fs.writeFileSync("tmp/" + uuid, rows[0].albumcover);
  356. await webp.cwebp("tmp/" + uuid, "tmp/" + uuid + ".webp", "-q 50 -mt", "-v")
  357. rows[0].albumcover = fs.readFileSync("tmp/" + uuid + ".webp")
  358. }
  359. }
  360. res.end(JSON.stringify(rows[0]));
  361. try {
  362. fs.rmSync("tmp/" + uuid)
  363. fs.rmSync("tmp/" + uuid + ".webp")
  364. } catch (err) {
  365. }
  366. })
  367. } else
  368. db.execute("SELECT song_name, album_name, artist, kind FROM instrunet_entry WHERE uuid = ?", [req.query.id], async function (err, rows) {
  369. if (err) {
  370. console.log(err);
  371. }
  372. res.contentType("application/json");
  373. res.header("Access-Control-Allow-Origin", "*");
  374. res.header("Access-Control-Allow-Headers", "Content-Type");
  375. res.end(JSON.stringify(rows[0]));
  376. })
  377. } else {
  378. res.contentType("application/json");
  379. res.header("Access-Control-Allow-Origin", "*");
  380. res.header("Access-Control-Allow-Headers", "Content-Type");
  381. res.end("{}")
  382. }
  383. })
  384. // 163
  385. app.options('/ncm/url', function (req, res) {
  386. res.header("Access-Control-Allow-Origin", "*");
  387. res.header("Access-Control-Allow-Headers", "Content-Type");
  388. res.end()
  389. })
  390. app.post("/ncm/url", function (req, res) {
  391. if (Number.isNaN(Number(req.body.id))) {
  392. res.status(500).header("Access-Control-Allow-Origin", "*").end("格式错误");
  393. return
  394. }
  395. if (req.body.id !== undefined && req.body.kind !== undefined) {
  396. try {
  397. let id = req.body.id
  398. fetch(ncmAPIUrl + "/song/download/url/v1?id=" + id + "&level=hires", {
  399. headers: {
  400. Cookie: "MUSIC_U=0087F9D8E102A1C1661EBE1792412F3351DA64D1BD3D862BA77E45E9024524725F3A1983345D9B5A4014C725D19C069DD71081F6FE3659F9E1FD412DC427FB809FAF7789AEEA10E9DE6F06C58D1959BA209D2A83C3FA753261036C4CFD0D143B6C7748B8A6D2DD5C2E96E75D1E847E4AAE035CB2C86B175D9AFC6A164C522ED76E24AE654740AB6BAF5B29597F7E3B0158B2EC1C37F2688279871873FA7ADAEF8280A059E84C4BBFB9E4F225F9A2065DF652247D5496587A7B1E3D35DB0CD3F825C06FE5BFE5CFEF1770847099704360504B73C9B396E37CECE4F9DDEE6001588C3C4F5B2861D9ADF339FC47DD480858CA800620785EA032215B63B81025304DB3331F384793FF8EE681247E34C7931176F2F618B66C122F0602F1EA15F963E422DEC79C257F3577A197BECE71E316C751C3B9F5F3CD07BFDC0270A287A1BB6576"
  401. }
  402. }).then(async result => {
  403. if (result.status === 200 || result.status === 304) {
  404. let result_json = await result.json()
  405. if (result_json.data.url !== null) {
  406. let infos = await (await fetch(ncmAPIUrl + "/song/detail?ids=" + id)).json();
  407. req.body.file = "data:audio/flac;base64," + Buffer.from(await (await fetch(result_json.data.url)).arrayBuffer()).toString("base64")
  408. req.body.name = infos.songs[0].name;
  409. req.body.albumName = infos.songs[0].al.name
  410. req.body.albumCover = infos.songs[0].al.picUrl
  411. req.body.link = result_json.data.url
  412. req.body.artist = infos.songs[0].ar[0].name
  413. /// Complete
  414. SubmitWrapper(req, res)
  415. } else {
  416. res.status(500).header("Access-Control-Allow-Origin", "*").end("不存在");
  417. }
  418. } else {
  419. res.status(404).header("Access-Control-Allow-Origin", "*").send("未找到或出现错误")
  420. }
  421. })
  422. } catch (err) {
  423. res.status(500).header("Access-Control-Allow-Origin", "*");
  424. console.log(err)
  425. }
  426. } else {
  427. res.contentType("application/json");
  428. res.header("Access-Control-Allow-Origin", "*");
  429. res.header("Access-Control-Allow-Headers", "Content-Type");
  430. res.end("傻逼。")
  431. }
  432. })
  433. app.get("/favicon.ico", function (req, res) {
  434. res.end("");
  435. })
  436. let availCache = {};
  437. // setInterval(() => {
  438. // availCache = {};
  439. // }, 1800000)
  440. // Fetch
  441. app.get('/:uuid', function (req, res) {
  442. let uuid = req.params.uuid;
  443. function Provider(err, rows) {
  444. try {
  445. availCache[uuid] = rows[0]
  446. res.contentType("audio/mp3");
  447. res.header("Access-Control-Allow-Origin", "*");
  448. res.header("Access-Control-Allow-Headers", "Content-Type");
  449. res.header('Content-Disposition', `attachment; filename="${encodeURI(availCache[uuid].song_name)}.mp3"`);
  450. /** @type {ArrayBuffer}*/
  451. res.sendSeekable(availCache[uuid].databinary)
  452. } catch (e) {
  453. res.contentType("text/plain");
  454. res.status(500).header("Access-Control-Allow-Origin", "*").end("Err.");
  455. console.log(e)
  456. console.log("Triggered err");
  457. }
  458. }
  459. if (availCache[uuid] !== undefined) {
  460. res.contentType("audio/mp3");
  461. res.header("Access-Control-Allow-Origin", "*");
  462. res.header("Access-Control-Allow-Headers", "Content-Type");
  463. res.header("Access-Control-Allow-Origin", "*");
  464. res.header('Content-Disposition', `attachment; filename="${encodeURI(availCache[uuid].song_name)}.mp3"`);
  465. res.sendSeekable(availCache[uuid].databinary)
  466. } else {
  467. db.execute("SELECT song_name, databinary FROM instrunet_entry WHERE uuid = ?", [uuid], Provider)
  468. }
  469. })
  470. if (argv.https === "true") {
  471. https.createServer({
  472. key: fs.readFileSync('andyxie.cn.key'), cert: fs.readFileSync('andyxie.cn.pem')
  473. }, app).listen(8080)
  474. console.log("Listening on port 8080 with TLS")
  475. } else {
  476. app.listen(8080)
  477. console.log("Listening on port 8080")
  478. }