task.py 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649
  1. """
  2. Get interface description of GRASS commands
  3. Based on gui/wxpython/gui_modules/menuform.py
  4. Usage:
  5. ::
  6. from grass.script import task as gtask
  7. gtask.command_info('r.info')
  8. (C) 2011 by the GRASS Development Team
  9. This program is free software under the GNU General Public
  10. License (>=v2). Read the file COPYING that comes with GRASS
  11. for details.
  12. .. sectionauthor:: Martin Landa <landa.martin gmail.com>
  13. """
  14. import types
  15. import string
  16. try:
  17. import xml.etree.ElementTree as etree
  18. except ImportError:
  19. import elementtree.ElementTree as etree # Python <= 2.4
  20. from utils import decode, split
  21. from core import *
  22. class grassTask:
  23. """This class holds the structures needed for filling by the parser
  24. Parameter blackList is a dictionary with fixed structure, eg.
  25. ::
  26. blackList = {'items' : {'d.legend' : { 'flags' : ['m'], 'params' : [] }},
  27. 'enabled': True}
  28. :param str path: full path
  29. :param blackList: hide some options in the GUI (dictionary)
  30. """
  31. def __init__(self, path=None, blackList=None):
  32. self.path = path
  33. self.name = _('unknown')
  34. self.params = list()
  35. self.description = ''
  36. self.label = ''
  37. self.flags = list()
  38. self.keywords = list()
  39. self.errorMsg = ''
  40. self.firstParam = None
  41. if blackList:
  42. self.blackList = blackList
  43. else:
  44. self.blackList = {'enabled': False, 'items': {}}
  45. if path is not None:
  46. try:
  47. processTask(tree=etree.fromstring(get_interface_description(path)),
  48. task=self)
  49. except ScriptError as e:
  50. self.errorMsg = e.value
  51. self.define_first()
  52. def define_first(self):
  53. """Define first parameter
  54. :return: name of first parameter
  55. """
  56. if len(self.params) > 0:
  57. self.firstParam = self.params[0]['name']
  58. return self.firstParam
  59. def get_error_msg(self):
  60. """Get error message ('' for no error)
  61. """
  62. return self.errorMsg
  63. def get_name(self):
  64. """Get task name
  65. """
  66. if sys.platform == 'win32':
  67. name, ext = os.path.splitext(self.name)
  68. if ext in ('.py', '.sh'):
  69. return name
  70. else:
  71. return self.name
  72. return self.name
  73. def get_description(self, full=True):
  74. """Get module's description
  75. :param bool full: True for label + desc
  76. """
  77. if self.label:
  78. if full:
  79. return self.label + ' ' + self.description
  80. else:
  81. return self.label
  82. else:
  83. return self.description
  84. def get_keywords(self):
  85. """Get module's keywords
  86. """
  87. return self.keywords
  88. def get_list_params(self, element='name'):
  89. """Get list of parameters
  90. :param str element: element name
  91. """
  92. params = []
  93. for p in self.params:
  94. params.append(p[element])
  95. return params
  96. def get_list_flags(self, element='name'):
  97. """Get list of flags
  98. :param str element: element name
  99. """
  100. flags = []
  101. for p in self.flags:
  102. flags.append(p[element])
  103. return flags
  104. def get_param(self, value, element='name', raiseError=True):
  105. """Find and return a param by name
  106. :param value: param's value
  107. :param str element: element name
  108. :param bool raiseError: True for raise on error
  109. """
  110. try:
  111. for p in self.params:
  112. val = p[element]
  113. if val is None:
  114. continue
  115. if type(val) in (types.ListType, types.TupleType):
  116. if value in val:
  117. return p
  118. elif type(val) == types.StringType:
  119. if p[element][:len(value)] == value:
  120. return p
  121. else:
  122. if p[element] == value:
  123. return p
  124. except KeyError:
  125. pass
  126. if raiseError:
  127. raise ValueError, _("Parameter element '%(element)s' not found: '%(value)s'") % \
  128. { 'element' : element, 'value' : value }
  129. else:
  130. return None
  131. def get_flag(self, aFlag):
  132. """Find and return a flag by name
  133. Raises ValueError when the flag is not found.
  134. :param str aFlag: name of the flag
  135. """
  136. for f in self.flags:
  137. if f['name'] == aFlag:
  138. return f
  139. raise ValueError, _("Flag not found: %s") % aFlag
  140. def get_cmd_error(self):
  141. """Get error string produced by get_cmd(ignoreErrors = False)
  142. :return: list of errors
  143. """
  144. errorList = list()
  145. # determine if suppress_required flag is given
  146. for f in self.flags:
  147. if f['value'] and f['suppress_required']:
  148. return errorList
  149. for p in self.params:
  150. if not p.get('value', '') and p.get('required', False):
  151. if not p.get('default', ''):
  152. desc = p.get('label', '')
  153. if not desc:
  154. desc = p['description']
  155. errorList.append(_("Parameter '%(name)s' (%(desc)s) is missing.") % \
  156. {'name': p['name'], 'desc': desc})
  157. return errorList
  158. def get_cmd(self, ignoreErrors=False, ignoreRequired=False,
  159. ignoreDefault=True):
  160. """Produce an array of command name and arguments for feeding
  161. into some execve-like command processor.
  162. :param bool ignoreErrors: True to return whatever has been built so
  163. far, even though it would not be a correct
  164. command for GRASS
  165. :param bool ignoreRequired: True to ignore required flags, otherwise
  166. '@<required@>' is shown
  167. :param bool ignoreDefault: True to ignore parameters with default values
  168. """
  169. cmd = [self.get_name()]
  170. suppress_required = False
  171. for flag in self.flags:
  172. if flag['value']:
  173. if len(flag['name']) > 1: # e.g. overwrite
  174. cmd += ['--' + flag['name']]
  175. else:
  176. cmd += ['-' + flag['name']]
  177. if flag['suppress_required']:
  178. suppress_required = True
  179. for p in self.params:
  180. if p.get('value', '') == '' and p.get('required', False):
  181. if p.get('default', '') != '':
  182. cmd += ['%s=%s' % (p['name'], p['default'])]
  183. elif ignoreErrors and not suppress_required and not ignoreRequired:
  184. cmd += [('%s=%s' % (p['name'], _('<required>'))).decode('utf-8')]
  185. elif p.get('value', '') == '' and p.get('default', '') != '' and not ignoreDefault:
  186. cmd += ['%s=%s' % (p['name'], p['default'])]
  187. elif p.get('value', '') != '' and \
  188. (p['value'] != p.get('default', '') or not ignoreDefault):
  189. # output only values that have been set, and different from defaults
  190. cmd += ['%s=%s' % (p['name'], p['value'])]
  191. errList = self.get_cmd_error()
  192. if ignoreErrors is False and errList:
  193. raise ValueError, '\n'.join(errList)
  194. return cmd
  195. def get_options(self):
  196. """Get options
  197. """
  198. return {'flags': self.flags, 'params': self.params}
  199. def has_required(self):
  200. """Check if command has at least one required parameter
  201. """
  202. for p in self.params:
  203. if p.get('required', False):
  204. return True
  205. return False
  206. def set_param(self, aParam, aValue, element='value'):
  207. """Set param value/values.
  208. """
  209. try:
  210. param = self.get_param(aParam)
  211. except ValueError:
  212. return
  213. param[element] = aValue
  214. def set_flag(self, aFlag, aValue, element='value'):
  215. """Enable / disable flag.
  216. """
  217. try:
  218. param = self.get_flag(aFlag)
  219. except ValueError:
  220. return
  221. param[element] = aValue
  222. def set_options(self, opts):
  223. """Set flags and parameters
  224. :param opts list of flags and parameters"""
  225. for opt in opts:
  226. if opt[0] == '-': # flag
  227. self.set_flag(opt.lstrip('-'), True)
  228. else: # parameter
  229. key, value = opt.split('=', 1)
  230. self.set_param(key, value)
  231. class processTask:
  232. """A ElementTree handler for the --interface-description output,
  233. as defined in grass-interface.dtd. Extend or modify this and the
  234. DTD if the XML output of GRASS' parser is extended or modified.
  235. :param tree: root tree node
  236. :param task: grassTask instance or None
  237. :param blackList: list of flags/params to hide
  238. :return: grassTask instance
  239. """
  240. def __init__(self, tree, task=None, blackList=None):
  241. if task:
  242. self.task = task
  243. else:
  244. self.task = grassTask()
  245. if blackList:
  246. self.task.blackList = blackList
  247. self.root = tree
  248. self._process_module()
  249. self._process_params()
  250. self._process_flags()
  251. self.task.define_first()
  252. def _process_module(self):
  253. """Process module description
  254. """
  255. self.task.name = self.root.get('name', default='unknown')
  256. # keywords
  257. for keyword in self._get_node_text(self.root, 'keywords').split(','):
  258. self.task.keywords.append(keyword.strip())
  259. self.task.label = self._get_node_text(self.root, 'label')
  260. self.task.description = self._get_node_text(self.root, 'description')
  261. def _process_params(self):
  262. """Process parameters
  263. """
  264. for p in self.root.findall('parameter'):
  265. # gisprompt
  266. node_gisprompt = p.find('gisprompt')
  267. gisprompt = False
  268. age = element = prompt = None
  269. if node_gisprompt is not None:
  270. gisprompt = True
  271. age = node_gisprompt.get('age', '')
  272. element = node_gisprompt.get('element', '')
  273. prompt = node_gisprompt.get('prompt', '')
  274. # value(s)
  275. values = []
  276. values_desc = []
  277. node_values = p.find('values')
  278. if node_values is not None:
  279. for pv in node_values.findall('value'):
  280. values.append(self._get_node_text(pv, 'name'))
  281. desc = self._get_node_text(pv, 'description')
  282. if desc:
  283. values_desc.append(desc)
  284. # keydesc
  285. key_desc = []
  286. node_key_desc = p.find('keydesc')
  287. if node_key_desc is not None:
  288. for ki in node_key_desc.findall('item'):
  289. key_desc.append(ki.text)
  290. if p.get('multiple', 'no') == 'yes':
  291. multiple = True
  292. else:
  293. multiple = False
  294. if p.get('required', 'no') == 'yes':
  295. required = True
  296. else:
  297. required = False
  298. if self.task.blackList['enabled'] and \
  299. self.task.name in self.task.blackList['items'] and \
  300. p.get('name') in self.task.blackList['items'][self.task.name].get('params', []):
  301. hidden = True
  302. else:
  303. hidden = False
  304. self.task.params.append( {
  305. "name" : p.get('name'),
  306. "type" : p.get('type'),
  307. "required" : required,
  308. "multiple" : multiple,
  309. "label" : self._get_node_text(p, 'label'),
  310. "description" : self._get_node_text(p, 'description'),
  311. 'gisprompt' : gisprompt,
  312. 'age' : age,
  313. 'element' : element,
  314. 'prompt' : prompt,
  315. "guisection" : self._get_node_text(p, 'guisection'),
  316. "guidependency" : self._get_node_text(p, 'guidependency'),
  317. "default" : self._get_node_text(p, 'default'),
  318. "values" : values,
  319. "values_desc" : values_desc,
  320. "value" : '',
  321. "key_desc" : key_desc,
  322. "hidden" : hidden
  323. })
  324. def _process_flags(self):
  325. """Process flags
  326. """
  327. for p in self.root.findall('flag'):
  328. if self.task.blackList['enabled'] and \
  329. self.task.name in self.task.blackList['items'] and \
  330. p.get('name') in self.task.blackList['items'][self.task.name].get('flags', []):
  331. hidden = True
  332. else:
  333. hidden = False
  334. if p.find('suppress_required') is not None:
  335. suppress_required = True
  336. else:
  337. suppress_required = False
  338. self.task.flags.append( {
  339. "name" : p.get('name'),
  340. "label" : self._get_node_text(p, 'label'),
  341. "description" : self._get_node_text(p, 'description'),
  342. "guisection" : self._get_node_text(p, 'guisection'),
  343. "suppress_required" : suppress_required,
  344. "value" : False,
  345. "hidden" : hidden
  346. } )
  347. def _get_node_text(self, node, tag, default=''):
  348. """Get node text"""
  349. p = node.find(tag)
  350. if p is not None:
  351. return string.join(string.split(p.text), ' ')
  352. return default
  353. def get_task(self):
  354. """Get grassTask instance"""
  355. return self.task
  356. def convert_xml_to_utf8(xml_text):
  357. # enc = locale.getdefaultlocale()[1]
  358. # modify: fetch encoding from the interface description text(xml)
  359. # e.g. <?xml version="1.0" encoding="GBK"?>
  360. pattern = re.compile('<\?xml[^>]*\Wencoding="([^"]*)"[^>]*\?>')
  361. m = re.match(pattern, xml_text)
  362. if m is None:
  363. return xml_text
  364. #
  365. enc = m.groups()[0]
  366. # modify: change the encoding to "utf-8", for correct parsing
  367. xml_text_utf8 = xml_text.decode(enc).encode("utf-8")
  368. p = re.compile('encoding="' + enc + '"', re.IGNORECASE)
  369. xml_text_utf8 = p.sub('encoding="utf-8"', xml_text_utf8)
  370. return xml_text_utf8
  371. def get_interface_description(cmd):
  372. """Returns the XML description for the GRASS cmd (force text encoding to
  373. "utf-8").
  374. The DTD must be located in $GISBASE/gui/xml/grass-interface.dtd,
  375. otherwise the parser will not succeed.
  376. :param cmd: command (name of GRASS module)
  377. """
  378. try:
  379. p = Popen([cmd, '--interface-description'], stdout=PIPE,
  380. stderr=PIPE)
  381. cmdout, cmderr = p.communicate()
  382. # TODO: do it better (?)
  383. if not cmdout and sys.platform == 'win32':
  384. # we in fact expect pure module name (without extension)
  385. # so, lets remove extension
  386. if cmd.endswith('.py'):
  387. cmd = os.path.splitext(cmd)[0]
  388. if cmd == 'd.rast3d':
  389. sys.path.insert(0, os.path.join(os.getenv('GISBASE'), 'etc',
  390. 'gui', 'scripts'))
  391. p = Popen([sys.executable, get_real_command(cmd),
  392. '--interface-description'],
  393. stdout=PIPE, stderr=PIPE)
  394. cmdout, cmderr = p.communicate()
  395. if cmd == 'd.rast3d':
  396. del sys.path[0] # remove gui/scripts from the path
  397. if p.returncode != 0:
  398. raise ScriptError, _("Unable to fetch interface description for command '%(cmd)s'."
  399. "\n\nDetails: %(det)s") % {'cmd': cmd, 'det': decode(cmderr)}
  400. except OSError as e:
  401. raise ScriptError, _("Unable to fetch interface description for command '%(cmd)s'."
  402. "\n\nDetails: %(det)s") % {'cmd': cmd, 'det': e}
  403. desc = cmdout.replace('grass-interface.dtd',
  404. os.path.join(os.getenv('GISBASE'),
  405. 'gui', 'xml',
  406. 'grass-interface.dtd'))
  407. return convert_xml_to_utf8(desc)
  408. def parse_interface(name, parser=processTask, blackList=None):
  409. """Parse interface of given GRASS module
  410. :param str name: name of GRASS module to be parsed
  411. :param parser:
  412. :param blackList:
  413. """
  414. tree = etree.fromstring(get_interface_description(name))
  415. return parser(tree, blackList=blackList).get_task()
  416. def command_info(cmd):
  417. """Returns meta information for any GRASS command as dictionary
  418. with entries for description, keywords, usage, flags, and
  419. parameters, e.g.
  420. >>> command_info('g.tempfile') # doctest: +NORMALIZE_WHITESPACE
  421. {'keywords': ['general', 'support'], 'params': [{'gisprompt': False,
  422. 'multiple': False, 'name': 'pid', 'guidependency': '', 'default': '',
  423. 'age': None, 'required': True, 'value': '', 'label': '', 'guisection': '',
  424. 'key_desc': [], 'values': [], 'values_desc': [], 'prompt': None,
  425. 'hidden': False, 'element': None, 'type': 'integer', 'description':
  426. 'Process id to use when naming the tempfile'}], 'flags': [{'description':
  427. "Dry run - don't create a file, just prints it's file name", 'value':
  428. False, 'label': '', 'guisection': '', 'suppress_required': False,
  429. 'hidden': False, 'name': 'd'}, {'description': 'Print usage summary',
  430. 'value': False, 'label': '', 'guisection': '', 'suppress_required': False,
  431. 'hidden': False, 'name': 'help'}, {'description': 'Verbose module output',
  432. 'value': False, 'label': '', 'guisection': '', 'suppress_required': False,
  433. 'hidden': False, 'name': 'verbose'}, {'description': 'Quiet module output',
  434. 'value': False, 'label': '', 'guisection': '', 'suppress_required': False,
  435. 'hidden': False, 'name': 'quiet'}], 'description': "Creates a temporary
  436. file and prints it's file name.", 'usage': 'g.tempfile pid=integer [--help]
  437. [--verbose] [--quiet]'}
  438. >>> command_info('v.buffer')
  439. ['vector', 'geometry', 'buffer']
  440. :param str cmd: the command to query
  441. """
  442. task = parse_interface(cmd)
  443. cmdinfo = {}
  444. cmdinfo['description'] = task.get_description()
  445. cmdinfo['keywords'] = task.get_keywords()
  446. cmdinfo['flags'] = flags = task.get_options()['flags']
  447. cmdinfo['params'] = params = task.get_options()['params']
  448. usage = task.get_name()
  449. flags_short = list()
  450. flags_long = list()
  451. for f in flags:
  452. fname = f.get('name', 'unknown')
  453. if len(fname) > 1:
  454. flags_long.append(fname)
  455. else:
  456. flags_short.append(fname)
  457. if len(flags_short) > 1:
  458. usage += ' [-' + ''.join(flags_short) + ']'
  459. for p in params:
  460. ptype = ','.join(p.get('key_desc', []))
  461. if not ptype:
  462. ptype = p.get('type', '')
  463. req = p.get('required', False)
  464. if not req:
  465. usage += ' ['
  466. else:
  467. usage += ' '
  468. usage += p['name'] + '=' + ptype
  469. if p.get('multiple', False):
  470. usage += '[,' + ptype + ',...]'
  471. if not req:
  472. usage += ']'
  473. for key in flags_long:
  474. usage += ' [--' + key + ']'
  475. cmdinfo['usage'] = usage
  476. return cmdinfo
  477. def cmdtuple_to_list(cmd):
  478. """Convert command tuple to list.
  479. :param tuple cmd: GRASS command to be converted
  480. :return: command in list
  481. """
  482. cmdList = []
  483. if not cmd:
  484. return cmdList
  485. cmdList.append(cmd[0])
  486. if 'flags' in cmd[1]:
  487. for flag in cmd[1]['flags']:
  488. cmdList.append('-' + flag)
  489. for flag in ('help', 'verbose', 'quiet', 'overwrite'):
  490. if flag in cmd[1] and cmd[1][flag] is True:
  491. cmdList.append('--' + flag)
  492. for k, v in cmd[1].iteritems():
  493. if k in ('flags', 'help', 'verbose', 'quiet', 'overwrite'):
  494. continue
  495. cmdList.append('%s=%s' % (k, v))
  496. return cmdList
  497. def cmdlist_to_tuple(cmd):
  498. """Convert command list to tuple for run_command() and others
  499. :param list cmd: GRASS command to be converted
  500. :return: command as tuple
  501. """
  502. if len(cmd) < 1:
  503. return None
  504. dcmd = {}
  505. for item in cmd[1:]:
  506. if '=' in item: # params
  507. key, value = item.split('=', 1)
  508. dcmd[str(key)] = str(value).replace('"', '')
  509. elif item[:2] == '--': # long flags
  510. flag = item[2:]
  511. if flag in ('help', 'verbose', 'quiet', 'overwrite'):
  512. dcmd[str(flag)] = True
  513. elif len(item) == 2 and item[0] == '-': # -> flags
  514. if 'flags' not in dcmd:
  515. dcmd['flags'] = ''
  516. dcmd['flags'] += item[1]
  517. else: # unnamed parameter
  518. module = gtask.parse_interface(cmd[0])
  519. dcmd[module.define_first()] = item
  520. return (cmd[0], dcmd)
  521. def cmdstring_to_tuple(cmd):
  522. """Convert command string to tuple for run_command() and others
  523. :param str cmd: command to be converted
  524. :return: command as tuple
  525. """
  526. return cmdlist_to_tuple(split(cmd))