Ei kuvausta
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Backup.php 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. <?php
  2. namespace think;
  3. use think\Db;
  4. //数据导出模型
  5. class Backup{
  6. /**
  7. * 文件指针
  8. * @var resource
  9. */
  10. private $fp;
  11. /**
  12. * 备份文件信息 part - 卷号,name - 文件名
  13. * @var array
  14. */
  15. private $file;
  16. /**
  17. * 当前打开文件大小
  18. * @var integer
  19. */
  20. private $size = 0;
  21. /**
  22. * 备份配置
  23. * @var integer
  24. */
  25. private $config;
  26. /**
  27. * 数据库备份构造方法
  28. * @param array $file 备份或还原的文件信息
  29. * @param array $config 备份配置信息
  30. * @param string $type 执行类型,export - 备份数据, import - 还原数据
  31. */
  32. public function __construct($file, $config, $type = 'export'){
  33. $this->file = $file;
  34. $this->config = $config;
  35. }
  36. /**
  37. * 打开一个卷,用于写入数据
  38. * @param integer $size 写入数据的大小
  39. */
  40. private function open($size){
  41. if($this->fp){
  42. $this->size += $size;
  43. if($this->size > $this->config['part']){
  44. $this->config['compress'] ? @gzclose($this->fp) : @fclose($this->fp);
  45. $this->fp = null;
  46. $this->file['part']++;
  47. session('backup_file', $this->file);
  48. $this->create();
  49. }
  50. } else {
  51. $backuppath = $this->config['path'];
  52. $filename = "{$backuppath}{$this->file['name']}-{$this->file['part']}-{$this->file['version']}.sql";
  53. if($this->config['compress']){
  54. $filename = "{$filename}.gz";
  55. $this->fp = @gzopen($filename, "a{$this->config['level']}");
  56. } else {
  57. $this->fp = @fopen($filename, 'a');
  58. }
  59. $this->size = filesize($filename) + $size;
  60. }
  61. }
  62. /**
  63. * 写入初始数据
  64. * @return boolean true - 写入成功,false - 写入失败
  65. */
  66. public function create(){
  67. $mysqlinfo = Db::query("SELECT VERSION() as version");
  68. $mysql_version = $mysqlinfo[0]['version'];
  69. $sql = "-- ----------------------------------------\n";
  70. $sql .= "-- EyouCms MySQL Data Transfer \n";
  71. $sql .= "-- \n";
  72. $sql .= "-- Server : " . config('database.hostname').'_'.config('database.hostport') . "\n";
  73. $sql .= "-- Server Version : " . $mysql_version . "\n";
  74. $sql .= "-- Host : " . config('database.hostname').':'.config('database.hostport') . "\n";
  75. $sql .= "-- Database : " . config('database.database') . "\n";
  76. $sql .= "-- \n";
  77. $sql .= "-- Part : #{$this->file['part']}\n";
  78. $sql .= "-- Version : #{$this->file['version']}\n";
  79. $sql .= "-- Date : " . date("Y-m-d H:i:s") . "\n";
  80. $sql .= "-- -----------------------------------------\n\n";
  81. $sql .= "SET FOREIGN_KEY_CHECKS = 0;\n\n";
  82. return $this->write($sql);
  83. }
  84. /**
  85. * 写入SQL语句
  86. * @param string $sql 要写入的SQL语句
  87. * @return boolean true - 写入成功,false - 写入失败!
  88. */
  89. private function write($sql){
  90. $size = strlen($sql);
  91. //由于压缩原因,无法计算出压缩后的长度,这里假设压缩率为50%,
  92. //一般情况压缩率都会高于50%;
  93. $size = $this->config['compress'] ? $size / 2 : $size;
  94. $this->open($size);
  95. return $this->config['compress'] ? @gzwrite($this->fp, $sql) : @fwrite($this->fp, $sql);
  96. }
  97. /**
  98. * 备份表结构
  99. * @param string $table 表名
  100. * @param integer $start 起始行数
  101. * @return boolean false - 备份失败
  102. */
  103. public function backup($table, $start){
  104. //备份表结构
  105. if(0 == $start){
  106. $result = Db::query("SHOW CREATE TABLE `{$table}`");
  107. $sql = "\n";
  108. $sql .= "-- -----------------------------\n";
  109. $sql .= "-- Table structure for `{$table}`\n";
  110. $sql .= "-- -----------------------------\n";
  111. $sql .= "DROP TABLE IF EXISTS `{$table}`;\n";
  112. $sql .= trim($result[0]['Create Table']) . ";\n\n";
  113. if(false === $this->write($sql)){
  114. return false;
  115. }
  116. }
  117. //数据总数
  118. $result = Db::query("SELECT COUNT(*) AS count FROM `{$table}`");
  119. $count = $result['0']['count'];
  120. //备份表数据
  121. if($count){
  122. //写入数据注释
  123. if(0 == $start){
  124. $sql = "-- -----------------------------\n";
  125. $sql .= "-- Records of `{$table}`\n";
  126. $sql .= "-- -----------------------------\n";
  127. $this->write($sql);
  128. }
  129. //备份数据记录
  130. $result = Db::query("SELECT * FROM `{$table}` LIMIT {$start}, 1000");
  131. foreach ($result as $row) {
  132. $row = array_map('addslashes', $row);
  133. $sql = "INSERT INTO `{$table}` VALUES ('" . str_replace(array("\r","\n"),array('\r','\n'),implode("', '", $row)) . "');\n";
  134. if(false === $this->write($sql)){
  135. return false;
  136. }
  137. }
  138. //还有更多数据
  139. if($count > $start + 1000){
  140. return array($start + 1000, $count);
  141. }
  142. }
  143. //备份下一表
  144. return 0;
  145. }
  146. public function import($start){
  147. //还原数据
  148. if($this->config['compress']){
  149. $gz = gzopen($this->file[1], 'r');
  150. $size = 0;
  151. } else {
  152. $size = filesize($this->file[1]);
  153. $gz = fopen($this->file[1], 'r');
  154. }
  155. $sql = '';
  156. if($start){
  157. $this->config['compress'] ? gzseek($gz, $start) : fseek($gz, $start);
  158. }
  159. $totalsize = 0;
  160. // for($i = 0; $i < 1000; $i++){
  161. // $sql .= $this->config['compress'] ? gzgets($gz) : fgets($gz);
  162. // if(preg_match('/.*;$/', trim($sql))){
  163. // /**
  164. // * 统计SQL文件字符长度
  165. // */
  166. // $sqlFormat = $this->sql_split($sql, config('database.prefix'));
  167. // $counts = count($sqlFormat);
  168. // for ($j = 0; $j < $counts; $j++) {
  169. // $tmp_sql = trim($sqlFormat[$j]);
  170. // if (strstr($tmp_sql, 'CREATE TABLE') || !empty($tmp_sql)) {
  171. // $totalsize += strlen($tmp_sql);
  172. // }
  173. // }
  174. // }
  175. // $sql = '';
  176. // }
  177. $sql = '';
  178. for($i = 0; $i < 1000; $i++){
  179. $sql .= $this->config['compress'] ? gzgets($gz) : fgets($gz);
  180. if(preg_match('/.*;$/', trim($sql))){
  181. /**
  182. * 执行SQL语句
  183. */
  184. $sqlFormat = $this->sql_split($sql, config('database.prefix'));
  185. $counts = count($sqlFormat);
  186. for ($j = 0; $j < $counts; $j++) {
  187. $tmp_sql = trim($sqlFormat[$j]);
  188. if (strstr($tmp_sql, 'CREATE TABLE')) {
  189. if(false !== Db::execute($tmp_sql)){
  190. $start += strlen($tmp_sql);
  191. } else {
  192. return false;
  193. }
  194. } else {
  195. if(trim($tmp_sql) == '')
  196. continue;
  197. if(false !== Db::execute($tmp_sql)){
  198. $start += strlen($tmp_sql);
  199. } else {
  200. return false;
  201. }
  202. }
  203. }
  204. $sql = '';
  205. } elseif ($this->config['compress'] ? gzeof($gz) : feof($gz)) {
  206. return 0;
  207. }
  208. }
  209. return array($start, $size, $totalsize);
  210. }
  211. /**
  212. * 析构方法,用于关闭文件资源
  213. */
  214. public function __destruct(){
  215. $this->config['compress'] ? @gzclose($this->fp) : @fclose($this->fp);
  216. }
  217. /**
  218. * 解析
  219. * @access public
  220. * @param mixed $filename
  221. * @return mixed
  222. */
  223. public static function parseSql($filename){
  224. /*获取所有数据表名*/
  225. $tableList = [];
  226. $dbtables = Db::query('SHOW TABLE STATUS');
  227. foreach ($dbtables as $k => $v) {
  228. if (preg_match('/^'.config('database.prefix').'/i', $v['Name'])) {
  229. $tableList[] = $v['Name'];
  230. }
  231. }
  232. /*--end*/
  233. $lines=file($filename);
  234. $lines[0]=str_replace(chr(239).chr(187).chr(191),"",$lines[0]);//去除BOM头
  235. $flage = true;
  236. $sqls = array();
  237. $sql="";
  238. foreach($lines as $line)
  239. {
  240. $line=trim($line);
  241. $char=substr($line,0,1);
  242. if($char!='#' && strlen($line)>0)
  243. {
  244. $prefix=substr($line,0,2);
  245. switch($prefix)
  246. {
  247. case '/*':
  248. {
  249. $flage=(substr($line,-3)=='*/;'||substr($line,-2)=='*/')?true:false;
  250. break 1;
  251. }
  252. case '--': break 1;
  253. default :
  254. {
  255. if($flage)
  256. {
  257. $sql.=$line;
  258. if(substr($line,-1)==";")
  259. {
  260. /*处理安装sql与备份sql里表前缀的兼容性*/
  261. $ey_prefix = 'ey_';
  262. foreach ($tableList as $k2 => $v2) {
  263. if (stristr($sql, '`'.$v2.'`')) {
  264. $ey_prefix = config('database.prefix');
  265. break;
  266. }
  267. }
  268. /*--end*/
  269. if ('ey_' != config('database.prefix')) {
  270. $sql = str_replace("`{$ey_prefix}", '`'.config('database.prefix'), $sql);
  271. }
  272. $sql = preg_replace("/TYPE=(InnoDB|MyISAM|MEMORY)( DEFAULT CHARSET=[^; ]+)?/", "ENGINE=\\1 DEFAULT CHARSET=utf8", $sql);
  273. $sqls[]=$sql;
  274. $sql="";
  275. }
  276. }
  277. if(!$flage)$flage=(substr($line,-3)=='*/;'||substr($line,-2)=='*/')?true:false;
  278. }
  279. }
  280. }
  281. }
  282. return $sqls;
  283. }
  284. /**
  285. * 安装
  286. * @access public
  287. * @param mixed $sqls
  288. * @return mixed
  289. */
  290. public static function install($sqls){
  291. $flag = true;
  292. if(is_array($sqls))
  293. {
  294. foreach($sqls as $sql)
  295. {
  296. if(Db::execute($sql) === false){ $flag = false;}
  297. }
  298. }
  299. return $flag;
  300. }
  301. public function sql_split($sql, $tablepre) {
  302. if ($tablepre != "ey_")
  303. $sql = str_replace("`ey_", '`'.$tablepre, $sql);
  304. $sql = preg_replace("/TYPE=(InnoDB|MyISAM|MEMORY)( DEFAULT CHARSET=[^; ]+)?/", "ENGINE=\\1 DEFAULT CHARSET=utf8", $sql);
  305. $sql = str_replace("\r", "\n", $sql);
  306. $ret = array();
  307. $num = 0;
  308. $queriesarray = explode(";\n", trim($sql));
  309. unset($sql);
  310. foreach ($queriesarray as $query) {
  311. $ret[$num] = '';
  312. $queries = explode("\n", trim($query));
  313. $queries = array_filter($queries);
  314. foreach ($queries as $query) {
  315. $str1 = substr($query, 0, 1);
  316. if ($str1 != '#' && $str1 != '-')
  317. $ret[$num] .= $query;
  318. }
  319. $num++;
  320. }
  321. return $ret;
  322. }
  323. }