app.py
Go to the documentation of this file.
1 # Software License Agreement (BSD License)
2 #
3 # Copyright (c) 2011, Willow Garage, Inc.
4 # All rights reserved.
5 #
6 # Redistribution and use in source and binary forms, with or without
7 # modification, are permitted provided that the following conditions
8 # are met:
9 #
10 # * Redistributions of source code must retain the above copyright
11 # notice, this list of conditions and the following disclaimer.
12 # * Redistributions in binary form must reproduce the above
13 # copyright notice, this list of conditions and the following
14 # disclaimer in the documentation and/or other materials provided
15 # with the distribution.
16 # * Neither the name of Willow Garage, Inc. nor the names of its
17 # contributors may be used to endorse or promote products derived
18 # from this software without specific prior written permission.
19 #
20 # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
21 # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
22 # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
23 # FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
24 # COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
25 # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
26 # BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
27 # LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
28 # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29 # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
30 # ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
31 # POSSIBILITY OF SUCH DAMAGE.
32 #
33 # Revision $Id: app.py 14667 2011-08-12 23:55:04Z pratkanis $
34 
35 # author: leibs, kwc
36 
37 import os
38 import errno
39 import yaml
40 
41 import roslaunch
42 import roslib.names
43 import rospkg
44 from rospkg import ResourceNotFound
45 from .exceptions import AppException, InvalidAppException, NotFoundException, InternalAppException
46 
47 class Interface(object):
48  def __init__(self, subscribed_topics, published_topics):
49  self.subscribed_topics = subscribed_topics
50  self.published_topics = published_topics
51 
52  def __eq__(self, other):
53  if not isinstance(other, Interface):
54  return False
55  return self.subscribed_topics == other.subscribed_topics and \
56  self.published_topics == other.published_topics
57 
58 class Client(object):
59  __slots__ = ['client_type', 'manager_data', 'app_data']
60  def __init__(self, client_type, manager_data, app_data):
61  self.client_type = client_type
62  self.manager_data = manager_data
63  self.app_data = app_data
64 
65  def as_dict(self):
66  return {'client_type': self.client_type, 'manager_data': self.manager_data, 'app_data': self.app_data}
67 
68  def __eq__(self, other):
69  if not isinstance(other, Client):
70  return False
71  return self.client_type == other.client_type and \
72  self.manager_data == other.manager_data and \
73  self.app_data == other.app_data
74 
75  def __repr__(self):
76  return yaml.dump(self.as_dict())
77 
78 class AppDefinition(object):
79  __slots__ = ['name', 'display_name', 'description', 'platform',
80  'launch', 'run', 'interface', 'clients', 'icon', 'plugins', 'plugin_order',
81  'timeout', 'allow_parallel']
82  def __init__(self, name, display_name, description, platform,
83  interface, clients, launch=None, run=None, icon=None, plugins=None, plugin_order=None,
84  timeout=None, allow_parallel=True):
85  self.name = name
86  self.display_name = display_name
87  self.description = description
88  self.platform=platform
89  self.interface = interface
90  self.clients = clients
91  self.launch = launch
92  self.run = run
93  self.icon = icon
94  self.plugins = plugins
95  self.plugin_order = plugin_order
96  self.timeout = timeout
97  self.allow_parallel = allow_parallel
98 
99  def __repr__(self):
100  d = {}
101  for s in self.__slots__:
102  if s == 'clients':
103  d[s] = [c.as_dict() for c in self.clients]
104  else:
105  d[s] = getattr(self, s)
106  return yaml.dump(d)
107  # return "name: %s\ndisplay: %s\ndescription: %s\nplatform: %s\nlaunch: %s\ninterface: %s\nclients: %s"%(self.name, self.display_name, self.description, self.platform, self.launch, self.interface, self.clients)
108 
109  def __eq__(self, other):
110  if not isinstance(other, AppDefinition):
111  return False
112  return self.name == other.name and \
113  self.display_name == other.display_name and \
114  self.description == other.description and \
115  self.platform == other.platform and \
116  self.interface == other.interface and \
117  self.clients == other.clients and \
118  self.launch == other.launch and \
119  self.run == other.run and \
120  self.icon == other.icon
121 
122 def find_resource(resource, rospack=None):
123  """
124  @return: filepath of resource. Does not validate if filepath actually exists.
125 
126  @raise ValueError: if resource is not a valid resource name.
127  @raise rospkg.ResourceNotFound: if package referred
128  to in resource name cannot be found.
129  @raise NotFoundException: if resource does not exist.
130  """
131  p, a = roslib.names.package_resource_name(resource)
132  if not p:
133  raise ValueError("Resource is missing package name: %s"%(resource))
134 
135  if rospack is None:
136  rospack = rospkg.RosPack()
137  matches = roslib.packages.find_resource(p, a, rospack=rospack)
138 
139  # TODO: convert ValueError to better type for better error messages
140  if len(matches) == 1:
141  return matches[0]
142  elif not matches:
143  raise NotFoundException("No resource [%s]"%(resource))
144  else:
145  raise ValueError("Multiple resources named [%s]"%(resource))
146 
148  """
149  @raise IOError: I/O error reading file (e.g. does not exist)
150  @raise InvalidAppException: if app file is invalid
151  """
152  with open(filename,'r') as f:
153  y = yaml.load(f.read())
154  y = y or {} #coerce to dict
155  try:
156  subscribed_topics = y.get('subscribed_topics', {})
157  published_topics = y.get('published_topics', {})
158  except KeyError:
159  raise InvalidAppException("Malformed interface, missing keys")
160  return Interface(published_topics=published_topics, subscribed_topics=subscribed_topics)
161 
162 def _AppDefinition_load_icon_entry(app_data, appfile="UNKNOWN", rospack=None):
163  """
164  @raise InvalidAppExcetion: if app definition is invalid.
165  """
166 
167  if rospack is None:
168  rospack = rospkg.RosPack()
169  # load/validate launch entry
170  try:
171  icon_resource = app_data.get('icon', '')
172  if icon_resource == '':
173  return None
174  icon_filename = find_resource(icon_resource, rospack=rospack)
175  if not icon_filename or not os.path.exists(icon_filename):
176  return None
177  return icon_filename
178  except ValueError as e:
179  raise InvalidAppException("Malformed appfile [%s]: bad icon entry: %s"%(appfile, e))
180  except NotFoundException:
181  # TODO: make this a soft fail?
182  raise InvalidAppException("App file [%s] refers to icon that cannot be found"%(appfile))
183  except ResourceNotFound as e:
184  raise InvalidAppException("App file [%s] refers to package that is not installed: %s"%(appfile, str(e)))
185 
186 def _AppDefinition_load_launch_entry(app_data, appfile="UNKNOWN", rospack=None):
187  """
188  @raise InvalidAppExcetion: if app definition is invalid.
189  """
190  # load/validate launch entry
191  if rospack is None:
192  rospack = rospkg.RosPack()
193  try:
194  launch_resource = app_data.get('launch', '')
195  if launch_resource == '':
196  return None
197  launch = find_resource(launch_resource, rospack=rospack)
198  if not os.path.exists(launch):
199  raise InvalidAppException("Malformed appfile [%s]: refers to launch that does not exist."%(appfile))
200  return launch
201  except ValueError as e:
202  raise InvalidAppException("Malformed appfile [%s]: bad launch entry: %s"%(appfile, e))
203  except NotFoundException:
204  raise InvalidAppException("App file [%s] refers to launch that is not installed"%(appfile))
205  except ResourceNotFound as e:
206  raise InvalidAppException("App file [%s] refers to package that is not installed: %s"%(appfile, str(e)))
207 
208 def _AppDefinition_load_run_args_entry(app_data, appfile="UNKNOWN"):
209  """
210  @raise InvalidAppException: if app definition is invalid.
211  """
212  # load/validate launch entry
213  try:
214  run_args = app_data.get('run_args', '')
215  if run_args == '':
216  return None
217  return run_args
218  except ValueError as e:
219  raise InvalidAppException("Malformed appfile [%s]: bad run_args entry: %s"%(appfile, e))
220 
221 def _AppDefinition_load_run_entry(app_data, appfile="UNKNOWN", rospack=None):
222  """
223  @raise InvalidAppExcetion: if app definition is invalid.
224  """
225  # load/validate run entry
226  if rospack is None:
227  rospack = rospkg.RosPack()
228  try:
229  run_resource = app_data.get('run', '')
230  if run_resource == '':
231  return None
232 
233  # check if file exists
234  run = find_resource(run_resource, rospack=rospack)
235  if not os.path.exists(run):
236  raise InvalidAppException("Malformed appfile [%s]: refers to run that does not exist."%(appfile))
237  # create node
238  p, a = roslib.names.package_resource_name(run_resource)
239  args = _AppDefinition_load_run_args_entry(app_data, appfile)
240  node = roslaunch.core.Node(p, a, args=args, output='screen')
241  return node
242  except ValueError as e:
243  raise InvalidAppException("Malformed appfile [%s]: bad run entry: %s"%(appfile, e))
244  except NotFoundException:
245  raise InvalidAppException("App file [%s] refers to run that is not installed"%(appfile))
246  except ResourceNotFound as e:
247  raise InvalidAppException("App file [%s] refers to package that is not installed: %s"%(appfile, str(e)))
248 
249 def _AppDefinition_load_interface_entry(app_data, appfile="UNKNOWN", rospack=None):
250  """
251  @raise InvalidAppExcetion: if app definition is invalid.
252  """
253  # load/validate interface entry
254  if rospack is None:
255  rospack = rospkg.RosPack()
256  try:
258  find_resource(app_data['interface'], rospack=rospack))
259  except IOError as e:
260  if e.errno == errno.ENOENT:
261  raise InvalidAppException("Malformed appfile [%s]: refers to interface file that does not exist"%(appfile))
262  else:
263  raise InvalidAppException("Error with appfile [%s]: cannot read interface file"%(appfile))
264  except ValueError:
265  raise InvalidAppException("Malformed appfile [%s]: bad interface entry"%(appfile))
266  except ResourceNotFound as e:
267  raise InvalidAppException("App file [%s] refers to package that is not installed: %s"%(appfile, str(e)))
268 
269 def _AppDefinition_load_clients_entry(app_data, appfile="UNKNOWN"):
270  """
271  @raise InvalidAppExcetion: if app definition is invalid.
272  """
273  clients_data = app_data.get('clients', [])
274  clients = []
275  for c in clients_data:
276  for reqd in ['type', 'manager']:
277  if not reqd in c:
278  raise InvalidAppException("Malformed appfile [%s], missing required key [%s]"%(appfile, reqd))
279  client_type = c['type']
280  manager_data = c['manager']
281  if not type(manager_data) == dict:
282  raise InvalidAppException("Malformed appfile [%s]: manager data must be a map"%(appfile))
283 
284  app_data = c.get('app', {})
285  if not type(app_data) == dict:
286  raise InvalidAppException("Malformed appfile [%s]: app data must be a map"%(appfile))
287 
288  clients.append(Client(client_type, manager_data, app_data))
289  return clients
290 
291 
292 def _AppDefinition_load_plugins_entry(app_data, appfile="UNKNOWN"):
293  """
294  @raise InvalidAppException: if app definition is invalid.
295  """
296  # load/validate launch entry
297  try:
298  plugins = app_data.get('plugins', '')
299  if plugins == '':
300  return None
301  for plugin in plugins:
302  for key in ['launch_args', 'plugin_args', 'start_plugin_args', 'stop_plugin_args']:
303  if key in plugin and not type(plugin[key]) == dict:
304  raise InvalidAppException("Malformed appfile [%s]: plugin data(%s) must be a map"%(appfile, key))
305  return plugins
306  except ValueError as e:
307  raise InvalidAppException("Malformed appfile [%s]: bad plugins entry: %s"%(appfile, e))
308 
309 
310 def _AppDefinition_load_plugin_order_entry(app_data, appfile="UNKNOWN"):
311  """
312  @raise InvalidAppException: if app definition is invalid.
313  """
314  # load/validate launch entry
315  try:
316  plugin_order = app_data.get('plugin_order', '')
317  if plugin_order == '':
318  return []
319  return plugin_order
320  except ValueError as e:
321  raise InvalidAppException("Malformed appfile [%s]: bad plugin_order entry: %s"%(appfile, e))
322 
323 
324 def _AppDefinition_load_timeout_entry(app_data, appfile="UNKNOWN"):
325  """
326  @raise InvalidAppException: if app definition is invalid.
327  """
328  # load/validate launch entry
329  try:
330  timeout = app_data.get('timeout', '')
331  if timeout == '':
332  return None
333  return timeout
334  except ValueError as e:
335  raise InvalidAppException("Malformed appfile [%s]: bad timeout entry: %s"%(appfile, e))
336 
337 
338 def _AppDefinition_load_allow_parallel_entry(app_data, appfile="UNKNOWN"):
339  """
340  @raise InvalidAppException: if app definition is invalid.
341  """
342  # load/validate launch entry
343  try:
344  allow_parallel = app_data.get('allow_parallel', '')
345  if allow_parallel == '':
346  return True
347  return allow_parallel
348  except ValueError as e:
349  raise InvalidAppException("Malformed appfile [%s]: bad allow_parallel entry: %s"%(appfile, e))
350 
351 
352 def load_AppDefinition_from_file(appfile, appname, rospack=None):
353  """
354  @raise InvalidAppExcetion: if app definition is invalid.
355  @raise IOError: I/O error reading appfile (e.g. file does not exist).
356  """
357  with open(appfile,'r') as f:
358  app_data = yaml.load(f.read())
359  for reqd in ['interface', 'platform']:
360  if not reqd in app_data:
361  raise InvalidAppException("Malformed appfile [%s], missing required key [%s]"%(appfile, reqd))
362  if not 'launch' in app_data and not 'run' in app_data:
363  raise InvalidAppException("Malformed appfile [%s], must have a [launch] or a [run] key"%(appfile))
364  if 'launch' in app_data and 'run' in app_data:
365  raise InvalidAppException("Malformed appfile [%s], cannot have both [launch] and [run] keys"%(appfile))
366 
367  display_name = app_data.get('display', appname)
368  description = app_data.get('description', '')
369  platform = app_data['platform']
370 
371  if rospack is None:
372  rospack = rospkg.RosPack()
374  app_data, appfile, rospack=rospack)
376  app_data, appfile, rospack=rospack)
378  app_data, appfile, rospack=rospack)
379  clients = _AppDefinition_load_clients_entry(app_data, appfile)
381  app_data, appfile, rospack=rospack)
382  plugins = _AppDefinition_load_plugins_entry(app_data, appfile)
383  plugin_order = _AppDefinition_load_plugin_order_entry(app_data, appfile)
384  timeout = _AppDefinition_load_timeout_entry(app_data, appfile)
386  app_data, appfile)
387 
388  return AppDefinition(appname, display_name, description, platform,
389  interface, clients, launch, run, icon,
390  plugins, plugin_order, timeout, allow_parallel)
391 
392 def load_AppDefinition_by_name(appname, rospack=None):
393  """
394  @raise InvalidAppExcetion: if app definition is invalid.
395  @raise NotFoundExcetion: if app definition is not installed.
396  @raise ValueError: if appname is invalid.
397  """
398 
399  if not appname:
400  raise ValueError("app name is empty")
401 
402  if rospack is None:
403  rospack = rospkg.RosPack()
404  try:
405  appfile = find_resource(appname + '.app', rospack=rospack)
406  except ResourceNotFound as e:
407  raise NotFoundException("Cannot locate app file for %s: package is not installed."%(appname))
408 
409  try:
410  return load_AppDefinition_from_file(appfile, appname, rospack=rospack)
411  except IOError as e:
412  if e.errno == errno.ENOENT:
413  raise NotFoundException("Cannot locate app file for %s."%(appname))
414  else:
415  raise InternalAppException("I/O error loading AppDefinition file: %s."%(e.errno))
def find_resource(resource, rospack=None)
Definition: app.py:122
def _AppDefinition_load_icon_entry(app_data, appfile="UNKNOWN", rospack=None)
Definition: app.py:162
def __init__(self, name, display_name, description, platform, interface, clients, launch=None, run=None, icon=None, plugins=None, plugin_order=None, timeout=None, allow_parallel=True)
Definition: app.py:84
def _AppDefinition_load_launch_entry(app_data, appfile="UNKNOWN", rospack=None)
Definition: app.py:186
def _AppDefinition_load_run_entry(app_data, appfile="UNKNOWN", rospack=None)
Definition: app.py:221
def _AppDefinition_load_run_args_entry(app_data, appfile="UNKNOWN")
Definition: app.py:208
def as_dict(self)
Definition: app.py:65
def load_AppDefinition_by_name(appname, rospack=None)
Definition: app.py:392
def __eq__(self, other)
Definition: app.py:109
def __init__(self, subscribed_topics, published_topics)
Definition: app.py:48
def _AppDefinition_load_allow_parallel_entry(app_data, appfile="UNKNOWN")
Definition: app.py:338
def __eq__(self, other)
Definition: app.py:52
def _AppDefinition_load_clients_entry(app_data, appfile="UNKNOWN")
Definition: app.py:269
def _AppDefinition_load_interface_entry(app_data, appfile="UNKNOWN", rospack=None)
Definition: app.py:249
def __repr__(self)
Definition: app.py:75
def __init__(self, client_type, manager_data, app_data)
Definition: app.py:60
def load_AppDefinition_from_file(appfile, appname, rospack=None)
Definition: app.py:352
def load_Interface_from_file(filename)
Definition: app.py:147
def _AppDefinition_load_plugin_order_entry(app_data, appfile="UNKNOWN")
Definition: app.py:310
def __eq__(self, other)
Definition: app.py:68
def _AppDefinition_load_timeout_entry(app_data, appfile="UNKNOWN")
Definition: app.py:324
def _AppDefinition_load_plugins_entry(app_data, appfile="UNKNOWN")
Definition: app.py:292


app_manager
Author(s): Jeremy Leibs, Ken Conley, Yuki Furuta
autogenerated on Thu Oct 13 2022 02:59:17