mirror of
				https://github.com/qgis/QGIS.git
				synced 2025-10-26 00:04:03 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			488 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			488 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """
 | |
| /***************************************************************************
 | |
| Name                 : DB Manager
 | |
| Description          : Database manager plugin for QGIS
 | |
| Date                 : May 23, 2011
 | |
| copyright            : (C) 2011 by Giuseppe Sucameli
 | |
| email                : brush.tyler@gmail.com
 | |
| 
 | |
|  ***************************************************************************/
 | |
| 
 | |
| /***************************************************************************
 | |
|  *                                                                         *
 | |
|  *   This program is free software; you can redistribute it and/or modify  *
 | |
|  *   it under the terms of the GNU General Public License as published by  *
 | |
|  *   the Free Software Foundation; either version 2 of the License, or     *
 | |
|  *   (at your option) any later version.                                   *
 | |
|  *                                                                         *
 | |
|  ***************************************************************************/
 | |
| """
 | |
| 
 | |
| # this will disable the dbplugin if the connector raise an ImportError
 | |
| from .connector import PostGisDBConnector
 | |
| 
 | |
| from qgis.PyQt.QtCore import (
 | |
|     Qt,
 | |
|     QRegularExpression,
 | |
|     QCoreApplication
 | |
| )
 | |
| from qgis.PyQt.QtGui import QIcon
 | |
| from qgis.PyQt.QtWidgets import QAction, QApplication, QMessageBox
 | |
| from qgis.core import Qgis, QgsApplication, QgsSettings
 | |
| from qgis.gui import QgsMessageBar
 | |
| 
 | |
| from ..plugin import ConnectionError, InvalidDataException, DBPlugin, Database, Schema, Table, VectorTable, RasterTable, \
 | |
|     TableField, TableConstraint, TableIndex, TableTrigger, TableRule
 | |
| 
 | |
| import re
 | |
| 
 | |
| 
 | |
| def classFactory():
 | |
|     return PostGisDBPlugin
 | |
| 
 | |
| 
 | |
| class PostGisDBPlugin(DBPlugin):
 | |
| 
 | |
|     @classmethod
 | |
|     def icon(self):
 | |
|         return QgsApplication.getThemeIcon("/mIconPostgis.svg")
 | |
| 
 | |
|     @classmethod
 | |
|     def typeName(self):
 | |
|         return 'postgis'
 | |
| 
 | |
|     @classmethod
 | |
|     def typeNameString(self):
 | |
|         return QCoreApplication.translate('db_manager', 'PostGIS')
 | |
| 
 | |
|     @classmethod
 | |
|     def providerName(self):
 | |
|         return 'postgres'
 | |
| 
 | |
|     @classmethod
 | |
|     def connectionSettingsKey(self):
 | |
|         return '/PostgreSQL/connections'
 | |
| 
 | |
|     def databasesFactory(self, connection, uri):
 | |
|         return PGDatabase(connection, uri)
 | |
| 
 | |
|     def connect(self, parent=None):
 | |
|         conn_name = self.connectionName()
 | |
|         settings = QgsSettings()
 | |
|         settings.beginGroup("/%s/%s" % (self.connectionSettingsKey(), conn_name))
 | |
| 
 | |
|         if not settings.contains("database"):  # non-existent entry?
 | |
|             raise InvalidDataException(self.tr('There is no defined database connection "{0}".').format(conn_name))
 | |
| 
 | |
|         from qgis.core import QgsDataSourceUri
 | |
| 
 | |
|         uri = QgsDataSourceUri()
 | |
| 
 | |
|         settingsList = ["service", "host", "port", "database", "username", "password", "authcfg"]
 | |
|         service, host, port, database, username, password, authcfg = (settings.value(x, "", type=str) for x in settingsList)
 | |
| 
 | |
|         useEstimatedMetadata = settings.value("estimatedMetadata", False, type=bool)
 | |
|         try:
 | |
