亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術(shù)文章
文章詳情頁

python實(shí)現(xiàn)的B站直播錄制工具

瀏覽:103日期:2022-06-21 08:50:30
項(xiàng)目地址:

https://github.com/Redlnn/blive_record

前言 作者: Red_lnn 不允許將本項(xiàng)目運(yùn)用于非法以及違反B站用戶協(xié)議的用途 僅支持單個(gè)主播,多個(gè)主播請復(fù)制多份并分開單獨(dú)啟動 運(yùn)行時(shí)如要停止錄制并退出,請按鍵盤 Ctrl+C 如要修改錄制設(shè)置,請以純文本方式打開.py文件 利用 FFmpeg 直接抓取主播推送的流,無需打開瀏覽器 有新功能需求請直接提 Pull requests 建議錄制為 flv 格式(默認(rèn)),以防止意外中斷導(dǎo)致錄制文件損壞,若要進(jìn)行剪輯可使用 FFmpeg 轉(zhuǎn)換為 mp4 文件后再倒入到剪輯軟件(使用 FFmpeg 轉(zhuǎn)換 flv 為 mp4 : ffmpeg -i {input}.flv -c:v copy -c:a copy {output}.mp4)使用方式

1.安裝 Python(>=3.7) 并設(shè)置環(huán)境變量

2.打開終端或命令行進(jìn)入本腳本所在目錄

3.通過 pip 安裝必須的第三方庫

Windows:

pip install -r requirements.txt

Linux:

python3 -m pip install -r requirements.txt

4.下載 ffmpeg 并正確設(shè)置環(huán)境變量(下載地址)5.Windows 直接雙擊運(yùn)行start.bat6.Linux 先運(yùn)行 chmod +x start.sh 再運(yùn)行 ./start.sh

主要代碼blive_record.py

