commit5c22600d98Merge:bc12ca7bcdbb4aAuthor: Antoine Nguyen <tonio@ngyn.org> Date: Tue Nov 29 16:54:28 2022 +0100 Merge pull request #462 from Spitfireap/randomize-api-call-time randomize api call time commitbcdbb4a2ceAuthor: Spitap <dev@asdrip.fr> Date: Tue Nov 29 14:53:05 2022 +0100 fix typo commitbd1ddcef21Author: Spitap <dev@asdrip.fr> Date: Tue Nov 29 13:45:31 2022 +0100 randomize api call time commitbc12ca7327Merge:d364239bd0ecd0Author: Antoine Nguyen <tonio@ngyn.org> Date: Mon Nov 14 15:49:41 2022 +0100 Merge pull request #458 from Spitfireap/fix-include_try fix typo in dovecot configuration file commitbd0ecd0949Author: Spitap <dev@asdrip.fr> Date: Thu Nov 10 14:57:43 2022 +0100 fix typo in dovecot configuration file commitd364239348Merge:61838db3763300Author: Antoine Nguyen <tonio@ngyn.org> Date: Wed Nov 9 10:51:30 2022 +0100 Merge pull request #456 from modoboa/feature/improved_backup_restore WIP: Improved backup/restore system. commit37633008cbAuthor: Antoine Nguyen <tonio@ngyn.org> Date: Wed Nov 9 10:30:44 2022 +0100 Fixed restore mode commitd6f9a5b913Author: Antoine Nguyen <tonio@ngyn.org> Date: Tue Nov 8 17:20:25 2022 +0100 Few fixes. commit8b1d60ee59Author: Antoine Nguyen <tonio@ngyn.org> Date: Tue Nov 8 17:19:23 2022 +0100 Few fixes commit2b5edae5d5Author: Antoine Nguyen <tonio@ngyn.org> Date: Sun Nov 6 10:30:24 2022 +0100 WIP: Improved backup/restore system. commit61838dbe4dAuthor: Antoine Nguyen <tonio@ngyn.org> Date: Sat Nov 5 09:30:50 2022 +0100 Check if restore is defined before doing anything else. fix #453 commit962cac3ad9Merge:1b192c5ef2359aAuthor: Antoine Nguyen <tonio@ngyn.org> Date: Fri Nov 4 09:41:20 2022 +0100 Merge pull request #450 from Spitfireap/fixed-super-call fixed super call in modoboa's script commitef2359a2a8Author: Spitap <dev@asdrip.fr> Date: Thu Nov 3 23:10:21 2022 +0100 fixed super call commit1b192c5fd5Merge:754d652b0b0146Author: Antoine Nguyen <tonio@ngyn.org> Date: Thu Nov 3 15:34:48 2022 +0100 Merge pull request #449 from Spitfireap/fixed-import-typo fixed constants import commitb0b01465d9Author: Spitap <dev@asdrip.fr> Date: Thu Nov 3 15:00:07 2022 +0100 fixed constants import commit754d652fc2Author: Antoine Nguyen <tonio@ngyn.org> Date: Thu Nov 3 12:27:04 2022 +0100 Few fixes commitcb5fa75693Merge:1afb8e6e01265aAuthor: Antoine Nguyen <tonio@ngyn.org> Date: Thu Nov 3 12:20:25 2022 +0100 Merge pull request #444 from Spitfireap/tighter-config-file-perm tighter config file permission commit1afb8e61fcMerge:15c17798dd0b7dAuthor: Antoine Nguyen <tonio@ngyn.org> Date: Thu Nov 3 12:17:16 2022 +0100 Merge pull request #424 from Spitfireap/restore Backup & restore system commit8dd0b7d497Author: Spitap <dev@asdrip.fr> Date: Thu Nov 3 10:57:03 2022 +0100 Last camelCase commit554611b366Author: Spitap <dev@asdrip.fr> Date: Thu Nov 3 10:54:06 2022 +0100 review fix commit15c17796f2Merge:ce8e7e684d1363Author: Antoine Nguyen <tonio@ngyn.org> Date: Fri Oct 28 09:43:30 2022 +0200 Merge pull request #446 from Spitfireap/fix-ssl-min-protocol fixed ssl_min_protocol setting commit84d13633a1Author: Spitap <dev@asdrip.fr> Date: Thu Oct 27 22:37:47 2022 +0200 fixed ssl_min_protocol setting commitce8e7e6027Merge:8e8ae5ffe7df27Author: Antoine Nguyen <tonio@ngyn.org> Date: Thu Oct 27 17:56:37 2022 +0200 Merge pull request #445 from Spitfireap/dovecot-fixes Fixes ssl permission error, updated ssl_protocol parameter commite01265a4eeMerge:a5fba03235ef3bAuthor: Spitap <dev@asdrip.fr> Date: Thu Oct 27 17:44:37 2022 +0200 Merge branch 'tighter-config-file-perm' of https://github.com/Spitfireap/modoboa-installer into tighter-config-file-perm commita5fba03264Author: Spitap <dev@asdrip.fr> Date: Thu Oct 27 11:13:47 2022 +0200 tighter config file permission commitfe7df276fcAuthor: Spitap <dev@asdrip.fr> Date: Thu Oct 27 17:25:39 2022 +0200 Check dovecot version greater commit8f34f0af6fAuthor: Spitap <dev@asdrip.fr> Date: Thu Oct 27 17:00:58 2022 +0200 Fixes ssl permission error, updated ssl_protocol parameter commit8e8ae5fb9cMerge:67f6ceefefbf54Author: Antoine Nguyen <tonio@ngyn.org> Date: Thu Oct 27 16:49:20 2022 +0200 Merge pull request #439 from stefaweb/master Update config_dict_template.py for default max_servers value commit235ef3befbAuthor: Spitap <dev@asdrip.fr> Date: Thu Oct 27 11:13:47 2022 +0200 thighter config file permission commit67f6cee8eaMerge:b84abbb53f7f8eAuthor: Antoine Nguyen <tonio@ngyn.org> Date: Tue Oct 25 19:32:37 2022 +0200 Merge pull request #442 from Spitfireap/patch-1 Set $max_server to 2 to avoid amavis crash commit5c9d5c9a03Author: Spitap <dev@asdrip.fr> Date: Tue Oct 25 16:58:57 2022 +0200 DKIM keys restore, Radicale backup/restore, fixes commit4c1f8710b5Author: Spitap <dev@asdrip.fr> Date: Tue Oct 25 16:04:55 2022 +0200 Added dkim key backup commite34eb4b337Author: Spitap <dev@asdrip.fr> Date: Tue Oct 25 13:59:28 2022 +0200 fix database path commit53f7f8ef9dAuthor: Spitfireap <45575529+Spitfireap@users.noreply.github.com> Date: Wed Oct 19 08:19:40 2022 +0000 Update config_dict_template.py commit35778cd614Merge:6726f5bb84abbbAuthor: Spitfireap <45575529+Spitfireap@users.noreply.github.com> Date: Tue Oct 18 17:17:48 2022 +0200 Merge branch 'modoboa:master' into restore commitfefbf549a4Author: Stephane Leclerc <sleclerc@actionweb.fr> Date: Thu Oct 6 13:36:13 2022 +0200 Update config_dict_template.py for default max_server value commit6726f5b1a2Author: Spitap <dev@asdrip.fr> Date: Mon Sep 26 13:39:28 2022 +0200 Improved path generation, path mistake proofing commita192cbcbd0Author: Spitap <dev@asdrip.fr> Date: Mon Sep 19 16:40:25 2022 +0200 Updated doc, default path on conf file commit5bed9655eaAuthor: Spitap <dev@asdrip.fr> Date: Mon Sep 19 15:53:19 2022 +0200 fixed typo commit6b096a7470Author: Spitap <dev@asdrip.fr> Date: Mon Sep 19 15:50:03 2022 +0200 Simplified db dumps restore commite30add03fdMerge:d75d83f1f8dd1bAuthor: Spitap <dev@asdrip.fr> Date: Mon Sep 19 15:39:05 2022 +0200 Update from master commitd75d83f202Author: Spitap <dev@asdrip.fr> Date: Mon Sep 19 15:13:44 2022 +0200 more refactoring commitf3811b4b39Author: Spitap <dev@asdrip.fr> Date: Mon Sep 19 14:59:43 2022 +0200 refactoring commitb0d56b3989Author: Spitap <dev@asdrip.fr> Date: Thu Sep 15 11:32:57 2022 +0200 PEP formating commit53e3e3ec58Author: Spitap <dev@asdrip.fr> Date: Fri Aug 5 15:20:11 2022 +0200 Better UX, use of os to concatenate path commite546d2cb23Author: Spitap <dev@asdrip.fr> Date: Wed Jul 27 16:32:59 2022 +0200 Better UX commit70faa1c5cbAuthor: Spitap <dev@asdrip.fr> Date: Wed Jul 27 15:58:41 2022 +0200 Fixed backupdir index commit563979a7ddAuthor: Spitap <dev@asdrip.fr> Date: Wed Jul 27 15:51:22 2022 +0200 fixed mail backup/restore commitee2ccf0647Author: Spitap <dev@asdrip.fr> Date: Wed Jul 27 14:35:48 2022 +0200 Fixed postfix install, added restore to readme commit2077c94b52Author: Spitap <dev@asdrip.fr> Date: Tue Jul 26 17:05:00 2022 +0200 Fix amavis config file not copied to right location commit4a7222bd24Author: Spitap <dev@asdrip.fr> Date: Tue Jul 26 16:53:24 2022 +0200 Fixed nginx call to uwsgi commite7b6104195Author: Spitap <dev@asdrip.fr> Date: Tue Jul 26 16:39:41 2022 +0200 fixed install within class commit4a00590354Author: Spitap <dev@asdrip.fr> Date: Tue Jul 26 16:20:03 2022 +0200 fixed restore disclamer commit15768c429eAuthor: Spitap <dev@asdrip.fr> Date: Tue Jul 26 12:07:42 2022 +0200 Restore workflow done commit439ffb94c4Author: Spitap <dev@asdrip.fr> Date: Mon Jul 25 18:54:47 2022 +0200 initial commit commit37bc21dfd3Author: Spitap <dev@asdrip.fr> Date: Tue Jul 26 10:36:08 2022 +0200 Backup postewhite.conf instead of custom whitelist Postwhite.conf contains a custom host list commit26204143afMerge:2097055d495afdAuthor: Spitap <dev@asdrip.fr> Date: Mon Jul 25 22:10:26 2022 +0200 Merge branch 'master' into backup commit20970557deAuthor: Spitap <dev@asdrip.fr> Date: Mon Jul 25 22:05:35 2022 +0200 Allow to disable mail backup commit632c26596eAuthor: Spitap <dev@asdrip.fr> Date: Mon Jul 25 21:52:15 2022 +0200 Update backup readme commit9e1c18cd6bAuthor: Spitap <dev@asdrip.fr> Date: Thu Jul 21 19:09:53 2022 +0200 Fix argument passed as list instead of string commitdb6457c5f5Author: Spitap <dev@asdrip.fr> Date: Thu Jul 21 19:07:18 2022 +0200 better path handling commit579faccfa5Author: Spitap <dev@asdrip.fr> Date: Thu Jul 21 19:00:32 2022 +0200 added an automatic bash option (no path provided) or a path provided bash (for cron job) commit5318fa279bAuthor: Spitap <dev@asdrip.fr> Date: Thu Jul 21 18:00:50 2022 +0200 bash option commit74de6a9bb1Author: Spitap <dev@asdrip.fr> Date: Thu Jul 21 17:31:56 2022 +0200 Reset pgpass before trying to backup secondary dbs commit54185a7c5aAuthor: Spitap <dev@asdrip.fr> Date: Thu Jul 21 17:26:40 2022 +0200 Fix database backup logic issue commit1f9d69c37cAuthor: Spitap <dev@asdrip.fr> Date: Thu Jul 21 17:21:59 2022 +0200 Fix copy issue commit8d02d2a9fbAuthor: Spitap <dev@asdrip.fr> Date: Thu Jul 21 17:09:23 2022 +0200 added safe mkdir in utils, use utils.mkdir_safe() in backup commit6f604a5fecAuthor: Spitap <dev@asdrip.fr> Date: Thu Jul 21 16:53:56 2022 +0200 Fix loop logic commit568c4a65a0Author: Spitap <dev@asdrip.fr> Date: Thu Jul 21 16:51:32 2022 +0200 fix none-type passed to os.path commitdc84a79528Author: Spitap <dev@asdrip.fr> Date: Thu Jul 21 14:12:35 2022 +0200 Note : capitalize affects only first letter commit304e25fa3cAuthor: Spitap <dev@asdrip.fr> Date: Thu Jul 21 14:10:57 2022 +0200 Fix getattr commit070efd61c4Author: Spitap <dev@asdrip.fr> Date: Thu Jul 21 14:08:39 2022 +0200 Fix import commit9917d8023eAuthor: Spitap <dev@asdrip.fr> Date: Thu Jul 21 14:02:41 2022 +0200 Edited README, fix backup run process commit27b9de6755Author: Spitap <dev@asdrip.fr> Date: Thu Jul 21 13:48:44 2022 +0200 database backup commit56ed214fb5Author: Spitap <dev@asdrip.fr> Date: Tue Jul 19 19:06:53 2022 +0200 Starting work on backup system
287 lines
9.9 KiB
Python
287 lines
9.9 KiB
Python
"""Database related tools."""
|
|
|
|
import os
|
|
import pwd
|
|
import stat
|
|
|
|
from . import package
|
|
from . import system
|
|
from . import utils
|
|
|
|
|
|
class Database(object):
|
|
|
|
"""Common database backend."""
|
|
|
|
default_port = None
|
|
packages = None
|
|
service = None
|
|
|
|
def __init__(self, config):
|
|
"""Install if necessary."""
|
|
self.config = config
|
|
engine = self.config.get("database", "engine")
|
|
self.dbhost = self.config.get("database", "host")
|
|
self.dbport = self.config.getint(
|
|
"database", "port", fallback=self.default_port)
|
|
self.dbuser = config.get(engine, "user")
|
|
self.dbpassword = config.get(engine, "password")
|
|
if self.config.getboolean("database", "install"):
|
|
self.install_package()
|
|
|
|
def install_package(self):
|
|
"""Install database package if required."""
|
|
package.backend.install_many(self.packages[package.backend.FORMAT])
|
|
system.enable_and_start_service(self.service)
|
|
|
|
|
|
class PostgreSQL(Database):
|
|
|
|
"""Postgres."""
|
|
|
|
default_port = 5432
|
|
packages = {
|
|
"deb": ["postgresql", "postgresql-server-dev-all"],
|
|
"rpm": ["postgresql-server", "postgresql-server-devel", "postgresql"]
|
|
}
|
|
service = "postgresql"
|
|
|
|
def __init__(self, config):
|
|
super().__init__(config)
|
|
self._pgpass_done = False
|
|
|
|
def install_package(self):
|
|
"""Install database if required."""
|
|
name, version = utils.dist_info()
|
|
if "CentOS" in name:
|
|
initdb_cmd = "postgresql-setup --initdb"
|
|
cfgfile = "/var/lib/pgsql/data/pg_hba.conf"
|
|
package.backend.install_many(self.packages[package.backend.FORMAT])
|
|
utils.exec_cmd(initdb_cmd)
|
|
pattern = "s/^host(.+)ident$/host$1md5/"
|
|
utils.exec_cmd("perl -pi -e '{}' {}".format(pattern, cfgfile))
|
|
else:
|
|
package.backend.install_many(self.packages[package.backend.FORMAT])
|
|
system.enable_and_start_service(self.service)
|
|
|
|
def _exec_query(self, query, dbname=None, dbuser=None, dbpassword=None):
|
|
"""Exec a postgresql query."""
|
|
cmd = "psql"
|
|
if dbname:
|
|
cmd += " -d {}".format(dbname)
|
|
if dbuser:
|
|
self._setup_pgpass(dbname, dbuser, dbpassword)
|
|
cmd += " -h {} -p {} -U {} -w".format(
|
|
self.dbhost, self.dbport, dbuser)
|
|
query = query.replace("'", "'\"'\"'")
|
|
cmd = "{} -c '{}' ".format(cmd, query)
|
|
utils.exec_cmd(cmd, sudo_user=self.dbuser)
|
|
|
|
def create_user(self, name, password):
|
|
"""Create a user."""
|
|
query = "SELECT 1 FROM pg_roles WHERE rolname='{}'".format(name)
|
|
code, output = utils.exec_cmd(
|
|
"""psql -tAc "{}" | grep -q 1""".format(query),
|
|
sudo_user=self.dbuser)
|
|
if not code:
|
|
return
|
|
query = "CREATE USER {} PASSWORD '{}'".format(name, password)
|
|
self._exec_query(query)
|
|
|
|
def create_database(self, name, owner):
|
|
"""Create a database."""
|
|
code, output = utils.exec_cmd(
|
|
"psql -lqt | cut -d \| -f 1 | grep -w {} | wc -l"
|
|
.format(name), sudo_user=self.dbuser)
|
|
if code:
|
|
return
|
|
utils.exec_cmd(
|
|
"createdb {} -O {}".format(name, owner),
|
|
sudo_user=self.dbuser)
|
|
|
|
def grant_access(self, dbname, user):
|
|
"""Grant access to dbname."""
|
|
query = "GRANT ALL ON DATABASE {} TO {}".format(dbname, user)
|
|
self._exec_query(query)
|
|
|
|
def grant_right_on_table(self, dbname, table, user, right):
|
|
"""Grant specific right to user on table."""
|
|
query = "GRANT {} ON {} TO {}".format(
|
|
right.upper(), table, user)
|
|
self._exec_query(query, dbname=dbname)
|
|
|
|
def _setup_pgpass(self, dbname, dbuser, dbpasswd):
|
|
"""Setup .pgpass file."""
|
|
if self._pgpass_done:
|
|
return
|
|
if self.dbhost not in ["localhost", "127.0.0.1"]:
|
|
self._pgpass_done = True
|
|
return
|
|
pw = pwd.getpwnam(self.dbuser)
|
|
target = os.path.join(pw[5], ".pgpass")
|
|
with open(target, "w") as fp:
|
|
fp.write("127.0.0.1:*:{}:{}:{}\n".format(
|
|
dbname, dbname, dbpasswd))
|
|
mode = stat.S_IRUSR | stat.S_IWUSR
|
|
os.chmod(target, mode)
|
|
os.chown(target, pw[2], pw[3])
|
|
self._pgpass_done = True
|
|
|
|
def load_sql_file(self, dbname, dbuser, dbpassword, path):
|
|
"""Load SQL file."""
|
|
self._setup_pgpass(dbname, dbuser, dbpassword)
|
|
cmd = "psql -h {} -p {} -d {} -U {} -w < {}".format(
|
|
self.dbhost, self.dbport, dbname, dbuser, path)
|
|
utils.exec_cmd(cmd, sudo_user=self.dbuser)
|
|
|
|
def dump_database(self, dbname, dbuser, dbpassword, path):
|
|
"""Dump DB to SQL file."""
|
|
# Reset pgpass since we backup multiple db (different secret set)
|
|
self._pgpass_done = False
|
|
self._setup_pgpass(dbname, dbuser, dbpassword)
|
|
cmd = "pg_dump -h {} -d {} -U {} -O -w > {}".format(
|
|
self.dbhost, dbname, dbuser, path)
|
|
utils.exec_cmd(cmd, sudo_user=self.dbuser)
|
|
|
|
|
|
class MySQL(Database):
|
|
|
|
"""MySQL backend."""
|
|
|
|
default_port = 3306
|
|
packages = {
|
|
"deb": ["mariadb-server"],
|
|
"rpm": ["mariadb", "mariadb-devel", "mariadb-server"],
|
|
}
|
|
service = "mariadb"
|
|
|
|
def _escape(self, query):
|
|
"""Replace special characters."""
|
|
return query.replace("'", "'\"'\"'")
|
|
|
|
def install_package(self):
|
|
"""Preseed package installation."""
|
|
name, version = utils.dist_info()
|
|
name = name.lower()
|
|
if name.startswith("debian"):
|
|
if version.startswith("8"):
|
|
self.packages["deb"].append("libmysqlclient-dev")
|
|
elif version.startswith("11"):
|
|
self.packages["deb"].append("libmariadb-dev")
|
|
else:
|
|
self.packages["deb"].append("libmariadbclient-dev")
|
|
elif name == "ubuntu":
|
|
self.packages["deb"].append("libmysqlclient-dev")
|
|
super(MySQL, self).install_package()
|
|
queries = []
|
|
if name.startswith("debian"):
|
|
if version.startswith("8"):
|
|
package.backend.preconfigure(
|
|
"mariadb-server", "root_password", "password",
|
|
self.dbpassword)
|
|
package.backend.preconfigure(
|
|
"mariadb-server", "root_password_again", "password",
|
|
self.dbpassword)
|
|
return
|
|
if version.startswith("11"):
|
|
queries = [
|
|
"SET PASSWORD FOR 'root'@'localhost' = PASSWORD('{}')"
|
|
.format(self.dbpassword),
|
|
"flush privileges"
|
|
]
|
|
if not queries:
|
|
queries = [
|
|
"UPDATE user SET plugin='' WHERE user='root'",
|
|
"UPDATE user SET password=PASSWORD('{}') WHERE USER='root'"
|
|
.format(self.dbpassword),
|
|
"flush privileges"
|
|
]
|
|
for query in queries:
|
|
utils.exec_cmd(
|
|
"mysql -D mysql -e '{}'".format(self._escape(query)))
|
|
|
|
def _exec_query(self, query, dbname=None, dbuser=None, dbpassword=None):
|
|
"""Exec a mysql query."""
|
|
if dbuser is None and dbpassword is None:
|
|
dbuser = self.dbuser
|
|
dbpassword = self.dbpassword
|
|
cmd = "mysql -h {} -P {} -u {}".format(
|
|
self.dbhost, self.dbport, dbuser)
|
|
if dbpassword:
|
|
cmd += " -p{}".format(dbpassword)
|
|
if dbname:
|
|
cmd += " -D {}".format(dbname)
|
|
utils.exec_cmd(cmd + """ -e '{}' """.format(self._escape(query)))
|
|
|
|
def create_user(self, name, password):
|
|
"""Create a user."""
|
|
self._exec_query(
|
|
"CREATE USER '{}'@'%' IDENTIFIED BY '{}'".format(
|
|
name, password))
|
|
self._exec_query(
|
|
"CREATE USER '{}'@'localhost' IDENTIFIED BY '{}'".format(
|
|
name, password))
|
|
|
|
def create_database(self, name, owner):
|
|
"""Create a database."""
|
|
self._exec_query(
|
|
"CREATE DATABASE IF NOT EXISTS {} "
|
|
"DEFAULT CHARACTER SET {} "
|
|
"DEFAULT COLLATE {}".format(
|
|
name, self.config.get("mysql", "charset"),
|
|
self.config.get("mysql", "collation"))
|
|
)
|
|
self.grant_access(name, owner)
|
|
|
|
def grant_access(self, dbname, user):
|
|
"""Grant access to dbname."""
|
|
self._exec_query(
|
|
"GRANT ALL PRIVILEGES ON {}.* to '{}'@'%'"
|
|
.format(dbname, user))
|
|
self._exec_query(
|
|
"GRANT ALL PRIVILEGES ON {}.* to '{}'@'localhost'"
|
|
.format(dbname, user))
|
|
|
|
def grant_right_on_table(self, dbname, table, user, right):
|
|
"""Grant specific right to user on table."""
|
|
query = "GRANT {} ON {}.{} TO '{}'@'%'".format(
|
|
right.upper(), dbname, table, user)
|
|
self._exec_query(query)
|
|
|
|
def load_sql_file(self, dbname, dbuser, dbpassword, path):
|
|
"""Load SQL file."""
|
|
utils.exec_cmd(
|
|
"mysql -h {} -P {} -u {} -p{} {} < {}".format(
|
|
self.dbhost, self.dbport, dbuser, dbpassword, dbname, path)
|
|
)
|
|
|
|
def dump_database(self, dbname, dbuser, dbpassword, path):
|
|
"""Dump DB to SQL file."""
|
|
cmd = "mysqldump -h {} -u {} -p{} {} > {}".format(
|
|
self.dbhost, dbuser, dbpassword, dbname, path)
|
|
utils.exec_cmd(cmd, sudo_user=self.dbuser)
|
|
|
|
|
|
def get_backend(config):
|
|
"""Return appropriate backend."""
|
|
engine = config.get("database", "engine")
|
|
if engine == "postgres":
|
|
backend = PostgreSQL
|
|
elif engine == "mysql":
|
|
backend = MySQL
|
|
else:
|
|
raise utils.FatalError("database engine not supported")
|
|
return backend(config)
|
|
|
|
|
|
def create(config, name, password):
|
|
"""Create a database and a user."""
|
|
backend = get_backend(config)
|
|
backend.create_user(name, password)
|
|
backend.create_database(name)
|
|
|
|
|
|
def grant_database_access(config, name, user):
|
|
"""Grant access to a database."""
|
|
get_backend(config).grant_access(name, user)
|