为微信小程序开发的网易云音乐api库

以前咱们已经开发过一款小程序适用的qq音乐api库 https://github.com/FisherWY/Q...,此次开发网易云音乐api库的缘由是qq音乐api库在小程序中iOS环境下没法使用小程序提供的背景音频播放器播放的问题

网易云的加密算法真的比其余几家api复杂太多了。。。完爆QQ和酷狗
想要直接用的话能够到Github直接取我封装好的api库。
Github地址https://github.com/JabinGP/Ne...javascript

依赖

本api库参考了Github上面开源的node库,由于咱们只想要查找音乐和播放音乐这两个功能,虽然Github那个库很方便,可是咱们不想为了两个接口特地去跑一个node.js服务。Github上的库java

  1. big-integer.js
    这里注意,不要使用最新版的,最新版的库再模拟器上运行没有问题,可是在真机调试的上传包阶段会报错说没法识别big-integer.js,最后在个人尝试下,选用了一个老版本的库解决了这个问题。
  2. crypto-js

这个库是用来aes加密的,在node上面有一个原生的crypto,可是在小程序里咱们没有,因此我照着Github上的源码一点一点用这个库翻译过来的,还有Buffer在小程序里也没有,我使用这个库的方法代替了。node

获取api的原理

网上不少帖子讲的很清楚了,这里推荐几篇文章,我只作一个简单的总结,方便你们理解这个库。
网易云的加密算法大概使用了两个:git

  1. AES加密+BASE64编码
  2. RSA加密

加密大体流程:github

  1. api请求信息先被转成json字符串格式,而后再使用一个固定的密钥aes+base64编码加密,获得了第一个加密结果a
  2. 客户端从abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789+/随机生成一个新的16位密钥,而后用这个密钥去加密加密结果a,获得加密结果b

3.这样咱们的数据就被双重加密了,可是咱们要发给服务器去查询对应的数据,服务器知道第一个固定的密钥是多少,能够解开第一个加密结果,可是服务器可不知道咱们第二次加密用的是什么,因此服务器还须要获得咱们的第二个生成的随机加密密钥。算法

  1. 第二个随机加密密钥要是直接发给服务器好像就不太安全了,因此客户端对第二个随机加密密钥也进行了加密,使用的就是RSA加密,加密后获得的数据咱们称为c
  2. bc发送给服务器,服务器就会返回给咱们对应的结果了。

加密核心代码

这段代码传入对象后能够直接加密成符合网易云api加密的结果。json

// 生成随机数,size默认16
function createSecretKey(size) {
    const keys = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"
    let key = ""
    for (let i = 0; i < size; i++) {
        let pos = Math.random() * keys.length
        pos = Math.floor(pos)
        key = key + keys.charAt(pos)
    }
    return key
}


// aes加密方法
function aesEncrypt(word, secKey) {
    let key = CryptoJS.enc.Utf8.parse(secKey);  //十六位十六进制数做为密钥
    let iv = CryptoJS.enc.Utf8.parse(aes_mv);   //十六位十六进制数做为密钥偏移量
    let srcs = CryptoJS.enc.Utf8.parse(word);
    let encrypted = CryptoJS.AES.encrypt(srcs, key, { iv: iv, mode: CryptoJS.mode.CBC, padding: CryptoJS.pad.Pkcs7 });
    let res = encrypted.toString();
    console.log(res);
    return res;
}

// 填充方法
function zfill(str, size) {
    while (str.length < size) str = "0" + str
    return str
}

// rsa加密方法
function rsaEncrypt(text, pubKey, modulus) {
    const _text = text.split('').reverse().join('')
    const biText = bigInt(CryptoJS.enc.Utf8.parse(_text).toString(), 16),
        biEx = bigInt(pubKey, 16),
        biMod = bigInt(modulus, 16),
        biRet = biText.modPow(biEx, biMod)
    return zfill(biRet.toString(16), 256)
}


// 加密总入口
function Encrypt(obj) {
    const text = JSON.stringify(obj)
    const secKey = createSecretKey(16)
    const encText = aesEncrypt(aesEncrypt(text, nonce), secKey)
    const encSecKey = rsaEncrypt(secKey, pubKey, modulus)
    return {
        params: encText,
        encSecKey: encSecKey
    }
}

封装好的Api库