|             sslmode = settings.enumValue("sslmode", QgsDataSourceUri.SslMode.SslPrefer)
 | |
|         except TypeError:
 | |
|             sslmode = QgsDataSourceUri.SslMode.SslPrefer
 | |
| 
 | |
|         settings.endGroup()
 | |
| 
 | |
|         if hasattr(authcfg, 'isNull') and authcfg.isNull():
 | |
|             authcfg = ''
 | |
| 
 | |
|         if service:
 | |
|             uri.setConnection(service, database, username, password, sslmode, authcfg)
 | |
|         else:
 | |
|             uri.setConnection(host, port, database, username, password, sslmode, authcfg)
 | |
| 
 | |
|         uri.setUseEstimatedMetadata(useEstimatedMetadata)
 | |
| 
 | |
|         try:
 | |
|             return self.connectToUri(uri)
 | |
|         except ConnectionError:
 | |
|             return False
 | |
| 
 | |
| 
 | |
| class PGDatabase(Database):
 | |
| 
 | |
|     def __init__(self, connection, uri):
 | |
|         Database.__init__(self, connection, uri)
 | |
| 
 | |
|     def connectorsFactory(self, uri):
 | |
|         return PostGisDBConnector(uri, self.connection())
 | |
| 
 | |
|     def dataTablesFactory(self, row, db, schema=None):
 | |
|         return PGTable(row, db, schema)
 | |
| 
 | |
|     def info(self):
 | |
|         from .info_model import PGDatabaseInfo
 | |
|         return PGDatabaseInfo(self)
 | |
| 
 | |
|     def vectorTablesFactory(self, row, db, schema=None):
 | |
|         return PGVectorTable(row, db, schema)
 | |
| 
 | |
|     def rasterTablesFactory(self, row, db, schema=None):
 | |
|         return PGRasterTable(row, db, schema)
 | |
| 
 | |
|     def schemasFactory(self, row, db):
 | |
|         return PGSchema(row, db)
 | |
| 
 | |
|     def sqlResultModel(self, sql, parent):
 | |
|         from .data_model import PGSqlResultModel
 | |
| 
 | |
|         return PGSqlResultModel(self, sql, parent)
 | |
| 
 | |
|     def sqlResultModelAsync(self, sql, parent):
 | |
|         from .data_model import PGSqlResultModelAsync
 | |
| 
 | |
|         return PGSqlResultModelAsync(self, sql, parent)
 | |
| 
 | |
|     def registerDatabaseActions(self, mainWindow):
 | |
|         Database.registerDatabaseActions(self, mainWindow)
 | |
| 
 | |
|         # add a separator
 | |
|         separator = QAction(self)
 | |
|         separator.setSeparator(True)
 | |
|         mainWindow.registerAction(separator, self.tr("&Table"))
 | |
| 
 | |
|         action = QAction(self.tr("Run &Vacuum Analyze"), self)
 | |
|         mainWindow.registerAction(action, self.tr("&Table"), self.runVacuumAnalyzeActionSlot)
 | |
| 
 | |
|         action = QAction(self.tr("Run &Refresh Materialized View"), self)
 | |
|         mainWindow.registerAction(action, self.tr("&Table"), self.runRefreshMaterializedViewSlot)
 | |
| 
 | |
|     def runVacuumAnalyzeActionSlot(self, item, action, parent):
 | |
|         QApplication.restoreOverrideCursor()
 | |
|         try:
 | |
|             if not isinstance(item, Table) or item.isView:
 | |
|                 parent.infoBar.pushMessage(self.tr("Select a table for vacuum analyze."), Qgis.MessageLevel.Info,
 | |
|                                            parent.iface.messageTimeout())
 | |
|                 return
 | |
|         finally:
 | |
|             QApplication.setOverrideCursor(Qt.CursorShape.WaitCursor)
 | |
| 
 | |
|         item.runVacuumAnalyze()
 | |
| 
 | |
