[Asterisk-code-review] testsuite: NOTIFY count tests unstable (testsuite[16])
Friendly Automation
asteriskteam at digium.com
Thu Sep 15 13:38:53 CDT 2022
Friendly Automation has submitted this change. ( https://gerrit.asterisk.org/c/testsuite/+/19209 )
Change subject: testsuite: NOTIFY count tests unstable
......................................................................
testsuite: NOTIFY count tests unstable
scenario_iterator - new asterisk testsuite py module to allow
iteration of tests that require a series of sipp scenario(s)
+ ami event(s) pairs. Each iteration is started at the stop
of the previous to prevent collision.
mwi_aggregate and mailbox_count_changes - modified to use
this new method
ASTERISK-30214
Change-Id: Ic7f318bc953416ff8d8673ed4768fef264046fe3
---
A lib/python/asterisk/scenario_iterator.py
M tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/mwi_check.py
M tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/test-config.yaml
M tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/mwi_check.py
M tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/test-config.yaml
5 files changed, 329 insertions(+), 73 deletions(-)
Approvals:
George Joseph: Looks good to me, approved
Friendly Automation: Approved for Submit
diff --git a/lib/python/asterisk/scenario_iterator.py b/lib/python/asterisk/scenario_iterator.py
new file mode 100644
index 0000000..cff1af3
--- /dev/null
+++ b/lib/python/asterisk/scenario_iterator.py
@@ -0,0 +1,231 @@
+'''
+Copyright (C) 2022, Sangoma Technologies Corp
+Mike Bradeen <mbradeen at sangoma.com>
+
+This program is free software, distributed under the terms of
+the GNU General Public License Version 2.
+
+The scenario_iterator is designed to let us start sipp, then perform an ami
+action as a cycle of start sipp->generate event(s)->sipp stops, start sipp...
+This is for tests that are performing the same test different ways or performing
+different iterations of the same test.
+
+An example would be to start a scenario that waits for a NOTIFY, then send the AMI
+to generate that NOTIFY, then starts a second scenario, send a second AMI, etc.
+
+There are two classes for this, the singleIterator and the multiIterator. The
+singleIterator is one-to-one, with each scenario start triggering the next ami and the
+scenario stop triggering the next scenario start. It is fed a pair of lists, one is a
+list of scenarios and the other is a list of AMI actions. After the final scenario,
+we add a 'done' indicator to the sipp side and a final event on the AMI side, example:
+
+scenarios = [
+ {'Name': 'none'},
+ {'Name': 'alice-is-notified-1.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'alice-is-notified-2.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'done'}
+]
+
+actions = [
+ {'Action': 'UserEvent', 'UserEvent': 'testStarted'}
+ {'Action': 'MWIUpdate', 'Mailbox': 'alice', 'NewMessages':'2', 'OldMessages':'0'},
+ {'Action': 'MWIDelete', 'Mailbox': 'alice'},
+ {'Action': 'UserEvent', 'UserEvent': 'testComplete'}
+]
+
+You then define a local instance and run it:
+
+def start_test(test_object, junk):
+ testrunner = singleIterator(test_object, scenarios, actions)
+ testrunner.run(junk)
+
+The multiIterator is a many-to-many mapper that allows multiple scenarios to be started, followed
+by multiple actions. The end of the last scenario in the list causes the next scenario to start.
+
+Multiple sequences can be tied to individual actions, or vice versa. In this example we start two
+scenarios but then only generate one corresponding AMI:
+
+scenarios = [
+ {'Name': 'mailbox_a', 'sequence': [
+ {'Name': 'alice-is-notified-1.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'bob-is-notified-1.xml', 'port': '5062', 'target': '127.0.0.1'} ]},
+ {'Name': 'done'}
+]
+
+actions = [
+ {'Messages': [{'Action': 'MWIUpdate', 'Mailbox': 'mailbox_a', 'NewMessages':'2', 'OldMessages':'1'}]},
+ {'Messages': [{'Action': 'UserEvent', 'UserEvent': 'testComplete'}]}
+]
+
+In the above examples, we end the test with a testComplete event that we then use to trigger the
+test stop event in the corresponding yaml file, example:
+ami-config:
+ -
+ ami-events:
+ conditions:
+ match:
+ Event: 'UserEvent'
+ UserEvent: 'testComplete'
+ count: 1
+ stop_test:
+
+Also, this test REQUIRES the following be set in the sipp configuration section of your yaml file if
+you are using a sipp scenario (like a REGISTER) to kick off the sequnce:
+ stop-after-scenarios: False
+
+A Name of 'none' for a scenario(list) or an Action of 'none' skips that particular scenario or action,
+but not the other. This is mostly to allow one or more AMI events to be triggered before the
+corresponding sipp scenario is started but could also allow for intermediate sipp scenarios. Setting
+both to 'none' would be functionally the same as waiting for 1 second before going on to the next
+iteration.
+
+'''
+from asterisk.sipp import SIPpScenarioSequence
+from asterisk.sipp import SIPpScenario
+from twisted.internet import reactor
+import sys
+import logging
+
+sys.path.append("lib/python")
+
+LOGGER = logging.getLogger(__name__)
+
+sipp_terminator = {'Name': 'done'}
+empty_action = {'Action': 'none'}
+empty_action_list = {'Messages': [{'Action': 'none'}] }
+
+class singleIterator(object):
+
+ def __init__(self, test_object, scenarios, actions):
+ self.test_object = test_object
+ self.scenarios = scenarios
+ self.actions = actions
+ self.iteration = 0
+
+ def __iterate(self):
+ try:
+ scenario = self.scenarios[self.iteration]
+ except IndexError:
+ LOGGER.warning("End of scenario list without proper termination")
+ scenario = sipp_terminator
+ try:
+ message = self.actions[self.iteration]
+ except IndexError:
+ LOGGER.warning("End of action list without proper termination")
+ message = empty_action
+
+ self.iteration += 1
+ if scenario['Name'] == 'none':
+ # skip ahead to the next iteration but send the AMI
+ # action if set. Speed up the iteration.
+ self.__sendMessage(message, 0)
+ reactor.callLater(1, self.run)
+ elif scenario['Name'] != 'done':
+ # A scenaro was specified so run it then schedule the
+ # AMI event if there is one.
+ self.__startScenario(scenario)
+ self.__sendMessage(message)
+ else:
+ # At the final iteration, send any final AMI immediately
+ self.__sendMessage(message, 0)
+
+ def __startScenario(self, scenario):
+ LOGGER.info("Starting sipp scenario %s" % scenario['Name'])
+ sipp_scenario = SIPpScenario(self.test_object.test_name,
+ {'scenario': scenario['Name'],
+ '-p': scenario['port']},
+ target=scenario['target'])
+ exiter = sipp_scenario.run(self.test_object)
+ exiter.addCallback(self.run)
+
+ def __sendMessage(self, message, delay=2):
+ if message['Action'] != 'none':
+ testami = self.test_object.ami[0]
+ LOGGER.info("Scheduling AMI %s" % message['Action'])
+ reactor.callLater(delay, testami.sendMessage, message)
+
+ def run(self, junk=None):
+ self.__iterate()
+
+
+class multiIterator(object):
+
+ def __init__(self, test_object, scenariosequences, actions):
+ self.test_object = test_object
+ self.scenariosequences = scenariosequences
+ self.actions = actions
+ self.iteration = 0
+ self.sequencecounter = 1
+
+ def __iterate(self):
+ try:
+ sippsequence = self.scenariosequences[self.iteration]
+ except IndexError:
+ LOGGER.warning("End of scenarios list without proper termination")
+ sippsequence = sipp_terminator
+ try:
+ messagesequence = self.actions[self.iteration]
+ except IndexError:
+ LOGGER.warning("End of action list without proper termination")
+ messagesequence = empty_action_list
+
+ self.iteration += 1
+ if sippsequence['Name'] == 'none':
+ # skip ahead to the next iteration but send any AMI actions if
+ # set, one second apart. Run the next iteration based on how far
+ # out we've scheduled messages.
+ sequencedelay = self.__sendMessages(messagesequence['Messages'], 1)
+ reactor.callLater(sequencedelay, self.run)
+ elif sippsequence['Name'] != 'done':
+ # A scenaro sequence was specified so run it then schedule the
+ # AMI event(s) normally. Set the delay equal to how many scenarios
+ # were registered (as seconds) to try and scale against complexity
+ LOGGER.info("Starting sipp sequence %s" % sippsequence['Name'])
+ self.__startScenarios(sippsequence['sequence'])
+ self.__sendMessages(messagesequence['Messages'], self.sequencecounter)
+ else:
+ # At the final iteration, send any final AMI
+ sequencedelay = self.__sendMessages(messagesequence['Messages'], 0)
+
+ def __startScenarios(self, sippscenarios):
+
+ sipp_sequence = SIPpScenarioSequence(self.test_object,
+ fail_on_any=True,
+ stop_on_done=False)
+ self.sequencecounter = 0
+ for scenario in sippscenarios:
+ LOGGER.info("Adding scenario %s to sequence" % scenario['Name'])
+ sipp_scenario = SIPpScenario(self.test_object.test_name,
+ {'scenario': scenario['Name'],
+ '-p': scenario['port']},
+ target=scenario['target'])
+ sipp_sequence.register_scenario(sipp_scenario)
+ # keep track of how many scenarios we register so we don't
+ # iterate until we have stop_callback hits for all of them
+ self.sequencecounter += 1
+
+ sipp_sequence.register_scenario_stop_callback(self.run)
+ sipp_sequence.execute()
+
+ def __sendMessages(self, messages, delay = 2):
+ testami = self.test_object.ami[0]
+ messagedelay = delay
+ for message in messages:
+ if message['Action'] !='none':
+ LOGGER.info("Scheduling AMI %s" % message['Action'])
+ reactor.callLater(delay, testami.sendMessage, message)
+ # spread out the messages by 2 seconds
+ messagedelay += 2
+ # return last action's delay + 2, ie when to run the next
+ # command if you want it evenly distributed and after the
+ # last message scheduled here
+ return messagedelay
+
+ def run(self, junk=None):
+ # only run the next iteration once there is a call-back for
+ # each sipp scenario in the sequence. Otherwise, decrement
+ # until we get there
+ if self.sequencecounter == 1:
+ self.__iterate()
+ else:
+ self.sequencecounter -= 1
diff --git a/tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/mwi_check.py b/tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/mwi_check.py
index 9a5d5e3..9fadeb0 100644
--- a/tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/mwi_check.py
+++ b/tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/mwi_check.py
@@ -8,32 +8,39 @@
the GNU General Public License Version 2.
"""
+import mailbox
import sys
import logging
sys.path.append("lib/python")
-from twisted.internet import reactor
+from asterisk.scenario_iterator import multiIterator
LOGGER = logging.getLogger(__name__)
-mwis = [
- {'mailbox': 'mailbox_a', 'new': '2', 'old': '1'},
- {'mailbox': 'mailbox_b', 'new': '3', 'old': '3'},
+mwiscenarios = [
+ {'Name': 'mailbox_a', 'sequence': [
+ {'Name': 'alice-is-notified-1.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'bob-is-notified-1.xml', 'port': '5062', 'target': '127.0.0.1'} ]},
+ {'Name': 'mailbox_b', 'sequence': [
+ {'Name': 'alice-is-notified-2.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'bob-is-notified-2.xml', 'port': '5062', 'target': '127.0.0.1'} ]},
+ {'Name': 'done'}
]
-def walk_states(test_object, accounts):
+mwis = [
+ {'Messages': [
+ {'Action': 'MWIUpdate', 'Mailbox': 'mailbox_a', 'NewMessages':'2', 'OldMessages':'1'} ]},
+ {'Messages': [
+ {'Action': 'MWIUpdate', 'Mailbox': 'mailbox_b', 'NewMessages':'3', 'OldMessages':'3'} ]},
+ {'Messages': [
+ {'Action': 'UserEvent', 'UserEvent': 'testComplete'} ]}
+]
- testami = test_object.ami[0]
- statedelay = 2
- for mwi in mwis:
- LOGGER.info("Sending MWI update. new: %s, old %s" %
- (mwi['new'],
- mwi['old']))
- message = {
- 'Action': 'MWIUpdate',
- 'Mailbox': mwi['mailbox'],
- 'NewMessages': mwi['new'],
- 'OldMessages': mwi['old']
- }
- reactor.callLater(statedelay, testami.sendMessage, message)
- statedelay += 1
+def start_test(test_object, junk):
+ LOGGER.info("Starting mwi_check")
+ testrunner = multiIterator(test_object, mwiscenarios, mwis)
+ testrunner.run(junk)
+
+def stop_test():
+ return
+
diff --git a/tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/test-config.yaml b/tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/test-config.yaml
index 48a0971..cabe5e2 100644
--- a/tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/test-config.yaml
+++ b/tests/channels/pjsip/subscriptions/mwi/mwi_aggregate/test-config.yaml
@@ -1,5 +1,4 @@
testinfo:
- skip: 'Unstable - ASTERISK-30214'
summary: 'Ensures mailbox state is aggregated or not aggregated when appropriate'
description: |
"Alice and Bob both receive mailbox updates for mailbox_a and mailbox_b. However, Alice
@@ -30,13 +29,18 @@
test-object:
config-section: sipp-config
typename: 'sipp.SIPpTestCase'
+ modules:
+ -
+ config-section: 'ami-config'
+ typename: 'pluggable_modules.EventActionModule'
sipp-config:
connect-ami: 'True'
reactor-timeout: 30
fail-on-any: True
- start_callback_module: 'mwi_check'
- start_callback_method: 'walk_states'
+ stop-after-scenarios: False
+ stop_callback_module: 'mwi_check'
+ stop_callback_method: 'start_test'
test-iterations:
# We pass the initial registers the -aa flag then let them run for a second so they can get the
# 1-2 initial NOTIFY messages we don't care about. The MWI AMI updates are set to start after
@@ -46,12 +50,12 @@
'ordered-args': {'-aa'} }
- { 'key-args': {'scenario': 'bob-registers.xml', '-p': '5062'},
'ordered-args': {'-aa'} }
- # Combining this with the -m makes the scripts regex conditional on call_number
- -
- scenarios:
- - { 'key-args': {'scenario': 'alice-is-notified-1.xml', '-p': '5061'} }
- - { 'key-args': {'scenario': 'bob-is-notified-1.xml', '-p': '5062'} }
- -
- scenarios:
- - { 'key-args': {'scenario': 'alice-is-notified-2.xml', '-p': '5061'} }
- - { 'key-args': {'scenario': 'bob-is-notified-2.xml', '-p': '5062'} }
+ami-config:
+ -
+ ami-events:
+ conditions:
+ match:
+ Event: 'UserEvent'
+ UserEvent: 'testComplete'
+ count: 1
+ stop_test:
diff --git a/tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/mwi_check.py b/tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/mwi_check.py
index 10a4e4f..28c4ded 100644
--- a/tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/mwi_check.py
+++ b/tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/mwi_check.py
@@ -6,36 +6,31 @@
sys.path.append("lib/python")
from twisted.internet import reactor
+from asterisk.scenario_iterator import singleIterator
LOGGER = logging.getLogger(__name__)
-mwis = [
- {'new': '2', 'old': '0'},
- {'new': '1', 'old': '1'},
- {'new': '0', 'old': '2'},
+mwiscenarios = [
+ {'Name': 'alice-is-notified-1.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'alice-is-notified-2.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'alice-is-notified-3.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'alice-is-notified-4.xml', 'port': '5061', 'target': '127.0.0.1'},
+ {'Name': 'done'}
]
-def walk_states(test_object, junk):
+mwis = [
+ {'Action': 'MWIUpdate', 'Mailbox': 'alice', 'NewMessages':'2', 'OldMessages':'0'},
+ {'Action': 'MWIUpdate', 'Mailbox': 'alice', 'NewMessages':'1', 'OldMessages':'1'},
+ {'Action': 'MWIUpdate', 'Mailbox': 'alice', 'NewMessages':'0', 'OldMessages':'2'},
+ {'Action': 'MWIDelete', 'Mailbox': 'alice'},
+ {'Action': 'UserEvent', 'UserEvent': 'testComplete'}
+]
- testami = test_object.ami[0]
- statedelay = 2
- for mwi in mwis:
- LOGGER.info("Sending MWI update. new: %s, old %s" %
- (mwi['new'],
- mwi['old']))
- message = {
- 'Action': 'MWIUpdate',
- 'Mailbox': 'alice',
- 'NewMessages': mwi['new'],
- 'OldMessages': mwi['old']
- }
- reactor.callLater(statedelay, testami.sendMessage, message)
- statedelay += 1
+def start_test(test_object, junk):
+ LOGGER.info("Starting mwi_check")
+ testrunner = singleIterator(test_object, mwiscenarios, mwis)
+ testrunner.run(junk)
- # delete mailbox after walking states
- LOGGER.info("Deleting Mailbox")
- message = {
- 'Action': 'MWIDelete',
- 'Mailbox': 'alice',
- }
- reactor.callLater(statedelay, testami.sendMessage, message)
+def stop_test():
+ return
+
diff --git a/tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/test-config.yaml b/tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/test-config.yaml
index 5cb1150..14f9456 100644
--- a/tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/test-config.yaml
+++ b/tests/channels/pjsip/subscriptions/mwi/unsolicited/mailbox_count_changes/test-config.yaml
@@ -1,5 +1,4 @@
testinfo:
- skip: 'See ASTERISK-30214'
summary: 'Ensures MWI bodies consist of accurate information'
description: |
"Unsolicited MWI notifications are sent to an endpoint as mailbox state updates. sipp
@@ -26,13 +25,18 @@
test-object:
config-section: sipp-config
typename: 'sipp.SIPpTestCase'
+ modules:
+ -
+ config-section: 'ami-config'
+ typename: 'pluggable_modules.EventActionModule'
sipp-config:
connect-ami: 'True'
reactor-timeout: 30
fail-on-any: True
- start_callback_module: 'mwi_check'
- start_callback_method: 'walk_states'
+ stop-after-scenarios: False
+ stop_callback_module: 'mwi_check'
+ stop_callback_method: 'start_test'
test-iterations:
# We pass the initial registers the -aa flag then let them run for a second so they can get the
# 1-2 initial NOTIFY messages we don't care about. The MWI AMI updates are set to start after
@@ -40,16 +44,12 @@
scenarios:
- { 'key-args': {'scenario': 'alice-registers.xml', '-p': '5061'},
'ordered-args': {'-aa'} }
- # Combining this with the -m makes the scripts regex conditional on call_number
- -
- scenarios:
- - { 'key-args': {'scenario': 'alice-is-notified-1.xml', '-p': '5061'} }
- -
- scenarios:
- - { 'key-args': {'scenario': 'alice-is-notified-2.xml', '-p': '5061'} }
- -
- scenarios:
- - { 'key-args': {'scenario': 'alice-is-notified-3.xml', '-p': '5061'} }
- -
- scenarios:
- - { 'key-args': {'scenario': 'alice-is-notified-4.xml', '-p': '5061'} }
+ami-config:
+ -
+ ami-events:
+ conditions:
+ match:
+ Event: 'UserEvent'
+ UserEvent: 'testComplete'
+ count: 1
+ stop_test:
--
To view, visit https://gerrit.asterisk.org/c/testsuite/+/19209
To unsubscribe, or for help writing mail filters, visit https://gerrit.asterisk.org/settings
Gerrit-Project: testsuite
Gerrit-Branch: 16
Gerrit-Change-Id: Ic7f318bc953416ff8d8673ed4768fef264046fe3
Gerrit-Change-Number: 19209
Gerrit-PatchSet: 4
Gerrit-Owner: Michael Bradeen <mbradeen at sangoma.com>
Gerrit-Reviewer: Friendly Automation
Gerrit-Reviewer: George Joseph <gjoseph at digium.com>
Gerrit-MessageType: merged
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.digium.com/pipermail/asterisk-code-review/attachments/20220915/25c3d30c/attachment-0001.html>
More information about the asterisk-code-review
mailing list