main.js 19 KB

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