# coding: utf-8 # ------------------------------------------------------------------- # YakPanel # ------------------------------------------------------------------- # Copyright (c) 2015-2099 YakPanel(www.yakpanel.com) All rights reserved. # ------------------------------------------------------------------- # Author: hwliang # ------------------------------------------------------------------- # ------------------------------ # 数据备份模块 # ------------------------------ import json import os import re import sys import time from typing import Tuple, Union, Optional sys.stdout.reconfigure(encoding="utf-8") os.chdir('/www/server/panel') if not 'class/' in sys.path: sys.path.insert(0, 'class/') if not 'class_v2/' in sys.path: sys.path.insert(0, 'class_v2/') import public _VERSION = 2.0 class backup: _BACKUP_DIR = public.M('config').where("id=?", (1,)).getField('backup_path') if not _BACKUP_DIR: _BACKUP_DIR = '/www/backup/' _exclude = "" _err_log = '/tmp/backup_err.log' _inode_min = 10 _cloud = None # _is_save_local = os.path.exists('data/is_save_local_backup.pl') _is_save_local = True # 本地保留得权限, 默认允许 _error_msg = "" _backup_all = False # 数据库备份 _DB_BACKUP_DIR: str = os.path.join(_BACKUP_DIR, "database") # mysql 备份 _MYSQL_BACKUP_DIR = os.path.join(_DB_BACKUP_DIR, "mysql", "crontab_backup") _MYSQLDUMP_BIN = public.get_mysqldump_bin() # mongodb 备份 _MONGODB_BACKUP_DIR = os.path.join(_DB_BACKUP_DIR, "mongodb", "crontab_backup") _MONGODBDUMP_BIN = os.path.join(public.get_setup_path(), "mongodb/bin/mongodump") _MONGOEXPORT_BIN = os.path.join(public.get_setup_path(), "mongodb/bin/mongoexport") # redis 备份 _REDIS_BACKUP_DIR = os.path.join(_DB_BACKUP_DIR, "redis", "crontab_backup") # pgsql 备份 _PGSQL_BACKUP_DIR = os.path.join(_DB_BACKUP_DIR, "pgsql", "crontab_backup") _PGDUMP_BIN = os.path.join(public.get_setup_path(), "pgsql/bin/pg_dump") _SPLIT_SIZE = 5 * 1024 * 1024 * 1024 # 拆分文件大小 _BACKUP_TIME_FORMAT = "%Y%m%d_%H%M%S" _CLOUD_OBJ = { "qiniu": "Qiniu Cloud Storage", "alioss": "Alibaba Cloud OSS", "ftp": "FTP Storage", "bos": "Baidu Cloud BOS", "obs": "Huawei Cloud OBS", "aws_s3": "AWS S3", "gdrive": "Google Drive", "msonedrive": "Microsoft OneDrive", "gcloud_storage": "Google Cloud Storage", "upyun": "Upyun Storage", "jdcloud": "JD Cloud Storage", "txcos": "Tencent Cloud COS", 'tianyiyun': "Tianyi Cloud ZOS", 'webdav': "WebDav", 'minio': "MinIO Storage", 'dogecloud': "Duoji Cloud COS", 'localhost': 'Local Disk' } def __init__(self, cloud_object=None, cron_info: dict = None): """ @name 数据备份对象 @param cloud_object 远程上传对象,需具备以下几个属性和方法: _title = '中文名称,如:阿里云OSS' _name = '英文名称,如:alioss' upload_file(filename,data_type = None) 文件名 , 数据类型 site/database/path # 新函数 cloud_delete_file(abs_file_name) # 弃置 delete_file delete_file(filename,data_type = None) 文件名 , 数据类型 site/database/path 给_error_msg赋值,传递错误消息: _error_msg = "错误消息" """ if not cron_info: cron_info = {} for i in [ self._BACKUP_DIR, self._MYSQL_BACKUP_DIR, self._MONGODB_BACKUP_DIR, self._REDIS_BACKUP_DIR, self._PGSQL_BACKUP_DIR, ]: if not os.path.exists(i): os.makedirs(i) self._cloud_new = None self._cloud = cloud_object self.cron_info = None self.echo_id = cron_info.get('echo') if cron_info and 'echo' in cron_info.keys(): self.cron_info = self.get_cron_info(cron_info["echo"]) if not public.M('sqlite_master').db('backup').where( 'type=? AND name=? AND sql LIKE ?', ('table', 'backup', '%cron_id%') ).count(): public.M('backup').execute("ALTER TABLE 'backup' ADD 'cron_id' INTEGER DEFAULT 0", ()) self.check_databases() self.db_data_type = "database" self.site_data_type = "site" self.path_data_type = "path" def echo_start(self): print("=" * 90) print(public.lang("★start backup[{}]", str(public.format_date()))) print("=" * 90) def echo_end(self): print("=" * 90) print(public.lang("☆backup is complete[{}]", str(public.format_date()))) print("=" * 90) print("\n") def echo_info(self, msg): print("|-{}".format(public.lang(msg))) def echo_error(self, msg): print("=" * 90) print(public.lang("|-Error:{}", str(msg))) if self._error_msg: self._error_msg += "\n" self._error_msg += msg # 取排除列表用于计算排除目录大小 def get_exclude_list(self, exclude: list = None): if exclude is None: exclude = [] if not exclude: tmp_exclude = os.getenv('BT_EXCLUDE') if tmp_exclude: exclude = tmp_exclude.split(',') if not exclude: return [] return exclude # 构造排除 def get_exclude(self, exclude: list = None): if not exclude: exclude = [] self._exclude = "" if not exclude: tmp_exclude = os.getenv('BT_EXCLUDE') if tmp_exclude: exclude = tmp_exclude.split(',') if not exclude: return "" for ex in exclude: if ex[-1] == '/': ex = ex[:-1] self._exclude += " --exclude='" + ex + "'" self._exclude += " " return self._exclude def GetDiskInfo2(self): # 取磁盘分区信息 temp = public.ExecShell("df -T -P|grep '/'|grep -v tmpfs|grep -v 'snap/core'|grep -v udev")[0] tempInodes = public.ExecShell("df -i -P|grep '/'|grep -v tmpfs|grep -v 'snap/core'|grep -v udev")[0] temp1 = temp.split('\n') tempInodes1 = tempInodes.split('\n') diskInfo = [] n = 0 cuts = [] for tmp in temp1: n += 1 try: inodes = tempInodes1[n - 1].split() # noinspection RegExpRedundantEscape disk = re.findall( r"^(.+)\s+([\w\.]+)\s+([\w\.]+)\s+([\w\.]+)\s+([\w\.]+)\s+([\d%]{2,4})\s+(/.{0,50})$", tmp.strip() ) if disk: disk = disk[0] if len(disk) < 6: continue if disk[2].find('M') != -1: continue if disk[2].find('K') != -1: continue if len(disk[6].split('/')) > 10: continue if disk[6] in cuts: continue if disk[6].find('docker') != -1: continue if disk[1].strip() in ['tmpfs']: continue arr = {'filesystem': disk[0].strip(), 'type': disk[1].strip(), 'path': disk[6]} tmp1 = [disk[2], disk[3], disk[4], disk[5]] arr['size'] = tmp1 if int(inodes[1]) == 0 and int(inodes[2]) == 0: arr['inodes'] = [inodes[1], 10000, 10000, 0] else: arr['inodes'] = [inodes[1], inodes[2], inodes[3], inodes[4]] diskInfo.append(arr) except: continue return diskInfo # 取磁盘可用空间 def get_disk_free(self, dfile): diskInfo = self.GetDiskInfo2() if not diskInfo: return '', 0, 0 _root = None for d in diskInfo: if d['path'] == '/': _root = d continue if re.match("^{}/.+".format(d['path']), dfile): return d['path'], float(d['size'][2]) * 1024, int(d['inodes'][2]) if _root: return _root['path'], float(_root['size'][2]) * 1024, int(_root['inodes'][2]) return '', 0, 0 # ======================== Public Func ============================= @property def chose_local_flag(self) -> bool: """本地备份快速标识""" if not self.cron_info or not isinstance(self.cron_info, dict): return False return self.cron_info.get("save_local", 0) == 1 or self.cron_info.get("backupTo", "") in [ "Local Disk", "localhost" ] def get_cloud_file_path(self, abs_file, data_type) -> str: """ 构造备份文件云端绝对路径 云对象中前缀要求 prefix_dict = { "site": "web", "database": "db", "path": "path", } 并以正则规则切割目录, 详情云对象 func build_object_name """ file_name = os.path.basename(abs_file) try: final = self._cloud.build_object_name(data_type, file_name) except: final = f"{data_type}/{file_name}" final = final.lstrip("/") if final.startswith("/") else final return f"{self.cron_info.get("backupTo", "unknown")}|{final}" def cloud_init_ctrller(self, cloud_name: str, obj_name: str, data_type: str) -> None: """云初始化控制器""" if self._cloud is None: from cloud_stora_upload_v2 import CloudStoraUpload self._cloud_new = CloudStoraUpload() self._cloud = self._cloud_new.run(cloud_name) if self._cloud is False: if not (cloud_name == 'Local Disk' or cloud_name == 'localhost'): # 没有 self._cloud 云存储对象, 且不是本地存储 self._cloud is False self.echo_info("Failed to link to cloud storage, please check if the configuration is correct!") self.send_failture_notification( '"Failed to link to cloud storage, please check if the configuration is correct!"', target=f"{obj_name}|{data_type}", remark='' ) def cloud_uploader(self, abs_file, upload_path, obj_name, data_type: str) -> Optional[bool]: """ 上传 :param abs_file: 备份文件绝对路径 :param upload_path: 上传路径 :param obj_name: 对象名称 :param data_type: 数据类型 site/database/path, 需遵循云对象 build obj name 要求 """ if self._cloud._title == "Google Drive": # Google Drive upload_path 特殊处理 upload_path = upload_path.rstrip('/') upload_path = os.path.dirname(upload_path) backup_size = os.path.getsize(abs_file) # 判断是否 大于 5 GB 进行切割上传 split_type_flag = self.cron_info.get("split_type", "") != "size" split_val_flag = backup_size // 1024 // 1024 > self.cron_info.get("split_value", 0) if all([ self.cron_info.get("split_type"), self.cron_info.get("split_value"), backup_size >= self._SPLIT_SIZE, (split_type_flag or split_val_flag), ]): is_status, err = self.split_upload_file(abs_file, upload_path) if is_status is False: remark = "Backup to " + self._cloud._title self.send_failture_notification(err, target=f"{obj_name}|{data_type}", remark=remark) return False else: # 正常上传 error_msg = "" if self._cloud._title != "AWS S3": # Google Drive , Ftp, other if hasattr(self._cloud_new, "cloud_upload_file"): # 统一使用新函数上传, 使用upload_path作为绝对路径 res = self._cloud_new.cloud_upload_file( file_name=abs_file, upload_path=upload_path ) else: # 兜底, 由云函数路径规则生成路径 res = self._cloud.upload_file(abs_file, data_type) else: # AWS S3 & Minio, minio使用S3_Endpoit兼容, 使用upload_path作为绝对路径 res = self._cloud.upload_file1( local_file_name=abs_file, object_name=upload_path ) if not res: if hasattr(self._cloud, "error_msg"): if self._cloud.error_msg: error_msg = self._cloud.error_msg if not error_msg: error_msg = public.lang("Backup task execution failed.") self.echo_error(error_msg) remark = public.lang("Backup to " + self._cloud._title) self.send_failture_notification( error_msg, target=f"{obj_name}|{data_type}", remark=remark ) return False self.echo_info("Successfully uploaded to {}".format(self._cloud._title)) return True def cloud_upload_ctrller(self, *args, **kwargs) -> Optional[bool]: """异常处理重试的上传控制器, (失败重试需要云对象支持)""" fial_count = 0 res = None while fial_count <= 2: # 防止CloudApi限流 try: res = self.cloud_uploader(*args, **kwargs) break except Exception as e: self.echo_info(public.lang( f"upload to cloud fail...please check if the configuration is correct! error: {e}") ) self.echo_info(public.lang("try again after few second...please wait...")) public.WriteLog("Cloud_Upload", f"upload to cloud fail...error:{e}") time.sleep(10 * (fial_count + 1)) fial_count += 1 continue return res def retain_local_ctrller(self, backup_path: str) -> bool: """本地保留控制器""" save_msg = public.lang("Cron task settings keep local. Local backup is retained.") drop_msg = public.lang("Cron task settings do not keep local. Local backup is not retained.") setting_limit_msg = public.lang("User setting to don't keep any local backup") if self.cron_info["save_local"] == 1: # 保留 self.echo_info(save_msg) return True if self._is_save_local is False: # 全局关闭保留本地权限(未实现) self.echo_info(f"{setting_limit_msg}, deleted {backup_path}") else: if self.chose_local_flag: # 保留 self.echo_info(save_msg) return True else: self.echo_info(drop_msg) # 本地不保留 try: os.remove(backup_path) except: pass return False # ============================ Path ================================================ # 备份指定目录 def backup_path(self, spath, dfile=None, exclude: list = None, save: int = 3, echo_id: str = None): # spath 备份的目录, dfile 备份到指定路径, 绝对路径abs if not exclude: exclude = [] self.echo_start() error_msg = "" echo_id = self.echo_id if echo_id is None else echo_id self.cron_info = public.M('crontab').where('echo=?', (echo_id,)).find() if not self.cron_info: error_msg = public.lang("The scheduled task does not exist! {}", str(echo_id)) self.echo_error(error_msg) self.send_failture_notification(error_msg) return False, error_msg if not os.path.exists(spath): error_msg = 'The specified directory {} does not exist!'.format(spath) self.echo_error(error_msg) self.send_failture_notification(error_msg, target="{}|path".format(spath)) return False spath = spath.rstrip('/') basename = os.path.basename(spath) if not dfile: # 构造特殊文件名 time_format = public.format_date(self._BACKUP_TIME_FORMAT) file_name = f'path_{basename}_{time_format}_{public.GetRandomString(6)}.tar.gz' dfile = os.path.join(self._BACKUP_DIR, 'path', basename, file_name) # 压缩目录, 保存到本地 if not self.backup_path_to(spath, dfile, exclude): if self._error_msg: error_msg = self._error_msg self.send_failture_notification(error_msg, target="{}|path".format(spath)) return False backup_path = dfile # 本地备份文件绝对路径 backup_size = os.path.getsize(dfile) nick_name = self._CLOUD_OBJ[self.cron_info.get("backupTo")] # 预构造入库信息 pdata = { 'cron_id': self.cron_info["id"], 'type': '2', 'name': spath, 'pid': 0, 'filename': backup_path, 'addtime': public.format_date(), 'ps': f"Plan task backup to {nick_name}", 'size': backup_size } # ============================= 上传阶段 ============================= # 上传初始化 self.cloud_init_ctrller( cloud_name=self.cron_info['backupTo'], obj_name=spath, data_type=self.path_data_type ) if self._cloud: self.echo_info("Uploading to {}, please wait...".format(self._cloud._title)) # 使用upload_path upload_path = os.path.join(self._cloud_new.backup_path, "path", os.path.basename(backup_path)) res = self.cloud_upload_ctrller( abs_file=backup_path, upload_path=upload_path, obj_name=spath, data_type=self.path_data_type ) if res: upload_path = upload_path.lstrip("/") # 云端路径入库 pdata["filename"] = f"{self.cron_info.get("backupTo", "unknown")}|{upload_path}" public.M("backup").insert(pdata) # 是否本地保留 if self.retain_local_ctrller(backup_path): # 修改为 本地路径入库 pdata["filename"] = backup_path pdata["ps"] = "Plan task backup to localhost" public.M("backup").insert(pdata) # ============================ 清理阶段 ============================= if self._cloud: # 云端清理 cloud_backups = public.M('backup').where( "cron_id=? and type=? and pid=? and name=? and filename LIKE ?", (self.cron_info['id'], '2', 0, spath, f'{self.cron_info.get("backupTo")}%') ).field('id,name,filename').select() self.delete_old(cloud_backups, save, self.path_data_type, nick_name) if self.chose_local_flag: # 本地清理 local_backups = public.M('backup').where( "cron_id=? and type=? and pid=? and name=? and filename NOT LIKE '%|%'", (self.cron_info['id'], '2', 0, spath) ).field('id,name,filename').select() self.delete_old(local_backups, save, self.path_data_type, "Local Disk") self.echo_end() self.save_backup_status(True, target="{}|path".format(spath)) return dfile # 清理过期备份文件, 适用于Path, Database, Site兼容wp单独处理. def delete_old(self, backups, save, data_type=None, name: str = ""): if not backups or not isinstance(backups, list): # self.echo_info('Failed to clean expired backup, error:{} '.format(backups)) return name = name if name else "Local Disk" self.echo_info('-' * 88) self.echo_info(f'Keep the latest number of backups: {save} copies for {name}') # 跳过手动备份文件 new_backups = [] for i in range(len(backups)): if data_type == 'database': # 数据库备份 new_backups.append(backups[i]) elif data_type == 'site' and backups[i]['name'][:4] == 'web_' and backups[i]['name'][ -7:] == '.tar.gz': # 网站备份 new_backups.append(backups[i]) elif data_type == 'path' and backups[i]['name'][:5] == 'path_': # 目录备份 new_backups.append(backups[i]) if new_backups: backups = new_backups[:] num = len(backups) - int(save) if num > 0: for backup in backups: if backup['filename'].find('|') != -1: # 处理目录备份到云 tmp = backup['filename'].split('|') abs_file_name = tmp[-1] else: # 是本地 abs_file_name = backup['filename'] file_name = os.path.basename(abs_file_name) # 尝试删除本地文件 if os.path.exists(abs_file_name): os.remove(abs_file_name) # 尝试删除远程文件 if self._cloud: if file_name.endswith("_split"): # 是否为切割文件 self._cloud_new.cloud_delete_dir(file_name) else: # 统一删除云函数入口 self._cloud_new.cloud_delete_file(abs_file_name) if not abs_file_name.startswith("/"): abs_file_name = "/" + abs_file_name self.echo_info( public.lang(f'Expired backup files have been cleaned from {name}: {abs_file_name}') ) # 从数据库清理 public.M('backup').where('id=?', (backup['id'],)).delete() num -= 1 if num < 1: break # 压缩目录 def backup_path_to( self, spath, dfile, exclude: list = None, siteName: str = None, for_wp2: bool = False, cp_time: str = None ): if not exclude: exclude = [] if not os.path.exists(spath): error_msg = public.lang('The specified directory {} does not exist!', spath) self.echo_error(error_msg) return False if spath[-1] == '/': spath = spath[:-1] dirname = os.path.basename(spath) dpath = os.path.dirname(dfile) if not os.path.exists(dpath): os.makedirs(dpath, 384) self.get_exclude(exclude) if self._exclude: self._exclude = self._exclude.replace(spath + '/', '') exclude_config = self._exclude exclude_list = self.get_exclude_list(exclude) p_size = public.get_path_size(spath, exclude=exclude_list) if not self._exclude: exclude_config = "Not set" if siteName: self.echo_info(public.lang('Backup site: {}', siteName)) self.echo_info(public.lang('Website document root: {}', spath)) else: self.echo_info(public.lang('Backup directory: {}', spath)) self.echo_info(public.lang('Directory size: {}', str(public.to_size(p_size)))) self.echo_info(public.lang('Exclusion setting: {}', exclude_config)) check_status, check_msg = self.avilable_disk(p_size, dfile) if check_status is False: return False compression_time = "" if not for_wp2: stime = time.time() if os.path.exists(dfile): os.remove(dfile) public.ExecShell( "cd " + os.path.dirname( spath) + " && tar zcvhf '" + dfile + "' " + self._exclude + " '" + dirname + "' 2>{err_log} 1> /dev/null".format( err_log=self._err_log ) ) compression_time = str('{:.2f}'.format(time.time() - stime)) tar_size = os.path.getsize(dfile) if tar_size < 1: self.echo_error(public.lang("Compression failed!")) self.echo_info(public.readFile(self._err_log)) return False compression_time = cp_time if cp_time else compression_time self.echo_info( public.lang( 'Compression completed, took {} seconds, compressed package size: {}', compression_time, str(public.to_size(tar_size)) ) ) if siteName: self.echo_info(public.lang('Site backed up to: {}', dfile)) else: self.echo_info(public.lang('Directory has been backed up to: {}', dfile)) if os.path.exists(self._err_log): try: os.remove(self._err_log) except FileNotFoundError: pass return dfile # ============================ Site ================================================= def __backup_site_check_local_flag(self) -> bool: if self.cron_info['backupTo'] == "localhost" or self.cron_info['backupTo'] == "Local Disk": if_save_local = True else: if_save_local = self.cron_info["save_local"] if self.cron_info else 0 if if_save_local and not self._is_save_local: if not os.path.exists('data/is_save_local_backup.pl'): os.mknod('data/is_save_local_backup.pl') save_local = True if self._is_save_local and if_save_local else False return save_local def __backup_site_delete_old(self, target: str, backups: list, save: int, data_type=None, if_wp: bool = False): self.echo_info('-' * 88) self.echo_info(public.lang('Keep the latest number of backups: {} copies for {}'.format(save, target))) num = len(backups) - int(save) if num > 0: for backup in backups: try: # 从数据库清理 if not if_wp: public.M('backup').where('id=?', (backup['id'],)).delete() else: public.M('wordpress_backups').where('id=?', (backup['id'],)).delete() if os.path.exists(backup.get('filename', '')): os.remove(backup['filename']) self.echo_info(public.lang( 'Expired backup files have been cleaned from disk: {}', backup['filename']) ) # 尝试删除远程site文件 elif backup.get('filename', '').find('|') != -1 and self._cloud and data_type == 'site': try: tmp = backup['filename'].split('|') backup['filename'] = tmp[0] backup['name'] = tmp[-1] siteName = public.M('sites').where('id=?', (backup['pid'],)).getField('name') site_backup_dir = self._cloud_new.backup_path fname = backup['name'] dfile = os.path.join(site_backup_dir, 'site', siteName, fname) res = self._cloud_new.cloud_delete_file(dfile, 'site') if res: remote_name = backup.get('name', '') if not remote_name: remote_name = backup.get('filename', '').split('|')[-1] remote_path = os.path.join(self._cloud.backup_path, 'site', siteName, remote_name) self.echo_info( public.lang( 'Expired backup files have been cleaned from {}: {}', target, remote_path )) return res except Exception: continue else: continue except Exception: continue num -= 1 if num < 1: break # 备份指定站点 def backup_site(self, siteName, save: int = 3, exclude: list = None, echo_id=None) -> str | bool: panelPath = '/www/server/panel/' os.chdir(panelPath) sys.path.insert(0, panelPath) echo_id = self.echo_id if echo_id is None else echo_id cron_id = public.M("crontab").where('echo=?', (echo_id,)).getField('id') if echo_id else 0 self.echo_start() self.cron_info = public.M('crontab').where('echo=?', (echo_id,)).field( 'backupTo,name,save_local,notice,notice_channel,id,sType,split_type,split_value,db_backup_path' ).select()[0] if not self.cron_info: return False save_local = self.__backup_site_check_local_flag() find = public.M('sites').where('name=?', (siteName,)).select()[0] if not find: error_msg = 'The specified website [{}] does not exist!'.format(siteName) self.echo_error(error_msg) self.send_failture_notification(error_msg, target=siteName) return False site_backup_dir = self.get_site_backup_dir( self.cron_info['backupTo'], self.cron_info['db_backup_path'], siteName ) backup_dir = os.path.join(site_backup_dir, 'site') # save local file at first, insert db depend on 'save_local' bak_info = None if find['project_type'] != 'WP2': spath = find['path'] pid = find['id'] fname = 'web_{}_{}_{}.tar.gz'.format( siteName, public.format_date(self._BACKUP_TIME_FORMAT), public.GetRandomString(6) ) # save dfile = os.path.join(site_backup_dir, 'site', siteName, fname) if not self.backup_path_to( spath, dfile, exclude, siteName=siteName ): error_msg = self._error_msg if self._error_msg else "" self.send_failture_notification(error_msg, target=siteName) return False # insert if save_local: public.M('backup').insert({ 'cron_id': cron_id, 'type': 0, 'name': fname, 'pid': pid, 'filename': dfile, 'addtime': public.format_date(), 'size': os.path.getsize(dfile) }) else: # WP2 try: from wp_toolkit import wpbackup # save & insert start = time.time() bak_info = wpbackup(find['id']).backup_full_get_data(self.cron_info) cp_time = str('{:.2f}'.format(time.time() - start)) spath = find['path'] pid = bak_info.id fname = bak_info.bak_file.split('/')[-1] dfile = bak_info.bak_file if not self.backup_path_to( spath, dfile, exclude, siteName=siteName, for_wp2=True, cp_time=cp_time ): error_msg = self._error_msg if self._error_msg else "" self.send_failture_notification(error_msg, target=siteName) return False except Exception as e: self.echo_info(str(e)) self.echo_end() return False backup_size = os.path.getsize(dfile) if not self._backup_all: self.check_disk_space(backup_size, backup_dir) # init_cloud self.cloud_init_ctrller( cloud_name=self.cron_info['backupTo'], obj_name=siteName, data_type="site", ) # cloud part if self._cloud: cloud_file_name = dfile + '|' + self._cloud._name + '|' + fname basename = os.path.basename(dfile) upload_path = os.path.join(self._cloud_new.backup_path, "site", siteName, basename) self.echo_info(public.lang("Uploading to {}, please wait...".format(self._cloud._title))) res = self.cloud_upload_ctrller( abs_file=dfile, upload_path=upload_path, obj_name=siteName, data_type="site" ) if res: # insert data if find['project_type'] != 'WP2': public.M('backup').insert({ 'cron_id': cron_id, 'type': 0, 'name': fname, 'pid': pid, 'filename': cloud_file_name, 'addtime': public.format_date(), 'size': backup_size }) else: # wp public.M('wordpress_backups').insert({ 's_id': bak_info.site_id, 'bak_type': 3, 'bak_file': cloud_file_name, 'bak_time': bak_info.bak_time, 'size': backup_size, 'cron_id': cron_id, }) self.echo_info(public.lang("Local backup has been kept")) if not save_local: # 选了云但没选择保存本地, 移除 public.M('wordpress_backups').where('id=?', (pid,)).delete() if os.path.exists(dfile): os.remove(dfile) self.echo_info(public.lang( 'User settings do not retain local backups, deleted {}', dfile )) # 清理多余备份 if find['project_type'] != 'WP2': if save_local: local_backups = public.M('backup').where( "cron_id=? AND type=? AND pid=? AND filename NOT LIKE '%|%' AND filename LIKE?", (cron_id, '0', pid, f'{self._BACKUP_DIR}%') ).order('addtime').select() # 清理local self.__backup_site_delete_old('Local Disk', local_backups, save, 'site') if self._cloud: cloud_backups = public.M('backup').where( 'cron_id=? and type=? and pid=? and filename LIKE ?', (cron_id, '0', pid, f"%{self._cloud._name}%") ).order('addtime').select() self.__backup_site_delete_old(self._cloud._title, cloud_backups, save, 'site') # 清理cloud else: # wp site_id = public.M('sites').where('name=?', (siteName,)).getField('id') if save_local: local_wp_backups = public.M('wordpress_backups').where( "cron_id=? AND s_id=? AND bak_file NOT LIKE '%|%' AND bak_file LIKE?", (cron_id, site_id, f'{self._BACKUP_DIR}%') ).order('bak_time').select() local_wp_backups = [ { 'id': x.get('id'), 'pid': x.get('s_id'), 'name': x.get('bak_file', '').split('/')[-1], # 纯文件名 'filename': x.get('bak_file', ''), # 绝对路径 } for x in local_wp_backups if x ] # 清理wp2 local self.__backup_site_delete_old('Local Disk', local_wp_backups, save, 'site', True) if self._cloud: cloud_wp_backups = public.M('wordpress_backups').where( "cron_id=? AND s_id=? AND bak_file LIKE ? AND bak_file LIKE '%|%'", (cron_id, site_id, f"%{self._cloud._name}%") ).order('bak_time').select() cloud_wp_backups = [ { 'id': x.get('id'), 'pid': x.get('s_id'), 'name': x.get('bak_file', '').split('|')[0].split('/')[-1], # 纯文件名 'filename': x.get('bak_file', ''), # 绝对路径 } for x in cloud_wp_backups if x and isinstance(x, dict) ] # 清理wp2 cloud self.__backup_site_delete_old(self._cloud._title, cloud_wp_backups, save, 'site', True) self.echo_end() return dfile # 备份所有站点 def backup_site_all(self, save=3, echo_id=None): if echo_id is None: echo_id = self.echo_id self.cron_info = public.M('crontab').where('echo=?', (echo_id,)).field( 'backupto,name,save_local,notice,notice_channel,id,sType').select()[0] sites = public.M('sites').field('name').select() self._backup_all = True failture_count = 0 results = [] backup_files = [] # 用于存储成功备份的文件路径 if self._cloud is None: try: cloud_name = self.cron_info['backupTo'] except: cloud_name = self.cron_info['backupto'] from cloud_stora_upload_v2 import CloudStoraUpload self._cloud_new = CloudStoraUpload() self._cloud = self._cloud_new.run(cloud_name) for site in sites: self._error_msg = "" result = self.backup_site(site['name'], save, echo_id=echo_id) if not result: failture_count += 1 else: backup_files.append(result) # 收集成功备份的文件路径 results.append((site['name'], result, self._error_msg,)) self.save_backup_status(result, target="{}|site".format(site['name']), msg=self._error_msg) if failture_count > 0: self.send_all_failture_notification("site", results) else: size = sum(os.path.getsize(file) for file in backup_files if os.path.exists(file)) backup_dir = self._BACKUP_DIR self.check_disk_space(size, backup_dir) self._backup_all = False # ============================ Database ============================================== # 备份所有数据库 def backup_database_all(self, save, echo_id: str): if echo_id is None: echo_id = self.echo_id self.cron_info = public.M('crontab').where("echo=? and sType='database'", (echo_id,)).find() if not self.cron_info: print(public.lang("The scheduled task does not exist!")) sName = self.cron_info["sName"] save = self.cron_info["save"] db_type = str(self.cron_info.get("db_type", "mysql")).lower() if not db_type: public.M('crontab').where("echo=? and sType='database'", (echo_id,)).update({"db_type": "mysql"}) self.cron_info["db_type"] = "mysql" db_type = "mysql" if db_type == "mysql": try: from database import database database = database() mysql_server = public.M('database_servers').where("db_type=?", ('mysql',)).select() for i in mysql_server: database.SyncGetDatabases(public.to_dict_obj({'sid': i['id']})) except: pass database_list = [] if db_type in ["mysql", "mongodb", "pgsql"]: if sName == "ALL": # 备份所有数据库 data_list = public.M("databases").field("name").where("LOWER(type)=LOWER(?)", (db_type)).select() database_list = [data["name"] for data in data_list] elif db_type == "redis": database_list.append("redis") # 上传对象 self._cloud = None cloud_name = self.cron_info["backupTo"] if cloud_name.find("localhost") == -1: from cloud_stora_upload_v2 import CloudStoraUpload self._cloud_new = CloudStoraUpload() self._cloud = self._cloud_new.run(cloud_name) self._backup_all = True resp_list = [] failture_count = 0 for db_name in database_list: self._error_msg = "" status, backup_path = self.backup_database(db_name, save, echo_id=echo_id) if status is False: failture_count += 1 resp_list.append((db_name, status, backup_path)) self.save_backup_status(status, target=f"{db_name}|database", msg=self._error_msg) if failture_count > 0: self.send_all_failture_notification(self.db_data_type, resp_list) else: size = 0 backup_dir = self._DB_BACKUP_DIR self.check_disk_space(size, backup_dir) self._backup_all = False def __transfer_db_backup_format(self, org_path: str) -> str: # 用于cloud识别, 云备份路径规则 if not os.path.basename(org_path).startswith("db_"): backup_path = os.path.join( os.path.dirname(org_path), f"db_{os.path.basename(org_path)}" ) try: public.ExecShell(f"mv '{org_path}' '{backup_path}'") except: backup_path = org_path else: backup_path = org_path return backup_path # 备份单个数据库 def backup_database(self, sName, save, echo_id: str = None): self.echo_start() sName = sName.strip() echo_id = self.echo_id if echo_id is None else echo_id self.cron_info = public.M('crontab').where("echo=? and sType='database'", (echo_id,)).find() if not self.cron_info: error_msg = public.lang("The scheduled task does not exist! {}", str(echo_id)) self.echo_error(error_msg) self.send_failture_notification(error_msg) return False, error_msg db_type = str(self.cron_info.get("db_type", "mysql")).lower() if not db_type: public.M('crontab').where("echo=? and sType='database'", (echo_id,)).update({"db_type": "mysql"}) self.cron_info["db_type"] = "mysql" db_type = "mysql" if db_type in ["mysql", "mongodb", "pgsql"]: database = public.M("databases").where("name=? and LOWER(type)=LOWER(?)", (sName, db_type)).find() if not database: error_msg = public.lang("{} Backup database {} does not exist", db_type, sName) self.echo_error(error_msg) self.send_failture_notification(error_msg, target="{}|database".format(sName)) return False, error_msg elif db_type == "redis": database = {"id": 0, "name": "redis", "db_type": 0} else: database = {} db_name = database.get("name", "unknown") func_dict = { "mysql": self.mysql_backup_database, "mongodb": self.mongodb_backup_database, "redis": self.redis_backup_database, "pgsql": self.pgsql_backup_database, } func_backup_database = func_dict[db_type] table_list = self.cron_info["table_list"] args = { "backup_mode": self.cron_info["backup_mode"], "db_backup_path": self.cron_info["db_backup_path"], "save_local": self.cron_info["save_local"] } if table_list: # table_list 备份 args.update({"table_list": [table_list]}) status, msg = func_backup_database(database, args) if status is False: # 备份失败 self.send_failture_notification(msg, target="{}|database".format(db_name)) return status, msg # format backup_path = self.__transfer_db_backup_format(org_path=msg) backup_size = os.path.getsize(backup_path) self.echo_info("The database has been backed up to: {}".format(backup_path)) if not self._backup_all: self.check_disk_space(backup_size, backup_path) file_name = os.path.basename(backup_path) nick_name = self._CLOUD_OBJ[self.cron_info.get("backupTo")] # 预构造入库信息 pdata = { "type": "1", "name": file_name, "pid": database["id"], "filename": backup_path, "size": backup_size, "addtime": public.format_date(), "ps": f"Plan task backup to {nick_name}", "cron_id": self.cron_info["id"], } # ============================= 上传阶段 ============================= # 上传初始化 self.cloud_init_ctrller( cloud_name=self.cron_info['backupTo'], obj_name=db_name, data_type=self.db_data_type ) # 上传 if self._cloud: self.echo_info(public.lang("Uploading to {}, please wait...", self._cloud._title)) # upload_path = self.get_cloud_file_path(abs_file=backup_path, data_type=self.db_data_type) if db_type == "redis": upload_path = os.path.join(self._cloud_new.backup_path, "database", db_type, file_name) else: upload_path = os.path.join(self._cloud_new.backup_path, "database", db_type, db_name, file_name) res = self.cloud_upload_ctrller( abs_file=backup_path, upload_path=upload_path, obj_name=db_name, data_type=self.db_data_type ) if res: upload_path = upload_path.lstrip("/") # 云端路径入库 pdata["filename"] = f"{self.cron_info.get("backupTo", "unknown")}|{upload_path}" public.M("backup").insert(pdata) # 是否本地保留 if self.retain_local_ctrller(backup_path): # 修改为 本地路径入库 pdata["filename"] = backup_path pdata["ps"] = f"Plan task backup to localhost" public.M("backup").insert(pdata) # ============================ 清理阶段 ============================= # 清理多余备份 # 云 if self._cloud: cloud_backups = public.M("backup").where( "cron_id=? and type=? and pid=? and name LIKE ? and filename LIKE ?", (self.cron_info["id"], "1", database["id"], f"%{db_name}%", f"{self.cron_info.get("backupTo")}%") ).field("id,name,filename").select() self.delete_old(cloud_backups, save, self.db_data_type, nick_name) # 本地 if self.chose_local_flag: local_backups = public.M("backup").where( "cron_id=? and type=? and pid=? and name LIKE ? and filename not LIKE '%|%'", (self.cron_info["id"], "1", database["id"], f"%{db_name}%") ).field("id,name,filename").select() self.delete_old(local_backups, save, self.db_data_type, "Local Disk") self.echo_end() if os.path.exists(self._err_log): try: os.remove(self._err_log) except FileNotFoundError: pass return True, backup_path def avilable_disk(self, check_size: int, back_dir: str) -> Tuple[bool, str]: disk_path, disk_free, disk_inode = self.get_disk_free(back_dir) self.echo_info( f"The available disk space for partition {disk_path} is: {public.to_size(disk_free)}," f" and the available Inode is: {disk_inode}" ) if disk_path: err_tamp = """The available {} space of the target partition is less than {}, and the backup cannot be completed. Please increase the disk capacity or change the default backup directory on the settings page!""" if disk_free < check_size: error_msg = err_tamp.format("disk", public.to_size(check_size)) self.echo_error(error_msg) return False, error_msg if disk_inode < self._inode_min: error_msg = err_tamp.format("Inode", self._inode_min) self.echo_error(error_msg) return False, error_msg return True, "" # mysql 备份数据库 def mysql_backup_database(self, db_find: dict, args: dict) -> Tuple[bool, str]: import db_mysql storage_type = args.get("storage_type", "db") # 备份的文件数量, 按照数据库 | 按照表 table_list = args.get("table_list", []) # 备份的集合 db_name = db_find["name"] db_host = "localhost" db_user = "root" if db_find["db_type"] == 0: # 本地数据库 backup_mode = args.get("backup_mode", "0") if backup_mode == "1": # 使用非 root 账号 db_user = db_find["username"] # 获取数据库用户 db_password = db_find["password"] # 获取数据库密码 print("Using non root user {} to backup database {}".format(db_user, db_name)) else: # 使用 root 账号 # db_user = "root" db_password = public.M("config").where("id=?", (1,)).getField("mysql_root") if not db_password: error_msg = "The database password is empty! Please set the database password first!" self.echo_error(error_msg) return False, error_msg try: from panelMysql import panelMysql db_port = int(panelMysql().query("show global variables like 'port'")[0][1]) except: db_port = 3306 elif db_find["db_type"] == 1: # 远程数据库 conn_config = json.loads(db_find["conn_config"]) db_host = conn_config["db_host"] db_port = int(conn_config["db_port"]) db_user = conn_config["db_user"] db_password = conn_config["db_password"] elif db_find["db_type"] == 2: conn_config = public.M("database_servers").where("id=? AND LOWER(db_type)=LOWER('mysql')", db_find["sid"]).find() db_host = conn_config["db_host"] db_port = int(conn_config["db_port"]) db_user = conn_config["db_user"] db_password = conn_config["db_password"] else: error_msg = "Unknown database type" self.echo_error(error_msg) return False, error_msg if not db_password: error_msg = "The database password cannot be empty" self.echo_error(error_msg) return False, error_msg db_charset = public.get_database_character(db_name) mysql_obj = db_mysql.panelMysql().set_host(db_host, db_port, None, db_user, db_password) if isinstance(mysql_obj, bool): error_msg = "Failed to connect to database [{}: {}]".format(db_host, db_port) self.echo_error(error_msg) return False, error_msg db_data = mysql_obj.query( "select sum(DATA_LENGTH)+sum(INDEX_LENGTH) from information_schema.tables where table_schema='{}'".format( db_name ) ) db_size = 0 if isinstance(db_data, list) and len(db_data) != 0: if db_data[0][0] is not None: db_size = db_data[0][0] if not db_size: error_msg = "The specified database '{}' has no data!".format(db_name) self.echo_error(error_msg) return False, error_msg self.echo_info('Backup MySQL database: {}'.format(db_name)) self.echo_info("Database size: {}".format(public.to_size(db_size))) self.echo_info("Database Character Set: {}".format(db_charset)) check_status, check_msg = self.avilable_disk(db_size, self._MYSQL_BACKUP_DIR) if check_status is False: return False, check_msg stime = time.time() self.echo_info("Start exporting database:{}".format(public.format_date(times=stime))) mysql_backup_dir = self.get_backup_dir(db_find, args, "mysql") db_backup_dir = os.path.join(mysql_backup_dir, db_name) if not os.path.exists(db_backup_dir): os.makedirs(db_backup_dir) file_name = "{db_name}_{backup_time}_mysql_data".format( db_name=db_name, backup_time=public.format_date(self._BACKUP_TIME_FORMAT) ) backup_shell = "'{mysqldump_bin}' --opt --hex-blob --skip-lock-tables --single-transaction --routines --events --skip-triggers --default-character-set='{db_charset}' --force " \ "--host='{db_host}' --port={db_port} --user='{db_user}' --password='{db_password}' '{db_name}'".format( mysqldump_bin=self._MYSQLDUMP_BIN, db_charset=db_charset, db_user=db_user, db_password=db_password, db_host=db_host, db_port=db_port, db_name=db_name, ) if storage_type == "db": # 导出单个文件 file_name = file_name + ".sql.gz" backup_path = os.path.join(db_backup_dir, file_name) table_shell = "" if len(table_list) != 0: if table_list and isinstance(table_list[0], str): table_list = table_list[0].split(',') # print("正在备份表{}".format(table_list)) table_shell = "'" + "' '".join(table_list) + "'" backup_shell += " {table_shell} 2> '{err_log}' | gzip > '{backup_path}'".format(table_shell=table_shell, err_log=self._err_log, backup_path=backup_path) public.ExecShell(backup_shell, env={"MYSQL_PWD": db_password}) else: # 按表导出 export_dir = os.path.join(db_backup_dir, file_name) if not os.path.isdir(export_dir): os.makedirs(export_dir) for table_name in table_list: tb_backup_path = os.path.join(export_dir, "{table_name}.sql".format(table_name=table_name)) tb_shell = backup_shell + " '{table_name}' 2> '{err_log}' > '{tb_backup_path}'".format( table_name=table_name, err_log=self._err_log, tb_backup_path=tb_backup_path) public.ExecShell(tb_shell, env={"MYSQL_PWD": db_password}) backup_path = "{export_dir}.zip".format(export_dir=export_dir) public.ExecShell( "cd '{backup_dir}' && zip -m '{backup_path}' -r '{file_name}'".format(backup_dir=db_backup_dir, backup_path=backup_path, file_name=file_name)) if not os.path.exists(backup_path): public.ExecShell("rm -rf {}".format(export_dir)) if not os.path.exists(backup_path): error_msg = "Database backup failed!!" self.echo_error(error_msg) self.echo_info(public.readFile(self._err_log)) return False, error_msg # self.echo_info('backup_path:{}'.format(backup_path)) gz_size = os.path.getsize(backup_path) if gz_size < 400: error_msg = "Backup file size abnormal, backup failed!" self.echo_error(error_msg) self.echo_info(public.readFile(self._err_log)) os.remove(backup_path) return False, error_msg self.echo_info( "Database backup completed, taking {:.2f} seconds, compressed file size: {}".format(time.time() - stime, public.to_size( gz_size))) # self.check_disk_space(gz_size,self._MYSQL_BACKUP_DIR,type=1) return True, backup_path # mongodb 备份数据库 def mongodb_backup_database(self, db_find: dict, args: dict) -> Tuple[bool, str]: storage_type = args.get("storage_type", "db") # 备份的文件数量, 按照数据库 | 按照集合, 只支持整db file_type = db_find.get("file_type", "bson") collection_list = args.get("collection_list", []) # 备份的集合 field_list = args.get("field_list", []) from databaseModelV2.mongodbModel import panelMongoDB db_name = db_find["name"] db_host = "127.0.0.1" db_user = db_find.get("username") db_password = db_find.get("password") conn_data = {} auth_enabled = False if db_find["db_type"] == 0: # 本地 db_port = int(panelMongoDB.get_config_options("net", "port", 27017)) auth_enabled = panelMongoDB.get_auth_status() if auth_enabled: if not db_user or not db_password: err = public.lang("MongoDB has enabled security authentication. " "The database password cannot be empty. " "Please set the password and try again!") self.echo_error(err) return False, err elif db_find["db_type"] == 1: # 远程 auth_enabled = True if not db_password: error_msg = "The database password is empty! Please set the database password first!" self.echo_error(error_msg) return False, error_msg # 远程数据库 conn_config = json.loads(db_find["conn_config"]) db_host = conn_config["db_host"] db_port = int(conn_config["db_port"]) conn_data["host"] = conn_config["db_host"] conn_data["port"] = conn_config["db_port"] conn_data["username"] = conn_config["db_user"] conn_data["password"] = conn_config["db_password"] elif db_find["db_type"] == 2: # 服务器 auth_enabled = True if not db_password: error_msg = "The database password is empty! Please set the database password first!" self.echo_error(error_msg) return False, error_msg conn_config = public.M("database_servers").where("id=? AND LOWER(db_type)=LOWER('mongodb')", db_find["sid"]).find() db_host = conn_config["db_host"] db_port = int(conn_config["db_port"]) conn_data["host"] = conn_config["db_host"] conn_data["port"] = conn_config["db_port"] conn_data["username"] = conn_config["db_user"] conn_data["password"] = conn_config["db_password"] else: error_msg = "Unknown database type!" self.echo_error(error_msg) return False, error_msg mongodb_obj = panelMongoDB().set_host(**conn_data) status, err_msg = mongodb_obj.connect() if status is False: error_msg = "Failed to connect to database [{}: {}]".format(db_host, db_port) self.echo_error(error_msg) return False, error_msg db_collections = 0 db_storage_size = 0 try: status, db_obj = mongodb_obj.get_db_obj_new(db_name) if status is False: error_msg = db_obj self.echo_error(error_msg) return False, error_msg data = db_obj.command("dbStats") db_collections = data.get("collections", 0) # 获取集合数 db_storage_size = data.get("storageSize", 0) # 获取存储大小 # if len(collection_list) == 0: # collection_list = db_obj.list_collection_names() except: error_msg = "Failed to connect to database [{}: {}]".format(db_host, db_port) self.echo_error(error_msg) return False, error_msg if db_collections == 0: error_msg = "The specified database '{}' does not have any collections!".format(db_name) self.echo_error(error_msg) return False, error_msg self.echo_info("Backup MongoDB database: {}".format(db_name)) self.echo_info("Database size: {}".format(public.to_size(db_storage_size))) self.echo_info("Number of database collections: {}".format(db_collections)) check_status, check_msg = self.avilable_disk(db_storage_size, self._MONGODB_BACKUP_DIR) if check_status is False: return check_status, check_msg stime = time.time() self.echo_info("Start exporting database:{}".format(public.format_date(times=stime))) mongodb_backup_dir = self.get_backup_dir(db_find, args, "mongodb") db_backup_dir = os.path.join(mongodb_backup_dir, db_name) if not os.path.exists(db_backup_dir): os.makedirs(db_backup_dir) file_name = "{db_name}_{file_type}_{backup_time}_mongodb_data".format( db_name=db_find["name"], file_type=file_type, backup_time=public.format_date(self._BACKUP_TIME_FORMAT) ) export_dir = os.path.join(db_backup_dir, file_name) mongodump_shell = "'{mongodump_bin}' --host='{db_host}' --port={db_port} --db='{db_name}' --out='{out}' 2> '{err_log}'".format( mongodump_bin=self._MONGODBDUMP_BIN, db_host=db_host, db_port=db_port, db_name=db_name, out=export_dir, err_log=self._err_log, ) mongoexport_shell = "'{mongoexport_bin}' --host='{db_host}' --port={db_port} --db='{db_name}' 2> '{err_log}'".format( mongoexport_bin=self._MONGOEXPORT_BIN, db_host=db_host, db_port=db_port, db_name=db_name, err_log=self._err_log, ) if auth_enabled: auth_str = (f" --username='{db_user}'" f" --password={public.shell_quote(str(db_password.strip()))}" f" --authenticationDatabase='{db_name}'") mongodump_shell += auth_str mongoexport_shell += auth_str if file_type == "bson": if len(collection_list) == 0: public.ExecShell(mongodump_shell) else: for collection_name in collection_list: shell = f"{mongodump_shell} --collection='{collection_name}'" public.ExecShell(shell) else: # 导出 json csv 格式 fields = None if file_type == "csv": # csv fields = "--fields='{}'".format(",".join(field_list)) for collection_name in collection_list: file_path = os.path.join(export_dir, "{collection_name}.{file_type}".format(collection_name=collection_name, file_type=file_type)) shell = "{mongoexport_shell} --collection='{collection}' --type='{type}' --out='{out}'".format( mongoexport_shell=mongoexport_shell, collection=collection_name, type=file_type, out=file_path, ) if fields is not None: shell += " --fields='{fields}'".format(fields=fields) public.ExecShell(shell) backup_path = "{export_dir}.zip".format(export_dir=export_dir) public.ExecShell("cd {backup_dir} && zip -m {backup_path} -r {file_name}".format( backup_dir=db_backup_dir, backup_path=backup_path, file_name=file_name) ) if not os.path.exists(backup_path): public.ExecShell("rm -rf {}", format(export_dir)) error_msg = "Database backup failed!" self.echo_error(error_msg) self.echo_info(public.readFile(self._err_log)) return False, error_msg zip_size = os.path.getsize(backup_path) self.echo_info( "Database backup completed, taking {:.2f} seconds, compressed file size: {}".format( time.time() - stime, public.to_size(zip_size) ) ) return True, backup_path # redis 备份数据库 def redis_backup_database(self, db_find: dict, args: dict) -> Tuple[bool, str]: if db_find["db_type"] != 0: error_msg = 'We currently do not support backing up remote databases!' self.echo_error(error_msg) return False, error_msg from databaseModelV2.redisModel import panelRedisDB redis_obj = panelRedisDB() redis_config = redis_obj.get_options(None) db_num = redis_config.get("databases", 16) for db_idx in range(0, int(db_num)): if redis_obj.redis_conn(db_idx) is False: error_msg = "Failed to connect to local Redis database" self.echo_error(error_msg) return False, error_msg redis_obj.redis_conn(db_idx).save() redis_obj_t = redis_obj.redis_conn(0) if redis_obj_t is False: error_msg = "Failed to connect to local Redis database" self.echo_error(error_msg) return False, error_msg redis_dir = redis_obj_t.config_get().get("dir", "") src_path = os.path.join(redis_dir, "dump.rdb") if not os.path.exists(src_path): error_msg = 'The database file does not exist! {}'.format(src_path) self.echo_error(error_msg) return False, error_msg file_name = "{sid}_{backup_time}_redis_data.rdb".format( sid=db_find["id"], backup_time=public.format_date(self._BACKUP_TIME_FORMAT) ) # 调用 get_backup_dir 函数来获取备份目录的路径 redis_backup_dir = self.get_backup_dir(db_find, args, "redis") # 使用获取的路径来构建备份文件的路径 backup_path = os.path.join(redis_backup_dir, file_name) backup_dir = os.path.dirname(backup_path) if not os.path.exists(backup_dir): os.makedirs(backup_dir) db_size = os.path.getsize(src_path) self.echo_info('Backup Redis database') self.echo_info("Backup file size: {}".format(public.to_size(db_size))) self.echo_info("Backup path: {}".format(src_path)) check_status, check_msg = self.avilable_disk(db_size, self._REDIS_BACKUP_DIR) if check_status is False: return check_status, check_msg stime = time.time() self.echo_info("Start backing up the database:{}".format(public.format_date(times=stime))) import shutil shutil.copyfile(src_path, backup_path) if not os.path.exists(backup_path): error_msg = "Database backup failed!" self.echo_error(error_msg) self.echo_info(public.readFile(self._err_log)) return False, error_msg backup_size = os.path.getsize(backup_path) self.echo_info("Database backup completed, taking {:.2f} seconds, file size: {}".format(time.time() - stime, public.to_size( backup_size))) return True, backup_path # pgsql 备份数据库 def pgsql_backup_database(self, db_find: dict, args: dict) -> Tuple[bool, str]: from databaseModelV2.pgsqlModel import panelPgsql # storage_type = args.get("storage_type", "db") # 备份的文件数量, 按照数据库 | 按照表, 只支持整db table_list = args.get("table_list", []) # 备份的集合 db_name = db_find["name"] db_user = "postgres" db_host = "127.0.0.1" if db_find["db_type"] == 0: db_port = panelPgsql.get_config_options("port", int, 5432) t_path = os.path.join(public.get_panel_path(), "data/postgresAS.json") if not os.path.isfile(t_path): error_msg = "Administrator password not set!" self.echo_error(error_msg) return False, error_msg db_password = json.loads(public.readFile(t_path)).get("password", "") if not db_password: error_msg = "The database password is empty! Please set the database password first!" self.echo_error(error_msg) return False, error_msg elif db_find["db_type"] == 1: # 远程数据库 conn_config = json.loads(db_find["conn_config"]) db_host = conn_config["db_host"] db_port = conn_config["db_port"] db_user = conn_config["db_user"] db_password = conn_config["db_password"] elif db_find["db_type"] == 2: conn_config = public.M("database_servers").where("id=? AND LOWER(db_type)=LOWER('pgsql')", db_find["sid"]).find() db_host = conn_config["db_host"] db_port = conn_config["db_port"] db_user = conn_config["db_user"] db_password = conn_config["db_password"] else: error_msg = "Unknown database type" self.echo_error(error_msg) return False, error_msg pgsql_obj = panelPgsql().set_host(host=db_host, port=db_port, database=None, user=db_user, password=db_password) status, err_msg = pgsql_obj.connect() if status is False: error_msg = "Failed to connect to database [{}: {}]".format(db_host, int(db_port)) self.echo_error(error_msg) return False, error_msg db_size = 0 db_data = pgsql_obj.query("SELECT pg_database_size('{}') AS database_size;".format(db_name)) if isinstance(db_data, list) and len(db_data) != 0: db_size = db_data[0][0] if db_size == 0: error_msg = "The specified database '{}' has no data!".format(db_name) self.echo_error(error_msg) return False, error_msg if "ALL" in table_list: tb_l = pgsql_obj.query("SELECT tablename FROM pg_tables WHERE schemaname = 'public';") if isinstance(tb_l, list) and tb_l: table_list = [i[0] for i in tb_l] self.echo_info('Backup PgSQL database: {}'.format(db_name)) self.echo_info("Database size: {}".format(public.to_size(db_size))) check_status, check_msg = self.avilable_disk(db_size, self._PGSQL_BACKUP_DIR) if check_status is False: return check_status, check_msg stime = time.time() self.echo_info("Start exporting database:{}".format(public.format_date(times=stime))) # 调用 get_backup_dir 函数来获取备份目录的路径 pgsql_backup_dir = self.get_backup_dir(db_find, args, "pgsql") # 使用获取的路径来构建备份文件的路径 db_backup_dir = os.path.join(pgsql_backup_dir, db_name) if not os.path.exists(db_backup_dir): os.makedirs(db_backup_dir) file_name = "{db_name}_{backup_time}_pgsql_data".format( db_name=db_name, backup_time=public.format_date(self._BACKUP_TIME_FORMAT) ) shell = "'{pgdump_bin}' --host='{db_host}' --port={db_port} --username='{db_user}' --dbname='{db_name}' --clean".format( pgdump_bin=self._PGDUMP_BIN, db_host=db_host, db_port=int(db_port), db_user=db_user, db_name=db_name, ) file_name = file_name + ".sql.gz" backup_path = os.path.join(db_backup_dir, file_name) table_shell = "" if len(table_list) != 0: table_shell = "--table='" + "' --table='".join(table_list) + "'" shell += " {table_shell} | gzip > '{backup_path}'".format(table_shell=table_shell, backup_path=backup_path) public.ExecShell(shell, env={"PGPASSWORD": db_password}) if not os.path.exists(backup_path): error_msg = "Database backup failed!" self.echo_error(error_msg) self.echo_info(public.readFile(self._err_log)) return False, error_msg gz_size = os.path.getsize(backup_path) self.echo_info( "Database backup completed, taking {:.2f} seconds, compressed file size: {}".format( time.time() - stime, public.to_size(gz_size) ) ) return True, backup_path # =========================== Backup Object End ================================= def generate_success_title(self, task_name): from send_mail import send_mail sm = send_mail() now = public.format_date(format="%Y-%m-%d %H:%M") server_ip = sm.GetLocalIp() title = "{}-{} Task execution successful".format(server_ip, task_name) return title def generate_failture_title(self, task_name): title = "yakpanel Plan Task Backup Failure Reminder".format(task_name) return title def generate_all_failture_notice(self, task_name, msg, backup_type, remark=""): from send_mail import send_mail sm = send_mail() now = public.format_date(format="%Y-%m-%d %H:%M:%S") server_ip = sm.GetLocalIp() if remark: remark = "\n* Task note: {}".format(remark) notice_content = """####Planned task execution failed: * Server IP: {} * time: {} * Plan task name:{}{} * The following is a list of {} backup failures:{} Please handle it as soon as possible""".format( server_ip, now, task_name, remark, backup_type, msg) return notice_content def generate_failture_notice(self, task_name, msg, remark): from send_mail import send_mail sm = send_mail() now = public.format_date(format="%Y-%m-%d %H:%M:%S") server_ip = sm.GetLocalIp() if remark: remark = "\n* Task note: {}".format(remark) notice_content = """####Planned task execution failed: * Server IP: {} * time: {} * Plan task name:{}{} * Error message: {} Please handle it as soon as possible""".format( server_ip, now, task_name, remark, msg) return notice_content def generate_disk_notice(self, task_name, free_space_gb, remark=""): from send_mail import send_mail sm = send_mail() now = public.format_date(format="%Y-%m-%d %H:%M:%S") server_ip = sm.GetLocalIp() if remark: remark = "\n* Task note: {}".format(remark) notice_content = """####Warm reminder for backup files of planned tasks: * Server IP: {} * time: {} * Plan task name: {}{} * reminder: We have detected that your server's disk space is insufficient {} G, which may cause subsequent backup file failures. Please clean up the disk as soon as possible! """.format( server_ip, now, task_name, remark, free_space_gb) return notice_content def get_cron_info(self, cron_name): """ 通过计划任务名称查找计划任务配置参数 """ try: cron_info = public.M('crontab').where('echo=?', (cron_name,)) \ .field('name,save_local,notice,notice_channel,id,sType').find() return cron_info except Exception as e: pass return {} def send_success_notification(self, msg, target="", remark=""): pass def send_disk_notification(self, free_space_gb, remark=""): """发送任务失败消息 :error_msg 错误信息 :remark 备注 """ if not self.cron_info: return cron_info = self.cron_info cron_title = cron_info["name"] notice = cron_info["notice"] notice_channel = cron_info["notice_channel"] if notice == 0 or not notice_channel: return if notice == 1 or notice == 2: title = self.generate_failture_title(cron_title) task_name = cron_title msg = self.generate_disk_notice(task_name, free_space_gb, remark) res = self.send_notification(notice_channel, title, msg) if res: self.echo_info("The message notification has been sent.") def send_failture_notification(self, error_msg, target="", remark=""): """发送任务失败消息 :error_msg 错误信息 :remark 备注 """ if self._backup_all: return if not self.cron_info: return cron_info = self.cron_info cron_title = cron_info["name"] save_local = cron_info["save_local"] notice = cron_info["notice"] notice_channel = cron_info["notice_channel"] self.save_backup_status(False, target, msg=error_msg) if notice == 0 or not notice_channel: return if notice == 1 or notice == 2: title = self.generate_failture_title(cron_title) task_name = cron_title msg = self.generate_failture_notice(task_name, error_msg, remark) res = self.send_notification(notice_channel, title, msg) if res: self.echo_info("The message notification has been sent.") def send_all_failture_notification(self, backup_type, results, remark=""): """统一发送任务失败消息 :results [(备份对象, 备份结果,错误信息),...] :remark 备注 """ if not self.cron_info: return cron_info = self.cron_info cron_title = cron_info["name"] save_local = cron_info["save_local"] notice = cron_info["notice"] notice_channel = cron_info["notice_channel"] if notice == 0 or not notice_channel: return if notice == 1 or notice == 2: title = self.generate_failture_title(cron_title) type_desc = { "site": "site", "database": "database" } backup_type_desc = type_desc[backup_type] task_name = cron_title failture_count = 0 total = 0 content = "" for obj in results: total += 1 obj_name = obj[0] result = obj[1] if not result: failture_count += 1 # content += "{}".format(obj_name) content += "\n{}".format(obj_name) if failture_count > 0: if self._cloud: remark = "Backup to {}, total {} {}, failed {}.".format( self._cloud._title, total, backup_type_desc, failture_count) else: remark = "Backup failed for {}/total of {} sites".format( failture_count, total, backup_type_desc) msg = self.generate_all_failture_notice(task_name, content, backup_type_desc, remark) res = self.send_notification(notice_channel, title, msg=msg, total=total, failture_count=failture_count) if res: self.echo_info("The message notification has been sent.") else: self.echo_error("Message notification sending failed.") def send_notification(self, channel, title, msg="", total=0, failture_count=0, cron_info=None): """发送通知 Args: channel (str): 消息通道,多个用英文逗号隔开 title (str): 通知标题 msg (str, optional): 消息内容. Defaults to "". Returns: bool: 通知是否发送成功 """ try: from config import config from panelPush import panelPush tongdao = [] if channel.find(",") >= 0: tongdao = channel.split(",") else: tongdao = [channel] error_count = 0 con_obj = config() get = public.dict_obj() msg_channels = con_obj.get_msg_configs(get) error_channel = [] channel_data = {} msg_data = {} for ch in tongdao: # 根据不同的消息通道准备不同的内容 if ch == "mail": msg_data = { "msg": msg.replace("\n", "
"), "title": title } if ch in ["dingding", "weixin", "feishu", "wx_account", "tg"]: msg_data["msg"] = msg # print("msg",msg_data["msg"]) if ch in ["sms"]: if not cron_info: task_name = self.cron_info["name"] else: task_name = cron_info["name"] if total > 0 and failture_count > 0: msg_data["sm_type"] = "backup_all" msg_data["sm_args"] = { "panel_name": public.GetConfigValue('title'), "task_name": task_name, "failed_count": failture_count, "total": total } else: msg_data["sm_type"] = "backup" msg_data["sm_args"] = { "panel_name": public.GetConfigValue('title'), "task_name": task_name } channel_data[ch] = msg_data # print("channel data:") # print(channel_data) # 即时推送 pp = panelPush() push_res = pp.push_message_immediately(channel_data) if push_res["status"]: channel_res = push_res["msg"] for ch, res in channel_res.items(): if not res["status"]: if ch in msg_channels: error_channel.append(msg_channels[ch]["title"]) error_count += 1 if not push_res["status"] or error_count: self.echo_error("Message channel: {} failed to send!".format(",".join(error_channel))) else: self.echo_info("Message sent successfully.") if error_count == len(tongdao): return False return True except Exception as e: import traceback print(traceback.format_exc()) return False def send_notification2(self, channel, title, msg=""): try: from send_mail import send_mail tongdao = [] if channel.find(",") >= 0: tongdao = channel.split(",") else: tongdao = [channel] sm = send_mail() send_res = [] error_count = 0 channel_names = { "mail": "mail", "dingidng": "dingidng" } error_channel = [] settings = sm.get_settings() for td in tongdao: _res = False if td == "mail": if len(settings["user_mail"]['mail_list']) == 0: continue mail_list = settings['user_mail']['mail_list'] if len(mail_list) == 1: mail_list = mail_list[0] _res = sm.qq_smtp_send(mail_list, title=title, body=msg.replace("\n", "
")) if not _res: error_count += 1 error_channel.append(channel_names[td]) if td == "dingding": if len(settings["dingding"]['info']) == 0: continue _res = sm.dingding_send(msg) send_res.append(_res) if not _res: error_count += 1 error_channel.append(channel_names[td]) if error_count > 0: print("Message channel: {} failed to send!".format(",".join(error_channel))) if error_count == len(tongdao): return False return True except Exception as e: print(e) return False def save_backup_status(self, status, target="", msg=""): """保存备份的状态""" try: if not self.cron_info: return cron_id = self.cron_info["id"] sql = public.M("system").dbfile("system").table("backup_status") sql.add("id,target,status,msg,addtime", (cron_id, target, status, msg, time.time(),)) except Exception as e: print("Save backup status exception: {}.".format(e)) # 切割文件上传 def split_upload_file(self, backup_path: str, upload_path: str) -> Tuple[bool, str]: is_status = True data = "" self.echo_info("File splitting in progress, please wait...") if self._cloud is None or self._cloud_new is None: return False, "No cloud storage specified!" if self.cron_info.get("split_type") == "size": is_status, data = self.split_file( file_path=backup_path, split_size=self.cron_info.get("split_value"), ) elif self.cron_info.get("split_type") == "num": is_status, data = self.split_file( file_path=backup_path, split_num=self.cron_info.get("split_value"), ) if is_status is False: return False, data save_dir = data["save_dir"] upload_dir = os.path.join(self._cloud_new.backup_path, upload_path, os.path.basename(save_dir)) self.echo_info( "File splitting is in progress. The size of each individual file after splitting is {} M, and the number of files after splitting is {}".format( data.get("split_size"), data.get("split_num"))) join_msg = """Split file recovery method: 1. Download cloud storage directory: {upload_dir} On the panel 2. Enter the directory and click on 'File Merge'""".format( upload_dir=upload_dir ) self.echo_info(join_msg) # upload_path = os.path.join(upload_path, os.path.basename(save_dir)) for name in os.listdir(save_dir): path = os.path.join(save_dir, name) if not os.path.isfile(path): continue upload_path_temp = os.path.join(upload_dir, name) if not self._cloud_new.cloud_upload_file(path, upload_path_temp): public.ExecShell("chattr -i -R {split_dir}".format(split_dir=save_dir)) public.ExecShell("rm -rf {}".format(save_dir)) error_msg = "Backup task execution failed." if hasattr(self._cloud, "error_msg"): if self._cloud.error_msg: error_msg = self._cloud.error_msg self.echo_error(error_msg) if os.path.exists(backup_path): os.remove(backup_path) return False, error_msg self.echo_info("Successfully uploaded to {}".format(self._cloud._title)) public.ExecShell("chattr -i -R {split_dir}".format(split_dir=save_dir)) public.ExecShell("rm -rf {}".format(save_dir)) return True, upload_dir def split_file(self, file_path: str, # 文件路径 split_size: int = None, # 切割大小 MB split_num: int = None, # 切割数量 split_file_name: str = None, # 切割文件名,默认为切割目标文件名 split_file_ext: str = None, # 切割文件后缀 save_path: str = None, # 切割后保存的路径 ) -> Tuple[bool, Union[str, dict]]: import math if not os.path.isfile(file_path): return False, "The file does not exist!" totoal_size = os.path.getsize(file_path) split_file_option = "" totoal_split_size = math.ceil(totoal_size / 1024 / 1024) if split_size: # 默认使用大小 split_file_option = "--bytes={}M".format(split_size) split_num = math.ceil(totoal_split_size / split_size) elif split_num: # 按照数量拆分 split_file_option = "--number={}".format(split_num) split_size = round(totoal_split_size / split_num) else: # 默认按照大小拆分 100M split_size = 100 split_file_option = "--bytes={}M".format(split_size) split_num = math.ceil(totoal_split_size / split_size) if split_num < 2: return False, "The minimum number of file splits is 2!" file_name = os.path.basename(file_path) if not save_path: # 保存路径 save_path = os.path.dirname(file_path) save_dir_name = "{file_name}_split".format(file_name=file_name) save_dir = os.path.join(save_path, save_dir_name) i = 1 while os.path.isdir(save_dir): save_dir_name = "{file_name}_split-({ectype})".format(file_name=file_name, ectype=i) save_dir = os.path.join(save_path, save_dir_name) i += 1 os.makedirs(save_dir) if not split_file_name: # 切割文件名,默认为切割目标文件名 file_ext_temp = file_name.split(".") if len(file_ext_temp) != 0: split_file_name = "{file_name}_sqlit_".format(file_name=".".join(file_ext_temp[:-1])) else: split_file_name = "{file_name}_sqlit_".format(file_name=file_name) if not split_file_ext: split_file_ext = ".bt_split" split_file_shell = "--additional-suffix={split_file_ext}".format(split_file_ext=split_file_ext) shell = "cd '{save_dir}' && split {split_file_option} --numeric-suffixes=1 --suffix-length={split_length} {split_file_shell} '{file_path}' {split_file_name}".format( save_dir=save_dir, split_file_option=split_file_option, split_length=len(str(split_num)), file_path=file_path, split_file_name=split_file_name, split_file_shell=split_file_shell, ) public.ExecShell(shell) split_file_list = [] for name in os.listdir(save_dir): path = os.path.join(save_dir, name) if not os.path.isfile(path): continue split_file_list.append(name) split_config_info = { "name": file_name, "size": totoal_size, "split_size": split_size, "split_num": split_num, "md5": public.FileMd5(file_path), "split_file": split_file_list, "save_dir": save_dir, } split_config_path = os.path.join(save_dir, "split_config.bt_split_json") public.writeFile(split_config_path, json.dumps(split_config_info)) public.ExecShell("chattr +i -R {split_dir}".format(split_dir=save_dir)) return True, split_config_info def get_backup_dir(self, db_find: dict, args: dict, db_type: str) -> str: backup_dir = args.get("db_backup_path", "") save_local = args.get("save_local", "") if db_find["db_type"] == 0 or save_local == 1: # 本地数据库 if backup_dir: db_backup_dir = os.path.join(backup_dir, "database") specific_db_backup_dir = os.path.join(db_backup_dir, db_type, "crontab_backup") else: specific_db_backup_dir = os.path.join(self._DB_BACKUP_DIR, db_type, "crontab_backup") else: specific_db_backup_dir = os.path.join(self._DB_BACKUP_DIR, db_type, "crontab_backup") return specific_db_backup_dir def get_site_backup_dir(self, backupto: str, db_backup_path: str, site_name: str) -> str: site_backup_dir = self._BACKUP_DIR # 给 site_backup_dir 一个默认值 if backupto == "localhost" or backupto == "Local Disk": if db_backup_path: site_backup_dir = db_backup_path return site_backup_dir def check_disk_space(self, file_size, backup_dir): min_required_space = max(file_size * 5, 1 * 1024 * 1024 * 1024) # 1GB disk_usage = public.get_disk_usage(backup_dir) if disk_usage.free < min_required_space: free_space_gb = int(min_required_space / (1024 * 1024 * 1024)) self.send_disk_notification(free_space_gb) def check_databases(self): """检查数据表是否存在""" import sqlite3 db_path = "/www/server/panel/data/system.db" fixed_path = f"{db_path}.fixed" try: tables = ["backup_status"] with sqlite3.connect(db_path) as conn: cur = conn.cursor() table_key = ",".join([f"'{t}'" for t in tables]) cur.execute(f"SELECT name FROM sqlite_master WHERE type='table' AND name IN ({table_key})") exists_dbs = [row[0] for row in cur.fetchall()] if "backup_status" not in exists_dbs: cur.execute(''' CREATE TABLE `backup_status` ( `id` INTEGER PRIMARY KEY, `target` TEXT, `status` INTEGER, `msg` TEXT DEFAULT "", `addtime` DATETIME DEFAULT CURRENT_TIMESTAMP ) ''') return True except sqlite3.DatabaseError as e: public.print_log("System.db database is damaged, try repairing it!") # 尝试修复数据库 ok = self._repair_database(db_path, fixed_path) if not ok: # 修复失败跳过 public.print_log("System.db database repair failed!") return False except Exception: return False def _repair_database(self, db_path, fixed_path): """修复损坏的SQLite数据库,失败时直接删除原文件""" try: # 检查数据库文件是否存在 if not os.path.exists(db_path): public.print_log(f"Error: Database file does not exist- {db_path}") return False # 执行修复命令 repair_cmd = f"sqlite3 {db_path} '.recover' | sqlite3 {fixed_path}" if os.system(repair_cmd) == 0 and os.path.exists(fixed_path): # 验证修复后的数据库 check_cmd = f"sqlite3 {fixed_path} 'PRAGMA integrity_check;'" import subprocess result = subprocess.getoutput(check_cmd) if "ok" in result.lower(): os.replace(fixed_path, db_path) public.print_log("Database repair successful!") return self.check_databases() else: public.print_log(f"The repaired database still has issues:: {result}") raise Exception("Repair validation failed") # 修复失败,原数据库文件删除后重建 public.print_log(f"Monitoring data is damaged, will be rebuilt after deletion: {db_path}") os.remove(db_path) public.print_log(f"Monitoring data repair failed, file has been deleted: {db_path}") return False except Exception as e: public.print_log(f"An error occurred during the repair process: {str(e)}") # 原数据库文件删除 if os.path.exists(db_path): os.remove(db_path) public.print_log(f"Monitoring data repair failed, file has been deleted: {db_path}") return False