random_play.py
Go to the documentation of this file.
00001 #!/usr/bin/env python
00002 # Software License Agreement (BSD License)
00003 #
00004 # Copyright (c) 2008, Willow Garage, Inc.
00005 # All rights reserved.
00006 #
00007 # Redistribution and use in source and binary forms, with or without
00008 # modification, are permitted provided that the following conditions
00009 # are met:
00010 #
00011 #  * Redistributions of source code must retain the above copyright
00012 #    notice, this list of conditions and the following disclaimer.
00013 #  * Redistributions in binary form must reproduce the above
00014 #    copyright notice, this list of conditions and the following
00015 #    disclaimer in the documentation and/or other materials provided
00016 #    with the distribution.
00017 #  * Neither the name of Willow Garage, Inc. nor the names of its
00018 #    contributors may be used to endorse or promote products derived
00019 #    from this software without specific prior written permission.
00020 #
00021 # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
00022 # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
00023 # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
00024 # FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
00025 # COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
00026 # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
00027 # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
00028 # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
00029 # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
00030 # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
00031 # ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
00032 # POSSIBILITY OF SUCH DAMAGE.
00033 #
00034 
00035 import unittest
00036 import rospy
00037 import rostest
00038 import sys
00039 try:
00040     from cStringIO import StringIO
00041 except ImportError:
00042     from io import StringIO
00043 import time
00044 from random_messages import RandomMsgGen
00045 import subprocess
00046 import os
00047 
00048 import genpy
00049 
00050 DELAY = 0.5
00051 
00052 class RandomPlay(unittest.TestCase):
00053 
00054   def msg_cb_topic(self, topic):
00055     def msg_cb(msg):
00056       nowtime = rospy.Time.now()
00057 
00058       if self.start is None:
00059         self.start = nowtime
00060 
00061       nowtime -= self.start
00062 
00063       self.input.append((topic, msg, nowtime.to_sec()))
00064 
00065     return msg_cb
00066 
00067 
00068   def test_random_play(self):
00069     rospy.init_node('random_sub', anonymous=True)
00070 
00071     self.start = None
00072     self.input = []
00073 
00074     self.assertTrue(len(sys.argv[1]) > 3)
00075 
00076     seed    = int(sys.argv[1])
00077     topics  = int(sys.argv[2])
00078     length  = float(sys.argv[3])
00079     scale   = float(sys.argv[4])
00080     self.use_clock = bool(int(sys.argv[5]))
00081 
00082     rmg = RandomMsgGen(int(seed), topics, length)
00083 
00084     subscribers = {}
00085     for (topic, msg_class) in rmg.topics():
00086       subscribers[topic] = rospy.Subscriber(topic, msg_class, self.msg_cb_topic(topic))
00087 
00088     bagpath = os.path.join('/tmp', 'test_rosbag_random_record_%d.bag'%seed)
00089     cmd = ['rosbag', 'play', '-d', str(DELAY), '-r', str(scale)]
00090     
00091     rospy.loginfo(str(cmd))
00092 
00093     if (self.use_clock):
00094       cmd += ['--clock', '--hz', '100']
00095 
00096     cmd += [bagpath]
00097 
00098     try:
00099       f1 = subprocess.Popen(cmd)
00100 
00101       last_input_count = 0
00102       while (len(self.input) < rmg.message_count()):
00103 #        print "\n%d/%d\n"%(len(self.input), rmg.message_count())
00104         time.sleep(1.0)
00105         # abort loop if no input is coming in anymore and process has finished
00106         if len(self.input) == last_input_count:
00107           rc = f1.poll()
00108           if rc is not None:
00109             self.assertEqual(rc, 0)
00110             break
00111         last_input_count = len(self.input)
00112 
00113       self.assertEqual(len(self.input), rmg.message_count())
00114 
00115       max_late = 0
00116       max_early = 0
00117       avg_off = 0
00118       power = 0
00119 
00120       for (expect_topic, expect_msg, expect_time) in rmg.messages():
00121 
00122         if (not self.use_clock):
00123           expect_time /= scale
00124 
00125         buff = StringIO()
00126         expect_msg.serialize(buff)
00127         expect_msg.deserialize(buff.getvalue())
00128 
00129         msg_match = False
00130 
00131         for ind in range(0,100):
00132           (input_topic, input_msg, input_time) = self.input[ind]
00133 
00134           if (genpy.message.strify_message(expect_msg) == genpy.message.strify_message(input_msg)):
00135             msg_match = True
00136             del self.input[ind]
00137 
00138             # stats
00139             diff = input_time - expect_time
00140 
00141             if (diff < max_early):
00142               max_early = diff
00143 
00144             if (diff > max_late):
00145               max_late = diff
00146 
00147             avg_off += diff / rmg.message_count()
00148 
00149             power += (diff**2) / rmg.message_count()
00150 
00151             # Messages can arrive late, but never very early Both of
00152             # these bounds are much larger than they ought to be, but
00153             # you never know with a heavily loaded system.
00154             self.assertTrue(input_time - expect_time > -.5)
00155             self.assertTrue(abs(input_time - expect_time) < .5)
00156             break
00157 
00158         if not msg_match:
00159           print("No match at time: %f" % expect_time)
00160 
00161         self.assertTrue(msg_match)
00162 
00163       print("%f %f %f %f"%(max_early, max_late, avg_off, power))
00164 
00165     finally:
00166       f1.communicate()
00167 
00168     self.assertEqual(f1.returncode, 0)
00169 
00170 if __name__ == '__main__':
00171   rostest.rosrun('test_rosbag', 'random_record_play', RandomPlay, sys.argv)


test_rosbag
Author(s): Tim Field, Jeremy Leibs, James Bowman
autogenerated on Tue Mar 7 2017 03:46:01