|     def runRefreshMaterializedViewSlot(self, item, action, parent):
 | |
|         QApplication.restoreOverrideCursor()
 | |
|         try:
 | |
|             if not isinstance(item, PGTable) or item._relationType != 'm':
 | |
|                 parent.infoBar.pushMessage(self.tr("Select a materialized view for refresh."), Qgis.MessageLevel.Info,
 | |
|                                            parent.iface.messageTimeout())
 | |
|                 return
 | |
|         finally:
 | |
|             QApplication.setOverrideCursor(Qt.CursorShape.WaitCursor)
 | |
| 
 | |
|         item.runRefreshMaterializedView()
 | |
| 
 | |
|     def hasLowercaseFieldNamesOption(self):
 | |
|         return True
 | |
| 
 | |
|     def supportsComment(self):
 | |
|         return True
 | |
| 
 | |
|     def executeSql(self, sql):
 | |
|         return self.connector._executeSql(sql)
 | |
| 
 | |
| 
 | |
| class PGSchema(Schema):
 | |
| 
 | |
|     def __init__(self, row, db):
 | |
|         Schema.__init__(self, db)
 | |
|         self.oid, self.name, self.owner, self.perms, self.comment = row
 | |
| 
 | |
| 
 | |
| class PGTable(Table):
 | |
| 
 | |
|     def __init__(self, row, db, schema=None):
 | |
|         Table.__init__(self, db, schema)
 | |
|         self.name, schema_name, self._relationType, self.owner, self.estimatedRowCount, self.pages, self.comment = row
 | |
|         self.isView = self._relationType in {'v', 'm'}
 | |
|         self.estimatedRowCount = int(self.estimatedRowCount)
 | |
| 
 | |
|     def runVacuumAnalyze(self):
 | |
|         self.aboutToChange.emit()
 | |
|         self.database().connector.runVacuumAnalyze((self.schemaName(), self.name))
 | |
|         # TODO: change only this item, not re-create all the tables in the schema/database
 | |
|         self.schema().refresh() if self.schema() else self.database().refresh()
 | |
| 
 | |
|     def runRefreshMaterializedView(self):
 | |
|         self.aboutToChange.emit()
 | |
|         self.database().connector.runRefreshMaterializedView((self.schemaName(), self.name))
 | |
|         # TODO: change only this item, not re-create all the tables in the schema/database
 | |
|         self.schema().refresh() if self.schema() else self.database().refresh()
 | |
| 
 | |
|     def runAction(self, action):
 | |
|         action = str(action)
 | |
| 
 | |
|         if action.startswith("vacuumanalyze/"):
 | |
|             if action == "vacuumanalyze/run":
 | |
|                 self.runVacuumAnalyze()
 | |
|                 return True
 | |
| 
 | |
|         elif action.startswith("rule/"):
 | |
|             parts = action.split('/')
 | |
|             rule_name = parts[1]
 | |
|             rule_action = parts[2]
 | |
| 
 | |
|             msg = self.tr("Do you want to {0} rule {1}?").format(rule_action, rule_name)
 | |
| 
 | |
|             QApplication.restoreOverrideCursor()
 | |
| 
 | |
|             try:
 | |
|                 if QMessageBox.question(None, self.tr("Table rule"), msg,
 | |
|                                         QMessageBox.StandardButton.Yes | QMessageBox.StandardButton.No) == QMessageBox.StandardButton.No:
 | |
|                     return False
 | |
|             finally:
 | |
|                 QApplication.setOverrideCursor(Qt.CursorShape.WaitCursor)
 | |
| 
 | |
|             if rule_action == "delete":
 | |
|                 self.aboutToChange.emit()
 | |
|                 self.database().connector.deleteTableRule(rule_name, (self.schemaName(), self.name))
 | |
|                 self.refreshRules()
 | |
|                 return True
 | |
| 
 | |
|         elif action.startswith("refreshmaterializedview/"):
 | |
