本文利用Python2.7根据网易云音乐歌曲ID爬取了该歌曲的所有用户评论数据。以id是28875120的歌曲《小岁月太着急》为示例,通过Chrome的DevTools工具获取已加密评论数据,然后基于AES对称加密算法对已加密数据进行解密实现,最后使用Python成功实现了对用户评论数据的抓取与保存。本文部分算法实现参考于知乎回答,全部代码可在Github项目地址查看。

利用DevTools工具获取加密数据

进入 http://music.163.com/#/song?id=28875120 页面,打开Chrome的DevTools工具选择Network并重载页面,找到与评论数据相关的请求即name为R_SO_4_28875120?csrf_token=90e04572eb42b040167323ec2fcdd79f的POST请求,如下图所示:

查看该请求信息,可知Request Headers参数如下:

其中的POST Request URL完整地址为 :
http://music.163.com/weapi/v1/resource/comments/R_SO_4_28875120?csrf_token=90e04572eb42b040167323ec2fcdd79f
并且,该Form Data含有params和encSecKey两个参数,显然,这两个参数是经过js加密后的。服务器返回的和评论相关的数据为json格式的,里面含有非常丰富的信息(比如有关评论者的信息,评论日期,点赞数,评论内容等等),同时,通过查看第一张图可知该请求的Initiator为core.js,因此需要通过查看该js源码来分析两个参数的值。

根据源代码分析加密数据

AES加密算法分析

AES(Advanced Encryption Standard)对称加密算法是一种高级数据加密标准,可有效抵制针对DES的攻击算法。特点:密钥建立时间短、灵敏性好、内存需求低、安全性高。

将core.js文件clone到本地并对其格式化,发现params和encSecKey两个参数同时出现在以下代码中:

params的值为byw6q.encText,encSecKey的值为byw6q.encSecKey,而byw6q是由window.asrsea()这个函数得到,定位这个函数发现他其实是d(d, e, f, g)函数,该d函数代码如下图所示:

由代码可知,d函数对json格式的明文数据params调用了两次AES对称加密(即b函数):第一次对params的值d加密,key是第四个参数g,第二次对第一次加密结果h.encText值进行加密并更新encText值,key是长度为16的随机字符串i(不妨设其值为16个F)。而且,在b加密函数中,密钥偏移量iv值为”0102030405060708”,密码工作模式model值为CBC,即密文链接分组密码工作模式(明文加密前需要先和前面的密文进行异或运算,也就是相同的明文加密后产生不同的密文)。而d函数的第一个参数JSON.stringify(j5o)的值由j5o决定,在不同的请求下随着j5o的变化会有不同的值,而后面的三个参数均为定值。
获取 params值的代码如下:

# 第二个参数
second_param = "010001"
# 第三个参数
third_param = "00e0b509f6259df8642dbc35662901477df22677ec152b5ff68ace615bb7b725152b3ab17a876aea8a5aa76d2e417629ec4ee341f56135fccf695280104e0312ecbda92557c93870114af6c9d05c4f7f0c3685b7a46bee255932575cce10b424d813cfe4875d3e82047b97ddef52741d546b8e289dc6935b3ece0462db0a22b8e7"
# 第四个参数
forth_param = "0CoJUm6Qyw8W8jud"def get_params(page): # page为传入页数iv = "0102030405060708"first_key = forth_paramsecond_key = 16 * 'F'if(page == 1): # offset的取值为:(评论页数-1)*20,total第一页为true,其余页为falsefirst_param = '{rid:"", offset:"0", total:"true", limit:"20", csrf_token:""}'h_encText = AES_encrypt(first_param, first_key, iv)else:offset = str((page-1)*20)first_param = '{rid:"", offset:"%s", total:"%s", limit:"20", csrf_token:""}' %(offset,'false')h_encText = AES_encrypt(first_param, first_key, iv)h_encText = AES_encrypt(h_encText, second_key, iv)return h_encText

encSecKey的值由c(i, e, f)函数,其参数均为常数,因此无论歌曲id或评论页数如何变化,该值均不变。获取encSecKey值的代码如下:

def get_encSecKey():encSecKey = "257348aecb5e556c066de214e531faadd1c55d814f9be95fd06d6bff9f4c7a41f831f6394d5a3fd2e3881736d94a02ca919d952872e7d0a50ebfa1769a7a62d512f5f1ca21aec60bc3819a9c3ffca5eca9a0dba6d6f7249b06f5965ecfff3695b54e1c28f3f624750ed39e7de08fc8493242e26dbc4484a01c76f739e135637c"return encSecKey

AES解密算法实现

这里用到的AES算法为AES-128-CBC,密钥偏移量iv值为”0102030405060708”,分组密码填充方式为PKCS5Padding(即需对待加解密块按需补位),输出格式为base64。对称算法中加密与解密实现一致,代码如下:

