411 lines
13 KiB
PHP
411 lines
13 KiB
PHP
![]() |
<?php
|
||
|
/**
|
||
|
* tpshop
|
||
|
* ============================================================================
|
||
|
* 版权所有 2015-2027 深圳搜豹网络科技有限公司,并保留所有权利。
|
||
|
* 网站地址: http://www.tp-shop.cn
|
||
|
* ----------------------------------------------------------------------------
|
||
|
* 这不是一个自由软件!您只能在不用于商业目的的前提下对程序代码进行修改和使用 .
|
||
|
* 不允许对程序代码以任何形式任何目的的再发布。
|
||
|
* 采用最新Thinkphp5助手函数特性实现单字母函数M D U等简写方式
|
||
|
* ============================================================================
|
||
|
* Author: 当燃
|
||
|
*
|
||
|
* Date: 2016-03-09
|
||
|
*/
|
||
|
|
||
|
namespace app\admin\controller;
|
||
|
use think\db;
|
||
|
use think\Backup;
|
||
|
|
||
|
class Tools extends Base {
|
||
|
public function index(){
|
||
|
$dbtables = DB::query('SHOW TABLE STATUS');
|
||
|
$total = 0;
|
||
|
foreach ($dbtables as $k => $v) {
|
||
|
$dbtables[$k]['size'] = format_bytes($v['Data_length'] + $v['Index_length']);
|
||
|
$total += $v['Data_length'] + $v['Index_length'];
|
||
|
}
|
||
|
$this->assign('list', $dbtables);
|
||
|
$this->assign('total', format_bytes($total));
|
||
|
$this->assign('tableNum', count($dbtables));
|
||
|
return $this->fetch();
|
||
|
}
|
||
|
|
||
|
public function export($tables = null, $id = null, $start = null){
|
||
|
//防止备份数据过程超时
|
||
|
function_exists('set_time_limit') && set_time_limit(0);
|
||
|
if(IS_POST && !empty($tables) && is_array($tables)){ //初始化
|
||
|
$path = C('DATA_BACKUP_PATH');
|
||
|
if(!is_dir($path)){
|
||
|
mkdir($path, 0755, true);
|
||
|
}
|
||
|
//读取备份配置
|
||
|
$config = array(
|
||
|
'path' => realpath($path) . DIRECTORY_SEPARATOR,
|
||
|
'part' => C('DATA_BACKUP_PART_SIZE'),
|
||
|
'compress' => C('DATA_BACKUP_COMPRESS'),
|
||
|
'level' => C('DATA_BACKUP_COMPRESS_LEVEL'),
|
||
|
);
|
||
|
//检查是否有正在执行的任务
|
||
|
$lock = "{$config['path']}backup.lock";
|
||
|
if(is_file($lock)){
|
||
|
// $this->error('检测到有一个备份任务正在执行,请稍后再试!');
|
||
|
return json(array('info'=>'检测到有一个备份任务正在执行,请稍后再试!', 'status'=>0, 'url'=>''));
|
||
|
} else {
|
||
|
//创建锁文件
|
||
|
file_put_contents($lock, NOW_TIME);
|
||
|
}
|
||
|
|
||
|
//检查备份目录是否可写
|
||
|
// is_writeable($config['path']) || $this->error('备份目录不存在或不可写,请检查后重试!');
|
||
|
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,
|
||
|
);
|
||
|
session('backup_file', $file);
|
||
|
//缓存要备份的表
|
||
|
session('backup_tables', $tables);
|
||
|
//创建备份文件
|
||
|
$Database = new Backup($file, $config);
|
||
|
if(false !== $Database->create()){
|
||
|
$tab = array('id' => 0, 'start' => 0);
|
||
|
// $this->success('初始化成功!', '', array('tables' => $tables, 'tab' => $tab));
|
||
|
return json(array('tables' => $tables, 'tab' => $tab, 'info'=>'初始化成功!', 'status'=>1, 'url'=>''));
|
||
|
} else {
|
||
|
// $this->error('初始化失败,备份文件创建失败!');
|
||
|
return json(array('info'=>'初始化失败,备份文件创建失败!', 'status'=>0, 'url'=>''));
|
||
|
}
|
||
|
} elseif (IS_GET && is_numeric($id) && is_numeric($start)) { //备份数据
|
||
|
$tables = session('backup_tables');
|
||
|
//备份指定表
|
||
|
$Database = new Backup(session('backup_file'), session('backup_config'));
|
||
|
$start = $Database->backup($tables[$id], $start);
|
||
|
if(false === $start){ //出错
|
||
|
// $this->error('备份出错!');
|
||
|
return json(array('info'=>'备份出错!', 'status'=>0, 'url'=>''));
|
||
|
} elseif (0 === $start) { //下一表
|
||
|
if(isset($tables[++$id])){
|
||
|
$tab = array('id' => $id, 'start' => 0);
|
||
|
// $this->success('备份完成!', '', array('tab' => $tab));
|
||
|
return json(array('tab' => $tab, 'info'=>'备份完成!', 'status'=>1, 'url'=>''));
|
||
|
} else { //备份完成,清空缓存
|
||
|
unlink(session('backup_config.path') . 'backup.lock');
|
||
|
session('backup_tables', null);
|
||
|
session('backup_file', null);
|
||
|
session('backup_config', null);
|
||
|
// $this->success('备份完成!');
|
||
|
return json(array('info'=>'备份完成!', 'status'=>1, 'url'=>''));
|
||
|
}
|
||
|
} else {
|
||
|
$tab = array('id' => $id, 'start' => $start[0]);
|
||
|
$rate = floor(100 * ($start[0] / $start[1]));
|
||
|
// $this->success("正在备份...({$rate}%)", '', array('tab' => $tab));
|
||
|
return json(array('tab' => $tab, 'info'=>"正在备份...({$rate}%)", 'status'=>1, 'url'=>''));
|
||
|
}
|
||
|
|
||
|
} else {//出错
|
||
|
// $this->error('参数错误!');
|
||
|
return json(array('info'=>'参数错误!', 'status'=>0, 'url'=>''));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public function restore(){
|
||
|
$path = C('DATA_BACKUP_PATH');
|
||
|
if(!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+\.sql(?:\.gz)?$/', $name)){
|
||
|
$name = sscanf($name, '%4s%2s%2s-%2s%2s%2s-%d');
|
||
|
$date = "{$name[0]}-{$name[1]}-{$name[2]}";
|
||
|
$time = "{$name[3]}:{$name[4]}:{$name[5]}";
|
||
|
$part = $name[6];
|
||
|
$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}");
|
||
|
$filenum++;
|
||
|
$total += $info['size'];
|
||
|
$list["{$date} {$time}"] = $info;
|
||
|
}
|
||
|
}
|
||
|
$this->assign('list', $list);
|
||
|
$this->assign('filenum',$filenum);
|
||
|
$this->assign('total',$total);
|
||
|
return $this->fetch();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 执行还原数据库操作
|
||
|
* @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 = realpath(C('DATA_BACKUP_PATH')) . DIRECTORY_SEPARATOR . $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); //缓存备份列表
|
||
|
$this->success('初始化完成!', null, array('part' => 1, 'start' => 0));
|
||
|
} else {
|
||
|
$this->error('备份文件可能已经损坏,请检查!');
|
||
|
}
|
||
|
} elseif(is_numeric($part) && is_numeric($start)) {
|
||
|
$list = session('backup_list');
|
||
|
$db = new Backup($list[$part], array(
|
||
|
'path' => realpath(C('DATA_BACKUP_PATH')) . DIRECTORY_SEPARATOR,
|
||
|
'compress' => $list[$part][2]));
|
||
|
$start = $db->import($start);
|
||
|
if(false === $start){
|
||
|
$this->error('还原数据出错!');
|
||
|
} elseif(0 === $start) { //下一卷
|
||
|
if(isset($list[++$part])){
|
||
|
$data = array('part' => $part, 'start' => 0);
|
||
|
$this->success("正在还原...#{$part}", null, $data);
|
||
|
} else {
|
||
|
session('backup_list', null);
|
||
|
$this->success('还原完成!');
|
||
|
}
|
||
|
} else {
|
||
|
$data = array('part' => $part, 'start' => $start[0]);
|
||
|
if($start[1]){
|
||
|
$rate = floor(100 * ($start[0] / $start[1]));
|
||
|
$this->success("正在还原...#{$part} ({$rate}%)", null, $data);
|
||
|
} else {
|
||
|
$data['gz'] = 1;
|
||
|
$this->success("正在还原...#{$part}", null, $data);
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
$this->error('参数错误!');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 优化
|
||
|
*/
|
||
|
public function optimize() {
|
||
|
$batchFlag = I('get.batchFlag', 0, 'intval');
|
||
|
//批量删除
|
||
|
if ($batchFlag) {
|
||
|
$table = I('key', array());
|
||
|
}else {
|
||
|
$table[] = I('tablename' , '');
|
||
|
}
|
||
|
|
||
|
if (empty($table)) {
|
||
|
$this->error('请选择要优化的表');
|
||
|
}
|
||
|
|
||
|
$strTable = implode(',', $table);
|
||
|
if (!DB::query("OPTIMIZE TABLE {$strTable} ")) {
|
||
|
$strTable = '';
|
||
|
}
|
||
|
$this->success("优化表成功" . $strTable, U('Tools/index'));
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 修复
|
||
|
*/
|
||
|
public function repair() {
|
||
|
$batchFlag = I('get.batchFlag', 0, 'intval');
|
||
|
//批量删除
|
||
|
if ($batchFlag) {
|
||
|
$table = I('key', array());
|
||
|
}else {
|
||
|
$table[] = I('tablename' , '');
|
||
|
}
|
||
|
|
||
|
if (empty($table)) {
|
||
|
$this->error('请选择修复的表');
|
||
|
}
|
||
|
|
||
|
$strTable = implode(',', $table);
|
||
|
if (!DB::query("REPAIR TABLE {$strTable} ")) {
|
||
|
$strTable = '';
|
||
|
}
|
||
|
|
||
|
$this->success("修复表成功" . $strTable, U('Tools/index'));
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
public function restoreUpload()
|
||
|
{
|
||
|
$file = request()->file('sqlfile');
|
||
|
if(empty($file)){
|
||
|
$this->error('请先上传sql文件');
|
||
|
}
|
||
|
// 移动到框架应用根目录/public/uploads/ 目录下
|
||
|
$path = UPLOAD_PATH.'sqldata' ;
|
||
|
$info = $file->validate(['size'=>200000000,'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)){
|
||
|
// array_map("unlink", glob($path));
|
||
|
$this->success("导入sql文件成功", U('Tools/restore'));
|
||
|
}else{
|
||
|
$this->error('sql文件导入失败');
|
||
|
}
|
||
|
} else {
|
||
|
$this->error('sql文件上传失败');
|
||
|
}
|
||
|
} else {
|
||
|
//上传错误提示错误信息
|
||
|
$this->error($file->getError());
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 下载
|
||
|
* @param int $time
|
||
|
*/
|
||
|
public function downFile($time = 0) {
|
||
|
$name = date('Ymd-His', $time) . '-*.sql*';
|
||
|
$path = realpath(C('DATA_BACKUP_PATH')) . DIRECTORY_SEPARATOR . $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 = 0){
|
||
|
if($time){
|
||
|
$name = date('Ymd-His', $time) . '-*.sql*';
|
||
|
$path = realpath(C('DATA_BACKUP_PATH')) . DIRECTORY_SEPARATOR . $name;
|
||
|
array_map("unlink", glob($path));
|
||
|
if(count(glob($path))){
|
||
|
$this->error('备份文件删除失败,请检查权限!');
|
||
|
} else {
|
||
|
$this->success('备份文件删除成功!');
|
||
|
}
|
||
|
} else {
|
||
|
$this->error('参数错误!');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public function region(){
|
||
|
|
||
|
$parent_id = I('parent_id',0);
|
||
|
if($parent_id == 0){
|
||
|
$parent = array('id'=>0,'name'=>"中国省份地区",'level'=>0);
|
||
|
}else{
|
||
|
$parent = M('region')->where("id" ,$parent_id)->find();
|
||
|
}
|
||
|
$names = $this->getParentRegionList($parent_id);
|
||
|
if(count($names) > 0){
|
||
|
$names = array_reverse($names);
|
||
|
$parent_path = implode($names, '>');
|
||
|
}
|
||
|
$region = M('region')->where("parent_id" , $parent_id)->select();
|
||
|
$this->assign('parent',$parent);
|
||
|
$this->assign('parent_path',$parent_path);
|
||
|
$this->assign('region',$region);
|
||
|
return $this->fetch();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* 寻找Region_id的父级字段, $column可自己指定
|
||
|
* @param $parent_id
|
||
|
* @return array
|
||
|
*/
|
||
|
function getParentRegionList($parent_id){
|
||
|
$names = array();
|
||
|
$region = M('region')->where(array('id'=>$parent_id))->find();
|
||
|
array_push($names,$region['name']);
|
||
|
if($region['parent_id'] != 0){
|
||
|
$nregion = $this->getParentRegionList($region['parent_id']);
|
||
|
if(!empty($nregion)){
|
||
|
$names = array_merge($names, $nregion);
|
||
|
}
|
||
|
}
|
||
|
return $names;
|
||
|
}
|
||
|
|
||
|
public function regionHandle(){
|
||
|
$data = I('post.');
|
||
|
$id = I('id');
|
||
|
$referurl = isset($_SERVER['HTTP_REFERER']) ? $_SERVER['HTTP_REFERER'] : U("Tools/region");
|
||
|
if(empty($id)){
|
||
|
$data['level'] = $data['level']+1;
|
||
|
if(empty($data['name'])){
|
||
|
$this->error("请填写地区名称", $referurl);
|
||
|
}else{
|
||
|
$res = M('region')->where("parent_id = ".$data['parent_id']." and name='".$data['name']."'")->find();
|
||
|
if(empty($res)){
|
||
|
M('region')->cache(true)->add($data);
|
||
|
$this->success("操作成功", $referurl);
|
||
|
}else{
|
||
|
$this->error("该区域下已有该地区,请不要重复添加", $referurl);
|
||
|
}
|
||
|
}
|
||
|
}else{
|
||
|
M('region')->where("id=$id or parent_id=$id")->cache(true)->delete();
|
||
|
$this->success("操作成功", $referurl);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public function ecshop(){
|
||
|
return $this->fetch();
|
||
|
}
|
||
|
|
||
|
public function taobao(){
|
||
|
return $this->fetch();
|
||
|
}
|
||
|
public function clear_demo_data(){
|
||
|
return $this->fetch();
|
||
|
}
|
||
|
}
|