|             if action == "refreshmaterializedview/run":
 | |
|                 self.runRefreshMaterializedView()
 | |
|                 return True
 | |
| 
 | |
|         return Table.runAction(self, action)
 | |
| 
 | |
|     def tableFieldsFactory(self, row, table):
 | |
|         return PGTableField(row, table)
 | |
| 
 | |
|     def tableConstraintsFactory(self, row, table):
 | |
|         return PGTableConstraint(row, table)
 | |
| 
 | |
|     def tableIndexesFactory(self, row, table):
 | |
|         return PGTableIndex(row, table)
 | |
| 
 | |
|     def tableTriggersFactory(self, row, table):
 | |
|         return PGTableTrigger(row, table)
 | |
| 
 | |
|     def tableRulesFactory(self, row, table):
 | |
|         return PGTableRule(row, table)
 | |
| 
 | |
|     def info(self):
 | |
|         from .info_model import PGTableInfo
 | |
| 
 | |
|         return PGTableInfo(self)
 | |
| 
 | |
|     def crs(self):
 | |
|         return self.database().connector.getCrs(self.srid)
 | |
| 
 | |
|     def tableDataModel(self, parent):
 | |
|         from .data_model import PGTableDataModel
 | |
| 
 | |
|         return PGTableDataModel(self, parent)
 | |
| 
 | |
|     def delete(self):
 | |
|         self.aboutToChange.emit()
 | |
|         if self.isView:
 | |
|             ret = self.database().connector.deleteView((self.schemaName(), self.name), self._relationType == 'm')
 | |
|         else:
 | |
|             ret = self.database().connector.deleteTable((self.schemaName(), self.name))
 | |
|         if not ret:
 | |
|             self.deleted.emit()
 | |
|         return ret
 | |
| 
 | |
| 
 | |
| class PGVectorTable(PGTable, VectorTable):
 | |
| 
 | |
|     def __init__(self, row, db, schema=None):
 | |
|         PGTable.__init__(self, row[:-4], db, schema)
 | |
|         VectorTable.__init__(self, db, schema)
 | |
|         self.geomColumn, self.geomType, self.geomDim, self.srid = row[-4:]
 | |
| 
 | |
|     def info(self):
 | |
|         from .info_model import PGVectorTableInfo
 | |
| 
 | |
|         return PGVectorTableInfo(self)
 | |
| 
 | |
|     def runAction(self, action):
 | |
|         if PGTable.runAction(self, action):
 | |
|             return True
 | |
|         return VectorTable.runAction(self, action)
 | |
| 
 | |
|     def geometryType(self):
 | |
|         """ Returns the proper WKT type.
 | |
|         PostGIS records type like this:
 | |
|         | WKT Type     | geomType    | geomDim |
 | |
|         |--------------|-------------|---------|
 | |
|         | LineString   | LineString  | 2       |
 | |
|         | LineStringZ  | LineString  | 3       |
 | |
|         | LineStringM  | LineStringM | 3       |
 | |
|         | LineStringZM | LineString  | 4       |
 | |
|         """
 | |
|         geometryType = self.geomType
 | |
|         if self.geomDim == 3 and self.geomType[-1] != "M":
 | |
|             geometryType += "Z"
 | |
|         elif self.geomDim == 4:
 | |
|             geometryType += "ZM"
 | |
| 
 | |
|         return geometryType
 | |
| 
 | |
| 
 | |
| class PGRasterTable(PGTable, RasterTable):
 | |
| 
 | |
|     def __init__(self, row, db, schema=None):
 | |
|         PGTable.__init__(self, row[:-6], db, schema)
 | |
|         RasterTable.__init__(self, db, schema)
 | |
|         self.geomColumn, self.pixelType, self.pixelSizeX, self.pixelSizeY, self.isExternal, self.srid = row[-6:]
 | |
|         self.geomType = 'RASTER'
 | |
| 
 | |
|     def info(self):
 | |
