vector.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447
  1. """
  2. Vector related functions to be used in Python scripts.
  3. Usage:
  4. ::
  5. from grass.script import vector as grass
  6. grass.vector_db(map)
  7. (C) 2008-2010 by the GRASS Development Team
  8. This program is free software under the GNU General Public
  9. License (>=v2). Read the file COPYING that comes with GRASS
  10. for details.
  11. .. sectionauthor:: Glynn Clements
  12. .. sectionauthor:: Martin Landa <landa.martin gmail.com>
  13. """
  14. from __future__ import absolute_import
  15. import os
  16. from .utils import parse_key_val
  17. from .core import *
  18. from grass.exceptions import CalledModuleError
  19. unicode = str
  20. def vector_db(map, env=None, **kwargs):
  21. """Return the database connection details for a vector map
  22. (interface to `v.db.connect -g`). Example:
  23. >>> vector_db('geology') # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
  24. {1: {'layer': 1, ... 'table': 'geology'}}
  25. :param str map: vector map
  26. :param kwargs: other v.db.connect's arguments
  27. :param env: environment
  28. :return: dictionary
  29. """
  30. s = read_command('v.db.connect', quiet=True, flags='g', map=map, sep=';',
  31. env=env, **kwargs)
  32. result = {}
  33. for l in s.splitlines():
  34. f = l.split(';')
  35. if len(f) != 5:
  36. continue
  37. if '/' in f[0]:
  38. f1 = f[0].split('/')
  39. layer = f1[0]
  40. name = f1[1]
  41. else:
  42. layer = f[0]
  43. name = ''
  44. result[int(layer)] = {
  45. 'layer': int(layer),
  46. 'name': name,
  47. 'table': f[1],
  48. 'key': f[2],
  49. 'database': f[3],
  50. 'driver': f[4] }
  51. return result
  52. def vector_layer_db(map, layer, env=None):
  53. """Return the database connection details for a vector map layer.
  54. If db connection for given layer is not defined, fatal() is called.
  55. :param str map: map name
  56. :param layer: layer number
  57. :param env: environment
  58. :return: parsed output
  59. """
  60. try:
  61. f = vector_db(map, env=env)[int(layer)]
  62. except KeyError:
  63. fatal(_("Database connection not defined for layer %s") % layer)
  64. return f
  65. # run "v.info -c ..." and parse output
  66. def vector_columns(map, layer=None, getDict=True, env=None, **kwargs):
  67. """Return a dictionary (or a list) of the columns for the
  68. database table connected to a vector map (interface to `v.info -c`).
  69. >>> vector_columns('geology', getDict=True) # doctest: +NORMALIZE_WHITESPACE
  70. {'PERIMETER': {'index': 2, 'type': 'DOUBLE PRECISION'}, 'GEOL250_':
  71. {'index': 3, 'type': 'INTEGER'}, 'SHAPE_area': {'index': 6, 'type':
  72. 'DOUBLE PRECISION'}, 'onemap_pro': {'index': 1, 'type': 'DOUBLE
  73. PRECISION'}, 'SHAPE_len': {'index': 7, 'type': 'DOUBLE PRECISION'},
  74. 'cat': {'index': 0, 'type': 'INTEGER'}, 'GEOL250_ID': {'index': 4, 'type':
  75. 'INTEGER'}, 'GEO_NAME': {'index': 5, 'type': 'CHARACTER'}}
  76. >>> vector_columns('geology', getDict=False) # doctest: +NORMALIZE_WHITESPACE
  77. ['cat',
  78. 'onemap_pro',
  79. 'PERIMETER',
  80. 'GEOL250_',
  81. 'GEOL250_ID',
  82. 'GEO_NAME',
  83. 'SHAPE_area',
  84. 'SHAPE_len']
  85. :param str map: map name
  86. :param layer: layer number or name (None for all layers)
  87. :param bool getDict: True to return dictionary of columns otherwise list
  88. of column names is returned
  89. :param kwargs: (v.info's arguments)
  90. :param env: environment
  91. :return: dictionary/list of columns
  92. """
  93. s = read_command('v.info', flags='c', map=map, layer=layer, quiet=True,
  94. env=env, **kwargs)
  95. if getDict:
  96. result = dict()
  97. else:
  98. result = list()
  99. i = 0
  100. for line in s.splitlines():
  101. ctype, cname = line.split('|')
  102. if getDict:
  103. result[cname] = {'type': ctype, 'index': i}
  104. else:
  105. result.append(cname)
  106. i += 1
  107. return result
  108. def vector_history(map, replace=False, env=None):
  109. """Set the command history for a vector map to the command used to
  110. invoke the script (interface to `v.support`).
  111. :param str map: mapname
  112. :param bool replace: Replace command line instead of appending it
  113. :param env: environment
  114. :return: v.support output
  115. """
  116. run_command('v.support', map=map, cmdhist=os.environ['CMDLINE'],
  117. flags='h' if replace else None, env=env)
  118. def vector_info_topo(map, layer=1, env=None):
  119. """Return information about a vector map (interface to `v.info -t`).
  120. Example:
  121. >>> vector_info_topo('geology') # doctest: +NORMALIZE_WHITESPACE
  122. {'lines': 0, 'centroids': 1832, 'boundaries': 3649, 'points': 0,
  123. 'primitives': 5481, 'islands': 907, 'nodes': 2724, 'map3d': False,
  124. 'areas': 1832}
  125. :param str map: map name
  126. :param int layer: layer number
  127. :param env: environment
  128. :return: parsed output
  129. """
  130. s = read_command('v.info', flags='t', layer=layer, map=map,
  131. env=env)
  132. ret = parse_key_val(s, val_type=int)
  133. if 'map3d' in ret:
  134. ret['map3d'] = bool(ret['map3d'])
  135. return ret
  136. def vector_info(map, layer=1, env=None):
  137. """Return information about a vector map (interface to
  138. `v.info`). Example:
  139. >>> vector_info('geology') # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
  140. {'comment': '', 'projection': 'Lambert Conformal Conic' ... 'south': 10875.8272320917}
  141. :param str map: map name
  142. :param int layer: layer number
  143. :param env: environment
  144. :return: parsed vector info
  145. """
  146. s = read_command('v.info', flags='get', layer=layer, map=map,
  147. env=env)
  148. kv = parse_key_val(s)
  149. for k in ['north', 'south', 'east', 'west', 'top', 'bottom']:
  150. kv[k] = float(kv[k])
  151. for k in ['level', 'num_dblinks']:
  152. kv[k] = int(kv[k])
  153. for k in ['nodes', 'points', 'lines', 'boundaries', 'centroids', 'areas',
  154. 'islands', 'primitives']:
  155. kv[k] = int(kv[k])
  156. if 'map3d' in kv:
  157. kv['map3d'] = bool(int(kv['map3d']))
  158. if kv['map3d']:
  159. for k in ['faces', 'kernels', 'volumes', 'holes']:
  160. kv[k] = int(kv[k])
  161. return kv
  162. def vector_db_select(map, layer=1, env=None, **kwargs):
  163. """Get attribute data of selected vector map layer.
  164. Function returns list of columns and dictionary of values ordered by
  165. key column value. Example:
  166. >>> print vector_db_select('geology')['columns']
  167. ['cat', 'onemap_pro', 'PERIMETER', 'GEOL250_', 'GEOL250_ID', 'GEO_NAME', 'SHAPE_area', 'SHAPE_len']
  168. >>> print vector_db_select('geology')['values'][3]
  169. ['3', '579286.875', '3335.55835', '4', '3', 'Zml', '579286.829631', '3335.557182']
  170. >>> print vector_db_select('geology', columns = 'GEO_NAME')['values'][3]
  171. ['Zml']
  172. :param str map: map name
  173. :param int layer: layer number
  174. :param kwargs: v.db.select options
  175. :param env: environment
  176. :return: dictionary ('columns' and 'values')
  177. """
  178. try:
  179. key = vector_db(map=map, env=env)[layer]['key']
  180. except KeyError:
  181. error(_('Missing layer %(layer)d in vector map <%(map)s>') %
  182. {'layer': layer, 'map': map})
  183. return {'columns': [], 'values': {}}
  184. include_key = True
  185. if 'columns' in kwargs:
  186. if key not in kwargs['columns'].split(','):
  187. # add key column if missing
  188. include_key = False
  189. debug("Adding key column to the output")
  190. kwargs['columns'] += ',' + key
  191. ret = read_command('v.db.select', map=map, layer=layer,
  192. env=env, **kwargs)
  193. if not ret:
  194. error(_('vector_db_select() failed'))
  195. return {'columns': [], 'values': {}}
  196. columns = []
  197. values = {}
  198. for line in ret.splitlines():
  199. if not columns:
  200. columns = line.split('|')
  201. key_index = columns.index(key)
  202. # discard key column
  203. if not include_key:
  204. columns = columns[:-1]
  205. continue
  206. value = line.split('|')
  207. key_value = int(value[key_index])
  208. if not include_key:
  209. # discard key column
  210. values[key_value] = value[:-1]
  211. else:
  212. values[key_value] = value
  213. return {'columns': columns, 'values': values}
  214. json = None
  215. orderedDict = None
  216. def vector_what(map, coord, distance=0.0, ttype=None,
  217. encoding=None, skip_attributes=False,
  218. layer=None, multiple=False, env=None):
  219. """Query vector map at given locations
  220. To query one vector map at one location
  221. ::
  222. print grass.vector_what(map='archsites', coord=(595743, 4925281),
  223. distance=250)
  224. [{'Category': 8, 'Map': 'archsites', 'Layer': 1, 'Key_column': 'cat',
  225. 'Database': '/home/martin/grassdata/spearfish60/PERMANENT/dbf/',
  226. 'Mapset': 'PERMANENT', 'Driver': 'dbf',
  227. 'Attributes': {'str1': 'No_Name', 'cat': '8'},
  228. 'Table': 'archsites', 'Type': 'Point', 'Id': 8}]
  229. To query one vector map with multiple layers (no additional parameters
  230. required)
  231. ::
  232. for q in grass.vector_what(map='some_map', distance=100.0,
  233. coord=(596532.357143,4920486.21429)):
  234. print q['Map'], q['Layer'], q['Attributes']
  235. new_bug_sites 1 {'str1': 'Beetle_site', 'GRASSRGB': '', 'cat': '80'}
  236. new_bug_sites 2 {'cat': '80'}
  237. To query more vector maps at one location
  238. ::
  239. for q in grass.vector_what(map=('archsites', 'roads'),
  240. coord=(595743, 4925281), distance=250):
  241. print q['Map'], q['Attributes']
  242. archsites {'str1': 'No_Name', 'cat': '8'}
  243. roads {'label': 'interstate', 'cat': '1'}
  244. To query one vector map at more locations
  245. ::
  246. for q in grass.vector_what(map='archsites', distance=250,
  247. coord=[(595743, 4925281), (597950, 4918898)]):
  248. print q['Map'], q['Attributes']
  249. archsites {'str1': 'No_Name', 'cat': '8'}
  250. archsites {'str1': 'Bob_Miller', 'cat': '22'}
  251. :param map: vector map(s) to query given as string or list/tuple
  252. :param coord: coordinates of query given as tuple (easting, northing) or
  253. list of tuples
  254. :param distance: query threshold distance (in map units)
  255. :param ttype: list of topology types (default of v.what are point, line,
  256. area, face)
  257. :param encoding: attributes encoding
  258. :param skip_attributes: True to skip quering attributes
  259. :param layer: layer number or list of layers (one for each vector),
  260. if None, all layers (-1) are used
  261. :param multiple: find multiple features within threshold distance
  262. :param env: environment
  263. :return: parsed list
  264. """
  265. if not env:
  266. env = os.environ.copy()
  267. if "LC_ALL" in env:
  268. env["LC_ALL"] = "C"
  269. if isinstance(map, (bytes, unicode)):
  270. map_list = [map]
  271. else:
  272. map_list = map
  273. if layer:
  274. if isinstance(layer, (tuple, list)):
  275. layer_list = [str(l) for l in layer]
  276. else:
  277. layer_list = [str(layer)]
  278. if len(layer_list) != len(map_list):
  279. raise ScriptError(_("Number of given vector maps ({m}) "
  280. "differs from number of layers ({l})").format(m=len(map_list),
  281. l=len(layer_list)))
  282. else:
  283. layer_list = ['-1'] * len(map_list)
  284. coord_list = list()
  285. if isinstance(coord, tuple):
  286. coord_list.append('%f,%f' % (coord[0], coord[1]))
  287. else:
  288. for e, n in coord:
  289. coord_list.append('%f,%f' % (e, n))
  290. flags = 'j'
  291. if not skip_attributes:
  292. flags += 'a'
  293. if multiple:
  294. flags += 'm'
  295. cmdParams = dict(quiet=True,
  296. flags=flags,
  297. map=','.join(map_list),
  298. layer=','.join(layer_list),
  299. coordinates=','.join(coord_list),
  300. distance=float(distance))
  301. if ttype:
  302. cmdParams['type'] = ','.join(ttype)
  303. try:
  304. ret = read_command('v.what', env=env,
  305. **cmdParams).strip()
  306. except CalledModuleError as e:
  307. raise ScriptError(e.msg)
  308. data = list()
  309. if not ret:
  310. return data
  311. # lazy import
  312. global json
  313. global orderedDict
  314. if json is None:
  315. import json
  316. if orderedDict is None:
  317. try:
  318. from collections import OrderedDict
  319. orderedDict = OrderedDict
  320. except ImportError:
  321. orderedDict = dict
  322. kwargs = {}
  323. if encoding:
  324. kwargs['encoding'] = encoding
  325. if sys.version_info[0:2] > (2, 6):
  326. kwargs['object_pairs_hook'] = orderedDict
  327. try:
  328. result = json.loads(ret, **kwargs)
  329. except ValueError:
  330. raise ScriptError(_("v.what output is not valid JSON format:\n {ret}").format(ret=ret))
  331. if multiple:
  332. for vmap in result['Maps']:
  333. features = vmap.pop('Features', None)
  334. if features:
  335. for feature in features:
  336. cats = feature.pop('Categories', None)
  337. if cats:
  338. for cat in cats:
  339. tmp = feature.copy()
  340. tmp.update(cat)
  341. tmp2 = vmap.copy()
  342. tmp2.update(tmp)
  343. data.append(tmp2)
  344. else:
  345. for vmap in result['Maps']:
  346. cats = vmap.pop('Categories', None)
  347. if cats:
  348. for cat in cats:
  349. tmp = vmap.copy()
  350. tmp.update(cat)
  351. data.append(tmp)
  352. else:
  353. data.append(vmap)
  354. return data