__init__.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645
  1. # -*- coding: utf-8 -*-
  2. """
  3. Created on Thu Jul 12 10:23:15 2012
  4. @author: pietro
  5. """
  6. from __future__ import print_function
  7. import subprocess
  8. import fnmatch
  9. import re
  10. try:
  11. from collections import OrderedDict
  12. except ImportError:
  13. from pygrass.orderdict import OrderedDict
  14. from itertools import izip_longest
  15. from xml.etree.ElementTree import fromstring
  16. import grass
  17. from pygrass.errors import GrassError
  18. #
  19. # this dictionary is used to extract the value of interest from the xml
  20. # the lambda experssion is used to define small simple functions,
  21. # is equivalent to: ::
  22. #
  23. # def f(p):
  24. # return p.text.strip()
  25. #
  26. # and then we call f(p)
  27. #
  28. _GETFROMTAG = {
  29. 'description': lambda p: p.text.strip(),
  30. 'keydesc': lambda p: p.text.strip(),
  31. 'gisprompt': lambda p: dict(p.items()),
  32. 'default': lambda p: p.text.strip(),
  33. 'values': lambda p: [e.text.strip() for e in p.findall('value/name')],
  34. 'value': lambda p: None,
  35. 'guisection': lambda p: p.text.strip(),
  36. 'label': lambda p: p.text.strip(),
  37. 'suppress_required': lambda p: None,
  38. 'keywords': lambda p: p.text.strip(),
  39. }
  40. _GETTYPE = {
  41. 'string': str,
  42. 'integer': int,
  43. 'float': float,
  44. 'double': float,
  45. }
  46. def stdout2dict(stdout, sep='=', default=None, val_type=None, vsep=None):
  47. """Return a dictionary where entries are separated
  48. by newlines and the key and value are separated by `sep' (default: `=').
  49. Use the grass.core.parse_key_val function
  50. sep: key/value separator
  51. default: default value to be used
  52. val_type: value type (None for no cast)
  53. vsep: vertical separator (default os.linesep)
  54. """
  55. return grass.script.core.parse_key_val(stdout, sep, default,
  56. val_type, vsep)
  57. class ParameterError(Exception):
  58. pass
  59. class FlagError(Exception):
  60. pass
  61. def _element2dict(xparameter):
  62. diz = dict(xparameter.items())
  63. for p in xparameter:
  64. if p.tag in _GETFROMTAG:
  65. diz[p.tag] = _GETFROMTAG[p.tag](p)
  66. else:
  67. print('New tag: %s, ignored' % p.tag)
  68. return diz
  69. # dictionary used to create docstring for the objects
  70. _DOC = {
  71. #------------------------------------------------------------
  72. # head
  73. 'head': """{cmd_name}({cmd_params})
  74. Parameters
  75. ----------
  76. """,
  77. #------------------------------------------------------------
  78. # param
  79. 'param': """{name}: {default}{required}{multi}{ptype}
  80. {description}{values}""",
  81. #------------------------------------------------------------
  82. # flag_head
  83. 'flag_head': """
  84. Flags
  85. ------
  86. """,
  87. #------------------------------------------------------------
  88. # flag
  89. 'flag': """{name}: {default}
  90. {description}""",
  91. #------------------------------------------------------------
  92. # foot
  93. 'foot': """
  94. Special Parameters
  95. ------------------
  96. The Module class have some optional parameters which are distinct using a final
  97. underscore.
  98. run_: True, optional
  99. If True execute the module.
  100. finish_: True, optional
  101. If True wait untill the end of the module execution, and store the module
  102. outputs into stdout, stderr attributes of the class.
  103. stdin_: PIPE,
  104. Set the standard input
  105. """}
  106. class Parameter(object):
  107. def __init__(self, xparameter=None, diz=None):
  108. self._value = None
  109. diz = _element2dict(xparameter) if xparameter is not None else diz
  110. if diz is None:
  111. raise TypeError('Xparameter or diz are required')
  112. self.name = diz['name']
  113. self.required = True if diz['required'] == 'yes' else False
  114. self.multiple = True if diz['multiple'] == 'yes' else False
  115. # check the type
  116. if diz['type'] in _GETTYPE:
  117. self.type = _GETTYPE[diz['type']]
  118. self.typedesc = diz['type']
  119. self._type = _GETTYPE[diz['type']]
  120. else:
  121. raise TypeError('New type: %s, ignored' % diz['type'])
  122. self.description = diz.get('description', None)
  123. self.keydesc = diz.get('keydesc', None)
  124. #
  125. # values
  126. #
  127. if 'values' in diz:
  128. try:
  129. # chek if it's a range string: "3-30"
  130. isrange = re.match("(?P<min>\d+)-(?P<max>\d+)",
  131. diz['values'][0])
  132. if isrange:
  133. range_min, range_max = isrange.groups()
  134. self.values = range(int(range_min), int(range_max) + 1)
  135. self.isrange = diz['values'][0]
  136. except TypeError:
  137. self.values = [self._type(i) for i in diz['values']]
  138. self.isrange = False
  139. #
  140. # default
  141. #
  142. self.default = self._type(
  143. diz['default']) if 'default' in diz else None
  144. if self.default is not None:
  145. self._value = self.default
  146. self.guisection = diz.get('guisection', None)
  147. #
  148. # gisprompt
  149. #
  150. if 'gisprompt' in diz:
  151. self.type = diz['gisprompt']['prompt']
  152. self.input = False if diz['gisprompt']['age'] == 'new' else True
  153. else:
  154. self.input = True
  155. def _get_value(self):
  156. return self._value
  157. def _set_value(self, value):
  158. if isinstance(value, list) or isinstance(value, tuple):
  159. if self.multiple:
  160. # check each value
  161. self._value = [
  162. val for val in value if isinstance(value, self._type)]
  163. else:
  164. str_err = 'The Parameter <%s> does not accept multiple inputs'
  165. raise TypeError(str_err % self.name)
  166. elif isinstance(value, self._type):
  167. if hasattr(self, 'values'):
  168. if value in self.values:
  169. self._value = value
  170. else:
  171. raise ValueError('The Parameter <%s>, must be one of: %r' %
  172. (self.name, self.values))
  173. else:
  174. self._value = value
  175. else:
  176. str_err = 'The Parameter <%s>, require: %s, get: %s instead'
  177. raise TypeError(str_err % (self.name, self.typedesc, type(value)))
  178. # here the property function is used to transform value in an attribute
  179. # in this case we define which function must be use to get/set the value
  180. value = property(fget=_get_value, fset=_set_value)
  181. def get_bash(self):
  182. if isinstance(self._value, list) or isinstance(self._value, tuple):
  183. value = ','.join([str(v) for v in self._value])
  184. else:
  185. value = str(self._value)
  186. return """%s=%s""" % (self.name, value)
  187. def get_python(self):
  188. if not self.value:
  189. return ''
  190. return """%s=%r""" % (self.name, self._value)
  191. def __str__(self):
  192. return self.get_bash()
  193. def __repr__(self):
  194. str_repr = "Parameter <%s> (required:%s, type:%s, multiple:%s)"
  195. return str_repr % (self.name,
  196. "yes" if self.required else "no",
  197. self.type if self.type in (
  198. 'raster', 'vector') else self.typedesc,
  199. "yes" if self.multiple else "no")
  200. # here we use property with a decorator, in this way we mask a method as
  201. # a class attribute
  202. @property
  203. def __doc__(self):
  204. """Return the docstring of the parameter
  205. {name}: {default}{required}{multi}{ptype}
  206. {description}{values}"","""
  207. if hasattr(self, 'values'):
  208. if self.isrange:
  209. vals = self.isrange
  210. else:
  211. vals = ', '.join([repr(val) for val in self.values])
  212. else:
  213. vals = False
  214. return _DOC['param'].format(name=self.name,
  215. default=repr(self.default) + ', ' if self.default else '',
  216. required='required, ' if self.required else 'optional, ',
  217. multi='multi' if self.multiple else '',
  218. ptype=self.typedesc, description=self.description,
  219. values='\n Values: {0}'.format(vals) if vals else '')
  220. class TypeDict(OrderedDict):
  221. def __init__(self, dict_type, *args, **kargs):
  222. self.type = dict_type
  223. super(TypeDict, self).__init__(*args, **kargs)
  224. def __setitem__(self, key, value):
  225. if isinstance(value, self.type):
  226. super(TypeDict, self).__setitem__(key, value)
  227. else:
  228. cl = repr(self.type).translate(None, "'<> ").split('.')
  229. str_err = 'The value: %r is not a %s object'
  230. raise TypeError(str_err % (value, cl[-1].title()))
  231. @property
  232. def __doc__(self):
  233. return '\n'.join([self.__getitem__(obj).__doc__
  234. for obj in self.__iter__()])
  235. def __call__(self):
  236. return [self.__getitem__(obj) for obj in self.__iter__()]
  237. class Flag(object):
  238. def __init__(self, xflag=None, diz=None):
  239. self.value = None
  240. diz = _element2dict(xflag) if xflag is not None else diz
  241. self.name = diz['name']
  242. self.special = True if self.name in (
  243. 'verbose', 'overwrite', 'quiet', 'run') else False
  244. self.description = diz['description']
  245. self.default = diz.get('default', None)
  246. self.guisection = diz.get('guisection', None)
  247. def get_bash(self):
  248. if self.value:
  249. if self.special:
  250. return '--%s' % self.name[0]
  251. else:
  252. return '-%s' % self.name
  253. else:
  254. return ''
  255. def get_python(self):
  256. if self.value:
  257. if self.special:
  258. return '%s=True' % self.name
  259. else:
  260. return self.name
  261. else:
  262. return ''
  263. def __str__(self):
  264. return self.get_bash()
  265. def __repr__(self):
  266. return "Flag <%s> (%s)" % (self.name, self.description)
  267. @property
  268. def __doc__(self):
  269. """
  270. {name}: {default}
  271. {description}"""
  272. return _DOC['flag'].format(name=self.name,
  273. default=repr(self.default),
  274. description=self.description)
  275. class Module(object):
  276. """
  277. Python allow developers to not specify all the arguments and
  278. keyword arguments of a method or function.
  279. ::
  280. def f(*args):
  281. for arg in args:
  282. print arg
  283. therefore if we call the function like: ::
  284. >>> f('grass', 'gis', 'modules')
  285. grass
  286. gis
  287. modules
  288. or we can define a new list: ::
  289. >>> words = ['grass', 'gis', 'modules']
  290. >>> f(*words)
  291. grass
  292. gis
  293. modules
  294. we can do the same with keyword arguments, rewrite the above function: ::
  295. def f(*args, **kargs):
  296. for arg in args:
  297. print arg
  298. for key, value in kargs.items():
  299. print "%s = %r" % (key, value)
  300. now we can use the new function, with: ::
  301. >>> f('grass', 'gis', 'modules', os = 'linux', language = 'python')
  302. grass
  303. gis
  304. modules
  305. os = 'linux'
  306. language = 'python'
  307. or, as before we can, define a dictionary and give the dictionary to
  308. the function, like: ::
  309. >>> keywords = {'os' : 'linux', 'language' : 'python'}
  310. >>> f(*words, **keywords)
  311. grass
  312. gis
  313. modules
  314. os = 'linux'
  315. language = 'python'
  316. In the Module class we heavily use this language feature to pass arguments
  317. and keyword arguments to the grass module.
  318. """
  319. def __init__(self, cmd, *args, **kargs):
  320. self.name = cmd
  321. try:
  322. # call the command with --interface-description
  323. get_cmd_xml = subprocess.Popen([cmd, "--interface-description"],
  324. stdout=subprocess.PIPE)
  325. except OSError:
  326. str_err = "Module %r not found, please check that the module exist"
  327. raise GrassError(str_err % self.name)
  328. # get the xml of the module
  329. self.xml = get_cmd_xml.communicate()[0]
  330. # transform and parse the xml into an Element class:
  331. # http://docs.python.org/library/xml.etree.elementtree.html
  332. tree = fromstring(self.xml)
  333. for e in tree:
  334. if e.tag not in ('parameter', 'flag'):
  335. self.__setattr__(e.tag, _GETFROMTAG[e.tag](e))
  336. #
  337. # extract parameters from the xml
  338. #
  339. self.params_list = [Parameter(p) for p in tree.findall("parameter")]
  340. self.inputs = TypeDict(Parameter)
  341. self.outputs = TypeDict(Parameter)
  342. self.required = []
  343. # Insert parameters into input/output and required
  344. for par in self.params_list:
  345. if par.input:
  346. self.inputs[par.name] = par
  347. else:
  348. self.outputs[par.name] = par
  349. if par.required:
  350. self.required.append(par)
  351. #
  352. # extract flags from the xml
  353. #
  354. flags_list = [Flag(f) for f in tree.findall("flag")]
  355. self.flags_dict = TypeDict(Flag)
  356. for flag in flags_list:
  357. self.flags_dict[flag.name] = flag
  358. #
  359. # Add new attributes to the class
  360. #
  361. self.run_ = True
  362. self.finish_ = True
  363. self.stdin_ = None
  364. self.stdout_ = None
  365. self.stderr_ = None
  366. self.popen = None
  367. if args or kargs:
  368. self.__call__(*args, **kargs)
  369. def _get_flags(self):
  370. return ''.join([flg.get_python() for flg in self.flags_dict.values()
  371. if not flg.special])
  372. def _set_flags(self, value):
  373. if isinstance(value, str):
  374. if value == '':
  375. for flg in self.flags_dict.values():
  376. if not flg.special:
  377. flg.value = False
  378. else:
  379. flgs = [flg.name for flg in self.flags_dict.values()
  380. if not flg.special]
  381. # we need to check if the flag is valid, special flags are not
  382. # allow
  383. for val in value:
  384. if val in flgs:
  385. self.flags_dict[val].value = True
  386. else:
  387. str_err = 'Flag not valid: %r, valid flag are: %r'
  388. raise ValueError(str_err % (val, flgs))
  389. else:
  390. raise TypeError('The flags attribute must be a string')
  391. flags = property(fget=_get_flags, fset=_set_flags)
  392. def __call__(self, *args, **kargs):
  393. if not args and not kargs:
  394. self.run()
  395. return
  396. #
  397. # check for extra kargs, set attribute and remove from dictionary
  398. #
  399. if 'flags' in kargs:
  400. self.flags = kargs['flags']
  401. del(kargs['flags'])
  402. if 'run_' in kargs:
  403. self.run_ = kargs['run_']
  404. del(kargs['run_'])
  405. if 'stdin_' in kargs:
  406. self.stdin_ = kargs['stdin_']
  407. del(kargs['stdin_'])
  408. if 'stdout_' in kargs:
  409. self.stdout_ = kargs['stdout_']
  410. del(kargs['stdout_'])
  411. if 'stderr_' in kargs:
  412. self.stderr_ = kargs['stderr_']
  413. del(kargs['stderr_'])
  414. if 'finish_' in kargs:
  415. self.finish_ = kargs['finish_']
  416. del(kargs['finish_'])
  417. #
  418. # check args
  419. #
  420. for param, arg in zip(self.params_list, args):
  421. param.value = arg
  422. for key, val in kargs.items():
  423. if key in self.inputs:
  424. self.inputs[key].value = val
  425. elif key in self.outputs:
  426. self.outputs[key].value = val
  427. elif key in self.flags_dict:
  428. # we need to add this, because some parameters (overwrite,
  429. # verbose and quiet) work like parameters
  430. self.flags_dict[key].value = val
  431. else:
  432. raise ParameterError('%s is not a valid parameter.' % key)
  433. #
  434. # check reqire parameters
  435. #
  436. for par in self.required:
  437. if par.value is None:
  438. raise ParameterError(
  439. "Required parameter <%s> not set." % par.name)
  440. #
  441. # check flags parameters
  442. #
  443. if self.flags:
  444. # check each character in flags
  445. for flag in self.flags:
  446. if flag in self.flags_dict:
  447. self.flags_dict[flag].value = True
  448. else:
  449. raise FlagError('Flag "%s" not valid.' % flag)
  450. #
  451. # check if execute
  452. #
  453. if self.run_:
  454. self.run()
  455. def get_bash(self):
  456. return ' '.join(self.make_cmd())
  457. def get_python(self):
  458. prefix = self.name.split('.')[0]
  459. name = '_'.join(self.name.split('.')[1:])
  460. params = ', '.join([par.get_python() for par in self.params_list
  461. if par.get_python() != ''])
  462. special = ', '.join([flg.get_python()
  463. for flg in self.flags_dict.values()
  464. if flg.special and flg.get_python() != ''])
  465. # pre name par flg special
  466. if self.flags and special:
  467. return "%s.%s(%s, flags=%r, %s)" % (prefix, name, params,
  468. self.flags, special)
  469. elif self.flags:
  470. return "%s.%s(%s, flags=%r)" % (prefix, name, params, self.flags)
  471. elif special:
  472. return "%s.%s(%s, %s)" % (prefix, name, params, special)
  473. else:
  474. return "%s.%s(%s)" % (prefix, name, params)
  475. def __str__(self):
  476. return ' '.join(self.make_cmd())
  477. def __repr__(self):
  478. return "Module(%r)" % self.name
  479. @property
  480. def __doc__(self):
  481. """{cmd_name}({cmd_params})
  482. """
  483. head = _DOC['head'].format(cmd_name=self.name,
  484. cmd_params=('\n' + # go to a new line
  485. # give space under the function name
  486. (' ' * (len(self.name) + 1))).join([', '.join(
  487. # transform each parameter in string
  488. [str(param) for param in line if param is not None])
  489. # make a list of parameters with only 3 param per line
  490. for line in izip_longest(*[iter(self.params_list)] * 3)]),)
  491. params = '\n'.join([par.__doc__ for par in self.params_list])
  492. flags = self.flags_dict.__doc__
  493. return '\n'.join([head, params, _DOC['flag_head'], flags])
  494. def make_cmd(self):
  495. args = [self.name, ]
  496. for par in self.params_list:
  497. if par.value is not None:
  498. args.append(str(par))
  499. for flg in self.flags_dict:
  500. if self.flags_dict[flg].value:
  501. args.append(str(self.flags_dict[flg]))
  502. return args
  503. def run(self, node=None):
  504. cmd = self.make_cmd()
  505. if self.stdin_:
  506. self.stdin = self.stdin_
  507. self.stdin_ = subprocess.PIPE
  508. self.popen = subprocess.Popen(cmd, stdin=self.stdin_,
  509. stdout=self.stdout_,
  510. stderr=self.stderr_)
  511. if self.stdin_:
  512. self.stdout, self.stderr = self.popen.communicate(input=self.stdin)
  513. else:
  514. if self.finish_:
  515. self.popen.wait()
  516. self.stdout, self.stderr = self.popen.communicate()
  517. _CMDS = list(grass.script.core.get_commands()[0])
  518. _CMDS.sort()
  519. class MetaModule(object):
  520. def __init__(self, prefix):
  521. self.prefix = prefix
  522. def __dir__(self):
  523. return [mod[(len(self.prefix) + 1):].replace('.', '_')
  524. for mod in fnmatch.filter(_CMDS, "%s.*" % self.prefix)]
  525. def __getattr__(self, name):
  526. return Module('%s.%s' % (self.prefix, name.replace('_', '.')))
  527. # http://grass.osgeo.org/grass70/manuals/html70_user/full_index.html
  528. #[ d.* | db.* | g.* | i.* | m.* | ps.* | r.* | r3.* | t.* | v.* ]
  529. #
  530. # d.* display commands
  531. # db.* database commands
  532. # g.* general commands
  533. # i.* imagery commands
  534. # m.* miscellaneous commands
  535. # ps.* postscript commands
  536. # r.* raster commands
  537. # r3.* raster3D commands
  538. # t.* temporal commands
  539. # v.* vector commands
  540. display = MetaModule('d')
  541. database = MetaModule('db')
  542. general = MetaModule('g')
  543. imagery = MetaModule('i')
  544. miscellaneous = MetaModule('m')
  545. postscript = MetaModule('ps')
  546. raster = MetaModule('r')
  547. raster3D = MetaModule('r3')
  548. temporal = MetaModule('t')
  549. vector = MetaModule('v')