26
26
#include <drizzled/sql_lex.h>
27
27
#include <drizzled/session.h>
28
28
#include <drizzled/sql_base.h>
29
#include <drizzled/strfunc.h>
29
#include "drizzled/strfunc.h"
30
#include <drizzled/db.h>
30
31
#include <drizzled/lock.h>
31
32
#include <drizzled/unireg.h>
32
33
#include <drizzled/item/int.h>
33
34
#include <drizzled/item/empty_string.h>
34
#include <drizzled/transaction_services.h>
35
#include <drizzled/transaction_services.h>
35
#include <drizzled/replication_services.h>
36
#include "drizzled/transaction_services.h"
36
37
#include <drizzled/table_proto.h>
37
38
#include <drizzled/plugin/client.h>
38
#include <drizzled/identifier.h>
39
#include <drizzled/internal/m_string.h>
40
#include <drizzled/global_charset_info.h>
41
#include <drizzled/charset.h>
43
#include <drizzled/definition/cache.h>
45
#include <drizzled/statement/alter_table.h>
46
#include <drizzled/sql_table.h>
47
#include <drizzled/pthread_globals.h>
48
#include <drizzled/typelib.h>
49
#include <drizzled/plugin/storage_engine.h>
39
#include <drizzled/table_identifier.h>
40
#include "drizzled/internal/m_string.h"
41
#include "drizzled/global_charset_info.h"
42
#include "drizzled/charset.h"
45
#include "drizzled/statement/alter_table.h"
46
#include "drizzled/sql_table.h"
47
#include "drizzled/pthread_globals.h"
51
49
#include <algorithm>
54
#include <boost/unordered_set.hpp>
56
52
using namespace std;
61
bool is_primary_key(KeyInfo *key_info)
57
extern plugin::StorageEngine *myisam_engine;
58
extern pid_t current_pid;
60
bool is_primary_key(KEY *key_info)
63
62
static const char * primary_key_name="PRIMARY";
64
63
return (strcmp(key_info->name, primary_key_name)==0);
108
void write_bin_log(Session *session, const std::string &query)
110
TransactionServices &transaction_services= TransactionServices::singleton();
111
transaction_services.rawStatement(*session, query);
106
void write_bin_log(Session *session,
109
ReplicationServices &replication_services= ReplicationServices::singleton();
110
replication_services.rawStatement(session, query);
114
/* Should should be refactored to go away */
115
void write_bin_log_drop_table(Session *session, bool if_exists, const char *db_name, const char *table_name)
117
ReplicationServices &replication_services= ReplicationServices::singleton();
121
built_query.append("DROP TABLE IF EXISTS ");
123
built_query.append("DROP TABLE ");
125
built_query.append("`");
126
if (session->db.empty() || strcmp(db_name, session->db.c_str()) != 0)
128
built_query.append(db_name);
129
built_query.append("`.`");
132
built_query.append(table_name);
133
built_query.append("`");
134
replication_services.rawStatement(session, built_query);
115
138
Execute the drop of a normal or temporary table
141
mysql_rm_table_part2()
119
142
session Thread Cursor
120
143
tables Tables to drop
121
144
if_exists If set, don't give an error if table doesn't exists.
122
145
In this case we give an warning of level 'NOTE'
123
146
drop_temporary Only drop temporary tables
126
149
When logging to the binary log, we should log
127
150
tmp_tables and transactional tables as separate statements if we
128
151
are in a transaction; This is needed to get these tables into the
138
161
-1 Thread was killed
141
int rm_table_part2(Session *session, TableList *tables, bool if_exists,
164
int mysql_rm_table_part2(Session *session, TableList *tables, bool if_exists,
144
167
TableList *table;
145
util::string::vector wrong_tables;
147
170
bool foreign_key_error= false;
151
boost::mutex::scoped_lock scopedLock(table::Cache::singleton().mutex());
153
if (not drop_temporary && session->lock_table_names_exclusively(tables))
158
/* Don't give warnings for not found errors, as we already generate notes */
159
session->no_warnings_for_error= 1;
161
for (table= tables; table; table= table->next_local)
163
identifier::Table tmp_identifier(table->getSchemaName(), table->getTableName());
165
error= session->drop_temporary_table(tmp_identifier);
169
// removed temporary table
172
pthread_mutex_lock(&LOCK_open); /* Part 2 of rm a table */
175
If we have the table in the definition cache, we don't have to check the
176
.frm cursor to find if the table is a normal table (not view) and what
180
for (table= tables; table; table= table->next_local)
183
table->db_type= NULL;
184
if ((share= TableShare::getShare(table->db, table->table_name)))
185
table->db_type= share->db_type();
188
if (not drop_temporary && lock_table_names_exclusively(session, tables))
190
pthread_mutex_unlock(&LOCK_open);
194
/* Don't give warnings for not found errors, as we already generate notes */
195
session->no_warnings_for_error= 1;
197
for (table= tables; table; table= table->next_local)
201
error= session->drop_temporary_table(table);
205
// removed temporary table
209
goto err_with_placeholders;
211
// temporary table not found
215
if (drop_temporary == false)
218
abort_locked_tables(session, db, table->table_name);
219
remove_table_from_cache(session, db, table->table_name,
220
RTFC_WAIT_OTHER_THREAD_FLAG |
221
RTFC_CHECK_KILLED_FLAG);
223
If the table was used in lock tables, remember it so that
224
unlock_table_names can free it
226
if ((locked_table= drop_locked_tables(session, db, table->table_name)))
227
table->table= locked_table;
232
goto err_with_placeholders;
235
TableIdentifier identifier(db, table->table_name, table->internal_tmp_table ? INTERNAL_TMP_TABLE : STANDARD_TABLE);
237
if (drop_temporary || not plugin::StorageEngine::doesTableExist(*session, identifier))
239
// Table was not found on disk and table can't be created from engine
241
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
242
ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR),
175
// temporary table not found
249
error= plugin::StorageEngine::dropTable(*session, identifier);
251
if ((error == ENOENT || error == HA_ERR_NO_SUCH_TABLE) && if_exists)
179
if (drop_temporary == false)
182
abort_locked_tables(session, tmp_identifier);
183
table::Cache::singleton().removeTable(session, tmp_identifier,
184
RTFC_WAIT_OTHER_THREAD_FLAG |
185
RTFC_CHECK_KILLED_FLAG);
187
If the table was used in lock tables, remember it so that
188
unlock_table_names can free it
190
if ((locked_table= drop_locked_tables(session, tmp_identifier)))
191
table->table= locked_table;
193
if (session->getKilled())
199
identifier::Table identifier(table->getSchemaName(), table->getTableName(), table->getInternalTmpTable() ? message::Table::INTERNAL : message::Table::STANDARD);
201
message::table::shared_ptr message= plugin::StorageEngine::getTableMessage(*session, identifier, true);
203
if (drop_temporary || not plugin::StorageEngine::doesTableExist(*session, identifier))
205
// Table was not found on disk and table can't be created from engine
208
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
209
ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR),
210
table->getTableName());
219
drizzled::error_t local_error;
221
/* Generate transaction event ONLY when we successfully drop */
222
if (plugin::StorageEngine::dropTable(*session, identifier, local_error))
224
if (message) // If we have no definition, we don't know if the table should have been replicated
226
TransactionServices &transaction_services= TransactionServices::singleton();
227
transaction_services.dropTable(*session, identifier, *message, if_exists);
232
if (local_error == HA_ERR_NO_SUCH_TABLE and if_exists)
235
session->clear_error();
238
if (local_error == HA_ERR_ROW_IS_REFERENCED)
240
/* the table is referenced by a foreign key constraint */
241
foreign_key_error= true;
249
wrong_tables.push_back(table->getTableName());
253
tables->unlock_table_names();
257
if (wrong_tables.size())
254
session->clear_error();
257
if (error == HA_ERR_ROW_IS_REFERENCED)
259
/* the table is referenced by a foreign key constraint */
260
foreign_key_error= true;
264
if (error == 0 || (if_exists && foreign_key_error == false))
266
ReplicationServices &replication_services= ReplicationServices::singleton();
267
replication_services.dropTable(session, string(db), string(table->table_name), if_exists);
272
if (wrong_tables.length())
273
wrong_tables.append(',');
274
wrong_tables.append(String(table->table_name,system_charset_info));
278
It's safe to unlock LOCK_open: we have an exclusive lock
281
pthread_mutex_unlock(&LOCK_open);
283
if (wrong_tables.length())
259
if (not foreign_key_error)
261
std::string table_error;
263
for (util::string::vector::iterator iter= wrong_tables.begin();
264
iter != wrong_tables.end();
270
table_error.resize(table_error.size() -1);
285
if (!foreign_key_error)
272
286
my_printf_error(ER_BAD_TABLE_ERROR, ER(ER_BAD_TABLE_ERROR), MYF(0),
273
table_error.c_str());
287
wrong_tables.c_ptr());
277
290
my_message(ER_ROW_IS_REFERENCED, ER(ER_ROW_IS_REFERENCED), MYF(0));
508
514
(*timestamps_with_niladic)++;
512
517
sql_field->unireg_check= Field::NONE;
515
519
else if (sql_field->unireg_check != Field::NONE)
517
520
(*timestamps_with_niladic)++;
524
case DRIZZLE_TYPE_BOOLEAN:
525
case DRIZZLE_TYPE_DATE: // Rest of string types
526
case DRIZZLE_TYPE_DATETIME:
527
case DRIZZLE_TYPE_DECIMAL:
528
case DRIZZLE_TYPE_DOUBLE:
529
case DRIZZLE_TYPE_LONG:
530
case DRIZZLE_TYPE_LONGLONG:
531
case DRIZZLE_TYPE_NULL:
532
case DRIZZLE_TYPE_TIME:
533
case DRIZZLE_TYPE_UUID:
534
case DRIZZLE_TYPE_VARCHAR:
525
sql_field->pack_flag=(0 |
526
f_settype((uint32_t) sql_field->sql_type));
541
static int prepare_create_table(Session *session,
542
HA_CREATE_INFO *create_info,
543
message::Table &create_proto,
544
AlterInfo *alter_info,
546
uint32_t *db_options,
547
KeyInfo **key_info_buffer,
549
int select_field_count)
532
static int mysql_prepare_create_table(Session *session,
533
HA_CREATE_INFO *create_info,
534
message::Table &create_proto,
535
AlterInfo *alter_info,
537
uint32_t *db_options,
538
KEY **key_info_buffer,
540
int select_field_count)
551
542
const char *key_name;
552
543
CreateField *sql_field,*dup_field;
553
544
uint field,null_fields,blob_columns,max_key_length;
554
545
ulong record_offset= 0;
556
KeyPartInfo *key_part_info;
547
KEY_PART_INFO *key_part_info;
557
548
int timestamps= 0, timestamps_with_niladic= 0;
559
550
int select_field_pos,auto_increment=0;
560
List<CreateField>::iterator it(alter_info->create_list.begin());
561
List<CreateField>::iterator it2(alter_info->create_list.begin());
551
List_iterator<CreateField> it(alter_info->create_list);
552
List_iterator<CreateField> it2(alter_info->create_list);
562
553
uint32_t total_uneven_bit_length= 0;
564
555
plugin::StorageEngine *engine= plugin::StorageEngine::findByName(create_proto.engine().name());
1200
1176
key_info->name=(char*) key_name;
1204
1179
if (!key_info->name || check_column_name(key_info->name))
1206
1181
my_error(ER_WRONG_NAME_FOR_INDEX, MYF(0), key_info->name);
1210
1184
if (!(key_info->flags & HA_NULL_PART_KEY))
1215
1186
key_info->key_length=(uint16_t) key_length;
1217
1187
if (key_length > max_key_length)
1219
1189
my_error(ER_TOO_LONG_KEY,MYF(0),max_key_length);
1226
1194
if (!unique_key && !primary_key &&
1227
1195
(engine->check_flag(HTON_BIT_REQUIRE_PRIMARY_KEY)))
1229
1197
my_message(ER_REQUIRES_PRIMARY_KEY, ER(ER_REQUIRES_PRIMARY_KEY), MYF(0));
1233
1200
if (auto_increment > 0)
1235
1202
my_message(ER_WRONG_AUTO_KEY, ER(ER_WRONG_AUTO_KEY), MYF(0));
1238
1205
/* Sort keys in optimized order */
1239
internal::my_qsort((unsigned char*) *key_info_buffer, *key_count, sizeof(KeyInfo),
1206
internal::my_qsort((unsigned char*) *key_info_buffer, *key_count, sizeof(KEY),
1240
1207
(qsort_cmp) sort_keys);
1242
1209
/* Check fields. */
1243
it= alter_info->create_list.begin();
1244
1211
while ((sql_field=it++))
1246
1213
Field::utype type= (Field::utype) MTYP_TYPENR(sql_field->unireg_check);
1248
1215
if (session->variables.sql_mode & MODE_NO_ZERO_DATE &&
1249
1216
!sql_field->def &&
1250
(sql_field->sql_type == DRIZZLE_TYPE_TIMESTAMP or sql_field->sql_type == DRIZZLE_TYPE_MICROTIME) &&
1217
sql_field->sql_type == DRIZZLE_TYPE_TIMESTAMP &&
1251
1218
(sql_field->flags & NOT_NULL_FLAG) &&
1252
1219
(type == Field::NONE || type == Field::TIMESTAMP_UN_FIELD))
1313
static bool locked_create_event(Session *session,
1314
const identifier::Table &identifier,
1315
HA_CREATE_INFO *create_info,
1316
message::Table &table_proto,
1317
AlterInfo *alter_info,
1318
bool is_if_not_exists,
1319
bool internal_tmp_table,
1322
KeyInfo *key_info_buffer)
1329
@note if we are building a temp table we need to check to see if a temp table
1330
already exists, otherwise we just need to find out if a normal table exists (aka it is fine
1331
to create a table under a temporary table.
1334
plugin::StorageEngine::doesTableExist(*session, identifier,
1335
identifier.getType() != message::Table::STANDARD );
1339
if (is_if_not_exists)
1342
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1343
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1344
identifier.getTableName().c_str());
1345
create_info->table_existed= 1; // Mark that table existed
1349
my_error(ER_TABLE_EXISTS_ERROR, identifier);
1354
if (identifier.getType() == message::Table::STANDARD) // We have a real table
1357
We don't assert here, but check the result, because the table could be
1358
in the table definition cache and in the same time the .frm could be
1359
missing from the disk, in case of manual intervention which deletes
1360
the .frm cursor. The user has to use FLUSH TABLES; to clear the cache.
1361
Then she could create the table. This case is pretty obscure and
1362
therefore we don't introduce a new error message only for it.
1365
@todo improve this error condition.
1367
if (definition::Cache::singleton().find(identifier.getKey()))
1369
my_error(ER_TABLE_EXISTS_ERROR, identifier);
1376
session->set_proc_info("creating table");
1377
create_info->table_existed= 0; // Mark that table is created
1379
create_info->table_options= db_options;
1381
if (not rea_create_table(session, identifier,
1383
create_info, alter_info->create_list,
1384
key_count, key_info_buffer))
1389
if (identifier.getType() == message::Table::TEMPORARY)
1391
/* Open table and put in temporary table list */
1392
if (not (session->open_temporary_table(identifier)))
1394
(void) session->rm_temporary_table(identifier);
1400
We keep this behind the lock to make sure ordering is correct for a table.
1401
This is a very unlikely problem where before we would write out to the
1402
trans log, someone would do a delete/create operation.
1405
if (table_proto.type() == message::Table::STANDARD && not internal_tmp_table)
1407
TransactionServices &transaction_services= TransactionServices::singleton();
1408
transaction_services.createTable(*session, table_proto);
1416
1282
Ignore the name of this function... it locks :(
1465
assert(identifier.getTableName() == table_proto.name());
1334
assert(strcmp(identifier.getTableName(), table_proto.name().c_str())==0);
1466
1335
db_options= create_info->table_options;
1468
set_table_default_charset(create_info, identifier.getSchemaName().c_str());
1470
/* Build a Table object to pass down to the engine, and the do the actual create. */
1471
if (not prepare_create_table(session, create_info, table_proto, alter_info,
1474
&key_info_buffer, &key_count,
1475
select_field_count))
1477
boost_unique_lock_t lock(table::Cache::singleton().mutex()); /* CREATE TABLE (some confussion on naming, double check) */
1478
error= locked_create_event(session,
1485
db_options, key_count,
1489
session->set_proc_info("After create");
1495
@note the following two methods implement create [temporary] table.
1497
static bool drizzle_create_table(Session *session,
1498
const identifier::Table &identifier,
1499
HA_CREATE_INFO *create_info,
1500
message::Table &table_proto,
1501
AlterInfo *alter_info,
1502
bool internal_tmp_table,
1503
uint32_t select_field_count,
1504
bool is_if_not_exists)
1506
Table *name_lock= NULL;
1509
if (session->lock_table_name_if_not_cached(identifier, &name_lock))
1513
else if (name_lock == NULL)
1337
if (create_info->row_type == ROW_TYPE_DYNAMIC)
1338
db_options|=HA_OPTION_PACK_RECORD;
1340
set_table_default_charset(create_info, identifier.getDBName());
1342
/* Check if table exists */
1343
if (mysql_prepare_create_table(session, create_info, table_proto, alter_info,
1346
&key_info_buffer, &key_count,
1347
select_field_count))
1350
/* Check if table already exists */
1351
if (lex_identified_temp_table &&
1352
session->find_temporary_table(identifier.getDBName(), identifier.getTableName()))
1515
1354
if (is_if_not_exists)
1356
create_info->table_existed= 1; // Mark that table existed
1517
1357
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1518
1358
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1519
identifier.getTableName().c_str());
1520
create_info->table_existed= 1;
1525
my_error(ER_TABLE_EXISTS_ERROR, identifier);
1531
result= create_table_no_lock(session,
1543
boost_unique_lock_t lock(table::Cache::singleton().mutex()); /* Lock for removing name_lock during table create */
1544
session->unlink_open_table(name_lock);
1359
identifier.getTableName());
1363
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1367
pthread_mutex_lock(&LOCK_open); /* CREATE TABLE (some confussion on naming, double check) */
1368
if (not internal_tmp_table && not lex_identified_temp_table)
1370
if (plugin::StorageEngine::doesTableExist(*session,
1371
identifier, false)==EEXIST)
1373
if (is_if_not_exists)
1376
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1377
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1378
identifier.getTableName());
1379
create_info->table_existed= 1; // Mark that table existed
1383
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1386
goto unlock_and_end;
1389
We don't assert here, but check the result, because the table could be
1390
in the table definition cache and in the same time the .frm could be
1391
missing from the disk, in case of manual intervention which deletes
1392
the .frm cursor. The user has to use FLUSH TABLES; to clear the cache.
1393
Then she could create the table. This case is pretty obscure and
1394
therefore we don't introduce a new error message only for it.
1396
if (TableShare::getShare(identifier.getDBName(), identifier.getTableName()))
1398
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1399
goto unlock_and_end;
1404
Check that table with given name does not already
1405
exist in any storage engine. In such a case it should
1406
be discovered and the error ER_TABLE_EXISTS_ERROR be returned
1407
unless user specified CREATE TABLE IF EXISTS
1408
The LOCK_open mutex has been locked to make sure no
1409
one else is attempting to discover the table. Since
1410
it's not on disk as a frm cursor, no one could be using it!
1412
if (not lex_identified_temp_table)
1414
bool exists= plugin::StorageEngine::doesTableExist(*session, identifier, false);
1418
if (is_if_not_exists)
1421
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1422
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1423
identifier.getTableName());
1424
create_info->table_existed= 1; // Mark that table existed
1425
goto unlock_and_end;
1428
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1429
goto unlock_and_end;
1433
session->set_proc_info("creating table");
1434
create_info->table_existed= 0; // Mark that table is created
1436
create_info->table_options= db_options;
1438
if (rea_create_table(session, identifier,
1440
create_info, alter_info->create_list,
1441
key_count, key_info_buffer))
1443
goto unlock_and_end;
1446
if (lex_identified_temp_table)
1448
/* Open table and put in temporary table list */
1449
if (not (session->open_temporary_table(identifier)))
1451
(void) session->rm_temporary_table(identifier);
1452
goto unlock_and_end;
1456
if (not internal_tmp_table && not lex_identified_temp_table)
1458
ReplicationServices &replication_services= ReplicationServices::singleton();
1459
replication_services.createTable(session, table_proto);
1463
pthread_mutex_unlock(&LOCK_open);
1466
session->set_proc_info("After create");
1552
Database locking aware wrapper for create_table_no_lock(),
1473
Database locking aware wrapper for mysql_create_table_no_lock(),
1554
bool create_table(Session *session,
1555
const identifier::Table &identifier,
1476
bool mysql_create_table(Session *session,
1477
TableIdentifier &identifier,
1556
1478
HA_CREATE_INFO *create_info,
1557
1479
message::Table &table_proto,
1558
1480
AlterInfo *alter_info,
1560
1482
uint32_t select_field_count,
1561
1483
bool is_if_not_exists)
1563
if (identifier.isTmp())
1565
return create_table_no_lock(session,
1575
return drizzle_create_table(session,
1485
Table *name_lock= NULL;
1487
bool lex_identified_temp_table=
1488
(table_proto.type() == message::Table::TEMPORARY);
1490
if (not lex_identified_temp_table)
1492
if (session->lock_table_name_if_not_cached(identifier.getDBName(),
1493
identifier.getTableName(),
1499
if (name_lock == NULL)
1501
if (is_if_not_exists)
1503
push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1504
ER_TABLE_EXISTS_ERROR, ER(ER_TABLE_EXISTS_ERROR),
1505
identifier.getTableName());
1506
create_info->table_existed= 1;
1511
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), identifier.getTableName());
1518
result= mysql_create_table_no_lock(session,
1530
pthread_mutex_lock(&LOCK_open); /* Lock for removing name_lock during table create */
1531
session->unlink_open_table(name_lock);
1532
pthread_mutex_unlock(&LOCK_open);
1649
rename_table(Session &session,
1650
plugin::StorageEngine *base,
1651
const identifier::Table &from,
1652
const identifier::Table &to)
1604
mysql_rename_table(plugin::StorageEngine *base, const char *old_db,
1605
const char *old_name, const char *new_db,
1606
const char *new_name, uint32_t flags)
1608
Session *session= current_session;
1609
char from[FN_REFLEN], to[FN_REFLEN];
1610
char *from_base= from, *to_base= to;
1658
if (not plugin::StorageEngine::doesSchemaExist(to))
1615
build_table_filename(from, sizeof(from), old_db, old_name,
1616
flags & FN_FROM_IS_TMP);
1617
build_table_filename(to, sizeof(to), new_db, new_name,
1618
flags & FN_TO_IS_TMP);
1620
if (!(error= base->renameTable(session, from_base, to_base)))
1660
my_error(ER_NO_DB_ERROR, MYF(0), to.getSchemaName().c_str());
1622
if (base->check_flag(HTON_BIT_HAS_DATA_DICTIONARY) == 0
1623
&& rename_table_proto_file(from_base, to_base))
1626
base->renameTable(session, to_base, from_base);
1664
error= base->renameTable(session, from, to);
1666
1630
if (error == HA_ERR_WRONG_COMMAND)
1668
1631
my_error(ER_NOT_SUPPORTED_YET, MYF(0), "ALTER Table");
1670
1632
else if (error)
1672
std::string from_path;
1673
std::string to_path;
1675
from.getSQLPath(from_path);
1676
to.getSQLPath(to_path);
1678
const char *from_identifier= from.isTmp() ? "#sql-temporary" : from_path.c_str();
1679
const char *to_identifier= to.isTmp() ? "#sql-temporary" : to_path.c_str();
1681
my_error(ER_ERROR_ON_RENAME, MYF(0), from_identifier, to_identifier, error);
1684
return error ? true : false;
1633
my_error(ER_ERROR_ON_RENAME, MYF(0), from, to, error);
1846
1797
char buff[FN_REFLEN + DRIZZLE_ERRMSG_SIZE];
1847
1798
uint32_t length;
1848
session->getClient()->store(table_name.c_str());
1849
session->getClient()->store(operator_name);
1850
session->getClient()->store(STRING_WITH_LEN("error"));
1799
session->client->store(table_name);
1800
session->client->store(operator_name);
1801
session->client->store(STRING_WITH_LEN("error"));
1851
1802
length= snprintf(buff, sizeof(buff), ER(ER_OPEN_AS_READONLY),
1852
table_name.c_str());
1853
session->getClient()->store(buff, length);
1854
transaction_services.autocommitOrRollback(*session, false);
1804
session->client->store(buff, length);
1805
transaction_services.ha_autocommit_or_rollback(session, false);
1855
1806
session->endTransaction(COMMIT);
1856
1807
session->close_thread_tables();
1857
session->getLex()->reset_query_tables_list(false);
1808
lex->reset_query_tables_list(false);
1858
1809
table->table=0; // For query cache
1859
if (session->getClient()->flush())
1810
if (session->client->flush())
1864
1815
/* Close all instances of the table to allow repair to rename files */
1865
if (lock_type == TL_WRITE && table->table->getShare()->getVersion())
1816
if (lock_type == TL_WRITE && table->table->s->version)
1867
table::Cache::singleton().mutex().lock(); /* Lock type is TL_WRITE and we lock to repair the table */
1868
const char *old_message=session->enter_cond(COND_refresh, table::Cache::singleton().mutex(),
1869
"Waiting to get writelock");
1870
session->abortLock(table->table);
1871
identifier::Table identifier(table->table->getShare()->getSchemaName(), table->table->getShare()->getTableName());
1872
table::Cache::singleton().removeTable(session, identifier, RTFC_WAIT_OTHER_THREAD_FLAG | RTFC_CHECK_KILLED_FLAG);
1818
pthread_mutex_lock(&LOCK_open); /* Lock type is TL_WRITE and we lock to repair the table */
1819
const char *old_message=session->enter_cond(&COND_refresh, &LOCK_open,
1820
"Waiting to get writelock");
1821
mysql_lock_abort(session,table->table);
1822
remove_table_from_cache(session, table->table->s->db.str,
1823
table->table->s->table_name.str,
1824
RTFC_WAIT_OTHER_THREAD_FLAG |
1825
RTFC_CHECK_KILLED_FLAG);
1873
1826
session->exit_cond(old_message);
1874
if (session->getKilled())
1827
if (session->killed)
1876
1829
open_for_modify= 0;
1883
session->getLex()->cleanup_after_one_table_open();
1836
lex->cleanup_after_one_table_open();
1884
1837
session->clear_error(); // these errors shouldn't get client
1886
List<DRIZZLE_ERROR>::iterator it(session->warn_list.begin());
1839
List_iterator_fast<DRIZZLE_ERROR> it(session->warn_list);
1887
1840
DRIZZLE_ERROR *err;
1888
1841
while ((err= it++))
1890
session->getClient()->store(table_name.c_str());
1891
session->getClient()->store(operator_name);
1892
session->getClient()->store(warning_level_names[err->level].str,
1843
session->client->store(table_name);
1844
session->client->store(operator_name);
1845
session->client->store(warning_level_names[err->level].str,
1893
1846
warning_level_names[err->level].length);
1894
session->getClient()->store(err->msg);
1895
if (session->getClient()->flush())
1847
session->client->store(err->msg);
1848
if (session->client->flush())
1898
1851
drizzle_reset_errors(session, true);
1900
session->getClient()->store(table_name.c_str());
1901
session->getClient()->store(operator_name);
1853
session->client->store(table_name);
1854
session->client->store(operator_name);
1903
1856
switch (result_code) {
1904
1857
case HA_ADMIN_NOT_IMPLEMENTED:
1906
1859
char buf[ERRMSGSIZE+20];
1907
1860
uint32_t length=snprintf(buf, ERRMSGSIZE,
1908
1861
ER(ER_CHECK_NOT_IMPLEMENTED), operator_name);
1909
session->getClient()->store(STRING_WITH_LEN("note"));
1910
session->getClient()->store(buf, length);
1862
session->client->store(STRING_WITH_LEN("note"));
1863
session->client->store(buf, length);
1914
1867
case HA_ADMIN_OK:
1915
session->getClient()->store(STRING_WITH_LEN("status"));
1916
session->getClient()->store(STRING_WITH_LEN("OK"));
1868
session->client->store(STRING_WITH_LEN("status"));
1869
session->client->store(STRING_WITH_LEN("OK"));
1919
1872
case HA_ADMIN_FAILED:
1920
session->getClient()->store(STRING_WITH_LEN("status"));
1921
session->getClient()->store(STRING_WITH_LEN("Operation failed"));
1873
session->client->store(STRING_WITH_LEN("status"));
1874
session->client->store(STRING_WITH_LEN("Operation failed"));
1924
1877
case HA_ADMIN_REJECT:
1925
session->getClient()->store(STRING_WITH_LEN("status"));
1926
session->getClient()->store(STRING_WITH_LEN("Operation need committed state"));
1878
session->client->store(STRING_WITH_LEN("status"));
1879
session->client->store(STRING_WITH_LEN("Operation need committed state"));
1927
1880
open_for_modify= false;
1930
1883
case HA_ADMIN_ALREADY_DONE:
1931
session->getClient()->store(STRING_WITH_LEN("status"));
1932
session->getClient()->store(STRING_WITH_LEN("Table is already up to date"));
1884
session->client->store(STRING_WITH_LEN("status"));
1885
session->client->store(STRING_WITH_LEN("Table is already up to date"));
1935
1888
case HA_ADMIN_CORRUPT:
1936
session->getClient()->store(STRING_WITH_LEN("error"));
1937
session->getClient()->store(STRING_WITH_LEN("Corrupt"));
1889
session->client->store(STRING_WITH_LEN("error"));
1890
session->client->store(STRING_WITH_LEN("Corrupt"));
1941
1894
case HA_ADMIN_INVALID:
1942
session->getClient()->store(STRING_WITH_LEN("error"));
1943
session->getClient()->store(STRING_WITH_LEN("Invalid argument"));
1895
session->client->store(STRING_WITH_LEN("error"));
1896
session->client->store(STRING_WITH_LEN("Invalid argument"));
1946
1899
default: // Probably HA_ADMIN_INTERNAL_ERROR
1958
1911
if (table->table)
1960
1913
if (fatal_error)
1962
table->table->getMutableShare()->resetVersion(); // Force close of table
1914
table->table->s->version=0; // Force close of table
1964
1915
else if (open_for_modify)
1966
if (table->table->getShare()->getType())
1917
if (table->table->s->tmp_table)
1968
1918
table->table->cursor->info(HA_STATUS_CONST);
1972
boost::unique_lock<boost::mutex> lock(table::Cache::singleton().mutex());
1973
identifier::Table identifier(table->table->getShare()->getSchemaName(), table->table->getShare()->getTableName());
1974
table::Cache::singleton().removeTable(session, identifier, RTFC_NO_FLAG);
1921
pthread_mutex_lock(&LOCK_open);
1922
remove_table_from_cache(session, table->table->s->db.str,
1923
table->table->s->table_name.str, RTFC_NO_FLAG);
1924
pthread_mutex_unlock(&LOCK_open);
1978
transaction_services.autocommitOrRollback(*session, false);
1928
transaction_services.ha_autocommit_or_rollback(session, false);
1979
1929
session->endTransaction(COMMIT);
1980
1930
session->close_thread_tables();
1981
1931
table->table=0; // For query cache
1982
if (session->getClient()->flush())
1932
if (session->client->flush())
1949
We have to write the query before we unlock the named table.
1951
Since temporary tables are not replicated under row-based
1952
replication, CREATE TABLE ... LIKE ... needs special
1953
treatement. We have four cases to consider, according to the
1954
following decision table:
1956
==== ========= ========= ==============================
1957
Case Target Source Write to binary log
1958
==== ========= ========= ==============================
1959
1 normal normal Original statement
1960
2 normal temporary Generated statement
1961
3 temporary normal Nothing
1962
4 temporary temporary Nothing
1963
==== ========= ========= ==============================
1965
static bool replicateCreateTableLike(Session *session, TableList *table, Table *name_lock,
1966
bool is_src_table_tmp, bool is_if_not_exists)
1968
if (is_src_table_tmp)
1971
String query(buf, sizeof(buf), system_charset_info);
1972
query.length(0); // Have to zero it since constructor doesn't
1976
Here we open the destination table, on which we already have
1977
name-lock. This is needed for store_create_info() to work.
1978
The table will be closed by unlink_open_table() at the end
1981
table->table= name_lock;
1982
pthread_mutex_lock(&LOCK_open); /* Open new table we have just acquired */
1983
if (session->reopen_name_locked_table(table, false))
1985
pthread_mutex_unlock(&LOCK_open);
1988
pthread_mutex_unlock(&LOCK_open);
1990
int result= store_create_info(table, &query, is_if_not_exists);
1992
assert(result == 0); // store_create_info() always return 0
1993
write_bin_log(session, query.ptr());
1997
write_bin_log(session, session->query.c_str());
1999
2004
Create a new table by copying from source table
2001
2006
Altough exclusive name-lock on target table protects us from concurrent
2002
2007
DML and DDL operations on it we still want to wrap .FRM creation and call
2003
2008
to plugin::StorageEngine::createTable() in critical section protected by
2004
table::Cache::singleton().mutex() in order to provide minimal atomicity against operations which
2009
LOCK_open in order to provide minimal atomicity against operations which
2005
2010
disregard name-locks, like I_S implementation, for example. This is a
2006
2011
temporary and should not be copied. Instead we should fix our code to
2007
2012
always honor name-locks.
2009
Also some engines (e.g. NDB cluster) require that table::Cache::singleton().mutex() should be held
2014
Also some engines (e.g. NDB cluster) require that LOCK_open should be held
2010
2015
during the call to plugin::StorageEngine::createTable().
2011
2016
See bug #28614 for more info.
2013
static bool create_table_wrapper(Session &session,
2014
const message::Table& create_table_proto,
2015
identifier::Table::const_reference destination_identifier,
2016
identifier::Table::const_reference source_identifier,
2018
static bool create_table_wrapper(Session &session, message::Table& create_table_proto,
2019
TableIdentifier &destination_identifier,
2020
TableIdentifier &src_table,
2021
bool lex_identified_temp_table, bool is_engine_set)
2019
// We require an additional table message because during parsing we used
2020
// a "new" message and it will not have all of the information that the
2021
// source table message would have.
2022
message::Table new_table_message;
2024
message::table::shared_ptr source_table_message= plugin::StorageEngine::getTableMessage(session, source_identifier);
2026
if (not source_table_message)
2028
my_error(ER_TABLE_UNKNOWN, source_identifier);
2023
int protoerr= EEXIST;
2024
message::Table new_proto;
2025
message::Table src_proto;
2027
protoerr= plugin::StorageEngine::getTableDefinition(session,
2030
new_proto.CopyFrom(src_proto);
2032
if (lex_identified_temp_table)
2034
new_proto.set_type(message::Table::TEMPORARY);
2038
new_proto.set_type(message::Table::STANDARD);
2043
message::Table::StorageEngine *protoengine;
2045
protoengine= new_proto.mutable_engine();
2046
protoengine->set_name(create_table_proto.engine().name());
2049
if (protoerr && protoerr != EEXIST)
2051
if (errno == ENOENT)
2052
my_error(ER_BAD_DB_ERROR,MYF(0), destination_identifier.getSchemaName());
2054
my_error(ER_CANT_CREATE_FILE, MYF(0), destination_identifier.getPath(), errno);
2032
new_table_message.CopyFrom(*source_table_message);
2034
if (destination_identifier.isTmp())
2036
new_table_message.set_type(message::Table::TEMPORARY);
2040
new_table_message.set_type(message::Table::STANDARD);
2045
new_table_message.mutable_engine()->set_name(create_table_proto.engine().name());
2048
{ // We now do a selective copy of elements on to the new table.
2049
new_table_message.set_name(create_table_proto.name());
2050
new_table_message.set_schema(create_table_proto.schema());
2051
new_table_message.set_catalog(create_table_proto.catalog());
2054
/* Fix names of foreign keys being added */
2055
for (int32_t j= 0; j < new_table_message.fk_constraint_size(); j++)
2057
if (new_table_message.fk_constraint(j).has_name())
2059
std::string name(new_table_message.name());
2062
name.append("_ibfk_");
2063
snprintf(number, sizeof(number), "%d", j+1);
2064
name.append(number);
2066
message::Table::ForeignKeyConstraint *pfkey= new_table_message.mutable_fk_constraint(j);
2067
pfkey->set_name(name);
2072
2060
As mysql_truncate don't work on a new table at this stage of
2073
creation, instead create the table directly (for both normal and temporary tables).
2061
creation, instead create the table directly (for both normal
2062
and temporary tables).
2075
bool success= plugin::StorageEngine::createTable(session,
2076
destination_identifier,
2079
if (success && not destination_identifier.isTmp())
2081
TransactionServices &transaction_services= TransactionServices::singleton();
2082
transaction_services.createTable(session, new_table_message);
2064
int err= plugin::StorageEngine::createTable(session,
2065
destination_identifier,
2068
return err ? false : true;
2089
2072
Create a table identical to the specified table
2075
mysql_create_like_table()
2093
2076
session Thread object
2094
2077
table Table list element for target table
2095
2078
src_table Table list element for source table
2103
bool create_like_table(Session* session,
2104
identifier::Table::const_reference destination_identifier,
2105
identifier::Table::const_reference source_identifier,
2106
message::Table &create_table_proto,
2107
bool is_if_not_exists,
2086
bool mysql_create_like_table(Session* session, TableList* table, TableList* src_table,
2087
message::Table& create_table_proto,
2088
bool is_if_not_exists,
2091
Table *name_lock= 0;
2092
char *db= table->db;
2093
char *table_name= table->table_name;
2110
2094
bool res= true;
2111
bool table_exists= false;
2096
bool lex_identified_temp_table=
2097
(create_table_proto.type() == message::Table::TEMPORARY);
2101
By opening source table we guarantee that it exists and no concurrent
2102
DDL operation will mess with it. Later we also take an exclusive
2103
name-lock on target table name, which makes copying of .frm cursor,
2104
call to plugin::StorageEngine::createTable() and binlogging atomic
2105
against concurrent DML and DDL operations on target table.
2106
Thus by holding both these "locks" we ensure that our statement is
2107
properly isolated from all concurrent operations which matter.
2109
if (session->open_tables_from_list(&src_table, ¬_used))
2112
TableIdentifier destination_identifier(db, table_name,
2113
lex_identified_temp_table ? TEMP_TABLE : STANDARD_TABLE);
2115
TableIdentifier src_identifier(src_table->table->s->db.str,
2116
src_table->table->s->table_name.str, src_table->table->s->tmp_table);
2114
2121
Check that destination tables does not exist. Note that its name
2115
2122
was already checked when it was added to the table list.
2117
For temporary tables we don't aim to grab locks.
2119
if (destination_identifier.isTmp())
2124
bool table_exists= false;
2125
if (lex_identified_temp_table)
2121
if (session->find_temporary_table(destination_identifier))
2127
if (session->find_temporary_table(db, table_name))
2123
2129
table_exists= true;
2127
bool was_created= create_table_wrapper(*session,
2129
destination_identifier,
2132
if (not was_created) // This is pretty paranoid, but we assume something might not clean up after itself
2134
(void) session->rm_temporary_table(destination_identifier, true);
2136
else if (not session->open_temporary_table(destination_identifier))
2138
// We created, but we can't open... also, a hack.
2139
(void) session->rm_temporary_table(destination_identifier, true);
2147
else // Standard table which will require locks.
2149
Table *name_lock= 0;
2151
if (session->lock_table_name_if_not_cached(destination_identifier, &name_lock))
2134
if (session->lock_table_name_if_not_cached(db, table_name, &name_lock))
2155
boost_unique_lock_t lock(table::Cache::singleton().mutex()); /* unlink open tables for create table like*/
2138
pthread_mutex_lock(&LOCK_open); /* unlink open tables for create table like*/
2156
2139
session->unlink_open_table(name_lock);
2140
pthread_mutex_unlock(&LOCK_open);
2202
2160
char warn_buff[DRIZZLE_ERRMSG_SIZE];
2203
2161
snprintf(warn_buff, sizeof(warn_buff),
2204
ER(ER_TABLE_EXISTS_ERROR), destination_identifier.getTableName().c_str());
2162
ER(ER_TABLE_EXISTS_ERROR), table_name);
2205
2163
push_warning(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
2206
ER_TABLE_EXISTS_ERROR, warn_buff);
2210
my_error(ER_TABLE_EXISTS_ERROR, destination_identifier);
2164
ER_TABLE_EXISTS_ERROR,warn_buff);
2169
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), table_name);
2172
else // Otherwise we create the table
2174
pthread_mutex_lock(&LOCK_open); /* We lock for CREATE TABLE LIKE to copy table definition */
2175
was_created= create_table_wrapper(*session, create_table_proto, destination_identifier,
2176
src_identifier, lex_identified_temp_table, is_engine_set);
2177
pthread_mutex_unlock(&LOCK_open);
2179
// So we blew the creation of the table, and we scramble to clean up
2180
// anything that might have been created (read... it is a hack)
2181
if (not was_created)
2183
if (lex_identified_temp_table)
2185
(void) session->rm_temporary_table(destination_identifier);
2189
TableIdentifier identifier(db, table_name, STANDARD_TABLE);
2190
quick_rm_table(*session, identifier);
2193
else if (lex_identified_temp_table && not session->open_temporary_table(destination_identifier))
2195
// We created, but we can't open... also, a hack.
2196
(void) session->rm_temporary_table(destination_identifier);
2200
if (not lex_identified_temp_table)
2202
bool rc= replicateCreateTableLike(session, table, name_lock, (src_table->table->s->tmp_table), is_if_not_exists);
2212
pthread_mutex_lock(&LOCK_open); /* unlink open tables for create table like*/
2213
session->unlink_open_table(name_lock);
2214
pthread_mutex_unlock(&LOCK_open);
2219
bool analyze_table(Session* session, TableList* tables, HA_CHECK_OPT* check_opt)
2221
bool mysql_analyze_table(Session* session, TableList* tables, HA_CHECK_OPT* check_opt)
2221
2223
thr_lock_type lock_type = TL_READ_NO_INSERT;
2223
return(admin_table(session, tables, check_opt,
2225
return(mysql_admin_table(session, tables, check_opt,
2224
2226
"analyze", lock_type, true,
2225
2227
&Cursor::ha_analyze));
2229
bool check_table(Session* session, TableList* tables,HA_CHECK_OPT* check_opt)
2231
bool mysql_check_table(Session* session, TableList* tables,HA_CHECK_OPT* check_opt)
2231
2233
thr_lock_type lock_type = TL_READ_NO_INSERT;
2233
return(admin_table(session, tables, check_opt,
2235
return(mysql_admin_table(session, tables, check_opt,
2234
2236
"check", lock_type,
2236
2238
&Cursor::ha_check));
2242
bool mysql_checksum_table(Session *session, TableList *tables,
2246
List<Item> field_list;
2249
field_list.push_back(item = new Item_empty_string("Table", NAME_LEN*2));
2250
item->maybe_null= 1;
2251
field_list.push_back(item= new Item_int("Checksum", (int64_t) 1,
2252
MY_INT64_NUM_DECIMAL_DIGITS));
2253
item->maybe_null= 1;
2254
if (session->client->sendFields(&field_list))
2257
/* Open one table after the other to keep lock time as short as possible. */
2258
for (table= tables; table; table= table->next_local)
2260
char table_name[NAME_LEN*2+2];
2263
sprintf(table_name,"%s.%s",table->db,table->table_name);
2265
t= table->table= session->openTableLock(table, TL_READ);
2266
session->clear_error(); // these errors shouldn't get client
2268
session->client->store(table_name);
2272
/* Table didn't exist */
2273
session->client->store();
2274
session->clear_error();
2279
@note if the engine keeps a checksum then we return the checksum, otherwise we calculate
2281
if (t->cursor->getEngine()->check_flag(HTON_BIT_HAS_CHECKSUM))
2283
session->client->store((uint64_t)t->cursor->checksum());
2287
/* calculating table's checksum */
2288
internal::ha_checksum crc= 0;
2289
unsigned char null_mask=256 - (1 << t->s->last_null_bit_pos);
2291
t->use_all_columns();
2293
if (t->cursor->ha_rnd_init(1))
2294
session->client->store();
2299
internal::ha_checksum row_crc= 0;
2300
int error= t->cursor->rnd_next(t->record[0]);
2301
if (unlikely(error))
2303
if (error == HA_ERR_RECORD_DELETED)
2307
if (t->s->null_bytes)
2309
/* fix undefined null bits */
2310
t->record[0][t->s->null_bytes-1] |= null_mask;
2311
if (!(t->s->db_create_options & HA_OPTION_PACK_RECORD))
2312
t->record[0][0] |= 1;
2314
row_crc= internal::my_checksum(row_crc, t->record[0], t->s->null_bytes);
2317
for (uint32_t i= 0; i < t->s->fields; i++ )
2319
Field *f= t->field[i];
2320
if ((f->type() == DRIZZLE_TYPE_BLOB) ||
2321
(f->type() == DRIZZLE_TYPE_VARCHAR))
2325
row_crc= internal::my_checksum(row_crc, (unsigned char*) tmp.ptr(), tmp.length());
2328
row_crc= internal::my_checksum(row_crc, f->ptr,
2334
session->client->store((uint64_t)crc);
2335
t->cursor->ha_rnd_end();
2338
session->clear_error();
2339
session->close_thread_tables();
2340
table->table=0; // For query cache
2342
if (session->client->flush())
2350
session->close_thread_tables(); // Shouldn't be needed
2239
2356
} /* namespace drizzled */