Python下载会员歌曲

Python下载会员歌曲

@(博客)[QQ|会员音乐|下载]

018.8.8

1. 由于本人是法盲,所以是否涉及侵权QQ音乐不知;如若侵权,相告即删
2. 相关代码仅作参考学习,不用于商业目的

前言

本来想先编个故事再进入正文的,这符合我的风格。但由于要下载QQ音乐的VIP歌曲,代码方面不难,而是分析文件的过程有点绕。我已经觉得这个过程我会说不清楚,继而意兴阑珊,故事什么的就了无趣味了


目标

QQ音乐中VIP才能下载的歌曲

使用库

主要使用的库:
- requests 向服务器发起请求
- urllib 构建url地址
- re 提取需要的数据

分析

(1)文件A

首先我们来到QQ音乐的网页端,播放一首歌曲,这里就以【小半】为例
这里写图片描述

利用chrome的开发者工具,勾选Preserver log,并且选中Media,刷新页面
这里写图片描述

刷新页面
这里写图片描述

此时会发现有这么一个不知道什么的文件出现,暂且称之为文件A。右下角红色方框内是请求这个数据时带上的query参数

点进来之后会发现其实这就是我们需要的歌曲文件
这里写图片描述

所以现在的问题成了如何请求文件A。我们已经有了请求参数,也可以找到服务器的接口
这里写图片描述

根据反复测试,发现只有关键字vkey的值在发生变化,所以只要我们获取了动态变化的vkey值,拿到文件A就易如反掌了


(2)文件B

通过开发者工具,我找到了一个JS文件,暂且称之为文件B,它在歌曲文件之前被请求,并且其返还数据里面有vkey
这里写图片描述

我们也发现,需要请求这个文件,需要的query参数不可谓少
这里写图片描述

同样,在反复测试以后会发现,songmid的值会根据歌曲的不同而发生改变;filename的值是在songmid值的左边加上C400,右边加上.m4a

于是问题变成了如何获取songmid的值


(3)文件C

继续顺藤摸瓜前边的文件,在一个JS文件,暂且称之为文件C中找到了
这里写图片描述

仔细分析会发现,关键字list是包含了【小半】所在专辑《小梦大半》里面的全部歌曲,而还有个关键字singername是歌手名字,为了确保我们下载的歌曲是我们想要的歌手唱的,所以我用正则提取出来。针对list,我的方法是将整个专辑中所有歌曲的songmid以及歌曲的名字全部提取出来,然后再从中确认我们需要的songmid

# 提取歌手名字
SINGER = re.search(r'"singername":"(.+?)"', data).group(1)
# 提取专辑中所有的songmid,以及对应的歌曲名字

results = re.findall(r'"songmid":"(\w+?)","songname":"(.+?)"', data)

# 我们知道,通过findall()方法得到的结果是由元组组成的列表(如:[(songmid1, songname1), (songmid2, songname2),...]),所以对其遍历,当歌曲名字SONGNAME在这个元组里边时,返回对应的songmid
if results:
    for result in results:
        if SONGNAME in result:
            return result[0]
    else:
        return None
else:
    return None

而如何获得这个文件呢?
这里写图片描述
可以看到,获取这个文件的关键点是albummid的值


(4)文件D

来到QQ音乐的搜索界面
这里写图片描述

当我们在搜索框中键入文字以后点击右边的搜索按钮,会发现浏览器接收到一个文件,我称之为文件D
这里写图片描述

文件D中的list里边就包含了我们搜索出来的结果,因为存在歌曲同名啊,翻唱之类的,所以一般list里边都包含多个值,而一般情况下,比较火的歌,且在QQ音乐中有版权的,都会存放在第一个(如果有其他目的,可自行在list的数据中进行取舍),这里我就只取出第一个
这里写图片描述

# 提取albummid的值
result = re.search(r'"mid":"(\w+?)"', data)
if result:
    return result.group(1)
else:
    return None

文件D的请求方式就比较简单了
这里写图片描述

尽管需要的参数很多,但最重要的就是w了,它对应的是歌曲名字

(5)注意事项

此时再回去看看请求文件A的接口,其实有一部分就是文件B中的关键字filename所对应的值,所以我们对这个接口要动态改变

# 构建下载歌曲的query参数
PARAMS_FOR_VIPSONG["vkey"] = vkey
url = parse.urljoin(URL_FOR_VIPSONG, "C400"+songmid+".m4a?")

最后

分析是从里到外,找到的文件是A->B->C->D;而代码的执行顺序应该是从外到里,请求文件的顺序是D->C->B->A

以下是我代码的主要结构
这里写图片描述

为了更加友好,我另写了一个main.py的文件,来提示程序的用法
这里写图片描述

代码运行效果如下
这里写图片描述

完整代码已上传Github(有详细注释)

posted @ 2018-08-08 15:27  有关心情  阅读(721)  评论(1编辑  收藏  举报