酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)
酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)
2024-06-30 03:55:13  作者:命样顾惜著  网址:https://m.xinb2b.cn/tech/oox106687.html
前言

写这篇博客的初衷是加深自己对网络请求发送和响应的理解,仅供学习使用,请勿用于非法用途!文明爬虫,从我做起。下面进入正题。

获取歌曲信息列表

在酷我的搜索框中输入关键词 aiko,回车之后可以看到所有和 aiko 相关的歌曲。打开开发者模式,在网络面板下按下 ctrl f,搜索 二人,可以找到响应结果中包含 二人 的请求,这个请求就是用来获取歌曲信息列表的。

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(1)

请求参数分析

请求的具体格式如下图所示,可以看到请求路径为 http://www.kuwo.cn/api/www/search/searchMusicBykeyWord,请求参数包括:

key: 搜索关键词,此处为 aikopn: 页码,page number 的缩写,此处为 1rn: 每页条目数,应该是 row number 的缩写,默认为 30httpsStatus:https 的状态?感觉没啥大用,看了源代码里面是直接写死 t.url = t.url "?reqId=".concat(n, "&httpsStatus=1")reqId:请求标识,刷新页面之后值会发生改变,不知道有啥用,待会儿模拟请求的时候试着不带上他会怎么样

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(2)

打开 Apifox(当然 postman 也行),新建一个接口,把请求路径和参数设置为下图所示的样子,为了让响应结果简短点,这里把每页的条目数设置为 1 而非默认的 30:

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(3)

在没有设置额外请求头的情况下发个请求试试,发现 403 Forbidden 了,emmmmm,应该是防盗链所致:

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(4)

可以看到浏览器发出的请求的请求头中有设置 Referer 字段,把它加上,应该不会再报错了吧:

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(5)

这次状态码为 200,但是没有收到任何数据,success 为 false 说明请求失败了,message 指明了失败原因是缺少 csrf token。问题不大,接着把浏览器发出的请求中的 csrf 加到 Apifox 请求头中,再发请求,还是报错 CSRF token Invalid!。算了,还是老老实实把 Cookie 也加上吧,但也不是全部加上,只加 kw_token=CCISYM2HV96 部分,因为 Cookie 里面只有这个字段和 token 有关系且它的值和 csrf 相同。

在源代码面板按下 ctrl shift f,搜索一下 csrf,可以看到 csrf 本来就是来自 Object(h.b)("kw_token"),这个函数用来取出 document.cookie 中的 kw_token 字段值。至于 Cookie 中的 kw_token 怎么计算得到的,那就是服务器的事情了,咱们只管 CV 操作即可。

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(6)

准备好参数和请求头,重新发送请求,可以得到想要的数据。如果去掉 reqId 参数,也可以拿到数据,但是会有略微的不同,这里就不贴出来了:

复制{ "code": 200, "curTime": 1649482287185, "data": { "total": "741", "list": [ { "musicrid": "MUSIC_11690555", "barrage": "0", "ad_type": "", "artist": "aiko", "mvpayinfo": {"play": 0,"vid": 8530326,"down": 0 }, "nationid": "0", "pic": "http://img4.kuwo.cn/star/starheads/500/24/88/4146545084.jpg", "isstar": 0, "rid": 11690555, "duration": 362, "score100": "42", "ad_subtype": "0", "content_type": "0", "track": 1, "hasLossless": true, "hasmv": 1, "releaseDate": "1970-01-01", "album": "", "albumid": 0, "pay": "16515324", "artistid": 1907, "albumpic": "http://img4.kuwo.cn/star/starheads/500/24/88/4146545084.jpg", "originalsongtype": 0, "songTimeMinutes": "06:02", "isListenFee": false, "pic120": "http://img4.kuwo.cn/star/starheads/120/24/88/4146545084.jpg", "name": "恋をしたのは", "online": 1, "payInfo": {"play": "1100","nplay": "00111","overseas_nplay": "11111","local_encrypt": "1","limitfree": 0,"refrain_start": 89150,"feeType": { "song": "1", "vip": "1"},"down": "1111","ndown": "11111","download": "1111","cannotDownload": 0,"overseas_ndown": "11111","refrain_end": 126247,"cannotOnlinePlay": 0 }, "tme_musician_adtype": "0" } ] }, "msg": "success", "profileId": "site", "reqId": "4b55cf4b0171253c33ce1d71b999c42f", "tId": ""}

