Db.php 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. <?php
  2. /**
  3. * Copyright © Magento, Inc. All rights reserved.
  4. * See COPYING.txt for license details.
  5. */
  6. namespace Magento\Backup\Model;
  7. use Magento\Backup\Helper\Data as Helper;
  8. use Magento\Backup\Model\ResourceModel\Table\GetListTables;
  9. use Magento\Backup\Model\ResourceModel\View\CreateViewsBackup;
  10. use Magento\Framework\App\ObjectManager;
  11. use Magento\Framework\Exception\RuntimeException;
  12. /**
  13. * Database backup model
  14. *
  15. * @api
  16. * @since 100.0.2
  17. * @deprecated 100.2.6 Backup module is to be removed.
  18. */
  19. class Db implements \Magento\Framework\Backup\Db\BackupDbInterface
  20. {
  21. /**
  22. * Buffer length for multi rows
  23. * default 100 Kb
  24. */
  25. const BUFFER_LENGTH = 102400;
  26. /**
  27. * Backup resource model
  28. *
  29. * @var \Magento\Backup\Model\ResourceModel\Db
  30. */
  31. protected $_resourceDb = null;
  32. /**
  33. * Core resource model
  34. *
  35. * @var \Magento\Framework\App\ResourceConnection
  36. */
  37. protected $_resource = null;
  38. /**
  39. * @var Helper
  40. */
  41. private $helper;
  42. /**
  43. * @var GetListTables
  44. */
  45. private $getListTables;
  46. /**
  47. * @var CreateViewsBackup
  48. */
  49. private $getViewsBackup;
  50. /**
  51. * Db constructor.
  52. * @param ResourceModel\Db $resourceDb
  53. * @param \Magento\Framework\App\ResourceConnection $resource
  54. * @param Helper|null $helper
  55. * @param GetListTables|null $getListTables
  56. * @param CreateViewsBackup|null $getViewsBackup
  57. */
  58. public function __construct(
  59. ResourceModel\Db $resourceDb,
  60. \Magento\Framework\App\ResourceConnection $resource,
  61. ?Helper $helper = null,
  62. ?GetListTables $getListTables = null,
  63. ?CreateViewsBackup $getViewsBackup = null
  64. ) {
  65. $this->_resourceDb = $resourceDb;
  66. $this->_resource = $resource;
  67. $this->helper = $helper ?? ObjectManager::getInstance()->get(Helper::class);
  68. $this->getListTables = $getListTables ?? ObjectManager::getInstance()->get(GetListTables::class);
  69. $this->getViewsBackup = $getViewsBackup ?? ObjectManager::getInstance()->get(CreateViewsBackup::class);
  70. }
  71. /**
  72. * List of tables which data should not be backed up
  73. *
  74. * @var array
  75. */
  76. protected $_ignoreDataTablesList = ['importexport/importdata'];
  77. /**
  78. * Retrieve resource model
  79. *
  80. * @return \Magento\Backup\Model\ResourceModel\Db
  81. */
  82. public function getResource()
  83. {
  84. return $this->_resourceDb;
  85. }
  86. /**
  87. * Tables list.
  88. *
  89. * @return array
  90. */
  91. public function getTables()
  92. {
  93. return $this->getResource()->getTables();
  94. }
  95. /**
  96. * Command to recreate given table.
  97. *
  98. * @param string $tableName
  99. * @param bool $addDropIfExists
  100. * @return string
  101. */
  102. public function getTableCreateScript($tableName, $addDropIfExists = false)
  103. {
  104. return $this->getResource()->getTableCreateScript($tableName, $addDropIfExists);
  105. }
  106. /**
  107. * Generate table's data dump.
  108. *
  109. * @param string $tableName
  110. * @return string
  111. */
  112. public function getTableDataDump($tableName)
  113. {
  114. return $this->getResource()->getTableDataDump($tableName);
  115. }
  116. /**
  117. * Header for dumps.
  118. *
  119. * @return string
  120. */
  121. public function getHeader()
  122. {
  123. return $this->getResource()->getHeader();
  124. }
  125. /**
  126. * Footer for dumps.
  127. *
  128. * @return string
  129. */
  130. public function getFooter()
  131. {
  132. return $this->getResource()->getFooter();
  133. }
  134. /**
  135. * Get backup SQL.
  136. *
  137. * @return string
  138. */
  139. public function renderSql()
  140. {
  141. ini_set('max_execution_time', 0);
  142. $sql = $this->getHeader();
  143. $tables = $this->getTables();
  144. foreach ($tables as $tableName) {
  145. $sql .= $this->getTableCreateScript($tableName, true);
  146. $sql .= $this->getTableDataDump($tableName);
  147. }
  148. $sql .= $this->getFooter();
  149. return $sql;
  150. }
  151. /**
  152. * @inheritDoc
  153. */
  154. public function createBackup(\Magento\Framework\Backup\Db\BackupInterface $backup)
  155. {
  156. if (!$this->helper->isEnabled()) {
  157. throw new RuntimeException(__('Backup functionality is disabled'));
  158. }
  159. $backup->open(true);
  160. $this->getResource()->beginTransaction();
  161. $tables = $this->getListTables->execute();
  162. $backup->write($this->getResource()->getHeader());
  163. $ignoreDataTablesList = $this->getIgnoreDataTablesList();
  164. foreach ($tables as $table) {
  165. $backup->write(
  166. $this->getResource()->getTableHeader($table) . $this->getResource()->getTableDropSql($table) . "\n"
  167. );
  168. $backup->write($this->getResource()->getTableCreateSql($table, false) . "\n");
  169. $tableStatus = $this->getResource()->getTableStatus($table);
  170. if ($tableStatus->getRows() && !in_array($table, $ignoreDataTablesList)) {
  171. $backup->write($this->getResource()->getTableDataBeforeSql($table));
  172. if ($tableStatus->getDataLength() > self::BUFFER_LENGTH) {
  173. if ($tableStatus->getAvgRowLength() < self::BUFFER_LENGTH) {
  174. $limit = floor(self::BUFFER_LENGTH / max($tableStatus->getAvgRowLength(), 1));
  175. $multiRowsLength = ceil($tableStatus->getRows() / $limit);
  176. } else {
  177. $limit = 1;
  178. $multiRowsLength = $tableStatus->getRows();
  179. }
  180. } else {
  181. $limit = $tableStatus->getRows();
  182. $multiRowsLength = 1;
  183. }
  184. for ($i = 0; $i < $multiRowsLength; $i++) {
  185. $backup->write($this->getResource()->getTableDataSql($table, $limit, $i * $limit));
  186. }
  187. $backup->write($this->getResource()->getTableDataAfterSql($table));
  188. }
  189. }
  190. $this->getViewsBackup->execute($backup);
  191. $backup->write($this->getResource()->getTableForeignKeysSql());
  192. $backup->write($this->getResource()->getTableTriggersSql());
  193. $backup->write($this->getResource()->getFooter());
  194. $this->getResource()->commitTransaction();
  195. $backup->close();
  196. }
  197. /**
  198. * Get database backup size
  199. *
  200. * @return int
  201. */
  202. public function getDBBackupSize()
  203. {
  204. $tables = $this->getResource()->getTables();
  205. $ignoreDataTablesList = $this->getIgnoreDataTablesList();
  206. $size = 0;
  207. foreach ($tables as $table) {
  208. $tableStatus = $this->getResource()->getTableStatus($table);
  209. if ($tableStatus->getRows() && !in_array($table, $ignoreDataTablesList)) {
  210. $size += $tableStatus->getDataLength() + $tableStatus->getIndexLength();
  211. }
  212. }
  213. return $size;
  214. }
  215. /**
  216. * Returns the list of tables which data should not be backed up
  217. *
  218. * @return string[]
  219. */
  220. public function getIgnoreDataTablesList()
  221. {
  222. $result = [];
  223. foreach ($this->_ignoreDataTablesList as $table) {
  224. $result[] = $this->_resource->getTableName($table);
  225. }
  226. return $result;
  227. }
  228. }