首先到Github下载个人Api库https://github.com/JabinGP/Ne...
下载完成后,这个库应该是能够直接导入微信小程序开发工具运行的,可是有几个注意事项小程序

  1. 这个库是用TypeScript写的,可是最后编译成了JS运行,可是编译后JS代码可读性不好,因此我保留了TypeScript源文件,就在NetEaseCloudMusicApi/ts_src里面,应用库的时候不须要使用到
  2. 关闭小程序开发工具的详情页的ES6转ES5,可使用await处理异步请求(由于库是用Promise写的,起码要能用Promise,实例代码使用的是await/async
  3. await关键字只能在async修饰过的函数体内部使用,不懂的能够查一下await和async的用法。
  4. NetEaseCloudMusicApi/Libary文件夹里面包含了项目依赖的js文件,应用的时候最好整个NetEaseCloudMusicApi文件夹复制到项目里面使用。
  5. 测试的时候能够勾选不校验合法域名。

1

开始使用以前的准备

  1. 找到NetEaseCloudMusicApi这个文件夹,里面应该包括Libary、src、ts_src三个文件夹,Libary是我引用的开源库,ts_src中是TypeScript源文件,src是ts_src编译后产生的JavaScript文件夹,也就是说不考虑读ts源文件的话,能够把ts_src删了,可是17.4 KB 的大小对应用包体积应该没有什么影响吧,留着也行。
  2. 在要使用到的库中以下引用
const {MusicManager} = require("../../NetEaseCloudMusicApi/src/MusicManager");

注意要用花括号吧MusicManager括起来,这一句能够须要变化的地方只有
../../NetEaseCloudMusicApi/src/MusicManager中的../../,后面的路径都表明了NetEaseCloudMusicApi文件夹和NetEaseCloudMusicApi里面文件的路径,由于个人库就是这样的结构,因此不须要改变,../../就要根据你项目中实际结构来改变了。微信小程序

MusicManager

该类有如下方法:该类提供了全部获取其余对象的方法,能够经过该类获取其余须要的对象而不是newapi

  • getMusicSearchHelper()

须要参数:{keyword:"搜索歌曲关键词",limit:数字}
返回:MusicSearchHelper搜索器

  • getMusicUrlHelper()
    须要参数:musicId(数字类型的歌曲id)

返回:MusicUrlHelperUrl获取器

  • getUserSearchHelper()

须要参数:{userName:"搜索用户的用户名关键词",limit:数字}
返回:UserSearchHelper用户查询器

  • getUserListHelper()

须要参数:userId(数字类型的用户id值)
返回:UserListHelper用户列表查询器

  • getUserListDetailHelper()

须要参数:listId(数字类型的列表id)
返回:UserListDetailHelper用户列表详情信息获取器

MusicSearchHelper

用于搜索音乐
可用方法:

  • getSearchResult()---获取数据(默认第一页)
  • nextPage()--- 下一页
  • previousPage()---上一页
  • getCurrentPage()---查看当前页数的

执行完切换页数后须要再次调用getSearchResult方法查看新的查询结果。

MusicUrlHelper

用于将搜索音乐结果中的id转换为url播放连接
可用方法:

  • getUrlResult() ---获取url播放连接

UserSearchHelper

用于根据用户名关键字搜索用户
可用方法:

  • getSearchResult()---获取搜索结果

UserListHelper

用于获取用户id后根据id获取用户歌单信息
可用方法:

  • getAllLists()---获取用户全部歌单
  • getILikeList()---获取用户的我喜欢歌单

UserListDetailHelper

用于获取歌单id后获取歌单内歌曲列表
可用方法:

  • getDeatil()---获取歌单内列表

搜索歌曲

  1. 经过MusicManager获取一个MusicSearchHelper搜索器
  2. MusicSearchHelper的方法:

    • getSearchResult()---获取数据(默认第一页)
    • nextPage()--- 下一页
    • previousPage()---上一页
    • getCurrentPage()---查看当前页数的
  3. 执行完切换页数后须要再次调用getSearchResult方法查看新的查询结果。

代码实例

const {MusicManager} = require("../../NetEaseCloudMusicApi/src/MusicManager");
async function test(){
     // 搜索歌曲
    let musicSearchHelper = MusicManager.getMusicSearchHelper({ keyword: "one more time one more chance", limit: 10 });
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    musicSearchHelper.nextPage();
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    musicSearchHelper.previousPage();
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    console.log(musicSearchHelper);
} 
test();

2

经过搜索歌曲的结果获取音乐Url

有了搜索结果,咱们还须要url才能播放资源

  1. 经过MusicManager获取一个MusicUrlHelperUrl获取器
  2. 经过MusicUrlHelpergetUrlResult方法获取url
  3. 须要注意的是,因为网易云接口时常返回空回复,因此这里我经过20之内的重复次请求直到有结果就中止,若是20次之后仍是没有结果(据我测试20次之内都请求到结果了),也就是返回一个空的字符串"",须要使用者本身从新调用一次urlHelpergetUrlResult方法(2019.04.27)如今不会返回空值了,返回空值发现问题出在使用微信请求时自做聪明将json转成了a=xxxx&b=xxx的格式,致使微信不能正常转换请求数据,如今每次请求都能获取结果。

代码实例

const {MusicManager} = require("../../NetEaseCloudMusicApi/src/MusicManager");

async function test(){
    // 搜索歌曲
    let musicSearchHelper = MusicManager.getMusicSearchHelper({ keyword: "one more time one more chance", limit: 10 });
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    musicSearchHelper.nextPage();
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    musicSearchHelper.previousPage();
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    console.log(musicSearchHelper);

    // 获取歌曲url
    let songs = await musicSearchHelper.getSearchResult();
    let musicId  = songs[0].id;
    let musicUrlHelper = MusicManager.getMusicUrlHelper(musicId);
    console.log(`歌曲的ID是:${musicId}`);
    let url = await musicUrlHelper.getUrlResult();
    console.log(`歌曲的url连接是:${url}`);
} 
test();

3

4.26更新

新增搜索用户以及用户歌单获取接口

搜索用户

  1. 经过MusicManager获取一个UserSearchHelper用户查询器
  2. 经过UserSearchHelpergetSearchResult()方法获取搜索结果
async function test(){
  // 搜索用户
  let userSearchHelper = MusicManager.getUserSearchHelper({ userName: "JabinGP", limit: 20 });
  let users = await userSearchHelper.getSearchResult();
  console.log(users);
}

获取用户歌单

  1. 经过MusicManager获取一个UserListHelper用户查询器
  2. 经过UserListHelper

    • getILikeList() ---获取我喜欢歌单,返回一个列表对象
    • getAllLists() ---获取全部歌单,返回一个列表对象的数组
async function test(){
  // 搜索用户
  let userSearchHelper = MusicManager.getUserSearchHelper({ userName: "JabinGP", limit: 20 });
  let users = await userSearchHelper.getSearchResult();
  console.log(users);
  
  // 获取我喜欢歌单
  let userListHelper = MusicManager.getUserListHelper(users[0].userId);
  let iLikeList = await userListHelper.getILikeList()
  console.log(iLikeList);
 }

经过歌单里的Id获取歌曲url

与前面一致,再也不赘述

完整实例

完整实例代码在项目page下的index.js中,运行项目就会自动执行输出结果。

async function test(){
    // 搜索歌曲
    let musicSearchHelper = MusicManager.getMusicSearchHelper({ keyword: "one more time one more chance", limit: 10 });
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    musicSearchHelper.nextPage();
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    musicSearchHelper.previousPage();
    console.log(`如今是第${musicSearchHelper.getCurrentPage()}页`);
    console.log(await musicSearchHelper.getSearchResult());
    console.log(musicSearchHelper);

    // 获取歌曲url
    let songs = await musicSearchHelper.getSearchResult();
    let musicId  = songs[0].id;
    let musicUrlHelper = MusicManager.getMusicUrlHelper(musicId);
    console.log(`歌曲的ID是:${musicId}`);
    let url = await musicUrlHelper.getUrlResult();
    console.log(`歌曲的url连接是:${url}`);

    // 搜索用户
    let userSearchHelper = MusicManager.getUserSearchHelper({ userName: "JabinGP", limit: 20 });
    let users = await userSearchHelper.getSearchResult();
    console.log(users);

    // 获取用户歌单
    let userListHelper = MusicManager.getUserListHelper(users[0].userId);
    let iLikeList = await userListHelper.getILikeList()
    console.log(iLikeList);
    
    // 获取我喜欢歌单
    let userListDeatilHelper = MusicManager.getUserListDetailHelper(iLikeList.id);
    let listDetail = await userListDeatilHelper.getDeatil();
    console.log(listDetail);
    let timer=0;
    for(let song of listDetail.tracks){
      musicUrlHelper.musicId=song.id;
      console.log(`歌曲的ID是:${musicUrlHelper.musicId}`);
      let url2 = await musicUrlHelper.getUrlResult();
      console.log(`歌曲的url连接是:${url2}`);
      if(timer++>20)break;
    }
  }   
test();

结尾

2019 4.25目前就只有这两个接口,由于咱们项目就只须要这两个接口,若是有须要更多接口的,能够在下方评论,以上示例代码都在Github项目上的index.js中,也就是你把文件导入微信开发者工具后,取消勾选一下详情的ES6转ES5以及取消勾选合法域名检验,就能够在控制台看到以上示例代码的输出了

2019 4.26更新搜索用户和获取用户歌单以及获取歌单详细三个接口。

若是对你有帮助,点个Star吧~

相关文章
相关标签/搜索