diff --git a/.gitignore b/.gitignore index 8b843de..b0b6f3a 100644 --- a/.gitignore +++ b/.gitignore @@ -1,160 +1,160 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -share/python-wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ -.nox/ -.coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -*.py,cover -.hypothesis/ -.pytest_cache/ -cover/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 -db.sqlite3-journal - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -.pybuilder/ -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# IPython -profile_default/ -ipython_config.py - -# pyenv -# For a library or package, you might want to ignore these files since the code is -# intended to run in multiple environments; otherwise, check them in: -# .python-version - -# pipenv -# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. -# However, in case of collaboration, if having platform-specific dependencies or dependencies -# having no cross-platform support, pipenv may install dependencies that don't work, or not -# install all needed dependencies. -#Pipfile.lock - -# poetry -# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control. -# This is especially recommended for binary packages to ensure reproducibility, and is more -# commonly ignored for libraries. -# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control -#poetry.lock - -# pdm -# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control. -#pdm.lock -# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it -# in version control. -# https://pdm.fming.dev/#use-with-ide -.pdm.toml - -# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm -__pypackages__/ - -# Celery stuff -celerybeat-schedule -celerybeat.pid - -# SageMath parsed files -*.sage.py - -# Environments -.env -.venv -env/ -venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ -.dmypy.json -dmypy.json - -# Pyre type checker -.pyre/ - -# pytype static type analyzer -.pytype/ - -# Cython debug symbols -cython_debug/ - -# PyCharm -# JetBrains specific template is maintained in a separate JetBrains.gitignore that can -# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore -# and can be added to the global gitignore or merged into this file. For a more nuclear -# option (not recommended) you can uncomment the following to ignore the entire idea folder. +# Byte-compiled / optimized / DLL files +__pycache__/ +*.py[cod] +*$py.class + +# C extensions +*.so + +# Distribution / packaging +.Python +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +share/python-wheels/ +*.egg-info/ +.installed.cfg +*.egg +MANIFEST + +# PyInstaller +# Usually these files are written by a python script from a template +# before PyInstaller builds the exe, so as to inject date/other infos into it. +*.manifest +*.spec + +# Installer logs +pip-log.txt +pip-delete-this-directory.txt + +# Unit test / coverage reports +htmlcov/ +.tox/ +.nox/ +.coverage +.coverage.* +.cache +nosetests.xml +coverage.xml +*.cover +*.py,cover +.hypothesis/ +.pytest_cache/ +cover/ + +# Translations +*.mo +*.pot + +# Django stuff: +*.log +local_settings.py +db.sqlite3 +db.sqlite3-journal + +# Flask stuff: +instance/ +.webassets-cache + +# Scrapy stuff: +.scrapy + +# Sphinx documentation +docs/_build/ + +# PyBuilder +.pybuilder/ +target/ + +# Jupyter Notebook +.ipynb_checkpoints + +# IPython +profile_default/ +ipython_config.py + +# pyenv +# For a library or package, you might want to ignore these files since the code is +# intended to run in multiple environments; otherwise, check them in: +# .python-version + +# pipenv +# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. +# However, in case of collaboration, if having platform-specific dependencies or dependencies +# having no cross-platform support, pipenv may install dependencies that don't work, or not +# install all needed dependencies. +#Pipfile.lock + +# poetry +# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control. +# This is especially recommended for binary packages to ensure reproducibility, and is more +# commonly ignored for libraries. +# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control +#poetry.lock + +# pdm +# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control. +#pdm.lock +# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it +# in version control. +# https://pdm.fming.dev/#use-with-ide +.pdm.toml + +# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm +__pypackages__/ + +# Celery stuff +celerybeat-schedule +celerybeat.pid + +# SageMath parsed files +*.sage.py + +# Environments +.env +.venv +env/ +venv/ +ENV/ +env.bak/ +venv.bak/ + +# Spyder project settings +.spyderproject +.spyproject + +# Rope project settings +.ropeproject + +# mkdocs documentation +/site + +# mypy +.mypy_cache/ +.dmypy.json +dmypy.json + +# Pyre type checker +.pyre/ + +# pytype static type analyzer +.pytype/ + +# Cython debug symbols +cython_debug/ + +# PyCharm +# JetBrains specific template is maintained in a separate JetBrains.gitignore that can +# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore +# and can be added to the global gitignore or merged into this file. For a more nuclear +# option (not recommended) you can uncomment the following to ignore the entire idea folder. .idea/ \ No newline at end of file diff --git a/Dockerfile b/Dockerfile index 1bf3af9..7d636cd 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,19 +1,19 @@ -# This Dockerfile is used to build an Python environment -FROM node:18-bullseye-slim - -LABEL maintainer="imgyh" - -WORKDIR /app - -ADD . $WORKDIR - -RUN sed -i s/deb.debian.org/mirrors.aliyun.com/g /etc/apt/sources.list - -RUN apt-get update && apt-get install -y python3.9 python3-pip - -RUN pip3 install -r requirements.txt - -ENV TZ=Asia/Shanghai - -CMD ["python3", "TikTokWeb.py"] - +# This Dockerfile is used to build an Python environment +FROM node:18-bullseye-slim + +LABEL maintainer="imgyh" + +WORKDIR /app + +ADD . $WORKDIR + +RUN sed -i s/deb.debian.org/mirrors.aliyun.com/g /etc/apt/sources.list + +RUN apt-get update && apt-get install -y python3.9 python3-pip + +RUN pip3 install -r requirements.txt + +ENV TZ=Asia/Shanghai + +CMD ["python3", "TikTokWeb.py"] + diff --git a/DouYinSelenium.py b/DouYinSelenium.py index d301e4c..c3b13e7 100644 --- a/DouYinSelenium.py +++ b/DouYinSelenium.py @@ -83,6 +83,7 @@ class TikTok(object): userVideoUrls.append(videoRealUrl) return userVideoUrls + tk = TikTok() # tk.oneVideoInfo() tk.userVideoInfo() diff --git a/TikTok.py b/TikTok.py index 0115b50..ecc996c 100644 --- a/TikTok.py +++ b/TikTok.py @@ -48,10 +48,10 @@ class TikTok(object): self.utils = Utils() self.result = Result() 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', - 'referer': 'https://www.douyin.com/', - 'accept-encoding': None, - 'Cookie': f"msToken={self.utils.generate_random_str(107)}; ttwid={self.utils.getttwid()}; odin_tt=324fb4ea4a89c0c05827e18a1ed9cf9bf8a17f7705fcc793fec935b637867e2a5a9b8168c885554d029919117a18ba69; passport_csrf_token=f61602fc63757ae0e4fd9d6bdcee4810;" + '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/', + 'accept-encoding': None, + 'Cookie': f"msToken={self.utils.generate_random_str(107)}; ttwid={self.utils.getttwid()}; odin_tt=324fb4ea4a89c0c05827e18a1ed9cf9bf8a17f7705fcc793fec935b637867e2a5a9b8168c885554d029919117a18ba69; passport_csrf_token=f61602fc63757ae0e4fd9d6bdcee4810;" } # 用于设置重复请求某个接口的最大时间 self.timeout = 10 @@ -71,7 +71,6 @@ class TikTok(object): # self.done_event = Event() # signal.signal(signal.SIGINT, self.handle_sigint) - # 从分享链接中提取网址 def getShareLink(self, string): # findall() 查找匹配正则表达式的字符串 @@ -145,7 +144,6 @@ class TikTok(object): return key_type, key - def getAwemeInfoApi(self, aweme_id): if aweme_id is None: return None @@ -205,7 +203,7 @@ class TikTok(object): if end - start > self.timeout: # raise RuntimeError("重复请求该接口" + str(self.timeout) + "s, 仍然未获取到数据") print("[ 提示 ]:重复请求该接口" + str(self.timeout) + "s, 仍然未获取到数据") - return {},{} + return {}, {} # print("[ 警告 ]:接口未返回数据, 正在重新请求!\r") # 清空self.awemeDict @@ -225,7 +223,6 @@ class TikTok(object): return self.result.awemeDict, datadict - def getUserInfoApi(self, sec_uid, mode="post", count=35, max_cursor=0): if sec_uid is None: return None @@ -346,10 +343,10 @@ class TikTok(object): awemeList.append(copy.deepcopy(self.result.awemeDict)) if numflag: - number-=1 - if number==0: + number -= 1 + if number == 0: break - if numflag and number==0: + if numflag and number == 0: print("\r\n[ 提示 ]: [主页] 下指定数量作品数据获取完成...\r\n") break @@ -399,7 +396,8 @@ class TikTok(object): self.result.liveDict["cover"] = live_json['data']['data'][0]['cover']['url_list'][0] # 头像 - self.result.liveDict["avatar"] = live_json['data']['data'][0]['owner']['avatar_thumb']['url_list'][0].replace("100x100", "1080x1080") + self.result.liveDict["avatar"] = live_json['data']['data'][0]['owner']['avatar_thumb']['url_list'][0].replace( + "100x100", "1080x1080") # 观看人数 self.result.liveDict["user_count"] = live_json['data']['data'][0]['user_count_str'] @@ -419,19 +417,20 @@ class TikTok(object): try: # 分区 self.result.liveDict["partition"] = live_json['data']['partition_road_map']['partition']['title'] - self.result.liveDict["sub_partition"] = live_json['data']['partition_road_map']['sub_partition']['partition'][ + self.result.liveDict["sub_partition"] = \ + live_json['data']['partition_road_map']['sub_partition']['partition'][ 'title'] except Exception as e: self.result.liveDict["partition"] = '无' self.result.liveDict["sub_partition"] = '无' - flv = [] for i, f in enumerate(self.result.liveDict["flv_pull_url"].keys()): flv.append(f) - self.result.liveDict["flv_pull_url0"] = self.result.liveDict["flv_pull_url"][flv[0]].replace("http://", "https://") + self.result.liveDict["flv_pull_url0"] = self.result.liveDict["flv_pull_url"][flv[0]].replace("http://", + "https://") return self.result.liveDict, live_json @@ -477,7 +476,8 @@ class TikTok(object): self.result.liveDict["cover"] = live_json['data']['data'][0]['cover']['url_list'][0] # 头像 - self.result.liveDict["avatar"] = live_json['data']['data'][0]['owner']['avatar_thumb']['url_list'][0].replace("100x100", "1080x1080") + self.result.liveDict["avatar"] = live_json['data']['data'][0]['owner']['avatar_thumb']['url_list'][0].replace( + "100x100", "1080x1080") # 观看人数 self.result.liveDict["user_count"] = live_json['data']['data'][0]['user_count_str'] @@ -497,7 +497,8 @@ class TikTok(object): try: # 分区 self.result.liveDict["partition"] = live_json['data']['partition_road_map']['partition']['title'] - self.result.liveDict["sub_partition"] = live_json['data']['partition_road_map']['sub_partition']['partition'][ + self.result.liveDict["sub_partition"] = \ + live_json['data']['partition_road_map']['sub_partition']['partition'][ 'title'] except Exception as e: self.result.liveDict["partition"] = '无' @@ -516,7 +517,8 @@ class TikTok(object): rate = int(input('[ 🎬 ]输入数字选择推流清晰度:')) - self.result.liveDict["flv_pull_url0"] = self.result.liveDict["flv_pull_url"][flv[rate]].replace("http://", "https://") + self.result.liveDict["flv_pull_url0"] = self.result.liveDict["flv_pull_url"][flv[rate]].replace("http://", + "https://") # 显示清晰度列表 print('[ %s ]:%s' % (flv[rate], self.result.liveDict["flv_pull_url"][flv[rate]])) @@ -544,7 +546,6 @@ class TikTok(object): if end - start > self.timeout: return None - for aweme in datadict["aweme_list"]: # 清空self.awemeDict @@ -672,13 +673,12 @@ class TikTok(object): return None for mix in datadict["mix_infos"]: - mixIdNameDict={} + mixIdNameDict = {} mixIdNameDict["https://www.douyin.com/collection/" + mix["mix_id"]] = mix["mix_name"] mixIdlist.append(mixIdNameDict) return mixIdlist, datadict, datadict["cursor"], datadict["has_more"] - def getUserAllMixInfo(self, sec_uid, count=35, number=0): print('[ 提示 ]:正在请求的用户 id = %s\r\n' % sec_uid) if sec_uid is None: @@ -762,7 +762,6 @@ class TikTok(object): if end - start > self.timeout: return None - for aweme in datadict["aweme_list"]: # 清空self.awemeDict self.result.clearDict(self.result.awemeDict) @@ -924,7 +923,7 @@ class TikTok(object): try: # 使用作品 创建时间+描述 当文件夹 - file_name = awemeDict["create_time"] + "_" + self.utils.replaceStr(awemeDict["desc"]) + file_name = awemeDict["create_time"] + "_" + self.utils.replaceStr(awemeDict["desc"]) aweme_path = os.path.join(savePath, file_name) if not os.path.exists(aweme_path): os.mkdir(aweme_path) @@ -937,7 +936,7 @@ class TikTok(object): f.write(json.dumps(awemeDict, ensure_ascii=False, indent=2)) f.close() except Exception as e: - print("[ 错误 ]:保存 result.json 失败... 作品名: " + file_name +"\r\n") + print("[ 错误 ]:保存 result.json 失败... 作品名: " + file_name + "\r\n") desc = file_name[:30] # 下载 视频 @@ -958,7 +957,7 @@ class TikTok(object): self.alltask.append( self.pool.submit(self.progressBarDownload, url, video_path, "[ 视频 ]:" + desc)) except Exception as e: - print("[ 警告 ]:视频下载失败,请重试... 作品名: " + file_name +"\r\n") + print("[ 警告 ]:视频下载失败,请重试... 作品名: " + file_name + "\r\n") # 下载 图集 if awemeDict["awemeType"] == 1: @@ -978,7 +977,7 @@ class TikTok(object): self.alltask.append( self.pool.submit(self.progressBarDownload, url, image_path, "[ 图集 ]:" + desc)) except Exception as e: - print("[ 警告 ]:图片下载失败,请重试... 作品名: " + file_name +"\r\n") + print("[ 警告 ]:图片下载失败,请重试... 作品名: " + file_name + "\r\n") # 下载 音乐 if music: @@ -999,7 +998,7 @@ class TikTok(object): self.alltask.append( self.pool.submit(self.progressBarDownload, url, music_path, "[ 原声 ]:" + desc)) except Exception as e: - print("[ 警告 ]:音乐(原声)下载失败,请重试... 作品名: " + file_name +"\r\n") + print("[ 警告 ]:音乐(原声)下载失败,请重试... 作品名: " + file_name + "\r\n") # 下载 cover if cover and awemeDict["awemeType"] == 0: @@ -1019,7 +1018,7 @@ class TikTok(object): self.alltask.append( self.pool.submit(self.progressBarDownload, url, cover_path, "[ 封面 ]:" + desc)) except Exception as e: - print("[ 警告 ]:cover下载失败,请重试... 作品名: " + file_name +"\r\n") + print("[ 警告 ]:cover下载失败,请重试... 作品名: " + file_name + "\r\n") # 下载 avatar if avatar: @@ -1039,7 +1038,7 @@ class TikTok(object): self.alltask.append( self.pool.submit(self.progressBarDownload, url, avatar_path, "[ 头像 ]:" + desc)) except Exception as e: - print("[ 警告 ]:avatar下载失败,请重试... 作品名: " + file_name +"\r\n") + print("[ 警告 ]:avatar下载失败,请重试... 作品名: " + file_name + "\r\n") except Exception as e: print("[ 错误 ]:下载作品时出错\r\n") @@ -1093,8 +1092,8 @@ class TikTok(object): # end = time.time() # 结束时间 # print('\n' + '[下载完成]:耗时: %d分钟%d秒\n' % (int((end - start) / 60), ((end - start) % 60))) # 输出下载用时时间 - - def userDownload(self, awemeList: list, music=True, cover=True, avatar=True, resjson=True, savePath=os.getcwd(), thread=5): + def userDownload(self, awemeList: list, music=True, cover=True, avatar=True, resjson=True, savePath=os.getcwd(), + thread=5): if awemeList is None: return if not os.path.exists(savePath): @@ -1106,7 +1105,8 @@ class TikTok(object): start = time.time() # 开始时间 for aweme in awemeList: - self.awemeDownload(awemeDict=aweme, music=music, cover=cover, avatar=avatar, resjson=resjson, savePath=savePath) + self.awemeDownload(awemeDict=aweme, music=music, cover=cover, avatar=avatar, resjson=resjson, + savePath=savePath) # time.sleep(0.5) wait(self.alltask, return_when=ALL_COMPLETED) @@ -1116,7 +1116,8 @@ class TikTok(object): self.isdwownload = True # 下载上一步失败的 for aweme in awemeList: - self.awemeDownload(awemeDict=aweme, music=music, cover=cover, avatar=avatar, resjson=resjson, savePath=savePath) + self.awemeDownload(awemeDict=aweme, music=music, cover=cover, avatar=avatar, resjson=resjson, + savePath=savePath) # time.sleep(0.5) wait(self.alltask, return_when=ALL_COMPLETED) @@ -1126,5 +1127,6 @@ class TikTok(object): end = time.time() # 结束时间 print('\n' + '[下载完成]:耗时: %d分钟%d秒\n' % (int((end - start) / 60), ((end - start) % 60))) # 输出下载用时时间 + if __name__ == "__main__": pass diff --git a/TikTokCommand.py b/TikTokCommand.py index dc364a5..6abc98c 100644 --- a/TikTokCommand.py +++ b/TikTokCommand.py @@ -42,6 +42,7 @@ configModel = { } + def argument(): parser = argparse.ArgumentParser(description='抖音批量下载工具 使用帮助') parser.add_argument("--cmd", "-C", help="使用命令行(True)或者配置文件(False), 默认为False", @@ -50,7 +51,7 @@ def argument(): help="作品(视频或图集)、直播、合集、音乐集合、个人主页的分享链接或者电脑浏览器网址, 可以设置多个链接(删除文案, 保证只有URL, https://v.douyin.com/kcvMpuN/ 或者 https://www.douyin.com/开头的)", type=str, required=False, default=[], action="append") parser.add_argument("--path", "-p", help="下载保存位置, 默认当前文件位置", - type=str, required=False,default=os.getcwd()) + type=str, required=False, default=os.getcwd()) parser.add_argument("--music", "-m", help="是否下载视频中的音乐(True/False), 默认为True", type=Utils().str2bool, required=False, default=True) parser.add_argument("--cover", "-c", help="是否下载视频的封面(True/False), 默认为True, 当下载视频时有效", @@ -82,12 +83,13 @@ def argument(): return args + def yamlConfig(): curPath = os.path.dirname(os.path.realpath(sys.argv[0])) yamlPath = os.path.join(curPath, "config.yml") f = open(yamlPath, 'r', encoding='utf-8') cfg = f.read() - configDict = yaml.load(stream=cfg,Loader=yaml.FullLoader) + configDict = yaml.load(stream=cfg, Loader=yaml.FullLoader) try: if configDict["link"] != None: @@ -159,7 +161,7 @@ def yamlConfig(): cookiekey = configDict["cookies"].keys() cookieStr = "" for i in cookiekey: - cookieStr = cookieStr + i + "=" + configDict["cookies"][i] + "; " + cookieStr = cookieStr + i + "=" + configDict["cookies"][i] + "; " configModel["cookie"] = cookieStr except Exception as e: pass @@ -184,7 +186,7 @@ def main(): configModel["avatar"] = args.avatar configModel["json"] = args.json if args.mode == None or args.mode == []: - args.mode=[] + args.mode = [] args.mode.append("post") configModel["mode"] = list(set(args.mode)) configModel["number"]["post"] = args.postnumber @@ -215,7 +217,7 @@ def main(): key_type, key = tk.getKey(url) if key_type == "user": print("[ 提示 ]:正在请求用户主页下作品\r\n") - userPath = os.path.join(configModel["path"], "user_"+key) + userPath = os.path.join(configModel["path"], "user_" + key) if not os.path.exists(userPath): os.mkdir(userPath) @@ -242,13 +244,15 @@ def main(): modePath = os.path.join(userPath, mode) if not os.path.exists(modePath): os.mkdir(modePath) - tk.userDownload(awemeList=datalist, music=configModel["music"], cover=configModel["cover"], + tk.userDownload(awemeList=datalist, music=configModel["music"], + cover=configModel["cover"], avatar=configModel["avatar"], resjson=configModel["json"], - savePath=os.path.join(modePath, mix_file_name), thread=configModel["thread"]) + savePath=os.path.join(modePath, mix_file_name), + thread=configModel["thread"]) print(f'[ 提示 ]:合集 [{mixIdNameDict[mix_id]}] 中的作品下载完成\r\n') elif key_type == "mix": print("[ 提示 ]:正在请求单个合集下作品\r\n") - datalist = tk.getMixInfo(key,35, configModel["number"]["mix"]) + datalist = tk.getMixInfo(key, 35, configModel["number"]["mix"]) if datalist is not None and datalist != []: mixPath = os.path.join(configModel["path"], "mix_" + key) if not os.path.exists(mixPath): @@ -258,7 +262,7 @@ def main(): savePath=mixPath, thread=configModel["thread"]) elif key_type == "music": print("[ 提示 ]:正在请求音乐(原声)下作品\r\n") - datalist = tk.getMusicInfo(key,35, configModel["number"]["music"]) + datalist = tk.getMusicInfo(key, 35, configModel["number"]["music"]) if datalist is not None and datalist != []: musicPath = os.path.join(configModel["path"], "music_" + key) if not os.path.exists(musicPath): @@ -281,19 +285,20 @@ def main(): elif key_type == "live": print("[ 提示 ]:正在进行直播解析\r\n") live_json = tk.getLiveInfo(key) - if configModel["json"]: + if configModel["json"]: livePath = os.path.join(configModel["path"], "live") if not os.path.exists(livePath): os.mkdir(livePath) live_file_name = utils.replaceStr(key + live_json["nickname"]) # 保存获取到json print("[ 提示 ]:正在保存获取到的信息到result.json\r\n") - with open(os.path.join(livePath, live_file_name + ".json"), "w", encoding='utf-8') as f: + with open(os.path.join(livePath, live_file_name + ".json"), "w", encoding='utf-8') as f: f.write(json.dumps(live_json, ensure_ascii=False, indent=2)) f.close() end = time.time() # 结束时间 print('\n' + '[下载完成]:总耗时: %d分钟%d秒\n' % (int((end - start) / 60), ((end - start) % 60))) # 输出下载用时时间 + if __name__ == "__main__": main() diff --git a/TikTokResult.py b/TikTokResult.py index a3e69e2..b13837d 100644 --- a/TikTokResult.py +++ b/TikTokResult.py @@ -1,283 +1,284 @@ -#!/usr/bin/env python -# -*- encoding: utf-8 -*- - -''' -@Description:TikTok.py -@Date :2023/02/11 13:06:23 -@Author :imgyh -@version :1.0 -@Github :https://github.com/imgyh -@Mail :admin@imgyh.com -------------------------------------------------- -Change Log : -------------------------------------------------- -''' - -import time -import copy - -class Result(object): - def __init__(self): - # 作者信息 - self.authorDict = { - "avatar_thumb": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - "avatar": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - "cover_url": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - # 喜欢的作品数 - "favoriting_count": "", - # 粉丝数 - "follower_count": "", - # 关注数 - "following_count": "", - # 昵称 - "nickname": "", - # 是否允许下载 - "prevent_download": "", - # 用户 url id - "sec_uid": "", - # 是否私密账号 - "secret": "", - # 短id - "short_id": "", - # 签名 - "signature": "", - # 总获赞数 - "total_favorited": "", - # 用户id - "uid": "", - # 用户自定义唯一id 抖音号 - "unique_id": "", - # 年龄 - "user_age": "", - - } - # 图片信息 - self.picDict = { - "height": "", - "mask_url_list": "", - "uri": "", - "url_list": [], - "width": "" - } - # 音乐信息 - self.musicDict = { - "cover_hd": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - "cover_large": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - "cover_medium": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - "cover_thumb": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - # 音乐作者抖音号 - "owner_handle": "", - # 音乐作者id - "owner_id": "", - # 音乐作者昵称 - "owner_nickname": "", - "play_url": { - "height": "", - "uri": "", - "url_key": "", - "url_list": [], - "width": "" - }, - # 音乐名字 - "title": "", - } - # 视频信息 - self.videoDict = { - "play_addr": { - "uri": "", - "url_list": "", - }, - "cover_original_scale": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - "dynamic_cover": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - "origin_cover": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - }, - "cover": { - "height": "", - "uri": "", - "url_list": [], - "width": "" - } - } - # 作品信息 - self.awemeDict = { - # 作品创建时间 - "create_time": "", - # awemeType=0 视频, awemeType=1 图集, awemeType=2 直播 - "awemeType": "", - # 作品 id - "aweme_id": "", - # 作者信息 - "author": self.authorDict, - # 作品描述 - "desc": "", - # 图片 - "images": [], - # 音乐 - "music": self.musicDict, - # 视频 - "video": self.videoDict, - # 作品信息统计 - "statistics": { - "admire_count": "", - "collect_count": "", - "comment_count": "", - "digg_count": "", - "play_count": "", - "share_count": "" - } - } - # 用户作品信息 - self.awemeList = [] - # 直播信息 - self.liveDict = { - # awemeType=0 视频, awemeType=1 图集, awemeType=2 直播 - "awemeType": "", - # 是否在播 - "status": "", - # 直播标题 - "title": "", - # 直播cover - "cover": "", - # 头像 - "avatar": "", - # 观看人数 - "user_count": "", - # 昵称 - "nickname": "", - # sec_uid - "sec_uid": "", - # 直播间观看状态 - "display_long": "", - # 推流 - "flv_pull_url": "", - # 分区 - "partition": "", - "sub_partition": "", - # 最清晰的地址 - "flv_pull_url0": "", - } - - # 将得到的json数据(dataRaw)精简成自己定义的数据(dataNew) - # 转换得到的数据 - def dataConvert(self, awemeType, dataNew, dataRaw): - for item in dataNew: - try: - # 作品创建时间 - if item == "create_time": - dataNew['create_time'] = time.strftime( - "%Y-%m-%d %H.%M.%S", time.localtime(dataRaw['create_time'])) - continue - # 设置 awemeType - if item == "awemeType": - dataNew["awemeType"] = awemeType - continue - # 当 解析的链接 是图片时 - if item == "images": - if awemeType == 1: - for image in dataRaw[item]: - for i in image: - self.picDict[i] = image[i] - # 字典要深拷贝 - self.awemeDict["images"].append(copy.deepcopy(self.picDict)) - continue - # 当 解析的链接 是视频时 - if item == "video": - if awemeType == 0: - self.dataConvert(awemeType, dataNew[item], dataRaw[item]) - continue - # 将小头像放大 - if item == "avatar": - for i in dataNew[item]: - if i == "url_list": - for j in self.awemeDict["author"]["avatar_thumb"]["url_list"]: - dataNew[item][i].append(j.replace("100x100", "1080x1080")) - elif i == "uri": - dataNew[item][i] = self.awemeDict["author"]["avatar_thumb"][i].replace("100x100", - "1080x1080") - else: - dataNew[item][i] = self.awemeDict["author"]["avatar_thumb"][i] - continue - - # 原来的json是[{}] 而我们的是 {} - if item == "cover_url": - self.dataConvert(awemeType, dataNew[item], dataRaw[item][0]) - continue - - # 根据 uri 获取 1080p 视频 - if item == "play_addr": - dataNew[item]["uri"] = dataRaw["bit_rate"][0]["play_addr"]["uri"] - # 使用 这个api 可以获得1080p - # dataNew[item]["url_list"] = "https://aweme.snssdk.com/aweme/v1/play/?video_id=%s&ratio=1080p&line=0" \ - # % dataNew[item]["uri"] - dataNew[item]["url_list"] = dataRaw["bit_rate"][0]["play_addr"]["url_list"][0] - continue - - # 常规 递归遍历 字典 - if isinstance(dataNew[item], dict): - self.dataConvert(awemeType, dataNew[item], dataRaw[item]) - else: - # 赋值 - dataNew[item] = dataRaw[item] - except Exception as e: - # 删除这个警告, 总是让人误会出错了 - # print("[ 警告 ]:转换数据时在接口中未找到 %s\r" % (item)) - pass - - def clearDict(self, data): - for item in data: - # 常规 递归遍历 字典 - if isinstance(data[item], dict): - self.clearDict(data[item]) - elif isinstance(data[item], list): - data[item] = [] - else: - data[item] = "" \ No newline at end of file +#!/usr/bin/env python +# -*- encoding: utf-8 -*- + +''' +@Description:TikTok.py +@Date :2023/02/11 13:06:23 +@Author :imgyh +@version :1.0 +@Github :https://github.com/imgyh +@Mail :admin@imgyh.com +------------------------------------------------- +Change Log : +------------------------------------------------- +''' + +import time +import copy + + +class Result(object): + def __init__(self): + # 作者信息 + self.authorDict = { + "avatar_thumb": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + "avatar": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + "cover_url": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + # 喜欢的作品数 + "favoriting_count": "", + # 粉丝数 + "follower_count": "", + # 关注数 + "following_count": "", + # 昵称 + "nickname": "", + # 是否允许下载 + "prevent_download": "", + # 用户 url id + "sec_uid": "", + # 是否私密账号 + "secret": "", + # 短id + "short_id": "", + # 签名 + "signature": "", + # 总获赞数 + "total_favorited": "", + # 用户id + "uid": "", + # 用户自定义唯一id 抖音号 + "unique_id": "", + # 年龄 + "user_age": "", + + } + # 图片信息 + self.picDict = { + "height": "", + "mask_url_list": "", + "uri": "", + "url_list": [], + "width": "" + } + # 音乐信息 + self.musicDict = { + "cover_hd": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + "cover_large": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + "cover_medium": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + "cover_thumb": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + # 音乐作者抖音号 + "owner_handle": "", + # 音乐作者id + "owner_id": "", + # 音乐作者昵称 + "owner_nickname": "", + "play_url": { + "height": "", + "uri": "", + "url_key": "", + "url_list": [], + "width": "" + }, + # 音乐名字 + "title": "", + } + # 视频信息 + self.videoDict = { + "play_addr": { + "uri": "", + "url_list": "", + }, + "cover_original_scale": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + "dynamic_cover": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + "origin_cover": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + }, + "cover": { + "height": "", + "uri": "", + "url_list": [], + "width": "" + } + } + # 作品信息 + self.awemeDict = { + # 作品创建时间 + "create_time": "", + # awemeType=0 视频, awemeType=1 图集, awemeType=2 直播 + "awemeType": "", + # 作品 id + "aweme_id": "", + # 作者信息 + "author": self.authorDict, + # 作品描述 + "desc": "", + # 图片 + "images": [], + # 音乐 + "music": self.musicDict, + # 视频 + "video": self.videoDict, + # 作品信息统计 + "statistics": { + "admire_count": "", + "collect_count": "", + "comment_count": "", + "digg_count": "", + "play_count": "", + "share_count": "" + } + } + # 用户作品信息 + self.awemeList = [] + # 直播信息 + self.liveDict = { + # awemeType=0 视频, awemeType=1 图集, awemeType=2 直播 + "awemeType": "", + # 是否在播 + "status": "", + # 直播标题 + "title": "", + # 直播cover + "cover": "", + # 头像 + "avatar": "", + # 观看人数 + "user_count": "", + # 昵称 + "nickname": "", + # sec_uid + "sec_uid": "", + # 直播间观看状态 + "display_long": "", + # 推流 + "flv_pull_url": "", + # 分区 + "partition": "", + "sub_partition": "", + # 最清晰的地址 + "flv_pull_url0": "", + } + + # 将得到的json数据(dataRaw)精简成自己定义的数据(dataNew) + # 转换得到的数据 + def dataConvert(self, awemeType, dataNew, dataRaw): + for item in dataNew: + try: + # 作品创建时间 + if item == "create_time": + dataNew['create_time'] = time.strftime( + "%Y-%m-%d %H.%M.%S", time.localtime(dataRaw['create_time'])) + continue + # 设置 awemeType + if item == "awemeType": + dataNew["awemeType"] = awemeType + continue + # 当 解析的链接 是图片时 + if item == "images": + if awemeType == 1: + for image in dataRaw[item]: + for i in image: + self.picDict[i] = image[i] + # 字典要深拷贝 + self.awemeDict["images"].append(copy.deepcopy(self.picDict)) + continue + # 当 解析的链接 是视频时 + if item == "video": + if awemeType == 0: + self.dataConvert(awemeType, dataNew[item], dataRaw[item]) + continue + # 将小头像放大 + if item == "avatar": + for i in dataNew[item]: + if i == "url_list": + for j in self.awemeDict["author"]["avatar_thumb"]["url_list"]: + dataNew[item][i].append(j.replace("100x100", "1080x1080")) + elif i == "uri": + dataNew[item][i] = self.awemeDict["author"]["avatar_thumb"][i].replace("100x100", + "1080x1080") + else: + dataNew[item][i] = self.awemeDict["author"]["avatar_thumb"][i] + continue + + # 原来的json是[{}] 而我们的是 {} + if item == "cover_url": + self.dataConvert(awemeType, dataNew[item], dataRaw[item][0]) + continue + + # 根据 uri 获取 1080p 视频 + if item == "play_addr": + dataNew[item]["uri"] = dataRaw["bit_rate"][0]["play_addr"]["uri"] + # 使用 这个api 可以获得1080p + # dataNew[item]["url_list"] = "https://aweme.snssdk.com/aweme/v1/play/?video_id=%s&ratio=1080p&line=0" \ + # % dataNew[item]["uri"] + dataNew[item]["url_list"] = dataRaw["bit_rate"][0]["play_addr"]["url_list"][0] + continue + + # 常规 递归遍历 字典 + if isinstance(dataNew[item], dict): + self.dataConvert(awemeType, dataNew[item], dataRaw[item]) + else: + # 赋值 + dataNew[item] = dataRaw[item] + except Exception as e: + # 删除这个警告, 总是让人误会出错了 + # print("[ 警告 ]:转换数据时在接口中未找到 %s\r" % (item)) + pass + + def clearDict(self, data): + for item in data: + # 常规 递归遍历 字典 + if isinstance(data[item], dict): + self.clearDict(data[item]) + elif isinstance(data[item], list): + data[item] = [] + else: + data[item] = "" diff --git a/TikTokTest.py b/TikTokTest.py index c43a65b..57847bb 100644 --- a/TikTokTest.py +++ b/TikTokTest.py @@ -1,101 +1,109 @@ -#!/usr/bin/env python -# -*- encoding: utf-8 -*- - -''' -@Description:TikTok.py -@Date :2023/02/11 13:06:23 -@Author :imgyh -@version :1.0 -@Github :https://github.com/imgyh -@Mail :admin@imgyh.com -------------------------------------------------- -Change Log : -------------------------------------------------- -''' -import TikTokUtils -from TikTok import TikTok - -def getAwemeInfo(): - share_link_video = "3.56 uSy:/ 复制打开抖音,看看【小透明的作品】没有女朋友就用我的吧哈哈哈哈 # 表情包锁屏 https://v.douyin.com/BugmVVD/" - share_link_pic = "8.20 MJI:/ 复制打开抖音,看看【舍溪的图文作品】我又来放图集啦~还有你们要的小可爱大图也放啦~# ... https://v.douyin.com/BugrFTN/" - tk = TikTok() - - url = tk.getShareLink(share_link_pic) - key_type, key = tk.getKey(url) - datanew, dataraw = tk.getAwemeInfo(key) - print(datanew) - -def getUserInfo(): - share_link_post = "1- 长按复制此条消息,打开抖音搜索,查看TA的更多作品。 https://v.douyin.com/BupCppt/" - share_link_like = "2- 长按复制此条消息,打开抖音搜索,查看TA的更多作品。 https://v.douyin.com/BusJrfr/" - tk = TikTok() - - url = tk.getShareLink(share_link_like) - key_type, key = tk.getKey(url) - awemeList = tk.getUserInfo(key, mode="like", count=35) - print(awemeList) - -def getLiveInfo(): - live_link = "https://live.douyin.com/40768897856" - tk = TikTok() - - url = tk.getShareLink(live_link) - key_type, key = tk.getKey(url) - live_json = tk.getLiveInfo(key) - print(live_json) - -def getMixInfo(): - mix_link = 'https://v.douyin.com/B3J63Le/' - tk = TikTok() - - url = tk.getShareLink(mix_link) - key_type, key = tk.getKey(url) - awemeList = tk.getMixInfo(key, count=35) - print(len(awemeList)) - -def getUserAllMixInfo(): - user_all_mix_link = 'https://v.douyin.com/B38oovu/' - tk = TikTok() - - url = tk.getShareLink(user_all_mix_link) - key_type, key = tk.getKey(url) - mixIdNameDict = tk.getUserAllMixInfo(key, count=35) - print(mixIdNameDict) - -def getMusicInfo(): - music_link = 'https://v.douyin.com/S6YMNXs/' - tk = TikTok() - - url = tk.getShareLink(music_link) - key_type, key = tk.getKey(url) - awemeList = tk.getMusicInfo(key,count=35) - print(len(awemeList)) - -def test(): - utils=TikTokUtils.Utils() - user_all_mix_link = 'https://www.douyin.com/aweme/v1/web/aweme/favorite/?'+\ - utils.getXbogus(url='device_platform=webapp&aid=6383&sec_user_id=MS4wLjABAAAAjQn6ONfaGgUpk0Q1ep8dPiD3W4T_lxTJmemfy3MTJ64&max_cursor=1676441180000&count=10') - 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', - 'referer': 'https://www.douyin.com/', - 'accept-encoding': None, - 'Cookie': 'ttwid=1|oLudm-Hi5ikxQQhmAnv4Km4LjwwvLa4Qk_JGrKffuYU|1681878460|b0f581d97797bb67d2260bf92d65e8808b90713afc08bf5d8100f571fd70a275; passport_csrf_token=570d671dcd8d8598fcde4c3f7c99664d; passport_csrf_token_default=570d671dcd8d8598fcde4c3f7c99664d; s_v_web_id=verify_lgn70gzw_sAVtjJdD_Clyk_43UL_8M9L_6JiTtPC2TyU0; n_mh=vrLGYVtwqutbPLOGNTDUGahwaD9AyYjn4iVvAO2Xt0s; sso_uid_tt=92e014dfb6653bdc319ecc6a6ceea870; sso_uid_tt_ss=92e014dfb6653bdc319ecc6a6ceea870; toutiao_sso_user=d0bd8d5cc0f75420799903572941ce83; toutiao_sso_user_ss=d0bd8d5cc0f75420799903572941ce83; passport_auth_status=5c576b088f4a19448eb4efd7aaeb7c5e,; passport_auth_status_ss=5c576b088f4a19448eb4efd7aaeb7c5e,; uid_tt=564215aecc985785d033c9e4c9d00fc4; uid_tt_ss=564215aecc985785d033c9e4c9d00fc4; sid_tt=f14390d924c81856fde84b1cd534bf09; sessionid=f14390d924c81856fde84b1cd534bf09; sessionid_ss=f14390d924c81856fde84b1cd534bf09; odin_tt=0aa1c100d17bf3541dce9e8dd62c6353302b8abaaa0a5998d5c437313feb245d8f7b6391a69772e4afa7d3c718878837; passport_assist_user=CjwC3fi9JyApG4DN-HiFI6n5FcGAdzNDT29nR-nSDJYAVfqh2BYI77qdTN2GRfgagkYLRi1wxNp1akHBwGcaSAo8XDkNT-UWnkFmc_0eXMYCb8OIh4G_YnH8pylwwdfS-7PCTekX3trj0JyENUVWLWFxnKuG5HhSS4FB2CtxEJ7yrg0Yia_WVCIBA9bY7IU=; sid_ucp_sso_v1=1.0.0-KDc0NmY3NzA5ZWFhNzI2YzMyOWMxMDMwNjAwMDg3YzRjYzg3ZjlhODcKHQjG-7zT9QEQ5dv9oQYY7zEgDDCL_7nMBTgGQPQHGgJscSIgZDBiZDhkNWNjMGY3NTQyMDc5OTkwMzU3Mjk0MWNlODM; ssid_ucp_sso_v1=1.0.0-KDc0NmY3NzA5ZWFhNzI2YzMyOWMxMDMwNjAwMDg3YzRjYzg3ZjlhODcKHQjG-7zT9QEQ5dv9oQYY7zEgDDCL_7nMBTgGQPQHGgJscSIgZDBiZDhkNWNjMGY3NTQyMDc5OTkwMzU3Mjk0MWNlODM; publish_badge_show_info="0,0,0,1681878505305"; LOGIN_STATUS=1; store-region=cn-sc; store-region-src=uid; sid_guard=f14390d924c81856fde84b1cd534bf09|1681878504|5183998|Sun,+18-Jun-2023+04:28:22+GMT; sid_ucp_v1=1.0.0-KGY5OGQ3OTQ4YmVkYzczODgzYzM0MmJmOWYxMzhkMDliMTc5NGI3NjMKGQjG-7zT9QEQ6Nv9oQYY7zEgDDgGQPQHSAQaAmxmIiBmMTQzOTBkOTI0YzgxODU2ZmRlODRiMWNkNTM0YmYwOQ; ssid_ucp_v1=1.0.0-KGY5OGQ3OTQ4YmVkYzczODgzYzM0MmJmOWYxMzhkMDliMTc5NGI3NjMKGQjG-7zT9QEQ6Nv9oQYY7zEgDDgGQPQHSAQaAmxmIiBmMTQzOTBkOTI0YzgxODU2ZmRlODRiMWNkNTM0YmYwOQ; download_guide="3/20230419"; pwa2="3|0"; FOLLOW_NUMBER_YELLOW_POINT_INFO="MS4wLjABAAAA-jD2lukp--I21BF8VQsmYUqJDbj3FmU-kGQTHl2y1Cw/1681920000000/0/0/1681900423333"; __ac_nonce=0644234d60042cacebbb6; __ac_signature=_02B4Z6wo00f01eZN0dAAAIDAhUcRuQz7dNHmbdVAAB3TRUC7i8VqXQejR8jv-D8UggBx1MBLH564PE.cCZf00m7Cw640CNUvcc5jJfhgn8u5FhvVndykvwbQb.HEpSfGN-8eqql7GpuGZJ8f1d; strategyABtestKey="1682060514.261"; passport_fe_beating_status=true; csrf_session_id=405b0ec4f6338a5d893a5f14f6fd1a64; bd_ticket_guard_client_data=eyJiZC10aWNrZXQtZ3VhcmQtdmVyc2lvbiI6MiwiYmQtdGlja2V0LWd1YXJkLWl0ZXJhdGlvbi12ZXJzaW9uIjoxLCJiZC10aWNrZXQtZ3VhcmQtY2xpZW50LWNlcnQiOiItLS0tLUJFR0lOIENFUlRJRklDQVRFLS0tLS1cbk1JSUNGRENDQWJxZ0F3SUJBZ0lVR1ljQ3FuQUh0UUJBZm5WWkYxQW84cUtjY2Zrd0NnWUlLb1pJemowRUF3SXdcbk1URUxNQWtHQTFVRUJoTUNRMDR4SWpBZ0JnTlZCQU1NR1hScFkydGxkRjluZFdGeVpGOWpZVjlsWTJSellWOHlcbk5UWXdIaGNOTWpNd016STNNRE15T0RBeldoY05Nek13TXpJM01URXlPREF6V2pBbk1Rc3dDUVlEVlFRR0V3SkRcblRqRVlNQllHQTFVRUF3d1BZbVJmZEdsamEyVjBYMmQxWVhKa01Ga3dFd1lIS29aSXpqMENBUVlJS29aSXpqMERcbkFRY0RRZ0FFNmZ1Z3d0MEJnZUh5akVub1FvNWtXUS9qc2daTTV1YXBiNTQ4KytTV0dRSjMwb2lSTHNtYlVFSUZcblFIYzh3UEthZzZmdXNPTm91WncxOEdNYm5vTStwNk9CdVRDQnRqQU9CZ05WSFE4QkFmOEVCQU1DQmFBd01RWURcblZSMGxCQ293S0FZSUt3WUJCUVVIQXdFR0NDc0dBUVVGQndNQ0JnZ3JCZ0VGQlFjREF3WUlLd1lCQlFVSEF3UXdcbktRWURWUjBPQkNJRUlPV3Y3d01ZUGhoeUNPL2ZwenJGNDJNeEQ4ZGIzN0YyTDgxaW8zVTVlVFpaTUNzR0ExVWRcbkl3UWtNQ0tBSURLbForcU9aRWdTamN4T1RVQjdjeFNiUjIxVGVxVFJnTmQ1bEpkN0lrZURNQmtHQTFVZEVRUVNcbk1CQ0NEbmQzZHk1a2IzVjVhVzR1WTI5dE1Bb0dDQ3FHU000OUJBTUNBMGdBTUVVQ0lCbm9xRDBRbVdUSlNLOVNcbkFZRjJ6YkljYzBsZjFRMDVTUGxXMURDQ0FMVUpBaUVBazJFRWpKdkdFRnl0YzBWbXRoRTA5bFpGeFFkUmlGN21cbk9pdE5IZzBOZUJrPVxuLS0tLS1FTkQgQ0VSVElGSUNBVEUtLS0tLVxuIn0=; msToken=qSfWwAdwksuBS5wmQAvpzUsf2ovkFFKefOLSvZDKA1Z_FX7ith-wCknpVQB08kft4ISWp00GHeQBaPwV9tcWJq6xBC-mPnQKNjBVINeOQGvFtSdsfacWMtWpa8x1RJE=; FOLLOW_LIVE_POINT_INFO="MS4wLjABAAAA-jD2lukp--I21BF8VQsmYUqJDbj3FmU-kGQTHl2y1Cw/1682092800000/0/0/1682061497980"; msToken=jDwC5gjTRXV6hrFvGMG-AkOBXHGrt_Mp5NaltB1upOUm0aQnZ7sy7qlSEn2tQbGHShp2X7ayNMDQQlPekSTV9MkxBv56LR9zepTlYNOoqbH_RdjDvbl-MZDrmui3OEE=; tt_scid=hERl4ibL-B89BGXb9wUfsmVkQh-G2dvL2wI0QvEDYLooFsvwoz.q6J4ZA0WErn0Tb9fb; home_can_add_dy_2_desktop="1"' } - - import requests - - res = requests.get(user_all_mix_link,headers=headers).text - import json - datadict = json.loads(res) - print(datadict["aweme_list"][0]["video"]["bit_rate"]) - print(len(datadict["aweme_list"][0]["video"]["bit_rate"])) - -if __name__ == "__main__": - # test() - # getMusicInfo() - # getUserAllMixInfo() - # getMixInfo() - # getAwemeInfo() - # getUserInfo() - # getLiveInfo() - pass - +#!/usr/bin/env python +# -*- encoding: utf-8 -*- + +''' +@Description:TikTok.py +@Date :2023/02/11 13:06:23 +@Author :imgyh +@version :1.0 +@Github :https://github.com/imgyh +@Mail :admin@imgyh.com +------------------------------------------------- +Change Log : +------------------------------------------------- +''' +import TikTokUtils +from TikTok import TikTok + + +def getAwemeInfo(): + share_link_video = "3.56 uSy:/ 复制打开抖音,看看【小透明的作品】没有女朋友就用我的吧哈哈哈哈 # 表情包锁屏 https://v.douyin.com/BugmVVD/" + share_link_pic = "8.20 MJI:/ 复制打开抖音,看看【舍溪的图文作品】我又来放图集啦~还有你们要的小可爱大图也放啦~# ... https://v.douyin.com/BugrFTN/" + tk = TikTok() + + url = tk.getShareLink(share_link_pic) + key_type, key = tk.getKey(url) + datanew, dataraw = tk.getAwemeInfo(key) + print(datanew) + + +def getUserInfo(): + share_link_post = "1- 长按复制此条消息,打开抖音搜索,查看TA的更多作品。 https://v.douyin.com/BupCppt/" + share_link_like = "2- 长按复制此条消息,打开抖音搜索,查看TA的更多作品。 https://v.douyin.com/BusJrfr/" + tk = TikTok() + + url = tk.getShareLink(share_link_like) + key_type, key = tk.getKey(url) + awemeList = tk.getUserInfo(key, mode="like", count=35) + print(awemeList) + + +def getLiveInfo(): + live_link = "https://live.douyin.com/40768897856" + tk = TikTok() + + url = tk.getShareLink(live_link) + key_type, key = tk.getKey(url) + live_json = tk.getLiveInfo(key) + print(live_json) + + +def getMixInfo(): + mix_link = 'https://v.douyin.com/B3J63Le/' + tk = TikTok() + + url = tk.getShareLink(mix_link) + key_type, key = tk.getKey(url) + awemeList = tk.getMixInfo(key, count=35) + print(len(awemeList)) + + +def getUserAllMixInfo(): + user_all_mix_link = 'https://v.douyin.com/B38oovu/' + tk = TikTok() + + url = tk.getShareLink(user_all_mix_link) + key_type, key = tk.getKey(url) + mixIdNameDict = tk.getUserAllMixInfo(key, count=35) + print(mixIdNameDict) + + +def getMusicInfo(): + music_link = 'https://v.douyin.com/S6YMNXs/' + tk = TikTok() + + url = tk.getShareLink(music_link) + key_type, key = tk.getKey(url) + awemeList = tk.getMusicInfo(key, count=35) + print(len(awemeList)) + + +def test(): + utils = TikTokUtils.Utils() + user_all_mix_link = 'https://www.douyin.com/aweme/v1/web/aweme/favorite/?' + \ + utils.getXbogus( + url='device_platform=webapp&aid=6383&sec_user_id=MS4wLjABAAAAjQn6ONfaGgUpk0Q1ep8dPiD3W4T_lxTJmemfy3MTJ64&max_cursor=1676441180000&count=10') + 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', + 'referer': 'https://www.douyin.com/', + 'accept-encoding': None, + 'Cookie': 'ttwid=1|oLudm-Hi5ikxQQhmAnv4Km4LjwwvLa4Qk_JGrKffuYU|1681878460|b0f581d97797bb67d2260bf92d65e8808b90713afc08bf5d8100f571fd70a275; passport_csrf_token=570d671dcd8d8598fcde4c3f7c99664d; passport_csrf_token_default=570d671dcd8d8598fcde4c3f7c99664d; s_v_web_id=verify_lgn70gzw_sAVtjJdD_Clyk_43UL_8M9L_6JiTtPC2TyU0; n_mh=vrLGYVtwqutbPLOGNTDUGahwaD9AyYjn4iVvAO2Xt0s; sso_uid_tt=92e014dfb6653bdc319ecc6a6ceea870; sso_uid_tt_ss=92e014dfb6653bdc319ecc6a6ceea870; toutiao_sso_user=d0bd8d5cc0f75420799903572941ce83; toutiao_sso_user_ss=d0bd8d5cc0f75420799903572941ce83; passport_auth_status=5c576b088f4a19448eb4efd7aaeb7c5e,; passport_auth_status_ss=5c576b088f4a19448eb4efd7aaeb7c5e,; uid_tt=564215aecc985785d033c9e4c9d00fc4; uid_tt_ss=564215aecc985785d033c9e4c9d00fc4; sid_tt=f14390d924c81856fde84b1cd534bf09; sessionid=f14390d924c81856fde84b1cd534bf09; sessionid_ss=f14390d924c81856fde84b1cd534bf09; odin_tt=0aa1c100d17bf3541dce9e8dd62c6353302b8abaaa0a5998d5c437313feb245d8f7b6391a69772e4afa7d3c718878837; passport_assist_user=CjwC3fi9JyApG4DN-HiFI6n5FcGAdzNDT29nR-nSDJYAVfqh2BYI77qdTN2GRfgagkYLRi1wxNp1akHBwGcaSAo8XDkNT-UWnkFmc_0eXMYCb8OIh4G_YnH8pylwwdfS-7PCTekX3trj0JyENUVWLWFxnKuG5HhSS4FB2CtxEJ7yrg0Yia_WVCIBA9bY7IU=; sid_ucp_sso_v1=1.0.0-KDc0NmY3NzA5ZWFhNzI2YzMyOWMxMDMwNjAwMDg3YzRjYzg3ZjlhODcKHQjG-7zT9QEQ5dv9oQYY7zEgDDCL_7nMBTgGQPQHGgJscSIgZDBiZDhkNWNjMGY3NTQyMDc5OTkwMzU3Mjk0MWNlODM; ssid_ucp_sso_v1=1.0.0-KDc0NmY3NzA5ZWFhNzI2YzMyOWMxMDMwNjAwMDg3YzRjYzg3ZjlhODcKHQjG-7zT9QEQ5dv9oQYY7zEgDDCL_7nMBTgGQPQHGgJscSIgZDBiZDhkNWNjMGY3NTQyMDc5OTkwMzU3Mjk0MWNlODM; publish_badge_show_info="0,0,0,1681878505305"; LOGIN_STATUS=1; store-region=cn-sc; store-region-src=uid; sid_guard=f14390d924c81856fde84b1cd534bf09|1681878504|5183998|Sun,+18-Jun-2023+04:28:22+GMT; sid_ucp_v1=1.0.0-KGY5OGQ3OTQ4YmVkYzczODgzYzM0MmJmOWYxMzhkMDliMTc5NGI3NjMKGQjG-7zT9QEQ6Nv9oQYY7zEgDDgGQPQHSAQaAmxmIiBmMTQzOTBkOTI0YzgxODU2ZmRlODRiMWNkNTM0YmYwOQ; ssid_ucp_v1=1.0.0-KGY5OGQ3OTQ4YmVkYzczODgzYzM0MmJmOWYxMzhkMDliMTc5NGI3NjMKGQjG-7zT9QEQ6Nv9oQYY7zEgDDgGQPQHSAQaAmxmIiBmMTQzOTBkOTI0YzgxODU2ZmRlODRiMWNkNTM0YmYwOQ; download_guide="3/20230419"; pwa2="3|0"; FOLLOW_NUMBER_YELLOW_POINT_INFO="MS4wLjABAAAA-jD2lukp--I21BF8VQsmYUqJDbj3FmU-kGQTHl2y1Cw/1681920000000/0/0/1681900423333"; __ac_nonce=0644234d60042cacebbb6; __ac_signature=_02B4Z6wo00f01eZN0dAAAIDAhUcRuQz7dNHmbdVAAB3TRUC7i8VqXQejR8jv-D8UggBx1MBLH564PE.cCZf00m7Cw640CNUvcc5jJfhgn8u5FhvVndykvwbQb.HEpSfGN-8eqql7GpuGZJ8f1d; strategyABtestKey="1682060514.261"; passport_fe_beating_status=true; csrf_session_id=405b0ec4f6338a5d893a5f14f6fd1a64; bd_ticket_guard_client_data=eyJiZC10aWNrZXQtZ3VhcmQtdmVyc2lvbiI6MiwiYmQtdGlja2V0LWd1YXJkLWl0ZXJhdGlvbi12ZXJzaW9uIjoxLCJiZC10aWNrZXQtZ3VhcmQtY2xpZW50LWNlcnQiOiItLS0tLUJFR0lOIENFUlRJRklDQVRFLS0tLS1cbk1JSUNGRENDQWJxZ0F3SUJBZ0lVR1ljQ3FuQUh0UUJBZm5WWkYxQW84cUtjY2Zrd0NnWUlLb1pJemowRUF3SXdcbk1URUxNQWtHQTFVRUJoTUNRMDR4SWpBZ0JnTlZCQU1NR1hScFkydGxkRjluZFdGeVpGOWpZVjlsWTJSellWOHlcbk5UWXdIaGNOTWpNd016STNNRE15T0RBeldoY05Nek13TXpJM01URXlPREF6V2pBbk1Rc3dDUVlEVlFRR0V3SkRcblRqRVlNQllHQTFVRUF3d1BZbVJmZEdsamEyVjBYMmQxWVhKa01Ga3dFd1lIS29aSXpqMENBUVlJS29aSXpqMERcbkFRY0RRZ0FFNmZ1Z3d0MEJnZUh5akVub1FvNWtXUS9qc2daTTV1YXBiNTQ4KytTV0dRSjMwb2lSTHNtYlVFSUZcblFIYzh3UEthZzZmdXNPTm91WncxOEdNYm5vTStwNk9CdVRDQnRqQU9CZ05WSFE4QkFmOEVCQU1DQmFBd01RWURcblZSMGxCQ293S0FZSUt3WUJCUVVIQXdFR0NDc0dBUVVGQndNQ0JnZ3JCZ0VGQlFjREF3WUlLd1lCQlFVSEF3UXdcbktRWURWUjBPQkNJRUlPV3Y3d01ZUGhoeUNPL2ZwenJGNDJNeEQ4ZGIzN0YyTDgxaW8zVTVlVFpaTUNzR0ExVWRcbkl3UWtNQ0tBSURLbForcU9aRWdTamN4T1RVQjdjeFNiUjIxVGVxVFJnTmQ1bEpkN0lrZURNQmtHQTFVZEVRUVNcbk1CQ0NEbmQzZHk1a2IzVjVhVzR1WTI5dE1Bb0dDQ3FHU000OUJBTUNBMGdBTUVVQ0lCbm9xRDBRbVdUSlNLOVNcbkFZRjJ6YkljYzBsZjFRMDVTUGxXMURDQ0FMVUpBaUVBazJFRWpKdkdFRnl0YzBWbXRoRTA5bFpGeFFkUmlGN21cbk9pdE5IZzBOZUJrPVxuLS0tLS1FTkQgQ0VSVElGSUNBVEUtLS0tLVxuIn0=; msToken=qSfWwAdwksuBS5wmQAvpzUsf2ovkFFKefOLSvZDKA1Z_FX7ith-wCknpVQB08kft4ISWp00GHeQBaPwV9tcWJq6xBC-mPnQKNjBVINeOQGvFtSdsfacWMtWpa8x1RJE=; FOLLOW_LIVE_POINT_INFO="MS4wLjABAAAA-jD2lukp--I21BF8VQsmYUqJDbj3FmU-kGQTHl2y1Cw/1682092800000/0/0/1682061497980"; msToken=jDwC5gjTRXV6hrFvGMG-AkOBXHGrt_Mp5NaltB1upOUm0aQnZ7sy7qlSEn2tQbGHShp2X7ayNMDQQlPekSTV9MkxBv56LR9zepTlYNOoqbH_RdjDvbl-MZDrmui3OEE=; tt_scid=hERl4ibL-B89BGXb9wUfsmVkQh-G2dvL2wI0QvEDYLooFsvwoz.q6J4ZA0WErn0Tb9fb; home_can_add_dy_2_desktop="1"'} + + import requests + + res = requests.get(user_all_mix_link, headers=headers).text + import json + datadict = json.loads(res) + print(datadict["aweme_list"][0]["video"]["bit_rate"]) + print(len(datadict["aweme_list"][0]["video"]["bit_rate"])) + + +if __name__ == "__main__": + # test() + # getMusicInfo() + # getUserAllMixInfo() + # getMixInfo() + # getAwemeInfo() + # getUserInfo() + # getLiveInfo() + pass diff --git a/TikTokUrls.py b/TikTokUrls.py index ccedc78..2536171 100644 --- a/TikTokUrls.py +++ b/TikTokUrls.py @@ -1,89 +1,90 @@ -#!/usr/bin/env python -# -*- encoding: utf-8 -*- - -''' -@Description:TikTok.py -@Date :2023/02/11 13:06:23 -@Author :imgyh -@version :1.0 -@Github :https://github.com/imgyh -@Mail :admin@imgyh.com -------------------------------------------------- -Change Log : -------------------------------------------------- -''' - - -class Urls(object): - def __init__(self): - # https://langyue.cc/APIdocV1.0.html - ######################################### WEB ######################################### - # 首页推荐 - self.TAB_FEED = 'https://www.douyin.com/aweme/v1/web/tab/feed/?' - - # 用户短信息(给多少个用户secid就返回多少的用户信息) - self.USER_SHORT_INFO = 'https://www.douyin.com/aweme/v1/web/im/user/info/?' - - # 用户详细信息 - self.USER_DETAIL = 'https://www.douyin.com/aweme/v1/web/user/profile/other/?' - - # 用户作品 - # cookies 暂时只需要 __ac_signature, s_v_web_id两个参数, 好像会过期 - # url 暂时不需要携带 msToken, X-Bogus, _signature - # 每次返回数据很少 - # self.USER_POST = 'https://m.douyin.com/web/api/v2/aweme/post/?' - # 2023/02/19 失效 - self.USER_POST = 'https://www.douyin.com/aweme/v1/web/aweme/post/?' - - # 作品信息 - self.POST_DETAIL = 'https://www.douyin.com/aweme/v1/web/aweme/detail/?' - - # 用户喜欢A - # 需要 odin_tt - self.USER_FAVORITE_A = 'https://www.douyin.com/aweme/v1/web/aweme/favorite/?' - - # 用户喜欢B - self.USER_FAVORITE_B = 'https://www.iesdouyin.com/web/api/v2/aweme/like/?' - - # 用户历史 - self.USER_HISTORY = 'https://www.douyin.com/aweme/v1/web/history/read/?' - - # 用户收藏 - self.USER_COLLECTION = 'https://www.douyin.com/aweme/v1/web/aweme/listcollection/?' - - # 用户评论 - self.COMMENT = 'https://www.douyin.com/aweme/v1/web/comment/list/?' - - # 首页朋友作品 - self.FRIEND_FEED = 'https://www.douyin.com/aweme/v1/web/familiar/feed/?' - - # 关注用户作品 - self.FOLLOW_FEED = 'https://www.douyin.com/aweme/v1/web/follow/feed/?' - - # 合集下所有作品 - # 只需要X-Bogus - self.USER_MIX = 'https://www.douyin.com/aweme/v1/web/mix/aweme/?' - - # 用户所有合集列表 - # 需要 ttwid - self.USER_MIX_LIST = 'https://www.douyin.com/aweme/v1/web/mix/list/?' - - # 直播 - self.LIVE = 'https://live.douyin.com/webcast/room/web/enter/?' - self.LIVE2 = 'https://webcast.amemv.com/webcast/room/reflow/info/?' - - # 音乐 - self.MUSIC = 'https://www.douyin.com/aweme/v1/web/music/aweme/?' - - # X-Bogus Path - # 60 秒内,请求同一URI累计超过 600 次,封锁IP 300 秒 - # 两个都可以用 - # 服务器在国外 - # self.GET_XB_PATH = 'https://tiktok.199933.xyz/xb' - # 服务器在国内 - self.GET_XB_PATH = 'http://47.115.208.101:9090/xb' - - ####################################################################################### - -if __name__ == '__main__': - Urls() \ No newline at end of file +#!/usr/bin/env python +# -*- encoding: utf-8 -*- + +''' +@Description:TikTok.py +@Date :2023/02/11 13:06:23 +@Author :imgyh +@version :1.0 +@Github :https://github.com/imgyh +@Mail :admin@imgyh.com +------------------------------------------------- +Change Log : +------------------------------------------------- +''' + + +class Urls(object): + def __init__(self): + # https://langyue.cc/APIdocV1.0.html + ######################################### WEB ######################################### + # 首页推荐 + self.TAB_FEED = 'https://www.douyin.com/aweme/v1/web/tab/feed/?' + + # 用户短信息(给多少个用户secid就返回多少的用户信息) + self.USER_SHORT_INFO = 'https://www.douyin.com/aweme/v1/web/im/user/info/?' + + # 用户详细信息 + self.USER_DETAIL = 'https://www.douyin.com/aweme/v1/web/user/profile/other/?' + + # 用户作品 + # cookies 暂时只需要 __ac_signature, s_v_web_id两个参数, 好像会过期 + # url 暂时不需要携带 msToken, X-Bogus, _signature + # 每次返回数据很少 + # self.USER_POST = 'https://m.douyin.com/web/api/v2/aweme/post/?' + # 2023/02/19 失效 + self.USER_POST = 'https://www.douyin.com/aweme/v1/web/aweme/post/?' + + # 作品信息 + self.POST_DETAIL = 'https://www.douyin.com/aweme/v1/web/aweme/detail/?' + + # 用户喜欢A + # 需要 odin_tt + self.USER_FAVORITE_A = 'https://www.douyin.com/aweme/v1/web/aweme/favorite/?' + + # 用户喜欢B + self.USER_FAVORITE_B = 'https://www.iesdouyin.com/web/api/v2/aweme/like/?' + + # 用户历史 + self.USER_HISTORY = 'https://www.douyin.com/aweme/v1/web/history/read/?' + + # 用户收藏 + self.USER_COLLECTION = 'https://www.douyin.com/aweme/v1/web/aweme/listcollection/?' + + # 用户评论 + self.COMMENT = 'https://www.douyin.com/aweme/v1/web/comment/list/?' + + # 首页朋友作品 + self.FRIEND_FEED = 'https://www.douyin.com/aweme/v1/web/familiar/feed/?' + + # 关注用户作品 + self.FOLLOW_FEED = 'https://www.douyin.com/aweme/v1/web/follow/feed/?' + + # 合集下所有作品 + # 只需要X-Bogus + self.USER_MIX = 'https://www.douyin.com/aweme/v1/web/mix/aweme/?' + + # 用户所有合集列表 + # 需要 ttwid + self.USER_MIX_LIST = 'https://www.douyin.com/aweme/v1/web/mix/list/?' + + # 直播 + self.LIVE = 'https://live.douyin.com/webcast/room/web/enter/?' + self.LIVE2 = 'https://webcast.amemv.com/webcast/room/reflow/info/?' + + # 音乐 + self.MUSIC = 'https://www.douyin.com/aweme/v1/web/music/aweme/?' + + # X-Bogus Path + # 60 秒内,请求同一URI累计超过 600 次,封锁IP 300 秒 + # 两个都可以用 + # 服务器在国外 + # self.GET_XB_PATH = 'https://tiktok.199933.xyz/xb' + # 服务器在国内 + self.GET_XB_PATH = 'http://47.115.208.101:9090/xb' + + ####################################################################################### + + +if __name__ == '__main__': + Urls() diff --git a/TikTokUtils.py b/TikTokUtils.py index 91ed986..dcd5b67 100644 --- a/TikTokUtils.py +++ b/TikTokUtils.py @@ -22,6 +22,7 @@ import sys import json from TikTokUrls import Urls + class Utils(object): def __init__(self): pass @@ -52,7 +53,7 @@ class Utils(object): # 去除前后空格 return result - def resource_path(self,relative_path): + def resource_path(self, relative_path): if getattr(sys, 'frozen', False): # 是否Bundle Resource base_path = sys._MEIPASS else: @@ -62,9 +63,10 @@ class Utils(object): def getXbogus(self, url, headers=None): # getXbogus算法开源地址https://github.com/B1gM8c/tiktok user_agent = headers.get( - 'User-Agent') if headers else "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36" + 'User-Agent') if headers else "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/109.0.0.0 Safari/537.36" try: - xbogus = execjs.compile(open(self.resource_path(os.path.join("X-Bogus.js"))).read()).call('sign', url, user_agent) + xbogus = execjs.compile(open(self.resource_path(os.path.join("X-Bogus.js"))).read()).call('sign', url, + user_agent) params = url + "&X-Bogus=" + xbogus except Exception as e: # print('[ 错误 ]:X-Bogus算法异常或者本地没有JS环境') @@ -79,8 +81,6 @@ class Utils(object): return return params - - def str2bool(self, v): if isinstance(v, bool): return v @@ -97,7 +97,7 @@ class Utils(object): data = '{"region":"cn","aid":1768,"needFid":false,"service":"www.ixigua.com","migrate_info":{"ticket":"","source":"node"},"cbUrlProtocol":"https","union":true}' res = requests.post(url=url, data=data) - for i,j in res.cookies.items(): + for i, j in res.cookies.items(): return j diff --git a/TikTokWeb.py b/TikTokWeb.py index 41df28e..6ccab0a 100644 --- a/TikTokWeb.py +++ b/TikTokWeb.py @@ -33,7 +33,8 @@ def work(share_link, max_cursor, mode, cookie): has_more = None if key_type == "user": if mode == 'post' or mode == 'like': - datalist, rawdatalist, cursor, has_more = tk.getUserInfoApi(sec_uid=key, mode=mode, count=35, max_cursor=max_cursor) + datalist, rawdatalist, cursor, has_more = tk.getUserInfoApi(sec_uid=key, mode=mode, count=35, + max_cursor=max_cursor) elif mode == 'mix': datalist, rawdatalist, cursor, has_more = tk.getUserAllMixInfoApi(sec_uid=key, count=35, cursor=max_cursor) elif key_type == "mix": @@ -45,7 +46,7 @@ def work(share_link, max_cursor, mode, cookie): elif key_type == "live": datalist, rawdatalist = tk.getLiveInfoApi(web_rid=key) - datadict={} + datadict = {} if datalist is not None and datalist != []: datadict["data"] = datalist @@ -57,6 +58,7 @@ def work(share_link, max_cursor, mode, cookie): datadict["status_code"] = 500 return datadict + def deal(mode=None): usefuldict = {} if request.headers.get("content_type") == "application/json": @@ -83,10 +85,12 @@ def deal(mode=None): usefuldict["status_code"] = 500 return jsonify(usefuldict) + app = Flask(__name__) # 设置编码 app.config['JSON_AS_ASCII'] = False + def argument(): parser = argparse.ArgumentParser(description='抖音去水印工具 使用帮助') parser.add_argument("--port", "-p", help="Web端口", @@ -95,34 +99,42 @@ def argument(): return args + @app.route("/douyin/music", methods=["POST"]) def douyinMusic(): return deal() + @app.route("/douyin/mix", methods=["POST"]) def douyinMix(): return deal() + @app.route("/douyin/user/mix", methods=["POST"]) def douyinUserMix(): return deal(mode="mix") + @app.route("/douyin/user/like", methods=["POST"]) def douyinUserLike(): return deal(mode="like") + @app.route("/douyin/user/post", methods=["POST"]) def douyinUserPost(): return deal(mode="post") + @app.route("/douyin/aweme", methods=["POST"]) def douyinAweme(): return deal() + @app.route("/douyin/live", methods=["POST"]) def douyinLive(): return deal() + @app.route("/douyin", methods=["POST"]) def douyin(): return deal() diff --git a/config.yml b/config.yml index ba5f389..67e63c0 100644 --- a/config.yml +++ b/config.yml @@ -1,81 +1,81 @@ -####################################### -# 说明: -# 1. 井号(#)为注释 -# 2. 缩进严格对齐,使用空格缩进, 注意有些冒号后面有一个空格, 有些没有空格 -# 3. 请使用英文字符 -# 4. 更多yaml语法请上网查看 -####################################### - - -# 作品(视频或图集)、直播、合集、音乐集合、个人主页的分享链接或者电脑浏览器网址 -# (删除文案, 保证只有URL, https://v.douyin.com/kcvMpuN/ 或者 https://www.douyin.com/开头的) -# 可以设置多个链接, 确保至少一个链接 -# 必选 -link: - - https://live.douyin.com/759547612580 - - https://v.douyin.com/BugmVVD/ - - https://v.douyin.com/BugrFTN/ - - https://v.douyin.com/B72pdU5/ - - https://v.douyin.com/B72QgDw/ - - https://v.douyin.com/AJp8D3f/ - - https://v.douyin.com/B38oovu/ - - https://v.douyin.com/S6YMNXs/ - -# 下载保存位置, 默认当前文件位置 -# 必选 -path: /mnt/c/project/test333 - -# 是否下载视频中的音乐(True/False), 默认为True -# 可选 -music: True - -# 是否下载视频的封面(True/False), 默认为True, 当下载视频时有效 -# 可选 -cover: True - -# 是否下载作者的头像(True/False), 默认为True -# 可选 -avatar: True - -# 是否保存获取到的数据(True/False), 默认为True -# 可选 -json: True - -# link是个人主页时, 设置下载发布的作品(post)或喜欢的作品(like)或者用户所有合集(mix), 默认为post, 可以设置多种模式 -# 可选 -mode: - - post - - like - - mix - -# 下载作品个数设置 -# 可选 -number: - post: 5 # 主页下作品下载个数设置, 默认为0 全部下载 - like: 5 # 主页下喜欢下载个数设置, 默认为0 全部下载 - allmix: 1 # 主页下合集下载个数设置, 默认为0 全部下载 - mix: 5 # 单个合集下作品下载个数设置, 默认为0 全部下载 - music: 5 # 音乐(原声)下作品下载个数设置, 默认为0 全部下载 - -# 设置线程数, 默认5个线程 -# 可选 -thread: 5 - -# cookie 请登录网页抖音后F12查看 -# cookies 和 cookie 二选一, 要使用这种形式, 请注释下面的cookie -# 目前只需要msToken、ttwid、odin_tt、passport_csrf_token、sid_guard -# 可以动态添加, 程序会根据填的键查找,并没有写死, 如果抖音需要更多的cookie自己加上就行了 -cookies: - msToken: xxx - ttwid: xxx - odin_tt: xxx - passport_csrf_token: xxx - sid_guard: xxx - -# cookie 请登录网页抖音后F12查看 -# cookies 和 cookie 二选一, 要使用这种形式, 请注释上面的cookies及包含的所有键值对 -# 设置了这个后上面的cookies选项自动失效, 这个优先级更高 -# 格式: "name1=value1; name2=value2;" 注意要加冒号 -# 冒号中的内容包括不限于以下键值对, 如果抖音需要更多的cookie自己加上就行了 -#cookie: "msToken=xxx; ttwid=xxx; odin_tt=xxx; passport_csrf_token=xxx; sid_guard=xxx;" - +####################################### +# 说明: +# 1. 井号(#)为注释 +# 2. 缩进严格对齐,使用空格缩进, 注意有些冒号后面有一个空格, 有些没有空格 +# 3. 请使用英文字符 +# 4. 更多yaml语法请上网查看 +####################################### + + +# 作品(视频或图集)、直播、合集、音乐集合、个人主页的分享链接或者电脑浏览器网址 +# (删除文案, 保证只有URL, https://v.douyin.com/kcvMpuN/ 或者 https://www.douyin.com/开头的) +# 可以设置多个链接, 确保至少一个链接 +# 必选 +link: + - https://live.douyin.com/759547612580 + - https://v.douyin.com/BugmVVD/ + - https://v.douyin.com/BugrFTN/ + - https://v.douyin.com/B72pdU5/ + - https://v.douyin.com/B72QgDw/ + - https://v.douyin.com/AJp8D3f/ + - https://v.douyin.com/B38oovu/ + - https://v.douyin.com/S6YMNXs/ + +# 下载保存位置, 默认当前文件位置 +# 必选 +path: /mnt/c/project/test333 + +# 是否下载视频中的音乐(True/False), 默认为True +# 可选 +music: True + +# 是否下载视频的封面(True/False), 默认为True, 当下载视频时有效 +# 可选 +cover: True + +# 是否下载作者的头像(True/False), 默认为True +# 可选 +avatar: True + +# 是否保存获取到的数据(True/False), 默认为True +# 可选 +json: True + +# link是个人主页时, 设置下载发布的作品(post)或喜欢的作品(like)或者用户所有合集(mix), 默认为post, 可以设置多种模式 +# 可选 +mode: + - post + - like + - mix + +# 下载作品个数设置 +# 可选 +number: + post: 5 # 主页下作品下载个数设置, 默认为0 全部下载 + like: 5 # 主页下喜欢下载个数设置, 默认为0 全部下载 + allmix: 1 # 主页下合集下载个数设置, 默认为0 全部下载 + mix: 5 # 单个合集下作品下载个数设置, 默认为0 全部下载 + music: 5 # 音乐(原声)下作品下载个数设置, 默认为0 全部下载 + +# 设置线程数, 默认5个线程 +# 可选 +thread: 5 + +# cookie 请登录网页抖音后F12查看 +# cookies 和 cookie 二选一, 要使用这种形式, 请注释下面的cookie +# 目前只需要msToken、ttwid、odin_tt、passport_csrf_token、sid_guard +# 可以动态添加, 程序会根据填的键查找,并没有写死, 如果抖音需要更多的cookie自己加上就行了 +cookies: + msToken: xxx + ttwid: xxx + odin_tt: xxx + passport_csrf_token: xxx + sid_guard: xxx + +# cookie 请登录网页抖音后F12查看 +# cookies 和 cookie 二选一, 要使用这种形式, 请注释上面的cookies及包含的所有键值对 +# 设置了这个后上面的cookies选项自动失效, 这个优先级更高 +# 格式: "name1=value1; name2=value2;" 注意要加冒号 +# 冒号中的内容包括不限于以下键值对, 如果抖音需要更多的cookie自己加上就行了 +#cookie: "msToken=xxx; ttwid=xxx; odin_tt=xxx; passport_csrf_token=xxx; sid_guard=xxx;" + diff --git a/static/css/style.css b/static/css/style.css index 1808c5f..396588d 100755 --- a/static/css/style.css +++ b/static/css/style.css @@ -1,632 +1,632 @@ -html { - scroll-behavior: smooth; -} - -body, -html { - margin: 0; - padding: 0; - font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, - Oxygen-Sans, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; -} - -* { - box-sizing: border-box; -} - -.d-grid { - display: grid; -} - -.d-flex { - display: flex; - display: -webkit-flex; -} - -.text-center { - text-align: center; -} - -.text-left { - text-align: left; -} - -.text-right { - text-align: right; -} - -button, -input, -select { - -webkit-appearance: none; - outline: none; -} - -button, -.btn, -select { - cursor: pointer; -} - -a { - text-decoration: none; -} - -iframe { - border: none; -} - -ul { - margin: 0; - padding: 0 -} - -h1, -h2, -h3, -h4, -h5, -h6, -p { - margin: 0; - padding: 0 -} - -p { - color: #485460; -} - -.p-relative { - position: relative; -} - -.p-absolute { - position: absolute; -} - -.p-fixed { - position: fixed; -} - -.p-sticky { - position: sticky; -} - -.btn, -button, -.actionbg { - border-radius: 2px; - -webkit-border-radius: 2px; - -moz-border-radius: 2px; - -o-border-radius: 2px; - -ms-border-radius: 2px; -} - -.btn:hover, -button:hover { - transition: 0.5s ease; - -webkit-transition: 0.5s ease; - -o-transition: 0.5s ease; - -ms-transition: 0.5s ease; - -moz-transition: 0.5s ease; -} - -/*--/wrapper--*/ -.wrapper { - width: 100%; - padding-right: 15px; - padding-left: 15px; - margin-right: auto; - margin-left: auto; -} - -@media (min-width: 576px) { - .wrapper { - max-width: 540px; - } -} - -@media (min-width: 768px) { - .wrapper { - max-width: 720px; - } -} - -@media (min-width: 992px) { - .wrapper { - max-width: 960px; - } -} - -@media (min-width: 1200px) { - .wrapper { - max-width: 1140px; - } -} - -.wrapper-full { - width: 100%; - padding-right: 15px; - padding-left: 15px; - margin-right: auto; - margin-left: auto; -} - -/*--//wrapper--*/ - .error-61-mian { - background: url(https://testingcf.jsdelivr.net/gh/imgyh/tiktok/static/img/banner.jpg) no-repeat bottom; - background-size: cover; - -webkit-background-size: cover; - -moz-background-size: cover; - -o-background-size: cover; - -moz-background-size: cover; - height: 100vh; - z-index: 0; - position: relative; - display: grid; - align-items: center; -} - - .error-61-mian:before { - content: ""; - background: rgba(66, 21, 2, 0.45); - position: absolute; - top: 0; - min-height: 100%; - left: 0; - right: 0; - z-index: -1; -} - - .errors-16-top { - max-width: 600px; - margin: 0 auto; - text-align: center -} - - .errors-16-mid { - max-width: 600px; - margin: 0 auto; - text-align: center -} - - .error-61-mian h3 { - font-size: 223px; - line-height: 200px; - color: #fff; - font-weight: bold; - margin-bottom: 20px; - text-shadow: 0 1px 2px rgba(0, 0, 0, .6); -} - - .error-61-mian p { - font-size: 18px; - line-height: 30px; - color: #fff; - letter-spacing: 1px; -} - - .error-page-form { - border: 2px solid #fff; - border-radius: 25px; - padding: 5px; - margin: 20px 0; -} - - .error-page-form input { - color: #fff; - font-size: 18px; - border: none; - outline: none; - display: block; - padding: 0.4em 1em; - background: rgba(114, 51, 23, 0.75); - letter-spacing: 1px; - width: 79%; - border-radius: 25px; - margin-right: 1%; -} - -::-webkit-input-placeholder { /* Chrome/Opera/Safari */ - color: #fff; - background: transparent; -} - -::-moz-placeholder { /* Firefox 19+ */ - color: #fff; - background: transparent; -} - -:-ms-input-placeholder { /* IE 10+ */ - color: #fff; - background: transparent; -} - -:-moz-placeholder { /* Firefox 18- */ - color: #fff; - background: transparent; -} - - .error-page-form button { - color: var(--theme-green); - background: #fff; - border: none; - padding: 10px 15px; - text-decoration: none; - cursor: pointer; - font-size: 18px; - width: 20%; - display: block; - border-radius: 25px; -} - - .error-page-form button:hover { - color: #fff; - background: #0099CC; /*#394247*/ - -} - - .social-coming-icons a { - background: transparent; - border-radius: 50%; - width: 34px; - height: 34px; - text-align: center; - display: inline-block; - margin-right: 12px; - border: 2px solid #fff; - color: #fff; -} - - .social-coming-icons a span { - font-size: 18px; - line-height: 30px; - -} - - .social-coming-icons a:hover { - color: #0099CC; /*#394247*/ - border: 2px solid #0099CC; /*#394247*/ - transition: 0.5s ease; - -webkit-transition: 0.5s ease; - -o-transition: 0.5s ease; - -ms-transition: 0.5s ease; - -moz-transition: 0.5s ease; -} - - .copy-right { - margin: 0 auto; - text-align: center -} - - .copy-right p { - font-size: 18px; - line-height: 29px; - color: #f9f9f9; -} - - .copy-right a { - color: #00c4b6; -} - -@media (max-width: 1280px) { - .error-61-mian h3 { - font-size: 180px; - line-height: 170px; - } -} - -@media (max-width: 1280px) { - .error-61-mian h3 { - font-size: 125px; - line-height: 120px; - color: #fff; - font-weight: 700; - margin-bottom: 20px; - } -} - -@media (max-width: 990px) { - .error-61-mian h3 { - font-size: 125px; - line-height: 110px; - } -} - -@media (max-width: 667px) { - .error-61-mian h3 { - font-size: 115px; - line-height: 110px; - } -} - -@media (max-width: 600px) { - .error-61-mian { - background: url(https://testingcf.jsdelivr.net/gh/imgyh/tiktok/static/img/banner.jpg) no-repeat right; - background-size: cover; - -webkit-background-size: cover; - -moz-background-size: cover; - -o-background-size: cover; - -moz-background-size: cover; - } -} - -@media (max-width: 440px) { - .error-61-mian h3 { - font-size: 105px; - line-height: 100px; - } -} - -@media (max-width: 384px) { - .error-61-mian h3 { - font-size: 95px; - line-height: 90px; - } -} - -/*加载动画*/ -.loading { - width: 30px; - height: 30px; - position: relative; - transform: rotate(45deg); - animation: animationContainer 1s ease infinite; - margin: 0 auto; - display: none; -} - -.shape { - width: 10px; - height: 10px; - border-radius: 50%; - position: absolute; -} - -.shape-1 { - background-color: #1875e5; - left: 0; - animation: animationShape1 0.3s ease infinite alternate; -} - -.shape-2 { - background-color: #c5523f; - right: 0; - animation: animationShape2 0.3s ease infinite 0.3s alternate; -} - -.shape-3 { - background-color: #499255; - bottom: 0; - animation: animationShape3 0.3s ease infinite 0.3s alternate; -} - -.shape-4 { - background-color: #f2b736; - right: 0; - bottom: 0; - animation: animationShape4 0.3s ease infinite alternate; -} - -@keyframes animationContainer { - 0% { - transform: rotate(0); - } - - 100% { - transform: rotate(360deg); - } -} - -@keyframes animationShape1 { - 0% { - transform: translate(5px, 5px); - } - - 100% { - transform: translate(-3px, -3px); - } -} - -@keyframes animationShape2 { - 0% { - transform: translate(-5px, 5px); - } - - 100% { - transform: translate(3px, -3px); - } -} - -@keyframes animationShape3 { - 0% { - transform: translate(5px, -5px); - } - - 100% { - transform: translate(-3px, 3px); - } -} - -@keyframes animationShape4 { - 0% { - transform: translate(-5px, -5px); - } - - 100% { - transform: translate(3px, 3px); - } -} - -/*加载动画*/ - -/*按钮样式*/ -.btn { - /* 文字颜色 */ - color: #0099CC; - /* 清除背景色 */ - background: transparent; - /* 边框样式、颜色、宽度 */ - border: 2px solid #0099CC; - /* 给边框添加圆角 */ - border-radius: 6px; - /* 字母转大写 */ - border: none; - color: white; - padding: 10px 20px; - text-align: center; - display: inline-block; - font-size: 18px; - margin: 4px 2px; - -webkit-transition-duration: 0.4s; /* Safari */ - transition-duration: 0.4s; - cursor: pointer; - text-decoration: none; - text-transform: uppercase; -} - -.btn1 { - background-color: white; - color: black; - border: 2px solid #0099CC; -} - -/* 悬停样式 */ -.btn1:hover { - background-color: #0099CC; - color: white; -} - -/*按钮样式*/ - -/*头像框*/ -.avatar { - width: 100%; - height: 100%; - display: flex; - border-radius: 50%; - align-items: center; - justify-content: center; - overflow: hidden; -} - -.nickname { - display: flex; - /*border-radius: 50%;*/ - align-items: center; - justify-content: center; - overflow: hidden; -} - -.photo { - float: left; - width: 30%; - height: 100%; -} - -.info { - float: right; - width: 70%; - height: 100%; -} - -.icons { - display: -webkit-flex; /* Safari */ - display: flex; - font-size: 30px; -} - -.icon { - flex: 1; -} - -/*头像框*/ - -/*select 样式*/ -.select { - display: inline-block; - width: 100px; - position: relative; - vertical-align: middle; - padding: 0; - overflow: hidden; - background-color: #fff; - color: #555; - border: 1px solid #aaa; - text-shadow: none; - border-radius: 25px; - transition: box-shadow 0.25s ease; - z-index: 2; -} - -.select:hover { - box-shadow: 0 1px 4px #0099CC; /*rgba(0, 0, 0, 0.15)*/ - background-color: #0099CC; - color: white; -} - -.select:before { - content: ""; - position: absolute; - width: 0; - height: 0; - border: 10px solid transparent; - border-top-color: #ccc; - top: 14px; - right: 10px; - cursor: pointer; - z-index: -2; -} - -.select select { - cursor: pointer; - padding: 10px; - width: 100%; - border: none; - background: transparent; - background-image: none; - -webkit-appearance: none; - -moz-appearance: none; - font-size: 18px; -} - -.select select:focus { - outline: none; -} - -/*select 样式*/ - -/*视频播放相关*/ -#show-video { - position: fixed; - top: 0; - bottom: 0; - right: 0; - left: 0; - z-index: 10 !important; - background: rgba(0, 0, 0, .85); - display: none; -} - -.video-close { - width: 45px; - height: 45px; - color: #211d1e; - position: absolute; - right: 50px; - top: 50px; - z-index: 10 ; - cursor: pointer; - -} - -#show-video video { - outline: none; - max-width: 85%; - max-height: 88vh; - position: absolute; - left: 50%; - top: 50%; - transform: translate(-50%, -50%); - box-shadow: 0 20px 40px rgb(0 0 0 / 50%); -} +html { + scroll-behavior: smooth; +} + +body, +html { + margin: 0; + padding: 0; + font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, + Oxygen-Sans, Ubuntu, Cantarell, "Helvetica Neue", sans-serif; +} + +* { + box-sizing: border-box; +} + +.d-grid { + display: grid; +} + +.d-flex { + display: flex; + display: -webkit-flex; +} + +.text-center { + text-align: center; +} + +.text-left { + text-align: left; +} + +.text-right { + text-align: right; +} + +button, +input, +select { + -webkit-appearance: none; + outline: none; +} + +button, +.btn, +select { + cursor: pointer; +} + +a { + text-decoration: none; +} + +iframe { + border: none; +} + +ul { + margin: 0; + padding: 0 +} + +h1, +h2, +h3, +h4, +h5, +h6, +p { + margin: 0; + padding: 0 +} + +p { + color: #485460; +} + +.p-relative { + position: relative; +} + +.p-absolute { + position: absolute; +} + +.p-fixed { + position: fixed; +} + +.p-sticky { + position: sticky; +} + +.btn, +button, +.actionbg { + border-radius: 2px; + -webkit-border-radius: 2px; + -moz-border-radius: 2px; + -o-border-radius: 2px; + -ms-border-radius: 2px; +} + +.btn:hover, +button:hover { + transition: 0.5s ease; + -webkit-transition: 0.5s ease; + -o-transition: 0.5s ease; + -ms-transition: 0.5s ease; + -moz-transition: 0.5s ease; +} + +/*--/wrapper--*/ +.wrapper { + width: 100%; + padding-right: 15px; + padding-left: 15px; + margin-right: auto; + margin-left: auto; +} + +@media (min-width: 576px) { + .wrapper { + max-width: 540px; + } +} + +@media (min-width: 768px) { + .wrapper { + max-width: 720px; + } +} + +@media (min-width: 992px) { + .wrapper { + max-width: 960px; + } +} + +@media (min-width: 1200px) { + .wrapper { + max-width: 1140px; + } +} + +.wrapper-full { + width: 100%; + padding-right: 15px; + padding-left: 15px; + margin-right: auto; + margin-left: auto; +} + +/*--//wrapper--*/ +.error-61-mian { + background: url(https://testingcf.jsdelivr.net/gh/imgyh/tiktok/static/img/banner.jpg) no-repeat bottom; + background-size: cover; + -webkit-background-size: cover; + -moz-background-size: cover; + -o-background-size: cover; + -moz-background-size: cover; + height: 100vh; + z-index: 0; + position: relative; + display: grid; + align-items: center; +} + +.error-61-mian:before { + content: ""; + background: rgba(66, 21, 2, 0.45); + position: absolute; + top: 0; + min-height: 100%; + left: 0; + right: 0; + z-index: -1; +} + +.errors-16-top { + max-width: 600px; + margin: 0 auto; + text-align: center +} + +.errors-16-mid { + max-width: 600px; + margin: 0 auto; + text-align: center +} + +.error-61-mian h3 { + font-size: 223px; + line-height: 200px; + color: #fff; + font-weight: bold; + margin-bottom: 20px; + text-shadow: 0 1px 2px rgba(0, 0, 0, .6); +} + +.error-61-mian p { + font-size: 18px; + line-height: 30px; + color: #fff; + letter-spacing: 1px; +} + +.error-page-form { + border: 2px solid #fff; + border-radius: 25px; + padding: 5px; + margin: 20px 0; +} + +.error-page-form input { + color: #fff; + font-size: 18px; + border: none; + outline: none; + display: block; + padding: 0.4em 1em; + background: rgba(114, 51, 23, 0.75); + letter-spacing: 1px; + width: 79%; + border-radius: 25px; + margin-right: 1%; +} + +::-webkit-input-placeholder { /* Chrome/Opera/Safari */ + color: #fff; + background: transparent; +} + +::-moz-placeholder { /* Firefox 19+ */ + color: #fff; + background: transparent; +} + +:-ms-input-placeholder { /* IE 10+ */ + color: #fff; + background: transparent; +} + +:-moz-placeholder { /* Firefox 18- */ + color: #fff; + background: transparent; +} + +.error-page-form button { + color: var(--theme-green); + background: #fff; + border: none; + padding: 10px 15px; + text-decoration: none; + cursor: pointer; + font-size: 18px; + width: 20%; + display: block; + border-radius: 25px; +} + +.error-page-form button:hover { + color: #fff; + background: #0099CC; /*#394247*/ + +} + +.social-coming-icons a { + background: transparent; + border-radius: 50%; + width: 34px; + height: 34px; + text-align: center; + display: inline-block; + margin-right: 12px; + border: 2px solid #fff; + color: #fff; +} + +.social-coming-icons a span { + font-size: 18px; + line-height: 30px; + +} + +.social-coming-icons a:hover { + color: #0099CC; /*#394247*/ + border: 2px solid #0099CC; /*#394247*/ + transition: 0.5s ease; + -webkit-transition: 0.5s ease; + -o-transition: 0.5s ease; + -ms-transition: 0.5s ease; + -moz-transition: 0.5s ease; +} + +.copy-right { + margin: 0 auto; + text-align: center +} + +.copy-right p { + font-size: 18px; + line-height: 29px; + color: #f9f9f9; +} + +.copy-right a { + color: #00c4b6; +} + +@media (max-width: 1280px) { + .error-61-mian h3 { + font-size: 180px; + line-height: 170px; + } +} + +@media (max-width: 1280px) { + .error-61-mian h3 { + font-size: 125px; + line-height: 120px; + color: #fff; + font-weight: 700; + margin-bottom: 20px; + } +} + +@media (max-width: 990px) { + .error-61-mian h3 { + font-size: 125px; + line-height: 110px; + } +} + +@media (max-width: 667px) { + .error-61-mian h3 { + font-size: 115px; + line-height: 110px; + } +} + +@media (max-width: 600px) { + .error-61-mian { + background: url(https://testingcf.jsdelivr.net/gh/imgyh/tiktok/static/img/banner.jpg) no-repeat right; + background-size: cover; + -webkit-background-size: cover; + -moz-background-size: cover; + -o-background-size: cover; + -moz-background-size: cover; + } +} + +@media (max-width: 440px) { + .error-61-mian h3 { + font-size: 105px; + line-height: 100px; + } +} + +@media (max-width: 384px) { + .error-61-mian h3 { + font-size: 95px; + line-height: 90px; + } +} + +/*加载动画*/ +.loading { + width: 30px; + height: 30px; + position: relative; + transform: rotate(45deg); + animation: animationContainer 1s ease infinite; + margin: 0 auto; + display: none; +} + +.shape { + width: 10px; + height: 10px; + border-radius: 50%; + position: absolute; +} + +.shape-1 { + background-color: #1875e5; + left: 0; + animation: animationShape1 0.3s ease infinite alternate; +} + +.shape-2 { + background-color: #c5523f; + right: 0; + animation: animationShape2 0.3s ease infinite 0.3s alternate; +} + +.shape-3 { + background-color: #499255; + bottom: 0; + animation: animationShape3 0.3s ease infinite 0.3s alternate; +} + +.shape-4 { + background-color: #f2b736; + right: 0; + bottom: 0; + animation: animationShape4 0.3s ease infinite alternate; +} + +@keyframes animationContainer { + 0% { + transform: rotate(0); + } + + 100% { + transform: rotate(360deg); + } +} + +@keyframes animationShape1 { + 0% { + transform: translate(5px, 5px); + } + + 100% { + transform: translate(-3px, -3px); + } +} + +@keyframes animationShape2 { + 0% { + transform: translate(-5px, 5px); + } + + 100% { + transform: translate(3px, -3px); + } +} + +@keyframes animationShape3 { + 0% { + transform: translate(5px, -5px); + } + + 100% { + transform: translate(-3px, 3px); + } +} + +@keyframes animationShape4 { + 0% { + transform: translate(-5px, -5px); + } + + 100% { + transform: translate(3px, 3px); + } +} + +/*加载动画*/ + +/*按钮样式*/ +.btn { + /* 文字颜色 */ + color: #0099CC; + /* 清除背景色 */ + background: transparent; + /* 边框样式、颜色、宽度 */ + border: 2px solid #0099CC; + /* 给边框添加圆角 */ + border-radius: 6px; + /* 字母转大写 */ + border: none; + color: white; + padding: 10px 20px; + text-align: center; + display: inline-block; + font-size: 18px; + margin: 4px 2px; + -webkit-transition-duration: 0.4s; /* Safari */ + transition-duration: 0.4s; + cursor: pointer; + text-decoration: none; + text-transform: uppercase; +} + +.btn1 { + background-color: white; + color: black; + border: 2px solid #0099CC; +} + +/* 悬停样式 */ +.btn1:hover { + background-color: #0099CC; + color: white; +} + +/*按钮样式*/ + +/*头像框*/ +.avatar { + width: 100%; + height: 100%; + display: flex; + border-radius: 50%; + align-items: center; + justify-content: center; + overflow: hidden; +} + +.nickname { + display: flex; + /*border-radius: 50%;*/ + align-items: center; + justify-content: center; + overflow: hidden; +} + +.photo { + float: left; + width: 30%; + height: 100%; +} + +.info { + float: right; + width: 70%; + height: 100%; +} + +.icons { + display: -webkit-flex; /* Safari */ + display: flex; + font-size: 30px; +} + +.icon { + flex: 1; +} + +/*头像框*/ + +/*select 样式*/ +.select { + display: inline-block; + width: 100px; + position: relative; + vertical-align: middle; + padding: 0; + overflow: hidden; + background-color: #fff; + color: #555; + border: 1px solid #aaa; + text-shadow: none; + border-radius: 25px; + transition: box-shadow 0.25s ease; + z-index: 2; +} + +.select:hover { + box-shadow: 0 1px 4px #0099CC; /*rgba(0, 0, 0, 0.15)*/ + background-color: #0099CC; + color: white; +} + +.select:before { + content: ""; + position: absolute; + width: 0; + height: 0; + border: 10px solid transparent; + border-top-color: #ccc; + top: 14px; + right: 10px; + cursor: pointer; + z-index: -2; +} + +.select select { + cursor: pointer; + padding: 10px; + width: 100%; + border: none; + background: transparent; + background-image: none; + -webkit-appearance: none; + -moz-appearance: none; + font-size: 18px; +} + +.select select:focus { + outline: none; +} + +/*select 样式*/ + +/*视频播放相关*/ +#show-video { + position: fixed; + top: 0; + bottom: 0; + right: 0; + left: 0; + z-index: 10 !important; + background: rgba(0, 0, 0, .85); + display: none; +} + +.video-close { + width: 45px; + height: 45px; + color: #211d1e; + position: absolute; + right: 50px; + top: 50px; + z-index: 10; + cursor: pointer; + +} + +#show-video video { + outline: none; + max-width: 85%; + max-height: 88vh; + position: absolute; + left: 50%; + top: 50%; + transform: translate(-50%, -50%); + box-shadow: 0 20px 40px rgb(0 0 0 / 50%); +} diff --git a/static/js/custom.js b/static/js/custom.js index 34a2730..59a3433 100644 --- a/static/js/custom.js +++ b/static/js/custom.js @@ -1,229 +1,229 @@ -// 发 post 请求 -function SendAjax() { - var data = {}; - data = $('#form1').serialize(); - $.ajax({ - type: 'POST', - url: "/douyin", - data: data, - dataType: 'json', - beforeSend: function () { - $("#loading").attr("style", "display:block;");//在请求后台数据之前显示loading图标 - $("#download").attr("style", "display:none;");//隐藏 download - }, - success: function (result) { - // console.log(result);//打印服务端返回的数据(调试用) - if (result.status_code === 200) { - result = result.data - if (result.awemeType === 0) { - $("#awemeType").html("预览视频"); - $("#AwemeOrLive").html("下载视频"); - $("#video").attr("href", result.video.play_addr.url_list); - $("#pre_video").attr("src", result.video.play_addr.url_list); - $("#video").attr("style", "display:inline;");//显示 video - } - if (result.awemeType === 1) { - $("#awemeType").html("预览图集"); - var images = result.images; - var licontent = ""; // 拼接输入的 li 标签的字符串 - for (var i = 0; i < images.length; i++) { - licontent += "
  • " - } - document.getElementById("images").innerHTML = licontent; - $("#video").attr("style", "display:none;");//隐藏 video - } - if (result.awemeType === 0 || result.awemeType === 1) { - $("#cover").attr("href", result.video.cover_original_scale.url_list[0]); - $("#pre_video").attr("poster", result.video.dynamic_cover.url_list[0]); - $("#music").attr("href", result.music.play_url.url_list[0]); - - $("#avatar").attr("src", result.author.avatar.url_list[0]); - $("#avatar").attr("alt", result.author.nickname); - $("#nickname").html(result.author.nickname); - $("#desc").html(result.desc); - - - var count = result.statistics.digg_count; - var digg_count; - if (count < 1000) { - digg_count = count - } else if (count >= 1000 && count < 10000) { - digg_count = (count / 1000).toFixed(1) + "K" - } else { - digg_count = (count / 10000).toFixed(1) + "W" - } - $("#aweme_digg_count").html(digg_count); - count = result.statistics.comment_count; - var comment_count; - if (count < 1000) { - comment_count = count - } else if (count >= 1000 && count < 10000) { - comment_count = (count / 1000).toFixed(1) + "K" - } else { - comment_count = (count / 10000).toFixed(1) + "W" - } - $("#aweme_comment_count").html(comment_count); - count = result.statistics.collect_count; - var collect_count; - if (count < 1000) { - collect_count = count - } else if (count >= 1000 && count < 10000) { - collect_count = (count / 1000).toFixed(1) + "K" - } else { - collect_count = (count / 10000).toFixed(1) + "W" - } - $("#aweme_collect_count").html(collect_count); - count = result.statistics.share_count; - var share_count; - if (count < 1000) { - share_count = count - } else if (count >= 1000 && count < 10000) { - share_count = (count / 1000).toFixed(1) + "K" - } else { - share_count = (count / 10000).toFixed(1) + "W" - } - $("#aweme_share_count").html(share_count); - - $("#icons").attr("style", "display:flex;");//显示 icons - $("#icon").attr("style", "display:table-row;");//显示 icon - $("#music").attr("style", "display:inline;");//显示 music - - $("#loading").attr("style", "display:none;");//隐藏 loading - $("#download").attr("style", "display:block;");//显示 download - // alert("SUCCESS"); - // 执行弹框 - narnSuccess(); - } - - if (result.awemeType === 2) { - if (result.status === 4) { - $("#loading").attr("style", "display:none;");//隐藏 loading - $("#download").attr("style", "display:none;");//隐藏 download - // 执行弹框 - narnWarn() - } else { - $("#AwemeOrLive").html("下载直播"); - $("#awemeType").html("预览直播"); - $("#video").attr("href", result.flv_pull_url0); - $("#pre_video").attr("src", result.flv_pull_url0); - - $("#cover").attr("href", result.cover); - $("#pre_video").attr("poster", result.cover); - $("#avatar").attr("src", result.avatar); - $("#avatar").attr("alt", result.nickname); - $("#nickname").html(result.nickname); - $("#desc").html(result.title); - - $("#video").attr("style", "display:inline;");//显示 video - $("#icons").attr("style", "display:none;");//隐藏 icons - $("#icon").attr("style", "display:none;");//隐藏 icon - $("#music").attr("style", "display:none;");//隐藏 music - - $("#loading").attr("style", "display:none;");//隐藏 loading - $("#download").attr("style", "display:block;");//显示 download - // alert("SUCCESS"); - // 执行弹框 - narnSuccess(); - } - } - } else { - $("#loading").attr("style", "display:none;");//隐藏 loading - $("#download").attr("style", "display:none;");//隐藏 download - - // 执行弹框 - narnFail(); - } - ; - }, - error: function (xhr, type) { - $("#loading").attr("style", "display:none;");//隐藏 loading - $("#download").attr("style", "display:none;");//隐藏 download - // alert("异常!"); - - // 执行弹框 - narnFail(); - } - }); -} - -// 右上角弹框 -function narnSuccess() { - naranja().success({ - title: '解析成功', - text: '请及时下载音视频', - icon: true, - timeout: 5000, - buttons: [] - }) -} - -function narnFail() { - naranja().error({ - title: '解析失败', - text: '直播/视频/图集不存在或接口失效', - icon: true, - timeout: 5000, - buttons: [] - }) -} - -function narnWarn() { - naranja().warn({ - title: '提示', - text: '直播未开始', - icon: true, - timeout: 5000, - buttons: [] - }) -} - - -window.addEventListener('DOMContentLoaded', function () { - document.getElementById('view_aweme').addEventListener('click', function () { - var awemeType = document.getElementById("awemeType").innerText; - - if (awemeType === "预览视频") { - // 调小音量 - var videoElement = document.getElementById("pre_video"); - videoElement.volume = 0.6 - /*弹出视频播放层*/ - $("#show-video").show(); - } - // 图片查看器 - if (awemeType === "预览图集") { - - var viewer = new Viewer(document.getElementById('images'), { - hidden: function () { - viewer.destroy(); - }, - }); - - // image.click(); - viewer.show(); - } - // 预览直播 - if (awemeType === "预览直播") { - if (flvjs.isSupported()) {//检查flvjs能否正常使用 - var videoElement = document.getElementById('pre_video');//使用id选择器找到第二步设置的dom元素 - var flvPlayer = flvjs.createPlayer({//创建一个新的flv播放器对象 - type: 'flv',//类型flv - url: $("#video").attr("href")//flv文件地址 - }); - flvPlayer.attachMediaElement(videoElement);//将flv视频装载进video元素内 - flvPlayer.load();//载入视频 - flvPlayer.play();//播放视频,如果不想要自动播放,去掉本行 - - /*弹出视频播放层*/ - $("#show-video").show(); - } - } - - - }); - /*关闭视频播放层*/ - $(".video-close").click(function () { - var videoElement = document.getElementById("pre_video"); - videoElement.pause() - $("#show-video").hide(); - }) -}); +// 发 post 请求 +function SendAjax() { + var data = {}; + data = $('#form1').serialize(); + $.ajax({ + type: 'POST', + url: "/douyin", + data: data, + dataType: 'json', + beforeSend: function () { + $("#loading").attr("style", "display:block;");//在请求后台数据之前显示loading图标 + $("#download").attr("style", "display:none;");//隐藏 download + }, + success: function (result) { + // console.log(result);//打印服务端返回的数据(调试用) + if (result.status_code === 200) { + result = result.data + if (result.awemeType === 0) { + $("#awemeType").html("预览视频"); + $("#AwemeOrLive").html("下载视频"); + $("#video").attr("href", result.video.play_addr.url_list); + $("#pre_video").attr("src", result.video.play_addr.url_list); + $("#video").attr("style", "display:inline;");//显示 video + } + if (result.awemeType === 1) { + $("#awemeType").html("预览图集"); + var images = result.images; + var licontent = ""; // 拼接输入的 li 标签的字符串 + for (var i = 0; i < images.length; i++) { + licontent += "
  • " + } + document.getElementById("images").innerHTML = licontent; + $("#video").attr("style", "display:none;");//隐藏 video + } + if (result.awemeType === 0 || result.awemeType === 1) { + $("#cover").attr("href", result.video.cover_original_scale.url_list[0]); + $("#pre_video").attr("poster", result.video.dynamic_cover.url_list[0]); + $("#music").attr("href", result.music.play_url.url_list[0]); + + $("#avatar").attr("src", result.author.avatar.url_list[0]); + $("#avatar").attr("alt", result.author.nickname); + $("#nickname").html(result.author.nickname); + $("#desc").html(result.desc); + + + var count = result.statistics.digg_count; + var digg_count; + if (count < 1000) { + digg_count = count + } else if (count >= 1000 && count < 10000) { + digg_count = (count / 1000).toFixed(1) + "K" + } else { + digg_count = (count / 10000).toFixed(1) + "W" + } + $("#aweme_digg_count").html(digg_count); + count = result.statistics.comment_count; + var comment_count; + if (count < 1000) { + comment_count = count + } else if (count >= 1000 && count < 10000) { + comment_count = (count / 1000).toFixed(1) + "K" + } else { + comment_count = (count / 10000).toFixed(1) + "W" + } + $("#aweme_comment_count").html(comment_count); + count = result.statistics.collect_count; + var collect_count; + if (count < 1000) { + collect_count = count + } else if (count >= 1000 && count < 10000) { + collect_count = (count / 1000).toFixed(1) + "K" + } else { + collect_count = (count / 10000).toFixed(1) + "W" + } + $("#aweme_collect_count").html(collect_count); + count = result.statistics.share_count; + var share_count; + if (count < 1000) { + share_count = count + } else if (count >= 1000 && count < 10000) { + share_count = (count / 1000).toFixed(1) + "K" + } else { + share_count = (count / 10000).toFixed(1) + "W" + } + $("#aweme_share_count").html(share_count); + + $("#icons").attr("style", "display:flex;");//显示 icons + $("#icon").attr("style", "display:table-row;");//显示 icon + $("#music").attr("style", "display:inline;");//显示 music + + $("#loading").attr("style", "display:none;");//隐藏 loading + $("#download").attr("style", "display:block;");//显示 download + // alert("SUCCESS"); + // 执行弹框 + narnSuccess(); + } + + if (result.awemeType === 2) { + if (result.status === 4) { + $("#loading").attr("style", "display:none;");//隐藏 loading + $("#download").attr("style", "display:none;");//隐藏 download + // 执行弹框 + narnWarn() + } else { + $("#AwemeOrLive").html("下载直播"); + $("#awemeType").html("预览直播"); + $("#video").attr("href", result.flv_pull_url0); + $("#pre_video").attr("src", result.flv_pull_url0); + + $("#cover").attr("href", result.cover); + $("#pre_video").attr("poster", result.cover); + $("#avatar").attr("src", result.avatar); + $("#avatar").attr("alt", result.nickname); + $("#nickname").html(result.nickname); + $("#desc").html(result.title); + + $("#video").attr("style", "display:inline;");//显示 video + $("#icons").attr("style", "display:none;");//隐藏 icons + $("#icon").attr("style", "display:none;");//隐藏 icon + $("#music").attr("style", "display:none;");//隐藏 music + + $("#loading").attr("style", "display:none;");//隐藏 loading + $("#download").attr("style", "display:block;");//显示 download + // alert("SUCCESS"); + // 执行弹框 + narnSuccess(); + } + } + } else { + $("#loading").attr("style", "display:none;");//隐藏 loading + $("#download").attr("style", "display:none;");//隐藏 download + + // 执行弹框 + narnFail(); + } + ; + }, + error: function (xhr, type) { + $("#loading").attr("style", "display:none;");//隐藏 loading + $("#download").attr("style", "display:none;");//隐藏 download + // alert("异常!"); + + // 执行弹框 + narnFail(); + } + }); +} + +// 右上角弹框 +function narnSuccess() { + naranja().success({ + title: '解析成功', + text: '请及时下载音视频', + icon: true, + timeout: 5000, + buttons: [] + }) +} + +function narnFail() { + naranja().error({ + title: '解析失败', + text: '直播/视频/图集不存在或接口失效', + icon: true, + timeout: 5000, + buttons: [] + }) +} + +function narnWarn() { + naranja().warn({ + title: '提示', + text: '直播未开始', + icon: true, + timeout: 5000, + buttons: [] + }) +} + + +window.addEventListener('DOMContentLoaded', function () { + document.getElementById('view_aweme').addEventListener('click', function () { + var awemeType = document.getElementById("awemeType").innerText; + + if (awemeType === "预览视频") { + // 调小音量 + var videoElement = document.getElementById("pre_video"); + videoElement.volume = 0.6 + /*弹出视频播放层*/ + $("#show-video").show(); + } + // 图片查看器 + if (awemeType === "预览图集") { + + var viewer = new Viewer(document.getElementById('images'), { + hidden: function () { + viewer.destroy(); + }, + }); + + // image.click(); + viewer.show(); + } + // 预览直播 + if (awemeType === "预览直播") { + if (flvjs.isSupported()) {//检查flvjs能否正常使用 + var videoElement = document.getElementById('pre_video');//使用id选择器找到第二步设置的dom元素 + var flvPlayer = flvjs.createPlayer({//创建一个新的flv播放器对象 + type: 'flv',//类型flv + url: $("#video").attr("href")//flv文件地址 + }); + flvPlayer.attachMediaElement(videoElement);//将flv视频装载进video元素内 + flvPlayer.load();//载入视频 + flvPlayer.play();//播放视频,如果不想要自动播放,去掉本行 + + /*弹出视频播放层*/ + $("#show-video").show(); + } + } + + + }); + /*关闭视频播放层*/ + $(".video-close").click(function () { + var videoElement = document.getElementById("pre_video"); + videoElement.pause() + $("#show-video").hide(); + }) +}); diff --git a/templates/index.html b/templates/index.html index bca4321..fea6a2e 100755 --- a/templates/index.html +++ b/templates/index.html @@ -1,172 +1,175 @@ - - - - - 抖音去水印工具 - - - - - - - - - - - - - - - - - - - - - - - - - - - -
    -
    -
    - -

    抖音去水印工具

    -
    -

    支持视频/图集/直播解析,粘贴视频/图集/直播分享链接时无需删除文案,但如果链接正确但解析失败请删掉文案后重试 https://v.douyin.com/kcvMpuN/

    -{#

    2.支持直播解析,需要网页版直播链接 https://live.douyin.com/343806013144

    #} -

    关于抖音批量下载与去水印工具的更多实现细节请点击: 抖音批量下载与去水印工具

    -
    -{# 以前需要手动选择 图片 或者 视频 现在加了自动判断#} -{#
    #} -{# #} -{#
    #} - - - -
    - {# #} - -
    -
    -
    -
    -
    -
    -
    -
    - - - -
    - -
    -
    -

    Copyright © 2021-2023.我的博客 GYH's Blog && 项目地址 Github - All rights reserved.

    -
    -
    - -{# 视频预览效果 https://blog.csdn.net/qq_45140694/article/details/115266928 #} -
    - - - - - - - - {# https://blog.csdn.net/seeeeeeeeeee/article/details/119981594 #} - -
    - -
    -
      -{#
    • Picture 1
    • #} -{#
    • Picture 2
    • #} -{#
    • Picture 3
    • #} -
    -
    - - - - - - - - + + + + + 抖音去水印工具 + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    +
    +
    + +

    抖音去水印工具

    +
    +

    支持视频/图集/直播解析,粘贴视频/图集/直播分享链接时无需删除文案,但如果链接正确但解析失败请删掉文案后重试 https://v.douyin.com/kcvMpuN/

    + {#

    2.支持直播解析,需要网页版直播链接 https://live.douyin.com/343806013144

    #} +

    关于抖音批量下载与去水印工具的更多实现细节请点击: 抖音批量下载与去水印工具

    +
    + {# 以前需要手动选择 图片 或者 视频 现在加了自动判断#} + {#
    #} + {# #} + {#
    #} + + + +
    + {# #} + +
    +
    +
    +
    +
    +
    +
    +
    + + + +
    + +
    +
    +

    Copyright © 2021-2023.我的博客 GYH's Blog && 项目地址 Github + All rights reserved.

    +
    +
    + +{# 视频预览效果 https://blog.csdn.net/qq_45140694/article/details/115266928 #} +
    + + + + + + + + {# https://blog.csdn.net/seeeeeeeeeee/article/details/119981594 #} + +
    + +
    +
      + {#
    • Picture 1
    • #} + {#
    • Picture 2
    • #} + {#
    • Picture 3
    • #} +
    +
    + + + + + + + + \ No newline at end of file