Files
Koneko_api_for_QZ-Music/6a2d8996cc974039d1dfbbf7_Koneko_插件开发文档_v0.0.3.md
2026-06-20 12:48:28 +08:00

84 KiB

Koneko 鎻掍欢寮€鍙戣俯鍧戜笌鏈€缁堜唬鐮佹枃妗?

鐗堟湰锛?.0.3 浣滆€咃細浜戞眬(Miao-moe) 鏁寸悊鏃ユ湡锛?026-06-19 鐢ㄩ€旓細姹囨€讳笌 QZ Music v2 鎷撳睍鎻掍欢寮€鍙戠浉鍏崇殑鍏ㄩ儴韪╁潙鐐广€佹渶缁堜唬鐮併€侀摼鎺?API锛屼究浜庤縼绉诲埌鍏朵粬 AI/宸ュ叿缁х画杩唬銆?


涓€銆侀」鐩儗鏅笌鐩爣

涓?QZ Music v2 缂栧啓 6 涓煶涔愬钩鍙版嫇灞曟彃浠讹細QQ闊充箰銆侀叿鐙楅煶涔愩€侀叿鎴戦煶涔愩€佺綉鏄撲簯闊充箰銆佸挭鍜曢煶涔愩€丟IT闊虫簮銆? 鏍稿績闇€姹傦細

  • 瀹樻柟鎼滅储鍙敤
  • 澶?API 闊虫簮娴嬮€熷鐏?- 鐜鍙橀噺璇诲彇 global.env
  • 鐗堟湰鍙蜂粠 0.0.1 寮€濮嬶紝姣忔淇敼缁熶竴鍗囩増
  • 浠呮敮鎸佸崟 .js 鏂囦欢鏍煎紡锛圦Z Music v2 瀹為檯鍔犺浇鐨勬槸 .js 鏂囦欢锛宍plugin.json` 鍐呭祵鍦ㄤ唬鐮侀噷锛?

浜屻€侀噸澶ц俯鍧戠偣锛堟寜鏃堕棿椤哄簭锛?

1. axios 涓嶅彲鐢?QZ Music v2 杩愯鐜娌℃湁 axios锛屽繀椤荤敤 Node.js 鍐呯疆 http/https 妯″潡銆?

2. 鎻掍欢鏍煎紡婕斿彉

  • 鏈€鍒濈洿鎺ュ啓 module.exports = { ... }锛屽悗鏉ュ彂鐜?app 鏈熸湜 plugin.musicSearch.search(...) 杩欐牱鐨勫璞℃柟娉曟牸寮忋€?- 12-4 鐗堟湰涔嬪墠灏濊瘯杩?folder + plugin.json + index.js 缁撴瀯锛屼絾鐢ㄦ埛鏈€缁堢‘璁ゅ彧瑕佸崟 .js 鏂囦欢銆?- pluginInfo 蹇呴』鍐呭祵鍦?module.exports 閲岋紝鍖呭惈 info銆乣env銆乣ext銆乣quality銆乣supportFunc銆?

3. 鐜鍙橀噺璇诲彇

鐢?global.env锛屼笉鏄?process.env銆?

var env = global.env || {}
var CERU_KEY = env.ceru_key || ''

4. Javet / V8 璇硶鍏煎鎬э紙瀵艰嚧"闂€€"锛?蹇呴』浣跨敤鏈€淇濆畧鐨?ES5 椋庢牸璇硶锛?- var 浠f浛 let/const

  • 浼犵粺 function 澹版槑锛屼笉鐢ㄧ澶村嚱鏁?- 涓嶇敤 async/await锛屽彧鐢?Promise 閾惧紡璋冪敤
  • catch 蹇呴』甯﹀弬鏁帮細catch (e) { }锛屼笉鑳藉啓 catch { }
  • 涓嶇敤 Promise.allSettled锛岃嚜宸辩敤 Promise.all + .then/.catch 鍖呰
  • 涓嶇敤瀵硅薄瑙f瀯绛夌幇浠h娉?

5. 鎼滅储缁撴灉瀛楁蹇呴』涓ユ牸鍖归厤

app 鐨?MusicListResponse 瑕佹眰杩斿洖锛?

{
  list: [...],
  allPage: number,
  limit: number,
  total: number,
  source: string
}

鍗曢姝屾洸瀛楁瑕佸寘鍚細

  • id锛氭瓕鏇插敮涓€鏍囪瘑
  • name锛氭瓕鍚?- artists锛氭瓕鎵嬪悕锛堢敤 / 鎴?銆乣鍒嗛殧锛?- albumName/albumId`
  • pic銆乣mPic銆乣sPic锛氬皝闈㈠浘
  • interval锛氭椂闀垮瓧绗︿覆 m:ss
  • qualities锛氶煶璐ㄥぇ灏忔槧灏?- source锛氬钩鍙版爣璇? 缃戞槗浜戞悳绱箣鍓嶇敤 { songs: [], total: 0 } 瀵艰嚧鎶ラ敊锛歚Field 'list' is required for type ...`銆?

6. 闊宠川鍙傛暟鏍煎紡

HUIBQ 绛夌涓夋柟 API 瑕佹眰闊宠川鍙傛暟甯?k锛屽 320k銆乣128k銆乣999k銆?mapBr 杩斿洖 '320' 鏃惰繑鍥炵殑鏄?fallback 鏃犳晥 URL锛屽鑷存墍鏈夊钩鍙版挱鏀惧け璐ャ€?

7. 缃戞槗浜戝皝闈㈠浘

缃戞槗浜戞悳绱?API 杩斿洖鐨?album.picId 鏄暟瀛楋紝闇€瑕?Base64 缂栫爜鍚庢嫾鎺ワ細

var picIdStr = String(s.album.picId)
var picIdB64 = Buffer.from(picIdStr).toString('base64').replace(/=/g, '')
var pic = 'https://p2.music.126.net/' + picIdB64 + '/' + picIdStr + '.jpg'

8. 閰风嫍鎼滅储鎺ュ彛瀛楁鍚?mobilecdn.kugou.com 鎺ュ彛杩斿洖鐨勫瓧娈垫槸 errcode 涓嶆槸 error_code锛屼笖姝屾洸淇℃伅瀛楁涓?hash銆乣songname銆乣singername銆乣imgurl`銆?

9. 閰风嫍灏侀潰鍥?鏈€鍒濋敊璇湴鐢ㄤ簡閰锋垜鐨?img2.kuwo.cn 鍩熷悕銆傚簲浣跨敤閰风嫍鎼滅储鎺ュ彛杩斿洖鐨?imgurl锛屾浛鎹?{size} 涓?400銆?

10. 娴嬮€熷鐏炬ā寮?鎵€鏈夊钩鍙?getUrl 閲囩敤骞跺彂璇锋眰澶氫釜 API锛屽彇绗竴涓垚鍔熺粨鏋滐細

Promise.all(promises.map(function(p) {
  return p.then(function(v) { return { status: 'fulfilled', value: v } })
          .catch(function(e) { return { status: 'rejected', reason: e } })
})).then(function(results) {
  for (var i = 0; i < results.length; i++) {
    if (results[i].status === 'fulfilled') return results[i].value.url
  }
  return ''
})

11. 鐗堟湰鍙风鐞?姣忔淇敼鎵€鏈夋彃浠剁粺涓€鍗囩骇鐗堟湰鍙枫€傚綋鍓嶆渶缁堢増鏈細0.0.3銆?


涓夈€佺幆澧冨彉閲忚鏄?

key 鐢ㄩ€? 蹇呭~
ceru_key 鑱嗘緶闊虫簮 API 瀵嗛挜 鍚?
playlist_url 缃戞槗浜戜釜浜轰富椤甸摼鎺ワ紝鐢ㄤ簬 userPlaylist 鍚?
cookie 缃戞槗浜?Cookie锛岀敤浜庢悳绱㈠拰 ext 鍔熻兘 鍚?

鍥涖€佹墍鏈夌浉鍏抽摼鎺?API

4.1 鏂囨。/鍙傝€冮摼鎺?- QZ Music v2 鎻掍欢鏂囨。锛堢敤鎴疯嚜缁存姢锛岄潪鍏紑閾炬帴锛?- CSDN 缃戞槗浜戞悳绱?API锛歨ttps://blog.csdn.net/2301_79279502/article/details/135568447

  • CSDN 閰风嫍 API 鏁寸悊锛歨ttps://blog.csdn.net/2301_78245299/article/details/140352615
  • 鏃х増鎻掍欢鍙傝€冿細Koneko_閰风嫍闊充箰_v1.0.9.js
  • 鏃ф棩蹇楋細log_2026-06-19_0.txt

4.2 QQ闊充箰鐩稿叧 API

4.3 閰风嫍闊充箰鐩稿叧 API

4.4 閰锋垜闊充箰鐩稿叧 API

4.5 缃戞槗浜戦煶涔愮浉鍏?API

  • 缃戞槗浜戞悳绱紙GET锛夛細https://music.163.com/api/search/get/web?csrf_token=&hlposttag=&s={kw}&type=1&offset={o}&total=true&limit={n}
  • 缃戞槗浜戞悳绱紙weapi POST锛屽鐢級锛歚https://music.163.com/weapi/cloudsearch/get/web`
  • 缃戞槗浜戠儹璇嶏紙weapi锛夛細https://music.163.com/weapi/search/hot
  • 缃戞槗浜戝畼鏂规挱鏀鹃摼鎺ワ紙weapi锛夛細https://music.163.com/weapi/song/enhance/player/url
  • 缃戞槗浜戠敤鎴锋瓕鍗曪紙weapi锛夛細https://music.163.com/weapi/user/playlist
  • 缃戞槗浜戞瘡鏃ユ帹鑽愶紙weapi锛夛細https://music.163.com/weapi/v1/discovery/recommend/songs
  • 缃戞槗浜戠浜?FM锛坵eapi锛夛細https://music.163.com/weapi/v1/radio/get
  • 缃戞槗浜戝枩娆㈡瓕鏇诧紙weapi锛夛細https://music.163.com/weapi/song/like/get
  • 缃戞槗浜戞瓕鏇茶鎯咃紙weapi锛夛細https://music.163.com/weapi/v3/song/detail
  • 缃戞槗浜戝皝闈㈠浘锛歚https://p2.music.126.net/{base64(picId)}/{picId}.jpg`
  • 鑱嗘緶 缃戞槗浜戦煶婧愶細https://ceruapi.lol/meting-api-0/?server=netease&type=url&id={id}&auth={key}&br={br}
  • bb 缃戞槗浜戦煶婧愶細https://api.bbdcz.cn/music/netease/url?id={id}&br={br}
  • lx 缃戞槗浜戦煶婧愶細https://lxmusicapi.onrender.com/url/wy/{id}/{br}
  • ymc 缃戞槗浜戦煶婧愶細https://api.ymusic.icu/netease/song?id={id}&quality={br}
  • unms 缃戞槗浜戦煶婧愶細https://unms.zeabur.app/netease/url?id={id}&br={br}

