summaryrefslogtreecommitdiff
path: root/lib/sqlalchemy/testing/plugin/provision.py
diff options
context:
space:
mode:
authorMike Bayer <mike_mp@zzzcomputing.com>2014-07-26 20:50:57 -0400
committerMike Bayer <mike_mp@zzzcomputing.com>2014-07-26 20:50:57 -0400
commitd2358629c9ea7f12718f32b64c7de7f2030dcd32 (patch)
tree1933ad41f00c94c00b9590e72f275e94e04f69ce /lib/sqlalchemy/testing/plugin/provision.py
parente65cb04974cc35e6f8301b9b7bc2a02d11edf139 (diff)
downloadsqlalchemy-d2358629c9ea7f12718f32b64c7de7f2030dcd32.tar.gz
- scale up for mysql, sqlite
Diffstat (limited to 'lib/sqlalchemy/testing/plugin/provision.py')
-rw-r--r--lib/sqlalchemy/testing/plugin/provision.py166
1 files changed, 145 insertions, 21 deletions
diff --git a/lib/sqlalchemy/testing/plugin/provision.py b/lib/sqlalchemy/testing/plugin/provision.py
index e6790f877..7c54cd643 100644
--- a/lib/sqlalchemy/testing/plugin/provision.py
+++ b/lib/sqlalchemy/testing/plugin/provision.py
@@ -1,11 +1,73 @@
from sqlalchemy.engine import url as sa_url
+from sqlalchemy import text
+from sqlalchemy.util import compat
+from .. import config, engines
+import os
+
+
+class register(object):
+ def __init__(self):
+ self.fns = {}
+
+ @classmethod
+ def init(cls, fn):
+ return register().for_db("*")(fn)
+
+ def for_db(self, dbname):
+ def decorate(fn):
+ self.fns[dbname] = fn
+ return self
+ return decorate
+
+ def __call__(self, cfg, *arg):
+ if isinstance(cfg, compat.string_types):
+ url = sa_url.make_url(cfg)
+ elif isinstance(cfg, sa_url.URL):
+ url = cfg
+ else:
+ url = cfg.db.url
+ backend = url.get_backend_name()
+ if backend in self.fns:
+ return self.fns[backend](cfg, *arg)
+ else:
+ return self.fns['*'](cfg, *arg)
def create_follower_db(follower_ident):
- from .. import config, engines
- follower_ident = "test_%s" % follower_ident
+ for cfg in _configs_for_db_operation():
+ url = cfg.db.url
+ backend = url.get_backend_name()
+ _create_db(cfg, cfg.db, follower_ident)
+
+ new_url = sa_url.make_url(str(url))
+
+ new_url.database = follower_ident
+
+
+def configure_follower(follower_ident):
+ for cfg in config.Config.all_configs():
+ _configure_follower(cfg, follower_ident)
+
+
+def setup_config(db_url, db_opts, options, file_config, follower_ident):
+ if follower_ident:
+ db_url = _follower_url_from_main(db_url, follower_ident)
+ eng = engines.testing_engine(db_url, db_opts)
+ eng.connect().close()
+ cfg = config.Config.register(eng, db_opts, options, file_config)
+ if follower_ident:
+ _configure_follower(cfg, follower_ident)
+ return cfg
+
+
+def drop_follower_db(follower_ident):
+ for cfg in _configs_for_db_operation():
+ url = cfg.db.url
+ _drop_db(cfg, cfg.db, follower_ident)
+
+def _configs_for_db_operation():
hosts = set()
for cfg in config.Config.all_configs():
@@ -19,47 +81,109 @@ def create_follower_db(follower_ident):
url.username, url.host, url.database)
if host_conf not in hosts:
- if backend.startswith("postgresql"):
- _pg_create_db(cfg.db, follower_ident)
- elif backend.startswith("mysql"):
- _mysql_create_db(cfg.db, follower_ident)
+ yield cfg
+ hosts.add(host_conf)
- new_url = sa_url.make_url(str(url))
+ for cfg in config.Config.all_configs():
+ cfg.db.dispose()
- new_url.database = follower_ident
- eng = engines.testing_engine(new_url, cfg.db_opts)
- if backend.startswith("postgresql"):
- _pg_init_db(eng)
- elif backend.startswith("mysql"):
- _mysql_init_db(eng)
+@register.init
+def _create_db(cfg, eng, ident):
+ raise NotImplementedError("no DB creation routine for cfg: %s" % eng.url)
- hosts.add(host_conf)
+
+@register.init
+def _drop_db(cfg, eng, ident):
+ raise NotImplementedError("no DB drop routine for cfg: %s" % eng.url)
+
+
+@register.init
+def _configure_follower(cfg, ident):
+ pass
+
+
+@register.init
+def _follower_url_from_main(url, ident):
+ url = sa_url.make_url(url)
+ url.database = ident
+ return url
+
+
+@_follower_url_from_main.for_db("sqlite")
+def _sqlite_follower_url_from_main(url, ident):
+ return sa_url.make_url("sqlite:///%s.db" % ident)
-def _pg_create_db(eng, ident):
+@_create_db.for_db("postgresql")
+def _pg_create_db(cfg, eng, ident):
with eng.connect().execution_options(
isolation_level="AUTOCOMMIT") as conn:
try:
- conn.execute("DROP DATABASE %s" % ident)
+ _pg_drop_db(cfg, conn, ident)
except:
pass
currentdb = conn.scalar("select current_database()")
conn.execute("CREATE DATABASE %s TEMPLATE %s" % (ident, currentdb))
-def _pg_init_db(eng):
+@_create_db.for_db("mysql")
+def _mysql_create_db(cfg, eng, ident):
+ with eng.connect() as conn:
+ try:
+ _mysql_drop_db(cfg, conn, ident)
+ except:
+ pass
+ conn.execute("CREATE DATABASE %s" % ident)
+ conn.execute("CREATE DATABASE %s_test_schema" % ident)
+ conn.execute("CREATE DATABASE %s_test_schema_2" % ident)
+
+
+@_configure_follower.for_db("mysql")
+def _mysql_configure_follower(config, ident):
+ config.test_schema = "%s_test_schema" % ident
+ config.test_schema_2 = "%s_test_schema_2" % ident
+
+
+@_create_db.for_db("sqlite")
+def _sqlite_create_db(cfg, eng, ident):
pass
-def _mysql_create_db(eng, ident):
+@_drop_db.for_db("postgresql")
+def _pg_drop_db(cfg, eng, ident):
+ with eng.connect().execution_options(
+ isolation_level="AUTOCOMMIT") as conn:
+ conn.execute(
+ text(
+ "select pg_terminate_backend(pid) from pg_stat_activity "
+ "where usename=current_user and pid != pg_backend_pid() "
+ "and datname=:dname"
+ ), dname=ident)
+ conn.execute("DROP DATABASE %s" % ident)
+
+
+@_drop_db.for_db("sqlite")
+def _sqlite_drop_db(cfg, eng, ident):
+ os.remove("%s.db" % ident)
+
+
+@_drop_db.for_db("mysql")
+def _mysql_drop_db(cfg, eng, ident):
with eng.connect() as conn:
try:
+ conn.execute("DROP DATABASE %s_test_schema" % ident)
+ except:
+ pass
+ try:
+ conn.execute("DROP DATABASE %s_test_schema_2" % ident)
+ except:
+ pass
+ try:
conn.execute("DROP DATABASE %s" % ident)
except:
pass
- conn.execute("CREATE DATABASE %s" % ident)
-def _mysql_init_db(eng):
- pass
+
+