ntp_monitor.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 the Willow Garage 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 import roslib
00035 roslib.load_manifest('pr2_computer_monitor')
00036 
00037 from diagnostic_msgs.msg import DiagnosticArray, DiagnosticStatus, KeyValue
00038 
00039 import sys
00040 import rospy
00041 import socket
00042 from subprocess import Popen, PIPE
00043 
00044 import time
00045 
00046 import re
00047 
00048 NAME = 'ntp_monitor'
00049 
00050 def ntp_monitor(ntp_hostname, offset=500, self_offset=500, diag_hostname = None, error_offset = 5000000):
00051     pub = rospy.Publisher("/diagnostics", DiagnosticArray)
00052     rospy.init_node(NAME, anonymous=True)
00053     
00054     hostname = socket.gethostname()
00055     if diag_hostname is None:
00056         diag_hostname = hostname
00057 
00058     stat = DiagnosticStatus()
00059     stat.level = 0
00060     stat.name = "NTP offset from "+ diag_hostname + " to " + ntp_hostname
00061     stat.message = "OK"
00062     stat.hardware_id = hostname
00063     stat.values = []
00064 
00065     self_stat = DiagnosticStatus()
00066     self_stat.level = DiagnosticStatus.OK
00067     self_stat.name = "NTP self-offset for "+ diag_hostname
00068     self_stat.message = "OK"
00069     self_stat.hardware_id = hostname
00070     self_stat.values = []
00071     
00072     while not rospy.is_shutdown():
00073         for st,host,off in [(stat,ntp_hostname,offset), (self_stat, hostname,self_offset)]:
00074             try:
00075                 p = Popen(["ntpdate", "-q", host], stdout=PIPE, stdin=PIPE, stderr=PIPE)
00076                 res = p.wait()
00077                 (o,e) = p.communicate()
00078             except OSError, (errno, msg):
00079                 if errno == 4:
00080                     break #ctrl-c interrupt
00081                 else:
00082                     raise
00083             if (res == 0):
00084                 measured_offset = float(re.search("offset (.*),", o).group(1))*1000000
00085 
00086                 st.level = DiagnosticStatus.OK
00087                 st.message = "OK"
00088                 st.values = [ KeyValue("Offset (us)", str(measured_offset)),
00089                               KeyValue("Offset tolerance (us)", str(off)),
00090                               KeyValue("Offset tolerance (us) for Error", str(error_offset)) ]
00091             
00092                 if (abs(measured_offset) > off):
00093                     st.level = DiagnosticStatus.WARN
00094                     st.message = "NTP Offset Too High"
00095                 if (abs(measured_offset) > error_offset):
00096                     st.level = DiagnosticStatus.ERROR
00097                     st.message = "NTP Offset Too High"
00098                                 
00099             else:
00100                 st.level = DiagnosticStatus.ERROR
00101                 st.message = "Error Running ntpdate. Returned %d" % res
00102                 st.values = [ KeyValue("Offset (us)", "N/A"),
00103                               KeyValue("Offset tolerance (us)", str(off)),
00104                               KeyValue("Offset tolerance (us) for Error", str(error_offset)),
00105                               KeyValue("Output", o),
00106                               KeyValue("Errors", e) ]
00107 
00108 
00109         msg = DiagnosticArray()
00110         msg.header.stamp = rospy.get_rostime()
00111         msg.status = [stat, self_stat]
00112         pub.publish(msg)
00113         time.sleep(1)
00114 
00115 def ntp_monitor_main(argv=sys.argv):
00116     import optparse
00117     parser = optparse.OptionParser(usage="usage: ntp_monitor ntp-hostname []")
00118     parser.add_option("--offset-tolerance", dest="offset_tol",
00119                       action="store", default=500,
00120                       help="Offset from NTP host", metavar="OFFSET-TOL")
00121     parser.add_option("--error-offset-tolerance", dest="error_offset_tol",
00122                       action="store", default=5000000,
00123                       help="Offset from NTP host. Above this is error", metavar="OFFSET-TOL")
00124     parser.add_option("--self_offset-tolerance", dest="self_offset_tol", 
00125                       action="store", default=500,
00126                       help="Offset from self", metavar="SELF_OFFSET-TOL")
00127     parser.add_option("--diag-hostname", dest="diag_hostname",
00128                       help="Computer name in diagnostics output (ex: 'c1')",
00129                       metavar="DIAG_HOSTNAME",
00130                       action="store", default=None)
00131     options, args = parser.parse_args(rospy.myargv())
00132 
00133     if (len(args) != 2):
00134         parser.error("Invalid arguments. Must have HOSTNAME [args]. %s" % args)
00135 
00136 
00137     try:
00138         offset = int(options.offset_tol)
00139         self_offset = int(options.self_offset_tol)
00140         error_offset = int(options.error_offset_tol)
00141     except:
00142         parser.error("Offsets must be numbers")        
00143     
00144     ntp_monitor(args[1], offset, self_offset, options.diag_hostname, error_offset)
00145     
00146 
00147 if __name__ == "__main__":
00148     try:
00149         ntp_monitor_main(rospy.myargv())
00150     except KeyboardInterrupt: pass
00151     except SystemExit: pass
00152     except:
00153         import traceback
00154         traceback.print_exc()


pr2_computer_monitor
Author(s): Kevin Watts (watts@willowgarage.com)
autogenerated on Fri Dec 6 2013 20:23:59