4 Star 7 Fork 0

Gitee 极速下载 / vsc-netease-music

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
此仓库是为了提升国内下载速度的镜像仓库,每日同步一次。 原始仓库: https://github.com/nondanee/vsc-netease-music
克隆/下载
request.js 8.99 KB
一键复制 编辑 原始数据 按行查看 历史
nondanee 提交于 2020-12-30 00:58 . mark
const http = require('http')
const https = require('https')
const crypto = require('crypto')
const parse = require('url').parse
const querystring = require('querystring')
let user = {}
const encrypt = object => {
const buffer = Buffer.from(JSON.stringify(object))
const cipher = crypto.createCipheriv('aes-128-ecb', 'rFgB&h#%2?^eDg:Q', '')
return { eparams: Buffer.concat([cipher.update(buffer), cipher.final()]).toString('hex').toUpperCase() }
}
const request = (method, url, headers, body = null) =>
new Promise((resolve, reject) => {
(url.startsWith('https://') ? https : http).request(Object.assign(parse(url), { method, headers }))
.on('response', response => resolve([201, 301, 302, 303, 307, 308].includes(response.statusCode) ? request(method, parse(url).resolve(response.headers.location), headers, body) : response))
.on('error', error => reject(error)).end(body)
})
const json = response =>
new Promise((resolve, reject) => {
const chunks = []
response
.on('data', chunk => chunks.push(chunk))
.on('end', () => resolve(Buffer.concat(chunks)))
.on('error', error => reject(error))
})
.then(body =>
JSON.parse(body.toString().replace(/([^\\]"\s*:\s*)(\d{16,})(\s*[}|,])/g, '$1"$2"$3'))
)
const apiRequest = (path, data, load = true) => {
const method = 'POST'
const url = `${runtime.preferenceReader.get('API.SSL') ? 'https:': 'http:'}//music.163.com/api/linux/forward`
const headers = {
'User-Agent': 'Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/60.0.3112.90 Safari/537.36',
'Content-Type': 'application/x-www-form-urlencoded',
'Referer': parse(url).resolve('/'),
'X-Real-IP': '118.88.88.88',
'Cookie': ['os=linux', user.cookie].join('; ')
}
data = querystring.stringify(encrypt({ method: method, url: parse(url).resolve(`/api/${path}`), params: data }))
return request(method, url, headers, data).then(load ? json : response => response)
}
const api = {
request,
user: {
detail: id => apiRequest(`v1/user/detail/${id || user.id}`, {}),
artist: () => apiRequest(`artist/sublist`, { limit: 1000, offset: 0 }),
album: () => apiRequest(`album/sublist`, { limit: 1000, offset: 0 }),
djradio: () => apiRequest('djradio/get/subed', { limit: 1000, offset: 0 }),
playlist: (id, compact) => apiRequest('user/playlist', { uid: id || user.id, limit: compact ? 1 : 100000 }),
record: id => Promise.all([0, 1].map(type => apiRequest('v1/play/record', { uid: id || user.id, type }))).then(data => Object.assign(data[0], data[1])),
likes: () => apiRequest('song/like/get', {}),
account: id => id ? user.id === id : user.id,
favorite: id => id ? user.favor === id : user.favor
},
artist: {
song: id => apiRequest(`v1/artist/${id}`, { top: 50 }),
album: id =>
Promise.all([apiRequest(`artist/albums/${id}`, { limit: 1000, offset: 0 }), apiRequest(`artist/detail/v4`, { id }), apiRequest(`artist/detail/dynamic`, { id })])
.then(data => (data[0].artist.fansNum = data[1].fansNum, data[0].artist.followed = data[2].followed, data[0])),
subscribe: (id, action) => apiRequest(`artist/${action ? 'sub' : 'unsub'}`, { artistId: id, artistIds: [id] })
},
djradio: {
program: id => apiRequest('dj/program/byradio', { radioId: id, limit: 500 })
.then(data =>
Promise.all(
Array.from(Array(Math.ceil(data.programs[0].radio.programCount / 500) - 1).keys()).map(group =>
apiRequest('dj/program/byradio', { radioId: id, limit: 500, offset: 500 * (group + 1) }))
)
.then(rest => rest.forEach(part => Array.prototype.push.apply(data.programs, part.programs)))
.then(() => data)
),
subscribe: (id, action) => apiRequest(`djradio/${action ? 'sub' : 'unsub'}`, { id })
},
program: {
detail: id => apiRequest('dj/program/detail', { id }),
listen: id => apiRequest('dj/program/listen', { id }),
url: id => api.program.detail(id).then(data => api.song.url(data.program.mainTrackId)), // cheating?
comment: id => apiRequest(`v1/resource/comments/A_DJ_1_${id}`, { rid: `A_DJ_1_${id}`, limit: 50, offset: 0 })
},
album: {
detail: id =>
Promise.all([apiRequest(`v1/album/${id}`, {}), apiRequest('album/detail/dynamic', { id })])
.then(data => (Object.assign(data[0].album.info, data[1]), data[0])),
subscribe: (id, action) => apiRequest(`album/${action ? 'sub' : 'unsub'}`, { id })
},
playlist: {
detail: id => apiRequest('v3/playlist/detail', { id, n: 1000 })
.then(data =>
Promise.all(
Array.from(Array(Math.ceil(data.playlist.trackCount / 1000) - 1).keys()).map(group =>
apiRequest('v3/song/detail', { c: JSON.stringify(data.playlist.trackIds.slice((group + 1) * 1000).slice(0, 1000).map(item => ({ id: item.id }))) })
))
.then(rest => rest.forEach(part => Array.prototype.push.apply(data.playlist.tracks, part.songs)))
.then(() => data)
),
highquality: () => apiRequest('playlist/highquality/list', { cat: '全部', limit: 50 }),
hot: () => apiRequest('playlist/list', { cat: '全部', limit: 50, offset: 0, order: 'hot' }),
subscribe: (id, action) => apiRequest(`playlist/${action ? 'subscribe' : 'unsubscribe'}`, { id }),
intelligence: (id, pid) => apiRequest('playmode/intelligence/list', { songId: id, startMusicId: id, playlistId: pid || user.favor, count: 1, type: 'fromPlayAll' })
},
toplist: () => apiRequest('toplist', {}),
song: {
detail: id => apiRequest('v3/song/detail', { c: JSON.stringify([{ id }]) }),
url: id => apiRequest('song/enhance/player/url', { ids: [id], br: 320000 }),
lyric: id => apiRequest('song/lyric', { id, lv: -1, tv: -1, cp: false }),
like: id => apiRequest('song/like', { trackId: id, like: true, time: 0, userid: 0 }),
dislike: id => apiRequest('song/like', { trackId: id, like: false, time: 0, userid: 0 }),
collect: (id, pid) => apiRequest('playlist/manipulate/tracks', { trackIds: [id], pid: pid, op: 'add' }),
comment: id => apiRequest(`v1/resource/comments/R_SO_4_${id}`, { rid: id, limit: 50, offset: 0 }),
log: data => apiRequest('feedback/weblog', { logs: JSON.stringify([{ action: 'play', json: data }]) }),
trash: (id, time = 0) => apiRequest('v1/radio/trash/add', { alg: 'itembased', songId: `${id}`, time: `${time}` })
},
recommend: {
song: () => apiRequest('v1/discovery/recommend/songs', { limit: 30, offset: 0 }),
playlist: () => apiRequest('personalized/playlist', { limit: 20, offset: 0, n: 1000 }),
radio: () => apiRequest('v1/radio/get', {})
},
new: {
song: () => apiRequest('v1/discovery/new/songs', { areaId: 0, limit: 50, offset: 0 }),
album: () => apiRequest('album/new', { area: 'ALL', limit: 50, offset: 0 })
},
search: {
keyword: text => apiRequest('search/suggest/keyword', { s: text }),
suggest: text => apiRequest('search/suggest/web', { s: text }),
type: (text, type) => apiRequest('search/get', { s: text, type: type, limit: 20, offset: 0 }),
hot: () => apiRequest('search/hot', { type: 1111 })
},
login: (account, password) => {
const email = account.includes('@')
const area = '+1 +1242 +1246 +1264 +1268 +1345 +1441 +1473 +1664 +1670 +1671 +1758 +1784 +1787 +1868 +1876 +1890 +20 +212 +213 +216 +218 +220 +221 +223 +224 +225 +226 +227 +228 +229 +230 +231 +232 +233 +234 +235 +236 +237 +239 +241 +242 +243 +244 +247 +248 +249 +251 +252 +253 +254 +255 +256 +257 +258 +260 +261 +262 +263 +264 +265 +266 +267 +268 +27 +30 +31 +32 +33 +34 +350 +351 +352 +353 +354 +355 +356 +357 +358 +359 +36 +370 +371 +372 +373 +374 +375 +376 +377 +378 +380 +381 +386 +39 +40 +41 +420 +421 +423 +43 +44 +45 +46 +47 +48 +49 +501 +502 +503 +504 +505 +506 +507 +509 +51 +52 +53 +54 +55 +56 +57 +58 +591 +592 +593 +594 +595 +596 +597 +598 +599 +60 +61 +62 +63 +64 +65 +66 +673 +674 +675 +676 +677 +679 +682 +684 +685 +689 +7 +81 +82 +84 +850 +852 +853 +855 +856 +86 +880 +886 +90 +91 +92 +93 +94 +95 +960 +961 +962 +963 +964 +965 +966 +967 +968 +971 +972 +973 +974 +976 +977 +98 +992 +993 +994 +995 +996 +998'.split(/\s/).find(code => account.startsWith(code))
const path = email ? 'login' : 'login/cellphone'
const data = Object.assign(
{ password: crypto.createHash('md5').update(password).digest('hex'), rememberLogin: 'true' },
email ? { username: account } : { phone: account.replace(area, ''), countrycode: (area || '').replace('+', '') }
)
return apiRequest(path, data, false)
.then(response =>
response.headers['set-cookie'] ? api.refresh(response.headers['set-cookie'].map(cookie => cookie.replace(/;.*/, '')).join('; ')) : json(response).then(data => Promise.reject(data))
)
},
logout: () => {
user = {}
return sync()
},
sign: () => apiRequest('point/dailyTask', { type: 1 }),
refresh: cookie => {
user = cookie ? { cookie } : (runtime.globalStorage.get('user') || {})
return apiRequest('user/info', {}).then(data => data.code === 200 ? user.id = data.userPoint.userId : user = {}).then(sync)
}
}
const sync = () => {
runtime.globalStorage.set('user', user)
runtime.stateManager.set('logged', !!user.id)
return user.id ? Promise.all([api.user.detail(), api.user.playlist(null, true)]).then(data => (runtime.stateManager.set('signed', !!data[0].pcSign), user.favor = data[1].playlist[0].id, data[0])) : Promise.resolve()
}
module.exports = api
const runtime = require('./runtime.js')
1
https://gitee.com/mirrors/vsc-netease-music.git
git@gitee.com:mirrors/vsc-netease-music.git
mirrors
vsc-netease-music
vsc-netease-music
master

搜索帮助