# -*- coding: utf-8 -*- from queue import Queue from threading import Thread from time import time, sleep from traceback import format_exc from loguru import logger from enums.ExceptionEnum import ExceptionType from enums.RecordingStatusEnum import RecordingStatus from exception.CustomerException import ServiceException from util.Cv2Utils import check_video_stream, clear_pull_p, build_video_info2, pull_read_video_stream2 from util.QueUtil import put_queue, get_no_block_queue, clear_queue, put_queue_result class PullStreamThread(Thread): __slots__ = ('_command', '_pull_queue', '_hb_queue', '_fb_queue', '_msg', '_context') def __init__(self, *args): super().__init__() self._msg, self._context, self._pull_queue, self._hb_queue, self._fb_queue, self._frame_num = args self._command = Queue() def sendEvent(self, result): put_queue(self._command, result, timeout=10, is_ex=False) class RecordingPullStreamThread(PullStreamThread): def run(self): msg, context, frame_num = self._msg, self._context, self._frame_num request_id, pull_url = msg["request_id"], msg['pull_url'] service = context["service"] pull_stream_timeout = int(service["recording_pull_stream_timeout"]) read_stream_timeout = int(service["cv2_read_stream_timeout"]) service_timeout = int(service["timeout"]) command_queue, pull_queue, fb_queue, hb_queue = self._command, self._pull_queue, self._fb_queue, self._hb_queue width, height, width_height_3, all_frames, w, h = None, None, None, 0, None, None read_start_time, pull_p, ex = None, None, None frame_list, frame_index_list = [], [] stop_ex = True pull_stream_start_time = time() try: logger.info("录屏拉流线程开始启动, requestId: {}", request_id) cv2_init_num, init_pull_num, concurrent_frame = 0, 1, 1 start_time = time() while True: # 检查任务是否超时 if time() - start_time > service_timeout: logger.error("录屏拉流超时, requestId: {}", request_id) raise ServiceException(ExceptionType.TASK_EXCUTE_TIMEOUT.value[0], ExceptionType.TASK_EXCUTE_TIMEOUT.value[1]) # 最终停止拉流 event = get_no_block_queue(command_queue) if event is not None: # 当接收到停止指令,说明不会再处理视频帧了, 直接退出 if 'stop' == event.get("command"): if len(frame_list) > 0: put_queue(pull_queue, (4, (frame_list, frame_index_list, all_frames)), timeout=1) logger.info("录屏拉流线程开始停止, requestId: {}", request_id) break # 主进程异常,停止子线程 if 'stop_ex' == event.get("command"): logger.info("录屏异常拉开始停止拉流线程, requestId: {}", request_id) stop_ex = False break # 如果是离线拉流 if pull_url.startswith('http'): if check_video_stream(width, height): logger.info("开始重新获取视频信息: {}次, requestId: {}", cv2_init_num, request_id) # 当是离线地址重试3次还是拉取不到视频流,关闭拉流管道,返回失败信息 # 目前改为等待5分钟 # if cv2_init_num > 3: if time() - start_time > 300: logger.info("离线拉流重试失败, 重试次数: {}, requestId: {}", cv2_init_num, request_id) raise ServiceException(ExceptionType.OR_VIDEO_ADDRESS_EXCEPTION.value[0], ExceptionType.OR_VIDEO_ADDRESS_EXCEPTION.value[1]) cv2_init_num += 1 width, height, width_height_3, all_frames, w, h = build_video_info2(pull_url, request_id) if width is not None: put_queue(hb_queue, {"status": RecordingStatus.RECORDING_RUNNING.value[0]}, timeout=2) else: # if cv2_init_num < 2: if time() - start_time < 300: put_queue(hb_queue, {"status": RecordingStatus.RECORDING_RETRYING.value[0]}, timeout=2) continue # 当离线视频时, 队列满了, 等待1秒后再试 if pull_queue.full(): logger.info("pull拉流队列满了: {}, requestId: {}", pull_queue.qsize(), request_id) sleep(1) continue # 如果是实时拉流 else: if check_video_stream(width, height): logger.info("开始重新获取视频信息: {}次, requestId: {}", cv2_init_num, request_id) pull_stream_init_timeout = time() - pull_stream_start_time if len(frame_list) > 0: put_queue(pull_queue, (4, (frame_list, frame_index_list, all_frames)), timeout=1) frame_list, frame_index_list = [], [] if pull_stream_init_timeout > pull_stream_timeout: logger.error("开始拉流超时, 超时时间:{}, requestId:{}", pull_stream_init_timeout, request_id) raise ServiceException(ExceptionType.PULLSTREAM_TIMEOUT_EXCEPTION.value[0], ExceptionType.PULLSTREAM_TIMEOUT_EXCEPTION.value[1]) cv2_init_num += 1 width, height, width_height_3, all_frames, w, h = build_video_info2(pull_url, request_id) if width is not None: put_queue(hb_queue, {"status": RecordingStatus.RECORDING_RUNNING.value[0]}, timeout=1) else: if cv2_init_num < 3: put_queue(hb_queue, {"status": RecordingStatus.RECORDING_RETRYING.value[0]}, timeout=1) sleep(1) continue pull_stream_start_time = time() cv2_init_num = 1 frame, pull_p, width, height = pull_read_video_stream2(pull_p, pull_url, width, height, width_height_3, w, h, request_id) if frame is None: if pull_url.startswith('http'): clear_pull_p(pull_p, request_id) logger.info("总帧数: {}, 当前帧数: {}, requestId: {}", all_frames, concurrent_frame, request_id) if len(frame_list) > 0: put_queue(pull_queue, (4, (frame_list, frame_index_list, all_frames)), timeout=1) if concurrent_frame < all_frames - 100: logger.info("离线拉流异常结束:requestId: {}", request_id) raise ServiceException(ExceptionType.READSTREAM_TIMEOUT_EXCEPTION.value[0], ExceptionType.READSTREAM_TIMEOUT_EXCEPTION.value[1]) logger.info("离线拉流线程结束, requestId: {}", request_id) break else: logger.info("获取帧为空, 开始重试: {}次, requestId: {}", init_pull_num, request_id) if len(frame_list) > 0: put_queue(pull_queue, (4, (frame_list, frame_index_list, all_frames)), timeout=1) frame_list, frame_index_list = [], [] if read_start_time is None: read_start_time = time() pull_stream_read_timeout = time() - read_start_time if pull_stream_read_timeout > read_stream_timeout: logger.info("拉流过程中断了重试超时, 超时时间: {}, requestId: {}", pull_stream_read_timeout, request_id) raise ServiceException(ExceptionType.READSTREAM_TIMEOUT_EXCEPTION.value[0], ExceptionType.READSTREAM_TIMEOUT_EXCEPTION.value[1]) init_pull_num += 1 continue init_pull_num = 1 read_start_time = None if pull_queue.full(): sleep(1) logger.info("pull拉流队列满了:{}, requestId: {}", pull_queue.qsize(), request_id) continue frame_list.append(frame) frame_index_list.append(concurrent_frame) if len(frame_list) >= frame_num: put_queue(pull_queue, (4, (frame_list, frame_index_list, all_frames)), timeout=1) frame_list, frame_index_list = [], [] concurrent_frame += 1 del frame except ServiceException as s: ex = s.code, s.msg except Exception: logger.exception("实时拉流异常: {}, requestId:{}", format_exc(), request_id) ex = ExceptionType.SERVICE_INNER_EXCEPTION.value[0], ExceptionType.SERVICE_INNER_EXCEPTION.value[1] finally: clear_pull_p(pull_p, request_id) if stop_ex: if ex: error_code, error_msg = ex result = put_queue_result(pull_queue, (1, error_code, error_msg), timeout=3) else: result = put_queue_result(pull_queue, (2,), timeout=3) if result: # 3分钟超时时间 cr_time = time() while time() - cr_time < 180: event = get_no_block_queue(command_queue) if event is not None: # 当接收到停止指令,说明不会再处理视频帧了, 直接退出 if 'stop' == event.get("command"): logger.info("录屏拉流线程开始停止, requestId: {}", request_id) break sleep(1) clear_queue(command_queue) clear_queue(pull_queue) clear_queue(hb_queue) del frame_list, frame_index_list logger.info("录屏拉流线程结束, requestId: {}", request_id)