|         from .info_model import PGRasterTableInfo
 | |
| 
 | |
|         return PGRasterTableInfo(self)
 | |
| 
 | |
|     def uri(self, uri=None):
 | |
|         """Returns the datasource URI for postgresraster provider"""
 | |
| 
 | |
|         if not uri:
 | |
|             uri = self.database().uri()
 | |
|         service = ('service=\'%s\'' % uri.service()) if uri.service() else ''
 | |
|         dbname = ('dbname=\'%s\'' % uri.database()) if uri.database() else ''
 | |
|         host = ('host=%s' % uri.host()) if uri.host() else ''
 | |
|         user = ('user=%s' % uri.username()) if uri.username() else ''
 | |
|         passw = ('password=%s' % uri.password()) if uri.password() else ''
 | |
|         port = ('port=%s' % uri.port()) if uri.port() else ''
 | |
| 
 | |
|         schema = self.schemaName() if self.schemaName() else 'public'
 | |
|         table = '"%s"."%s"' % (schema, self.name)
 | |
| 
 | |
|         if not dbname:
 | |
|             # postgresraster provider *requires* a dbname
 | |
|             connector = self.database().connector
 | |
|             r = connector._execute(None, "SELECT current_database()")
 | |
|             dbname = ('dbname=\'%s\'' % connector._fetchone(r)[0])
 | |
|             connector._close_cursor(r)
 | |
| 
 | |
|         # Find first raster field
 | |
|         col = ''
 | |
|         for fld in self.fields():
 | |
|             if fld.dataType == "raster":
 | |
|                 col = 'column=\'%s\'' % fld.name
 | |
|                 break
 | |
| 
 | |
|         uri = '%s %s %s %s %s %s %s table=%s' % \
 | |
|             (service, dbname, host, user, passw, port, col, table)
 | |
| 
 | |
|         return uri
 | |
| 
 | |
|     def mimeUri(self):
 | |
|         uri = "raster:postgresraster:{}:{}".format(self.name, re.sub(":", r"\:", self.uri()))
 | |
|         return uri
 | |
| 
 | |
|     def toMapLayer(self, geometryType=None, crs=None):
 | |
|         from qgis.core import QgsRasterLayer, QgsContrastEnhancement, QgsDataSourceUri, QgsCredentials
 | |
| 
 | |
|         rl = QgsRasterLayer(self.uri(), self.name, "postgresraster")
 | |
|         if not rl.isValid():
 | |
|             err = rl.error().summary()
 | |
|             uri = QgsDataSourceUri(self.database().uri())
 | |
|             conninfo = uri.connectionInfo(False)
 | |
|             username = uri.username()
 | |
|             password = uri.password()
 | |
| 
 | |
|             for i in range(3):
 | |
|                 (ok, username, password) = QgsCredentials.instance().get(conninfo, username, password, err)
 | |
|                 if ok:
 | |
|                     uri.setUsername(username)
 | |
|                     uri.setPassword(password)
 | |
|                     rl = QgsRasterLayer(self.uri(uri), self.name)
 | |
|                     if rl.isValid():
 | |
|                         break
 | |
| 
 | |
|         if rl.isValid():
 | |
|             rl.setContrastEnhancement(QgsContrastEnhancement.ContrastEnhancementAlgorithm.StretchToMinimumMaximum)
 | |
|         return rl
 | |
| 
 | |
| 
 | |
| class PGTableField(TableField):
 | |
| 
 | |
|     def __init__(self, row, table):
 | |
|         TableField.__init__(self, table)
 | |
|         self.num, self.name, self.dataType, self.charMaxLen, self.modifier, self.notNull, self.hasDefault, self.default, typeStr = row
 | |
|         self.primaryKey = False
 | |
| 
 | |
|         # get modifier (e.g. "precision,scale") from formatted type string
 | |
|         trimmedTypeStr = typeStr.strip()
 | |
|         regex = QRegularExpression(r"\((.+)\)$")
 | |
