forked from openlp/openlp
Merge branch 'sqlalchemy-update' into 'master'
Update database_exists method. maybe it will support multiple versions. See merge request openlp/openlp!311
This commit is contained in:
commit
5a55b28bb3
@ -31,7 +31,7 @@ from urllib.parse import quote_plus as urlquote
|
|||||||
from alembic.migration import MigrationContext
|
from alembic.migration import MigrationContext
|
||||||
from alembic.operations import Operations
|
from alembic.operations import Operations
|
||||||
from sqlalchemy import Column, MetaData, Table, UnicodeText, create_engine, types
|
from sqlalchemy import Column, MetaData, Table, UnicodeText, create_engine, types
|
||||||
from sqlalchemy.engine.url import make_url
|
from sqlalchemy.engine.url import make_url, URL
|
||||||
from sqlalchemy.exc import DBAPIError, InvalidRequestError, OperationalError, ProgrammingError, SQLAlchemyError
|
from sqlalchemy.exc import DBAPIError, InvalidRequestError, OperationalError, ProgrammingError, SQLAlchemyError
|
||||||
from sqlalchemy.orm import mapper, scoped_session, sessionmaker
|
from sqlalchemy.orm import mapper, scoped_session, sessionmaker
|
||||||
from sqlalchemy.pool import NullPool
|
from sqlalchemy.pool import NullPool
|
||||||
@ -47,62 +47,94 @@ from openlp.core.lib.ui import critical_error_message_box
|
|||||||
log = logging.getLogger(__name__)
|
log = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def _set_url_database(url, database):
|
||||||
|
try:
|
||||||
|
ret = URL.create(
|
||||||
|
drivername=url.drivername,
|
||||||
|
username=url.username,
|
||||||
|
password=url.password,
|
||||||
|
host=url.host,
|
||||||
|
port=url.port,
|
||||||
|
database=database,
|
||||||
|
query=url.query
|
||||||
|
)
|
||||||
|
except AttributeError: # SQLAlchemy <1.4
|
||||||
|
url.database = database
|
||||||
|
ret = url
|
||||||
|
assert ret.database == database, ret
|
||||||
|
return ret
|
||||||
|
|
||||||
|
|
||||||
|
def _get_scalar_result(engine, sql):
|
||||||
|
with engine.connect() as conn:
|
||||||
|
return conn.scalar(sql)
|
||||||
|
|
||||||
|
|
||||||
|
def _sqlite_file_exists(database):
|
||||||
|
if not os.path.isfile(database) or os.path.getsize(database) < 100:
|
||||||
|
return False
|
||||||
|
|
||||||
|
with open(database, 'rb') as f:
|
||||||
|
header = f.read(100)
|
||||||
|
|
||||||
|
return header[:16] == b'SQLite format 3\x00'
|
||||||
|
|
||||||
|
|
||||||
def database_exists(url):
|
def database_exists(url):
|
||||||
"""Check if a database exists.
|
"""Check if a database exists.
|
||||||
|
|
||||||
:param url: A SQLAlchemy engine URL.
|
:param url: A SQLAlchemy engine URL.
|
||||||
|
|
||||||
Performs backend-specific testing to quickly determine if a database
|
Performs backend-specific testing to quickly determine if a database
|
||||||
exists on the server. ::
|
exists on the server. ::
|
||||||
|
database_exists('postgresql://postgres@localhost/name') #=> False
|
||||||
database_exists('postgres://postgres@localhost/name') #=> False
|
create_database('postgresql://postgres@localhost/name')
|
||||||
create_database('postgres://postgres@localhost/name')
|
database_exists('postgresql://postgres@localhost/name') #=> True
|
||||||
database_exists('postgres://postgres@localhost/name') #=> True
|
|
||||||
|
|
||||||
Supports checking against a constructed URL as well. ::
|
Supports checking against a constructed URL as well. ::
|
||||||
|
engine = create_engine('postgresql://postgres@localhost/name')
|
||||||
engine = create_engine('postgres://postgres@localhost/name')
|
|
||||||
database_exists(engine.url) #=> False
|
database_exists(engine.url) #=> False
|
||||||
create_database(engine.url)
|
create_database(engine.url)
|
||||||
database_exists(engine.url) #=> True
|
database_exists(engine.url) #=> True
|
||||||
|
|
||||||
Borrowed from SQLAlchemy_Utils (v0.32.14) since we only need this one function.
|
Borrowed from SQLAlchemy_Utils since we only need this one function.
|
||||||
|
Copied from a fork/pull request since SQLAlchemy_Utils didn't supprt SQLAlchemy 1.4 when it was released:
|
||||||
|
https://github.com/nsoranzo/sqlalchemy-utils/blob/4f52578/sqlalchemy_utils/functions/database.py
|
||||||
"""
|
"""
|
||||||
|
|
||||||
url = copy(make_url(url))
|
url = copy(make_url(url))
|
||||||
database = url.database
|
database = url.database
|
||||||
if url.drivername.startswith('postgresql'):
|
dialect_name = url.get_dialect().name
|
||||||
url.database = 'template1'
|
|
||||||
else:
|
|
||||||
url.database = None
|
|
||||||
|
|
||||||
engine = create_engine(url)
|
if dialect_name == 'postgresql':
|
||||||
|
text = "SELECT 1 FROM pg_database WHERE datname='%s'" % database
|
||||||
|
for db in (database, 'postgres', 'template1', 'template0', None):
|
||||||
|
url = _set_url_database(url, database=db)
|
||||||
|
engine = create_engine(url, poolclass=NullPool)
|
||||||
|
try:
|
||||||
|
return bool(_get_scalar_result(engine, text))
|
||||||
|
except (ProgrammingError, OperationalError):
|
||||||
|
pass
|
||||||
|
return False
|
||||||
|
|
||||||
if engine.dialect.name == 'postgresql':
|
elif dialect_name == 'mysql':
|
||||||
text = "SELECT 1 FROM pg_database WHERE datname='{db}'".format(db=database)
|
url = _set_url_database(url, database=None)
|
||||||
return bool(engine.execute(text).scalar())
|
engine = create_engine(url, poolclass=NullPool)
|
||||||
|
|
||||||
elif engine.dialect.name == 'mysql':
|
|
||||||
text = ("SELECT SCHEMA_NAME FROM INFORMATION_SCHEMA.SCHEMATA "
|
text = ("SELECT SCHEMA_NAME FROM INFORMATION_SCHEMA.SCHEMATA "
|
||||||
"WHERE SCHEMA_NAME = '{db}'".format(db=database))
|
"WHERE SCHEMA_NAME = '%s'" % database)
|
||||||
return bool(engine.execute(text).scalar())
|
return bool(_get_scalar_result(engine, text))
|
||||||
|
|
||||||
elif engine.dialect.name == 'sqlite':
|
elif dialect_name == 'sqlite':
|
||||||
|
url = _set_url_database(url, database=None)
|
||||||
|
engine = create_engine(url, poolclass=NullPool)
|
||||||
if database:
|
if database:
|
||||||
return database == ':memory:' or os.path.exists(database)
|
return database == ':memory:' or _sqlite_file_exists(database)
|
||||||
else:
|
else:
|
||||||
# The default SQLAlchemy database is in memory,
|
# The default SQLAlchemy database is in memory,
|
||||||
# and :memory is not required, thus we should support that use-case
|
# and :memory is not required, thus we should support that use-case
|
||||||
return True
|
return True
|
||||||
|
|
||||||
else:
|
else:
|
||||||
text = 'SELECT 1'
|
text = 'SELECT 1'
|
||||||
try:
|
try:
|
||||||
url.database = database
|
engine = create_engine(url, poolclass=NullPool)
|
||||||
engine = create_engine(url)
|
return bool(_get_scalar_result(engine, text))
|
||||||
engine.execute(text)
|
|
||||||
return True
|
|
||||||
|
|
||||||
except (ProgrammingError, OperationalError):
|
except (ProgrammingError, OperationalError):
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user