mirror of
https://github.com/Evil0ctal/Douyin_TikTok_Download_API.git
synced 2025-04-19 09:54:46 +08:00
🪄 Add Bilibili Support
This commit is contained in:
parent
2a46a70251
commit
6a42f56503
539
scraper.py
539
scraper.py
@ -2,13 +2,16 @@
|
||||
# -*- encoding: utf-8 -*-
|
||||
# @Author: https://github.com/Evil0ctal/
|
||||
# @Time: 2021/11/06
|
||||
# @Update: 2023/06/27
|
||||
# @Version: 3.3.0
|
||||
# @Function:
|
||||
# @Update: 2023/09/22
|
||||
# @Version: 3.4.0
|
||||
# @Note:
|
||||
# Core code, valued at 1 bucks (๑•̀ㅂ•́)و✧
|
||||
# For scraping Douyin/TikTok/Bilibili data and returning it in dictionary form.
|
||||
# If this project is helpful to you, please give me a star, thank you!
|
||||
# @备注:
|
||||
# 核心代码,估值1块(๑•̀ㅂ•́)و✧
|
||||
# 用于爬取Douyin/TikTok数据并以字典形式返回。
|
||||
# input link, output dictionary.
|
||||
|
||||
# 用于爬取Douyin/TikTok/Bilibili的数据并以字典形式返回。
|
||||
# 如果本项目对您有帮助,请给我一个star,谢谢!
|
||||
|
||||
import re
|
||||
import os
|
||||
@ -37,7 +40,6 @@ class Scraper:
|
||||
'accept-encoding': 'gzip, deflate, br',
|
||||
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36',
|
||||
'referer': 'https://www.douyin.com/',
|
||||
'cookie': "ttwid=1%7C0YBAnAwiC5T3U5yJi8RVXEK3DOwF_2vpJ7kVJJZe8HU%7C1666668932%7C21048e6555b73e8801d3956afc6130b4a05ae73a2eefe4d3fef5ef1b61caf0e9; __live_version__=%221.1.1.2586%22; odin_tt=a77b90afad5db31e86fe004b39c5f35423292023ce7837cde82fd1f7fe54278890ce24dc89e09c8a2e55b1f4904950a7b0fca6b4fbff3b549ba6d55a335373ec; pwa2=%223%7C0%7C0%7C0%22; s_v_web_id=verify_lkagpdq1_IuHpxJyS_q6YH_4AvH_8aNH_zhvGPr95Jrc8; passport_csrf_token=301cf539fb735ab77de7e382b0dd93e5; passport_csrf_token_default=301cf539fb735ab77de7e382b0dd93e5; bd_ticket_guard_client_data=eyJiZC10aWNrZXQtZ3VhcmQtdmVyc2lvbiI6MiwiYmQtdGlja2V0LWd1YXJkLWl0ZXJhdGlvbi12ZXJzaW9uIjoxLCJiZC10aWNrZXQtZ3VhcmQtcmVlLXB1YmxpYy1rZXkiOiJCRXhuWUdqREVBa3ErdjRsT2l3anRIWi9HU2hRNXFseWdJMklLanIxM0orRHozYnA0M2pXc3M3N25CUzdnbE5tTXhHbWU3cldoSE9pdkJvVmNnT2JiWFU9IiwiYmQtdGlja2V0LWd1YXJkLXdlYi12ZXJzaW9uIjoxfQ==; passport_assist_user=CkHJzB17Xsy3FUHyNfX2Dyb8IFKKA_0pu1SKYG0OAT_av3ImQyCbEmGJV7b8MJep4l9MjeCRK1FPY9k9yAkVHbIbvhpICjzS68aPlRjIsUzHLIEM-5jMbp9awcdJnkACni5Nnc_PBm4ljAlEqChbF4nYPpn4xyh4kY2hBvRikmXs0sgQ4fq2DRiJr9ZUIgEDbm8-yw%3D%3D; n_mh=13KNPUKNEzoW3A4J-OLRxfal2zj1GbF-vJUFPs3WSIY; sso_uid_tt=2581aab41d03156c0b7fee9c7e865c6c; sso_uid_tt_ss=2581aab41d03156c0b7fee9c7e865c6c; toutiao_sso_user=b2556b53ed5cee89e947b154b17645f1; toutiao_sso_user_ss=b2556b53ed5cee89e947b154b17645f1; sid_ucp_sso_v1=1.0.0-KDhlZjRhMmJhZGU0OTVmOWM0YzBkMTY5ZGNkZmI4NTFjNTk2ODU5OTkKHwiPluCxqYzbAhC29OKmBhjvMSAMMLDIpZkGOAZA9AcaAmhsIiBiMjU1NmI1M2VkNWNlZTg5ZTk0N2IxNTRiMTc2NDVmMQ; ssid_ucp_sso_v1=1.0.0-KDhlZjRhMmJhZGU0OTVmOWM0YzBkMTY5ZGNkZmI4NTFjNTk2ODU5OTkKHwiPluCxqYzbAhC29OKmBhjvMSAMMLDIpZkGOAZA9AcaAmhsIiBiMjU1NmI1M2VkNWNlZTg5ZTk0N2IxNTRiMTc2NDVmMQ; sid_guard=c1d1ac1d22198149dfc6cac74938b14a%7C1691925046%7C5184000%7CThu%2C+12-Oct-2023+11%3A10%3A46+GMT; uid_tt=7e39a426dac7802b2448fa2266ca1b85; uid_tt_ss=7e39a426dac7802b2448fa2266ca1b85; sid_tt=c1d1ac1d22198149dfc6cac74938b14a; sessionid=c1d1ac1d22198149dfc6cac74938b14a; sessionid_ss=c1d1ac1d22198149dfc6cac74938b14a; sid_ucp_v1=1.0.0-KDc4Y2VkZjIyN2JlMDNhYmNhYTFlYTE5ODM1YzI2YjVlZDNmMGY0N2YKGwiPluCxqYzbAhC29OKmBhjvMSAMOAZA9AdIBBoCbHEiIGMxZDFhYzFkMjIxOTgxNDlkZmM2Y2FjNzQ5MzhiMTRh; ssid_ucp_v1=1.0.0-KDc4Y2VkZjIyN2JlMDNhYmNhYTFlYTE5ODM1YzI2YjVlZDNmMGY0N2YKGwiPluCxqYzbAhC29OKmBhjvMSAMOAZA9AdIBBoCbHEiIGMxZDFhYzFkMjIxOTgxNDlkZmM2Y2FjNzQ5MzhiMTRh; LOGIN_STATUS=1; _bd_ticket_crypt_cookie=861cdca903469f36dd23fc1ecfe847c1; __security_server_data_status=1; store-region=us; store-region-src=uid; d_ticket=28acd5a9c6df4227b13582669694acded6ede; __ac_nonce=064ec4f3a00901157c769; __ac_signature=_02B4Z6wo00f01ve8HKgAAIDD6.-iFWbfM-r3jRgAANkQTCm7UjsJOQlMGY7o-iPsCIAe0kuriDaQ15lHcML.nW.cGNWpSBLUJzdr6s8KHRbqh5ywvupCeAKBEHKKbji7hD1-Z0x3DI-n0KKx34; douyin.com; device_web_cpu_core=16; device_web_memory_size=-1; webcast_local_quality=null; publish_badge_show_info=%220%2C0%2C0%2C1693208382348%22; IsDouyinActive=true; home_can_add_dy_2_desktop=%220%22; strategyABtestKey=%221693208382.387%22; stream_recommend_feed_params=%22%7B%5C%22cookie_enabled%5C%22%3Atrue%2C%5C%22screen_width%5C%22%3A1344%2C%5C%22screen_height%5C%22%3A756%2C%5C%22browser_online%5C%22%3Atrue%2C%5C%22cpu_core_num%5C%22%3A16%2C%5C%22device_memory%5C%22%3A0%2C%5C%22downlink%5C%22%3A%5C%22%5C%22%2C%5C%22effective_type%5C%22%3A%5C%22%5C%22%2C%5C%22round_trip_time%5C%22%3A0%7D%22; VIDEO_FILTER_MEMO_SELECT=%7B%22expireTime%22%3A1693813183367%2C%22type%22%3A1%7D; volume_info=%7B%22isUserMute%22%3Afalse%2C%22isMute%22%3Atrue%2C%22volume%22%3A1%7D; my_rd=1; passport_fe_beating_status=true; msToken=ESPx4FwNhcdEvr36-bmhWde9xupU_c64WeeqvvzqzLCtmEsvGPXhkwsKM8miaoC2w8gWSzNAfqxPEju4w3jzopIFompVSmwemq9-z1F8V-2vLNhTxLlYCUVdXkzNj6zM; download_guide=%221%2F20230828%2F0%22; csrf_session_id=3c194edf7f2cee968b0df65f97a11648; msToken=XFIGWeX20IGrrEUGYr_4SR2DPrduwK5zxB3gOp8FfbxW_Ng-w9uNh8wQRUIoPUtkSblL6msqte55jyfcrKPb8eDZekS9Q1P9hkdkPFiV4Ni-l9Vmsr0KgFo5MOkLaBZy; tt_scid=-i-7N5fAMRj8pGg4drGXbjasutdtD4tzIeqRnm6OJ1LoXRRZGl8FNhORnEuY3id.b3b7"
|
||||
}
|
||||
self.tiktok_api_headers = {
|
||||
'User-Agent': 'com.ss.android.ugc.trill/494+Mozilla/5.0+(Linux;+Android+12;+2112123G+Build/SKQ1.211006.001;+wv)+AppleWebKit/537.36+(KHTML,+like+Gecko)+Version/4.0+Chrome/107.0.5304.105+Mobile+Safari/537.36'
|
||||
@ -211,21 +213,6 @@ class Scraper:
|
||||
|
||||
"""__________________________________________⬇️Douyin methods(抖音方法)⬇️______________________________________"""
|
||||
|
||||
"""
|
||||
Credits: https://github.com/Johnserf-Seed
|
||||
[中文]
|
||||
感谢John为本项目提供了非常多的帮助
|
||||
大家可以去他的仓库点个star :)
|
||||
顺便打个广告, 如果需要更稳定、快速、长期维护的抖音/TikTok API, 或者需要更多的数据(APP端),
|
||||
请移步: https://api.tikhub.io
|
||||
|
||||
[English]
|
||||
Thanks to John for providing a lot of help to this project
|
||||
You can go to his repository and give him a star :)
|
||||
By the way, if you need a more stable, fast and long-term maintenance Douyin/TikTok API, or need more data (APP side),
|
||||
Please go to: https://api.tikhub.io
|
||||
"""
|
||||
|
||||
# 生成抖音X-Bogus签名/Generate Douyin X-Bogus signature
|
||||
# 下面的代码不能保证稳定性,随时可能失效/ The code below cannot guarantee stability and may fail at any time
|
||||
def generate_x_bogus_url(self, url: str) -> str:
|
||||
@ -238,7 +225,7 @@ class Scraper:
|
||||
query = urllib.parse.urlparse(url).query
|
||||
xbogus = execjs.compile(open(self.relpath('./X-Bogus.js')).read()).call('sign', query,
|
||||
self.headers['User-Agent'])
|
||||
print('生成的X-Bogus签名为: {}'.format(xbogus))
|
||||
# print('生成的X-Bogus签名为: {}'.format(xbogus))
|
||||
new_url = url + "&X-Bogus=" + xbogus
|
||||
return new_url
|
||||
|
||||
@ -256,25 +243,25 @@ class Scraper:
|
||||
# 视频页 https://www.douyin.com/video/7086770907674348841
|
||||
if '/video/' in video_url:
|
||||
key = re.findall('/video/(\d+)?', video_url)[0]
|
||||
print('获取到的抖音视频ID为: {}'.format(key))
|
||||
# print('获取到的抖音视频ID为: {}'.format(key))
|
||||
return key
|
||||
# 发现页 https://www.douyin.com/discover?modal_id=7086770907674348841
|
||||
elif 'discover?' in video_url:
|
||||
key = re.findall('modal_id=(\d+)', video_url)[0]
|
||||
print('获取到的抖音视频ID为: {}'.format(key))
|
||||
# print('获取到的抖音视频ID为: {}'.format(key))
|
||||
return key
|
||||
# 直播页
|
||||
elif 'live.douyin' in video_url:
|
||||
# https://live.douyin.com/1000000000000000000
|
||||
video_url = video_url.split('?')[0] if '?' in video_url else video_url
|
||||
key = video_url.replace('https://live.douyin.com/', '')
|
||||
print('获取到的抖音直播ID为: {}'.format(key))
|
||||
# print('获取到的抖音直播ID为: {}'.format(key))
|
||||
return key
|
||||
# note
|
||||
elif 'note' in video_url:
|
||||
# https://www.douyin.com/note/7086770907674348841
|
||||
key = re.findall('/note/(\d+)?', video_url)[0]
|
||||
print('获取到的抖音笔记ID为: {}'.format(key))
|
||||
# print('获取到的抖音笔记ID为: {}'.format(key))
|
||||
return key
|
||||
except Exception as e:
|
||||
print('获取抖音视频ID出错了:{}'.format(e))
|
||||
@ -287,37 +274,32 @@ class Scraper:
|
||||
:param video_id: str - 抖音视频id
|
||||
:return:dict - 包含信息的字典
|
||||
"""
|
||||
print('正在获取抖音视频数据...')
|
||||
try:
|
||||
# 构造访问链接/Construct the access link
|
||||
api_url = f"https://www.douyin.com/aweme/v1/web/aweme/detail/?device_platform=webapp&aid=6383&channel=channel_pc_web&aweme_id={video_id}&pc_client_type=1&version_code=190500&version_name=19.5.0&cookie_enabled=true&screen_width=1344&screen_height=756&browser_language=zh-CN&browser_platform=Win32&browser_name=Firefox&browser_version=110.0&browser_online=true&engine_name=Gecko&engine_version=109.0&os_name=Windows&os_version=10&cpu_core_num=16&device_memory=&platform=PC&webid=7158288523463362079&msToken=abL8SeUTPa9-EToD8qfC7toScSADxpg6yLh2dbNcpWHzE0bT04txM_4UwquIcRvkRb9IU8sifwgM1Kwf1Lsld81o9Irt2_yNyUbbQPSUO8EfVlZJ_78FckDFnwVBVUVK"
|
||||
api_url = self.generate_x_bogus_url(api_url)
|
||||
# 访问API/Access API
|
||||
print("正在获取视频数据API: {}".format(api_url))
|
||||
print("正在请求抖音视频API: {}".format(api_url))
|
||||
async with aiohttp.ClientSession() as session:
|
||||
async with session.get(api_url, headers=self.douyin_api_headers, proxy=self.proxies,
|
||||
timeout=10) as response:
|
||||
response = await response.json()
|
||||
# 获取视频数据/Get video data
|
||||
video_data = response['aweme_detail']
|
||||
print('获取视频数据成功!')
|
||||
# print('获取视频数据成功!')
|
||||
# print("抖音API返回数据: {}".format(video_data))
|
||||
return video_data
|
||||
except Exception as e:
|
||||
print('获取抖音视频数据失败!原因:{}'.format(e))
|
||||
# return None
|
||||
raise e
|
||||
raise ValueError(f"获取抖音视频数据出错了: {e}")
|
||||
|
||||
# 获取单个抖音直播视频数据/Get single Douyin Live video data
|
||||
# 暂时不可用,待修复。
|
||||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||||
async def get_douyin_live_video_data(self, web_rid: str) -> Union[dict, None]:
|
||||
print('正在获取抖音视频数据...')
|
||||
try:
|
||||
# 构造访问链接/Construct the access link
|
||||
api_url = f"https://live.douyin.com/webcast/web/enter/?aid=6383&web_rid={web_rid}"
|
||||
# 访问API/Access API
|
||||
print("正在获取视频数据API: {}".format(api_url))
|
||||
print("正在请求抖音直播API: {}".format(api_url))
|
||||
async with aiohttp.ClientSession() as session:
|
||||
async with session.get(api_url, headers=self.douyin_api_headers, proxy=self.proxies,
|
||||
timeout=10) as response:
|
||||
@ -333,51 +315,6 @@ class Scraper:
|
||||
# return None
|
||||
raise e
|
||||
|
||||
# 获取单个抖音视频数据/Get single Douyin video data
|
||||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||||
async def get_douyin_user_profile_videos(self, profile_url: str, tikhub_token: str) -> Union[dict, None]:
|
||||
try:
|
||||
api_url = f"https://api.tikhub.io/douyin_profile_videos/?douyin_profile_url={profile_url}&cursor=0&count=20"
|
||||
_headers = {"Authorization": f"Bearer {tikhub_token}"}
|
||||
async with aiohttp.ClientSession() as session:
|
||||
async with session.get(api_url, headers=_headers, proxy=self.proxies, timeout=10) as response:
|
||||
response = await response.json()
|
||||
return response
|
||||
except Exception as e:
|
||||
print('获取抖音视频数据失败!原因:{}'.format(e))
|
||||
# return None
|
||||
raise e
|
||||
|
||||
# 获取抖音主页点赞视频数据/Get Douyin profile like video data
|
||||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||||
async def get_douyin_profile_liked_data(self, profile_url: str, tikhub_token: str) -> Union[dict, None]:
|
||||
try:
|
||||
api_url = f"https://api.tikhub.io/douyin_profile_liked_videos/?douyin_profile_url={profile_url}&cursor=0&count=20"
|
||||
_headers = {"Authorization": f"Bearer {tikhub_token}"}
|
||||
async with aiohttp.ClientSession() as session:
|
||||
async with session.get(api_url, headers=_headers, proxy=self.proxies, timeout=10) as response:
|
||||
response = await response.json()
|
||||
return response
|
||||
except Exception as e:
|
||||
print('获取抖音视频数据失败!原因:{}'.format(e))
|
||||
# return None
|
||||
raise e
|
||||
|
||||
# 获取抖音视频评论数据/Get Douyin video comment data
|
||||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||||
async def get_douyin_video_comments(self, video_url: str, tikhub_token: str) -> Union[dict, None]:
|
||||
try:
|
||||
api_url = f"https://api.tikhub.io/douyin_video_comments/?douyin_video_url={video_url}&cursor=0&count=20"
|
||||
_headers = {"Authorization": f"Bearer {tikhub_token}"}
|
||||
async with aiohttp.ClientSession() as session:
|
||||
async with session.get(api_url, headers=_headers, proxy=self.proxies, timeout=10) as response:
|
||||
response = await response.json()
|
||||
return response
|
||||
except Exception as e:
|
||||
print('获取抖音视频数据失败!原因:{}'.format(e))
|
||||
# return None
|
||||
raise e
|
||||
|
||||
"""__________________________________________⬇️TikTok methods(TikTok方法)⬇️______________________________________"""
|
||||
|
||||
# 获取TikTok视频ID/Get TikTok video ID
|
||||
@ -395,7 +332,7 @@ class Scraper:
|
||||
video_id = re.findall('/video/(\d+)', original_url)[0]
|
||||
elif '/v/' in original_url:
|
||||
video_id = re.findall('/v/(\d+)', original_url)[0]
|
||||
print('获取到的TikTok视频ID是{}'.format(video_id))
|
||||
# print('获取到的TikTok视频ID是{}'.format(video_id))
|
||||
# 返回视频ID/Return video ID
|
||||
return video_id
|
||||
except Exception as e:
|
||||
@ -409,7 +346,7 @@ class Scraper:
|
||||
:param video_id: 视频id
|
||||
:return: 视频信息
|
||||
"""
|
||||
print('正在获取TikTok视频数据...')
|
||||
# print('正在获取TikTok视频数据...')
|
||||
try:
|
||||
# 构造访问链接/Construct the access link
|
||||
api_url = f'https://api16-normal-c-useast1a.tiktokv.com/aweme/v1/feed/?aweme_id={video_id}'
|
||||
@ -419,41 +356,13 @@ class Scraper:
|
||||
timeout=10) as response:
|
||||
response = await response.json()
|
||||
video_data = response['aweme_list'][0]
|
||||
print('获取视频信息成功!')
|
||||
# print('获取视频信息成功!')
|
||||
return video_data
|
||||
except Exception as e:
|
||||
print('获取视频信息失败!原因:{}'.format(e))
|
||||
# return None
|
||||
raise e
|
||||
|
||||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||||
async def get_tiktok_user_profile_videos(self, tiktok_video_url: str, tikhub_token: str) -> Union[dict, None]:
|
||||
try:
|
||||
api_url = f"https://api.tikhub.io/tiktok_profile_videos/?tiktok_video_url={tiktok_video_url}&cursor=0&count=20"
|
||||
_headers = {"Authorization": f"Bearer {tikhub_token}"}
|
||||
async with aiohttp.ClientSession() as session:
|
||||
async with session.get(api_url, headers=_headers, proxy=self.proxies, timeout=10) as response:
|
||||
response = await response.json()
|
||||
return response
|
||||
except Exception as e:
|
||||
print('获取抖音视频数据失败!原因:{}'.format(e))
|
||||
# return None
|
||||
raise e
|
||||
|
||||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||||
async def get_tiktok_user_profile_liked_videos(self, tiktok_video_url: str, tikhub_token: str) -> Union[dict, None]:
|
||||
try:
|
||||
api_url = f"https://api.tikhub.io/tiktok_profile_liked_videos/?tiktok_video_url={tiktok_video_url}&cursor=0&count=20"
|
||||
_headers = {"Authorization": f"Bearer {tikhub_token}"}
|
||||
async with aiohttp.ClientSession() as session:
|
||||
async with session.get(api_url, headers=_headers, proxy=self.proxies, timeout=10) as response:
|
||||
response = await response.json()
|
||||
return response
|
||||
except Exception as e:
|
||||
print('获取抖音视频数据失败!原因:{}'.format(e))
|
||||
# return None
|
||||
raise e
|
||||
|
||||
"""__________________________________________⬇️bilibili methods(Bilibili方法)⬇️______________________________________"""
|
||||
|
||||
# 获取TikTok视频ID/Get TikTok video ID
|
||||
@ -466,18 +375,18 @@ class Scraper:
|
||||
try:
|
||||
# 转换链接/Convert link
|
||||
original_url = await self.convert_share_urls(original_url)
|
||||
# 获取视频ID/Get video ID
|
||||
if "video/BV" in original_url:
|
||||
video_id = str('video/BV'.join(re.findall(r"BV([0-9,a-z,A-Z]+)[?]{0,1}.*", original_url)))
|
||||
elif "video/av" in original_url: #
|
||||
video_id = str('video/av'.join(re.findall(r"av([0-9,a-z,A-Z]+)[?]{0,1}.*", original_url)))
|
||||
|
||||
print('获取到的BiliBili视频ID是{}'.format(video_id))
|
||||
# 返回视频ID/Return video ID
|
||||
return video_id
|
||||
except Exception as e:
|
||||
print('获取BiliBili视频ID出错了:{}'.format(e))
|
||||
match = re.search(r"video/BV(?P<id>[0-9a-zA-Z]+)", original_url)
|
||||
if match:
|
||||
return f"video/BV{match.group('id')}"
|
||||
elif "video/av" in original_url:
|
||||
match = re.search(r"video/av(?P<id>[0-9a-zA-Z]+)", original_url)
|
||||
if match:
|
||||
return f"video/av{match.group('id')}"
|
||||
# 如果未找到匹配项,可以选择抛出异常或返回 None,这里我选择返回 None
|
||||
return None
|
||||
except Exception as e:
|
||||
raise ValueError(f'获取BiliBili视频ID出错了:{e}')
|
||||
|
||||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||||
async def get_bilibili_video_data(self, video_id: str) -> Union[dict, None]:
|
||||
@ -492,7 +401,7 @@ class Scraper:
|
||||
api_url = f'https://api.bilibili.com/x/web-interface/view?bvid={video_id.replace("video/BV", "")}'
|
||||
if "video/av" in video_id:
|
||||
api_url = f'https://api.bilibili.com/x/web-interface/view?aid={video_id.replace("video/av", "")}'
|
||||
print("正在获取视频数据API: {}".format(api_url))
|
||||
print(f"正在获取视频数据API: {api_url}")
|
||||
# 这里获取的是m端端播放地址,清晰度不高,需要请求两次 第一次拿需要端参数第二次才能拿到最终的播放地址
|
||||
async with aiohttp.ClientSession() as session:
|
||||
async with session.get(api_url, headers=self.bilibili_api_headers, proxy=self.proxies,
|
||||
@ -507,189 +416,216 @@ class Scraper:
|
||||
timeout=10) as response:
|
||||
response = await response.json()
|
||||
video_data = response.get("data", {}).get("durl", [])[0]["url"]
|
||||
video_data = {
|
||||
'status': 'success',
|
||||
'message': "更多接口请查看(More API see): https://api.tikhub.io/",
|
||||
'type': 'video',
|
||||
'platform': 'bilibili',
|
||||
'video_url': video_data,
|
||||
}
|
||||
return video_data
|
||||
|
||||
|
||||
except Exception as e:
|
||||
print('获取视频信息失败!原因:{}'.format(e))
|
||||
# return None
|
||||
raise e
|
||||
raise ValueError(f'获取BiliBili视频数据出错了:{e}')
|
||||
|
||||
"""__________________________________________⬇️Hybrid methods(混合方法)⬇️______________________________________"""
|
||||
|
||||
# 判断链接平台/Judge link platform
|
||||
async def judge_url_platform(self, video_url: str) -> str:
|
||||
if 'douyin' in video_url:
|
||||
url_platform = 'douyin'
|
||||
elif 'bilibili' in video_url:
|
||||
url_platform = 'bilibili'
|
||||
elif 'tiktok' in video_url:
|
||||
url_platform = 'tiktok'
|
||||
else:
|
||||
url_platform = None
|
||||
return url_platform
|
||||
|
||||
# 自定义获取数据/Custom data acquisition
|
||||
async def hybrid_parsing(self, video_url: str) -> dict:
|
||||
# URL平台判断/Judge URL platform
|
||||
url_platform = 'douyin' if 'douyin' in video_url else 'tiktok'
|
||||
print('当前链接平台为:{}'.format(url_platform))
|
||||
url_platform = await self.judge_url_platform(video_url)
|
||||
|
||||
# 如果不是指定平台抛出异常/If it is not the specified platform, an exception is thrown
|
||||
if not url_platform:
|
||||
raise ValueError(f"链接**{video_url}**不是抖音、Bilibili、TikTok链接!")
|
||||
|
||||
print(f"正在解析**{url_platform}**视频链接...")
|
||||
|
||||
# 获取视频ID/Get video ID
|
||||
print("正在获取视频ID...")
|
||||
video_id = await self.get_douyin_video_id(
|
||||
video_url) if url_platform == 'douyin' else await self.get_tiktok_video_id(
|
||||
video_url)
|
||||
if video_id:
|
||||
print("获取视频ID成功,视频ID为:{}".format(video_id))
|
||||
# 获取视频数据/Get video data
|
||||
print("正在获取视频数据...")
|
||||
data = await self.get_douyin_video_data(
|
||||
video_id) if url_platform == 'douyin' else await self.get_tiktok_video_data(
|
||||
video_id)
|
||||
if data:
|
||||
print("获取视频数据成功,正在判断数据类型...")
|
||||
url_type_code = data['aweme_type']
|
||||
"""以下为抖音/TikTok类型代码/Type code for Douyin/TikTok"""
|
||||
url_type_code_dict = {
|
||||
# 抖音/Douyin
|
||||
2: 'image',
|
||||
4: 'video',
|
||||
68: 'image',
|
||||
# TikTok
|
||||
0: 'video',
|
||||
51: 'video',
|
||||
55: 'video',
|
||||
58: 'video',
|
||||
61: 'video',
|
||||
150: 'image'
|
||||
}
|
||||
# 获取视频类型/Get video type
|
||||
# 如果类型代码不存在,则默认为视频类型/If the type code does not exist, it is assumed to be a video type
|
||||
print("数据类型代码: {}".format(url_type_code))
|
||||
# 判断链接类型/Judge link type
|
||||
url_type = url_type_code_dict.get(url_type_code, 'video')
|
||||
print("数据类型: {}".format(url_type))
|
||||
print("准备开始判断并处理数据...")
|
||||
video_id = await self.get_douyin_video_id(video_url) if url_platform == 'douyin' \
|
||||
else await self.get_tiktok_video_id(video_url) if url_platform == 'tiktok' \
|
||||
else await self.get_bilibili_video_id(video_url) if url_platform == 'bilibili' \
|
||||
else None
|
||||
|
||||
"""
|
||||
以下为(视频||图片)数据处理的四个方法,如果你需要自定义数据处理请在这里修改.
|
||||
The following are four methods of (video || image) data processing.
|
||||
If you need to customize data processing, please modify it here.
|
||||
"""
|
||||
# 如果获取不到视频ID抛出异常/If the video ID cannot be obtained, an exception is thrown
|
||||
if not video_id:
|
||||
raise ValueError(f"获取**{url_platform}**视频ID失败!")
|
||||
|
||||
"""
|
||||
创建已知数据字典(索引相同),稍后使用.update()方法更新数据
|
||||
Create a known data dictionary (index the same),
|
||||
and then use the .update() method to update the data
|
||||
"""
|
||||
print(f"获取到的**{url_platform}**视频ID是{video_id}")
|
||||
# 获取视频数据/Get video data
|
||||
data = await self.get_douyin_video_data(video_id) if url_platform == 'douyin' \
|
||||
else await self.get_tiktok_video_data(video_id) if url_platform == 'tiktok' \
|
||||
else await self.get_bilibili_video_data(video_id) if url_platform == 'bilibili' \
|
||||
else None
|
||||
|
||||
result_data = {
|
||||
'status': 'success',
|
||||
'message': "更多接口请查看(More API see): https://api.tikhub.io/docs",
|
||||
'type': url_type,
|
||||
'platform': url_platform,
|
||||
'aweme_id': video_id,
|
||||
'official_api_url':
|
||||
{
|
||||
"User-Agent": self.headers["User-Agent"],
|
||||
"api_url": f"https://www.iesdouyin.com/aweme/v1/web/aweme/detail/?aweme_id={video_id}&aid=1128&version_name=23.5.0&device_platform=android&os_version=2333&Github=Evil0ctal&words=FXXK_U_ByteDance"
|
||||
} if url_platform == 'douyin'
|
||||
else
|
||||
{
|
||||
"User-Agent": self.tiktok_api_headers["User-Agent"],
|
||||
"api_url": f'https://api16-normal-c-useast1a.tiktokv.com/aweme/v1/feed/?aweme_id={video_id}'
|
||||
},
|
||||
'desc': data.get("desc"),
|
||||
'create_time': data.get("create_time"),
|
||||
'author': data.get("author"),
|
||||
'music': data.get("music"),
|
||||
'statistics': data.get("statistics"),
|
||||
'cover_data': {
|
||||
'cover': data.get("video").get("cover"),
|
||||
'origin_cover': data.get("video").get("origin_cover"),
|
||||
'dynamic_cover': data.get("video").get("dynamic_cover")
|
||||
if data:
|
||||
|
||||
# 如果是Bilibili平台则返回视频数据/If it is a Bilibili platform, return video data
|
||||
if url_platform == 'bilibili':
|
||||
print("获取Bilibili视频数据成功!")
|
||||
return data
|
||||
|
||||
# 如果是抖音/TikTok平台则继续进行数据解析/If it is a Douyin/TikTok platform, continue to parse the data
|
||||
print(f"获取**{url_platform}**视频数据成功,正在判断数据类型...")
|
||||
url_type_code = data['aweme_type']
|
||||
# 以下为抖音或TikTok类型代码/Type code for Douyin or TikTok
|
||||
url_type_code_dict = {
|
||||
# 抖音/Douyin
|
||||
2: 'image',
|
||||
4: 'video',
|
||||
68: 'image',
|
||||
# TikTok
|
||||
0: 'video',
|
||||
51: 'video',
|
||||
55: 'video',
|
||||
58: 'video',
|
||||
61: 'video',
|
||||
150: 'image'
|
||||
}
|
||||
# 判断链接类型/Judge link type
|
||||
url_type = url_type_code_dict.get(url_type_code, 'video')
|
||||
print(f"获取到的**{url_platform}**的链接类型是{url_type}")
|
||||
print("准备开始判断并处理数据...")
|
||||
|
||||
"""
|
||||
以下为(视频||图片)数据处理的四个方法,如果你需要自定义数据处理请在这里修改.
|
||||
The following are four methods of (video || image) data processing.
|
||||
If you need to customize data processing, please modify it here.
|
||||
"""
|
||||
|
||||
"""
|
||||
创建已知数据字典(索引相同),稍后使用.update()方法更新数据
|
||||
Create a known data dictionary (index the same),
|
||||
and then use the .update() method to update the data
|
||||
"""
|
||||
|
||||
result_data = {
|
||||
'status': 'success',
|
||||
'message': "更多接口请查看(More API see): https://api.tikhub.io/",
|
||||
'type': url_type,
|
||||
'platform': url_platform,
|
||||
'aweme_id': video_id,
|
||||
'official_api_url':
|
||||
{
|
||||
"User-Agent": self.headers["User-Agent"],
|
||||
"api_url": f"https://www.iesdouyin.com/aweme/v1/web/aweme/detail/?aweme_id={video_id}&aid=1128&version_name=23.5.0&device_platform=android&os_version=2333&Github=Evil0ctal&words=FXXK_U_ByteDance"
|
||||
} if url_platform == 'douyin'
|
||||
else
|
||||
{
|
||||
"User-Agent": self.tiktok_api_headers["User-Agent"],
|
||||
"api_url": f'https://api16-normal-c-useast1a.tiktokv.com/aweme/v1/feed/?aweme_id={video_id}'
|
||||
},
|
||||
'hashtags': data.get('text_extra'),
|
||||
}
|
||||
# 创建一个空变量,稍后使用.update()方法更新数据/Create an empty variable and use the .update() method to update the data
|
||||
api_data = None
|
||||
# 判断链接类型并处理数据/Judge link type and process data
|
||||
try:
|
||||
# 抖音数据处理/Douyin data processing
|
||||
if url_platform == 'douyin':
|
||||
# 抖音视频数据处理/Douyin video data processing
|
||||
if url_type == 'video':
|
||||
print("正在处理抖音视频数据...")
|
||||
# 将信息储存在字典中/Store information in a dictionary
|
||||
uri = data['video']['play_addr']['uri']
|
||||
wm_video_url = data['video']['play_addr']['url_list'][0]
|
||||
wm_video_url_HQ = f"https://aweme.snssdk.com/aweme/v1/playwm/?video_id={uri}&radio=1080p&line=0"
|
||||
nwm_video_url = wm_video_url.replace('playwm', 'play')
|
||||
nwm_video_url_HQ = f"https://aweme.snssdk.com/aweme/v1/play/?video_id={uri}&ratio=1080p&line=0"
|
||||
api_data = {
|
||||
'video_data':
|
||||
{
|
||||
'wm_video_url': wm_video_url,
|
||||
'wm_video_url_HQ': wm_video_url_HQ,
|
||||
'nwm_video_url': nwm_video_url,
|
||||
'nwm_video_url_HQ': nwm_video_url_HQ
|
||||
}
|
||||
}
|
||||
# 抖音图片数据处理/Douyin image data processing
|
||||
elif url_type == 'image':
|
||||
print("正在处理抖音图片数据...")
|
||||
# 无水印图片列表/No watermark image list
|
||||
no_watermark_image_list = []
|
||||
# 有水印图片列表/With watermark image list
|
||||
watermark_image_list = []
|
||||
# 遍历图片列表/Traverse image list
|
||||
for i in data['images']:
|
||||
no_watermark_image_list.append(i['url_list'][0])
|
||||
watermark_image_list.append(i['download_url_list'][0])
|
||||
api_data = {
|
||||
'image_data':
|
||||
{
|
||||
'no_watermark_image_list': no_watermark_image_list,
|
||||
'watermark_image_list': watermark_image_list
|
||||
}
|
||||
}
|
||||
# TikTok数据处理/TikTok data processing
|
||||
elif url_platform == 'tiktok':
|
||||
# TikTok视频数据处理/TikTok video data processing
|
||||
if url_type == 'video':
|
||||
print("正在处理TikTok视频数据...")
|
||||
# 将信息储存在字典中/Store information in a dictionary
|
||||
wm_video = data['video']['download_addr']['url_list'][0]
|
||||
api_data = {
|
||||
'video_data':
|
||||
{
|
||||
'wm_video_url': wm_video,
|
||||
'wm_video_url_HQ': wm_video,
|
||||
'nwm_video_url': data['video']['play_addr']['url_list'][0],
|
||||
'nwm_video_url_HQ': data['video']['bit_rate'][0]['play_addr']['url_list'][0]
|
||||
}
|
||||
}
|
||||
# TikTok图片数据处理/TikTok image data processing
|
||||
elif url_type == 'image':
|
||||
print("正在处理TikTok图片数据...")
|
||||
# 无水印图片列表/No watermark image list
|
||||
no_watermark_image_list = []
|
||||
# 有水印图片列表/With watermark image list
|
||||
watermark_image_list = []
|
||||
for i in data['image_post_info']['images']:
|
||||
no_watermark_image_list.append(i['display_image']['url_list'][0])
|
||||
watermark_image_list.append(i['owner_watermark_image']['url_list'][0])
|
||||
api_data = {
|
||||
'image_data':
|
||||
{
|
||||
'no_watermark_image_list': no_watermark_image_list,
|
||||
'watermark_image_list': watermark_image_list
|
||||
}
|
||||
}
|
||||
# 更新数据/Update data
|
||||
result_data.update(api_data)
|
||||
# print("数据处理完成,最终数据: \n{}".format(result_data))
|
||||
# 返回数据/Return data
|
||||
return result_data
|
||||
except Exception as e:
|
||||
traceback.print_exc()
|
||||
print("数据处理失败!")
|
||||
return {'status': 'failed', 'message': '数据处理失败!/Data processing failed!'}
|
||||
else:
|
||||
print("[抖音|TikTok方法]返回数据为空,无法处理!")
|
||||
return {'status': 'failed',
|
||||
'message': '返回数据为空,无法处理!/Return data is empty and cannot be processed!'}
|
||||
'desc': data.get("desc"),
|
||||
'create_time': data.get("create_time"),
|
||||
'author': data.get("author"),
|
||||
'music': data.get("music"),
|
||||
'statistics': data.get("statistics"),
|
||||
'cover_data': {
|
||||
'cover': data.get("video").get("cover"),
|
||||
'origin_cover': data.get("video").get("origin_cover"),
|
||||
'dynamic_cover': data.get("video").get("dynamic_cover")
|
||||
},
|
||||
'hashtags': data.get('text_extra'),
|
||||
}
|
||||
# 创建一个空变量,稍后使用.update()方法更新数据/Create an empty variable and use the .update() method to update the data
|
||||
api_data = None
|
||||
# 判断链接类型并处理数据/Judge link type and process data
|
||||
try:
|
||||
# 抖音数据处理/Douyin data processing
|
||||
if url_platform == 'douyin':
|
||||
# 抖音视频数据处理/Douyin video data processing
|
||||
if url_type == 'video':
|
||||
print("正在处理抖音视频数据...")
|
||||
# 将信息储存在字典中/Store information in a dictionary
|
||||
uri = data['video']['play_addr']['uri']
|
||||
wm_video_url = data['video']['play_addr']['url_list'][0]
|
||||
wm_video_url_HQ = f"https://aweme.snssdk.com/aweme/v1/playwm/?video_id={uri}&radio=1080p&line=0"
|
||||
nwm_video_url = wm_video_url.replace('playwm', 'play')
|
||||
nwm_video_url_HQ = f"https://aweme.snssdk.com/aweme/v1/play/?video_id={uri}&ratio=1080p&line=0"
|
||||
api_data = {
|
||||
'video_data':
|
||||
{
|
||||
'wm_video_url': wm_video_url,
|
||||
'wm_video_url_HQ': wm_video_url_HQ,
|
||||
'nwm_video_url': nwm_video_url,
|
||||
'nwm_video_url_HQ': nwm_video_url_HQ
|
||||
}
|
||||
}
|
||||
# 抖音图片数据处理/Douyin image data processing
|
||||
elif url_type == 'image':
|
||||
print("正在处理抖音图片数据...")
|
||||
# 无水印图片列表/No watermark image list
|
||||
no_watermark_image_list = []
|
||||
# 有水印图片列表/With watermark image list
|
||||
watermark_image_list = []
|
||||
# 遍历图片列表/Traverse image list
|
||||
for i in data['images']:
|
||||
no_watermark_image_list.append(i['url_list'][0])
|
||||
watermark_image_list.append(i['download_url_list'][0])
|
||||
api_data = {
|
||||
'image_data':
|
||||
{
|
||||
'no_watermark_image_list': no_watermark_image_list,
|
||||
'watermark_image_list': watermark_image_list
|
||||
}
|
||||
}
|
||||
# TikTok数据处理/TikTok data processing
|
||||
elif url_platform == 'tiktok':
|
||||
# TikTok视频数据处理/TikTok video data processing
|
||||
if url_type == 'video':
|
||||
print("正在处理TikTok视频数据...")
|
||||
# 将信息储存在字典中/Store information in a dictionary
|
||||
wm_video = data['video']['download_addr']['url_list'][0]
|
||||
api_data = {
|
||||
'video_data':
|
||||
{
|
||||
'wm_video_url': wm_video,
|
||||
'wm_video_url_HQ': wm_video,
|
||||
'nwm_video_url': data['video']['play_addr']['url_list'][0],
|
||||
'nwm_video_url_HQ': data['video']['bit_rate'][0]['play_addr']['url_list'][0]
|
||||
}
|
||||
}
|
||||
# TikTok图片数据处理/TikTok image data processing
|
||||
elif url_type == 'image':
|
||||
print("正在处理TikTok图片数据...")
|
||||
# 无水印图片列表/No watermark image list
|
||||
no_watermark_image_list = []
|
||||
# 有水印图片列表/With watermark image list
|
||||
watermark_image_list = []
|
||||
for i in data['image_post_info']['images']:
|
||||
no_watermark_image_list.append(i['display_image']['url_list'][0])
|
||||
watermark_image_list.append(i['owner_watermark_image']['url_list'][0])
|
||||
api_data = {
|
||||
'image_data':
|
||||
{
|
||||
'no_watermark_image_list': no_watermark_image_list,
|
||||
'watermark_image_list': watermark_image_list
|
||||
}
|
||||
}
|
||||
# 更新数据/Update data
|
||||
result_data.update(api_data)
|
||||
# print("数据处理完成,最终数据: \n{}".format(result_data))
|
||||
# 返回数据/Return data
|
||||
return result_data
|
||||
except Exception as e:
|
||||
traceback.print_exc()
|
||||
print("数据处理失败!")
|
||||
return {'status': 'failed', 'message': '数据处理失败!/Data processing failed!'}
|
||||
else:
|
||||
print('获取视频ID失败!')
|
||||
return {'status': 'failed', 'message': '获取视频ID失败!/Failed to get video ID!'}
|
||||
print("[抖音|TikTok方法]返回数据为空,无法处理!")
|
||||
return {'status': 'failed',
|
||||
'message': '返回数据为空,无法处理!/Return data is empty and cannot be processed!'}
|
||||
|
||||
# 处理数据方便快捷指令使用/Process data for easy-to-use shortcuts
|
||||
@staticmethod
|
||||
@ -721,29 +657,40 @@ class Scraper:
|
||||
async def async_test(_douyin_url: str = None, _tiktok_url: str = None, _bilibili_url: str = None) -> None:
|
||||
# 异步测试/Async test
|
||||
start_time = time.time()
|
||||
print("正在进行异步测试...")
|
||||
print("<异步测试/Async test>")
|
||||
|
||||
print('\n--------------------------------------------------')
|
||||
print("正在测试异步获取哔哩哔哩视频ID方法...")
|
||||
bilibili_id = await api.get_bilibili_video_id(_bilibili_url)
|
||||
print(f"哔哩哔哩视频ID: {bilibili_id}")
|
||||
print("正在测试异步获取哔哩哔哩视频数据方法...")
|
||||
bilibili_data = await api.get_bilibili_video_data(bilibili_id)
|
||||
print(bilibili_data)
|
||||
print(f"哔哩哔哩视频数据: {str(bilibili_data)[:100]}")
|
||||
|
||||
print('\n--------------------------------------------------')
|
||||
print("正在测试异步获取抖音视频ID方法...")
|
||||
douyin_id = await api.get_douyin_video_id(_douyin_url)
|
||||
print(f"抖音视频ID: {douyin_id}")
|
||||
print("正在测试异步获取抖音视频数据方法...")
|
||||
douyin_data = await api.get_douyin_video_data(douyin_id)
|
||||
print(douyin_data)
|
||||
print(f"抖音视频数据: {str(douyin_data)[:100]}")
|
||||
|
||||
print('\n--------------------------------------------------')
|
||||
print("正在测试异步获取TikTok视频ID方法...")
|
||||
tiktok_id = await api.get_tiktok_video_id(_tiktok_url)
|
||||
print(f"TikTok视频ID: {tiktok_id}")
|
||||
print("正在测试异步获取TikTok视频数据方法...")
|
||||
tiktok_data = await api.get_tiktok_video_data(tiktok_id)
|
||||
print(f"TikTok视频数据: {str(tiktok_data)[:100]}")
|
||||
|
||||
print('\n--------------------------------------------------')
|
||||
print("正在测试异步混合解析方法...")
|
||||
douyin_hybrid_data = await api.hybrid_parsing(_douyin_url)
|
||||
tiktok_hybrid_data = await api.hybrid_parsing(_tiktok_url)
|
||||
bilibili_hybrid_data = await api.hybrid_parsing(_bilibili_url)
|
||||
print(f"抖音、TikTok、哔哩哔哩混合解析全部成功!")
|
||||
|
||||
print('\n--------------------------------------------------')
|
||||
# 总耗时/Total time
|
||||
total_time = round(time.time() - start_time, 2)
|
||||
print("异步测试完成,总耗时: {}s".format(total_time))
|
||||
@ -756,5 +703,5 @@ if __name__ == '__main__':
|
||||
# api.generate_x_bogus(params)
|
||||
douyin_url = 'https://v.douyin.com/rLyrQxA/6.66'
|
||||
tiktok_url = 'https://www.tiktok.com/@evil0ctal/video/7217027383390555438'
|
||||
bilibili_url = "https://b23.tv/Ya65brl"
|
||||
bilibili_url = "https://www.bilibili.com/video/BV1Th411x7ii/"
|
||||
asyncio.run(async_test(_douyin_url=douyin_url, _tiktok_url=tiktok_url, _bilibili_url=bilibili_url))
|
||||
|
Loading…
x
Reference in New Issue
Block a user