rtp.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. from .lib import b4ToInt, b2ToInt
  2. __all__ = ["RTP"]
  3. class Header:
  4. """
  5. Reference:
  6. 1. https://datatracker.ietf.org/doc/html/rfc7798
  7. 2. https://blog.csdn.net/u010140427/article/details/127773028
  8. Structure:
  9. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  10. |0 1 2 3 | 10进制分列
  11. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  12. |0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1| 10进制分列
  13. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  14. | Byte | Byte | Byte | Byte | 字节分列,以下为真实header结构
  15. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  16. |V=2|P|X| CC=4 |M| PT=7 | SN=16 |
  17. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  18. | timestamp |
  19. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  20. | SSRC |
  21. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  22. | CSRC |
  23. | .... |
  24. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  25. Notes:
  26. 1、V:RTP协议的版本号,占2位,当前协议版本号为2
  27. 2、P:填充标志,占1位,如果P=1,则在该报文的尾部填充一个或多个额外的八位组,它们不是有效载荷的一部分。
  28. 3、X:扩展标志,占1位,如果X=1,则在RTP报头后跟有一个扩展报头
  29. 4、CC:CSRC计数器,占4位,指示CSRC 标识符的个数
  30. 5、M: 标记,占1位,不同的有效载荷有不同的含义,对于视频,标记一帧的结束;对于音频,标记会话的开始。
  31. 6、PT: 有效荷载类型,占7位,用于说明RTP报文中有效载荷的类型,如GSM音频、JPEM图像等,
  32. 在流媒体中大部分是用来区分音频流和视频流的,这样便于客户端进行解析。
  33. 7、序列号SN:占16位,用于标识发送者所发送的RTP报文的序列号,每发送一个报文,序列号增1。
  34. 这个字段当下层的承载协议用UDP的时候,网络状况不好的时候可以用来检查丢包。同时出现网络抖动的情况可以用来对数据进行重新排序,
  35. 序列号的初始值是随机的,同时音频包和视频包的sequence是分别记数的。
  36. 8、时间戳(timestamp):占32位,必须使用90 kHz时钟频率。时戳反映了该RTP报文的第一个八位组的采样时刻。
  37. 接收者使用时戳来计算延迟和延迟抖动,并进行同步控制。
  38. 9、同步信源(SSRC)标识符:占32位,用于标识同步信源。该标识符是随机选择的,参加同一视频会议的两个同步信源不能有相同的SSRC。
  39. 10、特约信源(CSRC)标识符:每个CSRC标识符占32位,可以有0~15个。每个CSRC标识了包含在该RTP报文有效载荷中的所有特约信源。
  40. !!! 注:基本的RTP说明并不定义任何头扩展本身,如果遇到X=1,需要特殊处理。
  41. Demo:
  42. 假设header数据为:
  43. 80 60 00 01 ==> V_P_X_CC M_PT SN
  44. 00 00 00 00 ==> timestamp
  45. 39 6e d3 46 ==> SSRC
  46. ...
  47. 对应二进制:
  48. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  49. |0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1|
  50. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-| 第一行数据
  51. | 80 | 60 | 00 | 01 | 16进制
  52. |1 0 0 0 0 0 0 0|0 1 1 0 0 0 0 0|0 0 0 0 0 0 0 0|0 0 0 0 0 0 0 1| 2进制
  53. |V=2|P|X| CC=4 |M| PT=7 | SN=16 | 对应位
  54. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-| 第二行数据
  55. | 00 | 00 | 00 | 00 | 16进制
  56. | timestamp |
  57. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-| 第三行数据
  58. | 39 | 6e | d3 | 46 | 16进制
  59. | SSRC |
  60. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  61. ...
  62. """
  63. __SIZE = 12
  64. def __init__(self, header: "bytes"):
  65. self.V = header[0] >> 6
  66. self.P = (header[0] & 0x20) > 0
  67. self.X = (header[0] & 0x10) > 0
  68. self.CC = header[0] & 0x0F
  69. self.M = (header[1] & 0x80) > 0
  70. self.PT = header[1] & 0x7F
  71. self.SN = b2ToInt(header[2:4])
  72. self.timestamp = b4ToInt(header[4:8])
  73. self.SSRC = b4ToInt(header[8:12])
  74. self.size = self.__SIZE + self.CC * 4
  75. self.CSRC = header[self.__SIZE:self.size]
  76. def __str__(self):
  77. return (
  78. f"<RTP-Header: V:{self.V}, P:{self.P}, X:{self.X}, CC:{self.CC}, M:{self.M}, PT:{self.PT}, "
  79. f"SN:{self.SN}, Time:{self.timestamp}, SSRC:{self.SSRC}, CSRC:{self.CSRC}, size:{self.size}>"
  80. )
  81. class Extension:
  82. """
  83. Structure:
  84. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  85. |0 1 2 3 |
  86. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  87. |0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1|
  88. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  89. | defined by profile | length | 长度为本行之后的内容长度,可为0
  90. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  91. | header extension |
  92. | .... |
  93. |-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-|
  94. """
  95. __SIZE = 4
  96. def __init__(self):
  97. self.dbp, self.len, self.size, self.body = b"", 0, 0, b""
  98. def parse(self, ext: "bytes"):
  99. self.dbp = b2ToInt(ext[0:2])
  100. self.len = b2ToInt(ext[2:4])
  101. self.size = self.__SIZE + self.len
  102. self.body = ext[self.__SIZE:self.size]
  103. def __str__(self):
  104. return f"<RTP-Extension: dbp:{self.dbp}, len:{self.len}, body:{self.body}, size:{self.size}>"
  105. class RTP:
  106. def __init__(self, packet: "bytes"):
  107. self.header = Header(packet)
  108. self.extension = Extension()
  109. if self.header.X:
  110. self.extension.parse(packet[self.header.size:])
  111. if self.header.P:
  112. self.payload = packet[self.header.size + self.extension.size:packet[-1]]
  113. else:
  114. self.payload = packet[self.header.size + self.extension.size:]
  115. def __str__(self):
  116. return f"<RTP-Decoder:\n\t{self.header}\n\t{self.extension}\n\tpayload length: {len(self.payload)}\n>"