ogr2ogrtopostgislist.py 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461
  1. """
  2. ***************************************************************************
  3. ogr2ogrtopostgislist.py
  4. ---------------------
  5. Date : November 2012
  6. Copyright : (C) 2012 by Victor Olaya
  7. Email : volayaf at gmail dot com
  8. ***************************************************************************
  9. * *
  10. * This program is free software; you can redistribute it and/or modify *
  11. * it under the terms of the GNU General Public License as published by *
  12. * the Free Software Foundation; either version 2 of the License, or *
  13. * (at your option) any later version. *
  14. * *
  15. ***************************************************************************
  16. """
  17. __author__ = 'wanger'
  18. __date__ = 'November 2024'
  19. __copyright__ = '(C) 2024, wanger'
  20. import os
  21. from PyQt5.QtGui import QIcon
  22. from PyQt5.QtWidgets import QMessageBox
  23. from qgis._core import QgsProcessingParameterDefinition
  24. from qgis.core import (QgsProcessing,
  25. QgsProcessingParameterFeatureSource,
  26. QgsProcessingParameterString,
  27. QgsProcessingParameterDateTime,
  28. QgsProcessingParameterEnum,
  29. QgsProcessingParameterCrs,
  30. QgsProcessingParameterField,
  31. QgsProcessingParameterExtent,
  32. QgsProcessingParameterBoolean,
  33. QgsProcessingParameterProviderConnection,
  34. QgsProcessingParameterDatabaseSchema,
  35. QgsProcessingParameterDatabaseTable,
  36. QgsProviderRegistry,
  37. QgsProcessingException,
  38. QgsCoordinateReferenceSystem,
  39. QgsProviderConnectionException,
  40. QgsDataSourceUri)
  41. from processing.algs.gdal.GdalAlgorithm import GdalAlgorithm
  42. from processing.algs.gdal.GdalUtils import GdalUtils
  43. from processing.tools.PrintUtils import printStr
  44. from processing.tools.PostgreSQL.PostgreSQL import PostgreSQL
  45. from processing.tools.system import isWindows
  46. pluginPath = os.path.normpath(os.path.join(
  47. os.path.split(os.path.dirname(__file__))[0], os.pardir))
  48. class Ogr2OgrToPostGisList(GdalAlgorithm):
  49. DATABASE = 'DATABASE'
  50. INPUT = 'INPUT'
  51. SHAPE_ENCODING = 'SHAPE_ENCODING'
  52. GTYPE = 'GTYPE'
  53. GEOMTYPE = ['', 'NONE', 'GEOMETRY', 'POINT', 'LINESTRING', 'POLYGON', 'GEOMETRYCOLLECTION', 'MULTIPOINT',
  54. 'MULTIPOLYGON', 'MULTILINESTRING', 'CIRCULARSTRING', 'COMPOUNDCURVE', 'CURVEPOLYGON', 'MULTICURVE',
  55. 'MULTISURFACE']
  56. S_SRS = 'S_SRS'
  57. T_SRS = 'T_SRS'
  58. A_SRS = 'A_SRS'
  59. HOST = 'HOST'
  60. PORT = 'PORT'
  61. USER = 'USER'
  62. DBNAME = 'DBNAME'
  63. PASSWORD = 'PASSWORD'
  64. SCHEMA = 'SCHEMA'
  65. TABLE = 'TABLE'
  66. TABLE_ALIAS = 'TABLE_ALIAS'
  67. PK = 'PK'
  68. PRIMARY_KEY = 'PRIMARY_KEY'
  69. GEOCOLUMN = 'GEOCOLUMN'
  70. DIM = 'DIM'
  71. DIMLIST = ['2', '3', '4']
  72. SIMPLIFY = 'SIMPLIFY'
  73. SEGMENTIZE = 'SEGMENTIZE'
  74. SPAT = 'SPAT'
  75. CLIP = 'CLIP'
  76. WHERE = 'WHERE'
  77. GT = 'GT'
  78. OVERWRITE = 'OVERWRITE'
  79. APPEND = 'APPEND'
  80. ADDFIELDS = 'ADDFIELDS'
  81. LAUNDER = 'LAUNDER'
  82. INDEX = 'INDEX'
  83. SKIPFAILURES = 'SKIPFAILURES'
  84. PRECISION = 'PRECISION'
  85. PROMOTETOMULTI = 'PROMOTETOMULTI'
  86. OPTIONS = 'OPTIONS'
  87. # TODO 三亚项目新追加入库属性
  88. VECTOR_YEAR = 'VECTOR_YEAR'
  89. VECTOR_XZQH = 'VECTOR_XZQH'
  90. VECTOR_XMLX = 'VECTOR_XMLX'
  91. VECTOR_SJLY = 'VECTOR_SJLY'
  92. VECTOR_YWLX = 'VECTOR_YWLX'
  93. SOURCE_TYPE = 'SOURCE_TYPE'
  94. XZQH_FIELD = 'XZQH_FIELD'
  95. VECTOR_GLBM = 'VECTOR_GLBM'
  96. VECTOR_ZYML = 'VECTOR_ZYML'
  97. Metadata_storage = 'Metadata_storage'
  98. XLK = 'XLK'
  99. XLKXZX = ['选择一', '选择二', '选择三', '选择四', '选择五']
  100. selectedValue = "selectedValue"
  101. def __init__(self):
  102. super().__init__()
  103. def initAlgorithm(self, config=None):
  104. self.addParameter(QgsProcessingParameterFeatureSource(self.INPUT,
  105. self.tr('Input layer'),
  106. types=[QgsProcessing.TypeVector]))
  107. db_param = QgsProcessingParameterProviderConnection(
  108. self.DATABASE,
  109. self.tr('数据库'), provider='postgres',
  110. defaultValue=self.pgcoon["host"])
  111. self.addParameter(db_param)
  112. schema_param = QgsProcessingParameterDatabaseSchema(
  113. self.SCHEMA,
  114. self.tr('模式'), defaultValue=self.pgcoon["schema"], connectionParameterName=self.DATABASE,
  115. optional=False)
  116. self.addParameter(schema_param)
  117. table_param = QgsProcessingParameterString(
  118. self.TABLE,
  119. self.tr('表名'), defaultValue="", optional=True)
  120. self.addParameter(table_param)
  121. tablealias_param = QgsProcessingParameterString(
  122. self.TABLE_ALIAS,
  123. self.tr('别名'), defaultValue="", optional=True)
  124. self.addParameter(tablealias_param)
  125. # 行政区划字段选择
  126. self.addParameter(QgsProcessingParameterField(self.XZQH_FIELD,
  127. self.tr('行政区划字段'),
  128. None,
  129. self.INPUT,
  130. QgsProcessingParameterField.Any,
  131. optional=True))
  132. # 数据来源
  133. self.addParameter(QgsProcessingParameterString(self.VECTOR_SJLY,
  134. self.tr('数据来源'), "",
  135. optional=False))
  136. self.addParameter(
  137. QgsProcessingParameterDateTime(self.VECTOR_YEAR, '数据时效', type=QgsProcessingParameterDateTime.Type.Date,
  138. defaultValue=None))
  139. pgconn = PostgreSQL(schema='base')
  140. rows = pgconn.getVectorYwlx()
  141. self.ywlxs = []
  142. for row in rows:
  143. self.ywlxs.append(row[0])
  144. self.addParameter(QgsProcessingParameterEnum(name=self.VECTOR_YWLX,
  145. description=self.tr('业务类型'), options=self.ywlxs))
  146. rows = pgconn.getDeptList()
  147. self.depts = []
  148. for row in rows:
  149. self.depts.append(row[0])
  150. self.addParameter(QgsProcessingParameterEnum(name=self.VECTOR_GLBM,
  151. description=self.tr('管理部门'), options=self.depts))
  152. rows = pgconn.getVectorZyml()
  153. self.zymls = []
  154. for row in rows:
  155. self.zymls.append(row[1])
  156. self.addParameter(QgsProcessingParameterEnum(name=self.VECTOR_ZYML,
  157. description=self.tr('资源目录'), options=self.zymls))
  158. # self.addParameter(QgsProcessingParameterEnum(self.XLK,
  159. # self.tr('下拉框'), options=self.XLKXZX,
  160. # defaultValue=0))
  161. # 数据年份
  162. # self.addParameter(QgsProcessingParameterString(self.VECTOR_YEAR,
  163. # self.tr('数据年份'), "",
  164. # optional=False))
  165. # 行政区划
  166. # self.addParameter(QgsProcessingParameterString(self.VECTOR_XZQH,
  167. # self.tr('行政区划'), "",
  168. # optional=False))
  169. # 项目类型
  170. # self.addParameter(QgsProcessingParameterString(self.VECTOR_XMLX,
  171. # self.tr('项目类型'), "",
  172. # optional=False))
  173. crs = QgsCoordinateReferenceSystem("EPSG:4525")
  174. crs_parameter = QgsProcessingParameterCrs(self.T_SRS,
  175. self.tr('指定入库坐标系'),
  176. defaultValue=crs,
  177. optional=False)
  178. self.addParameter(crs_parameter)
  179. string = QgsProcessingParameterString(self.SOURCE_TYPE,
  180. self.tr('数据源类型'), "vector",
  181. optional=False)
  182. self.addParameter(string)
  183. # self.addParameter(QgsProcessingParameterString(self.SHAPE_ENCODING,
  184. # self.tr('Shape encoding'), "", optional=True))
  185. #
  186. # self.addParameter(QgsProcessingParameterCrs(self.A_SRS,
  187. # self.tr('Assign an output CRS'), defaultValue='', optional=True))
  188. # self.addParameter(QgsProcessingParameterCrs(self.S_SRS,
  189. # self.tr('Override source CRS'), defaultValue='', optional=True))
  190. #
  191. # self.addParameter(QgsProcessingParameterEnum(self.GTYPE,
  192. # self.tr('Output geometry type'), options=self.GEOMTYPE,
  193. # defaultValue=0))
  194. #
  195. self.addParameter(QgsProcessingParameterString(self.PK,
  196. self.tr('Primary key (new field)'), defaultValue='id',
  197. optional=True))
  198. # self.addParameter(QgsProcessingParameterField(self.PRIMARY_KEY,
  199. # self.tr(
  200. # 'Primary key (existing field, used if the above option is left empty)'),
  201. # parentLayerParameterName=self.INPUT, optional=True))
  202. self.addParameter(QgsProcessingParameterString(self.GEOCOLUMN,
  203. self.tr('Geometry column name'), defaultValue='geom',
  204. optional=True))
  205. # self.addParameter(QgsProcessingParameterEnum(self.DIM,
  206. # self.tr('Vector dimensions'), options=self.DIMLIST,
  207. # defaultValue=0))
  208. # self.addParameter(QgsProcessingParameterString(self.SIMPLIFY,
  209. # self.tr('Distance tolerance for simplification'),
  210. # defaultValue='', optional=True))
  211. # self.addParameter(QgsProcessingParameterString(self.SEGMENTIZE,
  212. # self.tr('Maximum distance between 2 nodes (densification)'),
  213. # defaultValue='', optional=True))
  214. # self.addParameter(QgsProcessingParameterExtent(self.SPAT,
  215. # self.tr(
  216. # 'Select features by extent (defined in input layer CRS)'),
  217. # optional=True))
  218. # self.addParameter(QgsProcessingParameterBoolean(self.CLIP,
  219. # self.tr(
  220. # 'Clip the input layer using the above (rectangle) extent'),
  221. # defaultValue=False))
  222. # self.addParameter(QgsProcessingParameterString(self.WHERE,
  223. # self.tr(
  224. # 'Select features using a SQL "WHERE" statement (Ex: column=\'value\')'),
  225. # defaultValue='', optional=True))
  226. # self.addParameter(QgsProcessingParameterString(self.GT,
  227. # self.tr('Group N features per transaction (Default: 20000)'),
  228. # defaultValue='', optional=True))
  229. metadata = QgsProcessingParameterBoolean(self.Metadata_storage,
  230. self.tr('元数据入库'), defaultValue=True)
  231. # metadata.setEnable(False)
  232. self.addParameter(metadata)
  233. self.addParameter(QgsProcessingParameterBoolean(self.OVERWRITE,
  234. self.tr('Overwrite existing table'), defaultValue=True))
  235. self.addParameter(QgsProcessingParameterBoolean(self.APPEND,
  236. self.tr('Append to existing table'), defaultValue=False))
  237. self.addParameter(QgsProcessingParameterBoolean(self.ADDFIELDS,
  238. self.tr('Append and add new fields to existing table'),
  239. defaultValue=False))
  240. self.addParameter(QgsProcessingParameterBoolean(self.LAUNDER,
  241. self.tr('Do not launder columns/table names'),
  242. defaultValue=False))
  243. self.addParameter(QgsProcessingParameterBoolean(self.INDEX,
  244. self.tr('Do not create spatial index'), defaultValue=False))
  245. self.addParameter(QgsProcessingParameterBoolean(self.SKIPFAILURES,
  246. self.tr(
  247. 'Continue after a failure, skipping the failed feature'),
  248. defaultValue=False))
  249. self.addParameter(QgsProcessingParameterBoolean(self.PROMOTETOMULTI,
  250. self.tr('Promote to Multipart'),
  251. defaultValue=True))
  252. self.addParameter(QgsProcessingParameterBoolean(self.PRECISION,
  253. self.tr('Keep width and precision of input attributes'),
  254. defaultValue=False))
  255. self.addParameter(QgsProcessingParameterString(self.OPTIONS,
  256. self.tr('Additional creation options'), defaultValue='',
  257. optional=True))
  258. pgconn.close()
  259. def name(self):
  260. return 'importvectorintopostgisdatabaseavailableconnections'
  261. def icon(self):
  262. return QIcon(os.path.join(pluginPath, 'images', 'dbms', 'importvector.png'))
  263. def displayName(self):
  264. return self.tr('矢量数据入库')
  265. def shortDescription(self):
  266. return self.tr('Exports a vector layer to an existing PostgreSQL database connection')
  267. def tags(self):
  268. t = self.tr('import,into,postgis,database,vector').split(',')
  269. t.extend(super().tags())
  270. return t
  271. def group(self):
  272. return self.tr('数据入库工具')
  273. def groupId(self):
  274. return 'vector2postgis'
  275. def setSelectedValue(self, v):
  276. printStr(v)
  277. self.selectedValue = v
  278. def getSelectedValue(self):
  279. return self.selectedValue
  280. def getConsoleCommands(self, parameters, context, feedback, executing=True):
  281. print(parameters)
  282. print(self.ywlxs)
  283. connection_name = self.parameterAsConnectionName(parameters, self.DATABASE, context)
  284. if not connection_name:
  285. raise QgsProcessingException(
  286. self.tr('No connection specified'))
  287. # resolve connection details to uri
  288. try:
  289. md = QgsProviderRegistry.instance().providerMetadata('postgres')
  290. conn = md.createConnection(connection_name)
  291. except QgsProviderConnectionException:
  292. raise QgsProcessingException(
  293. self.tr('Could not retrieve connection details for {}').format(connection_name))
  294. uri = conn.uri()
  295. ogrLayer, layername = self.getOgrCompatibleSource(self.INPUT, parameters, context, feedback, executing)
  296. print(ogrLayer)
  297. print(layername)
  298. shapeEncoding = self.parameterAsString(parameters, self.SHAPE_ENCODING, context)
  299. ssrs = self.parameterAsCrs(parameters, self.S_SRS, context)
  300. tsrs = self.parameterAsCrs(parameters, self.T_SRS, context)
  301. asrs = self.parameterAsCrs(parameters, self.A_SRS, context)
  302. table = self.parameterAsDatabaseTableName(parameters, self.TABLE, context)
  303. schema = self.parameterAsSchema(parameters, self.SCHEMA, context)
  304. pk = self.parameterAsString(parameters, self.PK, context)
  305. pkstring = "-lco FID=" + pk
  306. primary_key = self.parameterAsString(parameters, self.PRIMARY_KEY, context)
  307. geocolumn = self.parameterAsString(parameters, self.GEOCOLUMN, context)
  308. geocolumnstring = "-lco GEOMETRY_NAME=" + geocolumn
  309. dim = self.DIMLIST[self.parameterAsEnum(parameters, self.DIM, context)]
  310. dimstring = "-lco DIM=" + dim
  311. simplify = self.parameterAsString(parameters, self.SIMPLIFY, context)
  312. segmentize = self.parameterAsString(parameters, self.SEGMENTIZE, context)
  313. spat = self.parameterAsExtent(parameters, self.SPAT, context)
  314. clip = self.parameterAsBoolean(parameters, self.CLIP, context)
  315. where = self.parameterAsString(parameters, self.WHERE, context)
  316. wherestring = '-where "' + where + '"'
  317. gt = self.parameterAsString(parameters, self.GT, context)
  318. overwrite = self.parameterAsBoolean(parameters, self.OVERWRITE, context)
  319. append = self.parameterAsBoolean(parameters, self.APPEND, context)
  320. addfields = self.parameterAsBoolean(parameters, self.ADDFIELDS, context)
  321. launder = self.parameterAsBoolean(parameters, self.LAUNDER, context)
  322. launderstring = "-lco LAUNDER=NO"
  323. index = self.parameterAsBoolean(parameters, self.INDEX, context)
  324. indexstring = "-lco SPATIAL_INDEX=OFF"
  325. skipfailures = self.parameterAsBoolean(parameters, self.SKIPFAILURES, context)
  326. promotetomulti = self.parameterAsBoolean(parameters, self.PROMOTETOMULTI, context)
  327. precision = self.parameterAsBoolean(parameters, self.PRECISION, context)
  328. options = self.parameterAsString(parameters, self.OPTIONS, context)
  329. table_alias = self.parameterAsString(parameters, self.TABLE_ALIAS, context)
  330. arguments = [
  331. '-progress',
  332. '--config PG_USE_COPY YES'
  333. ]
  334. if shapeEncoding:
  335. arguments.append('--config')
  336. arguments.append('SHAPE_ENCODING')
  337. arguments.append(shapeEncoding)
  338. arguments.append('-f')
  339. arguments.append('PostgreSQL')
  340. connection_parts = QgsDataSourceUri(uri).connectionInfo(executing).split(' ')
  341. connection_parts.append('active_schema={}'.format(schema or 'public'))
  342. arguments.append('PG:{}'.format(' '.join(connection_parts)))
  343. # 检查数据库中该模式下是否存在相同的表
  344. pgconn = PostgreSQL(schema='base')
  345. resbool = pgconn.checkTableName(schemaname=schema,
  346. tablename=layername.lower() if len(table) == 0 else table.lower())
  347. pgconn.close()
  348. if resbool == False:
  349. QMessageBox.warning(None, "提示",
  350. self.tr('数据库中已存在相同表名,请输入或修改表名!'))
  351. arguments.append(dimstring)
  352. arguments.append(ogrLayer)
  353. arguments.append(layername)
  354. if index:
  355. arguments.append(indexstring)
  356. if launder:
  357. arguments.append(launderstring)
  358. if append:
  359. arguments.append('-append')
  360. if addfields:
  361. arguments.append('-addfields')
  362. if overwrite:
  363. arguments.append('-overwrite')
  364. if len(self.GEOMTYPE[self.parameterAsEnum(parameters, self.GTYPE, context)]) > 0:
  365. arguments.append('-nlt')
  366. arguments.append(self.GEOMTYPE[self.parameterAsEnum(parameters, self.GTYPE, context)])
  367. if len(geocolumn) > 0:
  368. arguments.append(geocolumnstring)
  369. if pk:
  370. arguments.append(pkstring)
  371. elif primary_key:
  372. arguments.append("-lco FID=" + primary_key)
  373. if len(table) == 0:
  374. # table = layername.lower()
  375. table = layername
  376. if schema:
  377. table = f'{schema}.{table}'
  378. # if len(table_alias) > 0:
  379. # arguments.append("-lco layer_name=" + table_alias)
  380. arguments.append('-nln')
  381. arguments.append(table)
  382. if ssrs.isValid():
  383. arguments.append('-s_srs')
  384. arguments.append(GdalUtils.gdal_crs_string(ssrs))
  385. if tsrs.isValid():
  386. arguments.append('-t_srs')
  387. arguments.append(GdalUtils.gdal_crs_string(tsrs))
  388. if asrs.isValid():
  389. arguments.append('-a_srs')
  390. arguments.append(GdalUtils.gdal_crs_string(asrs))
  391. if not spat.isNull():
  392. arguments.append('-spat')
  393. arguments.append(spat.xMinimum())
  394. arguments.append(spat.yMinimum())
  395. arguments.append(spat.xMaximum())
  396. arguments.append(spat.yMaximum())
  397. if clip:
  398. arguments.append('-clipsrc spat_extent')
  399. if skipfailures:
  400. arguments.append('-skipfailures')
  401. if where:
  402. arguments.append(wherestring)
  403. if len(simplify) > 0:
  404. arguments.append('-simplify')
  405. arguments.append(simplify)
  406. if len(segmentize) > 0:
  407. arguments.append('-segmentize')
  408. arguments.append(segmentize)
  409. if len(gt) > 0:
  410. arguments.append('-gt')
  411. arguments.append(gt)
  412. if promotetomulti:
  413. arguments.append('-nlt PROMOTE_TO_MULTI')
  414. if precision is False:
  415. arguments.append('-lco PRECISION=NO')
  416. if len(options) > 0:
  417. arguments.append(options)
  418. if isWindows():
  419. return ['cmd.exe', '/C ', 'ogr2ogr.exe',
  420. GdalUtils.escapeAndJoin(arguments)]
  421. else:
  422. return ['ogr2ogr', GdalUtils.escapeAndJoin(arguments)]
  423. def getTableName(self, table):
  424. sp = table.split(".")
  425. res = f'{sp[0]}."{sp[1]}"'
  426. print(res)
  427. return res
  428. def commandName(self):
  429. return "ogr2ogr"