<p>Joshua Colp has uploaded this change for <strong>review</strong>.</p><p><a href="https://gerrit.asterisk.org/6364">View Change</a></p><pre style="font-family: monospace,monospace; white-space: pre-wrap;">tests/rtp/strict_rtp: Add test for strict RTP support.<br><br>This adds a test that sets up a call between two Asterisk<br>instances with strict RTP enabled and symmetric RTP enabled.<br>Once media has started flowing a flood of minimal RTP packets<br>are sent to each instance. If any traffic is received on the<br>port that the RTP packets were sent from the test fails as<br>strict RTP protection did not discard the packets.<br><br>ASTERISK-27013<br><br>Change-Id: Id24462bed062424295d32618658d94ef4209a934<br>---<br>A tests/rtp/strict_rtp/configs/ast1/extensions.conf<br>A tests/rtp/strict_rtp/configs/ast1/pjsip.conf<br>A tests/rtp/strict_rtp/configs/ast1/rtp.conf<br>A tests/rtp/strict_rtp/configs/ast2/extensions.conf<br>A tests/rtp/strict_rtp/configs/ast2/pjsip.conf<br>A tests/rtp/strict_rtp/configs/ast2/rtp.conf<br>A tests/rtp/strict_rtp/strict_rtp.py<br>A tests/rtp/strict_rtp/test-config.yaml<br>A tests/rtp/tests.yaml<br>M tests/tests.yaml<br>10 files changed, 224 insertions(+), 0 deletions(-)<br><br></pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">git pull ssh://gerrit.asterisk.org:29418/testsuite refs/changes/64/6364/1</pre><pre style="font-family: monospace,monospace; white-space: pre-wrap;">diff --git a/tests/rtp/strict_rtp/configs/ast1/extensions.conf b/tests/rtp/strict_rtp/configs/ast1/extensions.conf<br>new file mode 100644<br>index 0000000..eabe04d<br>--- /dev/null<br>+++ b/tests/rtp/strict_rtp/configs/ast1/extensions.conf<br>@@ -0,0 +1,15 @@<br>+<br>+[default]<br>+<br>+; -- Audio Source --<br>+<br>+exten => playback,1,NoOp()<br>+ same => n,Answer()<br>+ same => n,Playback(tt-monkeys)<br>+ same => n,Hangup()<br>+<br>+; -- Sender --<br>+<br>+exten => 1000,1,NoOp()<br>+ same => n,Dial(PJSIP/ast2)<br>+ same => n,Hangup()<br>diff --git a/tests/rtp/strict_rtp/configs/ast1/pjsip.conf b/tests/rtp/strict_rtp/configs/ast1/pjsip.conf<br>new file mode 100644<br>index 0000000..c633496<br>--- /dev/null<br>+++ b/tests/rtp/strict_rtp/configs/ast1/pjsip.conf<br>@@ -0,0 +1,19 @@<br>+[global]<br>+debug=yes<br>+<br>+[local-transport-udp]<br>+type=transport<br>+bind=127.0.0.1:5060<br>+protocol=udp<br>+<br>+[ast2]<br>+type=aor<br>+contact=sip:ast2@127.0.0.2:5060<br>+<br>+[ast2]<br>+type=endpoint<br>+aors=ast2<br>+context=default<br>+allow=!all,ulaw,alaw<br>+direct_media=no<br>+rtp_symmetric=yes<br>diff --git a/tests/rtp/strict_rtp/configs/ast1/rtp.conf b/tests/rtp/strict_rtp/configs/ast1/rtp.conf<br>new file mode 100644<br>index 0000000..25f04dd<br>--- /dev/null<br>+++ b/tests/rtp/strict_rtp/configs/ast1/rtp.conf<br>@@ -0,0 +1,4 @@<br>+[general]<br>+rtpstart=10000<br>+rtpend=10002<br>+strictrtp=yes<br>diff --git a/tests/rtp/strict_rtp/configs/ast2/extensions.conf b/tests/rtp/strict_rtp/configs/ast2/extensions.conf<br>new file mode 100644<br>index 0000000..4848420<br>--- /dev/null<br>+++ b/tests/rtp/strict_rtp/configs/ast2/extensions.conf<br>@@ -0,0 +1,9 @@<br>+<br>+[default]<br>+<br>+; -- Receiver --<br>+<br>+exten => ast2,1,NoOp()<br>+ same => n,Answer()<br>+ same => n,Echo()<br>+<br>diff --git a/tests/rtp/strict_rtp/configs/ast2/pjsip.conf b/tests/rtp/strict_rtp/configs/ast2/pjsip.conf<br>new file mode 100644<br>index 0000000..4db6013<br>--- /dev/null<br>+++ b/tests/rtp/strict_rtp/configs/ast2/pjsip.conf<br>@@ -0,0 +1,24 @@<br>+[global]<br>+debug=yes<br>+<br>+[local-transport-udp]<br>+type=transport<br>+bind=127.0.0.2:5060<br>+protocol=udp<br>+<br>+[ast1]<br>+type=identify<br>+endpoint=ast1<br>+match=127.0.0.1<br>+<br>+[ast1]<br>+type=aor<br>+contact=sip:ast1@127.0.0.1:5060<br>+<br>+[ast1]<br>+type=endpoint<br>+aors=ast1<br>+context=default<br>+allow=!all,ulaw,alaw<br>+direct_media=no<br>+rtp_symmetric=yes<br>diff --git a/tests/rtp/strict_rtp/configs/ast2/rtp.conf b/tests/rtp/strict_rtp/configs/ast2/rtp.conf<br>new file mode 100644<br>index 0000000..cb64420<br>--- /dev/null<br>+++ b/tests/rtp/strict_rtp/configs/ast2/rtp.conf<br>@@ -0,0 +1,4 @@<br>+[general]<br>+rtpstart=10100<br>+rtpend=10102<br>+strictrtp=yes<br>diff --git a/tests/rtp/strict_rtp/strict_rtp.py b/tests/rtp/strict_rtp/strict_rtp.py<br>new file mode 100644<br>index 0000000..a782152<br>--- /dev/null<br>+++ b/tests/rtp/strict_rtp/strict_rtp.py<br>@@ -0,0 +1,96 @@<br>+"""Strict RTP verification<br>+<br>+This module sends a flood of RTP packets to a target and considers the<br>+test failed if we receive any traffic back.<br>+<br>+Copyright (C) 2017, Digium, Inc.<br>+Joshua Colp <jcolp@digium.com><br>+<br>+This program is free software, distributed under the terms of<br>+the GNU General Public License Version 2.<br>+"""<br>+<br>+import logging<br>+from datetime import datetime<br>+<br>+from twisted.internet.protocol import DatagramProtocol<br>+from twisted.internet import reactor<br>+<br>+LOGGER = logging.getLogger(__name__)<br>+<br>+<br>+class StrictRtpTester(object):<br>+ """A pluggable module for verifying the strict RTP functionality"""<br>+<br>+ class NoAnswerProtocol(DatagramProtocol):<br>+ """The twisted NoAnswerProtocol that fails the test if any packets are received<br>+ """<br>+<br>+ def __init__(self, test_object):<br>+ """Constructor<br>+<br>+ Keyword Arguments:<br>+ test_object Our one and only test object<br>+ """<br>+ self.test_object = test_object<br>+<br>+ def datagramReceived(self, data, (host, port)):<br>+ """Callback for when a datagram is received<br>+<br>+ Keyword Arguments:<br>+ data The actual packet<br>+ (host, port) Tuple of source host and port<br>+ """<br>+ LOGGER.debug('Packet received from {0}:{1}\n{2}'.format(<br>+ host, port, data))<br>+<br>+ self.test_object.set_passed(False)<br>+<br>+ def __init__(self, module_config, test_object):<br>+ """Constructor<br>+<br>+ Keyword Arguments:<br>+ module_config The configuration for this pluggable module<br>+ test_object The one and only test object<br>+ """<br>+<br>+ self.packet_count = 8<br>+<br>+ # Use the AMI callback to know for sure we are fully booted<br>+ self.test_object = test_object<br>+ test_object.register_ami_observer(self.ami_connect_cb)<br>+<br>+ def ami_connect_cb(self, ami):<br>+ """Callback called when AMI connects<br>+<br>+ Keyword Arguments:<br>+ ami The AMI manager object for our Asterisk instance<br>+ """<br>+ ami.registerEvent('RTCPReceived', self.rtcp_received_handler)<br>+<br>+ def rtcp_received_handler(self, ami, event):<br>+ """RTCPReceived callback<br>+<br>+ Keyword Arguments:<br>+ ami The AMI protocol instance<br>+ event The Newchannel event<br>+ """<br>+ if event['sentpackets'] != '250':<br>+ return<br>+<br>+ self.test_object.set_passed(True)<br>+ protocol = StrictRtpTester.NoAnswerProtocol(self.test_object)<br>+ reactor.listenUDP(0, protocol)<br>+<br>+ # Determine the target of the packets from the RTCPReceived event<br>+ (host, port) = event["to"].split(":")<br>+<br>+ # Construct a minimal RTP header by setting the version to 2<br>+ header = bytearray(12)<br>+ header[0] = (2 << 6) & 0xC0<br>+<br>+ for packet in range(self.packet_count):<br>+ # Set the sequence number to the packet number<br>+ header[2] = (packet & 0xFF00) >> 8<br>+ header[3] = (packet & 0xFF)<br>+ protocol.transport.write(header, (host, int(port) - 1))<br>diff --git a/tests/rtp/strict_rtp/test-config.yaml b/tests/rtp/strict_rtp/test-config.yaml<br>new file mode 100644<br>index 0000000..e5d3c88<br>--- /dev/null<br>+++ b/tests/rtp/strict_rtp/test-config.yaml<br>@@ -0,0 +1,49 @@<br>+testinfo:<br>+ summary: 'Verify the strict RTP functionality of the RTP stack'<br>+ description: |<br>+ This test verifies that during a call with strict RTP and symmetric RTP<br>+ enabled that the act of sending multiple RTP packets at the same time<br>+ does not cause media to be taken away from the existing target. The test<br>+ listens for RTCP events which indicate media is flowing. The target RTP<br>+ address information is extracted from these and multiple basic RTP packets<br>+ sent. If any UDP traffic is received on the port that the packets were<br>+ sent from the test fails. If no packets are received the test passes.<br>+<br>+test-modules:<br>+ add-test-to-search-path: True<br>+ test-object:<br>+ config-section: test-object-config<br>+ typename: 'test_case.TestCaseModule'<br>+ modules:<br>+ -<br>+ typename: 'pluggable_modules.Originator'<br>+ config-section: originator-config<br>+ -<br>+ config-section: 'hangup-monitor'<br>+ typename: 'pluggable_modules.HangupMonitor'<br>+ -<br>+ typename: 'strict_rtp.StrictRtpTester'<br>+ config-section: dummy-config<br>+<br>+test-object-config:<br>+ asterisk-instances: 2<br>+ connect-ami: True<br>+<br>+hangup-monitor:<br>+ ids: '0'<br>+<br>+originator-config:<br>+ channel: 'Local/1000@default'<br>+ exten: 'playback'<br>+ context: 'default'<br>+ priority: 1<br>+ async: True<br>+ trigger: 'ami_connect'<br>+ async: True<br>+<br>+dummy-config:<br>+<br>+properties:<br>+ minversion: '13.16.1'<br>+ dependencies:<br>+ - asterisk : 'chan_pjsip'<br>diff --git a/tests/rtp/tests.yaml b/tests/rtp/tests.yaml<br>new file mode 100644<br>index 0000000..4a5db65<br>--- /dev/null<br>+++ b/tests/rtp/tests.yaml<br>@@ -0,0 +1,3 @@<br>+# Enter tests here in the order they should be considered for execution:<br>+tests:<br>+ - test: 'strict_rtp'<br>diff --git a/tests/tests.yaml b/tests/tests.yaml<br>index e3a0b07..9d7ef0f 100644<br>--- a/tests/tests.yaml<br>+++ b/tests/tests.yaml<br>@@ -37,3 +37,4 @@<br> - dir: 'sorcery'<br> - test: 'remote-test'<br> - dir: 'codecs'<br>+ - dir: 'rtp'<br></pre><p>To view, visit <a href="https://gerrit.asterisk.org/6364">change 6364</a>. To unsubscribe, visit <a href="https://gerrit.asterisk.org/settings">settings</a>.</p><div itemscope itemtype="http://schema.org/EmailMessage"><div itemscope itemprop="action" itemtype="http://schema.org/ViewAction"><link itemprop="url" href="https://gerrit.asterisk.org/6364"/><meta itemprop="name" content="View Change"/></div></div>
<div style="display:none"> Gerrit-Project: testsuite </div>
<div style="display:none"> Gerrit-Branch: master </div>
<div style="display:none"> Gerrit-MessageType: newchange </div>
<div style="display:none"> Gerrit-Change-Id: Id24462bed062424295d32618658d94ef4209a934 </div>
<div style="display:none"> Gerrit-Change-Number: 6364 </div>
<div style="display:none"> Gerrit-PatchSet: 1 </div>
<div style="display:none"> Gerrit-Owner: Joshua Colp <jcolp@digium.com> </div>