#!/usr/bin/env python3# -*- coding:utf-8 -*-'''*--------------------------------------* B站直播錄播姬 By: Red_lnn 僅支持單個(gè)主播,多個(gè)主播請復(fù)制多份并分開單獨(dú)啟動 運(yùn)行時(shí)如要停止錄制并退出,請按鍵盤 Ctrl+C 如要修改錄制設(shè)置,請以純文本方式打開.py文件 利用ffmpeg直接抓取主播推送的流,不需要打開瀏覽器*--------------------------------------*'''# import ffmpy3 # noqaimport loggingimport osimport signalimport sysimport threadingimport timeimport tracebackfrom json import loadsfrom logging import handlersfrom subprocess import PIPE, Popen, STDOUTimport requestsfrom regex import match# 導(dǎo)入配置from config import * # noqarecord_status = False # 錄制狀態(tài),True為錄制中kill_times = 0 # 嘗試強(qiáng)制結(jié)束FFmpeg的次數(shù)logging.addLevelName(15, ’FFmpeg’) # 自定義FFmpeg的日志級別logger = logging.getLogger(’Record’)logger.setLevel(logging.DEBUG)fms = ’[%(asctime)s %(levelname)s] %(message)s’# datefmt = '%Y-%m-%d %H:%M:%S'datefmt = '%H:%M:%S'default_handler = logging.StreamHandler(sys.stdout)if debug: default_handler.setLevel(logging.DEBUG)elif verbose: default_handler.setLevel(15)else: default_handler.setLevel(logging.INFO)default_handler.setFormatter(logging.Formatter(fms, datefmt=datefmt))logger.addHandler(default_handler)if save_log: # file_handler = logging.FileHandler('debug.log', mode=’w+’, encoding=’utf-8’) if not os.path.exists(os.path.join(’logs’)):os.mkdir(os.path.join(’logs’)) file_handler = handlers.TimedRotatingFileHandler(os.path.join(’logs’, ’debug.log’), ’midnight’, encoding=’utf-8’) if debug:default_handler.setLevel(logging.DEBUG) else:default_handler.setLevel(15) file_handler.setFormatter(logging.Formatter(fms, datefmt=datefmt)) logger.addHandler(file_handler)def get_timestamp() -> int: ''' 獲取當(dāng)前時(shí)間戳 ''' return int(time.time())def get_time() -> str: ''' 獲取格式化后的時(shí)間 ''' time_now = get_timestamp() time_local = time.localtime(time_now) dt = time.strftime('%Y%m%d_%H%M%S', time_local) return dtdef record(): ''' 錄制過程中要執(zhí)行的檢測與判斷 ''' global p, record_status, last_record_time, kill_times # noqa while True:line = p.stdout.readline().decode()p.stdout.flush()logger.log(15, line.rstrip())if match(’video:[0-9kmgB]* audio:[0-9kmgB]* subtitle:[0-9kmgB]*’, line) or ’Exiting normally’ in line: record_status = False # 如果FFmpeg正常結(jié)束錄制則退出本循環(huán) breakelif match(’frame=[0-9]’, line) or ’Opening’ in line: last_record_time = get_timestamp() # 獲取最后錄制的時(shí)間elif ’Failed to read handshake response’ in line: time.sleep(5) # FFmpeg讀取m3u8流失敗,等個(gè)5s康康會不會恢復(fù) continuetime_diff = get_timestamp() - last_record_time # 計(jì)算上次錄制到目前的時(shí)間差if time_diff >= 65: logger.error(’最后一次錄制到目前已超65s,將嘗試發(fā)送終止信號’) logger.debug(f’間隔時(shí)間:{time_diff}s’) kill_times += 1 p.send_signal(signal.SIGTERM) # 若最后一次錄制到目前已超過65s,則認(rèn)為FFmpeg卡死,嘗試發(fā)送終止信號 time.sleep(0.5) if kill_times >= 3:logger.critical(’由于無法結(jié)束FFmpeg進(jìn)程,將嘗試自我了結(jié)’)sys.exit(1)if ’Immediate exit requested’ in line: logger.info(’FFmpeg已被強(qiáng)制結(jié)束’) breakif p.poll() is not None: # 如果FFmpeg已退出但沒有被上一個(gè)判斷和本循環(huán)第一個(gè)判斷捕捉到,則當(dāng)作異常退出 logger.error(’ffmpeg未正常退出,請檢查日志文件!’) record_status = False breakdef main(): global p, room_id, record_status, last_record_time, kill_times # noqa while True:record_status = Falsewhile True: logger.info(’------------------------------’) logger.info(f’正在檢測直播間:{room_id}’) try:room_info = requests.get(f’https://api.live.bilibili.com/room/v1/Room/get_info?room_id={room_id}’, timeout=5) except (requests.exceptions.ReadTimeout, requests.exceptions.Timeout, requests.exceptions.ConnectTimeout):logger.error(f’無法連接至B站API,等待{check_time}s后重新開始檢測’)time.sleep(check_time)continue live_status = loads(room_info.text)[’data’][’live_status’] if live_status == 1:break elif live_status == 0:logger.info(f’沒有開播,等待{check_time}s重新開始檢測’) time.sleep(check_time)if not os.path.exists(os.path.join(’download’)): try:os.mkdir(os.path.join(’download’)) except: # noqalogger.error(f’無法創(chuàng)建下載文件夾 ↓n{traceback.format_exc()}’)sys.exit(1)if os.path.isfile(os.path.join(’download’)): logger.error(’存在與下載文件夾同名的文件’) sys.exit(1)logger.info(’正在直播,準(zhǔn)備開始錄制’)m3u8_list = requests.get( f’https://api.live.bilibili.com/xlive/web-room/v1/playUrl/playUrl?cid={room_id}&platform=h5&qn=10000’)m3u8_address = loads(m3u8_list.text)[’data’][’durl’][0][’url’]# 下面命令中的timeout單位為微秒,10000000us為10s(https://www.cnblogs.com/zhifa/p/12345376.html)command = [’ffmpeg’, ’-rw_timeout’, ’10000000’, ’-timeout’, ’10000000’, ’-listen_timeout’, ’10000000’, ’-headers’, ’'Accept: */*? Accept-Encoding: gzip, deflate, br? Accept-Language: zh,zh-TW;q=0.9,en-US;q=0.8,en;’ f’q=0.7,zh-CN;q=0.6,ru;q=0.5? Origin: https://live.bilibili.com/{room_id}? ’ ’User-Agent: Mozilla/5.0 (Windows NT 10.0;Win64; x64) ’ ’AppleWebKit/537.36 (KHTML, like Gecko) Chrome/83.0.4103.106 Safari/537.36?'’, ’-i’, m3u8_address, ’-c:v’, ’copy’, ’-c:a’, ’copy’, ’-bsf:a’, ’aac_adtstoasc’, ’-f’, ’segment’, ’-segment_time’, str(segment_time), ’-segment_start_number’, ’1’, os.path.join(’download’, f’[{room_id}]_{get_time()}_part%03d.{file_extensions}’), ’-y’]if debug: logger.debug(’FFmpeg命令如下 ↓’) command_str = ’’ for _ in command:command_str += _ logger.debug(command_str)p = Popen(command, stdin=PIPE, stdout=PIPE, stderr=STDOUT, shell=False)record_status = Truestart_time = last_record_time = get_timestamp()try: t = threading.Thread(target=record) t.start() while True:if not record_status: breakif verbose or debug: time.sleep(20) logger.info(f’--==>>> 已錄制 {round((get_timestamp() - start_time) / 60, 2)} 分鐘 <<<==--’)else: time.sleep(60) logger.info(f’--==>>> 已錄制 {int((get_timestamp() - start_time) / 60)} 分鐘 <<<==--’)if not record_status: breakexcept KeyboardInterrupt: # p.send_signal(signal.CTRL_C_EVENT) logger.info(’停止錄制,等待ffmpeg退出后本程序會自動退出’) logger.info(’若長時(shí)間卡住,請?jiān)俅伟聪耤trl+c (可能會損壞視頻文件)’) logger.info(’Bye!’) sys.exit(0)kill_times = 0logger.info(’FFmpeg已退出,重新開始檢測直播間’)# time.sleep(check_time)if __name__ == ’__main__’: logger.info(’B站直播錄播姬 By: Red_lnn’) logger.info(’如要停止錄制并退出,請按鍵盤 Ctrl+C’) logger.info(’如要修改錄制設(shè)置,請以純文本方式打開.py文件’) logger.info(’準(zhǔn)備開始錄制...’) time.sleep(0.3) try:main() except KeyboardInterrupt:logger.info(’Bye!’)sys.exit(0)config.py(配置文件)