4.6 鍜挄闊充箰鐩稿叧 API

4.7 GIT闊虫簮鐩稿叧 API

4.8 鍏朵粬鎻愬強浣嗘湭浣跨敤/澶囩敤

  • https://music.163.com/weapi 绯诲垪锛坵eapi 澶囩敤锛?- https://m.kugou.com/rank/list&json=true
  • http://mobilecdnbj.kugou.com/api/v3/tag/list?pid=0&apiver=2&plat=0

浜斻€佹渶缁堜唬鐮?

浠ヤ笅 6 涓枃浠剁増鏈彿鍧囦负 0.0.3锛屾槸鎴嚦 2026-06-19 鐨勬渶缁堝彲鐢ㄤ唬鐮併€?


Koneko_QQ闊充箰_v0.0.3

'use strict'
var https = require('https')
var http = require('http')
var crypto = require('crypto')

var env = global.env || {}
var CERU_KEY = env.ceru_key || ''

var HEADERS_COMMON = {
  'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
}

function mapBr(q) {
  if (q === '128k' || q === 'standard') return '128k'
  if (q === '320k' || q === 'exhigh') return '320k'
  return '999k'
}

function httpGet(url, headers, timeout) {
  return new Promise(function(resolve, reject) {
    var mod = url.indexOf('https') === 0 ? https : http
    var req = mod.get(url, { headers: headers || {}, timeout: timeout || 8000 }, function(res) {
      var data = ''
      res.on('data', function(chunk) { data += chunk })
      res.on('end', function() {
        try { resolve(JSON.parse(data)) } catch(e) { resolve(data) }
      })
    })
    req.on('error', function(err) { reject(err) })
    req.on('timeout', function() { req.destroy(); reject(new Error('timeout')) })
  })
}

function httpPost(url, body, headers, timeout) {
  return new Promise(function(resolve, reject) {
    var mod = url.indexOf('https') === 0 ? https : http
    var postData = typeof body === 'string' ? body : JSON.stringify(body)
    var opts = {
      method: 'POST',
      headers: {},
      timeout: timeout || 10000
    }
    if (headers) {
      for (var k in headers) { opts.headers[k] = headers[k] }
    }
    opts.headers['Content-Type'] = 'application/json'
    opts.headers['Content-Length'] = Buffer.byteLength(postData)
    var req = mod.request(url, opts, function(res) {
      var data = ''
      res.on('data', function(chunk) { data += chunk })
      res.on('end', function() {
        try { resolve(JSON.parse(data)) } catch(e) { resolve(data) }
      })
    })
    req.on('error', function(err) { reject(err) })
    req.on('timeout', function() { req.destroy(); reject(new Error('timeout')) })
    req.write(postData)
    req.end()
  })
}

var PART_1_INDEXES = [23, 14, 6, 36, 16, 40, 7, 19]
var PART_2_INDEXES = [16, 1, 32, 12, 19, 27, 8, 5]
var SCRAMBLE_VALUES = [89, 39, 179, 150, 218, 82, 58, 252, 177, 52, 186, 123, 120, 64, 242, 133, 143, 161, 121, 179]

function zzcSign(text) {
  var hash = crypto.createHash('sha1').update(text).digest('hex')
  var part1 = ''
  for (var i = 0; i < PART_1_INDEXES.length; i++) { part1 += hash[PART_1_INDEXES[i]] }
  var part2 = ''
  for (var i = 0; i < PART_2_INDEXES.length; i++) { part2 += hash[PART_2_INDEXES[i]] }
  var part3 = []
  for (var i = 0; i < SCRAMBLE_VALUES.length; i++) {
    part3.push(SCRAMBLE_VALUES[i] ^ parseInt(hash.slice(i * 2, i * 2 + 2), 16))
  }
  var b64Part = Buffer.from(part3).toString('base64').replace(/[\/+=]/g, '')
  return ('zzc' + part1 + b64Part + part2).toLowerCase()
}

function signRequest(data) {
  var sign = zzcSign(JSON.stringify(data))
  return httpPost('https://u.y.qq.com/cgi-bin/musics.fcg?sign=' + sign, data, {
    'User-Agent': 'QQMusic 14090508(android 12)'
  })
}

function formatPlayTime(seconds) {
  if (!seconds || isNaN(seconds)) return '--/--'
  var m = Math.floor(seconds / 60)
  var s = seconds % 60
  return m + ':' + (s < 10 ? '0' : '') + s
}

function formatSingerName(singers) {
  if (!singers || singers.length === 0) return ''
  var names = []
  for (var i = 0; i < singers.length; i++) {
    if (singers[i].name) names.push(singers[i].name)
  }
  return names.join('銆?)
}

function formatSize(bytes) {
  if (!bytes) return ''
  var n = parseFloat(bytes)
  if (isNaN(n) || n < 0) return ''
  return (n / (1024 * 1024)).toFixed(2) + 'MB'
}

var musicSearch = {
  limit: 30,
  total: 0,
  page: 0,
  allPage: 1,
  musicSearch: function(str, page, limit, retryNum) {
    var self = this
    if (retryNum === undefined) retryNum = 0
    if (retryNum > 3) return Promise.reject(new Error('鎼滅储澶辫触'))
    var data = {
      comm: {
        ct: '11', cv: '14090508', v: '14090508', tmeAppID: 'qqmusic',
        phonetype: 'EBG-AN10', deviceScore: '553.47', devicelevel: '50', newdevicelevel: '20',
        rom: 'HuaWei/EMOTION/EmotionUI_14.2.0', os_ver: '12',
        OpenUDID: '0', OpenUDID2: '0', QIMEI36: '0', udid: '0',
        chid: '0', aid: '0', oaid: '0', taid: '0', tid: '0', wid: '0', uid: '0', sid: '0',
        modeSwitch: '6', teenMode: '0', ui_mode: '2', nettype: '1020', v4ip: ''
      },
      req: {
        module: 'music.search.SearchCgiService',
        method: 'DoSearchForQQMusicMobile',
        param: {
          search_type: 0, searchid: Math.random().toString().slice(2),
          query: str, page_num: page, num_per_page: limit,
          highlight: 0, nqc_flag: 0, multi_zhida: 0, cat: 2, grp: 1, sin: 0, sem: 0
        }
      }
    }
    return signRequest(data).then(function(body) {
      if (!body || typeof body !== 'object' || body.code !== 0 || !body.req || body.req.code !== 0) {
        return self.musicSearch(str, page, limit, retryNum + 1)
      }
      return body.req.data
    })
  },
  handleResult: function(rawList) {
    if (!rawList || rawList.length === 0) return []
    var list = []
    for (var i = 0; i < rawList.length; i++) {
      var item = rawList[i]
      if (!item.file || !item.file.media_mid) continue
      var albumId = ''
      var albumName = ''
      if (item.album) { albumName = item.album.name; albumId = item.album.mid }
      var picUrl = ''
      if (albumId === '' || albumId === '?) {
        if (item.singer && item.singer.length) {
          picUrl = 'https://y.gtimg.cn/music/photo_new/T001R500x500M000' + item.singer[0].mid + '.jpg'
        }
      } else {
        picUrl = 'https://y.gtimg.cn/music/photo_new/T002R500x500M000' + albumId + '.jpg'
      }
      var qualities = {}
      if (item.file && item.file.size_128mp3) qualities.standard = formatSize(item.file.size_128mp3)
      if (item.file && item.file.size_320mp3) qualities.exhigh = formatSize(item.file.size_320mp3)
      if (item.file && item.file.size_flac) qualities.lossless = formatSize(item.file.size_flac)
      if (item.file && item.file.size_hires) qualities.hires = formatSize(item.file.size_hires)
      list.push({
        id: String(item.mid),
        name: item.name + (item.title_extra || ''),
        artists: formatSingerName(item.singer),
        source: 'tx',
        pic: picUrl,
        mPic: picUrl,
        sPic: picUrl,
        albumName: albumName,
        albumId: String(albumId || ''),
        interval: String(formatPlayTime(item.interval) || '--/--'),
        qualities: qualities
      })
    }
    return list
  },
  search: function(str, page, limit) {
    var self = this
    if (!page) page = 1
    if (limit == null) limit = this.limit
    return this.musicSearch(str, page, limit).then(function(data) {
      if (!data || typeof data !== 'object') {
        return { list: [], allPage: 1, limit: 30, total: 0, source: 'tx' }
      }
      var list = self.handleResult(data.body && data.body.item_song ? data.body.item_song : [])
      if (!list || list.length === 0) {
        return { list: [], allPage: 1, limit: 30, total: 0, source: 'tx' }
      }
      self.total = data.meta && data.meta.estimate_sum ? data.meta.estimate_sum : 0
      self.page = page
      self.allPage = Math.ceil(self.total / limit)
      return { list: list, allPage: self.allPage, limit: limit, total: self.total, source: 'tx' }
    }).catch(function(e) {
      return { list: [], allPage: 1, limit: 30, total: 0, source: 'tx' }
    })
  }
}

var tipSearch = {
  getList: function(str) {
    return httpGet(
      'https://c.y.qq.com/splcloud/fcgi-bin/smartbox_new.fcg?is_xml=0&format=json&key=' + encodeURIComponent(str) + '&loginUin=0&hostUin=0&inCharset=utf8&outCharset=utf-8&notice=0&platform=yqq&needNewCode=0',
      { Referer: 'https://y.qq.com/portal/player.html' }
    ).then(function(body) {
      if (!body || typeof body !== 'object' || body.code !== 0) return []
      var result = { order: [], songs: [], artists: [], albums: [] }
      if (body.data && body.data.song && body.data.song.count > 0) result.order.push('songs')
      if (body.data && body.data.singer && body.data.singer.count > 0) result.order.push('artists')
      if (body.data && body.data.album && body.data.album.count > 0) result.order.push('albums')
      if (body.data && body.data.song && body.data.song.itemlist) {
        for (var i = 0; i < body.data.song.itemlist.length; i++) {
          var item = body.data.song.itemlist[i]
          result.songs.push({ name: item.name, artist: { name: item.singer } })
        }
      }
      if (body.data && body.data.singer && body.data.singer.itemlist) {
        for (var i = 0; i < body.data.singer.itemlist.length; i++) {
          result.artists.push({ name: body.data.singer.itemlist[i].name })
        }
      }
      if (body.data && body.data.album && body.data.album.itemlist) {
        for (var i = 0; i < body.data.album.itemlist.length; i++) {
          result.albums.push({ name: body.data.album.itemlist[i].name })
        }
      }
      return result
    }).catch(function(e) { return [] })
  }
}

var hotSearch = {
  getList: function() {
    var data = {
      comm: { ct: '19', cv: '1803', guid: '0', patch: '118' },
      hotkey: {
        method: 'GetHotkeyForQQMusicPC',
        module: 'tencent_musicsoso_hotkey.HotkeyService',
        param: { search_id: '', uin: 0 }
      }
    }
    return httpPost('https://u.y.qq.com/cgi-bin/musicu.fcg', data, {
      Referer: 'https://y.qq.com/portal/player.html'
    }).then(function(body) {
      if (!body || typeof body !== 'object' || body.code !== 0 || !body.hotkey || !body.hotkey.data) return []
      var list = []
      for (var i = 0; i < body.hotkey.data.vec_hotkey.length; i++) {
        list.push(body.hotkey.data.vec_hotkey[i].query)
      }
      return list
    }).catch(function(e) { return [] })
  }
}

function buildApis(songId, q) {
  var br = mapBr(q)
  var apis = []
  if (CERU_KEY) {
    apis.push({
      name: '鑱嗘緶',
      url: 'https://source.shiqianjiang.cn/api/music/url?source=tx&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-API-Key': CERU_KEY },
      extract: function(res) { return res && res.code === 200 && res.url ? res.url : null }
    })
  }
  apis.push(
    {
      name: 'HUIBQ',
      url: 'https://lxmusicapi.onrender.com/url/tx/' + songId + '/' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Request-Key': 'share-v3' },
      extract: function(res) { return res && res.code === 0 && res.url ? res.url : null }
    },
    {
      name: '蹇嗛煶',
      url: 'https://music.3e0.cn/?server=tencent&type=url&id=' + songId,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (typeof res === 'string' && res.indexOf('http') === 0) return res
        return null
      }
    },
    {
      name: '鏄熸捣',
      url: 'https://music-api.gdstudio.xyz/api.php?types=url&source=tencent&id=' + songId + '&br=' + br,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '鏀堕泦QQ',
      url: 'https://cyapi.top/API/qq_music.php?apikey=4d6f7369632d6170692e63656e6775696769692e636f6d&type=json&mid=' + songId,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '蹇靛績',
      url: 'https://music.nxinxz.com/kgqq/tx.php?id=' + songId + '&level=' + q + '&type=mp3',
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '闀块潚',
      url: 'http://175.27.166.236/kgqq/qq.php?type=mp3&id=' + songId + '&level=' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '鏄熸捣澶?,
      url: 'https://music-dl.sayqz.com/api/?source=qq&id=' + songId + '&type=url&br=' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: 'fish',
      url: 'https://m-api.ceseet.me/url/tx/' + songId + '/' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: 'HYW',
      url: 'https://music.bxa241d4.shop/api/music/url?source=tx&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Card-Key': 'TF-VSS0-8Y73-U1AW-GEXJ' },
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.code === 200 && res.data && res.data.url) return res.data.url
        return null
      }
    }
  )
  return apis
}

