main.js 24 KB

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