Files
yakpanel-core/class_v2/panel_restore_v2.py

229 lines
11 KiB
Python
Raw Permalink Normal View History

2026-04-07 02:04:22 +05:30
# coding: utf-8
# -------------------------------------------------------------------
# YakPanel
# -------------------------------------------------------------------
# Copyright (c) 2015-2099 YakPanel(www.yakpanel.com) All rights reserved.
# -------------------------------------------------------------------
# Author: zhwwen <zhw@yakpanel.com>
# -------------------------------------------------------------------
#
# ------------------------------
# 网站恢复
# ------------------------------
import public,os,files,sys
from time import sleep
from public.validate import Param
class panel_restore:
_local_file = '/tmp/{}'
_progress_file = '/tmp/restore_site.log'
# def __init__(self):
# # 清空日志文件
def _progress_rewrite(self,content,mothed='a+'):
sleep(2)
public.writeFile(self._progress_file,content+'\n',mothed)
def _get_local_backup_path(self):
local_backdir = public.M('config').field('backup_path').find()['backup_path']
return local_backdir
def _build_aws_backup_path(self,btype,file_name,domain):
config_file = "/www/server/panel/plugin/aws_s3/config.conf"
conf = public.readFile(config_file)
backup_path = conf.split('|')[-1].strip()+btype+'/'+ domain + '/' + file_name
return backup_path
def _build_google_backup_path(self,btype,file_name,domain):
object_name = 'bt_backup/{}/{}/{}'.format(btype,domain,file_name)
return object_name
def _get_backfile_method(self,filename):
backup_info = public.M('backup').where("name=?", (filename,)).getField('filename')
backup_info = backup_info.split('|')
if len(backup_info) >= 3:
method = backup_info[1]
else:
method = 'local'
return method
def _remove_old_website_file_to_trush(self,args):
# 将原来目录移至回收站
files.files().DeleteDir(args)
def _get_website_info(self,site_id):
site_name = public.M('sites').where("id=?",(site_id,)).getField('name')
site_path = public.M('sites').where("id=?",(site_id,)).getField('path')
return {'site_name':site_name,'site_path':site_path}
def _restore_backup(self,local_backup_file_path,site_info,args):
# 判断备份文件是否存在,如果不存在继续检查是否远程备份
if not os.path.exists(local_backup_file_path):
self._progress_rewrite('No backup file found: {}'.format(str(local_backup_file_path)))
return public.return_message(-1, 0, public.lang("Panel does not find the backup file: {}",local_backup_file_path))
# 将网站目录移至回收站
self._progress_rewrite('Move the current website directory to the recycle bin: {}'.format(str(args.path)))
self._remove_old_website_file_to_trush(args)
if not os.path.exists(args.path):
self._progress_rewrite('Create an empty directory for the site: {}'.format(str(args.path)))
os.makedirs(site_info['site_path'])
if 'zip' in args.file_name:
uncompress_comand = 'unzip'
else:
uncompress_comand = 'tar -zxvf'
self._progress_rewrite('The decompression command is: {}'.format(str(uncompress_comand)))
self._progress_rewrite('Start to restore data......')
public.ExecShell('cd {} && {} {} >> /tmp/restore_site.log'.format(site_info['site_path'], uncompress_comand, local_backup_file_path))
if len(os.listdir(site_info['site_path'])) == 2:
public.ExecShell('cd {s} && mv {s}/{d}/{{*,.*}} .'.format(s=site_info['site_path'],d=site_info['site_name']))
public.ExecShell('cd {s} && rmdir {d}'.format(s=site_info['site_path'],d=site_info['site_name']))
# 将文件全新设置为644文件夹设置为755
self._progress_rewrite('Setting site permissions......')
files.files().fix_permissions(args)
def _download_aws_file(self,args,btype='site'):
sys.path.append('/www/server/panel/plugin/aws_s3')
import aws_s3_main
aws3 = aws_s3_main.aws_s3_main()
self._progress_rewrite('Building S3 download path...')
download_file = self._build_aws_backup_path(btype,args.file_name,args.obj_name)
self._progress_rewrite('The download path is:{}'.format(download_file))
self._local_file = self._local_file.format(args.file_name)
self._progress_rewrite('Backup file will be downloaded to:{}'.format(self._local_file))
self._progress_rewrite('Starting to download file:{}'.format(self._local_file))
args.object_name = download_file
args.local_file = self._local_file
aws3.download_file(args)
self._progress_rewrite('Download completed:{}'.format(self._local_file))
return self._local_file
def _download_google_cloud_file(self,args,btype='site'):
sys.path.append('/www/server/panel/plugin/gcloud_storage')
import gcloud_storage_main
gs = gcloud_storage_main.gcloud_storage_main()
self._progress_rewrite('Building Google Store download path...')
download_file = self._build_google_backup_path(btype,args.file_name,args.obj_name)
self._progress_rewrite('The download path is:{}'.format(download_file))
self._local_file = self._local_file.format(args.file_name)
self._progress_rewrite('Backup file will be downloaded to:{}'.format(self._local_file))
self._progress_rewrite('Starting to download file:{}'.format(self._local_file))
args.source_blob_name = download_file
args.destination_file_name = self._local_file
gs.download_blob(args)
self._progress_rewrite('Download completed:{}'.format(self._local_file))
return self._local_file
def _download_google_drive_file(self,args):
sys.path.append('/www/server/panel/plugin/gdrive')
import gdrive_main
gd = gdrive_main.gdrive_main()
self._local_file = self._local_file.format(args.file_name)
self._progress_rewrite('Backup file will be downloaded to:{}'.format(self._local_file))
self._progress_rewrite('Starting to download file:{}'.format(self._local_file))
gd.download_file(args.file_name)
self._progress_rewrite('Download completed:{}'.format(self._local_file))
return self._local_file
def restore_website_backup(self,args):
"""
@name 恢复站点文件
@author zhwen<zhw@yakpanel.com>
@parma file_name 备份得文件名
@parma site_id 网站id
"""
# 校验参数
try:
args.validate([
Param('file_name').String(),
Param('site_id').Integer(),
], [
public.validate.trim_filter(),
])
except Exception as ex:
public.print_log("error info: {}".format(ex))
return public.return_message(-1, 0, str(ex))
self._progress_rewrite('','w')
site_info = self._get_website_info(args.site_id)
self._progress_rewrite('Get site information:{}'.format(str(site_info)))
args.path = site_info['site_path']
args.obj_name = site_info['site_name']
self._progress_rewrite('Get the site path:{}'.format(str(site_info['site_path'])))
local_backup_path = self._get_local_backup_path()
local_backup_file_path = local_backup_path +'/site/'+ args.file_name
if not os.path.exists(local_backup_file_path):
local_backup_file_path = local_backup_path +'/site/'+ site_info['site_name']+'/'+args.file_name
self._progress_rewrite('Get the local backup file path: {}'.format(str(local_backup_path)))
backup_method = self._get_backfile_method(args.file_name)
self._progress_rewrite('Get the backup method: {}'.format(str(backup_method)))
if backup_method == 'local':
self._progress_rewrite('Start to restore local backup files: {}'.format(str(local_backup_file_path)))
result = self._restore_backup(local_backup_file_path,site_info,args)
if result:
self._progress_rewrite('Recovery failed: {}'.format(str(site_info['site_path'])))
return result
elif backup_method == 'aws_s3':
self._download_aws_file(args)
result = self._restore_backup(self._local_file, site_info, args)
elif backup_method == 'Google Cloud':
self._download_google_cloud_file(args)
result = self._restore_backup(self._local_file, site_info, args)
elif backup_method == 'Google Drive':
self._download_google_drive_file(args)
result = self._restore_backup(self._local_file, site_info, args)
else:
return public.return_msg_gettext(False,'Currently only supports restoring local, Google storage and AWS S3 backups')
if os.path.exists(self._local_file):
os.remove(self._local_file)
if result:
self._progress_rewrite('Recovery failed: {}'.format(str(site_info['site_path'])))
return result
self._progress_rewrite('Successful recovery: {}'.format(str(site_info['site_path'])))
return public.return_message(0, 0, public.lang("Restore Successful"))
# 取任务进度
def get_progress(self, get):
"""
@name 获取进度日志
@author zhwen<zhw@yakpanel.com>
"""
# result = public.GetNumLines(self._progress_file, 20)
result = public.ExecShell('tail -n 20 {}'.format(self._progress_file))[0]
if len(result) < 1:
return public.return_message(0, 0, public.lang("Wait for the restore to start"))
return public.return_message(0,0,result)
# 恢复数据库
def restore_db_backup(self,args):
"""
@name 恢复站点文件
@author zhwen<zhw@yakpanel.com>
@parma file_name 备份得文件名 /www/backup/database/db_test_com_20200817_112722.sql.gz|Google Drive|db_test_com_20200817_112722.sql.gz
@parma obj_name 数据库名
"""
if "|" not in args.file:
return public.returnMsg(True,'success')
try:
backup_info = args.file.split('|')
args.file_name = backup_info[-1]
args.obj_name = args.name
backup_method = backup_info[1]
self._progress_rewrite('','w')
self._progress_rewrite('Restoring database...')
self._progress_rewrite('Get the backup method: {}'.format(str(backup_method)))
if backup_method == 'aws_s3':
self._download_aws_file(args,'database')
elif backup_method == 'Google Cloud':
self._download_google_cloud_file(args,'database')
elif backup_method == 'Google Drive':
self._download_google_drive_file(args)
else:
return public.returnMsg(False,'Currently only supports restoring local, Google storage and AWS S3 backups')
public.ExecShell('mv {} {}/database'.format(self._local_file, self._get_local_backup_path()))
return public.returnMsg(True,'success')
except:
return public.returnMsg(False,"Download error!")