mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-25 18:38:00 +03:00 
			
		
		
		
	Merge ahristov@bk-internal.mysql.com:/home/bk/mysql-5.1-maint
into example.com:/work/bug22369-v2/my51 mysql-test/r/alter_table.result: Auto merged mysql-test/r/grant.result: Auto merged mysql-test/t/grant.test: Auto merged sql/mysql_priv.h: Auto merged sql/sql_parse.cc: Auto merged sql/sql_table.cc: Auto merged mysql-test/t/alter_table.test: manual merge
This commit is contained in:
		| @@ -854,3 +854,45 @@ Table	Create Table | ||||
|   `c1` int(11) DEFAULT NULL | ||||
| ) ENGINE=MyISAM DEFAULT CHARSET=latin1 | ||||
| DROP TABLE `#sql2`, `@0023sql1`; | ||||
| DROP TABLE IF EXISTS t1; | ||||
| DROP TABLE IF EXISTS t2; | ||||
| CREATE TABLE t1 ( | ||||
| int_field INTEGER UNSIGNED NOT NULL, | ||||
| char_field CHAR(10), | ||||
| INDEX(`int_field`) | ||||
| ); | ||||
| DESCRIBE t1; | ||||
| Field	Type	Null	Key	Default	Extra | ||||
| int_field	int(10) unsigned	NO	MUL		 | ||||
| char_field	char(10)	YES		NULL	 | ||||
| SHOW INDEXES FROM t1; | ||||
| Table	Non_unique	Key_name	Seq_in_index	Column_name	Collation	Cardinality	Sub_part	Packed	Null	Index_type	Comment | ||||
| t1	1	int_field	1	int_field	A	NULL	NULL	NULL		BTREE	 | ||||
| INSERT INTO t1 VALUES (1, "edno"), (1, "edno"), (2, "dve"), (3, "tri"), (5, "pet"); | ||||
| "Non-copy data change - new frm, but old data and index files" | ||||
| ALTER TABLE t1 | ||||
| CHANGE int_field unsigned_int_field INTEGER UNSIGNED NOT NULL, | ||||
| RENAME t2; | ||||
| SELECT * FROM t1 ORDER BY int_field; | ||||
| ERROR 42S02: Table 'test.t1' doesn't exist | ||||
| SELECT * FROM t2 ORDER BY unsigned_int_field; | ||||
| unsigned_int_field	char_field | ||||
| 1	edno | ||||
| 1	edno | ||||
| 2	dve | ||||
| 3	tri | ||||
| 5	pet | ||||
| DESCRIBE t2; | ||||
| Field	Type	Null	Key	Default	Extra | ||||
| unsigned_int_field	int(10) unsigned	NO	MUL		 | ||||
| char_field	char(10)	YES		NULL	 | ||||
| DESCRIBE t2; | ||||
| Field	Type	Null	Key	Default	Extra | ||||
| unsigned_int_field	int(10) unsigned	NO	MUL		 | ||||
| char_field	char(10)	YES		NULL	 | ||||
| ALTER TABLE t2 MODIFY unsigned_int_field BIGINT UNSIGNED NOT NULL; | ||||
| DESCRIBE t2; | ||||
| Field	Type	Null	Key	Default	Extra | ||||
| unsigned_int_field	bigint(20) unsigned	NO	MUL		 | ||||
| char_field	char(10)	YES		NULL	 | ||||
| DROP TABLE t2; | ||||
|   | ||||
| @@ -948,6 +948,71 @@ DROP USER 'mysqltest_1'@'localhost'; | ||||
| use test; | ||||
| create user mysqltest1_thisisreallytoolong; | ||||
| ERROR HY000: String 'mysqltest1_thisisreallytoolong' is too long for user name (should be no longer than 16) | ||||
| CREATE DATABASE mysqltest1; | ||||
| CREATE TABLE mysqltest1.t1 ( | ||||
| int_field INTEGER UNSIGNED NOT NULL, | ||||
| char_field CHAR(10), | ||||
| INDEX(`int_field`) | ||||
| ); | ||||
| CREATE TABLE mysqltest1.t2 (int_field INT); | ||||
| "Now check that we require equivalent grants for " | ||||
| "RENAME TABLE and ALTER TABLE" | ||||
| CREATE USER mysqltest_1@localhost; | ||||
| GRANT SELECT ON mysqltest1.t1 TO mysqltest_1@localhost; | ||||
| SELECT USER(); | ||||
| USER() | ||||
| mysqltest_1@localhost | ||||
| SHOW GRANTS; | ||||
| Grants for mysqltest_1@localhost | ||||
| GRANT USAGE ON *.* TO 'mysqltest_1'@'localhost' | ||||
| GRANT SELECT ON `mysqltest1`.`t1` TO 'mysqltest_1'@'localhost' | ||||
| RENAME TABLE t1 TO t2; | ||||
| ERROR 42000: ALTER command denied to user 'mysqltest_1'@'localhost' for table 't1' | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| ERROR 42000: DROP,ALTER command denied to user 'mysqltest_1'@'localhost' for table 't1' | ||||
| GRANT DROP ON mysqltest1.t1 TO mysqltest_1@localhost; | ||||
| RENAME TABLE t1 TO t2; | ||||
| ERROR 42000: ALTER command denied to user 'mysqltest_1'@'localhost' for table 't1' | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| ERROR 42000: ALTER command denied to user 'mysqltest_1'@'localhost' for table 't1' | ||||
| GRANT ALTER ON mysqltest1.t1 TO mysqltest_1@localhost; | ||||
| SHOW GRANTS; | ||||
| Grants for mysqltest_1@localhost | ||||
| GRANT USAGE ON *.* TO 'mysqltest_1'@'localhost' | ||||
| GRANT SELECT, DROP, ALTER ON `mysqltest1`.`t1` TO 'mysqltest_1'@'localhost' | ||||
| RENAME TABLE t1 TO t2; | ||||
| ERROR 42000: INSERT,CREATE command denied to user 'mysqltest_1'@'localhost' for table 't2' | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| ERROR 42000: INSERT,CREATE command denied to user 'mysqltest_1'@'localhost' for table 't2' | ||||
| GRANT INSERT, CREATE ON mysqltest1.t1 TO mysqltest_1@localhost; | ||||
| SHOW GRANTS; | ||||
| Grants for mysqltest_1@localhost | ||||
| GRANT USAGE ON *.* TO 'mysqltest_1'@'localhost' | ||||
| GRANT SELECT, INSERT, CREATE, DROP, ALTER ON `mysqltest1`.`t1` TO 'mysqltest_1'@'localhost' | ||||
| GRANT INSERT, SELECT, CREATE, ALTER, DROP ON mysqltest1.t2 TO mysqltest_1@localhost; | ||||
| DROP TABLE mysqltest1.t2; | ||||
| SHOW GRANTS; | ||||
| Grants for mysqltest_1@localhost | ||||
| GRANT USAGE ON *.* TO 'mysqltest_1'@'localhost' | ||||
| GRANT SELECT, INSERT, CREATE, DROP, ALTER ON `mysqltest1`.`t2` TO 'mysqltest_1'@'localhost' | ||||
| GRANT SELECT, INSERT, CREATE, DROP, ALTER ON `mysqltest1`.`t1` TO 'mysqltest_1'@'localhost' | ||||
| RENAME TABLE t1 TO t2; | ||||
| RENAME TABLE t2 TO t1; | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| ALTER TABLE t2 RENAME TO t1; | ||||
| REVOKE DROP, INSERT ON mysqltest1.t1 FROM mysqltest_1@localhost; | ||||
| REVOKE DROP, INSERT ON mysqltest1.t2 FROM mysqltest_1@localhost; | ||||
| SHOW GRANTS; | ||||
| Grants for mysqltest_1@localhost | ||||
| GRANT USAGE ON *.* TO 'mysqltest_1'@'localhost' | ||||
| GRANT SELECT, CREATE, ALTER ON `mysqltest1`.`t2` TO 'mysqltest_1'@'localhost' | ||||
| GRANT SELECT, CREATE, ALTER ON `mysqltest1`.`t1` TO 'mysqltest_1'@'localhost' | ||||
| RENAME TABLE t1 TO t2; | ||||
| ERROR 42000: INSERT command denied to user 'mysqltest_1'@'localhost' for table 't2' | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| ERROR 42000: DROP command denied to user 'mysqltest_1'@'localhost' for table 't1' | ||||
| DROP USER mysqltest_1@localhost; | ||||
| DROP DATABASE mysqltest1; | ||||
| GRANT CREATE ON mysqltest.* TO 1234567890abcdefGHIKL@localhost; | ||||
| ERROR HY000: String '1234567890abcdefGHIKL' is too long for user name (should be no longer than 16) | ||||
| GRANT CREATE ON mysqltest.* TO some_user_name@1234567890abcdefghij1234567890abcdefghij1234567890abcdefghijQWERTY; | ||||
|   | ||||
| @@ -101,7 +101,7 @@ create table mysqltest.t1 (name char(15)); | ||||
| insert into mysqltest.t1 (name) values ("mysqltest"); | ||||
| select * from t1; | ||||
| select * from mysqltest.t1; | ||||
| --error 1050 | ||||
| --error ER_TABLE_EXISTS_ERROR | ||||
| alter table t1 rename mysqltest.t1; | ||||
| select * from t1; | ||||
| select * from mysqltest.t1; | ||||
| @@ -231,9 +231,9 @@ DROP TABLE t1; | ||||
| # BUG#4717 - check for valid table names | ||||
| # | ||||
| create table t1 (a int); | ||||
| --error 1103 | ||||
| --error ER_WRONG_TABLE_NAME | ||||
| alter table t1 rename to ``; | ||||
| --error 1103 | ||||
| --error ER_WRONG_TABLE_NAME | ||||
| rename table t1 to ``; | ||||
| drop table t1; | ||||
|  | ||||
| @@ -325,14 +325,14 @@ drop table t1; | ||||
| CREATE TABLE t1 (a int PRIMARY KEY, b INT UNIQUE); | ||||
| ALTER TABLE t1 DROP PRIMARY KEY; | ||||
| SHOW CREATE TABLE t1; | ||||
| --error 1091 | ||||
| --error ER_CANT_DROP_FIELD_OR_KEY | ||||
| ALTER TABLE t1 DROP PRIMARY KEY; | ||||
| DROP TABLE t1; | ||||
|  | ||||
| # BUG#3899 | ||||
| create table t1 (a int, b int, key(a)); | ||||
| insert into t1 values (1,1), (2,2); | ||||
| --error 1091 | ||||
| --error ER_CANT_DROP_FIELD_OR_KEY | ||||
| alter table t1 drop key no_such_key; | ||||
| alter table t1 drop key a; | ||||
| drop table t1; | ||||
| @@ -343,7 +343,7 @@ drop table t1; | ||||
| # Some platforms (Mac OS X, Windows) will send the error message using small letters. | ||||
| CREATE TABLE T12207(a int) ENGINE=MYISAM; | ||||
| --replace_result t12207 T12207 | ||||
| --error 1031 | ||||
| --error ER_ILLEGAL_HA | ||||
| ALTER TABLE T12207 DISCARD TABLESPACE; | ||||
| DROP TABLE T12207; | ||||
|  | ||||
| @@ -367,7 +367,7 @@ drop table t1; | ||||
| # shorter than packed field length. | ||||
| # | ||||
| create table t1 ( a timestamp ); | ||||
| --error 1089 | ||||
| --error ER_WRONG_SUB_KEY | ||||
| alter table t1 add unique ( a(1) ); | ||||
| drop table t1; | ||||
|  | ||||
| @@ -477,7 +477,7 @@ create table t1 (c1 int); | ||||
| # Move table to other database. | ||||
| alter table t1 rename mysqltest.t1; | ||||
| # Assure that it has moved. | ||||
| --error 1051 | ||||
| --error ER_BAD_TABLE_ERROR | ||||
| drop table t1; | ||||
| # Move table back. | ||||
| alter table mysqltest.t1 rename t1; | ||||
| @@ -491,7 +491,7 @@ use mysqltest; | ||||
| # Drop the current db. This de-selects any db. | ||||
| drop database mysqltest; | ||||
| # Now test for correct message. | ||||
| --error 1046 | ||||
| --error ER_NO_DB_ERROR | ||||
| alter table test.t1 rename t1; | ||||
| # Check that explicit qualifying works even with no selected db. | ||||
| alter table test.t1 rename test.t1; | ||||
| @@ -650,3 +650,39 @@ INSERT INTO `@0023sql1`  VALUES (2); | ||||
| SHOW CREATE TABLE `#sql2`; | ||||
| SHOW CREATE TABLE `@0023sql1`; | ||||
| DROP TABLE `#sql2`, `@0023sql1`; | ||||
|  | ||||
| # | ||||
| # Bug #22369: Alter table rename combined with other alterations causes lost tables | ||||
| # | ||||
| # This problem happens if the data change is compatible. | ||||
| # Changing to the same type is compatible for example. | ||||
| # | ||||
| --disable_warnings | ||||
| DROP TABLE IF EXISTS t1; | ||||
| DROP TABLE IF EXISTS t2; | ||||
| --enable_warnings | ||||
| CREATE TABLE t1 ( | ||||
|   int_field INTEGER UNSIGNED NOT NULL, | ||||
|   char_field CHAR(10), | ||||
|   INDEX(`int_field`) | ||||
| ); | ||||
|  | ||||
| DESCRIBE t1; | ||||
|  | ||||
| SHOW INDEXES FROM t1; | ||||
|  | ||||
| INSERT INTO t1 VALUES (1, "edno"), (1, "edno"), (2, "dve"), (3, "tri"), (5, "pet");  | ||||
| --echo "Non-copy data change - new frm, but old data and index files" | ||||
| ALTER TABLE t1 | ||||
|   CHANGE int_field unsigned_int_field INTEGER UNSIGNED NOT NULL, | ||||
|   RENAME t2; | ||||
|  | ||||
| --error ER_NO_SUCH_TABLE | ||||
| SELECT * FROM t1 ORDER BY int_field; | ||||
| SELECT * FROM t2 ORDER BY unsigned_int_field; | ||||
| DESCRIBE t2; | ||||
| DESCRIBE t2; | ||||
| ALTER TABLE t2 MODIFY unsigned_int_field BIGINT UNSIGNED NOT NULL; | ||||
| DESCRIBE t2; | ||||
|  | ||||
| DROP TABLE t2; | ||||
|   | ||||
| @@ -822,6 +822,82 @@ create user mysqltest1_thisisreallytoolong; | ||||
| # statements. | ||||
| # | ||||
|  | ||||
| # | ||||
| # Bug #22369: Alter table rename combined with other alterations causes lost tables | ||||
| # | ||||
| CREATE DATABASE mysqltest1; | ||||
| CREATE TABLE mysqltest1.t1 ( | ||||
|   int_field INTEGER UNSIGNED NOT NULL, | ||||
|   char_field CHAR(10), | ||||
|   INDEX(`int_field`) | ||||
| ); | ||||
| CREATE TABLE mysqltest1.t2 (int_field INT); | ||||
|  | ||||
| --echo "Now check that we require equivalent grants for " | ||||
| --echo "RENAME TABLE and ALTER TABLE" | ||||
| CREATE USER mysqltest_1@localhost; | ||||
| GRANT SELECT ON mysqltest1.t1 TO mysqltest_1@localhost; | ||||
|  | ||||
| --connect (conn42,localhost,mysqltest_1,,mysqltest1); | ||||
| SELECT USER(); | ||||
| SHOW GRANTS; | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| RENAME TABLE t1 TO t2; | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| --disconnect conn42 | ||||
| --connection default | ||||
| GRANT DROP ON mysqltest1.t1 TO mysqltest_1@localhost; | ||||
|  | ||||
| --connect (conn42,localhost,mysqltest_1,,mysqltest1); | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| RENAME TABLE t1 TO t2; | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| --disconnect conn42 | ||||
| --connection default | ||||
| GRANT ALTER ON mysqltest1.t1 TO mysqltest_1@localhost; | ||||
|  | ||||
| --connect (conn42,localhost,mysqltest_1,,mysqltest1); | ||||
| SHOW GRANTS; | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| RENAME TABLE t1 TO t2; | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| --disconnect conn42 | ||||
| --connection default | ||||
| GRANT INSERT, CREATE ON mysqltest1.t1 TO mysqltest_1@localhost; | ||||
| --connect (conn42,localhost,mysqltest_1,,mysqltest1); | ||||
| SHOW GRANTS; | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| --disconnect conn42 | ||||
| --connection default | ||||
| GRANT INSERT, SELECT, CREATE, ALTER, DROP ON mysqltest1.t2 TO mysqltest_1@localhost; | ||||
| DROP TABLE mysqltest1.t2; | ||||
|  | ||||
| --connect (conn42,localhost,mysqltest_1,,mysqltest1); | ||||
| SHOW GRANTS; | ||||
| RENAME TABLE t1 TO t2; | ||||
| RENAME TABLE t2 TO t1; | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| ALTER TABLE t2 RENAME TO t1; | ||||
| --disconnect conn42 | ||||
| --connection default | ||||
| REVOKE DROP, INSERT ON mysqltest1.t1 FROM mysqltest_1@localhost; | ||||
| REVOKE DROP, INSERT ON mysqltest1.t2 FROM mysqltest_1@localhost; | ||||
|  | ||||
| --connect (conn42,localhost,mysqltest_1,,mysqltest1); | ||||
| SHOW GRANTS; | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| RENAME TABLE t1 TO t2; | ||||
| --error ER_TABLEACCESS_DENIED_ERROR | ||||
| ALTER TABLE t1 RENAME TO t2; | ||||
| --disconnect conn42 | ||||
| --connection default | ||||
|  | ||||
| DROP USER mysqltest_1@localhost; | ||||
| DROP DATABASE mysqltest1; | ||||
|  | ||||
| # Working with database-level privileges. | ||||
|  | ||||
| --error ER_WRONG_STRING_LENGTH | ||||
|   | ||||
| @@ -1844,6 +1844,7 @@ uint build_table_filename(char *buff, size_t bufflen, const char *db, | ||||
| #define FN_FROM_IS_TMP  (1 << 0) | ||||
| #define FN_TO_IS_TMP    (1 << 1) | ||||
| #define FN_IS_TMP       (FN_FROM_IS_TMP | FN_TO_IS_TMP) | ||||
| #define NO_FRM_RENAME   (1 << 2) | ||||
|  | ||||
| /* from hostname.cc */ | ||||
| struct in_addr; | ||||
|   | ||||
| @@ -3139,8 +3139,11 @@ end_with_restore_list: | ||||
|     { | ||||
|       ulong priv=0; | ||||
|       ulong priv_needed= ALTER_ACL; | ||||
|       /* We also require DROP priv for ALTER TABLE ... DROP PARTITION */ | ||||
|       if (lex->alter_info.flags & ALTER_DROP_PARTITION) | ||||
|       /* | ||||
|         We also require DROP priv for ALTER TABLE ... DROP PARTITION, as well | ||||
|         as for RENAME TO, as being done by SQLCOM_RENAME_TABLE | ||||
|       */ | ||||
|       if (lex->alter_info.flags & (ALTER_DROP_PARTITION | ALTER_RENAME)) | ||||
|         priv_needed|= DROP_ACL; | ||||
|  | ||||
|       /* Must be set in the parser */ | ||||
|   | ||||
| @@ -3654,10 +3654,12 @@ make_unique_key_name(const char *field_name,KEY *start,KEY *end) | ||||
|       flags                     flags for build_table_filename(). | ||||
|                                 FN_FROM_IS_TMP old_name is temporary. | ||||
|                                 FN_TO_IS_TMP   new_name is temporary. | ||||
|                                 NO_FRM_RENAME  Don't rename the FRM file | ||||
|                                 but only the table in the storage engine. | ||||
|  | ||||
|   RETURN | ||||
|     0           OK | ||||
|     != 0        Error | ||||
|     FALSE   OK | ||||
|     TRUE    Error | ||||
| */ | ||||
|  | ||||
| bool | ||||
| @@ -3706,7 +3708,7 @@ mysql_rename_table(handlerton *base, const char *old_db, | ||||
|  | ||||
|   if (!file || !(error=file->rename_table(from_base, to_base))) | ||||
|   { | ||||
|     if (rename_file_ext(from,to,reg_ext)) | ||||
|     if (!(flags & NO_FRM_RENAME) && rename_file_ext(from,to,reg_ext)) | ||||
|     { | ||||
|       error=my_errno; | ||||
|       /* Restore old file name */ | ||||
| @@ -5246,6 +5248,51 @@ bool alter_table_manage_keys(TABLE *table, int indexes_were_disabled, | ||||
|  | ||||
| /* | ||||
|   Alter table | ||||
|  | ||||
|   SYNOPSIS | ||||
|     mysql_alter_table() | ||||
|       thd              Thread handle | ||||
|       new_db           If there is a RENAME clause | ||||
|       new_name         If there is a RENAME clause | ||||
|       lex_create_info  Information from the parsing phase. Since some | ||||
|                        clauses are common to CREATE and ALTER TABLE, the | ||||
|                        data is stored in lex->create_info. The non-common | ||||
|                        is stored in lex->alter_info. | ||||
|       table_list       The table to change. | ||||
|       fields           lex->create_list - List of fields to be changed, | ||||
|                        added or dropped. | ||||
|       keys             lex->key_list - List of keys to be changed, added or | ||||
|                        dropped. | ||||
|       order_num        How many ORDER BY fields has been specified. | ||||
|       order            List of fields to ORDER BY. | ||||
|       ignore           Whether we have ALTER IGNORE TABLE | ||||
|       alter_info       Information from the parsing phase specific to ALTER | ||||
|                        TABLE and not shared with CREATE TABLE. | ||||
|       do_send_ok       Whether to call send_ok() on success. | ||||
|  | ||||
|   DESCRIPTION | ||||
|     This is a veery long function and is everything but the kitchen sink :) | ||||
|     It is used to alter a table and not only by ALTER TABLE but also | ||||
|     CREATE|DROP INDEX are mapped on this function. | ||||
|  | ||||
|     When the ALTER TABLE statement just does a RENAME or ENABLE|DISABLE KEYS, | ||||
|     or both, then this function short cuts its operation by renaming | ||||
|     the table and/or enabling/disabling the keys. In this case, the FRM is | ||||
|     not changed, directly by mysql_alter_table. However, if there is a | ||||
|     RENAME + change of a field, or an index, the short cut is not used. | ||||
|     See how `fields` is used to generate the new FRM regarding the structure | ||||
|     of the fields. The same is done for the indices of the table. | ||||
|  | ||||
|     Important is the fact, that this function tries to do as little work as | ||||
|     possible, by finding out whether a intermediate table is needed to copy | ||||
|     data into and when finishing the altering to use it as the original table. | ||||
|     For this reason the function compare_tables() is called, which decides | ||||
|     based on all kind of data how similar are the new and the original | ||||
|     tables.  | ||||
|  | ||||
|   RETURN VALUES | ||||
|     FALSE  OK | ||||
|     TRUE   Error | ||||
| */ | ||||
|  | ||||
| bool mysql_alter_table(THD *thd,char *new_db, char *new_name, | ||||
| @@ -5264,7 +5311,7 @@ bool mysql_alter_table(THD *thd,char *new_db, char *new_name, | ||||
|   char reg_path[FN_REFLEN+1]; | ||||
|   ha_rows copied,deleted; | ||||
|   uint db_create_options, used_fields; | ||||
|   handlerton *old_db_type, *new_db_type; | ||||
|   handlerton *old_db_type, *new_db_type, *save_old_db_type; | ||||
|   legacy_db_type table_type; | ||||
|   HA_CREATE_INFO *create_info; | ||||
|   frm_type_enum frm_type; | ||||
| @@ -5568,7 +5615,7 @@ view_err: | ||||
|     DBUG_RETURN(error); | ||||
|   } | ||||
|  | ||||
|   /* Full alter table */ | ||||
|   /* We have to do full alter table */ | ||||
|  | ||||
|   /* Let new create options override the old ones */ | ||||
|   if (!(used_fields & HA_CREATE_USED_MIN_ROWS)) | ||||
| @@ -6087,8 +6134,8 @@ view_err: | ||||
|       old data and index files.  Create also symlinks to point at | ||||
|       the new tables. | ||||
|       Copy data. | ||||
|       At end, rename temporary tables and symlinks to temporary table | ||||
|       to final table name. | ||||
|       At end, rename intermediate tables, and symlinks to intermediate | ||||
|       table, to final table name. | ||||
|       Remove old table and old symlinks | ||||
|  | ||||
|     If rename is made to another database: | ||||
| @@ -6149,6 +6196,7 @@ view_err: | ||||
|       /* table is a normal table: Create temporary table in same directory */ | ||||
|       build_table_filename(path, sizeof(path), new_db, tmp_name, "", | ||||
|                            FN_IS_TMP); | ||||
|       /* Open our intermediate table */ | ||||
|       new_table=open_temporary_table(thd, path, new_db, tmp_name,0); | ||||
|     } | ||||
|     if (!new_table) | ||||
| @@ -6365,7 +6413,7 @@ view_err: | ||||
|  | ||||
|   if (new_table) | ||||
|   { | ||||
|     /* close temporary table that will be the new table */ | ||||
|     /* Close the intermediate table that will be the new table */ | ||||
|     intern_close_table(new_table); | ||||
|     my_free((gptr) new_table,MYF(0)); | ||||
|   } | ||||
| @@ -6379,7 +6427,7 @@ view_err: | ||||
|  | ||||
|   /* | ||||
|     Data is copied.  Now we rename the old table to a temp name, | ||||
|     rename the new one to the old name, remove all entries from the old table | ||||
|     rename the new one to the old name, remove all entries about the old table | ||||
|     from the cache, free all locks, close the old table and remove it. | ||||
|   */ | ||||
|  | ||||
| @@ -6406,7 +6454,7 @@ view_err: | ||||
|   { | ||||
|     /* | ||||
|       Win32 and InnoDB can't drop a table that is in use, so we must | ||||
|       close the original table at before doing the rename | ||||
|       close the original table before doing the rename | ||||
|     */ | ||||
|     table->s->version= 0;                	// Force removal of table def | ||||
|     close_cached_table(thd, table); | ||||
| @@ -6420,6 +6468,21 @@ view_err: | ||||
|  | ||||
|  | ||||
|   error=0; | ||||
|   save_old_db_type= old_db_type; | ||||
|  | ||||
|   /* | ||||
|     This leads to the storage engine (SE) not being notified for renames in | ||||
|     mysql_rename_table(), because we just juggle with the FRM and nothing | ||||
|     more. If we have an intermediate table, then we notify the SE that | ||||
|     it should become the actual table. Later, we will recycle the old table. | ||||
|     However, in case of ALTER TABLE RENAME there might be no intermediate | ||||
|     table. This is when the old and new tables are compatible, according to | ||||
|     compare_table(). Then, we need one additional call to | ||||
|     mysql_rename_table() with flag NO_FRM_RENAME, which does nothing else but | ||||
|     actual rename in the SE and the FRM is not touched. Note that, if the | ||||
|     table is renamed and the SE is also changed, then an intermediate table | ||||
|     is created and the additional call will not take place. | ||||
|   */ | ||||
|   if (!need_copy_table) | ||||
|     new_db_type=old_db_type= NULL; // this type cannot happen in regular ALTER | ||||
|   if (mysql_rename_table(old_db_type, db, table_name, db, old_name, | ||||
| @@ -6429,8 +6492,11 @@ view_err: | ||||
|     VOID(quick_rm_table(new_db_type, new_db, tmp_name, FN_IS_TMP)); | ||||
|   } | ||||
|   else if (mysql_rename_table(new_db_type,new_db,tmp_name,new_db, | ||||
| 			      new_alias, FN_FROM_IS_TMP) || | ||||
|                               new_alias, FN_FROM_IS_TMP) || | ||||
|            (new_name != table_name || new_db != db) && // we also do rename | ||||
|            (need_copy_table || | ||||
|             mysql_rename_table(save_old_db_type, db, table_name, new_db, | ||||
|                                new_alias, NO_FRM_RENAME)) && | ||||
|            Table_triggers_list::change_table_name(thd, db, table_name, | ||||
|                                                   new_db, new_alias)) | ||||
|   { | ||||
| @@ -6441,6 +6507,7 @@ view_err: | ||||
|     VOID(mysql_rename_table(old_db_type, db, old_name, db, alias, | ||||
|                             FN_FROM_IS_TMP)); | ||||
|   } | ||||
|  | ||||
|   if (error) | ||||
|   { | ||||
|     /* | ||||
| @@ -6472,6 +6539,7 @@ view_err: | ||||
|       goto err; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   if (thd->lock || new_name != table_name || no_table_reopen)  // True if WIN32 | ||||
|   { | ||||
|     /* | ||||
| @@ -6538,10 +6606,7 @@ view_err: | ||||
|   DBUG_ASSERT(!(mysql_bin_log.is_open() && thd->current_stmt_binlog_row_based && | ||||
|                 (create_info->options & HA_LEX_CREATE_TMP_TABLE))); | ||||
|   write_bin_log(thd, TRUE, thd->query, thd->query_length); | ||||
|   /* | ||||
|     TODO RONM: This problem needs to handled for Berkeley DB partitions | ||||
|     as well | ||||
|   */ | ||||
|  | ||||
|   if (ha_check_storage_engine_flag(old_db_type,HTON_FLUSH_AFTER_RENAME)) | ||||
|   { | ||||
|     /* | ||||
|   | ||||
		Reference in New Issue
	
	Block a user