RTP - Real-time Transport Protocol

The RTP module recives and transmits, sound and phone-event data for a particular phone call.

The RTP module has two methods that are used by various classes for packet parsing.

RTP.byte_to_bits(byte)
This method converts a single byte into an eight character string of ones and zeros. The byte argument must be a single byte.
RTP.add_bytes(bytes)
This method takes multiple bytes and adds them together into an integer.

Errors

exception RTP.DynamicPayloadType
This may be thrown when you try to int cast a dynamic PayloadType. Most PayloadTypes have a number assigned in RFC 3551 Section 6. However, some are considered to be ‘dynamic’ meaning the PBX/VoIP server will pick an available number, and define it.
exception RTP.RTPParseError
This is thrown by RTPMessage is unable to parse a RTP message. It may also be thrown by RTPClient when it’s unable to encode or decode the RTP packet payload.

Enums

RTP.RTPProtocol

RTPProtocol is an Enum with three attributes. It defines the method that packets are to be sent with. Currently, only AVP is supported.

RTPProtocol.UDP
This means the audio should be sent with pure UDP. Returns 'udp' when string casted.
RTPProtocol.AVP
This means the audio should be sent with RTP Audio/Video Protocol described in :rfc:3551. Returns 'RTP/AVP' when string casted.
RTPProtocol.SAVP
This means the audio should be sent with RTP Secure Audio/Video Protocol described in :rfc:3711. Returns 'RTP/SAVP' when string casted.
RTP.TransmitType

TransmitType is an Enum with four attributes. It describes how the RTPClient should act.

TransmitType.RECVONLY
This means the RTPClient should only recive audio, not transmit it. Returns 'recvonly' when string casted.
TransmitType.SENDRECV
This means the RTPClient should send and receive audio. Returns 'sendrecv' when string casted.
TransmitType.SENDONLY
This means the RTPClient should only send audio, not receive it. Returns 'sendonly' when string casted.
TransmitType.INACTIVE
This means the RTP client should not send or receive audio, and instead wait to be activated. Returns 'inactive' when string casted.
RTP.PayloadType

PayloadType is an Enum with multiple attributes. It described the list of attributes in RFC 3551 Section 6. Currently, only one dynamic event is assigned: telephone-event. Telephone-event is used for sending and recieving DTMF codes. There are a few conflicing names in the RFC as they’re the same codec with varrying options so we will go over the conflicts here. PayloadType has the following attributes:

type.value
This is either the number assigned as PT in the RFC 3551 Section 6 chart, or it is the encoding name if it is dynamic. Int casting the PayloadType will return this number, or raise a DynamicPayloadType error if the protocol is dynamic.
type.rate
This will return the clock rate of the codec.
type.channel
This will return the number of channels the used in the codec, or for Non-codecs like telephone-event, it will return zero.
type.description
This will return the encoding name of the payload. String casting the PayloadType will return this value.
PayloadType.DVI4_8000
This variation of the DVI4 Codec has the attributes: value 5, rate 8000, channel 1, description “DVI4”
PayloadType.DVI4_16000
This variation of the DVI4 Codec has the attributes: value 6, rate 16000, channel 1, description “DVI4”
PayloadType.DVI4_11025
This variation of the DVI4 Codec has the attributes: value 16, rate 11025, channel 1, description “DVI4”
PayloadType.DVI4_22050
This variation of the DVI4 Codec has the attributes: value 17, rate 22050, channel 1, description “DVI4”
PayloadType.L16
This variation of the L16 Codec has the attributes: value 11, rate 44100, channel 1, description “L16”
PayloadType.L16_2
This variation of the L16 Codec has the attributes: value 11, rate 44100, channel 2, description “L16”
PayloadType.EVENT
This is the dynamic non-codec ‘telephone-event’. Telephone-event is used for sending and recieving DTMF codes.

Classes

RTPPacketManager

The RTPPacketManager class utilizes an io.ByteIO that stores either received payloads, or raw audio data waiting to be transmitted.

RTP.RTPPacketManager()

read(length=160)
Reads length bytes from the ByteIO. This will always return the length requested, and will append b'\x00'’s onto the end of the available bytes to achieve this length.
rebuild(reset, offset=0, data=b’’)
This rebuilds the ByteIO if packets are sent out of order. Setting the argument reset to true will wipe all data in the ByteIO and insert in the data in the argument data at the position in the argument offset.
write(offset, data)
Writes the data in the argument data to the ByteIO at the position in the argument offset. RTP data comes with a timestamp that is passed as the offset in this case. This makes it so a hole left by delayed packets can be filled later. If a packet with a timestamp sooner than any other timestamp received, it will rebuild the ByteIO with the new data. If this new position is over 100,000 bytes before the earliest byte, the ByteIO is completely wiped and starts over. This is to prevent Overflow errors.

