102 lines
3.1 KiB
Python
102 lines
3.1 KiB
Python
import frappe
|
|
from frappe import _
|
|
|
|
|
|
class DbManager:
|
|
def __init__(self, db):
|
|
"""
|
|
Pass root_conn here for access to all databases.
|
|
"""
|
|
if db:
|
|
self.db = db
|
|
|
|
def get_current_host(self):
|
|
return self.db.sql("select user()")[0][0].split("@")[1]
|
|
|
|
def create_user(self, user, password, host=None):
|
|
host = host or self.get_current_host()
|
|
password_predicate = f" IDENTIFIED BY '{password}'" if password else ""
|
|
self.db.sql(f"CREATE USER IF NOT EXISTS '{user}'@'{host}'{password_predicate}")
|
|
|
|
def delete_user(self, target, host=None):
|
|
host = host or self.get_current_host()
|
|
self.db.sql(f"DROP USER IF EXISTS '{target}'@'{host}'")
|
|
|
|
def create_database(self, target):
|
|
if target in self.get_database_list():
|
|
self.drop_database(target)
|
|
self.db.sql(f"CREATE DATABASE `{target}` CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci")
|
|
|
|
def drop_database(self, target):
|
|
self.db.sql_ddl(f"DROP DATABASE IF EXISTS `{target}`")
|
|
|
|
def grant_all_privileges(self, target, user, host=None):
|
|
host = host or self.get_current_host()
|
|
permissions = (
|
|
(
|
|
"SELECT, INSERT, UPDATE, DELETE, CREATE, DROP, INDEX, ALTER, "
|
|
"CREATE TEMPORARY TABLES, CREATE VIEW, EVENT, TRIGGER, SHOW VIEW, "
|
|
"CREATE ROUTINE, ALTER ROUTINE, EXECUTE, LOCK TABLES"
|
|
)
|
|
if frappe.conf.rds_db
|
|
else "ALL PRIVILEGES"
|
|
)
|
|
self.db.sql(f"GRANT {permissions} ON `{target}`.* TO '{user}'@'{host}'")
|
|
|
|
def flush_privileges(self):
|
|
self.db.sql("FLUSH PRIVILEGES")
|
|
|
|
def get_database_list(self):
|
|
return self.db.sql("SHOW DATABASES", pluck=True)
|
|
|
|
@staticmethod
|
|
def restore_database(verbose: bool, target: str, source: str, user: str, password: str) -> None:
|
|
"""
|
|
Function to restore the given SQL file to the target database.
|
|
:param target: The database to restore to.
|
|
:param source: The SQL dump to restore
|
|
:param user: The database username
|
|
:param password: The database password
|
|
:return: Nothing
|
|
"""
|
|
|
|
import shlex
|
|
from shutil import which
|
|
|
|
from frappe.database import get_command
|
|
from frappe.utils import execute_in_shell
|
|
|
|
# Ensure that the entire process fails if any part of the pipeline fails
|
|
command: list[str] = ["set -o pipefail;"]
|
|
|
|
# Handle gzipped backups
|
|
if source.endswith(".gz"):
|
|
if gzip := which("gzip"):
|
|
command.extend([gzip, "-cd", source, "|"])
|
|
else:
|
|
raise Exception("`gzip` not installed")
|
|
else:
|
|
command.extend(["cat", source, "|"])
|
|
|
|
# Newer versions of MariaDB add in a line that'll break on older versions, so remove it
|
|
command.extend(["sed", r"'/\/\*M\{0,1\}!999999\\- enable the sandbox mode \*\//d'", "|"])
|
|
|
|
# Generate the restore command
|
|
bin, args, bin_name = get_command(
|
|
socket=frappe.conf.db_socket,
|
|
host=frappe.conf.db_host,
|
|
port=frappe.conf.db_port,
|
|
user=user,
|
|
password=password,
|
|
db_name=target,
|
|
)
|
|
if not bin:
|
|
return frappe.throw(
|
|
_("{} not found in PATH! This is required to restore the database.").format(bin_name),
|
|
exc=frappe.ExecutableNotFound,
|
|
)
|
|
command.append(bin)
|
|
command.append(shlex.join(args))
|
|
|
|
execute_in_shell(" ".join(command), check_exit_code=True, verbose=verbose)
|
|
frappe.cache.delete_keys("") # Delete all keys associated with this site.
|