def AES_encrypt(text, key, iv):pad = 16 - len(text) % 16text = text + pad * chr(pad)encryptor = AES.new(key, AES.MODE_CBC, iv)encrypt_text = encryptor.encrypt(text)encrypt_text = base64.b64encode(encrypt_text)return encrypt_text

使用Python抓取评论数据

抓取评论数据

获取json格式的评论数据的代码如下:

# 设置代理服务器
proxies= {'http:':'http://121.232.146.184','https:':'https://144.255.48.197'}def get_json(url, params, encSecKey):data = {"params": params,"encSecKey": encSecKey}response = requests.post(url, headers=headers, data=data,proxies = proxies)return response.content

抓取热门评论的代码如下:

def get_hot_comments(url):hot_comments_list = []hot_comments_list.append(u"用户ID 用户昵称 用户头像地址 评论时间 点赞总数 评论内容\n")params = get_params(1) # 第一页encSecKey = get_encSecKey()json_text = get_json(url,params,encSecKey)json_dict = json.loads(json_text)hot_comments = json_dict['hotComments'] # 热门评论print("共有%d条热门评论!" % len(hot_comments))for item in hot_comments:comment = item['content'] # 评论内容likedCount = item['likedCount'] # 点赞总数comment_time = item['time'] # 评论时间(时间戳)userID = item['user']['userID'] # 评论者idnickname = item['user']['nickname'] # 昵称avatarUrl = item['user']['avatarUrl'] # 头像地址comment_info = userID + " " + nickname + " " + avatarUrl + " " + comment_time + " " + likedCount + " " + comment + u"\n"hot_comments_list.append(comment_info)return hot_comments_list

抓取全部评论的代码如下:

def get_all_comments(url):all_comments_list = [] # 存放所有评论all_comments_list.append(u"用户ID 用户昵称 用户头像地址 评论时间 点赞总数 评论内容\n") # 头部信息params = get_params(1)encSecKey = get_encSecKey()json_text = get_json(url,params,encSecKey)json_dict = json.loads(json_text)comments_num = int(json_dict['total'])if(comments_num % 20 == 0):page = comments_num / 20else:page = int(comments_num / 20) + 1print("共有%d页评论!" % page)for i in range(page):  # 逐页抓取params = get_params(i+1)encSecKey = get_encSecKey()json_text = get_json(url,params,encSecKey)json_dict = json.loads(json_text)if i == 0:print("共有%d条评论!" % comments_num) # 全部评论总数for item in json_dict['comments']:comment = item['content'] # 评论内容likedCount = item['likedCount'] # 点赞总数comment_time = item['time'] # 评论时间(时间戳)userID = item['user']['userId'] # 评论者idnickname = item['user']['nickname'] # 昵称avatarUrl = item['user']['avatarUrl'] # 头像地址comment_info = unicode(userID) + u" " + nickname + u" " + avatarUrl + u" " + unicode(comment_time) + u" " + unicode(likedCount) + u" " + comment + u"\n"all_comments_list.append(comment_info)print("第%d页抓取完毕!" % (i+1))return all_comments_list

写入文本文件

将已获得的评论数据写入文本文件,代码如下:

def save_to_file(list,filename):with codecs.open(filename,'a',encoding='utf-8') as f:f.writelines(list)print("写入文件成功!")

利用Python获得数据结果

获取Headers数据

获取headers代码如下:

headers = {'Accept':"*/*",'Accept-Encoding':"gzip, deflate",'Accept-Language':"zh-CN,zh;q=0.8",'Connection':"keep-alive",'Content-Length':"416",'Content-Type':"application/x-www-form-urlencoded",'Cookie':"_ntes_nnid=61528ed156a887c721f86bb28fb76864,1498012702495; ...; __csrf=880488a01f19e0b9f25a81842477c87b",'Host':"music.163.com",'Origin':"http://music.163.com",'Referer':"http://music.163.com/",'User-Agent':"Mozilla/5.0 (Macintosh; Intel Mac OS X 10_12_5) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/58.0.3029.110 Safari/537.36"}

数据抓取实现

安装pycrypto

#pip uninstall Crypto
#pip uninstall pycrypto
pip install pycrypto

抓取数据

#!/usr/bin/env python2.7
# -*- coding: utf-8 -*-from Crypto.Cipher import AES
import base64
import requests
import json
import codecs
import timeif __name__ == "__main__":start_time = time.time() # 开始时间url = "http://music.163.com/weapi/v1/resource/comments/R_SO_4_28875120?csrf_token=90e04572eb42b040167323ec2fcdd79f"filename = u"小岁月太着急.txt"all_comments_list = get_all_comments(url)save_to_file(all_comments_list,filename)end_time = time.time() #结束时间print("程序耗时%f秒." % (end_time - start_time))

数据结果输出

代码实现