请求代码

响应结果的 data 字段中有很多东西,这里只提取需要的部分。在提取之前先来定义一下歌曲信息实体类,这样在其他函数中要一首歌曲的信息时只要把实体类的实例传入即可。

复制# coding:utf-8from copy import deepcopyfrom dataclasses import dataclassclass Entity: """ Entity abstract class """ def __setitem__(self, key, value): self.__dict__[key] = value def __getitem__(self, key): return self.__dict__[key] def get(self, key, default=None): return self.__dict__.get(key, default) def copy(self): return deepcopy(self)@dataclassclass SongInfo(Entity): """ Song information """ file: str = None title: str = None singer: str = None album: str = None year: int = None genre: str = None duration: int = None track: int = None trackTotal: int = None disc: int = None discTotal: int = None createTime: int = None modifiedTime: int = None

上述代码显示定义了实体类的基类,并且重写了 __getitem__ 和 __setitem__ 魔法方法,这样我们可以像访问字典一样来访问实体类对象的属性。接着让歌曲信息实体类继承了实体类基类,并且使用 @dataclass 装饰器,这是 python 3.7 引入的新特性,使用它装饰之后的实体类无需实现构造函数、__str__等常用函数,python 会帮我们自动生成。

在发送请求的过程中可能会遇到各种异常,如果在代码里面写 try except 语句会显得很乱,这里同样可以用装饰器来解决这个问题。

复制# coding:utf-8from copy import deepcopydef ExceptionHandler(*default): """ decorator for exception handling Parameters ---------- *default: the default value returned when an exception occurs """ def outer(func): def inner(*args, **kwargs): try: return func(*args, **kwargs) except BaseException as e: print(e) value = deepcopy(default) if len(value) == 0:return None elif len(value) == 1:return value[0] else:return value return inner return outer

下面是发送获取歌曲信息请求的代码,使用 exception_handler 装饰了 getSongInfos 方法,这样发生异常时会打印异常信息并返回默认值:

复制# coding:utf-8import jsonfrom urllib import parsefrom typing import List, Tupleimport requestsclass KuWoMusicCrawler: """ Crawler of KuWo Music """ def __init__(self): super().__init__() self.headers = { 'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) ' 'AppleWebKit/537.36 (KHTML, like Gecko) Chrome/71.0.3578.98 Safari/537.36', 'Cookie': 'kw_token=C713RK6IJ8J', 'csrf': 'C713RK6IJ8J', 'Host': 'www.kuwo.cn', 'Referer': '' } @exceptionHandler([], 0) def getSongInfos(self, key_word: str, page_num=1, page_size=10) -> Tuple[List[SongInfo], int]: key_word = parse.quote(key_word) # configure request header headers = self.headers.copy() headers["Referer"] = 'http://www.kuwo.cn/search/list?key=' key_word # send request for song information url = f'http://www.kuwo.cn/api/www/search/searchMusicBykeyWord?key={key_word}&pn={page_num}&rn={page_size}&reqId=c06e0e50-fe7c-11eb-9998-47e7e13a7206' response = requests.get(url, headers=headers) response.raise_for_status() # parse the response data song_infos = [] data = json.loads(response.text)['data'] for info in data['list']: song_info = SongInfo() song_info['rid'] = info['rid'] song_info.title = info['name'] song_info.singer = info['artist'] song_info.album = info['album'] song_info.year = info['releaseDate'].split('-')[0] song_info.track = info['track'] song_info.trackTotal = info['track'] song_info.duration = info["duration"] song_info.genre = 'Pop' song_info['coverPath'] = info.get('albumpic', '') song_infos.append(song_info) return song_infos, int(data['total'])

