树莓派使用python-librtmp实现rtmp推流h264

[复制链接]
 楼主| gaoyang9992006 发表于 2018-2-3 16:07 | 显示全部楼层 |阅读模式
目的是能使用Python进行rtmp推流,方便在h264帧里加入弹幕等操作。
librtmp使用的是0.3.0,使用树莓派noir官方摄像头适配的。
通过wireshark抓ffmpeg的包一点点改动,最终可以在red5和斗鱼上推流了。
  1. # -- coding: utf-8 --
  2. # http://blog.csdn.net/luhanglei
  3. import picamera
  4. import time
  5. import traceback
  6. import ctypes
  7. from librtmp import *

  8. global meta_packet
  9. global start_time


  10. class Writer():  # camera可以通过一个类文件的对象来输出,实现write方法即可
  11.     conn = None  # rtmp连接
  12.     sps = None  # 记录sps帧,发过以后就不需要再发了(抓包看到ffmpeg是这样的)
  13.     pps = None  # 同上
  14.     sps_len = 0  # 同上
  15.     pps_len = 0  # 同上

  16.     time_stamp = 0

  17.     def __init__(self, conn):
  18.         self.conn = conn

  19.     def write(self, data):
  20.         try:
  21.             # 寻找h264帧间隔符
  22.             indexs = []
  23.             index = 0
  24.             data_len = len(data)
  25.             while index < data_len - 3:
  26.                 if ord(data[index]) == 0x00 and ord(data[index + 1]) == 0x00 and ord(
  27.                         data[index + 2]) == 0x00 and ord(data[index + 3]) == 0x01:
  28.                     indexs.append(index)
  29.                     index = index + 3
  30.                 index = index + 1
  31.             # 寻找h264帧间隔符 完成
  32.             # 通过间隔符个数确定类型,树莓派摄像头的第一帧是sps+pps同时发的
  33.             if len(indexs) == 1:  # 非sps pps帧
  34.                 buf = data[4: len(data)]  # 裁掉原来的头(00 00 00 01),把帧内容拿出来
  35.                 buf_len = len(buf)
  36.                 type = ord(buf[0]) & 0x1f
  37.                 if type == 0x05:  # 关键帧,根据wire shark抓包结果,需要拼装sps pps 帧内容 三部分,长度都用4个字节表示
  38.                     body0 = 0x17
  39.                     data_body_array = [bytes(bytearray(
  40.                         [body0, 0x01, 0x00, 0x00, 0x00, (self.sps_len >> 24) & 0xff, (self.sps_len >> 16) & 0xff,
  41.                          (self.sps_len >> 8) & 0xff,
  42.                          self.sps_len & 0xff])), self.sps,
  43.                         bytes(bytearray(
  44.                             [(self.pps_len >> 24) & 0xff, (self.pps_len >> 16) & 0xff, (self.pps_len >> 8) & 0xff,
  45.                              self.pps_len & 0xff])),
  46.                         self.pps,
  47.                         bytes(bytearray(
  48.                             [(buf_len >> 24) & 0xff, (buf_len >> 16) & 0xff, (buf_len >> 8) & 0xff, (buf_len) & 0xff])),
  49.                         buf
  50.                     ]
  51.                     mbody = ''.join(data_body_array)
  52.                     time_stamp = 0  # 第一次发出的时候,发时间戳0,此后发真时间戳
  53.                     if self.time_stamp != 0:
  54.                         time_stamp = int((time.time() - start_time) * 1000)
  55.                     packet_body = RTMPPacket(type=PACKET_TYPE_VIDEO, format=PACKET_SIZE_LARGE, channel=0x06,
  56.                                              timestamp=time_stamp, body=mbody)
  57.                     packet_body.packet.m_nInfoField2 = 1
  58.                 else:  # 非关键帧
  59.                     body0 = 0x27
  60.                     data_body_array = [bytes(bytearray(
  61.                         [body0, 0x01, 0x00, 0x00, 0x00, (buf_len >> 24) & 0xff, (buf_len >> 16) & 0xff,
  62.                          (buf_len >> 8) & 0xff,
  63.                          (buf_len) & 0xff])), buf]
  64.                     mbody = ''.join(data_body_array)
  65.                     # if (self.time_stamp == 0):
  66.                     self.time_stamp = int((time.time() - start_time) * 1000)
  67.                     packet_body = RTMPPacket(type=PACKET_TYPE_VIDEO, format=PACKET_SIZE_MEDIUM, channel=0x06,
  68.                                              timestamp=self.time_stamp, body=mbody)
  69.                 self.conn.send_packet(packet_body)
  70.             elif len(indexs) == 2:  # sps pps帧
  71.                 if self.sps is not None:
  72.                     return
  73.                 data_body_array = [bytes(bytearray([0x17, 0x00, 0x00, 0x00, 0x00, 0x01]))]
  74.                 sps = data[indexs[0] + 4: indexs[1]]
  75.                 sps_len = len(sps)
  76.                 pps = data[indexs[1] + 4: len(data)]
  77.                 pps_len = len(pps)
  78.                 self.sps = sps
  79.                 self.sps_len = sps_len
  80.                 self.pps = pps
  81.                 self.pps_len = pps_len
  82.                 data_body_array.append(sps[1:4])
  83.                 data_body_array.append(bytes(bytearray([0xff, 0xe1, (sps_len >> 8) & 0xff, sps_len & 0xff])))
  84.                 data_body_array.append(sps)
  85.                 data_body_array.append(bytes(bytearray([0x01, (pps_len >> 8) & 0xff, pps_len & 0xff])))
  86.                 data_body_array.append(pps)
  87.                 data_body = ''.join(data_body_array)
  88.                 body_packet = RTMPPacket(type=PACKET_TYPE_VIDEO, format=PACKET_SIZE_LARGE, channel=0x06,
  89.                                          timestamp=0, body=data_body)
  90.                 body_packet.packet.m_nInfoField2 = 1

  91.                 self.conn.send_packet(meta_packet, queue=True)
  92.                 self.conn.send_packet(body_packet, queue=True)
  93.         except Exception, e:
  94.             traceback.print_exc()

  95.     def flush(self):
  96.         pass


  97. def get_property_string(string):  # 返回两字节string长度及string
  98.     length = len(string)
  99.     return ''.join([chr((length >> 8) & 0xff), chr(length & 0xff), string])


  100. def get_meta_string(string):  # 按照meta packet要求格式返回bytes,带02前缀
  101.     return ''.join([chr(0x02), get_property_string(string)])


  102. def get_meta_double(db):
  103.     nums = [0x00]
  104.     fp = ctypes.pointer(ctypes.c_double(db))
  105.     cp = ctypes.cast(fp, ctypes.POINTER(ctypes.c_longlong))
  106.     for i in range(7, -1, -1):
  107.         nums.append((cp.contents.value >> (i * 8)) & 0xff)
  108.     return ''.join(bytes(bytearray(nums)))


  109. def get_meta_boolean(isTrue):
  110.     nums = [0x01]
  111.     if (isTrue):
  112.         nums.append(0x01)
  113.     else:
  114.         nums.append(0x00)
  115.     return ''.join(bytes(bytearray(nums)))


  116. conn = RTMP(
  117.     'rtmp://192.168.199.154/oflaDemo/test',  # 推流地址
  118.     live=True)
  119. librtmp.RTMP_EnableWrite(conn.rtmp)
  120. conn.connect()
  121. start_time = time.time()
  122. # 拼装视频格式的数据包
  123. meta_body_array = [get_meta_string('@setDataFrame'), get_meta_string('onMetaData'),
  124.                    bytes(bytearray([0x08, 0x00, 0x00, 0x00, 0x06])),  # 两个字符串和ECMA array头,共计6个元素,注释掉了音频相关数据
  125.                    get_property_string('width'), get_meta_double(640.0),
  126.                    get_property_string('height'), get_meta_double(480.0),
  127.                    get_property_string('videodatarate'), get_meta_double(0.0),
  128.                    get_property_string('framerate'), get_meta_double(25.0),
  129.                    get_property_string('videocodecid'), get_meta_double(7.0),
  130.                    # get_property_string('audiodatarate'), get_meta_double(125.0),
  131.                    # get_property_string('audiosamplerate'), get_meta_double(44100.0),
  132.                    # get_property_string('audiosamplesize'), get_meta_double(16.0),
  133.                    # get_property_string('stereo'), get_meta_boolean(True),
  134.                    # get_property_string('audiocodecid'), get_meta_double(10.0),
  135.                    get_property_string('encoder'), get_meta_string('Lavf57.56.101'),
  136.                    bytes(bytearray([0x00, 0x00, 0x09]))
  137.                    ]
  138. meta_body = ''.join(meta_body_array)
  139. print meta_body.encode('hex')
  140. meta_packet = RTMPPacket(type=PACKET_TYPE_INFO, format=PACKET_SIZE_LARGE, channel=0x04,
  141.                          timestamp=0, body=meta_body)
  142. meta_packet.packet.m_nInfoField2 = 1  # 修改stream id
  143. stream = conn.create_stream(writeable=True)
  144. with picamera.PiCamera() as camera:
  145.     camera.start_preview()
  146.     time.sleep(2)
  147.     camera.start_recording(Writer(conn), format='h264', resize=(640, 480), intra_period=25,
  148.                            quality=25)  # 开始录制,数据输出到Writer的对象里
  149.     while True:#永远不停止
  150.         time.sleep(60)
  151.     camera.stop_recording()
  152.     camera.stop_preview()


您需要登录后才可以回帖 登录 | 注册

本版积分规则

个人签名:如果你觉得我的分享或者答复还可以,请给我点赞,谢谢。

2046

主题

16356

帖子

221

粉丝
快速回复 在线客服 返回列表 返回顶部