前言

突然听到了一首周董的歌,想在 5sing 上下载一首歌
然而,我好几年前的账号密码都忘得一干二净了,又不想为了下载一首歌重新注册,于是就只能通过官方的API来下载了
为了一劳永逸,我决定研究一下油猴脚本,用于实现在歌曲页面里能够绕过登录并且一键下载的功能

什么是油猴脚本?

油猴,英文(Tampermonkey),又名Greasemonkey,是一款非常流行的浏览器扩展,ChromeFirefox均支持,其原理有点像Windows编程里的hook,它的可自定义性非常高,它可以运行由广大社区编写的扩展脚本,来实现各式各样的功能,常见的去广告、修改样式文件、甚至是下载视频,能给用户带来无限可能

新建一个油猴脚本,默认内容如下,无需过多配置,填写好UserScript就可以开始编写了

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// ==UserScript==
// @name New Userscript
// @namespace http://tampermonkey.net/
// @version 0.1
// @description try to take over the world!
// @author You
// @match http://*/*
// @grant none
// ==/UserScript==

(function() {
'use strict';

// Your code here...
})();

流程思维导图

第一次接触写油猴,虽然没系统学过JS,但也还有一丢丢的JS基础,得亏天下编程语言万变不离其宗,凭着“语感”一边摸索一边“借鉴”,居然也完成了

大致的流程思维图,具体看代码

获取歌曲ID及类型

ID和歌曲类型可以在歌曲页面网址上直接获得

例如:http://5sing.kugou.com/bz/88538.html
bz就是类型,表示此歌曲是伴奏歌曲,同理yc则是原唱歌曲88538就是歌曲ID

1
2
var id = window.location.href.split('/')[4].split(".")[0];
var type = window.location.href.split('/')[3].split(".")[0];

获取歌曲下载链接

向接口http://service.5sing.kugou.com/song/getsongurl?&songid=[id]&songtype=[bz]发送GET请求,其中的两个参数songid和songtype的值,上面已讲,不再赘述
成功后会返回一组Json,里边有各种音质、歌曲MD5等等信息,反序列化一下提取对应音质的下载链接即可

1
2
3
4
5
6
7
8
9
10
11
var url = 'http://service.5sing.kugou.com/song/getsongurl?&songid=' + id + '&songtype='+type;
GM_xmlhttpRequest({
method: "GET",
url: url,
onload: function (res) {
if (res.status == 200) {
var text = res.responseText;
var ret = jQuery.parseJSON(text)
}
}
}

添加下载事件

1
2
3
4
5
var oDivNode = document.getElementById("func_Down");
oDivNode.addEventListener("click",function(){
down_music(music_url,id+".mp3");
$('.new_login_bg').remove();
});

问题难点

由于我不知道的某个原因,对于视频、音频、文本等等文件直链访问并不会触发下载,而是直接在浏览器中预览,这就很烦,尝试了几个方法,貌似只能创建blob对象来下载

可能对前端大佬来说这不是什么难点,但对我这种半桶水的来说可能就难一点了

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
function down_music(url, name) {
let that = this
let oReq = new XMLHttpRequest();
oReq.open("GET", url, true);
oReq.responseType = "blob";
//oReq.withCredentials = true;//如果跨域
oReq.onload = function (oEvent) {
let content = oReq.response;
let elink = document.createElement('a');
elink.download = name;
elink.style.display = 'none';
let blob = new Blob([content])
elink.href = URL.createObjectURL(blob);
document.body.appendChild(elink);
elink.click();
document.body.removeChild(elink);
};
oReq.send();
}

最终效果

出现跨域提醒时,点击总是允许此域名即可,以后使用时就不会弹出了

下载使用演示