* Date: 2018-4-3 */ namespace app\admin\controller; use think\Db; use think\Backup; class Tools extends Base { public function _initialize() { parent::_initialize(); $this->language_access(); // 多语言功能操作权限 } /** * 数据表列表 */ public function index() { $dbtables = Db::query('SHOW TABLE STATUS'); $total = 0; $list = array(); foreach ($dbtables as $k => $v) { if (preg_match('/^'.PREFIX.'/i', $v['Name'])) { $v['size'] = format_bytes($v['Data_length'] + $v['Index_length']); $list[$k] = $v; $total += $v['Data_length'] + $v['Index_length']; } } $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); @unlink(realpath(trim($path, '/')) . DS . 'backup.lock'); // if (session('?backup_config.path')) { //备份完成,清空缓存 session('backup_tables', null); session('backup_file', null); session('backup_config', null); // } $this->assign('list', $list); $this->assign('total', format_bytes($total)); $this->assign('tableNum', count($list)); return $this->fetch(); } /** * 数据备份 */ public function export($tables = null, $id = null, $start = null,$optstep = 0) { //防止备份数据过程超时 function_exists('set_time_limit') && set_time_limit(0); @ini_set('memory_limit','-1'); /*升级完自动备份所有数据表*/ if ('all' == $tables) { $dbtables = Db::query('SHOW TABLE STATUS'); $list = array(); foreach ($dbtables as $k => $v) { if (preg_match('/^'.PREFIX.'/i', $v['Name'])) { $list[] = $v['Name']; } } $tables = $list; unlink(session('backup_config.path') . 'backup.lock'); } /*--end*/ if(IS_POST && !empty($tables) && is_array($tables) && empty($optstep)){ //初始化 $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); $path = trim($path, '/'); if(!empty($path) && !is_dir($path)){ mkdir($path, 0755, true); } //读取备份配置 $config = array( 'path' => realpath($path) . DS, 'part' => config('DATA_BACKUP_PART_SIZE'), 'compress' => config('DATA_BACKUP_COMPRESS'), 'level' => config('DATA_BACKUP_COMPRESS_LEVEL'), ); //检查是否有正在执行的任务 $lock = "{$config['path']}backup.lock"; if(is_file($lock)){ return json(array('info'=>'检测到有一个备份任务正在执行,请稍后再试!', 'status'=>0, 'url'=>'')); } else { //创建锁文件 file_put_contents($lock, $_SERVER['REQUEST_TIME']); } //检查备份目录是否可写 if(!is_writeable($config['path'])){ return json(array('info'=>'备份目录不存在或不可写,请检查后重试!', 'status'=>0, 'url'=>'')); } session('backup_config', $config); //生成备份文件信息 $file = array( 'name' => date('Ymd-His', $_SERVER['REQUEST_TIME']), 'part' => 1, 'version' => getCmsVersion(), ); session('backup_file', $file); //缓存要备份的表 session('backup_tables', $tables); //创建备份文件 $Database = new Backup($file, $config); if(false !== $Database->create()){ $speed = (floor((1/count($tables))*10000)/10000*100); $speed = sprintf("%.2f", $speed); $tab = array('id' => 0, 'start' => 0, 'speed'=>$speed, 'table'=>$tables[0], 'optstep'=>1); return json(array('tables' => $tables, 'tab' => $tab, 'info'=>'初始化成功!', 'status'=>1, 'url'=>'')); } else { return json(array('info'=>'初始化失败,备份文件创建失败!', 'status'=>0, 'url'=>'')); } } elseif (IS_POST && is_numeric($id) && is_numeric($start) && 1 == intval($optstep)) { //备份数据 $tables = session('backup_tables'); //备份指定表 $Database = new Backup(session('backup_file'), session('backup_config')); $start = $Database->backup($tables[$id], $start); if(false === $start){ //出错 return json(array('info'=>'备份出错!', 'status'=>0, 'url'=>'')); } elseif (0 === $start) { //下一表 if(isset($tables[++$id])){ $speed = (floor((($id+1)/count($tables))*10000)/10000*100); $speed = sprintf("%.2f", $speed); $tab = array('id' => $id, 'start' => 0, 'speed' => $speed, 'table'=>$tables[$id], 'optstep'=>1); return json(array('tab' => $tab, 'info'=>'备份完成!', 'status'=>1, 'url'=>'')); } else { //备份完成,清空缓存 /*自动覆盖安装目录下的eyoucms.sql*/ $install_time = DEFAULT_INSTALL_DATE; $constsant_path = APP_PATH.MODULE_NAME.'/conf/constant.php'; if (file_exists($constsant_path)) { require_once($constsant_path); defined('INSTALL_DATE') && $install_time = INSTALL_DATE; } $install_path = ROOT_PATH.'install'; if (!is_dir($install_path) || !file_exists($install_path)) { $install_path = ROOT_PATH.'install_'.$install_time; } if (is_dir($install_path) && file_exists($install_path)) { $srcfile = session('backup_config.path').session('backup_file.name').'-'.session('backup_file.part').'-'.session('backup_file.version').'.sql'; $dstfile = $install_path.'/eyoucms.sql'; if(@copy($srcfile, $dstfile)){ /*替换所有表的前缀为官方默认ey_,并重写安装数据包里*/ $eyouDbStr = file_get_contents($dstfile); $dbtables = Db::query('SHOW TABLE STATUS'); $tableName = $eyTableName = []; foreach ($dbtables as $k => $v) { if (preg_match('/^'.PREFIX.'/i', $v['Name'])) { $tableName[] = "`{$v['Name']}`"; $eyTableName[] = preg_replace('/^`'.PREFIX.'/i', '`ey_', "`{$v['Name']}`"); } } $eyouDbStr = str_replace($tableName, $eyTableName, $eyouDbStr); @file_put_contents($dstfile, $eyouDbStr); unset($eyouDbStr); /*--end*/ } else { @unlink($dstfile); // 复制失败就删掉,避免安装错误的数据包 } } /*--end*/ unlink(session('backup_config.path') . 'backup.lock'); session('backup_tables', null); session('backup_file', null); session('backup_config', null); return json(array('info'=>'备份完成!', 'status'=>1, 'url'=>'')); } } else { $rate = floor(100 * ($start[0] / $start[1])); $speed = floor((($id+1)/count($tables))*10000)/10000*100 + ($rate/100); $speed = sprintf("%.2f", $speed); $tab = array('id' => $id, 'start' => $start[0], 'speed' => $speed, 'table'=>$tables[$id], 'optstep'=>1); return json(array('tab' => $tab, 'info'=>"正在备份...({$rate}%)", 'status'=>1, 'url'=>'')); } } else {//出错 return json(array('info'=>'参数有误', 'tab'=>['speed'=>-1], 'status'=>0, 'url'=>'')); } } /** * 优化 */ public function optimize() { $batchFlag = input('get.batchFlag', 0, 'intval'); //批量删除 if ($batchFlag) { $table = input('key', array()); }else { $table[] = input('tablename' , ''); } if (empty($table)) { $this->error('请选择数据表'); } $strTable = implode(',', $table); if (!DB::query("OPTIMIZE TABLE {$strTable} ")) { $strTable = ''; } $this->success("操作成功" . $strTable, url('Tools/index')); } /** * 修复 */ public function repair() { $batchFlag = input('get.batchFlag', 0, 'intval'); //批量删除 if ($batchFlag) { $table = input('key', array()); }else { $table[] = input('tablename' , ''); } if (empty($table)) { $this->error('请选择数据表'); } $strTable = implode(',', $table); if (!DB::query("REPAIR TABLE {$strTable} ")) { $strTable = ''; } $this->success("操作成功" . $strTable, url('Tools/index')); } /** * 数据还原 */ public function restore() { $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); $path = trim($path, '/'); if(!empty($path) && !is_dir($path)){ mkdir($path, 0755, true); } $path = realpath($path); $flag = \FilesystemIterator::KEY_AS_FILENAME; $glob = new \FilesystemIterator($path, $flag); $list = array(); $filenum = $total = 0; foreach ($glob as $name => $file) { if(preg_match('/^\d{8,8}-\d{6,6}-\d+-v\d+\.\d+\.\d+(.*)\.sql(?:\.gz)?$/', $name)){ $name = sscanf($name, '%4s%2s%2s-%2s%2s%2s-%d-%s'); $date = "{$name[0]}-{$name[1]}-{$name[2]}"; $time = "{$name[3]}:{$name[4]}:{$name[5]}"; $part = $name[6]; $version = preg_replace('#\.sql(.*)#i', '', $name[7]); $info = pathinfo($file); if(isset($list["{$date} {$time}"])){ $info = $list["{$date} {$time}"]; $info['part'] = max($info['part'], $part); $info['size'] = $info['size'] + $file->getSize(); } else { $info['part'] = $part; $info['size'] = $file->getSize(); } $info['compress'] = ($info['extension'] === 'sql') ? '-' : $info['extension']; $info['time'] = strtotime("{$date} {$time}"); $info['version'] = $version; $filenum++; $total += $info['size']; $list["{$date} {$time}"] = $info; } } array_multisort($list, SORT_DESC); $this->assign('list', $list); $this->assign('filenum',$filenum); $this->assign('total',$total); return $this->fetch(); } /** * 上传sql文件 */ public function restoreUpload() { $this->error('该功能仅限技术人员使用!'); $file = request()->file('sqlfile'); if(empty($file)){ $this->error('请上传sql文件'); } // 移动到框架应用根目录/data/sqldata/ 目录下 $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); $path = trim($path, '/'); $image_upload_limit_size = intval(tpCache('basic.file_size') * 1024 * 1024); $info = $file->validate(['size'=>$image_upload_limit_size,'ext'=>'sql,gz'])->move($path, $_FILES['sqlfile']['name']); if ($info) { //上传成功 获取上传文件信息 $file_path_full = $info->getPathName(); if (file_exists($file_path_full)) { $sqls = Backup::parseSql($file_path_full); if(Backup::install($sqls)){ /*清除缓存*/ delFile(RUNTIME_PATH); /*--end*/ $this->success("执行sql成功", url('Tools/restore')); }else{ $this->error('执行sql失败'); } } else { $this->error('sql文件上传失败'); } } else { //上传错误提示错误信息 $this->error($file->getError()); } } /** * 执行还原数据库操作 * @param int $time * @param null $part * @param null $start */ public function import($time = 0, $part = null, $start = null) { function_exists('set_time_limit') && set_time_limit(0); if(is_numeric($time) && is_null($part) && is_null($start)){ //初始化 //获取备份文件信息 $name = date('Ymd-His', $time) . '-*.sql*'; $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); $path = trim($path, '/'); $path = realpath($path) . DS . $name; $files = glob($path); $list = array(); foreach($files as $name){ $basename = basename($name); $match = sscanf($basename, '%4s%2s%2s-%2s%2s%2s-%d'); $gz = preg_match('/^\d{8,8}-\d{6,6}-\d+\.sql.gz$/', $basename); $list[$match[6]] = array($match[6], $name, $gz); } ksort($list); //检测文件正确性 $last = end($list); if(count($list) === $last[0]){ session('backup_list', $list); //缓存备份列表 $part = 1; $start = 0; $data = array('part' => $part, 'start' => $start); // $this->success('初始化完成!', null, array('part' => $part, 'start' => $start)); respose(array('code'=>1, 'msg'=>"初始化完成!准备还原#{$part}...", 'rate'=>'', 'data'=>$data)); } else { // $this->error('备份文件可能已经损坏,请检查!'); respose(array('code'=>0, 'msg'=>"备份文件可能已经损坏,请检查!")); } } elseif(is_numeric($part) && is_numeric($start)) { $list = session('backup_list'); $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); $path = trim($path, '/'); $db = new Backup($list[$part], array( 'path' => realpath($path) . DS, 'compress' => $list[$part][2])); $start = $db->import($start); if(false === $start){ // $this->error('还原数据出错!'); respose(array('code'=>0, 'msg'=>"还原数据出错!", 'rate'=>'0%')); } elseif(0 === $start) { //下一卷 if(isset($list[++$part])){ $data = array('part' => $part, 'start' => 0); // $this->success("正在还g原...#{$part}", null, $data); $rate = (floor((($start+1)/count($list))*10000)/10000*100).'%'; respose(array('code'=>1, 'msg'=>"正在还原#{$part}...", 'rate'=>$rate, 'data'=>$data)); } else { session('backup_list', null); delFile(RUNTIME_PATH); respose(array('code'=>1, 'msg'=>"还原完成...", 'rate'=>'100%')); // $this->success('还原完成!'); } } else { $data = array('part' => $part, 'start' => $start[0]); if($start[1]){ $rate = floor(100 * ($start[0] / $start[1])).'%'; respose(array('code'=>1, 'msg'=>"正在还原#{$part}...", 'rate'=>$rate, 'data'=>$data)); // $this->success("正在还d原...#{$part} ({$rate}%)", null, $data); } else { $data['gz'] = 1; respose(array('code'=>1, 'msg'=>"正在还原#{$part}...", 'data'=>$data, 'start'=>$start)); // $this->success("正在还s原...#{$part}", null, $data); } } } else { // $this->error('参数错误!'); respose(array('code'=>0, 'msg'=>"参数有误", 'rate'=>'0%')); } } /** * (新)执行还原数据库操作 * @param int $time */ public function new_import($time = 0) { function_exists('set_time_limit') && set_time_limit(0); @ini_set('memory_limit','-1'); if(is_numeric($time) && intval($time) > 0){ //获取备份文件信息 $name = date('Ymd-His', $time) . '-*.sql*'; $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); $path = trim($path, '/'); $path = realpath($path) . DS . $name; $files = glob($path); $list = array(); foreach($files as $name){ $basename = basename($name); $match = sscanf($basename, '%4s%2s%2s-%2s%2s%2s-%d-%s'); $gz = preg_match('/^\d{8,8}-\d{6,6}-\d+-v\d+\.\d+\.\d+(.*)\.sql.gz$/', $basename); $list[$match[6]] = array($match[6], $name, $gz); } ksort($list); //检测文件正确性 $last = end($list); $file_path_full = !empty($last[1]) ? $last[1] : ''; if (file_exists($file_path_full)) { /*校验sql文件是否属于当前CMS版本*/ preg_match('/(\d{8,8})-(\d{6,6})-(\d+)-(v\d+\.\d+\.\d+(.*))\.sql/i', $file_path_full, $matches); $version = getCmsVersion(); if ($matches[4] != $version) { $this->error('sql不兼容当前版本:'.$version, url('Tools/restore')); } /*--end*/ $sqls = Backup::parseSql($file_path_full); if(Backup::install($sqls)){ /*清除缓存*/ delFile(RUNTIME_PATH); /*--end*/ $this->success('操作成功', request()->baseFile(), '', 1, [], '_parent'); }else{ $this->error('操作失败!', url('Tools/restore')); } } } else { $this->error("参数有误", url('Tools/restore')); } exit; } /** * 下载 * @param int $time */ public function downFile($time = 0) { $name = date('Ymd-His', $time) . '-*.sql*'; $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); $path = trim($path, '/'); $path = realpath($path) . DS . $name; $files = glob($path); if(is_array($files)){ foreach ($files as $filePath){ if (!file_exists($filePath)) { $this->error("该文件不存在,可能是被删除"); }else{ $filename = basename($filePath); header("Content-type: application/octet-stream"); header('Content-Disposition: attachment; filename="' . $filename . '"'); header("Content-Length: " . filesize($filePath)); readfile($filePath); } } } } /** * 删除备份文件 * @param Integer $time 备份时间 */ public function del() { $time_arr = input('del_id/a'); $time_arr = eyIntval($time_arr); if(is_array($time_arr) && !empty($time_arr)){ foreach ($time_arr as $key => $val) { $name = date('Ymd-His', $val) . '-*.sql*'; $path = tpCache('global.web_sqldatapath'); $path = !empty($path) ? $path : config('DATA_BACKUP_PATH'); $path = trim($path, '/'); $path = realpath($path) . DS . $name; array_map("unlink", glob($path)); if(count(glob($path))){ $this->error('备份文件删除失败,请检查目录权限!'); } } $this->success('删除成功!'); } else { $this->error('参数有误'); } } }