mirror of
https://github.com/qgis/QGIS.git
synced 2025-10-24 00:04:44 -04:00
This reverts commit 26e9ec98e76591286576d2dde098578877bef9c0. It results in unfixable crashes on many platforms, likely due to some issue in sip itself Fixes #38393, reopens #38245 The original bug (being asked twice for credentials) is preferable over a hard crash
471 lines
17 KiB
Python
471 lines
17 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
"""
|
|
/***************************************************************************
|
|
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. *
|
|
* *
|
|
***************************************************************************/
|
|
"""
|
|
from builtins import str
|
|
from builtins import map
|
|
from builtins import range
|
|
|
|
# this will disable the dbplugin if the connector raise an ImportError
|
|
from .connector import PostGisDBConnector
|
|
|
|
from qgis.PyQt.QtCore import Qt, QRegExp, 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(u"/%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.value("sslmode", QgsDataSourceUri.SslPrefer, type=int)
|
|
except TypeError:
|
|
sslmode = QgsDataSourceUri.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.Info,
|
|
parent.iface.messageTimeout())
|
|
return
|
|
finally:
|
|
QApplication.setOverrideCursor(Qt.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.Info,
|
|
parent.iface.messageTimeout())
|
|
return
|
|
finally:
|
|
QApplication.setOverrideCursor(Qt.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 set(['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(u"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.Yes | QMessageBox.No) == QMessageBox.No:
|
|
return False
|
|
finally:
|
|
QApplication.setOverrideCursor(Qt.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)
|
|
|
|
|
|
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 = (u'service=\'%s\'' % uri.service()) if uri.service() else ''
|
|
dbname = (u'dbname=\'%s\'' % uri.database()) if uri.database() else ''
|
|
host = (u'host=%s' % uri.host()) if uri.host() else ''
|
|
user = (u'user=%s' % uri.username()) if uri.username() else ''
|
|
passw = (u'password=%s' % uri.password()) if uri.password() else ''
|
|
port = (u'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 = (u'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 = u'column=\'%s\'' % fld.name
|
|
break
|
|
|
|
uri = u'%s %s %s %s %s %s %s table=%s' % \
|
|
(service, dbname, host, user, passw, port, col, table)
|
|
|
|
return uri
|
|
|
|
def mimeUri(self):
|
|
uri = u"raster:postgresraster:{}:{}".format(self.name, re.sub(":", r"\:", self.uri()))
|
|
return uri
|
|
|
|
def toMapLayer(self):
|
|
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.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 = QRegExp("\\((.+)\\)$")
|
|
startpos = regex.indexIn(trimmedTypeStr)
|
|
if startpos >= 0:
|
|
self.modifier = regex.cap(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
|