获取歌曲下载链接免费歌曲

虽然我们实现了搜索歌曲的功能,但是没拿到每一首歌的播放地址,也就没办法把歌曲下载下来。我们先来播放一首不收费的歌曲试试。可以看到浏览器发送了一个获取播放链接的请求,路径为 http://www.kuwo.cn/api/v1/www/music/playUrl,有两个需要关注的参数:

mid:音乐 Id,此处的值为 941583,和页面 url 中的编号一致,由于我们是通过点击搜索结果页面中 二人 跳转过来的,而 二人 这条结果也是动态加载出来的,超链接中的 Id 肯定也来自于上一节中响应结果的某个字段。二人 是第四条记录,通过对比可以发现 data.list[3].rid 就是 mid;type:音乐类型?此处的值为 music,发送请求的时候也设置为 music 即可

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(7)

在 Apifox 中新建一个获取歌曲播放地址的请求,如下所示,发现可以成功拿到播放地址:

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(8)

付费歌曲

现在换一首歌,比如 aiko - 横颜,点击歌曲页面上的播放按钮时会弹出要求在客户端中付费收听的对话框。直接发送请求,响应结果会是下面这个样子,状态码为 403:

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(9)

其实酷我在 2021 年 9 月份的时候换过获取播放地址的接口,那时候的请求接口为 http://www.kuwo.cn/url,支持以下几个参数:

format: 在线音乐的格式,可以是 mp3type: 和现在的接口中的 type 参数一样,但是值为 convert_url3rid: 音乐 Id,和 mid 一样br: 在线音乐的比特率,越大则音质越高,可选的有 128kmp3、 192kmp3 和 320kmp3

这个接口不管是付费音乐还是免费音乐都可以用。如果将现在这个接口的 type 参数的值换成 convert_url3,请求结果如下所示,说明成功了:

酷我音乐网络爬虫(如何使用python爬取酷我在线音乐)(10)

请求代码

下面是获取在线音乐播放链接的代码,只需调用 downloadSong 函数并把爬取到的歌曲传入就能完成歌曲的下载:

复制@exceptionHandler('')def getSongUrl(self, song_info: SongInfo) -> str: # configure request header headers = self.headers.copy() headers.pop('Referer') headers.pop('csrf') # send request for play url url = f"http://www.kuwo.cn/api/v1/www/music/playUrl?mid={song_info['rid']}&type=convert_url3" response = requests.get(url, headers=headers) response.raise_for_status() play_url = json.loads(response.text)['data']['url'] return play_url@exceptionHandler('')def downloadSong(self, song_info: SongInfo, save_dir: str) -> str: # get play url url = self.getSongUrl(song_info) if not url: return '' # send request for binary data of audio headers = self.headers.copy() headers.pop('Referer') headers.pop('csrf') headers.pop('Host') response = requests.get(url, headers=headers) response.raise_for_status() # save audio file song_path = os.path.join( save_dir, f"{song_info.singer} - {song_info.title}.mp3") with open(song_path, 'wb') as f: f.write(data) return song

后记

除了获取歌曲的详细信息和播放地址外,我们还能拿到歌词、歌手信息等,方法是类似的,在我的 Groove 中提供了在线歌曲的功能,一部分接口就是来自酷我,还有一些来自酷狗和网易云,爬虫的代码在 app/common/crawler 目录下,喜欢的话可以给个 star 哦,以上~~

文章作者: 之一Yo

