BackUpDatabase.php 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. <?php
  2. declare(strict_types=1);
  3. // +----------------------------------------------------------------------
  4. // | CatchAdmin [Just Like ~ ]
  5. // +----------------------------------------------------------------------
  6. // | Copyright (c) 2017~2020 http://catchadmin.com All rights reserved.
  7. // +----------------------------------------------------------------------
  8. // | Licensed ( https://github.com/yanwenwu/catch-admin/blob/master/LICENSE.txt )
  9. // +----------------------------------------------------------------------
  10. // | Author: JaguarJack [ njphper@gmail.com ]
  11. // +----------------------------------------------------------------------
  12. namespace catcher\library;
  13. use catcher\CatchAdmin;
  14. use catcher\facade\FileSystem;
  15. use think\facade\Db;
  16. class BackUpDatabase
  17. {
  18. /**
  19. *
  20. * @time 2020年07月19日
  21. * @param $tables
  22. * @throws \think\db\exception\DataNotFoundException
  23. * @throws \think\db\exception\DbException
  24. * @throws \think\db\exception\ModelNotFoundException
  25. * @return void
  26. */
  27. public function done($tables)
  28. {
  29. $this->generator(explode(',', $tables));
  30. $this->zip();
  31. }
  32. /**
  33. *
  34. * @time 2019年09月30日
  35. * @param $tables
  36. * @return void
  37. * @throws \think\db\exception\DbException
  38. * @throws \think\db\exception\ModelNotFoundException
  39. * @throws \think\db\exception\DataNotFoundException
  40. */
  41. protected function generator($tables): void
  42. {
  43. foreach ($tables as $table) {
  44. $this->table = $table;
  45. $this->createDataFile();
  46. }
  47. }
  48. /**
  49. * 创建数据文件
  50. *
  51. * @time 2019年09月27日
  52. * @return void
  53. * @throws \think\db\exception\DbException
  54. * @throws \think\db\exception\ModelNotFoundException
  55. * @throws \think\db\exception\DataNotFoundException
  56. */
  57. public function createDataFile(): void
  58. {
  59. $file = CatchAdmin::backupDirectory() . $this->table . '.sql';
  60. $handle = fopen($file, 'wb+');
  61. fwrite($handle, $begin = "BEGIN;\r\n", \strlen($begin));
  62. $this->createClass($this->table, $handle);
  63. fwrite($handle, $end = 'COMMIT;', \strlen($end));
  64. fclose($handle);
  65. }
  66. /**
  67. * 创建了临时模型
  68. *
  69. * @time 2019年09月27日
  70. * @param $table
  71. * @param $handle
  72. * @return void
  73. * @throws \think\db\exception\DbException
  74. * @throws \think\db\exception\ModelNotFoundException
  75. * @throws \think\db\exception\DataNotFoundException
  76. */
  77. protected function createClass($table, $handle)
  78. {
  79. $this->setUnbuffered();
  80. // 防止 IO 多次写入
  81. $buffer = [];
  82. // 记录中记录
  83. $total = Db::table($table)->count();
  84. $limit = 1000;
  85. // 生成器减少内存
  86. while ($total > 0) {
  87. $items = Db::table($table)->limit($limit)->select();
  88. $this->writeIn($handle, $items);
  89. $total -= $limit;
  90. }
  91. }
  92. /**
  93. * sql 文件格式
  94. *
  95. * @time 2019年09月27日
  96. * @param $handle
  97. * @param $datas
  98. * @return void
  99. */
  100. protected function writeIn($handle, $datas)
  101. {
  102. $values = '';
  103. $sql = '';
  104. foreach ($datas as $data) {
  105. foreach ($data as $value) {
  106. $values .= sprintf("'%s'", $value) . ',';
  107. }
  108. $sql .= sprintf('INSERT INTO `%s` VALUE (%s);' . "\r\n", $this->table, rtrim($values, ','));
  109. $values = '';
  110. }
  111. fwrite($handle, $sql, strlen($sql));
  112. }
  113. /**
  114. * 设置未缓存模式
  115. *
  116. * @time 2019年09月29日
  117. * @return void
  118. */
  119. protected function setUnbuffered()
  120. {
  121. $connections = \config('database.connections');
  122. $connections['mysql']['params'] = [
  123. \PDO::MYSQL_ATTR_USE_BUFFERED_QUERY => false
  124. ];
  125. \config([
  126. 'connections' => $connections,
  127. ],'database.connections');
  128. }
  129. /**
  130. * 文件压缩
  131. *
  132. * @time 2020年07月19日
  133. * @throws \Exception
  134. * @return void
  135. */
  136. protected function zip()
  137. {
  138. $files = FileSystem::allFiles(CatchAdmin::backupDirectory());
  139. $storePath = runtime_path('database/');
  140. if (!FileSystem::isDirectory($storePath)) {
  141. FileSystem::makeDirectory($storePath);
  142. }
  143. (new Zip)->make($storePath . 'backup.zip')->addFiles($files)->close();
  144. FileSystem::deleteDirectory(CatchAdmin::backupDirectory());
  145. }
  146. }