function getUrl(songId, quality) {
  var q = quality || '320k'
  var apis = buildApis(songId, q)
  var promises = []
  for (var i = 0; i < apis.length; i++) {
    (function(api) {
      promises.push(
        httpGet(api.url, api.headers, 8000).then(function(res) {
          var url = api.extract(res)
          if (url) {
            console.log('[Koneko QQ闊充箰] ' + api.name + ' 鎴愬姛')
            return { name: api.name, url: url }
          }
          throw new Error(api.name + ' 鏃犳湁鏁圲RL')
        }).catch(function(err) {
          console.error('[Koneko QQ闊充箰] ' + api.name + ' 澶辫触: ' + err.message)
          throw err
        })
      )
    })(apis[i])
  }
  return Promise.all(promises.map(function(p) {
    return p.then(function(v) { return { status: 'fulfilled', value: v } }).catch(function(e) { return { status: 'rejected', reason: e } })
  })).then(function(results) {
    for (var i = 0; i < results.length; i++) {
      if (results[i].status === 'fulfilled') return results[i].value.url
    }
    console.error('[Koneko QQ闊充箰] 鎵€鏈堿PI鍧囧け璐?)
    return ''
  })
}

var leaderboard = { getList: function() { return Promise.resolve([]) } }
var songList = {
  getListDetail: function(id, page, limit) {
    return Promise.resolve({ list: [], page: page || 1, limit: limit || 30, total: 0, source: 'tx', info: { name: '', img: '', desc: '', author: '' } })
  }
}
var singer = { getInfo: function(id) { return Promise.resolve(null) } }
var album = {
  getListDetail: function(id) {
    return Promise.resolve({ list: [], page: 1, limit: 1000, total: 0, source: 'tx', info: { name: '', img: '', desc: '', author: '' } })
  },
  search: function(str, page, limit) { return Promise.resolve([]) }
}

function getLyric(id) { return Promise.resolve('') }
function getPic(songId) { return Promise.resolve('') }
function musicDetail(id) { return Promise.resolve(null) }
function musicInfo(id) { return Promise.resolve(null) }

var pluginInfo = {
  info: { id: 'koneko_tx', name: 'QQ闊充箰 - Koneko', version: '0.0.3', description: 'QQ闊充箰鑱氬悎闊虫簮鎻掍欢锛屽畼鏂规悳绱?澶欰PI闊虫簮锛岃嚜鍔ㄦ祴閫熷鐏惧垏鎹? },
  env: [{ key: 'ceru_key', name: '鑱嗘緶API Key', description: '鑱嗘緶闊虫簮API瀵嗛挜锛岀暀绌哄垯璺宠繃鑱嗘緶闊虫簮' }],
  ext: [],
  quality: [
    { name: '鏍囧噯闊宠川', ui: '鏍?, id: 'standard' },
    { name: '楂樺搧闊宠川', ui: 'HQ', id: 'exhigh' },
    { name: '鏃犳崯闊宠川', ui: 'SQ', id: 'lossless' },
    { name: 'Hi-Res', ui: 'HR', id: 'hires' }
  ],
  supportFunc: []
}

module.exports = {
  musicSearch: musicSearch,
  tipSearch: tipSearch,
  leaderboard: leaderboard,
  songList: songList,
  hotSearch: hotSearch,
  singer: singer,
  album: album,
  getLyric: getLyric,
  getPic: getPic,
  getUrl: getUrl,
  musicDetail: musicDetail,
  musicInfo: musicInfo,
  pluginInfo: pluginInfo
}


Koneko_閰风嫍闊充箰_v0.0.3

'use strict'
var https = require('https')
var http = require('http')

var env = global.env || {}
var CERU_KEY = env.ceru_key || ''

var HEADERS_COMMON = {
  'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
}

function mapBr(q) {
  if (q === '128k' || q === 'standard') return '128k'
  if (q === '320k' || q === 'exhigh') return '320k'
  return '999k'
}

function httpGet(url, headers, timeout) {
  return new Promise(function(resolve, reject) {
    var mod = url.indexOf('https') === 0 ? https : http
    var req = mod.get(url, { headers: headers || {}, timeout: timeout || 8000 }, function(res) {
      var data = ''
      res.on('data', function(chunk) { data += chunk })
      res.on('end', function() {
        try { resolve(JSON.parse(data)) } catch(e) { resolve(data) }
      })
    })
    req.on('error', function(err) { reject(err) })
    req.on('timeout', function() { req.destroy(); reject(new Error('timeout')) })
  })
}

function formatPlayTime(seconds) {
  if (!seconds || isNaN(seconds)) return '--/--'
  var m = Math.floor(seconds / 60)
  var s = seconds % 60
  return m + ':' + (s < 10 ? '0' : '') + s
}

var musicSearch = {
  limit: 30,
  total: 0,
  page: 0,
  allPage: 1,
  search: function(str, page, limit, retryNum) {
    var self = this
    if (retryNum === undefined) retryNum = 0
    if (++retryNum > 3) return Promise.reject(new Error('鎼滅储澶辫触'))
    if (!page) page = 1
    if (limit == null) limit = this.limit
    return httpGet(
      'http://mobilecdn.kugou.com/api/v3/search/song?format=json&keyword=' + encodeURIComponent(str) + '&page=' + page + '&pagesize=' + limit,
      HEADERS_COMMON
    ).then(function(result) {
      if (!result || typeof result !== 'object' || result.errcode !== 0 || !result.data || !result.data.info) {
        return self.search(str, page, limit, retryNum)
      }
      var list = []
      for (var i = 0; i < result.data.info.length; i++) {
        var item = result.data.info[i]
        var picUrl = ''
        if (item.imgurl && item.imgurl.indexOf('http') === 0) {
          picUrl = item.imgurl.replace('{size}', '400')
        }
        if (!picUrl && item.album_img) {
          picUrl = item.album_img
        }
        list.push({
          id: String(item.hash || ''),
          name: String(item.songname || item.song_name || ''),
          artists: String(item.singername || item.singer_name || ''),
          albumName: String(item.album_name || ''),
          albumId: String(item.album_id || ''),
          source: 'kg',
          interval: String(formatPlayTime(item.duration) || '--/--'),
          pic: picUrl,
          mPic: picUrl,
          sPic: picUrl,
          qualities: {}
        })
      }
      self.total = result.data.total
      self.page = page
      self.allPage = Math.ceil(self.total / limit)
      return { list: list, allPage: self.allPage, limit: limit, total: self.total, source: 'kg' }
    }).catch(function(e) {
      return { list: [], allPage: 1, limit: 30, total: 0, source: 'kg' }
    })
  }
}

var tipSearch = {
  getList: function(str) {
    return httpGet(
      'https://searchtip.kugou.com/getSearchTip?MusicTipCount=10&keyword=' + encodeURIComponent(str),
      { 'User-Agent': HEADERS_COMMON['User-Agent'], referer: 'https://www.kugou.com/' }
    ).then(function(body) {
      if (!body || !body.data) return []
      var result = { order: [], songs: [], artists: [], albums: [] }
      if (body.data.songs && body.data.songs.length > 0) result.order.push('songs')
      if (body.data.artists && body.data.artists.length > 0) result.order.push('artists')
      result.songs = []
      if (body.data.songs) {
        for (var i = 0; i < body.data.songs.length; i++) {
          result.songs.push({ name: body.data.songs[i].name, artist: { name: body.data.songs[i].artist } })
        }
      }
      result.artists = []
      if (body.data.artists) {
        for (var i = 0; i < body.data.artists.length; i++) {
          result.artists.push({ name: body.data.artists[i].name })
        }
      }
      return result
    }).catch(function(e) { return [] })
  }
}

var hotSearch = {
  getList: function() {
    return httpGet(
      'http://gateway.kugou.com/api/v3/search/hot_tab?signature=ee44edb9d7155821412d220bcaf509dd&appid=1005&clientver=10026&plat=0',
      HEADERS_COMMON
    ).then(function(body) {
      if (!body || body.error_code !== 0 || !body.data) return []
      var list = []
      for (var i = 0; i < body.data.length; i++) {
        list.push(body.data[i].keyword || body.data[i].searchword || '')
      }
      return list
    }).catch(function(e) { return [] })
  }
}

function buildApis(songId, q) {
  var br = mapBr(q)
  var apis = []
  if (CERU_KEY) {
    apis.push({
      name: '鑱嗘緶',
      url: 'https://source.shiqianjiang.cn/api/music/url?source=kg&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-API-Key': CERU_KEY },
      extract: function(res) { return res && res.code === 200 && res.url ? res.url : null }
    })
  }
  apis.push(
    {
      name: 'HUIBQ',
      url: 'https://lxmusicapi.onrender.com/url/kg/' + songId + '/' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Request-Key': 'share-v3' },
      extract: function(res) { return res && res.code === 0 && res.url ? res.url : null }
    },
    {
      name: '鏄熸捣',
      url: 'https://music-api.gdstudio.xyz/api.php?types=url&source=kugou&id=' + songId + '&br=' + br,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '蹇靛績',
      url: 'https://music.nxinxz.com/kgqq/kg.php?id=' + songId + '&level=' + q + '&type=mp3',
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '闀块潚',
      url: 'https://music.haitangw.cc/kgqq/kg.php?id=' + songId + '&level=' + q + '&type=mp3',
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '鏄熸捣澶?,
      url: 'https://music-dl.sayqz.com/api/?source=kugou&id=' + songId + '&type=url&br=' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: 'fish',
      url: 'https://m-api.ceseet.me/url/kg/' + songId + '/' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: 'HYW',
      url: 'https://music.bxa241d4.shop/api/music/url?source=kg&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Card-Key': 'TF-VSS0-8Y73-U1AW-GEXJ' },
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.code === 200 && res.data && res.data.url) return res.data.url
        return null
      }
    }
  )
  return apis
}

