__init__.py 20 KB

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