|         match = regex.match(trimmedTypeStr)
 | |
|         if match.hasMatch():
 | |
|             self.modifier = match.captured(1).strip()
 | |
|         else:
 | |
|             self.modifier = None
 | |
| 
 | |
|         # find out whether fields are part of primary key
 | |
|         for con in self.table().constraints():
 | |
|             if con.type == TableConstraint.TypePrimaryKey and self.num in con.columns:
 | |
|                 self.primaryKey = True
 | |
|                 break
 | |
| 
 | |
|     def getComment(self):
 | |
|         """Returns the comment for a field"""
 | |
|         tab = self.table()
 | |
|         # SQL Query checking if a comment exists for the field
 | |
|         sql_cpt = "Select count(*) from pg_description pd, pg_class pc, pg_attribute pa where relname = '%s' and attname = '%s' and pa.attrelid = pc.oid and pd.objoid = pc.oid and pd.objsubid = pa.attnum" % (tab.name, self.name)
 | |
|         # SQL Query that return the comment of the field
 | |
|         sql = "Select pd.description from pg_description pd, pg_class pc, pg_attribute pa where relname = '%s' and attname = '%s' and pa.attrelid = pc.oid and pd.objoid = pc.oid and pd.objsubid = pa.attnum" % (tab.name, self.name)
 | |
|         c = tab.database().connector._execute(None, sql_cpt)  # Execute Check query
 | |
|         res = tab.database().connector._fetchone(c)[0]  # Store result
 | |
|         if res == 1:
 | |
|             # When a comment exists
 | |
|             c = tab.database().connector._execute(None, sql)  # Execute query
 | |
|             res = tab.database().connector._fetchone(c)[0]  # Store result
 | |
|             tab.database().connector._close_cursor(c)  # Close cursor
 | |
|             return res  # Return comment
 | |
|         else:
 | |
|             return ''
 | |
| 
 | |
| 
 | |
| class PGTableConstraint(TableConstraint):
 | |
| 
 | |
|     def __init__(self, row, table):
 | |
|         TableConstraint.__init__(self, table)
 | |
|         self.name, constr_type_str, self.isDefferable, self.isDeffered, columns = row[:5]
 | |
|         self.columns = list(map(int, columns.split(' ')))
 | |
| 
 | |
|         if constr_type_str in TableConstraint.types:
 | |
|             self.type = TableConstraint.types[constr_type_str]
 | |
|         else:
 | |
|             self.type = TableConstraint.TypeUnknown
 | |
| 
 | |
|         if self.type == TableConstraint.TypeCheck:
 | |
|             self.checkSource = row[5]
 | |
|         elif self.type == TableConstraint.TypeForeignKey:
 | |
|             self.foreignTable = row[6]
 | |
|             self.foreignOnUpdate = TableConstraint.onAction[row[7]]
 | |
|             self.foreignOnDelete = TableConstraint.onAction[row[8]]
 | |
|             self.foreignMatchType = TableConstraint.matchTypes[row[9]]
 | |
|             self.foreignKeys = row[10]
 | |
| 
 | |
| 
 | |
| class PGTableIndex(TableIndex):
 | |
| 
 | |
|     def __init__(self, row, table):
 | |
|         TableIndex.__init__(self, table)
 | |
|         self.name, columns, self.isUnique = row
 | |
|         self.columns = list(map(int, columns.split(' ')))
 | |
| 
 | |
| 
 | |
| class PGTableTrigger(TableTrigger):
 | |
| 
 | |
|     def __init__(self, row, table):
 | |
|         TableTrigger.__init__(self, table)
 | |
|         self.name, self.function, self.type, self.enabled = row
 | |
| 
 | |
| 
 | |
| class PGTableRule(TableRule):
 | |
| 
 | |
|     def __init__(self, row, table):
 | |
|         TableRule.__init__(self, table)
 | |
|         self.name, self.definition = row
 |