ZerodeMBP:~ zero$ cd /Users/zero/Documents/163mc_spider
ZerodeMBP:163mc_spider zero$ python 163mc_spider.py
共有22页评论!
共有434条评论!
第1页抓取完毕!
第2页抓取完毕!
...
第22页抓取完毕!
写入文件成功!
程序耗时3.193853秒.

评论数据查看

全部代码可在Github项目地址查看。

利用Python爬取基于AES对称加密算法的网易云音乐用户评论数据相关推荐

  1. python画哪吒_用Python爬取暑期档大火的《哪吒》,20W+评论数据,我们分析一波...

    原标题:用Python爬取暑期档大火的<哪吒>,20W+评论数据,我们分析一波 说起这个暑期档的动画片,非<哪吒之魔童降世>莫属了! 上映第 1 天:89分钟,中国动画最快破 ...

  2. 抓取网易云音乐用户评论

    段子手与神编剧共存,矫情文艺与励志故事齐飞 用户热评是网易云音乐除智能推荐外的第二法宝 语料库是所有自然语言处理模型的知识基础,甚至决定了模型的适应范围,毕竟 Garbage in, garbage ...

  3. Python 爬取网易云音乐所有评论

    题图:by cfunk44 from Instagram 在使用 Ajax 技术加载数据的网站中, JavaScript 发起的 HTTP 请求通常需要带上参数,而且参数的值都是经过加密的.如果我们想 ...

  4. python 怎么爬桌软件数据_如何利用Python爬取并分析红岭创投的数据?

    第一步:爬取数据 通过 selenium + Firefox 的无头模式将需要的数据爬取出来,代码实现不在赘述,详细步骤可查看我的上一篇图文(如何利用Python爬取网易新闻), 由于 seleniu ...

  5. python 翻译库本地库_利用python爬取并翻译GEO数据库

    原标题:利用python爬取并翻译GEO数据库 GEO数据库是NCBI创建并维护的基因表达数据库,始于2000年,收录了世界各国研究机构提交的高通量基因表达数据,现芯片集数据量高达12万以上.想要从这 ...

  6. 利用python爬取东方财富网股吧评论并进行情感分析(一)

    利用python爬取东方财富网股吧评论(一) python-东方财富网贴吧文本数据爬取 分享一下写论文时爬数据用到的代码,有什么问题或者改善的建议的话小伙伴们一起评论区讨论.涉及内容在前人的研究基础之 ...

  7. python爬取股票信息_利用Python爬取网易上证所有股票数据(代码

    利用Python爬取网易上证所有股票数据(代码 发布时间:2018-04-14 17:30, 浏览次数:1261 , 标签: Python import urllib.request import r ...

  8. 利用python爬取豆瓣电影top250

    利用python爬取豆瓣电影top250: 注:本内容只是作为个人学习记录 1.业务分析 进入网页https://movie.douban.com/top250 可以看见每部电影都呈现在眼前,点击电影 ...

  9. 利用python爬取丁香医生上新型肺炎数据,并下载到本地,附带经纬度信息

    新增:国外疫情网站介绍 已更新:爬取国外疫情数据 已更新:新型肺炎历史数据下载 2020年3月27日补充: 制作了一个全球肺炎数据查询下载网站,效果如下: 访问地址:http://119.3.227. ...

最新文章

  1. 机器学习入门一定要避开的3个陷阱
  2. UVa11770 - Lighting Away(排序+DFS)
  3. 面试中经常出现的算法1(整理)
  4. 03_linux 常用命令(上)
  5. CentOS7安装VPP(FD.io)
  6. leetcode题库:5.最长回文子串Longest Palindrome string
  7. 12月数据库榜单,整体排名稳定如昨,Oracle 分数接连下降
  8. linux的shell键盘输入,linux shell获取键盘输入
  9. 关于AOP(面向方面的编程)
  10. ztree 后台异步加载_ztree 异步加载示例
  11. Visio绘制电路图
  12. 代码智能:问题与解法
  13. 数据结构c语言课程设计报告,数据结构c语言课程设计报告.doc
  14. spark学习之资源调度
  15. Fedora 30 将默认启用 DNF-best 模式
  16. OpenAI 开源语音识别模型 Whisper 初体验
  17. mysql中的left_MySQL left()函数
  18. [python]变量作用域问题
  19. java lint_Java代码规范与质量检测插件SonarLint
  20. Java初级程序员与ChatGPT(文心一言)使用感受

热门文章

  1. 无涯教程- Java 14 – Record类型介绍
  2. resteasy 上传表单文件名乱码
  3. 基于Lire库搜索相似图片
  4. 与腾讯、网易逐梦云游戏,三七互娱还少张王牌
  5. Unity 游戏Demo制作——地下城 一
  6. 彩色图像 psnr matlab,在matlab中的PSNR图像
  7. 笔记本上无计算机,笔记本电脑没声音怎么回事【解决措施】
  8. SHD-NMF+MV-Co-VH
  9. 8.Dubbo 原理
  10. 【CentOS7 Samba服务器配置】