Можно ли автоматически предсказать операции, которые следуют за DELETE CASCADE? В моем программном обеспечении я хотел бы дать пользователю предупреждение с подробностями о данных, которые будут удалены тогда.
Имитировать DELETE CASCADE в MySQL?
Ответ 1
Вы можете сделать копию базы данных и поместить триггеры на after delete
DELIMITER $$
CREATE TRIGGER ad_table1_each AFTER DELETE ON table1 FOR EACH ROW
BEGIN
INSERT INTO log VALUES (null /*autoinc id*/
, 'table1' /*tablename*/
, old.id /*tableid*/
, concat_ws(',',old.field1,old.field2 /*CSV of fields*/
, NOW() /*timestamp*/
, 'delete'); /*what action*/
REPLACE INTO restore_table1 VALUES (old.id,
, old.field1
, old.field2
, ... );
END $$
DELIMITER ;
Таблица журналов - это всего лишь таблица со следующими полями:
id integer autoincrement primary key
tablename varchar(45)
table_id integer
fields varchar(6000)
delete_time timestamp
action enum('insert','update','delete')
Если вы делаете SELECT @last_id:= max(id) FROM log
перед удалением каскада на копии.
Затем вы можете сделать SELECT * FROM log WHERE id > @last_id
и получить все строки, которые будут удалены в каскаде.
После этого вы можете использовать restore_table1 для воссоздания строк, которые были удалены в каскаде в базе данных копий.
Ответ 2
Я думаю, вы могли бы использовать решение запуска Johan в сочетании с транзакцией, которую вы откатываете назад. Это позволяет избежать необходимости во второй базе данных и для ручного восстановления удаленных записей.
- добавить триггер и таблицу журналов
- для каждого предпринятого удаления запустите транзакцию и удалите записи
- представить информацию из журнала для вашего пользователя для утверждения
- если пользователь согласен совершить транзакцию, в противном случае откат
Ответ 3
Я написал очень быстрый взлом, который делает именно то, что вам нужно на PHP, так как я хотел сделать то же самое и не нашел никаких ресурсов для этого онлайн.
Это может быть слишком поздно для вас, но это может помочь другим.
function get_referencing_foreign_keys ($database, $table) {
$query = 'SELECT TABLE_SCHEMA, TABLE_NAME, COLUMN_NAME, REFERENCED_COLUMN_NAME FROM INFORMATION_SCHEMA.KEY_COLUMN_USAGE WHERE REFERENCED_TABLE_SCHEMA = "'.$database.'" AND REFERENCED_TABLE_NAME = '.esc($table);
$result = rquery($query);
$foreign_keys = array();
while ($row = mysql_fetch_row($result)) {
$foreign_keys[] = array('database' => $row[0], 'table' => $row[1], 'column' => $row[2], 'reference_column' => $row[3]);
}
return $foreign_keys;
}
function get_foreign_key_deleted_data_html ($database, $table, $where) {
$data = get_foreign_key_deleted_data ($database, $table, $where);
$html = '';
foreach ($data as $key => $this_data) {
$html .= "<h2>$key</h2>\n";
$html .= "<table>\n";
$i = 0;
foreach ($this_data as $value) {
if($i == 0) {
$html .= "\t<tr>\n";
foreach ($value as $column => $column_value) {
$html .= "\t\t<th>".htmlentities($column)."</th>\n";
}
$html .= "\t</tr>\n";
}
$html .= "\t<tr>\n";
foreach ($value as $column => $column_value) {
$html .= "\t\t<td>".htmlentities($column_value)."</td>\n";
}
$html .= "\t</tr>\n";
$i++;
}
$html .= "</table>\n";
}
return $html;
}
function get_foreign_key_deleted_data ($database, $table, $where) {
$GLOBALS['get_data_that_would_be_deleted'] = array();
$data = get_data_that_would_be_deleted($database, $table, $where);
$GLOBALS['get_data_that_would_be_deleted'] = array();
return $data;
}
function get_data_that_would_be_deleted ($database, $table, $where, $recursion = 100) {
if($recursion <= 0) {
die("Deep recursion!");
}
if($recursion == 100) {
$GLOBALS['get_data_that_would_be_deleted'] = array();
}
if($table) {
if(is_array($where)) {
$foreign_keys = get_referencing_foreign_keys($database, $table);
$data = array();
$query = 'SELECT * FROM `'.$table.'`';
if(count($where)) {
$query .= ' WHERE 1';
foreach ($where as $name => $value) {
$query .= " AND `$name` = ".esc($value);
}
}
$result = rquery($query);
$to_check = array();
while ($row = mysql_fetch_row($result)) {
$new_row = array();
$i = 0;
foreach ($row as $this_row) {
$field_info = mysql_fetch_field($result, $i);
$new_row[$field_info->name] = $this_row;
foreach ($foreign_keys as $this_foreign_key) {
if($this_foreign_key['reference_column'] == $field_info->name) {
$to_check[] = array('value' => $this_row, 'foreign_key' => array('table' => $this_foreign_key['table'], 'column' => $this_foreign_key['column'], 'database' => $this_foreign_key['database']));
}
}
$i++;
}
$GLOBALS['get_data_that_would_be_deleted'][$table][] = $new_row;
}
foreach ($to_check as $this_to_check) {
if(isset($this_to_check['value']) && !is_null($this_to_check['value'])) {
get_data_that_would_be_deleted($database, $this_to_check['foreign_key']['table'], array($this_to_check['foreign_key']['column'] => $this_to_check['value']), $recursion - 1);;
}
}
$data = $GLOBALS['get_data_that_would_be_deleted'];
return $data;
} else {
die("\$where needs to be an array with column_name => value pairs");
}
} else {
die("\$table was not defined!");
}
}
Предположим, у меня есть таблица под названием "table" в базе данных "db", и я хочу удалить ее с идентификатором 180, затем я бы назвал:
print(get_foreign_key_deleted_data_html('db', 'table', array('id' => 180)));
и он печатает полную таблицу со всеми строками и всеми значениями, которые будут удалены.
Но, как я уже сказал, это очень, очень быстрый и грязный хак. Я был бы рад за любой отчет об ошибке (и их, конечно же, много!).