function getUrl(songId, quality) {
  var q = quality || '320k'
  var apis = buildApis(songId, q)
  var promises = []
  for (var i = 0; i < apis.length; i++) {
    (function(api) {
      promises.push(
        httpGet(api.url, api.headers, 8000).then(function(res) {
          var url = api.extract(res)
          if (url) {
            console.log('[Koneko 閰风嫍闊充箰] ' + api.name + ' 鎴愬姛')
            return { name: api.name, url: url }
          }
          throw new Error(api.name + ' 鏃犳湁鏁圲RL')
        }).catch(function(err) {
          console.error('[Koneko 閰风嫍闊充箰] ' + api.name + ' 澶辫触: ' + err.message)
          throw err
        })
      )
    })(apis[i])
  }
  return Promise.all(promises.map(function(p) {
    return p.then(function(v) { return { status: 'fulfilled', value: v } }).catch(function(e) { return { status: 'rejected', reason: e } })
  })).then(function(results) {
    for (var i = 0; i < results.length; i++) {
      if (results[i].status === 'fulfilled') return results[i].value.url
    }
    console.error('[Koneko 閰风嫍闊充箰] 鎵€鏈堿PI鍧囧け璐?)
    return ''
  })
}

var leaderboard = { getList: function() { return Promise.resolve([]) } }
var songList = {
  getListDetail: function(id, page, limit) {
    return Promise.resolve({ list: [], page: page || 1, limit: limit || 30, total: 0, source: 'kg', info: { name: '', img: '', desc: '', author: '' } })
  }
}
var singer = { getInfo: function(id) { return Promise.resolve(null) } }
var album = {
  getListDetail: function(id) {
    return Promise.resolve({ list: [], page: 1, limit: 1000, total: 0, source: 'kg', info: { name: '', img: '', desc: '', author: '' } })
  },
  search: function(str, page, limit) { return Promise.resolve([]) }
}

function getLyric(id) { return Promise.resolve('') }
function getPic(songId) { return Promise.resolve('') }
function musicDetail(id) { return Promise.resolve(null) }
function musicInfo(id) { return Promise.resolve(null) }

var pluginInfo = {
  info: { id: 'koneko_kg', name: '閰风嫍闊充箰 - Koneko', version: '0.0.3', description: '閰风嫍闊充箰鑱氬悎闊虫簮鎻掍欢锛屽畼鏂规悳绱?澶欰PI闊虫簮锛岃嚜鍔ㄦ祴閫熷鐏惧垏鎹? },
  env: [{ key: 'ceru_key', name: '鑱嗘緶API Key', description: '鑱嗘緶闊虫簮API瀵嗛挜锛岀暀绌哄垯璺宠繃鑱嗘緶闊虫簮' }],
  ext: [],
  quality: [
    { name: '鏍囧噯闊宠川', ui: '鏍?, id: 'standard' },
    { name: '楂樺搧闊宠川', ui: 'HQ', id: 'exhigh' },
    { name: '鏃犳崯闊宠川', ui: 'SQ', id: 'lossless' },
    { name: 'Hi-Res', ui: 'HR', id: 'hires' }
  ],
  supportFunc: []
}

module.exports = {
  musicSearch: musicSearch,
  tipSearch: tipSearch,
  leaderboard: leaderboard,
  songList: songList,
  hotSearch: hotSearch,
  singer: singer,
  album: album,
  getLyric: getLyric,
  getPic: getPic,
  getUrl: getUrl,
  musicDetail: musicDetail,
  musicInfo: musicInfo,
  pluginInfo: pluginInfo
}


Koneko_閰锋垜闊充箰_v0.0.3

'use strict'
var https = require('https')
var http = require('http')

var env = global.env || {}
var CERU_KEY = env.ceru_key || ''

var HEADERS_COMMON = {
  'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
}

function mapBr(q) {
  if (q === '128k' || q === 'standard') return '128k'
  if (q === '320k' || q === 'exhigh') return '320k'
  return '999k'
}

function httpGet(url, headers, timeout) {
  return new Promise(function(resolve, reject) {
    var mod = url.indexOf('https') === 0 ? https : http
    var req = mod.get(url, { headers: headers || {}, timeout: timeout || 8000 }, function(res) {
      var data = ''
      res.on('data', function(chunk) { data += chunk })
      res.on('end', function() {
        try { resolve(JSON.parse(data)) } catch(e) { resolve(data) }
      })
    })
    req.on('error', function(err) { reject(err) })
    req.on('timeout', function() { req.destroy(); reject(new Error('timeout')) })
  })
}

function formatPlayTime(seconds) {
  if (!seconds || isNaN(seconds)) return '--/--'
  var m = Math.floor(seconds / 60)
  var s = seconds % 60
  return m + ':' + (s < 10 ? '0' : '') + s
}

var musicSearch = {
  limit: 30,
  total: 0,
  page: 0,
  allPage: 1,
  search: function(str, page, limit, retryNum) {
    var self = this
    if (retryNum === undefined) retryNum = 0
    if (++retryNum > 3) return Promise.reject(new Error('鎼滅储澶辫触'))
    if (!page) page = 1
    if (limit == null) limit = this.limit
    return httpGet(
      'http://search.kuwo.cn/r.s?client=kt&all=' + encodeURIComponent(str) + '&pn=' + (page - 1) + '&rn=' + limit + '&uid=794762570&ver=kwplayer_ar_9.2.2.1&vipver=1&show_copyright_off=1&newver=1&ft=music&cluster=0&strategy=2012&encoding=utf8&rformat=json&vermerge=1&mobi=1&issubtitle=1',
      HEADERS_COMMON
    ).then(function(result) {
      if (!result || !result.abslist || result.abslist.length === 0) {
        return self.search(str, page, limit, retryNum)
      }
      var list = []
      for (var i = 0; i < result.abslist.length; i++) {
        var info = result.abslist[i]
        var songId = (info.MUSICRID || '').replace('MUSIC_', '')
        var qualities = {}
        if (info.N_MINFO) {
          var parts = info.N_MINFO.split(';')
          for (var j = 0; j < parts.length; j++) {
            var m = parts[j].match(/level:(\w+),bitrate:(\d+),format:(\w+),size:([\w.]+)/)
            if (m) {
              if (m[2] === '20900') qualities.jymaster = m[4]
              else if (m[2] === '4000') qualities.hires = m[4]
              else if (m[2] === '2000') qualities.lossless = m[4]
              else if (m[2] === '320') qualities.exhigh = m[4]
              else if (m[2] === '128') qualities.standard = m[4]
            }
          }
        }
        var picUrl = ''
        if (info.ALBUMID) {
          picUrl = 'https://img2.kuwo.cn/star/albumcover/300/' + info.ALBUMID + '.jpg'
        } else {
          picUrl = 'http://artistpicserver.kuwo.cn/pic.web?corp=kuwo&type=rid_pic&pictype=500&size=500&rid=' + songId
        }
        var artistStr = ''
        if (info.ARTIST) artistStr = info.ARTIST.replace(/&/g, '銆?)
        var duration = parseInt(info.DURATION)
        list.push({
          id: String(songId),
          name: String(info.SONGNAME || ''),
          artists: artistStr,
          source: 'kw',
          pic: picUrl,
          mPic: picUrl,
          sPic: picUrl,
          albumName: String(info.ALBUM || ''),
          albumId: String(info.ALBUMID || ''),
          interval: isNaN(duration) ? '--/--' : formatPlayTime(duration),
          qualities: qualities
        })
      }
      self.total = parseInt(result.TOTAL) || 0
      self.page = page
      self.allPage = Math.ceil(self.total / limit)
      return { list: list, allPage: self.allPage, limit: limit, total: self.total, source: 'kw' }
    }).catch(function(e) {
      return { list: [], allPage: 1, limit: 30, total: 0, source: 'kw' }
    })
  }
}

var tipSearch = {
  getList: function(str) {
    return httpGet(
      'https://tips.kuwo.cn/t.s?corp=kuwo&newver=3&p2p=1&notrace=0&c=mbox&w=' + encodeURIComponent(str) + '&encoding=utf8&rformat=json',
      { 'User-Agent': HEADERS_COMMON['User-Agent'], Referer: 'http://www.kuwo.cn/' }
    ).then(function(body) {
      if (!body || !body.abs) return []
      var result = { order: [], songs: [] }
      if (body.abs.length > 0) result.order.push('songs')
      result.songs = []
      for (var i = 0; i < body.abs.length; i++) {
        result.songs.push({ name: body.abs[i].name, artist: { name: body.abs[i].artist } })
      }
      return result
    }).catch(function(e) { return [] })
  }
}

var hotSearch = {
  getList: function() {
    return httpGet(
      'http://hotword.kuwo.cn/hotword.s?prod=kwplayer_ar_9.3.0.1&corp=kuwo&newver=2&vipver=9.3.0.1&source=kwplayer_ar_9.3.0.1_40.apk&p2p=1&notrace=0&uid=0&plat=kwplayer_ar&rformat=json&encoding=utf8&tabid=1',
      { 'User-Agent': 'Dalvik/2.1.0 (Linux; U; Android 9;)' }
    ).then(function(body) {
      if (!body || !body.data) return []
      var list = []
      for (var i = 0; i < body.data.length; i++) {
        list.push(body.data[i].keyword || body.data[i].searchWord || body.data[i].name || '')
      }
      return list
    }).catch(function(e) { return [] })
  }
}

function buildApis(songId, q) {
  var br = mapBr(q)
  var apis = []
  if (CERU_KEY) {
    apis.push({
      name: '鑱嗘緶',
      url: 'https://source.shiqianjiang.cn/api/music/url?source=kw&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-API-Key': CERU_KEY },
      extract: function(res) { return res && res.code === 200 && res.url ? res.url : null }
    })
  }
  apis.push(
    {
      name: 'HUIBQ',
      url: 'https://lxmusicapi.onrender.com/url/kw/' + songId + '/' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Request-Key': 'share-v3' },
      extract: function(res) { return res && res.code === 0 && res.url ? res.url : null }
    },
    {
      name: '鏄熸捣',
      url: 'https://music-api.gdstudio.xyz/api.php?types=url&source=kuwo&id=' + songId + '&br=' + br,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '鏀堕泦KW',
      url: 'https://kw-api.cenguigui.cn/api/song/url?id=' + songId + '&quality=' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '蹇靛績',
      url: 'https://music.nxinxz.com/kgqq/kw.php?id=' + songId + '&level=' + q + '&type=mp3',
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '闀块潚',
      url: 'https://musicapi.haitangw.net/music/kw.php?id=' + songId + '&level=' + q + '&type=mp3',
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '鏄熸捣澶?,
      url: 'https://music-dl.sayqz.com/api/?source=kuwo&id=' + songId + '&type=url&br=' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: 'fish',
      url: 'https://m-api.ceseet.me/url/kw/' + songId + '/' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: 'HYW',
      url: 'https://music.bxa241d4.shop/api/music/url?source=kw&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Card-Key': 'TF-VSS0-8Y73-U1AW-GEXJ' },
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.code === 200 && res.data && res.data.url) return res.data.url
        return null
      }
    }
  )
  return apis
}

