vector.py 13 KB

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