node_info.py
Go to the documentation of this file.
00001 # Copyright (c) 2013, Oregon State University
00002 # All rights reserved.
00003 
00004 # Redistribution and use in source and binary forms, with or without
00005 # modification, are permitted provided that the following conditions are met:
00006 #     * Redistributions of source code must retain the above copyright
00007 #       notice, this list of conditions and the following disclaimer.
00008 #     * Redistributions in binary form must reproduce the above copyright
00009 #       notice, this list of conditions and the following disclaimer in the
00010 #       documentation and/or other materials provided with the distribution.
00011 #     * Neither the name of the Oregon State University nor the
00012 #       names of its contributors may be used to endorse or promote products
00013 #       derived from this software without specific prior written permission.
00014 
00015 # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
00016 # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
00017 # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
00018 # DISCLAIMED. IN NO EVENT SHALL OREGON STATE UNIVERSITY BE LIABLE FOR ANY
00019 # DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
00020 # (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
00021 # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
00022 # ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
00023 # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
00024 # SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
00025 
00026 # Author Dan Lazewatsky/lazewatd@engr.orst.edu
00027 
00028 import rosnode
00029 import rospy
00030 import xmlrpclib
00031 import psutil
00032 
00033 ID = '/NODEINFO'
00034 
00035 class NodeInfo(object):
00036     nodes = dict()
00037     
00038     def get_node_info(self, node_name, skip_cache=False):
00039         node_api = rosnode.get_api_uri(rospy.get_master(), node_name, skip_cache=skip_cache)
00040         try:
00041             code, msg, pid = xmlrpclib.ServerProxy(node_api[2]).getPid(ID)
00042             if node_name in self.nodes:
00043                 return self.nodes[node_name]
00044             else:
00045                 try:
00046                     p = psutil.Process(pid)
00047                     self.nodes[node_name] = p
00048                     return p
00049                 except:
00050                     return False
00051         except xmlrpclib.socket.error:
00052             if not skip_cache:
00053                 return self.get_node_info(node_name, skip_cache=True)
00054             else:
00055                 return False
00056 
00057     def get_all_node_info(self):
00058         infos = []
00059         self.remove_dead_nodes()
00060         for node_name in rosnode.get_node_names():
00061             info = self.get_node_info(node_name)
00062             if info is not False: infos.append((node_name, info))
00063         return infos
00064 
00065     def get_all_node_fields(self, fields):
00066         processes = self.get_all_node_info()
00067         infos = []
00068         for name, p in processes:
00069             infos.append(self.as_dict(p, fields + ['cmdline', 'get_memory_info']))
00070             infos[-1]['node_name'] = name
00071         return infos
00072         
00073     def remove_dead_nodes(self):
00074         running_nodes = rosnode.get_node_names()
00075         dead_nodes = [node_name for node_name in self.nodes if node_name not in running_nodes]
00076         for node_name in dead_nodes:
00077             self.nodes.pop(node_name, None)
00078 
00079     def kill_node(self, node_name):
00080         success, fail = rosnode.kill_nodes([node_name])
00081         return node_name in success
00082 
00083     def as_dict(self, p, attrs=[], ad_value=None):
00084         # copied code from psutil.__init__ from a newer version
00085         excluded_names = set(['send_signal', 'suspend', 'resume', 'terminate',
00086                               'kill', 'wait', 'is_running', 'as_dict', 'parent',
00087                               'get_children', 'nice'])
00088         retdict = dict()
00089         for name in set(attrs or dir(p)):
00090             if name.startswith('_'):
00091                 continue
00092             if name.startswith('set_'):
00093                 continue
00094             if name in excluded_names:
00095                 continue
00096             try:
00097                 attr = getattr(p, name)
00098                 if callable(attr):
00099                     if name == 'get_cpu_percent':
00100                         ret = attr(interval=0)
00101                     else:
00102                         ret = attr()
00103                 else:
00104                     ret = attr
00105             except AccessDenied:
00106                 ret = ad_value
00107             except NotImplementedError:
00108                 # in case of not implemented functionality (may happen
00109                 # on old or exotic systems) we want to crash only if
00110                 # the user explicitly asked for that particular attr
00111                 if attrs:
00112                     raise
00113                 continue
00114             if name.startswith('get'):
00115                 if name[3] == '_':
00116                     name = name[4:]
00117                 elif name == 'getcwd':
00118                     name = 'cwd'
00119             retdict[name] = ret
00120         return retdict


rqt_top
Author(s): Dan Lazewatsky
autogenerated on Wed Sep 16 2015 06:58:27