mirror of
https://github.com/Evil0ctal/Douyin_TikTok_Download_API.git
synced 2025-04-22 18:36:56 +08:00
761 lines
44 KiB
Python
761 lines
44 KiB
Python
#!/usr/bin/env python
|
||
# -*- encoding: utf-8 -*-
|
||
# @Author: https://github.com/Evil0ctal/
|
||
# @Time: 2021/11/06
|
||
# @Update: 2023/06/27
|
||
# @Version: 3.3.0
|
||
# @Function:
|
||
# 核心代码,估值1块(๑•̀ㅂ•́)و✧
|
||
# 用于爬取Douyin/TikTok数据并以字典形式返回。
|
||
# input link, output dictionary.
|
||
|
||
|
||
import re
|
||
import os
|
||
import time
|
||
import execjs
|
||
import aiohttp
|
||
import platform
|
||
import asyncio
|
||
import traceback
|
||
import configparser
|
||
import urllib.parse
|
||
|
||
from typing import Union
|
||
from tenacity import *
|
||
|
||
|
||
class Scraper:
|
||
"""__________________________________________⬇️initialization(初始化)⬇️______________________________________"""
|
||
|
||
# 初始化/initialization
|
||
def __init__(self):
|
||
self.headers = {
|
||
'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"
|
||
}
|
||
self.douyin_api_headers = {
|
||
'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'
|
||
}
|
||
self.bilibili_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'
|
||
}
|
||
# 判断配置文件是否存在/Check if the configuration file exists
|
||
if os.path.exists('config.ini'):
|
||
self.config = configparser.ConfigParser()
|
||
self.config.read('config.ini', encoding='utf-8')
|
||
# 判断是否使用代理
|
||
if self.config['Scraper']['Proxy_switch'] == 'True':
|
||
# 判断是否区别协议选择代理
|
||
if self.config['Scraper']['Use_different_protocols'] == 'False':
|
||
self.proxies = {
|
||
'all': self.config['Scraper']['All']
|
||
}
|
||
else:
|
||
self.proxies = {
|
||
'http': self.config['Scraper']['Http_proxy'],
|
||
'https': self.config['Scraper']['Https_proxy'],
|
||
}
|
||
else:
|
||
self.proxies = None
|
||
# 配置文件不存在则不使用代理/If the configuration file does not exist, do not use the proxy
|
||
else:
|
||
self.proxies = None
|
||
# 针对Windows系统的异步事件规则/Asynchronous event rules for Windows systems
|
||
if platform.system() == 'Windows':
|
||
asyncio.set_event_loop_policy(asyncio.WindowsSelectorEventLoopPolicy())
|
||
|
||
"""__________________________________________⬇️utils(实用程序)⬇️______________________________________"""
|
||
|
||
# 检索字符串中的链接/Retrieve links from string
|
||
@staticmethod
|
||
def get_url(text: str) -> Union[str, None]:
|
||
try:
|
||
# 从输入文字中提取索引链接存入列表/Extract index links from input text and store in list
|
||
url = re.findall('http[s]?://(?:[a-zA-Z]|[0-9]|[$-_@.&+]|[!*\(\),]|(?:%[0-9a-fA-F][0-9a-fA-F]))+', text)
|
||
# 判断是否有链接/Check if there is a link
|
||
if len(url) > 0:
|
||
return url[0]
|
||
except Exception as e:
|
||
print('Error in get_url:', e)
|
||
return None
|
||
|
||
@staticmethod
|
||
def relpath(file):
|
||
""" Always locate to the correct relative path. """
|
||
from sys import _getframe
|
||
from pathlib import Path
|
||
frame = _getframe(1)
|
||
curr_file = Path(frame.f_code.co_filename)
|
||
return str(curr_file.parent.joinpath(file).resolve())
|
||
|
||
# 生成X-Bogus签名/Generate X-Bogus signature
|
||
@staticmethod
|
||
def generate_x_bogus_url(url: str, headers: dict) -> str:
|
||
query = urllib.parse.urlparse(url).query
|
||
xbogus = execjs.compile(open('./X-Bogus.js').read()).call('sign', query, headers['User-Agent'])
|
||
new_url = url + "&X-Bogus=" + xbogus
|
||
return new_url
|
||
|
||
# 转换链接/convert url
|
||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||
async def convert_share_urls(self, url: str) -> Union[str, None]:
|
||
"""
|
||
用于将分享链接(短链接)转换为原始链接/Convert share links (short links) to original links
|
||
:return: 原始链接/Original link
|
||
"""
|
||
# 检索字符串中的链接/Retrieve links from string
|
||
url = self.get_url(url)
|
||
# 判断是否有链接/Check if there is a link
|
||
if url is None:
|
||
print('无法检索到链接/Unable to retrieve link')
|
||
return None
|
||
# 判断是否为抖音分享链接/judge if it is a douyin share link
|
||
if 'douyin' in url:
|
||
"""
|
||
抖音视频链接类型(不全):
|
||
1. https://v.douyin.com/MuKhKn3/
|
||
2. https://www.douyin.com/video/7157519152863890719
|
||
3. https://www.iesdouyin.com/share/video/7157519152863890719/?region=CN&mid=7157519152863890719&u_code=ffe6jgjg&titleType=title×tamp=1600000000&utm_source=copy_link&utm_campaign=client_share&utm_medium=android&app=aweme&iid=123456789&share_id=123456789
|
||
抖音用户链接类型(不全):
|
||
1. https://www.douyin.com/user/MS4wLjABAAAAbLMPpOhVk441et7z7ECGcmGrK42KtoWOuR0_7pLZCcyFheA9__asY-kGfNAtYqXR?relation=0&vid=7157519152863890719
|
||
2. https://v.douyin.com/MuKoFP4/
|
||
抖音直播链接类型(不全):
|
||
1. https://live.douyin.com/88815422890
|
||
"""
|
||
if 'v.douyin' in url:
|
||
# 转换链接/convert url
|
||
# 例子/Example: https://v.douyin.com/rLyAJgf/8.74
|
||
url = re.compile(r'(https://v.douyin.com/)\w+', re.I).match(url).group()
|
||
print('正在通过抖音分享链接获取原始链接...')
|
||
try:
|
||
async with aiohttp.ClientSession() as session:
|
||
async with session.get(url, headers=self.headers, proxy=self.proxies, allow_redirects=False,
|
||
timeout=10) as response:
|
||
if response.status == 302:
|
||
url = response.headers['Location'].split('?')[0] if '?' in response.headers[
|
||
'Location'] else \
|
||
response.headers['Location']
|
||
print('获取原始链接成功, 原始链接为: {}'.format(url))
|
||
return url
|
||
except Exception as e:
|
||
print('获取原始链接失败!')
|
||
print(e)
|
||
# return None
|
||
raise e
|
||
else:
|
||
print('该链接为原始链接,无需转换,原始链接为: {}'.format(url))
|
||
return url
|
||
# 判断是否为TikTok分享链接/judge if it is a TikTok share link
|
||
elif 'tiktok' in url:
|
||
"""
|
||
TikTok视频链接类型(不全):
|
||
1. https://www.tiktok.com/@tiktok/video/6950000000000000000
|
||
2. https://www.tiktok.com/t/ZTRHcXS2C/
|
||
TikTok用户链接类型(不全):
|
||
1. https://www.tiktok.com/@tiktok
|
||
"""
|
||
if '@' in url:
|
||
print('该链接为原始链接,无需转换,原始链接为: {}'.format(url))
|
||
return url
|
||
else:
|
||
print('正在通过TikTok分享链接获取原始链接...')
|
||
try:
|
||
async with aiohttp.ClientSession() as session:
|
||
async with session.get(url, headers=self.headers, proxy=self.proxies, allow_redirects=False,
|
||
timeout=10) as response:
|
||
if response.status == 301:
|
||
url = response.headers['Location'].split('?')[0] if '?' in response.headers[
|
||
'Location'] else \
|
||
response.headers['Location']
|
||
print('获取原始链接成功, 原始链接为: {}'.format(url))
|
||
return url
|
||
except Exception as e:
|
||
print('获取原始链接失败!')
|
||
print(e)
|
||
return None
|
||
elif 'b23.tv' in url or "bilibili" in url:
|
||
"""
|
||
bilibili视频链接类型(不全):
|
||
1. https://b23.tv/Ya65brl
|
||
2. https://www.bilibili.com/video/BV1MK4y1w7MV/
|
||
bilibili用户链接类型(不全):
|
||
1. https://www.douyin.com/user/MS4wLjABAAAAbLMPpOhVk441et7z7ECGcmGrK42KtoWOuR0_7pLZCcyFheA9__asY-kGfNAtYqXR?relation=0&vid=7157519152863890719
|
||
bilibili直播链接类型(不全):
|
||
"""
|
||
if 'b23.tv' in url:
|
||
print('正在通过哔哩哔哩分享链接获取原始链接...')
|
||
try:
|
||
async with aiohttp.ClientSession() as session:
|
||
async with session.get(url, headers=self.headers, proxy=self.proxies, allow_redirects=False,
|
||
timeout=10) as response:
|
||
if response.status == 302:
|
||
url = response.headers['Location'].split('?')[0] if '?' in response.headers[
|
||
'Location'] else \
|
||
response.headers['Location']
|
||
print('获取原始链接成功, 原始链接为: {}'.format(url))
|
||
return url
|
||
except Exception as e:
|
||
print('获取原始链接失败!')
|
||
print(e)
|
||
# return None
|
||
raise e
|
||
else:
|
||
print('该链接为原始链接,无需转换,原始链接为: {}'.format(url))
|
||
return url
|
||
|
||
"""__________________________________________⬇️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:
|
||
"""
|
||
生成抖音X-Bogus签名
|
||
:param url: 视频链接
|
||
:return: 包含X-Bogus签名的URL
|
||
"""
|
||
# 调用JavaScript函数
|
||
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))
|
||
new_url = url + "&X-Bogus=" + xbogus
|
||
return new_url
|
||
|
||
# 获取抖音视频ID/Get Douyin video ID
|
||
async def get_douyin_video_id(self, original_url: str) -> Union[str, None]:
|
||
"""
|
||
获取视频id
|
||
:param original_url: 视频链接
|
||
:return: 视频id
|
||
"""
|
||
# 正则匹配出视频ID
|
||
try:
|
||
video_url = await self.convert_share_urls(original_url)
|
||
# 链接类型:
|
||
# 视频页 https://www.douyin.com/video/7086770907674348841
|
||
if '/video/' in video_url:
|
||
key = re.findall('/video/(\d+)?', video_url)[0]
|
||
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))
|
||
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))
|
||
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))
|
||
return key
|
||
except Exception as e:
|
||
print('获取抖音视频ID出错了:{}'.format(e))
|
||
return None
|
||
|
||
# 获取单个抖音视频数据/Get single Douyin video data
|
||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||
async def get_douyin_video_data(self, video_id: str) -> Union[dict, None]:
|
||
"""
|
||
: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))
|
||
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("抖音API返回数据: {}".format(video_data))
|
||
return video_data
|
||
except Exception as e:
|
||
print('获取抖音视频数据失败!原因:{}'.format(e))
|
||
# return None
|
||
raise 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))
|
||
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['data']
|
||
print(video_data)
|
||
print('获取视频数据成功!')
|
||
# print("抖音API返回数据: {}".format(video_data))
|
||
return video_data
|
||
except Exception as e:
|
||
print('获取抖音视频数据失败!原因:{}'.format(e))
|
||
# 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
|
||
async def get_tiktok_video_id(self, original_url: str) -> Union[str, None]:
|
||
"""
|
||
获取视频id
|
||
:param original_url: 视频链接
|
||
:return: 视频id
|
||
"""
|
||
try:
|
||
# 转换链接/Convert link
|
||
original_url = await self.convert_share_urls(original_url)
|
||
# 获取视频ID/Get video ID
|
||
if '/video/' in original_url:
|
||
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))
|
||
# 返回视频ID/Return video ID
|
||
return video_id
|
||
except Exception as e:
|
||
print('获取TikTok视频ID出错了:{}'.format(e))
|
||
return None
|
||
|
||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||
async def get_tiktok_video_data(self, video_id: str) -> Union[dict, None]:
|
||
"""
|
||
获取单个视频信息
|
||
:param video_id: 视频id
|
||
:return: 视频信息
|
||
"""
|
||
print('正在获取TikTok视频数据...')
|
||
try:
|
||
# 构造访问链接/Construct the access link
|
||
api_url = f'https://api16-normal-c-useast1a.tiktokv.com/aweme/v1/feed/?aweme_id={video_id}'
|
||
print("正在获取视频数据API: {}".format(api_url))
|
||
async with aiohttp.ClientSession() as session:
|
||
async with session.get(api_url, headers=self.tiktok_api_headers, proxy=self.proxies,
|
||
timeout=10) as response:
|
||
response = await response.json()
|
||
video_data = response['aweme_list'][0]
|
||
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
|
||
async def get_bilibili_video_id(self, original_url: str) -> Union[str, None]:
|
||
"""
|
||
获取视频id
|
||
:param original_url: 视频链接
|
||
:return: 视频id
|
||
"""
|
||
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))
|
||
return None
|
||
|
||
@retry(stop=stop_after_attempt(4), wait=wait_fixed(7))
|
||
async def get_bilibili_video_data(self, video_id: str) -> Union[dict, None]:
|
||
"""
|
||
获取单个视频信息
|
||
:param video_id: 视频id
|
||
:return: 视频信息
|
||
"""
|
||
print('正在获取BiliBili视频数据...')
|
||
try:
|
||
# 构造访问链接/Construct the access link
|
||
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))
|
||
# 这里获取的是m端端播放地址,清晰度不高,需要请求两次 第一次拿需要端参数第二次才能拿到最终的播放地址
|
||
async with aiohttp.ClientSession() as session:
|
||
async with session.get(api_url, headers=self.bilibili_api_headers, proxy=self.proxies,
|
||
timeout=10) as response:
|
||
response = await response.json()
|
||
avid = response.get("data", {}).get("aid", "")
|
||
cid = response.get("data", {}).get("cid", "")
|
||
print('获取视频信息成功!')
|
||
play_url_api = f"https://api.bilibili.com/x/player/playurl?avid={avid}&cid={cid}&platform=html5" # platform 参数得加上不加上获取的播放地址403 待解决
|
||
async with aiohttp.ClientSession() as session:
|
||
async with session.get(play_url_api, headers=self.bilibili_api_headers, proxy=self.proxies,
|
||
timeout=10) as response:
|
||
response = await response.json()
|
||
video_data = response.get("data", {}).get("durl", [])[0]["url"]
|
||
return video_data
|
||
|
||
|
||
except Exception as e:
|
||
print('获取视频信息失败!原因:{}'.format(e))
|
||
# return None
|
||
raise e
|
||
|
||
"""__________________________________________⬇️Hybrid methods(混合方法)⬇️______________________________________"""
|
||
|
||
# 自定义获取数据/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))
|
||
# 获取视频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("准备开始判断并处理数据...")
|
||
|
||
"""
|
||
以下为(视频||图片)数据处理的四个方法,如果你需要自定义数据处理请在这里修改.
|
||
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/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")
|
||
},
|
||
'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!'}
|
||
else:
|
||
print('获取视频ID失败!')
|
||
return {'status': 'failed', 'message': '获取视频ID失败!/Failed to get video ID!'}
|
||
|
||
# 处理数据方便快捷指令使用/Process data for easy-to-use shortcuts
|
||
@staticmethod
|
||
def hybrid_parsing_minimal(data: dict) -> dict:
|
||
# 如果数据获取成功/If the data is successfully obtained
|
||
if data['status'] == 'success':
|
||
result = {
|
||
'status': 'success',
|
||
'message': data.get('message'),
|
||
'platform': data.get('platform'),
|
||
'type': data.get('type'),
|
||
'desc': data.get('desc'),
|
||
'wm_video_url': data['video_data']['wm_video_url'] if data['type'] == 'video' else None,
|
||
'wm_video_url_HQ': data['video_data']['wm_video_url_HQ'] if data['type'] == 'video' else None,
|
||
'nwm_video_url': data['video_data']['nwm_video_url'] if data['type'] == 'video' else None,
|
||
'nwm_video_url_HQ': data['video_data']['nwm_video_url_HQ'] if data['type'] == 'video' else None,
|
||
'no_watermark_image_list': data['image_data']['no_watermark_image_list'] if data[
|
||
'type'] == 'image' else None,
|
||
'watermark_image_list': data['image_data']['watermark_image_list'] if data['type'] == 'image' else None
|
||
}
|
||
return result
|
||
else:
|
||
return data
|
||
|
||
|
||
"""__________________________________________⬇️Test methods(测试方法)⬇️______________________________________"""
|
||
|
||
|
||
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("正在测试异步获取哔哩哔哩视频ID方法...")
|
||
bilibili_id = await api.get_bilibili_video_id(_bilibili_url)
|
||
print("正在测试异步获取哔哩哔哩视频数据方法...")
|
||
bilibili_data = await api.get_bilibili_video_data(bilibili_id)
|
||
print(bilibili_data)
|
||
|
||
print("正在测试异步获取抖音视频ID方法...")
|
||
douyin_id = await api.get_douyin_video_id(_douyin_url)
|
||
print("正在测试异步获取抖音视频数据方法...")
|
||
douyin_data = await api.get_douyin_video_data(douyin_id)
|
||
print(douyin_data)
|
||
|
||
print("正在测试异步获取TikTok视频ID方法...")
|
||
tiktok_id = await api.get_tiktok_video_id(_tiktok_url)
|
||
print("正在测试异步获取TikTok视频数据方法...")
|
||
tiktok_data = await api.get_tiktok_video_data(tiktok_id)
|
||
|
||
print("正在测试异步混合解析方法...")
|
||
douyin_hybrid_data = await api.hybrid_parsing(_douyin_url)
|
||
tiktok_hybrid_data = await api.hybrid_parsing(_tiktok_url)
|
||
|
||
# 总耗时/Total time
|
||
total_time = round(time.time() - start_time, 2)
|
||
print("异步测试完成,总耗时: {}s".format(total_time))
|
||
|
||
|
||
if __name__ == '__main__':
|
||
api = Scraper()
|
||
# 运行测试
|
||
# params = "device_platform=webapp&aid=6383&channel=channel_pc_web&aweme_id=7153585499477757192&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"
|
||
# 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"
|
||
asyncio.run(async_test(_douyin_url=douyin_url, _tiktok_url=tiktok_url, _bilibili_url=bilibili_url))
|