function getUrl(songId, quality) {
  var q = quality || '320k'
  var apis = buildApis(songId, q)
  var promises = []
  for (var i = 0; i < apis.length; i++) {
    (function(api) {
      promises.push(
        httpGet(api.url, api.headers, 8000).then(function(res) {
          var url = api.extract(res)
          if (url) {
            console.log('[Koneko 閰锋垜闊充箰] ' + api.name + ' 鎴愬姛')
            return { name: api.name, url: url }
          }
          throw new Error(api.name + ' 鏃犳湁鏁圲RL')
        }).catch(function(err) {
          console.error('[Koneko 閰锋垜闊充箰] ' + api.name + ' 澶辫触: ' + err.message)
          throw err
        })
      )
    })(apis[i])
  }
  return Promise.all(promises.map(function(p) {
    return p.then(function(v) { return { status: 'fulfilled', value: v } }).catch(function(e) { return { status: 'rejected', reason: e } })
  })).then(function(results) {
    for (var i = 0; i < results.length; i++) {
      if (results[i].status === 'fulfilled') return results[i].value.url
    }
    console.error('[Koneko 閰锋垜闊充箰] 鎵€鏈堿PI鍧囧け璐?)
    return ''
  })
}

var leaderboard = { getList: function() { return Promise.resolve([]) } }
var songList = {
  getListDetail: function(id, page, limit) {
    return Promise.resolve({ list: [], page: page || 1, limit: limit || 30, total: 0, source: 'kw', info: { name: '', img: '', desc: '', author: '' } })
  }
}
var singer = { getInfo: function(id) { return Promise.resolve(null) } }
var album = {
  getListDetail: function(id) {
    return Promise.resolve({ list: [], page: 1, limit: 1000, total: 0, source: 'kw', info: { name: '', img: '', desc: '', author: '' } })
  },
  search: function(str, page, limit) { return Promise.resolve([]) }
}

function getLyric(id) { return Promise.resolve('') }
function getPic(songId) { return Promise.resolve('') }
function musicDetail(id) { return Promise.resolve(null) }
function musicInfo(id) { return Promise.resolve(null) }

var pluginInfo = {
  info: { id: 'koneko_kw', name: '閰锋垜闊充箰 - Koneko', version: '0.0.3', description: '閰锋垜闊充箰鑱氬悎闊虫簮鎻掍欢锛屽畼鏂规悳绱?澶欰PI闊虫簮锛岃嚜鍔ㄦ祴閫熷鐏惧垏鎹? },
  env: [{ key: 'ceru_key', name: '鑱嗘緶API Key', description: '鑱嗘緶闊虫簮API瀵嗛挜锛岀暀绌哄垯璺宠繃鑱嗘緶闊虫簮' }],
  ext: [],
  quality: [
    { name: '鏍囧噯闊宠川', ui: '鏍?, id: 'standard' },
    { name: '楂樺搧闊宠川', ui: 'HQ', id: 'exhigh' },
    { name: '鏃犳崯闊宠川', ui: 'SQ', id: 'lossless' },
    { name: 'Hi-Res', ui: 'HR', id: 'hires' }
  ],
  supportFunc: []
}

module.exports = {
  musicSearch: musicSearch,
  tipSearch: tipSearch,
  leaderboard: leaderboard,
  songList: songList,
  hotSearch: hotSearch,
  singer: singer,
  album: album,
  getLyric: getLyric,
  getPic: getPic,
  getUrl: getUrl,
  musicDetail: musicDetail,
  musicInfo: musicInfo,
  pluginInfo: pluginInfo
}


Koneko_缃戞槗浜戦煶涔恄v0.0.3

'use strict'
var https = require('https')
var http = require('http')
var crypto = require('crypto')

var env = global.env || {}
var CERU_KEY = env.ceru_key || ''
var WY_COOKIE = env.cookie || ''

var HEADERS_COMMON = {
  'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
}

function mapBr(q) {
  if (q === '128k' || q === 'standard') return '128000'
  if (q === '320k' || q === 'exhigh') return '320000'
  if (q === '999k' || q === 'lossless') return '999000'
  if (q === 'hires') return '999000'
  return '320000'
}

function httpGet(url, headers, timeout) {
  return new Promise(function(resolve, reject) {
    var mod = url.indexOf('https') === 0 ? https : http
    var req = mod.get(url, { headers: headers || {}, timeout: timeout || 8000 }, function(res) {
      var data = ''
      res.on('data', function(chunk) { data += chunk })
      res.on('end', function() {
        try { resolve(JSON.parse(data)) } catch(e) { resolve(data) }
      })
    })
    req.on('error', function(err) { reject(err) })
    req.on('timeout', function() { req.destroy(); reject(new Error('timeout')) })
  })
}

function httpPost(url, body, headers, timeout) {
  return new Promise(function(resolve, reject) {
    var mod = url.indexOf('https') === 0 ? https : http
    var postData = typeof body === 'string' ? body : JSON.stringify(body)
    var opts = {
      method: 'POST',
      headers: {},
      timeout: timeout || 10000
    }
    if (headers) {
      for (var k in headers) { opts.headers[k] = headers[k] }
    }
    opts.headers['Content-Type'] = 'application/x-www-form-urlencoded'
    opts.headers['Content-Length'] = Buffer.byteLength(postData)
    var req = mod.request(url, opts, function(res) {
      var data = ''
      res.on('data', function(chunk) { data += chunk })
      res.on('end', function() {
        try { resolve(JSON.parse(data)) } catch(e) { resolve(data) }
      })
    })
    req.on('error', function(err) { reject(err) })
    req.on('timeout', function() { req.destroy(); reject(new Error('timeout')) })
    req.write(postData)
    req.end()
  })
}

var EAPI_KEY = 'e82ckenh8dichen8'

function aesEncryptEcb(text, key) {
  var cipher = crypto.createCipheriv('aes-128-ecb', key, '')
  cipher.setAutoPadding(true)
  var encrypted = cipher.update(text, 'utf8', 'hex')
  encrypted += cipher.final('hex')
  return encrypted
}

function eapiEncrypt(url, text) {
  var message = 'nobody' + url + 'use' + text + 'md5forencrypt'
  var digest = crypto.createHash('md5').update(message).digest('hex')
  var data = url + '-36cd479b6b5-' + text + '-36cd479b6b5-' + digest
  return aesEncryptEcb(data, EAPI_KEY)
}

function eapiRequest(url, params) {
  var text = JSON.stringify(params)
  var enc = eapiEncrypt(url, text)
  var body = 'params=' + encodeURIComponent(enc)
  var headers = {
    'User-Agent': 'Mozilla/5.0 (Linux; Android 10; SM-G960U) AppleWebKit/537.36',
    'Content-Type': 'application/x-www-form-urlencoded',
    'Referer': 'https://music.163.com',
    'Cookie': WY_COOKIE || 'os=android; appver=8.9.0;'
  }
  return httpPost('https://music.163.com' + url, body, headers, 15000)
}

var WEAPI_IV = '0102030405060708'
var WEAPI_PRESET_KEY = '0CoJUm6Qyw8W8jud'
var WEAPI_RSA_KEY = '010001'
var WEAPI_RSA_MODULUS = '00e0b509f6259df8642dbc35662901477df22677ec152b5ff68ace615bb7b725152b3ab17a876aea8a5aa76d2e417629ec4ee341f56135fccf695280104e0312ecbda92557c93870114af6c9d05c4f7f0c3685b7a46bee255932575cce10b424d813cfe4875d3e82047b97ddef52741d546b8e289dc6935b3ece0462db0a22b8e7'
var WEAPI_BASE62 = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789'

function aesEncryptCbc(text, key, iv) {
  var cipher = crypto.createCipheriv('aes-128-cbc', key, iv)
  cipher.setAutoPadding(true)
  var encrypted = cipher.update(text, 'utf8', 'base64')
  encrypted += cipher.final('base64')
  return encrypted
}

function rsaEncrypt(text, exponent, modulus) {
  var reversed = text.split('').reverse().join('')
  var bigInt = BigInt('0x' + Buffer.from(reversed).toString('hex'))
  var exp = BigInt('0x' + exponent)
  var mod = BigInt('0x' + modulus)
  var result = bigInt ** exp % mod
  var hex = result.toString(16).padStart(256, '0')
  return hex
}

function weapiEncrypt(text) {
  var secretKey = ''
  for (var i = 0; i < 16; i++) {
    secretKey += WEAPI_BASE62.charAt(Math.floor(Math.random() * 62))
  }
  var firstEnc = aesEncryptCbc(text, WEAPI_PRESET_KEY, WEAPI_IV)
  var secondEnc = aesEncryptCbc(firstEnc, secretKey, WEAPI_IV)
  var rsa = rsaEncrypt(secretKey, WEAPI_RSA_KEY, WEAPI_RSA_MODULUS)
  return {
    params: secondEnc,
    encSecKey: rsa
  }
}

function weapiRequest(url, params) {
  var text = JSON.stringify(params)
  var enc = weapiEncrypt(text)
  var body = 'params=' + encodeURIComponent(enc.params) + '&encSecKey=' + encodeURIComponent(enc.encSecKey)
  var headers = {
    'User-Agent': 'Mozilla/5.0 (Linux; Android 10; SM-G960U) AppleWebKit/537.36',
    'Content-Type': 'application/x-www-form-urlencoded',
    'Referer': 'https://music.163.com',
    'Cookie': WY_COOKIE || 'os=android; appver=8.9.0;'
  }
  return httpPost('https://music.163.com' + url, body, headers, 15000)
}

function formatPlayTime(ms) {
  if (!ms || isNaN(ms)) return '--/--'
  var totalSec = Math.floor(ms / 1000)
  var m = Math.floor(totalSec / 60)
  var s = totalSec % 60
  return m + ':' + (s < 10 ? '0' : '') + s
}

function wySearch(keyword, page, limit) {
  if (!page) page = 1
  if (!limit) limit = 30
  var offset = (page - 1) * limit
  var url = 'https://music.163.com/api/search/get/web?csrf_token=&hlposttag=&s=' + encodeURIComponent(keyword) + '&type=1&offset=' + offset + '&total=true&limit=' + limit
  return httpGet(url, {
    'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36',
    'Referer': 'https://music.163.com',
    'Cookie': WY_COOKIE || 'os=pc; appver=2.10.0;'
  }, 10000).then(function(res) {
    if (!res || !res.result || !res.result.songs) {
      return { list: [], allPage: 1, limit: limit, total: 0, source: 'wy' }
    }
    var list = []
    for (var i = 0; i < res.result.songs.length; i++) {
      var s = res.result.songs[i]
      var artists = []
      if (s.artists) {
        for (var j = 0; j < s.artists.length; j++) {
          artists.push(s.artists[j].name)
        }
      }
      var album = s.album ? s.album.name : ''
      var albumId = s.album ? String(s.album.id) : ''
      var pic = ''
      if (s.album && s.album.picId) {
        var picIdStr = String(s.album.picId)
        var picIdB64 = Buffer.from(picIdStr).toString('base64').replace(/=/g, '')
        pic = 'https://p2.music.126.net/' + picIdB64 + '/' + picIdStr + '.jpg'
      }
      list.push({
        id: s.id ? String(s.id) : '',
        name: s.name || '',
        artists: artists.join('銆?),
        albumName: album,
        albumId: albumId,
        source: 'wy',
        pic: pic,
        mPic: pic,
        sPic: pic,
        interval: formatPlayTime(s.duration),
        qualities: {}
      })
    }
    var total = res.result.songCount || 0
    var allPage = Math.ceil(total / limit)
    return { list: list, allPage: allPage, limit: limit, total: total, source: 'wy' }
  }).catch(function(e) {
    return { list: [], allPage: 1, limit: 30, total: 0, source: 'wy' }
  })
}

function wyTipSearch(keyword) {
  var url = 'https://music.163.com/api/search/get/web?csrf_token=&hlposttag=&s=' + encodeURIComponent(keyword) + '&type=1&offset=0&total=true&limit=10'
  return httpGet(url, {
    'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36',
    'Referer': 'https://music.163.com',
    'Cookie': WY_COOKIE || 'os=pc; appver=2.10.0;'
  }, 10000).then(function(res) {
    if (!res || !res.result || !res.result.songs) return []
    var tips = []
    for (var i = 0; i < res.result.songs.length; i++) {
      var s = res.result.songs[i]
      tips.push(s.name + ' - ' + (s.artists && s.artists[0] ? s.artists[0].name : '鏈煡'))
    }
    return tips
  }).catch(function(e) {
    return []
  })
}

function wyHotSearch() {
  return weapiRequest('/weapi/search/hot', { type: 1111 }).then(function(res) {
    if (!res || !res.result || !res.result.hots) return []
    var hots = []
    for (var i = 0; i < res.result.hots.length; i++) {
      hots.push(res.result.hots[i].first)
    }
    return hots
  }).catch(function(e) {
    return []
  })
}

function wyGetUrl(songId, quality) {
  var br = mapBr(quality)
  var params = {
    ids: '[' + songId + ']',
    br: br,
    csrf_token: ''
  }
  return weapiRequest('/weapi/song/enhance/player/url', params).then(function(res) {
    if (res && res.data && res.data[0] && res.data[0].url) {
      return { url: res.data[0].url, platform: 'wy' }
    }
    throw new Error('no url')
  })
}

function wyUserPlaylist() {
  var url = env.playlist_url || ''
  var uid = ''
  var match = url.match(/id=(\d+)/)
  if (match) uid = match[1]
  if (!uid) {
    match = url.match(/(\d+)/)
    if (match) uid = match[1]
  }
  if (!uid) return Promise.resolve({ playlists: [] })
  var params = { uid: uid, limit: 1000, offset: 0 }
  return weapiRequest('/weapi/user/playlist', params).then(function(res) {
    if (!res || !res.playlist) return { playlists: [] }
    var list = []
    for (var i = 0; i < res.playlist.length; i++) {
      var p = res.playlist[i]
      list.push({
        id: String(p.id),
        name: p.name,
        picUrl: p.coverImgUrl,
        trackCount: p.trackCount,
        description: p.description || ''
      })
    }
    return { playlists: list }
  }).catch(function(e) {
    return { playlists: [] }
  })
}

function wyDailyRecommend() {
  return weapiRequest('/weapi/v1/discovery/recommend/songs', {}).then(function(res) {
    if (!res || !res.data || !res.data.dailySongs) return { songs: [] }
    var songs = []
    for (var i = 0; i < res.data.dailySongs.length; i++) {
      var s = res.data.dailySongs[i]
      var artists = []
      if (s.ar) {
        for (var j = 0; j < s.ar.length; j++) {
          artists.push(s.ar[j].name)
        }
      }
      songs.push({
        id: String(s.id),
        name: s.name,
        artist: artists.join('/'),
        album: s.al ? s.al.name : '',
        picUrl: s.al ? s.al.picUrl : '',
        duration: s.dt || 0,
        platform: 'wy'
      })
    }
    return { songs: songs }
  }).catch(function(e) {
    return { songs: [] }
  })
}

function wyPersonalFm() {
  return weapiRequest('/weapi/v1/radio/get', {}).then(function(res) {
    if (!res || !res.data) return { songs: [] }
    var songs = []
    for (var i = 0; i < res.data.length; i++) {
      var s = res.data[i]
      var artists = []
      if (s.ar) {
        for (var j = 0; j < s.ar.length; j++) {
          artists.push(s.ar[j].name)
        }
      }
      songs.push({
        id: String(s.id),
        name: s.name,
        artist: artists.join('/'),
        album: s.al ? s.al.name : '',
        picUrl: s.al ? s.al.picUrl : '',
        duration: s.dt || 0,
        platform: 'wy'
      })
    }
    return { songs: songs }
  }).catch(function(e) {
    return { songs: [] }
  })
}

function wyMyLikedSongs() {
  return weapiRequest('/weapi/song/like/get', {}).then(function(res) {
    if (!res || !res.data || !res.data.checkPoint) return { songs: [] }
    var ids = res.data.checkPoint
    if (!ids || ids.length === 0) return { songs: [] }
    var idStr = ''
    for (var i = 0; i < ids.length; i++) {
      if (i > 0) idStr += ','
      idStr += ids[i]
    }
    var params = { ids: '[' + idStr + ']', csrf_token: '' }
    return weapiRequest('/weapi/v3/song/detail', params).then(function(detail) {
      if (!detail || !detail.songs) return { songs: [] }
      var songs = []
      for (var i = 0; i < detail.songs.length; i++) {
        var s = detail.songs[i]
        var artists = []
        if (s.ar) {
          for (var j = 0; j < s.ar.length; j++) {
            artists.push(s.ar[j].name)
          }
        }
        songs.push({
          id: String(s.id),
          name: s.name,
          artist: artists.join('/'),
          album: s.al ? s.al.name : '',
          picUrl: s.al ? s.al.picUrl : '',
          duration: s.dt || 0,
          platform: 'wy'
        })
      }
      return { songs: songs }
    })
  }).catch(function(e) {
    return { songs: [] }
  })
}

function ceruGetUrl(songId, quality) {
  if (!CERU_KEY) return Promise.reject(new Error('no key'))
  var br = mapBr(quality)
  var url = 'https://ceruapi.lol/meting-api-0/?server=netease&type=url&id=' + encodeURIComponent(songId) + '&auth=' + encodeURIComponent(CERU_KEY) + '&br=' + br
  return httpGet(url, HEADERS_COMMON, 10000).then(function(res) {
    if (res && res.url) return { url: res.url, platform: 'wy' }
    throw new Error('no url')
  })
}

function bbGetUrl(songId, quality) {
  var br = mapBr(quality)
  var url = 'https://api.bbdcz.cn/music/netease/url?id=' + encodeURIComponent(songId) + '&br=' + br
  return httpGet(url, HEADERS_COMMON, 10000).then(function(res) {
    if (res && res.data && res.data.url) return { url: res.data.url, platform: 'wy' }
    if (res && res.url) return { url: res.url, platform: 'wy' }
    throw new Error('no url')
  })
}

function lxGetUrl(songId, quality) {
  var br = mapBr(quality)
  var url = 'https://lxmusicapi.onrender.com/url/wy/' + encodeURIComponent(songId) + '/' + br
  return httpGet(url, HEADERS_COMMON, 15000).then(function(res) {
    if (res && res.url) return { url: res.url, platform: 'wy' }
    throw new Error('no url')
  })
}

function ymcGetUrl(songId, quality) {
  var br = mapBr(quality)
  var url = 'https://api.ymusic.icu/netease/song?id=' + encodeURIComponent(songId) + '&quality=' + br
  return httpGet(url, HEADERS_COMMON, 10000).then(function(res) {
    if (res && res.data && res.data.url) return { url: res.data.url, platform: 'wy' }
    if (res && res.url) return { url: res.url, platform: 'wy' }
    throw new Error('no url')
  })
}

function unmsGetUrl(songId, quality) {
  var br = mapBr(quality)
  var url = 'https://unms.zeabur.app/netease/url?id=' + encodeURIComponent(songId) + '&br=' + br
  return httpGet(url, HEADERS_COMMON, 10000).then(function(res) {
    if (res && res.url) return { url: res.url, platform: 'wy' }
    throw new Error('no url')
  })
}

function getUrl(songId, quality) {
  var apis = []
  apis.push({ name: 'ceru', fn: ceruGetUrl })
  apis.push({ name: 'bb', fn: bbGetUrl })
  apis.push({ name: 'lx', fn: lxGetUrl })
  apis.push({ name: 'ymc', fn: ymcGetUrl })
  apis.push({ name: 'unms', fn: unmsGetUrl })
  apis.push({ name: 'official', fn: wyGetUrl })

  var promises = []
  for (var i = 0; i < apis.length; i++) {
    var api = apis[i]
    promises.push(
      api.fn(songId, quality).then(function(result) {
        return { status: 'fulfilled', value: result }
      }).catch(function(err) {
        return { status: 'rejected', reason: err }
      })
    )
  }

  return new Promise(function(resolve, reject) {
    Promise.all(promises).then(function(results) {
      for (var i = 0; i < results.length; i++) {
        if (results[i].status === 'fulfilled') {
          resolve(results[i].value)
          return
        }
      }
      reject(new Error('all apis failed'))
    }).catch(function(e) {
      reject(e)
    })
  })
}

var leaderboard = { getList: function() { return Promise.resolve([]) } }
var songList = {
  getListDetail: function(id, page, limit) {
    return Promise.resolve({ list: [], page: page || 1, limit: limit || 30, total: 0, source: 'wy', info: { name: '', img: '', desc: '', author: '' } })
  }
}
var singer = { getInfo: function(id) { return Promise.resolve(null) } }
var album = {
  getListDetail: function(id) {
    return Promise.resolve({ list: [], page: 1, limit: 1000, total: 0, source: 'wy', info: { name: '', img: '', desc: '', author: '' } })
  },
  search: function(str, page, limit) { return Promise.resolve([]) }
}

function getLyric(id) { return Promise.resolve('') }
function getPic(songId) { return Promise.resolve('') }
function musicDetail(id) { return Promise.resolve(null) }
function musicInfo(id) { return Promise.resolve(null) }

var pluginInfo = {
  info: { id: 'koneko_wy', name: '缃戞槗浜戦煶涔?- Koneko', version: '0.0.3', description: '缃戞槗浜戦煶涔愯仛鍚堥煶婧愭彃浠讹紝瀹樻柟鎼滅储+澶欰PI闊虫簮锛岃嚜鍔ㄦ祴閫熷鐏惧垏鎹? },
  env: [
    { key: 'ceru_key', name: '鑱嗘緶API Key', description: '鑱嗘緶闊虫簮API瀵嗛挜锛岀暀绌哄垯璺宠繃鑱嗘緶闊虫簮' },
    { key: 'playlist_url', name: '汉涓婚閾炬帴', description: '缃戞槗浜戦煶涔愪釜浜轰富椤甸摼鎺ワ紝鐢ㄤ簬鑾峰彇涓汉姝屽崟' },
    { key: 'cookie', name: 'Cookie', description: '缃戞槗浜戦煶涔怌ookie锛岀敤浜庢悳绱?姣忔棩鎺ㄨ崘/绉佷汉FM/鎴戝枩娆㈢殑闊充箰' }
  ],
  ext: [
    { name: '汉姝屽崟', description: '閫氳繃鍒嗕韩閾炬帴鑾峰彇涓汉姝屽崟', entry: 'plugin.userPlaylist()', type: 'playlists' },
    { name: '姣忔棩鎺ㄨ崘', description: '鑾峰彇姣忔棩鎺ㄨ崘姝屾洸', entry: 'plugin.dailyRecommend()', type: 'songs' },
    { name: '绉佷汉FM', description: '鑾峰彇绉佷汉FM姝屾洸', entry: 'plugin.personalFm()', type: 'songs' },
    { name: '鎴戝枩娆㈢殑闊充箰', description: '鑾峰彇鎴戝枩娆㈢殑闊充箰鍒楄', entry: 'plugin.myLikedSongs()', type: 'songs' }
  ],
  quality: [
    { name: '鏍囧噯闊宠川', ui: '?, id: 'standard' },
    { name: '楂樺搧闊宠川', ui: 'HQ', id: 'exhigh' },
    { name: '鏃犳崯闊宠川', ui: 'SQ', id: 'lossless' },
    { name: 'Hi-Res', ui: 'HR', id: 'hires' },
    { name: '楂樻竻鐜粫澹?, ui: 'DB', id: 'jyeffect' },
    { name: '娌夋蹈鐜粫澹?, ui: 'SK', id: 'sky' },
    { name: '瓒呮竻姣嶅甫', ui: 'MT', id: 'jymaster' }
  ],
  supportFunc: []
}

var musicSearch = {
  search: function(keyword, page, limit) {
    return wySearch(keyword, page, limit).catch(function(e) {
      return { list: [], allPage: 1, limit: 30, total: 0, source: 'wy' }
    })
  },
  tipSearch: function(keyword) {
    return wyTipSearch(keyword).catch(function(e) {
      return []
    })
  },
  hotSearch: function() {
    return wyHotSearch().catch(function(e) {
      return []
    })
  }
}

var tipSearch = {
  getList: function(str) {
    return wyTipSearch(str).catch(function(e) {
      return []
    })
  }
}

var hotSearch = {
  getList: function() {
    return wyHotSearch().catch(function(e) {
      return []
    })
  }
}

module.exports = {
  musicSearch: musicSearch,
  tipSearch: tipSearch,
  leaderboard: leaderboard,
  songList: songList,
  hotSearch: hotSearch,
  singer: singer,
  album: album,
  getLyric: getLyric,
  getPic: getPic,
  getUrl: getUrl,
  musicDetail: musicDetail,
  musicInfo: musicInfo,
  pluginInfo: pluginInfo,
  userPlaylist: function() {
    return wyUserPlaylist().catch(function(e) {
      return { playlists: [] }
    })
  },
  dailyRecommend: function() {
    return wyDailyRecommend().catch(function(e) {
      return { songs: [] }
    })
  },
  personalFm: function() {
    return wyPersonalFm().catch(function(e) {
      return { songs: [] }
    })
  },
  myLikedSongs: function() {
    return wyMyLikedSongs().catch(function(e) {
      return { songs: [] }
    })
  }
}


Koneko_鍜挄闊充箰_v0.0.3

'use strict'
var https = require('https')
var http = require('http')
var crypto = require('crypto')

var env = global.env || {}
var CERU_KEY = env.ceru_key || ''

var HEADERS_COMMON = {
  'User-Agent': 'Mozilla/5.0 (Linux; U; Android 11.0.0; zh-cn; MI 11 Build/OPR1.170623.032) AppleWebKit/534.30'
}

function mapBr(q) {
  if (q === '128k' || q === 'standard') return '128k'
  if (q === '320k' || q === 'exhigh') return '320k'
  return '999k'
}

function httpGet(url, headers, timeout) {
  return new Promise(function(resolve, reject) {
    var mod = url.indexOf('https') === 0 ? https : http
    var req = mod.get(url, { headers: headers || {}, timeout: timeout || 8000 }, function(res) {
      var data = ''
      res.on('data', function(chunk) { data += chunk })
      res.on('end', function() {
        try { resolve(JSON.parse(data)) } catch(e) { resolve(data) }
      })
    })
    req.on('error', function(err) { reject(err) })
    req.on('timeout', function() { req.destroy(); reject(new Error('timeout')) })
  })
}

function formatPlayTime(seconds) {
  if (!seconds || isNaN(seconds)) return '--/--'
  var m = Math.floor(seconds / 60)
  var s = seconds % 60
  return m + ':' + (s < 10 ? '0' : '') + s
}

function formatSize(bytes) {
  if (!bytes) return ''
  var n = parseFloat(bytes)
  if (isNaN(n) || n < 0) return ''
  return (n / (1024 * 1024)).toFixed(2) + 'MB'
}

function createSignature(time, str) {
  var deviceId = '963B7AA0D21511ED807EE5846EC87D20'
  var signatureMd5 = '6cdc72a439cef99a3418d2a78aa28c73'
  var sign = crypto.createHash('md5').update(str + signatureMd5 + 'yyapp2d16148780a1dcc7408e06336b98cfd50' + deviceId + time).digest('hex')
  return { sign: sign, deviceId: deviceId }
}

var musicSearch = {
  limit: 30,
  total: 0,
  page: 0,
  allPage: 1,
  search: function(str, page, limit, retryNum) {
    var self = this
    if (retryNum === undefined) retryNum = 0
    if (++retryNum > 3) return Promise.reject(new Error('鎼滅储澶辫触'))
    if (!page) page = 1
    if (limit == null) limit = this.limit
    var time = Date.now().toString()
    var signData = createSignature(time, str)
    var searchSwitch = encodeURIComponent('{"song":1,"album":0,"singer":0,"tagSong":1,"mvSong":0,"bestShow":1,"songlist":0,"lyricSong":0}')
    return httpGet(
      'https://jadeite.migu.cn/music_search/v3/search/searchAll?isCorrect=0&isCopyright=1&searchSwitch=' + searchSwitch + '&pageSize=' + limit + '&text=' + encodeURIComponent(str) + '&pageNo=' + page + '&sort=0&sid=USS',
      {
        'User-Agent': HEADERS_COMMON['User-Agent'],
        uiVersion: 'A_music_3.6.1',
        deviceId: signData.deviceId,
        timestamp: time,
        sign: signData.sign,
        channel: '0146921'
      }
    ).then(function(result) {
      if (!result || typeof result !== 'object' || result.code !== '000000' || !result.songResultData) {
        return self.search(str, page, limit, retryNum)
      }
      var songData = result.songResultData || { resultList: [], totalCount: 0 }
      var ids = {}
      var list = []
      var resultList = songData.resultList || []
      for (var i = 0; i < resultList.length; i++) {
        var item = resultList[i]
        if (!Array.isArray(item)) continue
        for (var j = 0; j < item.length; j++) {
          var data = item[j]
          if (!data.songId || !data.copyrightId || ids[data.copyrightId]) continue
          ids[data.copyrightId] = true
          var qualities = {}
          if (data.audioFormats) {
            for (var k = 0; k < data.audioFormats.length; k++) {
              var type = data.audioFormats[k]
              var size = formatSize(type.asize || type.isize)
              if (type.formatType === 'PQ') qualities.standard = size
              else if (type.formatType === 'HQ') qualities.exhigh = size
              else if (type.formatType === 'SQ') qualities.lossless = size
              else if (type.formatType === 'ZQ24') qualities.hires = size
            }
          }
          var img = data.img3 || data.img2 || data.img1 || ''
          if (img && img.indexOf('http') !== 0) img = 'https://d.musicapp.migu.cn' + img
          var artists = ''
          if (data.singerList) {
            var names = []
            for (var k = 0; k < data.singerList.length; k++) {
              if (data.singerList[k].name) names.push(data.singerList[k].name)
            }
            artists = names.join('銆?)
          }
          list.push({
            artists: artists,
            name: data.name || '',
            albumName: data.album || '',
            albumId: data.albumId || '',
            id: data.contentId || '',
            source: 'mg',
            interval: formatPlayTime(data.duration),
            pic: img,
            mPic: img,
            sPic: img,
            qualities: qualities
          })
        }
      }
      self.total = parseInt(songData.totalCount) || 0
      self.page = page
      self.allPage = Math.ceil(self.total / limit)
      return { list: list, allPage: self.allPage, limit: limit, total: self.total, source: 'mg' }
    }).catch(function(e) {
      return { list: [], allPage: 1, limit: 30, total: 0, source: 'mg' }
    })
  }
}

var tipSearch = { getList: function(str) { return Promise.resolve([]) } }
var hotSearch = { getList: function() { return Promise.resolve([]) } }

function buildApis(songId, q) {
  var br = mapBr(q)
  var apis = []
  if (CERU_KEY) {
    apis.push({
      name: '鑱嗘緶',
      url: 'https://source.shiqianjiang.cn/api/music/url?source=mg&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-API-Key': CERU_KEY },
      extract: function(res) { return res && res.code === 200 && res.url ? res.url : null }
    })
  }
  apis.push(
    {
      name: 'HUIBQ',
      url: 'https://lxmusicapi.onrender.com/url/mg/' + songId + '/' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Request-Key': 'share-v3' },
      extract: function(res) { return res && res.code === 0 && res.url ? res.url : null }
    },
    {
      name: '鏄熸捣',
      url: 'https://music-api.gdstudio.xyz/api.php?types=url&source=migu&id=' + songId + '&br=' + br,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '蹇靛績',
      url: 'https://music.nxinxz.com/kgqq/mg.php?id=' + songId + '&level=' + q + '&type=mp3',
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '闀块潚',
      url: 'https://music.haitangw.cc/musicapi/mg.php?id=' + songId + '&level=' + q + '&type=mp3',
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: '鏄熸捣澶?,
      url: 'https://music-dl.sayqz.com/api/?source=migu&id=' + songId + '&type=url&br=' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: 'fish',
      url: 'https://m-api.ceseet.me/url/mg/' + songId + '/' + q,
      headers: HEADERS_COMMON,
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.data && res.data.url) return res.data.url
        return null
      }
    },
    {
      name: 'HYW',
      url: 'https://music.bxa241d4.shop/api/music/url?source=mg&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Card-Key': 'TF-VSS0-8Y73-U1AW-GEXJ' },
      extract: function(res) {
        if (res && res.url) return res.url
        if (res && res.code === 200 && res.data && res.data.url) return res.data.url
        return null
      }
    }
  )
  return apis
}

function getUrl(songId, quality) {
  var q = quality || '320k'
  var apis = buildApis(songId, q)
  var promises = []
  for (var i = 0; i < apis.length; i++) {
    (function(api) {
      promises.push(
        httpGet(api.url, api.headers, 8000).then(function(res) {
          var url = api.extract(res)
          if (url) {
            console.log('[Koneko 鍜挄闊充箰] ' + api.name + ' 鎴愬姛')
            return { name: api.name, url: url }
          }
          throw new Error(api.name + ' 鏃犳湁鏁圲RL')
        }).catch(function(err) {
          console.error('[Koneko 鍜挄闊充箰] ' + api.name + ' 澶辫触: ' + err.message)
          throw err
        })
      )
    })(apis[i])
  }
  return Promise.all(promises.map(function(p) {
    return p.then(function(v) { return { status: 'fulfilled', value: v } }).catch(function(e) { return { status: 'rejected', reason: e } })
  })).then(function(results) {
    for (var i = 0; i < results.length; i++) {
      if (results[i].status === 'fulfilled') return results[i].value.url
    }
    console.error('[Koneko 鍜挄闊充箰] 鎵€鏈堿PI鍧囧け璐?)
    return ''
  })
}

var leaderboard = { getList: function() { return Promise.resolve([]) } }
var songList = {
  getListDetail: function(id, page, limit) {
    return Promise.resolve({ list: [], page: page || 1, limit: limit || 30, total: 0, source: 'mg', info: { name: '', img: '', desc: '', author: '' } })
  }
}
var singer = { getInfo: function(id) { return Promise.resolve(null) } }
var album = {
  getListDetail: function(id) {
    return Promise.resolve({ list: [], page: 1, limit: 1000, total: 0, source: 'mg', info: { name: '', img: '', desc: '', author: '' } })
  },
  search: function(str, page, limit) { return Promise.resolve([]) }
}

function getLyric(id) { return Promise.resolve('') }
function getPic(songId) { return Promise.resolve('') }
function musicDetail(id) { return Promise.resolve(null) }
function musicInfo(id) { return Promise.resolve(null) }

var pluginInfo = {
  info: { id: 'koneko_mg', name: '挄闊充箰 - Koneko', version: '0.0.3', description: '挄闊充箰鑱氬悎闊虫簮鎻掍欢锛屽畼鏂规悳绱?澶欰PI闊虫簮锛岃嚜鍔ㄦ祴閫熷鐏惧垏鎹? },
  env: [{ key: 'ceru_key', name: '鑱嗘緶API Key', description: '鑱嗘緶闊虫簮API瀵嗛挜锛岀暀绌哄垯璺宠繃鑱嗘緶闊虫簮' }],
  ext: [],
  quality: [
    { name: '鏍囧噯闊宠川', ui: '鏍?, id: 'standard' },
    { name: '楂樺搧闊宠川', ui: 'HQ', id: 'exhigh' },
    { name: '鏃犳崯闊宠川', ui: 'SQ', id: 'lossless' },
    { name: 'Hi-Res', ui: 'HR', id: 'hires' }
  ],
  supportFunc: []
}

module.exports = {
  musicSearch: musicSearch,
  tipSearch: tipSearch,
  leaderboard: leaderboard,
  songList: songList,
  hotSearch: hotSearch,
  singer: singer,
  album: album,
  getLyric: getLyric,
  getPic: getPic,
  getUrl: getUrl,
  musicDetail: musicDetail,
  musicInfo: musicInfo,
  pluginInfo: pluginInfo
}


Koneko_GIT闊虫簮_v0.0.3

'use strict'
var https = require('https')
var http = require('http')

var env = global.env || {}
var CERU_KEY = env.ceru_key || ''

var HEADERS_COMMON = {
  'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36'
}

function httpGet(url, headers, timeout) {
  return new Promise(function(resolve, reject) {
    var mod = url.indexOf('https') === 0 ? https : http
    var req = mod.get(url, { headers: headers || {}, timeout: timeout || 8000 }, function(res) {
      var data = ''
      res.on('data', function(chunk) { data += chunk })
      res.on('end', function() {
        try { resolve(JSON.parse(data)) } catch(e) { resolve(data) }
      })
    })
    req.on('error', function(err) { reject(err) })
    req.on('timeout', function() { req.destroy(); reject(new Error('timeout')) })
  })
}

function buildApis(songId, q) {
  var apis = []
  if (CERU_KEY) {
    apis.push({
      name: '鑱嗘緶',
      url: 'https://source.shiqianjiang.cn/api/music/url?source=git&songId=' + songId + '&quality=' + q,
      headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-API-Key': CERU_KEY },
      extract: function(res) { return res && res.code === 200 && res.url ? res.url : null }
    })
  }
  apis.push({
    name: 'HUIBQ',
    url: 'https://lxmusicapi.onrender.com/url/git/' + songId + '/' + q,
    headers: { 'User-Agent': HEADERS_COMMON['User-Agent'], 'X-Request-Key': 'share-v3' },
    extract: function(res) { return res && res.code === 0 && res.url ? res.url : null }
  })
  return apis
}

function getUrl(songId, quality) {
  var q = quality || '320k'
  var apis = buildApis(songId, q)
  var promises = []
  for (var i = 0; i < apis.length; i++) {
    (function(api) {
      promises.push(
        httpGet(api.url, api.headers, 8000).then(function(res) {
          var url = api.extract(res)
          if (url) {
            console.log('[Koneko GIT闊虫簮] ' + api.name + ' 鎴愬姛')
            return { name: api.name, url: url }
          }
          throw new Error(api.name + ' 鏃犳湁鏁圲RL')
        }).catch(function(err) {
          console.error('[Koneko GIT闊虫簮] ' + api.name + ' 澶辫触: ' + err.message)
          throw err
        })
      )
    })(apis[i])
  }
  return Promise.all(promises.map(function(p) {
    return p.then(function(v) { return { status: 'fulfilled', value: v } }).catch(function(e) { return { status: 'rejected', reason: e } })
  })).then(function(results) {
    for (var i = 0; i < results.length; i++) {
      if (results[i].status === 'fulfilled') return results[i].value.url
    }
    console.error('[Koneko GIT闊虫簮] 鎵€鏈堿PI鍧囧け璐?)
    return ''
  })
}

var musicSearch = {
  search: function(str, page, limit) {
    return Promise.resolve({ list: [], total: 0, page: page || 1, limit: limit || 30, allPage: 0, source: 'koneko_git' })
  }
}
var tipSearch = { getList: function(str) { return Promise.resolve([]) } }
var leaderboard = { getList: function() { return Promise.resolve([]) } }
var songList = {
  getListDetail: function(id, page, limit) {
    return Promise.resolve({ list: [], page: page || 1, limit: limit || 30, total: 0, source: 'koneko_git', info: { name: '', img: '', desc: '', author: '' } })
  }
}
var hotSearch = { getList: function() { return Promise.resolve([]) } }
var singer = { getInfo: function(id) { return Promise.resolve(null) } }
var album = {
  getListDetail: function(id) {
    return Promise.resolve({ list: [], page: 1, limit: 1000, total: 0, source: 'koneko_git', info: { name: '', img: '', desc: '', author: '' } })
  },
  search: function(str, page, limit) { return Promise.resolve([]) }
}

function getLyric(id) { return Promise.resolve('') }
function getPic(songId) { return Promise.resolve('') }
function musicDetail(id) { return Promise.resolve(null) }
function musicInfo(id) { return Promise.resolve(null) }

var pluginInfo = {
  info: { id: 'koneko_git', name: 'GIT闊虫簮 - Koneko', version: '0.0.3', description: 'GIT闊虫簮鑱氬悎鎻掍欢锛岃仛鍚?+API闊虫簮锛岃嚜鍔ㄦ祴閫熷鐏惧垏鎹? },
  env: [{ key: 'ceru_key', name: '鑱嗘緶API Key', description: '鑱嗘緶闊虫簮API瀵嗛挜锛岀暀绌哄垯璺宠繃鑱嗘緶闊虫簮' }],
  ext: [],
  quality: [
    { name: '鏍囧噯闊宠川', ui: '鏍?, id: 'standard' },
    { name: '楂樺搧闊宠川', ui: 'HQ', id: 'exhigh' },
    { name: '鏃犳崯闊宠川', ui: 'SQ', id: 'lossless' },
    { name: 'Hi-Res', ui: 'HR', id: 'hires' }
  ],
  supportFunc: []
}

module.exports = {
  musicSearch: musicSearch,
  tipSearch: tipSearch,
  leaderboard: leaderboard,
  songList: songList,
  hotSearch: hotSearch,
  singer: singer,
  album: album,
  getLyric: getLyric,
  getPic: getPic,
  getUrl: getUrl,
  musicDetail: musicDetail,
  musicInfo: musicInfo,
  pluginInfo: pluginInfo
}