RTPMessage

The RTPMessage class is used to parse RTP packets and makes them easily processed by the RTPClient.

RTP.RTPMessage(data, assoc)

The data argument is the received RTP packet in bytes.

The assoc argument is a dictionary, using the payload number as a key and a PayloadType as the value. This way RTPMessage can determine what a number a dynamic payload is. This association dictionary is generated by VoIPCall.

RTPMessage has attributes that come from RFC 3550 Section 5.1. RTPMessage has the following attributes:

RTPMessage.version
This attribute is the RTP packet version, represented as an integer.
RTPMessage.padding
If this attribute is set to True the payload has padding.
RTPMessage.extension
If this attribute is set to True the packet has a header extension.
RTPMessage.CC
This attribute is the CSRC Count, represented as an integer.
RTPMessage.marker
This attribute is set to True if the marker bit is set.
RTPMessage.payload_type
This attribute is set to the PayloadType that corresponds to the payload codec.
RTPMessage.sequence
This attribute is set to the sequence number of the RTP packet, represented as an integer.
RTPMessage.timestamp
This attribute is set to the timestamp of the RTP packet, represented as an integer.
RTPMessage.SSRC
This attribute is set to the synchronization source of the RTP packet, represented as an integer.
RTPMessage.payload
This attribute is the payload data of the RTP packet, represented as bytes.
RTPMessage.raw
This attribute is the unparsed version of the data argument, in bytes.
summary()
This method returns a string representation of the RTP packet excluding the payload.
parse(data)
This method is called by the initialization of the class. It determines the RTP version, whether the packet has padding, has a header extension, and other information about the backet.

RTPClient

The RTPClient is used to send and receive RTP packets and encode/decode the audio codecs.

class RTP.RTPClient(assoc, inIP, inPort, outIP, outPort, sendrecv, dtmf = None):

The assoc argument is a dictionary, using the payload number as a key and a PayloadType as the value. This way, RTPMessage can determine what a number a dynamic payload is. This association dictionary is generated by VoIPCall.

The inIP argument is used to receive incoming RTP message.

The inPort argument is the port RTPClient will bind to, to receive incoming RTP packets.

The outIP argument is used to transmit RTP packets.

The outPort argument is used to transmit RTP packets.

The sendrecv argument describes how the RTPClient should act. Please reference TransmitType<enums> for more details.

The dtmf argument is set to the callback VoIPCall.dtmfCallback().

start()
This method is called by VoIPCall.answer(). It starts the recv() and trans() threads. It is also what initiates the bound port. This should not be called by the user.
stop()
This method is called by VoIPCall.hangup() and VoIPCall.bye(). It stops the recv() and trans() threads. It will also close the bound port. This should not be called by the user.
read(length=160, blocking=True)
This method is called by VoIPCall.readAudio(). It reads linear/raw audio data from the received buffer. Returns length amount of bytes. Default length is 160 as that is the amount of bytes sent per PCMU/PCMA packet. When blocking is set to true, this function will not return until data is available. When blocking is set to false and data is not available, this function will return bytes(length).
write(data)
This method is called by VoIPCall.writeAudio(). It queues the data written to be sent to the client.
recv()
This method is called by RTPClient.start() and is responsible for receiving and parsing through RTP packets. This should not be called by the user.
trans()
This method is called by RTPClient.start() and is responsible for transmitting RTP packets. This should not be called by the user.
parsePacket(packet)
This method is called by the recv() thread. It converts the argument packet into a RTPMessage, then sends it to the proper parse function depending on the PayloadType.
encodePacket(payload)
This method is called by the trans() thread. It encoded the argument payload into the prefered codec. Currently, PCMU is the hardcoded prefered codec. The trans() thread will use the payload to create the RTP packet before transmitting.
parsePCMU(packet)
This method is called by parsePacket(). It will decode the packet’s payload from PCMU to linear/raw audio and write it to the incoming RTPPacketManager.
encodePCMU(payload)
This method is called by encodePacket(). It will encode the payload into the PCMU audio codec.
parsePCMA(packet)
This method is called by parsePacket(). It will decode the packet’s payload from PCMA to linear/raw audio and write it to the incoming RTPPacketManager.
encodePCMA(payload)
This method is called by encodePacket(). It will encode the payload into the PCMA audio codec.
parseTelephoneEvent(packet)
This method is called by parsePacket(). It will decode the packet’s payload from the telephone-event non-codec to the string representation of the event. It will then call VoIPCall.dtmfCallback().