index.ts 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651
  1. import axios from "axios";
  2. import CryptoJs = require("crypto-js");
  3. const pageSize = 20;
  4. function formatMusicItem(_) {
  5. const albumid = _.albumid || _.album?.id;
  6. const albummid = _.albummid || _.album?.mid;
  7. const albumname = _.albumname || _.album?.title;
  8. return {
  9. id: _.id || _.songid,
  10. songmid: _.mid || _.songmid,
  11. title: _.title || _.songname,
  12. artist: _.singer.map((s) => s.name).join(", "),
  13. artwork: albummid
  14. ? `https://y.gtimg.cn/music/photo_new/T002R300x300M000${albummid}.jpg`
  15. : undefined,
  16. album: albumname,
  17. lrc: _.lyric || undefined,
  18. albumid: albumid,
  19. albummid: albummid,
  20. };
  21. }
  22. function formatAlbumItem(_) {
  23. return {
  24. id: _.albumID || _.albumid,
  25. albumMID: _.albumMID || _.album_mid,
  26. title: _.albumName || _.album_name,
  27. artwork:
  28. _.albumPic ||
  29. `https://y.gtimg.cn/music/photo_new/T002R300x300M000${
  30. _.albumMID || _.album_mid
  31. }.jpg`,
  32. date: _.publicTime || _.pub_time,
  33. singerID: _.singerID || _.singer_id,
  34. artist: _.singerName || _.singer_name,
  35. singerMID: _.singerMID || _.singer_mid,
  36. description: _.desc,
  37. };
  38. }
  39. function formatArtistItem(_) {
  40. return {
  41. name: _.singerName,
  42. id: _.singerID,
  43. singerMID: _.singerMID,
  44. avatar: _.singerPic,
  45. worksNum: _.songNum,
  46. };
  47. }
  48. const searchTypeMap = {
  49. 0: "song",
  50. 2: "album",
  51. 1: "singer",
  52. 3: "songlist",
  53. 7: "song", // 实际上是歌词
  54. 12: "mv",
  55. };
  56. const headers = {
  57. referer: "https://y.qq.com",
  58. "user-agent":
  59. "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/106.0.0.0 Safari/537.36",
  60. Cookie: "uin=",
  61. };
  62. const validSongFilter = (item) => {
  63. return item.pay.pay_play === 0 || item.pay.payplay === 0;
  64. };
  65. async function searchBase(query, page, type) {
  66. const res = (
  67. await axios({
  68. url: "https://u.y.qq.com/cgi-bin/musicu.fcg",
  69. method: "POST",
  70. data: {
  71. req_1: {
  72. method: "DoSearchForQQMusicDesktop",
  73. module: "music.search.SearchCgiService",
  74. param: {
  75. num_per_page: pageSize,
  76. page_num: page,
  77. query: query,
  78. search_type: type,
  79. },
  80. },
  81. },
  82. headers: headers,
  83. xsrfCookieName: "XSRF-TOKEN",
  84. withCredentials: true,
  85. })
  86. ).data;
  87. return {
  88. isEnd: res.req_1.data.meta.sum <= page * pageSize,
  89. data: res.req_1.data.body[searchTypeMap[type]].list,
  90. };
  91. }
  92. async function searchMusic(query, page) {
  93. const songs = await searchBase(query, page, 0);
  94. return {
  95. isEnd: songs.isEnd,
  96. data: songs.data.filter(validSongFilter).map(formatMusicItem),
  97. };
  98. }
  99. async function searchAlbum(query, page) {
  100. const albums = await searchBase(query, page, 2);
  101. return {
  102. isEnd: albums.isEnd,
  103. data: albums.data.map(formatAlbumItem),
  104. };
  105. }
  106. async function searchArtist(query, page) {
  107. const artists = await searchBase(query, page, 1);
  108. return {
  109. isEnd: artists.isEnd,
  110. data: artists.data.map(formatArtistItem),
  111. };
  112. }
  113. async function searchMusicSheet(query, page) {
  114. const musicSheet = await searchBase(query, page, 3);
  115. return {
  116. isEnd: musicSheet.isEnd,
  117. data: musicSheet.data.map((item) => ({
  118. title: item.dissname,
  119. createAt: item.createtime,
  120. description: item.introduction,
  121. playCount: item.listennum,
  122. worksNums: item.song_count,
  123. artwork: item.imgurl,
  124. id: item.dissid,
  125. artist: item.creator.name,
  126. })),
  127. };
  128. }
  129. async function searchLyric(query, page) {
  130. const songs = await searchBase(query, page, 7);
  131. return {
  132. isEnd: songs.isEnd,
  133. data: songs.data.map((it) => ({
  134. ...formatMusicItem(it),
  135. rawLrcTxt: it.content,
  136. })),
  137. };
  138. }
  139. // searchLyric("玫瑰花", 1).then(console.log);
  140. function getQueryFromUrl(key, search) {
  141. try {
  142. const sArr = search.split("?");
  143. let s = "";
  144. if (sArr.length > 1) {
  145. s = sArr[1];
  146. } else {
  147. return key ? undefined : {};
  148. }
  149. const querys = s.split("&");
  150. const result = {};
  151. querys.forEach((item) => {
  152. const temp = item.split("=");
  153. result[temp[0]] = decodeURIComponent(temp[1]);
  154. });
  155. return key ? result[key] : result;
  156. } catch (err) {
  157. // 除去search为空等异常
  158. return key ? "" : {};
  159. }
  160. }
  161. // geturl
  162. function changeUrlQuery(obj, baseUrl) {
  163. const query = getQueryFromUrl(null, baseUrl);
  164. let url = baseUrl.split("?")[0];
  165. const newQuery = { ...query, ...obj };
  166. let queryArr = [];
  167. Object.keys(newQuery).forEach((key) => {
  168. if (newQuery[key] !== undefined && newQuery[key] !== "") {
  169. queryArr.push(`${key}=${encodeURIComponent(newQuery[key])}`);
  170. }
  171. });
  172. return `${url}?${queryArr.join("&")}`.replace(/\?$/, "");
  173. }
  174. const typeMap = {
  175. m4a: {
  176. s: "C400",
  177. e: ".m4a",
  178. },
  179. 128: {
  180. s: "M500",
  181. e: ".mp3",
  182. },
  183. 320: {
  184. s: "M800",
  185. e: ".mp3",
  186. },
  187. ape: {
  188. s: "A000",
  189. e: ".ape",
  190. },
  191. flac: {
  192. s: "F000",
  193. e: ".flac",
  194. },
  195. };
  196. async function getSourceUrl(id, type = "128") {
  197. const mediaId = id;
  198. let uin = "";
  199. const guid = (Math.random() * 10000000).toFixed(0);
  200. const typeObj = typeMap[type];
  201. const file = `${typeObj.s}${id}${mediaId}${typeObj.e}`;
  202. const url = changeUrlQuery(
  203. {
  204. "-": "getplaysongvkey",
  205. g_tk: 5381,
  206. loginUin: uin,
  207. hostUin: 0,
  208. format: "json",
  209. inCharset: "utf8",
  210. outCharset: "utf-8¬ice=0",
  211. platform: "yqq.json",
  212. needNewCode: 0,
  213. data: JSON.stringify({
  214. req_0: {
  215. module: "vkey.GetVkeyServer",
  216. method: "CgiGetVkey",
  217. param: {
  218. filename: [file],
  219. guid: guid,
  220. songmid: [id],
  221. songtype: [0],
  222. uin: uin,
  223. loginflag: 1,
  224. platform: "20",
  225. },
  226. },
  227. comm: {
  228. uin: uin,
  229. format: "json",
  230. ct: 19,
  231. cv: 0,
  232. authst: "",
  233. },
  234. }),
  235. },
  236. "https://u.y.qq.com/cgi-bin/musicu.fcg"
  237. );
  238. return (
  239. await axios({
  240. method: "GET",
  241. url: url,
  242. xsrfCookieName: "XSRF-TOKEN",
  243. withCredentials: true,
  244. })
  245. ).data;
  246. }
  247. async function getAlbumInfo(albumItem) {
  248. const url = changeUrlQuery(
  249. {
  250. data: JSON.stringify({
  251. comm: {
  252. ct: 24,
  253. cv: 10000,
  254. },
  255. albumSonglist: {
  256. method: "GetAlbumSongList",
  257. param: {
  258. albumMid: albumItem.albumMID,
  259. albumID: 0,
  260. begin: 0,
  261. num: 999,
  262. order: 2,
  263. },
  264. module: "music.musichallAlbum.AlbumSongList",
  265. },
  266. }),
  267. },
  268. "https://u.y.qq.com/cgi-bin/musicu.fcg?g_tk=5381&format=json&inCharset=utf8&outCharset=utf-8"
  269. );
  270. const res = (
  271. await axios({
  272. url: url,
  273. headers: headers,
  274. xsrfCookieName: "XSRF-TOKEN",
  275. withCredentials: true,
  276. })
  277. ).data;
  278. return {
  279. musicList: res.albumSonglist.data.songList
  280. .filter((_) => validSongFilter(_.songInfo))
  281. .map((item) => {
  282. const _ = item.songInfo;
  283. return formatMusicItem(_);
  284. }),
  285. };
  286. }
  287. async function getArtistSongs(artistItem, page) {
  288. const url = changeUrlQuery(
  289. {
  290. data: JSON.stringify({
  291. comm: {
  292. ct: 24,
  293. cv: 0,
  294. },
  295. singer: {
  296. method: "get_singer_detail_info",
  297. param: {
  298. sort: 5,
  299. singermid: artistItem.singerMID,
  300. sin: (page - 1) * pageSize,
  301. num: pageSize,
  302. },
  303. module: "music.web_singer_info_svr",
  304. },
  305. }),
  306. },
  307. "http://u.y.qq.com/cgi-bin/musicu.fcg"
  308. );
  309. const res = (
  310. await axios({
  311. url: url,
  312. method: "get",
  313. headers: headers,
  314. xsrfCookieName: "XSRF-TOKEN",
  315. withCredentials: true,
  316. })
  317. ).data;
  318. return {
  319. isEnd: res.singer.data.total_song <= page * pageSize,
  320. data: res.singer.data.songlist.filter(validSongFilter).map(formatMusicItem),
  321. };
  322. }
  323. async function getArtistAlbums(artistItem, page) {
  324. const url = changeUrlQuery(
  325. {
  326. data: JSON.stringify({
  327. comm: {
  328. ct: 24,
  329. cv: 0,
  330. },
  331. singerAlbum: {
  332. method: "get_singer_album",
  333. param: {
  334. singermid: artistItem.singerMID,
  335. order: "time",
  336. begin: (page - 1) * pageSize,
  337. num: pageSize / 1,
  338. exstatus: 1,
  339. },
  340. module: "music.web_singer_info_svr",
  341. },
  342. }),
  343. },
  344. "http://u.y.qq.com/cgi-bin/musicu.fcg"
  345. );
  346. const res = (
  347. await axios({
  348. url,
  349. method: "get",
  350. headers: headers,
  351. xsrfCookieName: "XSRF-TOKEN",
  352. withCredentials: true,
  353. })
  354. ).data;
  355. return {
  356. isEnd: res.singerAlbum.data.total <= page * pageSize,
  357. data: res.singerAlbum.data.list.map(formatAlbumItem),
  358. };
  359. }
  360. async function getArtistWorks(artistItem, page, type) {
  361. if (type === "music") {
  362. return getArtistSongs(artistItem, page);
  363. }
  364. if (type === "album") {
  365. return getArtistAlbums(artistItem, page);
  366. }
  367. }
  368. async function getLyric(musicItem) {
  369. const result = (
  370. await axios({
  371. url: `http://c.y.qq.com/lyric/fcgi-bin/fcg_query_lyric_new.fcg?songmid=${
  372. musicItem.songmid
  373. }&pcachetime=${new Date().getTime()}&g_tk=5381&loginUin=0&hostUin=0&inCharset=utf8&outCharset=utf-8&notice=0&platform=yqq&needNewCode=0`,
  374. headers: { Referer: "https://y.qq.com", Cookie: "uin=" },
  375. method: "get",
  376. xsrfCookieName: "XSRF-TOKEN",
  377. withCredentials: true,
  378. })
  379. ).data;
  380. const res = JSON.parse(
  381. result.replace(/callback\(|MusicJsonCallback\(|jsonCallback\(|\)$/g, "")
  382. );
  383. return {
  384. rawLrc: CryptoJs.enc.Base64.parse(res.lyric).toString(CryptoJs.enc.Utf8),
  385. };
  386. }
  387. async function importMusicSheet(urlLike) {
  388. //
  389. let id;
  390. if (!id) {
  391. id = (urlLike.match(
  392. /https?:\/\/i\.y\.qq\.com\/n2\/m\/share\/details\/taoge\.html\?.*id=([0-9]+)/
  393. ) || [])[1];
  394. }
  395. if (!id) {
  396. id = (urlLike.match(/https?:\/\/y\.qq\.com\/n\/ryqq\/playlist\/([0-9]+)/) ||
  397. [])[1];
  398. }
  399. if (!id) {
  400. id = (urlLike.match(/^(\d+)$/) || [])[1];
  401. }
  402. if (!id) {
  403. return;
  404. }
  405. const result = (
  406. await axios({
  407. url: `http://c.y.qq.com/qzone/fcg-bin/fcg_ucc_getcdinfo_byids_cp.fcg?type=1&utf8=1&disstid=${id}&loginUin=0`,
  408. headers: { Referer: "https://y.qq.com/n/yqq/playlist", Cookie: "uin=" },
  409. method: "get",
  410. xsrfCookieName: "XSRF-TOKEN",
  411. withCredentials: true,
  412. })
  413. ).data;
  414. const res = JSON.parse(
  415. result.replace(/callback\(|MusicJsonCallback\(|jsonCallback\(|\)$/g, "")
  416. );
  417. return res.cdlist[0].songlist.filter(validSongFilter).map(formatMusicItem);
  418. }
  419. /// 榜单
  420. async function getTopLists() {
  421. const list = await axios({
  422. url: "https://u.y.qq.com/cgi-bin/musicu.fcg?_=1577086820633&data=%7B%22comm%22%3A%7B%22g_tk%22%3A5381%2C%22uin%22%3A123456%2C%22format%22%3A%22json%22%2C%22inCharset%22%3A%22utf-8%22%2C%22outCharset%22%3A%22utf-8%22%2C%22notice%22%3A0%2C%22platform%22%3A%22h5%22%2C%22needNewCode%22%3A1%2C%22ct%22%3A23%2C%22cv%22%3A0%7D%2C%22topList%22%3A%7B%22module%22%3A%22musicToplist.ToplistInfoServer%22%2C%22method%22%3A%22GetAll%22%2C%22param%22%3A%7B%7D%7D%7D",
  423. method: "get",
  424. headers: {
  425. Cookie: "uin=",
  426. },
  427. xsrfCookieName: "XSRF-TOKEN",
  428. withCredentials: true,
  429. });
  430. return list.data.topList.data.group.map((e) => ({
  431. title: e.groupName,
  432. data: e.toplist.map((_) => ({
  433. id: _.topId,
  434. description: _.intro,
  435. title: _.title,
  436. period: _.period,
  437. coverImg: _.headPicUrl || _.frontPicUrl,
  438. })),
  439. }));
  440. }
  441. async function getTopListDetail(topListItem: IMusicSheet.IMusicSheetItem) {
  442. const res = await axios({
  443. url: `https://u.y.qq.com/cgi-bin/musicu.fcg?g_tk=5381&data=%7B%22detail%22%3A%7B%22module%22%3A%22musicToplist.ToplistInfoServer%22%2C%22method%22%3A%22GetDetail%22%2C%22param%22%3A%7B%22topId%22%3A${
  444. topListItem.id
  445. }%2C%22offset%22%3A0%2C%22num%22%3A100%2C%22period%22%3A%22${
  446. topListItem.period ?? ""
  447. }%22%7D%7D%2C%22comm%22%3A%7B%22ct%22%3A24%2C%22cv%22%3A0%7D%7D`,
  448. method: "get",
  449. headers: {
  450. Cookie: "uin=",
  451. },
  452. xsrfCookieName: "XSRF-TOKEN",
  453. withCredentials: true,
  454. });
  455. return {
  456. ...topListItem,
  457. musicList: res.data.detail.data.songInfoList
  458. .filter(validSongFilter)
  459. .map(formatMusicItem),
  460. };
  461. }
  462. async function getRecommendSheetTags() {
  463. const res = (
  464. await axios.get(
  465. "https://c.y.qq.com/splcloud/fcgi-bin/fcg_get_diss_tag_conf.fcg?format=json&inCharset=utf8&outCharset=utf-8",
  466. {
  467. headers: {
  468. referer: "https://y.qq.com/",
  469. },
  470. }
  471. )
  472. ).data.data.categories;
  473. const data = res.slice(1).map((_) => ({
  474. title: _.categoryGroupName,
  475. data: _.items.map((tag) => ({
  476. id: tag.categoryId,
  477. title: tag.categoryName,
  478. })),
  479. }));
  480. const pinned = [];
  481. for (let d of data) {
  482. if (d.data.length) {
  483. pinned.push(d.data[0]);
  484. }
  485. }
  486. return {
  487. pinned,
  488. data,
  489. };
  490. }
  491. async function getRecommendSheetsByTag(tag, page) {
  492. const pageSize = 20;
  493. const rawRes = (
  494. await axios.get(
  495. "https://c.y.qq.com/splcloud/fcgi-bin/fcg_get_diss_by_tag.fcg",
  496. {
  497. headers: {
  498. referer: "https://y.qq.com/",
  499. },
  500. params: {
  501. inCharset: "utf8",
  502. outCharset: "utf-8",
  503. sortId: 5,
  504. categoryId: tag?.id || "10000000",
  505. sin: pageSize * (page - 1),
  506. ein: page * pageSize - 1,
  507. },
  508. }
  509. )
  510. ).data;
  511. const res = JSON.parse(
  512. rawRes.replace(/callback\(|MusicJsonCallback\(|jsonCallback\(|\)$/g, "")
  513. ).data;
  514. const isEnd = res.sum <= page * pageSize;
  515. const data = res.list.map((item) => ({
  516. id: item.dissid,
  517. createTime: item.createTime,
  518. title: item.dissname,
  519. artwork: item.imgurl,
  520. description: item.introduction,
  521. playCount: item.listennum,
  522. artist: item.creator?.name ?? "",
  523. }));
  524. return {
  525. isEnd,
  526. data,
  527. };
  528. }
  529. async function getMusicSheetInfo(sheet: IMusicSheet.IMusicSheetItem, page) {
  530. const data = await importMusicSheet(sheet.id);
  531. return {
  532. isEnd: true,
  533. musicList: data,
  534. };
  535. }
  536. // 接口参考:https://jsososo.github.io/QQMusicApi/#/
  537. module.exports = {
  538. platform: "QQ音乐",
  539. version: "0.2.1",
  540. srcUrl:
  541. "https://gitee.com/maotoumao/MusicFreePlugins/raw/v0.1/dist/qq/index.js",
  542. cacheControl: "no-cache",
  543. hints: {
  544. importMusicSheet: [
  545. "QQ音乐APP:自建歌单-分享-分享到微信好友/QQ好友;然后点开并复制链接,直接粘贴即可",
  546. "H5:复制URL并粘贴,或者直接输入纯数字歌单ID即可",
  547. "导入过程中会过滤掉所有VIP/试听/收费音乐,导入时间和歌单大小有关,请耐心等待",
  548. ],
  549. },
  550. primaryKey: ['id', 'songmid'],
  551. supportedSearchType: ["music", "album", "sheet", "artist", "lyric"],
  552. async search(query, page, type) {
  553. if (type === "music") {
  554. return await searchMusic(query, page);
  555. }
  556. if (type === "album") {
  557. return await searchAlbum(query, page);
  558. }
  559. if (type === "artist") {
  560. return await searchArtist(query, page);
  561. }
  562. if (type === "sheet") {
  563. return await searchMusicSheet(query, page);
  564. }
  565. if (type === "lyric") {
  566. return await searchLyric(query, page);
  567. }
  568. },
  569. async getMediaSource(musicItem, quality: IMusic.IQualityKey) {
  570. let purl = "";
  571. let domain = "";
  572. let type = "128";
  573. if (quality === "standard") {
  574. type = "320";
  575. } else if (quality === "high") {
  576. type = "m4a";
  577. } else if (quality === "super") {
  578. type = "flac";
  579. }
  580. const result = await getSourceUrl(musicItem.songmid, type);
  581. if (result.req_0 && result.req_0.data && result.req_0.data.midurlinfo) {
  582. purl = result.req_0.data.midurlinfo[0].purl;
  583. }
  584. if (!purl) {
  585. return null;
  586. }
  587. if (domain === "") {
  588. domain =
  589. result.req_0.data.sip.find((i) => !i.startsWith("http://ws")) ||
  590. result.req_0.data.sip[0];
  591. }
  592. return {
  593. url: `${domain}${purl}`,
  594. };
  595. },
  596. getLyric,
  597. getAlbumInfo,
  598. getArtistWorks,
  599. importMusicSheet,
  600. getTopLists,
  601. getTopListDetail,
  602. getRecommendSheetTags,
  603. getRecommendSheetsByTag,
  604. getMusicSheetInfo,
  605. };