#!/usr/bin/env python3# -*- coding:utf-8 -*-'''*------------以下為可配置項(xiàng)-------------*'''# room_id = 1151716 # 萵苣某人# room_id = 1857249 # Red_lnnroom_id = 1151716 # 要錄制的B站直播間的直播間IDsegment_time = 3600 # 錄播分段時(shí)長(單位:秒)check_time = 60 # 開播檢測間隔(單位:秒)file_extensions = ’flv’ # 錄制文件后綴名(文件格式)verbose = True # 是否打印ffmpeg輸出信息到控制臺debug = False # 是否顯示并保存調(diào)試信息(優(yōu)先級高于 verbose)save_log = True # 是否保存日志信息為文件,同一天多次啟動本腳本會共用同一個(gè)日志文件,每天凌晨分割一次日志文件'''*------------以上為可配置項(xiàng)-------------*'''

以上就是python實(shí)現(xiàn)的B站直播錄播工具的詳細(xì)內(nèi)容,更多關(guān)于python B站直播錄播的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: b站 嗶哩嗶哩 Python
相關(guān)文章:
主站蜘蛛池模板: 日韩精品一区二区三区中文3d | 黄色小视频在线观看免费 | 婷婷四房综合激情五月性色 | 国产99久久亚洲综合精品 | 中文字幕亚洲欧美 | 99视频在线精品自拍 | 国产 日韩 欧美 综合 | 国产在线视频资源 | 欧美黑人双插 | 国产视频www | 97视频在线播放 | 国产精品久久久亚洲第一牛牛 | 最新福利小视频在线播放 | 最新黄色地址 | aaa免费视频 | 国产精品久久久久久久久久久不卡 | 国产精品视频第一页 | 欧美高清在线视频一区二区 | 99免费视频 | 国产麻豆精品视频 | 久草在线视频播放 | 日韩中文字幕久久精品 | 无需付费大片免费在线观看 | 欲色影视天天一区二区三区色香欲 | 免费视频一级片 | 免费看片免 | 国产精品青草久久 | 日韩一级高清 | 国产性生大片免费观看性 | 国产精品国产三级国产爱网 | 午夜激情在线观看 | 国产在线日韩在线 | 午夜视频你懂的 | 97理论三级九七午夜在线观看 | 国产 网红 喷水 播放 | 曰本一级毛片免费播放 | 欧美日韩国产亚洲综合不卡 | 在线永久免费观看的a站视频 | 一道精品视频一区二区三区图片 | 欧美激情一区二区 | 99pao在线视频精品免费 |