main.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  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 db = mysql.createConnection({
  11. host: 'mc.andyxie.cn',
  12. user: 'instrunet',
  13. password: 'Moyingren2015',
  14. database: "instrunet_data",
  15. pool: pool,
  16. enableKeepAlive: true,
  17. keepAliveInitialDelay: 0,
  18. })
  19. const nodemailer = require('nodemailer')
  20. let Queue = require('js-queue');
  21. let queue = new Queue();
  22. queue.autoRun = false;
  23. const bodyParser = require("body-parser");
  24. const nrc = require('node-run-cmd')
  25. const https = require("node:https");
  26. const {OpenCC} = require("opencc");
  27. const converters2t = new OpenCC('s2t.json')
  28. const convertert2s = new OpenCC('t2s.json')
  29. app.use(bodyParser.json({"limit": "200mb"}));
  30. app.use(express.json());
  31. app.use(SendSeekable)
  32. const transporter = nodemailer.createTransport({
  33. host: 'smtp.qq.com', port: 465, secure: true, auth: {
  34. user: '3095864740@qq.com', pass: 'caemyuagapsadfff',
  35. }
  36. })
  37. /// TODO
  38. let ncmAPIUrl = "http://localhost:5999";
  39. let res_o = null;
  40. let req_o = null;
  41. let currentTask = [];
  42. function Submit() {
  43. let reqLocal = req_o;
  44. let resLocal = res_o;
  45. let uuid = crypto.randomUUID()
  46. db.connect(function (err) {
  47. if (err) {
  48. console.log(err)
  49. }
  50. })
  51. let albumCover = null;
  52. try {
  53. fetch(reqLocal.body.file).then(res => {
  54. res.arrayBuffer().then(r => fs.writeFileSync("./" + uuid, Buffer.from(r)));
  55. })
  56. fetch(reqLocal.body.albumCover).then(res => {
  57. res.arrayBuffer().then(r => {
  58. albumCover = r
  59. })
  60. })
  61. } catch (err) {
  62. console.log(err)
  63. return
  64. }
  65. const callback = function (d) {
  66. console.log(d.toString());
  67. }
  68. const errcb = function (d) {
  69. console.log(d.toString());
  70. }
  71. let kind_of = [];
  72. switch (reqLocal.body.kind) {
  73. case 0:
  74. 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 --output_format mp3 --mdx_batch_size 300 --output_dir output`
  75. kind_of[1] = `./output/${uuid}_(Instrumental)_UVR-MDX-NET-Inst_HQ_5.mp3`
  76. kind_of[2] = `${uuid}_(Instrumental)_UVR-MDX-NET-Inst_HQ_5.mp3`
  77. kind_of[3] = `./output/${uuid}_(Vocals)_UVR-MDX-NET-Inst_HQ_5.mp3`
  78. break;
  79. case 1:
  80. kind_of[0] = `audio-separator ./${uuid} --model_filename UVR_MDXNET_KARA.onnx --mdx_enable_denoise --mdx_segment_size 4000 --mdx_overlap 0.75 --output_format mp3 --mdx_batch_size 300 --output_dir output`
  81. kind_of[1] = `./output/${uuid}_(Instrumental)_UVR_MDXNET_KARA.mp3`
  82. kind_of[2] = `${uuid}_(Instrumental)_UVR_MDXNET_KARA.mp3`
  83. kind_of[3] = `./output/${uuid}_(Vocals)_UVR_MDXNET_KARA.mp3`
  84. break;
  85. case 3:
  86. kind_of[0] = `audio-separator ./${uuid} --model_filename kuielab_a_bass.onnx --mdx_enable_denoise --mdx_segment_size 4000 --mdx_overlap 0.75 --output_format mp3 --mdx_batch_size 300 --output_dir output`
  87. kind_of[1] = `./output/${uuid}_(Bass)_kuielab_a_bass.mp3`
  88. kind_of[2] = `${uuid}_(Bass)_kuielab_a_bass.mp3`
  89. kind_of[3] = `./output/${uuid}_(No Bass)_kuielab_a_bass.mp3`
  90. break;
  91. case 4:
  92. kind_of[0] = `audio-separator ./${uuid} --model_filename kuielab_a_drums.onnx --mdx_enable_denoise --mdx_segment_size 4000 --mdx_overlap 0.75 --output_format mp3 --mdx_batch_size 300 --output_dir output`
  93. kind_of[1] = `./output/${uuid}_(Drums)_kuielab_a_drums.mp3`
  94. kind_of[2] = `${uuid}_(Drums)_kuielab_a_drums.mp3`
  95. kind_of[3] = `./output/${uuid}_(No Drums)_kuielab_a_drums.mp3`
  96. break;
  97. }
  98. nrc.run([kind_of[0]], {
  99. onData: callback, onError: errcb
  100. }).then(() => {
  101. db.execute(("INSERT INTO instrunet_entry (uuid, song_name, album_name, link_to, databinary, artist,kind, albumcover) VALUES (?,?,?,?,?,?,?,?)"), [uuid, reqLocal.body.name, reqLocal.body.albumName, reqLocal.body.link, fs.readFileSync(kind_of[1]), reqLocal.body.artist, reqLocal.body.kind, albumCover])
  102. db.unprepare(() => {
  103. })
  104. fs.rm(kind_of[1], (err) => {
  105. if (err) {
  106. console.log(err);
  107. }
  108. })
  109. fs.rm(kind_of[3], (err) => {
  110. if (err) {
  111. console.log(err);
  112. }
  113. })
  114. fs.rm(uuid, (err) => {
  115. if (err) {
  116. console.log(err);
  117. }
  118. })
  119. if (reqLocal.body.email !== undefined) {
  120. console.log(reqLocal.body.email)
  121. transporter.sendMail({
  122. from: '"xiey0" <xiey0@qq.com>',
  123. to: reqLocal.body.email,
  124. subject: "你的音频已处理完成。",
  125. text: "你的音频已处理完成。",
  126. html: fs.readFileSync("./Template.html", "utf8").toString().replace("{song_name}", reqLocal.body.name).replace("{href_link}", "https://andyxie.cn:4000/player?play=" + uuid)
  127. }).then((result) => {
  128. console.log("Message sent: %s", result.messageId)
  129. })
  130. }
  131. currentTask.shift();
  132. queue.next()
  133. })
  134. }
  135. app.post('/submit', SubmitWrapper)
  136. function SubmitWrapper(req, res) {
  137. db.execute(`SELECT uuid, song_name, album_name, link_to, artist, kind
  138. from instrunet_entry
  139. where song_name = "${req.body.name}"
  140. and artist = "${req.body.artist}"
  141. and kind = ${req.body.kind} `, (err, result) => {
  142. if (result.length === 0) {
  143. let repeat = false;
  144. currentTask.forEach((item) => {
  145. if(req.body.name === item.name){
  146. if(req.body.albumName === item.albumName){
  147. if(req.body.kind === item.kind){
  148. repeat = true;
  149. res.header("Access-Control-Allow-Origin", "*");
  150. res.statusCode=500
  151. res.end("傻逼,重复了。请在盲目上传之前看看库里有没有好么傻逼?")
  152. }
  153. }
  154. }
  155. })
  156. if(repeat){
  157. return
  158. }
  159. req_o = req;
  160. res_o = res;
  161. if (req.body.file !== undefined) {
  162. queue.add(Submit)
  163. currentTask.push({
  164. name: req.body.name,
  165. albumName: req.body.albumName,
  166. kind: req.body.kind,
  167. })
  168. }
  169. console.log(queue.contents)
  170. console.log(currentTask)
  171. if (currentTask.length === 1) {
  172. queue.next()
  173. }
  174. res.header("Access-Control-Allow-Origin", "*");
  175. res.end("api_success")
  176. } else {
  177. res.header("Access-Control-Allow-Origin", "*");
  178. res.statusCode=500
  179. res.end("傻逼,重复了。请在盲目上传之前看看库里有没有好么傻逼?")
  180. }
  181. })
  182. }
  183. app.post('/lyric', async (req, res) => {
  184. let name = req.body.name
  185. let artist = req.body.artist;
  186. let album = req.body.albumName;
  187. let lrc = await (await fetch(`https://api.lrc.cx/api/v1/lyrics/single?title=${name}&album=${album}&artist=${artist}`)).text()
  188. res.header("Access-Control-Allow-Origin", "*");
  189. res.end(lrc)
  190. })
  191. app.options('/lyric', async (req, res) => {
  192. res.header("Access-Control-Allow-Origin", "*");
  193. res.header("Access-Control-Allow-Headers", "Content-Type");
  194. res.end()
  195. })
  196. app.options('/submit', function (req, res) {
  197. res.header("Access-Control-Allow-Origin", "*");
  198. res.header("Access-Control-Allow-Headers", "Content-Type");
  199. res.end()
  200. })
  201. app.post('/search_api', async function (req, res) {
  202. db.execute(`SELECT uuid, song_name, album_name, artist, kind
  203. FROM instrunet_entry
  204. WHERE song_name like '%${await converters2t.convertPromise(req.body.searchStr)}%'
  205. or album_name like '%${await converters2t.convertPromise(req.body.searchStr)}%'
  206. or artist like '%${await converters2t.convertPromise(req.body.searchStr)}%'`, async (err, rowsT) => {
  207. db.execute(`SELECT uuid, song_name, album_name, artist, kind
  208. FROM instrunet_entry
  209. WHERE song_name like '%${await convertert2s.convertPromise(req.body.searchStr)}%'
  210. or album_name like '%${await convertert2s.convertPromise(req.body.searchStr)}%'
  211. or artist like '%${await convertert2s.convertPromise(req.body.searchStr)}%'`, (err, rowsS) => {
  212. try {
  213. let prepare = [];
  214. rowsT.forEach(row => {
  215. prepare = prepare.concat(row);
  216. })
  217. rowsS.forEach(row => {
  218. if (JSON.stringify(prepare).indexOf(JSON.stringify(row)) === -1) {
  219. prepare = prepare.concat(row);
  220. }
  221. })
  222. res.header("Access-Control-Allow-Origin", "*");
  223. res.end(JSON.stringify(prepare));
  224. } catch (e) {
  225. console.log(e)
  226. }
  227. });
  228. })
  229. })
  230. app.options('/search_api', function (req, res) {
  231. res.header("Access-Control-Allow-Origin", "*");
  232. res.header("Access-Control-Allow-Headers", "Content-Type");
  233. res.end()
  234. })
  235. app.get("/getSingle", function (req, res) {
  236. if (req.query.id) {
  237. db.execute(`SELECT song_name, album_name, artist, kind, albumcover
  238. FROM instrunet_entry
  239. WHERE uuid = "${req.query.id}"`, function (err, rows) {
  240. if (err) {
  241. console.log(err);
  242. }
  243. res.contentType("application/json");
  244. res.header("Access-Control-Allow-Origin", "*");
  245. res.header("Access-Control-Allow-Headers", "Content-Type");
  246. res.end(JSON.stringify(rows[0]));
  247. })
  248. } else {
  249. res.contentType("application/json");
  250. res.header("Access-Control-Allow-Origin", "*");
  251. res.header("Access-Control-Allow-Headers", "Content-Type");
  252. res.end("{}")
  253. }
  254. })
  255. // 163
  256. app.options('/ncm/url', 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("/ncm/url", function (req, res) {
  262. if (req.body.id) {
  263. let id = req.body.id
  264. fetch(ncmAPIUrl + "/song/download/url/v1?id=" + id + "&level=hires", {
  265. headers: {
  266. Cookie: "MUSIC_U=0087F9D8E102A1C1661EBE1792412F3351DA64D1BD3D862BA77E45E9024524725F3A1983345D9B5A4014C725D19C069DD71081F6FE3659F9E1FD412DC427FB809FAF7789AEEA10E9DE6F06C58D1959BA209D2A83C3FA753261036C4CFD0D143B6C7748B8A6D2DD5C2E96E75D1E847E4AAE035CB2C86B175D9AFC6A164C522ED76E24AE654740AB6BAF5B29597F7E3B0158B2EC1C37F2688279871873FA7ADAEF8280A059E84C4BBFB9E4F225F9A2065DF652247D5496587A7B1E3D35DB0CD3F825C06FE5BFE5CFEF1770847099704360504B73C9B396E37CECE4F9DDEE6001588C3C4F5B2861D9ADF339FC47DD480858CA800620785EA032215B63B81025304DB3331F384793FF8EE681247E34C7931176F2F618B66C122F0602F1EA15F963E422DEC79C257F3577A197BECE71E316C751C3B9F5F3CD07BFDC0270A287A1BB6576"
  267. }
  268. }).then(async result => {
  269. let result_json = await result.json()
  270. if (result_json.data.url !== null) {
  271. let infos = await (await fetch(ncmAPIUrl + "/song/detail?ids=" + id)).json();
  272. req.body.file = "data:audio/flac;base64," + Buffer.from(await (await fetch(result_json.data.url)).arrayBuffer()).toString("base64")
  273. req.body.name = infos.songs[0].name;
  274. req.body.albumName = infos.songs[0].al.name
  275. req.body.albumCover = infos.songs[0].al.picUrl
  276. req.body.link = result_json.data.url
  277. req.body.artist = infos.songs[0].ar[0].name
  278. /// Complete
  279. SubmitWrapper(req, res)
  280. } else {
  281. res.status(404).header("Access-Control-Allow-Origin", "*").send("未找到")
  282. }
  283. })
  284. } else {
  285. res.contentType("application/json");
  286. res.header("Access-Control-Allow-Origin", "*");
  287. res.header("Access-Control-Allow-Headers", "Content-Type");
  288. res.end("{}")
  289. }
  290. })
  291. app.get("/favicon.ico", function (req, res) {
  292. res.end("");
  293. })
  294. let availCache = {};
  295. setInterval(() => {
  296. availCache = {};
  297. }, 1800000)
  298. // Fetch
  299. app.get('/:uuid', function (req, res) {
  300. let uuid = req.params.uuid;
  301. function Provider(err, rows) {
  302. try {
  303. res.contentType("audio/mp3");
  304. res.header("Access-Control-Allow-Origin", "*");
  305. res.header("Access-Control-Allow-Headers", "Content-Type");
  306. res.header("Access-Control-Allow-Origin", "*");
  307. res.header('Content-Disposition', `attachment; filename="${encodeURI(rows[0].song_name)}"`);
  308. /** @type {ArrayBuffer}*/
  309. availCache[uuid] = rows[0].databinary
  310. res.sendSeekable(availCache[uuid])
  311. } catch (e) {
  312. console.log(e)
  313. console.log("Triggered err");
  314. }
  315. }
  316. if (availCache[uuid] !== undefined) {
  317. db.execute(`SELECT song_name
  318. FROM instrunet_entry
  319. WHERE uuid = '${uuid}'`, (err, rows) => {
  320. res.contentType("audio/mp3");
  321. res.header("Access-Control-Allow-Origin", "*");
  322. res.header("Access-Control-Allow-Headers", "Content-Type");
  323. res.header("Access-Control-Allow-Origin", "*");
  324. res.header('Content-Disposition', `attachment; filename="${encodeURI(rows[0].song_name)}"`);
  325. res.sendSeekable(availCache[uuid])
  326. })
  327. } else {
  328. db.execute(`SELECT song_name, databinary
  329. FROM instrunet_entry
  330. WHERE uuid = '${uuid}'`, Provider)
  331. db.unprepare()
  332. }
  333. })
  334. if (argv.https === "true") {
  335. https.createServer({
  336. key: fs.readFileSync('andyxie.cn.key'), cert: fs.readFileSync('andyxie.cn.pem')
  337. }, app).listen(8080)
  338. console.log("Listening on port 8080 with TLS")
  339. } else {
  340. app.listen(8080)
  341. console.log("Listening on port 8080")
  342. }