文章链接: https://www.cnblogs.com/zhiyiYo/p/16122664.html

  • 区分新旧两种革命的标志是(区分新旧两种革命的标志解释)
  • 2024-06-30区分新旧两种革命的标志解释区分新旧两种革命的标志革命领导权问题革命领导权问题是区别新旧民主革命的主要标志,也是革命成败的关键新民主主义革命的核心问题是无产阶级领导权问题中国革命的敌人是异常强大的中国要获得民族独立和人民解放,必。
  • 牛肉片保鲜方法(牛肉片保鲜储藏方法介绍)
  • 2024-06-30牛肉片保鲜储藏方法介绍放进冷藏室保鲜把买回来的牛肉片盖上保鲜膜直接放入冷藏室保鲜,这种方法只适用于牛肉片在1至2天内能够吃完时保鲜用食用盐腌制牛肉片保鲜把买回来的牛肉片,放入盆里,再把盐撒抹在牛肉片上,揉搓均匀,放入冷藏,。
  • 交通标志标线图解知识(5分钟读懂20种道路交通标志线)
  • 2024-06-305分钟读懂20种道路交通标志线​大家都知道交通标志线多种多样,作为一种引导交通的指示标线,如果我们能够认真清楚的识别领会它的意思的话,对于我们平时驾车出行是非常有用的1、白色虚线和实线​白色虚线和白色实线是道路上最常见的标线白色虚。
  • 巴萨登贝莱何时复出(巴萨联赛大名单)
  • 2024-06-30巴萨联赛大名单巴萨公布了与比利亚雷亚尔的联赛大名单,贝莱林伤愈复出,阿劳霍,克里斯滕森,孟菲斯继续伤停巴萨大名单:门将:特尔施特根,伊尼亚基-佩尼亚,阿尔纳乌-代纳斯;后卫:贝莱林,皮克,马科斯-阿隆索,阿尔巴,孔。
  • 心中的另一个自己歌词(歌曲心中的另一个自己歌词)
  • 2024-06-30歌曲心中的另一个自己歌词给词无法遗忘这里玫瑰的盛放一路上有你在我身旁世间的花都在勇敢的歌唱超越一切穿梭时空带来光亮啊~夜空中闪烁的星光是我们汇聚梦想的光芒出发吧最天使挥动的翅膀只有相信世界充满希望盛开吧这世界最美丽的花用生命。
  • 胡海泉个人专辑歌单(胡海泉做客酷狗晚安电台)
  • 2024-06-30胡海泉做客酷狗晚安电台你,有多久没睡一个好觉了?你,还可以无忧无虑入梦吗?整天的嘈杂与喧嚣过后,宁静的声音与旋律成了夜晚最好的陪伴洗个热水澡,安静地躺在床上,打开电台,这或许已经成了不少都市职场工作者的习惯不知不觉中,酷狗。
  • dior口红色号推荐春夏(适合春夏的15支口红)
  • 2024-06-30适合春夏的15支口红这两天刷到了Angelababy为迪奥拍的美妆画报,黑色礼帽,烈焰红唇,高级感满满,鼻姐真的好绝惹!正好大概10086天没分享口红了,Iknow,Iknow~这次让你们种草个够!但选口红是门“技术活”。
  • 余承恩怎么出道的(余承恩即将合作)
  • 2024-06-30余承恩即将合作说起今年娱乐圈炙手可热的人气小花,虞书欣肯定榜上有名虽然虞书欣是女团出道,但是她的表演却很有感染力,得到了很多观众的高度认可无论是《下一站是幸福》中的蔡敏敏,还是《月光变奏曲》中的初礼,虞书欣总能用自。
  • 苹果肉质变黄(苹果变黄身边的知识)
  • 2024-06-30苹果变黄身边的知识苹果变黄——身边的知识苹果是大家常吃的水果之一,富含维生素A与维生素C,有排毒养颜的作用大家也知道,苹果切开或咬开之后,暴露在空气中的部分会变成黄色,那么,关于关于苹果变黄的问题,今天我们来具体讨论一。
  • 双十一什么时候开始的(双十一什么时间开始的)
  • 2024-06-30双十一什么时间开始的双十一是2009年开始的双十一购物狂欢节,是指每年11月11日的网络促销日,源于淘宝商城(天猫)2009年11月11日举办的网络促销活动,当时参与的商家数量和促销力度有限,但营业额远超预想的效果,于是。