~drizzle-trunk/drizzle/development

1 by brian
clean slate
1
/* Copyright (C) 2000-2003 MySQL AB
2
3
   This program is free software; you can redistribute it and/or modify
4
   it under the terms of the GNU General Public License as published by
5
   the Free Software Foundation; version 2 of the License.
6
7
   This program is distributed in the hope that it will be useful,
8
   but WITHOUT ANY WARRANTY; without even the implied warranty of
9
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10
   GNU General Public License for more details.
11
12
   You should have received a copy of the GNU General Public License
13
   along with this program; if not, write to the Free Software
14
   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
15
16
17
/* create and drop of databases */
549 by Monty Taylor
Took gettext.h out of header files.
18
#include <config.h>
481.1.9 by Monty Taylor
Added autoconf tests for location of cstdint and cinttypes. Use those in C++ programs now, so that we don't have to define _STDC_LIMIT_MACROS, etc by hand. Stop, in fact, defining those by hand.
19
#include CSTDINT_H
20
#include CINTTYPES_H
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
21
#include <string>
22
#include <fstream>
316 by Brian Aker
First pass of new sql_db.cc work
23
#include <drizzled/serialize/serialize.h>
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
24
using namespace std;
243.1.17 by Jay Pipes
FINAL PHASE removal of mysql_priv.h (Bye, bye my friend.)
25
#include <drizzled/server_includes.h>
212.5.13 by Monty Taylor
Moved my_sys/my_pthread/my_nosys and mysys_err to mysys.
26
#include <mysys/mysys_err.h>
212.5.38 by Monty Taylor
Moved my_dir (and removed references to a lot of places)
27
#include <mysys/my_dir.h>
1 by brian
clean slate
28
#include "log.h"
549 by Monty Taylor
Took gettext.h out of header files.
29
#include <drizzled/error.h>
538 by Monty Taylor
Moved gettext.h into drizzled in anticipation of the new client lib.
30
#include <drizzled/gettext.h>
584.1.15 by Monty Taylor
The mega-patch from hell. Renamed sql_class to session (since that's what it is) and removed it and field and table from common_includes.
31
#include <mysys/hash.h>
32
#include <drizzled/session.h>
33
#include <drizzled/db.h>
34
#include <drizzled/sql_base.h>
316 by Brian Aker
First pass of new sql_db.cc work
35
1 by brian
clean slate
36
#define MAX_DROP_TABLE_Q_LEN      1024
37
461 by Monty Taylor
Removed NullS. bu-bye.
38
const char *del_exts[]= {".frm", ".BAK", ".TMD",".opt", NULL};
1 by brian
clean slate
39
static TYPELIB deletable_extentions=
40
{array_elements(del_exts)-1,"del_exts", del_exts, NULL};
41
520.1.22 by Brian Aker
Second pass of thd cleanup
42
static long mysql_rm_known_files(Session *session, MY_DIR *dirp,
584.1.15 by Monty Taylor
The mega-patch from hell. Renamed sql_class to session (since that's what it is) and removed it and field and table from common_includes.
43
                                 const char *db, const char *path,
44
                                 uint32_t level,
327.2.4 by Brian Aker
Refactoring table.h
45
                                 TableList **dropped_tables);
584.1.15 by Monty Taylor
The mega-patch from hell. Renamed sql_class to session (since that's what it is) and removed it and field and table from common_includes.
46
253 by Brian Aker
Removed final my_bool from sql_db.
47
static bool rm_dir_w_symlink(const char *org_path, bool send_error);
520.1.22 by Brian Aker
Second pass of thd cleanup
48
static void mysql_change_db_impl(Session *session,
1 by brian
clean slate
49
                                 LEX_STRING *new_db_name,
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
50
                                 const CHARSET_INFO * const new_db_charset);
1 by brian
clean slate
51
52
53
/* Database lock hash */
54
HASH lock_db_cache;
55
pthread_mutex_t LOCK_lock_db;
56
int creating_database= 0;  // how many database locks are made
57
58
59
/* Structure for database lock */
60
typedef struct my_dblock_st
61
{
62
  char *name;        /* Database name        */
482 by Brian Aker
Remove uint.
63
  uint32_t name_length;  /* Database length name */
1 by brian
clean slate
64
} my_dblock_t;
65
66
67
/*
68
  lock_db key.
69
*/
70
481 by Brian Aker
Remove all of uchar.
71
extern "C" unsigned char* lock_db_get_key(my_dblock_t *, size_t *, bool not_used);
1 by brian
clean slate
72
481 by Brian Aker
Remove all of uchar.
73
unsigned char* lock_db_get_key(my_dblock_t *ptr, size_t *length,
253 by Brian Aker
Removed final my_bool from sql_db.
74
                       bool not_used __attribute__((unused)))
1 by brian
clean slate
75
{
76
  *length= ptr->name_length;
481 by Brian Aker
Remove all of uchar.
77
  return (unsigned char*) ptr->name;
1 by brian
clean slate
78
}
79
80
81
/*
82
  Free lock_db hash element.
83
*/
84
85
extern "C" void lock_db_free_element(void *ptr);
86
87
void lock_db_free_element(void *ptr)
88
{
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
89
  free(ptr);
1 by brian
clean slate
90
}
91
92
93
/*
94
  Delete a database lock entry from hash.
95
*/
96
482 by Brian Aker
Remove uint.
97
void lock_db_delete(const char *name, uint32_t length)
1 by brian
clean slate
98
{
99
  my_dblock_t *opt;
100
  safe_mutex_assert_owner(&LOCK_lock_db);
101
  if ((opt= (my_dblock_t *)hash_search(&lock_db_cache,
481 by Brian Aker
Remove all of uchar.
102
                                       (const unsigned char*) name, length)))
103
    hash_delete(&lock_db_cache, (unsigned char*) opt);
1 by brian
clean slate
104
}
105
106
107
/* Database options hash */
108
static HASH dboptions;
253 by Brian Aker
Removed final my_bool from sql_db.
109
static bool dboptions_init= 0;
1 by brian
clean slate
110
static rw_lock_t LOCK_dboptions;
111
112
/* Structure for database options */
113
typedef struct my_dbopt_st
114
{
115
  char *name;			/* Database name                  */
482 by Brian Aker
Remove uint.
116
  uint32_t name_length;		/* Database length name           */
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
117
  const CHARSET_INFO *charset;	/* Database default character set */
1 by brian
clean slate
118
} my_dbopt_t;
119
120
121
/*
122
  Function we use in the creation of our hash to get key.
123
*/
124
481 by Brian Aker
Remove all of uchar.
125
extern "C" unsigned char* dboptions_get_key(my_dbopt_t *opt, size_t *length,
253 by Brian Aker
Removed final my_bool from sql_db.
126
                                    bool not_used);
1 by brian
clean slate
127
481 by Brian Aker
Remove all of uchar.
128
unsigned char* dboptions_get_key(my_dbopt_t *opt, size_t *length,
253 by Brian Aker
Removed final my_bool from sql_db.
129
                         bool not_used __attribute__((unused)))
1 by brian
clean slate
130
{
131
  *length= opt->name_length;
481 by Brian Aker
Remove all of uchar.
132
  return (unsigned char*) opt->name;
1 by brian
clean slate
133
}
134
135
136
/*
137
  Helper function to write a query to binlog used by mysql_rm_db()
138
*/
139
520.1.22 by Brian Aker
Second pass of thd cleanup
140
static inline void write_to_binlog(Session *session, char *query, uint32_t q_len,
482 by Brian Aker
Remove uint.
141
                                   char *db, uint32_t db_len)
1 by brian
clean slate
142
{
520.1.22 by Brian Aker
Second pass of thd cleanup
143
  Query_log_event qinfo(session, query, q_len, 0, 0);
1 by brian
clean slate
144
  qinfo.error_code= 0;
145
  qinfo.db= db;
146
  qinfo.db_len= db_len;
586.1.1 by Yoshinori Sano
Rename mysql to drizzle, specifically mysql_bin_log to drizzle_bin_log.
147
  drizzle_bin_log.write(&qinfo);
1 by brian
clean slate
148
}  
149
150
151
/*
152
  Function to free dboptions hash element
153
*/
154
155
extern "C" void free_dbopt(void *dbopt);
156
157
void free_dbopt(void *dbopt)
158
{
481 by Brian Aker
Remove all of uchar.
159
  free((unsigned char*) dbopt);
1 by brian
clean slate
160
}
161
162
163
/* 
164
  Initialize database option hash and locked database hash.
165
166
  SYNOPSIS
167
    my_database_names()
168
169
  NOTES
170
    Must be called before any other database function is called.
171
172
  RETURN
173
    0	ok
174
    1	Fatal error
175
*/
176
177
bool my_database_names_init(void)
178
{
253 by Brian Aker
Removed final my_bool from sql_db.
179
  bool error= false;
1 by brian
clean slate
180
  (void) my_rwlock_init(&LOCK_dboptions, NULL);
181
  if (!dboptions_init)
182
  {
183
    dboptions_init= 1;
184
    error= hash_init(&dboptions, lower_case_table_names ? 
185
                     &my_charset_bin : system_charset_info,
186
                     32, 0, 0, (hash_get_key) dboptions_get_key,
187
                     free_dbopt,0) ||
188
           hash_init(&lock_db_cache, lower_case_table_names ? 
189
                     &my_charset_bin : system_charset_info,
190
                     32, 0, 0, (hash_get_key) lock_db_get_key,
191
                     lock_db_free_element,0);
192
193
  }
194
  return error;
195
}
196
197
198
199
/* 
200
  Free database option hash and locked databases hash.
201
*/
202
203
void my_database_names_free(void)
204
{
205
  if (dboptions_init)
206
  {
207
    dboptions_init= 0;
208
    hash_free(&dboptions);
209
    (void) rwlock_destroy(&LOCK_dboptions);
210
    hash_free(&lock_db_cache);
211
  }
212
}
213
214
215
/*
216
  Cleanup cached options
217
*/
218
219
void my_dbopt_cleanup(void)
220
{
221
  rw_wrlock(&LOCK_dboptions);
222
  hash_free(&dboptions);
223
  hash_init(&dboptions, lower_case_table_names ? 
224
            &my_charset_bin : system_charset_info,
225
            32, 0, 0, (hash_get_key) dboptions_get_key,
226
            free_dbopt,0);
227
  rw_unlock(&LOCK_dboptions);
228
}
229
230
231
/*
232
  Find database options in the hash.
233
  
234
  DESCRIPTION
235
    Search a database options in the hash, usings its path.
236
    Fills "create" on success.
237
  
238
  RETURN VALUES
239
    0 on success.
240
    1 on error.
241
*/
242
253 by Brian Aker
Removed final my_bool from sql_db.
243
static bool get_dbopt(const char *dbname, HA_CREATE_INFO *create)
1 by brian
clean slate
244
{
245
  my_dbopt_t *opt;
482 by Brian Aker
Remove uint.
246
  uint32_t length;
253 by Brian Aker
Removed final my_bool from sql_db.
247
  bool error= true;
1 by brian
clean slate
248
  
249
  length= (uint) strlen(dbname);
250
  
251
  rw_rdlock(&LOCK_dboptions);
481 by Brian Aker
Remove all of uchar.
252
  if ((opt= (my_dbopt_t*) hash_search(&dboptions, (unsigned char*) dbname, length)))
1 by brian
clean slate
253
  {
254
    create->default_table_charset= opt->charset;
253 by Brian Aker
Removed final my_bool from sql_db.
255
    error= true;
1 by brian
clean slate
256
  }
257
  rw_unlock(&LOCK_dboptions);
258
  return error;
259
}
260
261
262
/*
263
  Writes database options into the hash.
264
  
265
  DESCRIPTION
266
    Inserts database options into the hash, or updates
267
    options if they are already in the hash.
268
  
269
  RETURN VALUES
270
    0 on success.
271
    1 on error.
272
*/
273
253 by Brian Aker
Removed final my_bool from sql_db.
274
static bool put_dbopt(const char *dbname, HA_CREATE_INFO *create)
1 by brian
clean slate
275
{
276
  my_dbopt_t *opt;
482 by Brian Aker
Remove uint.
277
  uint32_t length;
253 by Brian Aker
Removed final my_bool from sql_db.
278
  bool error= false;
1 by brian
clean slate
279
280
  length= (uint) strlen(dbname);
281
  
282
  rw_wrlock(&LOCK_dboptions);
481 by Brian Aker
Remove all of uchar.
283
  if (!(opt= (my_dbopt_t*) hash_search(&dboptions, (unsigned char*) dbname, length)))
1 by brian
clean slate
284
  { 
285
    /* Options are not in the hash, insert them */
286
    char *tmp_name;
287
    if (!my_multi_malloc(MYF(MY_WME | MY_ZEROFILL),
288
                         &opt, (uint) sizeof(*opt), &tmp_name, (uint) length+1,
461 by Monty Taylor
Removed NullS. bu-bye.
289
                         NULL))
1 by brian
clean slate
290
    {
253 by Brian Aker
Removed final my_bool from sql_db.
291
      error= true;
1 by brian
clean slate
292
      goto end;
293
    }
294
    
295
    opt->name= tmp_name;
411.1.1 by Brian Aker
Work on removing GNU specific calls.
296
    my_stpcpy(opt->name, dbname);
1 by brian
clean slate
297
    opt->name_length= length;
298
    
481 by Brian Aker
Remove all of uchar.
299
    if ((error= my_hash_insert(&dboptions, (unsigned char*) opt)))
1 by brian
clean slate
300
    {
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
301
      free(opt);
1 by brian
clean slate
302
      goto end;
303
    }
304
  }
305
306
  /* Update / write options in hash */
307
  opt->charset= create->default_table_charset;
308
309
end:
310
  rw_unlock(&LOCK_dboptions);  
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
311
  return(error);
1 by brian
clean slate
312
}
313
314
315
/*
316
  Deletes database options from the hash.
317
*/
318
319
void del_dbopt(const char *path)
320
{
321
  my_dbopt_t *opt;
322
  rw_wrlock(&LOCK_dboptions);
481 by Brian Aker
Remove all of uchar.
323
  if ((opt= (my_dbopt_t *)hash_search(&dboptions, (const unsigned char*) path,
1 by brian
clean slate
324
                                      strlen(path))))
481 by Brian Aker
Remove all of uchar.
325
    hash_delete(&dboptions, (unsigned char*) opt);
1 by brian
clean slate
326
  rw_unlock(&LOCK_dboptions);
327
}
328
329
330
/*
331
  Create database options file:
332
333
  DESCRIPTION
334
    Currently database default charset is only stored there.
335
336
  RETURN VALUES
337
  0	ok
338
  1	Could not create file or write to it.  Error sent through my_error()
339
*/
340
520.1.22 by Brian Aker
Second pass of thd cleanup
341
static bool write_db_opt(Session *session, const char *path, const char *name, HA_CREATE_INFO *create)
1 by brian
clean slate
342
{
253 by Brian Aker
Removed final my_bool from sql_db.
343
  bool error= true;
323 by Brian Aker
Updated proto file for table (not FRM work).
344
  drizzle::Schema db;
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
345
346
  assert(name);
1 by brian
clean slate
347
348
  if (!create->default_table_charset)
520.1.22 by Brian Aker
Second pass of thd cleanup
349
    create->default_table_charset= session->variables.collation_server;
1 by brian
clean slate
350
351
  if (put_dbopt(path, create))
352
    return 1;
353
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
354
  db.set_name(name);
355
  db.set_characterset(create->default_table_charset->csname);
356
  db.set_collation(create->default_table_charset->name);
357
358
  fstream output(path, ios::out | ios::trunc | ios::binary);
359
  if (!db.SerializeToOstream(&output)) 
360
    error= false;
361
1 by brian
clean slate
362
  return error;
363
}
364
365
366
/*
367
  Load database options file
368
369
  load_db_opt()
370
  path		Path for option file
371
  create	Where to store the read options
372
373
  DESCRIPTION
374
375
  RETURN VALUES
376
  0	File found
377
  1	No database file or could not open it
378
379
*/
380
520.1.22 by Brian Aker
Second pass of thd cleanup
381
bool load_db_opt(Session *session, const char *path, HA_CREATE_INFO *create)
1 by brian
clean slate
382
{
383
  bool error=1;
323 by Brian Aker
Updated proto file for table (not FRM work).
384
  drizzle::Schema db;
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
385
  string buffer;
1 by brian
clean slate
386
212.6.6 by Mats Kindahl
Removing redundant use of casts in drizzled/ for memcmp(), memcpy(), memset(), and memmove().
387
  memset(create, 0, sizeof(*create));
520.1.22 by Brian Aker
Second pass of thd cleanup
388
  create->default_table_charset= session->variables.collation_server;
1 by brian
clean slate
389
390
  /* Check if options for this database are already in the hash */
391
  if (!get_dbopt(path, create))
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
392
    return(0);
1 by brian
clean slate
393
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
394
  fstream input(path, ios::in | ios::binary);
395
  if (!input)
396
    goto err1;
397
  else if (!db.ParseFromIstream(&input))
398
    goto err1;
399
400
  buffer= db.characterset();
401
  if (!(create->default_table_charset= get_charset_by_csname(buffer.c_str(), MY_CS_PRIMARY, MYF(0))))
402
  {
338 by Monty Taylor
Tagged more strings.
403
    sql_print_error(_("Error while loading database options: '%s':"),path);
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
404
    sql_print_error(ER(ER_UNKNOWN_COLLATION), buffer.c_str());
405
    create->default_table_charset= default_charset_info;
406
  }
407
408
  buffer= db.collation();
409
  if (!(create->default_table_charset= get_charset_by_name(buffer.c_str(), MYF(0))))
410
  {
338 by Monty Taylor
Tagged more strings.
411
    sql_print_error(_("Error while loading database options: '%s':"),path);
318 by Brian Aker
Modified sql_db to now use Google Proto buffers instead of MySQL type.
412
    sql_print_error(ER(ER_UNKNOWN_COLLATION), buffer.c_str());
413
    create->default_table_charset= default_charset_info;
414
  }
415
1 by brian
clean slate
416
  /*
417
    Put the loaded value into the hash.
418
    Note that another thread could've added the same
419
    entry to the hash after we called get_dbopt(),
420
    but it's not an error, as put_dbopt() takes this
421
    possibility into account.
422
  */
423
  error= put_dbopt(path, create);
424
425
err1:
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
426
  return(error);
1 by brian
clean slate
427
}
428
429
430
/*
431
  Retrieve database options by name. Load database options file or fetch from
432
  cache.
433
434
  SYNOPSIS
435
    load_db_opt_by_name()
436
    db_name         Database name
437
    db_create_info  Where to store the database options
438
439
  DESCRIPTION
440
    load_db_opt_by_name() is a shortcut for load_db_opt().
441
442
  NOTE
443
    Although load_db_opt_by_name() (and load_db_opt()) returns status of
444
    the operation, it is useless usually and should be ignored. The problem
445
    is that there are 1) system databases ("mysql") and 2) virtual
446
    databases ("information_schema"), which do not contain options file.
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
447
    So, load_db_opt[_by_name]() returns false for these databases, but this
1 by brian
clean slate
448
    is not an error.
449
450
    load_db_opt[_by_name]() clears db_create_info structure in any case, so
451
    even on failure it contains valid data. So, common use case is just
452
    call load_db_opt[_by_name]() without checking return value and use
453
    db_create_info right after that.
454
455
  RETURN VALUES (read NOTE!)
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
456
    false   Success
457
    true    Failed to retrieve options
1 by brian
clean slate
458
*/
459
520.1.22 by Brian Aker
Second pass of thd cleanup
460
bool load_db_opt_by_name(Session *session, const char *db_name,
1 by brian
clean slate
461
                         HA_CREATE_INFO *db_create_info)
462
{
463
  char db_opt_path[FN_REFLEN];
464
465
  /*
466
    Pass an empty file name, and the database options file name as extension
467
    to avoid table name to file name encoding.
468
  */
469
  (void) build_table_filename(db_opt_path, sizeof(db_opt_path),
470
                              db_name, "", MY_DB_OPT_FILE, 0);
471
520.1.22 by Brian Aker
Second pass of thd cleanup
472
  return load_db_opt(session, db_opt_path, db_create_info);
1 by brian
clean slate
473
}
474
475
476
/**
477
  Return default database collation.
478
520.1.22 by Brian Aker
Second pass of thd cleanup
479
  @param session     Thread context.
1 by brian
clean slate
480
  @param db_name Database name.
481
482
  @return CHARSET_INFO object. The operation always return valid character
483
    set, even if the database does not exist.
484
*/
485
520.1.22 by Brian Aker
Second pass of thd cleanup
486
const CHARSET_INFO *get_default_db_collation(Session *session, const char *db_name)
1 by brian
clean slate
487
{
488
  HA_CREATE_INFO db_info;
489
520.1.22 by Brian Aker
Second pass of thd cleanup
490
  if (session->db != NULL && strcmp(db_name, session->db) == 0)
491
    return session->db_charset;
1 by brian
clean slate
492
520.1.22 by Brian Aker
Second pass of thd cleanup
493
  load_db_opt_by_name(session, db_name, &db_info);
1 by brian
clean slate
494
495
  /*
496
    NOTE: even if load_db_opt_by_name() fails,
497
    db_info.default_table_charset contains valid character set
498
    (collation_server). We should not fail if load_db_opt_by_name() fails,
499
    because it is valid case. If a database has been created just by
500
    "mkdir", it does not contain db.opt file, but it is valid database.
501
  */
502
503
  return db_info.default_table_charset;
504
}
505
506
507
/*
508
  Create a database
509
510
  SYNOPSIS
511
  mysql_create_db()
520.1.22 by Brian Aker
Second pass of thd cleanup
512
  session		Thread handler
1 by brian
clean slate
513
  db		Name of database to create
514
		Function assumes that this is already validated.
515
  create_info	Database create options (like character set)
516
  silent	Used by replication when internally creating a database.
517
		In this case the entry should not be logged.
518
519
  SIDE-EFFECTS
520
   1. Report back to client that command succeeded (my_ok)
521
   2. Report errors to client
522
   3. Log event to binary log
523
   (The 'silent' flags turns off 1 and 3.)
524
525
  RETURN VALUES
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
526
  false ok
527
  true  Error
1 by brian
clean slate
528
529
*/
530
520.1.22 by Brian Aker
Second pass of thd cleanup
531
int mysql_create_db(Session *session, char *db, HA_CREATE_INFO *create_info, bool silent)
1 by brian
clean slate
532
{
533
  char	 path[FN_REFLEN+16];
534
  char	 tmp_query[FN_REFLEN+16];
535
  long result= 1;
536
  int error= 0;
15 by brian
Fix for stat, NETWARE removal
537
  struct stat stat_info;
482 by Brian Aker
Remove uint.
538
  uint32_t create_options= create_info ? create_info->options : 0;
539
  uint32_t path_len;
1 by brian
clean slate
540
541
  /* do not create 'information_schema' db */
575.4.7 by Monty Taylor
More header cleanup.
542
  if (!my_strcasecmp(system_charset_info, db, INFORMATION_SCHEMA_NAME.c_str()))
1 by brian
clean slate
543
  {
544
    my_error(ER_DB_CREATE_EXISTS, MYF(0), db);
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
545
    return(-1);
1 by brian
clean slate
546
  }
547
548
  /*
549
    Do not create database if another thread is holding read lock.
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
550
    Wait for global read lock before acquiring LOCK_drizzle_create_db.
1 by brian
clean slate
551
    After wait_if_global_read_lock() we have protection against another
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
552
    global read lock. If we would acquire LOCK_drizzle_create_db first,
1 by brian
clean slate
553
    another thread could step in and get the global read lock before we
554
    reach wait_if_global_read_lock(). If this thread tries the same as we
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
555
    (admin a db), it would then go and wait on LOCK_drizzle_create_db...
1 by brian
clean slate
556
    Furthermore wait_if_global_read_lock() checks if the current thread
557
    has the global read lock and refuses the operation with
558
    ER_CANT_UPDATE_WITH_READLOCK if applicable.
559
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
560
  if (wait_if_global_read_lock(session, 0, 1))
1 by brian
clean slate
561
  {
562
    error= -1;
563
    goto exit2;
564
  }
565
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
566
  pthread_mutex_lock(&LOCK_drizzle_create_db);
1 by brian
clean slate
567
568
  /* Check directory */
569
  path_len= build_table_filename(path, sizeof(path), db, "", "", 0);
570
  path[path_len-1]= 0;                    // Remove last '/' from path
571
15 by brian
Fix for stat, NETWARE removal
572
  if (!stat(path,&stat_info))
1 by brian
clean slate
573
  {
574
    if (!(create_options & HA_LEX_CREATE_IF_NOT_EXISTS))
575
    {
576
      my_error(ER_DB_CREATE_EXISTS, MYF(0), db);
577
      error= -1;
578
      goto exit;
579
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
580
    push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
581
			ER_DB_CREATE_EXISTS, ER(ER_DB_CREATE_EXISTS), db);
582
    if (!silent)
520.1.22 by Brian Aker
Second pass of thd cleanup
583
      my_ok(session);
1 by brian
clean slate
584
    error= 0;
585
    goto exit;
586
  }
587
  else
588
  {
15 by brian
Fix for stat, NETWARE removal
589
    if (errno != ENOENT)
1 by brian
clean slate
590
    {
15 by brian
Fix for stat, NETWARE removal
591
      my_error(EE_STAT, MYF(0), path, errno);
1 by brian
clean slate
592
      goto exit;
593
    }
594
    if (my_mkdir(path,0777,MYF(0)) < 0)
595
    {
596
      my_error(ER_CANT_CREATE_DB, MYF(0), db, my_errno);
597
      error= -1;
598
      goto exit;
599
    }
600
  }
601
602
  path[path_len-1]= FN_LIBCHAR;
603
  strmake(path+path_len, MY_DB_OPT_FILE, sizeof(path)-path_len-1);
520.1.22 by Brian Aker
Second pass of thd cleanup
604
  if (write_db_opt(session, path, db, create_info))
1 by brian
clean slate
605
  {
606
    /*
607
      Could not create options file.
608
      Restore things to beginning.
609
    */
610
    path[path_len]= 0;
611
    if (rmdir(path) >= 0)
612
    {
613
      error= -1;
614
      goto exit;
615
    }
616
    /*
617
      We come here when we managed to create the database, but not the option
618
      file.  In this case it's best to just continue as if nothing has
619
      happened.  (This is a very unlikely senario)
620
    */
621
  }
622
  
623
  if (!silent)
624
  {
625
    char *query;
482 by Brian Aker
Remove uint.
626
    uint32_t query_length;
1 by brian
clean slate
627
520.1.22 by Brian Aker
Second pass of thd cleanup
628
    if (!session->query)				// Only in replication
1 by brian
clean slate
629
    {
630
      query= 	     tmp_query;
631
      query_length= (uint) (strxmov(tmp_query,"create database `",
461 by Monty Taylor
Removed NullS. bu-bye.
632
                                    db, "`", NULL) - tmp_query);
1 by brian
clean slate
633
    }
634
    else
635
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
636
      query= 	    session->query;
637
      query_length= session->query_length;
1 by brian
clean slate
638
    }
639
586.1.1 by Yoshinori Sano
Rename mysql to drizzle, specifically mysql_bin_log to drizzle_bin_log.
640
    if (drizzle_bin_log.is_open())
1 by brian
clean slate
641
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
642
      Query_log_event qinfo(session, query, query_length, 0, 
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
643
			    /* suppress_use */ true);
1 by brian
clean slate
644
645
      /*
646
	Write should use the database being created as the "current
647
        database" and not the threads current database, which is the
648
        default. If we do not change the "current database" to the
649
        database being created, the CREATE statement will not be
650
        replicated when using --binlog-do-db to select databases to be
651
        replicated. 
652
653
	An example (--binlog-do-db=sisyfos):
654
       
655
          CREATE DATABASE bob;        # Not replicated
656
          USE bob;                    # 'bob' is the current database
657
          CREATE DATABASE sisyfos;    # Not replicated since 'bob' is
658
                                      # current database.
659
          USE sisyfos;                # Will give error on slave since
660
                                      # database does not exist.
661
      */
662
      qinfo.db     = db;
663
      qinfo.db_len = strlen(db);
664
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
665
      /* These DDL methods and logging protected with LOCK_drizzle_create_db */
586.1.1 by Yoshinori Sano
Rename mysql to drizzle, specifically mysql_bin_log to drizzle_bin_log.
666
      drizzle_bin_log.write(&qinfo);
1 by brian
clean slate
667
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
668
    my_ok(session, result);
1 by brian
clean slate
669
  }
670
671
exit:
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
672
  pthread_mutex_unlock(&LOCK_drizzle_create_db);
520.1.22 by Brian Aker
Second pass of thd cleanup
673
  start_waiting_global_read_lock(session);
1 by brian
clean slate
674
exit2:
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
675
  return(error);
1 by brian
clean slate
676
}
677
678
679
/* db-name is already validated when we come here */
680
520.1.22 by Brian Aker
Second pass of thd cleanup
681
bool mysql_alter_db(Session *session, const char *db, HA_CREATE_INFO *create_info)
1 by brian
clean slate
682
{
683
  char path[FN_REFLEN+16];
684
  long result=1;
253 by Brian Aker
Removed final my_bool from sql_db.
685
  int error= false;
1 by brian
clean slate
686
687
  /*
688
    Do not alter database if another thread is holding read lock.
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
689
    Wait for global read lock before acquiring LOCK_drizzle_create_db.
1 by brian
clean slate
690
    After wait_if_global_read_lock() we have protection against another
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
691
    global read lock. If we would acquire LOCK_drizzle_create_db first,
1 by brian
clean slate
692
    another thread could step in and get the global read lock before we
693
    reach wait_if_global_read_lock(). If this thread tries the same as we
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
694
    (admin a db), it would then go and wait on LOCK_drizzle_create_db...
1 by brian
clean slate
695
    Furthermore wait_if_global_read_lock() checks if the current thread
696
    has the global read lock and refuses the operation with
697
    ER_CANT_UPDATE_WITH_READLOCK if applicable.
698
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
699
  if ((error=wait_if_global_read_lock(session,0,1)))
1 by brian
clean slate
700
    goto exit2;
701
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
702
  pthread_mutex_lock(&LOCK_drizzle_create_db);
1 by brian
clean slate
703
704
  /* 
705
     Recreate db options file: /dbpath/.db.opt
706
     We pass MY_DB_OPT_FILE as "extension" to avoid
707
     "table name to file name" encoding.
708
  */
709
  build_table_filename(path, sizeof(path), db, "", MY_DB_OPT_FILE, 0);
520.1.22 by Brian Aker
Second pass of thd cleanup
710
  if ((error=write_db_opt(session, path, db, create_info)))
1 by brian
clean slate
711
    goto exit;
712
713
  /* Change options if current database is being altered. */
714
520.1.22 by Brian Aker
Second pass of thd cleanup
715
  if (session->db && !strcmp(session->db,db))
1 by brian
clean slate
716
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
717
    session->db_charset= create_info->default_table_charset ?
1 by brian
clean slate
718
		     create_info->default_table_charset :
520.1.22 by Brian Aker
Second pass of thd cleanup
719
		     session->variables.collation_server;
720
    session->variables.collation_database= session->db_charset;
1 by brian
clean slate
721
  }
722
586.1.1 by Yoshinori Sano
Rename mysql to drizzle, specifically mysql_bin_log to drizzle_bin_log.
723
  if (drizzle_bin_log.is_open())
1 by brian
clean slate
724
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
725
    Query_log_event qinfo(session, session->query, session->query_length, 0,
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
726
			  /* suppress_use */ true);
1 by brian
clean slate
727
728
    /*
729
      Write should use the database being created as the "current
730
      database" and not the threads current database, which is the
731
      default.
732
    */
733
    qinfo.db     = db;
734
    qinfo.db_len = strlen(db);
735
520.1.22 by Brian Aker
Second pass of thd cleanup
736
    session->clear_error();
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
737
    /* These DDL methods and logging protected with LOCK_drizzle_create_db */
586.1.1 by Yoshinori Sano
Rename mysql to drizzle, specifically mysql_bin_log to drizzle_bin_log.
738
    drizzle_bin_log.write(&qinfo);
1 by brian
clean slate
739
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
740
  my_ok(session, result);
1 by brian
clean slate
741
742
exit:
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
743
  pthread_mutex_unlock(&LOCK_drizzle_create_db);
520.1.22 by Brian Aker
Second pass of thd cleanup
744
  start_waiting_global_read_lock(session);
1 by brian
clean slate
745
exit2:
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
746
  return(error);
1 by brian
clean slate
747
}
748
749
750
/*
751
  Drop all tables in a database and the database itself
752
753
  SYNOPSIS
754
    mysql_rm_db()
520.1.22 by Brian Aker
Second pass of thd cleanup
755
    session			Thread handle
1 by brian
clean slate
756
    db			Database name in the case given by user
757
		        It's already validated and set to lower case
758
                        (if needed) when we come here
759
    if_exists		Don't give error if database doesn't exists
760
    silent		Don't generate errors
761
762
  RETURN
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
763
    false ok (Database dropped)
1 by brian
clean slate
764
    ERROR Error
765
*/
766
520.1.22 by Brian Aker
Second pass of thd cleanup
767
bool mysql_rm_db(Session *session,char *db,bool if_exists, bool silent)
1 by brian
clean slate
768
{
769
  long deleted=0;
253 by Brian Aker
Removed final my_bool from sql_db.
770
  int error= false;
1 by brian
clean slate
771
  char	path[FN_REFLEN+16];
772
  MY_DIR *dirp;
482 by Brian Aker
Remove uint.
773
  uint32_t length;
327.2.4 by Brian Aker
Refactoring table.h
774
  TableList* dropped_tables= 0;
1 by brian
clean slate
775
776
  if (db && (strcmp(db, "information_schema") == 0))
777
  {
575.4.7 by Monty Taylor
More header cleanup.
778
    my_error(ER_DBACCESS_DENIED_ERROR, MYF(0), "", "", INFORMATION_SCHEMA_NAME.c_str());
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
779
    return(true);
1 by brian
clean slate
780
  }
781
782
  /*
783
    Do not drop database if another thread is holding read lock.
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
784
    Wait for global read lock before acquiring LOCK_drizzle_create_db.
1 by brian
clean slate
785
    After wait_if_global_read_lock() we have protection against another
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
786
    global read lock. If we would acquire LOCK_drizzle_create_db first,
1 by brian
clean slate
787
    another thread could step in and get the global read lock before we
788
    reach wait_if_global_read_lock(). If this thread tries the same as we
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
789
    (admin a db), it would then go and wait on LOCK_drizzle_create_db...
1 by brian
clean slate
790
    Furthermore wait_if_global_read_lock() checks if the current thread
791
    has the global read lock and refuses the operation with
792
    ER_CANT_UPDATE_WITH_READLOCK if applicable.
793
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
794
  if (wait_if_global_read_lock(session, 0, 1))
1 by brian
clean slate
795
  {
796
    error= -1;
797
    goto exit2;
798
  }
799
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
800
  pthread_mutex_lock(&LOCK_drizzle_create_db);
1 by brian
clean slate
801
802
  /*
803
    This statement will be replicated as a statement, even when using
804
    row-based replication. The flag will be reset at the end of the
805
    statement.
806
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
807
  session->clear_current_stmt_binlog_row_based();
1 by brian
clean slate
808
809
  length= build_table_filename(path, sizeof(path), db, "", "", 0);
411.1.1 by Brian Aker
Work on removing GNU specific calls.
810
  my_stpcpy(path+length, MY_DB_OPT_FILE);		// Append db option file name
1 by brian
clean slate
811
  del_dbopt(path);				// Remove dboption hash entry
812
  path[length]= '\0';				// Remove file name
813
814
  /* See if the directory exists */
815
  if (!(dirp= my_dir(path,MYF(MY_DONT_SORT))))
816
  {
817
    if (!if_exists)
818
    {
819
      error= -1;
820
      my_error(ER_DB_DROP_EXISTS, MYF(0), db);
821
      goto exit;
822
    }
823
    else
520.1.22 by Brian Aker
Second pass of thd cleanup
824
      push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
825
			  ER_DB_DROP_EXISTS, ER(ER_DB_DROP_EXISTS), db);
826
  }
827
  else
828
  {
829
    pthread_mutex_lock(&LOCK_open);
830
    remove_db_from_cache(db);
831
    pthread_mutex_unlock(&LOCK_open);
832
833
    
834
    error= -1;
520.1.22 by Brian Aker
Second pass of thd cleanup
835
    if ((deleted= mysql_rm_known_files(session, dirp, db, path, 0,
1 by brian
clean slate
836
                                       &dropped_tables)) >= 0)
837
    {
838
      ha_drop_database(path);
839
      error = 0;
840
    }
841
  }
842
  if (!silent && deleted>=0)
843
  {
844
    const char *query;
308 by Brian Aker
ulong conversion
845
    uint32_t query_length;
520.1.22 by Brian Aker
Second pass of thd cleanup
846
    if (!session->query)
1 by brian
clean slate
847
    {
848
      /* The client used the old obsolete mysql_drop_db() call */
849
      query= path;
850
      query_length= (uint) (strxmov(path, "drop database `", db, "`",
461 by Monty Taylor
Removed NullS. bu-bye.
851
                                     NULL) - path);
1 by brian
clean slate
852
    }
853
    else
854
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
855
      query =session->query;
856
      query_length= session->query_length;
1 by brian
clean slate
857
    }
586.1.1 by Yoshinori Sano
Rename mysql to drizzle, specifically mysql_bin_log to drizzle_bin_log.
858
    if (drizzle_bin_log.is_open())
1 by brian
clean slate
859
    {
520.1.22 by Brian Aker
Second pass of thd cleanup
860
      Query_log_event qinfo(session, query, query_length, 0, 
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
861
			    /* suppress_use */ true);
1 by brian
clean slate
862
      /*
863
        Write should use the database being created as the "current
864
        database" and not the threads current database, which is the
865
        default.
866
      */
867
      qinfo.db     = db;
868
      qinfo.db_len = strlen(db);
869
520.1.22 by Brian Aker
Second pass of thd cleanup
870
      session->clear_error();
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
871
      /* These DDL methods and logging protected with LOCK_drizzle_create_db */
586.1.1 by Yoshinori Sano
Rename mysql to drizzle, specifically mysql_bin_log to drizzle_bin_log.
872
      drizzle_bin_log.write(&qinfo);
1 by brian
clean slate
873
    }
520.1.22 by Brian Aker
Second pass of thd cleanup
874
    session->clear_error();
875
    session->server_status|= SERVER_STATUS_DB_DROPPED;
876
    my_ok(session, (uint32_t) deleted);
877
    session->server_status&= ~SERVER_STATUS_DB_DROPPED;
1 by brian
clean slate
878
  }
586.1.1 by Yoshinori Sano
Rename mysql to drizzle, specifically mysql_bin_log to drizzle_bin_log.
879
  else if (drizzle_bin_log.is_open())
1 by brian
clean slate
880
  {
881
    char *query, *query_pos, *query_end, *query_data_start;
327.2.4 by Brian Aker
Refactoring table.h
882
    TableList *tbl;
482 by Brian Aker
Remove uint.
883
    uint32_t db_len;
1 by brian
clean slate
884
520.1.22 by Brian Aker
Second pass of thd cleanup
885
    if (!(query= (char*) session->alloc(MAX_DROP_TABLE_Q_LEN)))
1 by brian
clean slate
886
      goto exit; /* not much else we can do */
411.1.1 by Brian Aker
Work on removing GNU specific calls.
887
    query_pos= query_data_start= my_stpcpy(query,"drop table ");
1 by brian
clean slate
888
    query_end= query + MAX_DROP_TABLE_Q_LEN;
889
    db_len= strlen(db);
890
891
    for (tbl= dropped_tables; tbl; tbl= tbl->next_local)
892
    {
482 by Brian Aker
Remove uint.
893
      uint32_t tbl_name_len;
1 by brian
clean slate
894
895
      /* 3 for the quotes and the comma*/
896
      tbl_name_len= strlen(tbl->table_name) + 3;
897
      if (query_pos + tbl_name_len + 1 >= query_end)
898
      {
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
899
        /* These DDL methods and logging protected with LOCK_drizzle_create_db */
520.1.22 by Brian Aker
Second pass of thd cleanup
900
        write_to_binlog(session, query, query_pos -1 - query, db, db_len);
1 by brian
clean slate
901
        query_pos= query_data_start;
902
      }
903
904
      *query_pos++ = '`';
411.1.1 by Brian Aker
Work on removing GNU specific calls.
905
      query_pos= my_stpcpy(query_pos,tbl->table_name);
1 by brian
clean slate
906
      *query_pos++ = '`';
907
      *query_pos++ = ',';
908
    }
909
910
    if (query_pos != query_data_start)
911
    {
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
912
      /* These DDL methods and logging protected with LOCK_drizzle_create_db */
520.1.22 by Brian Aker
Second pass of thd cleanup
913
      write_to_binlog(session, query, query_pos -1 - query, db, db_len);
1 by brian
clean slate
914
    }
915
  }
916
917
exit:
918
  /*
919
    If this database was the client's selected database, we silently
920
    change the client's selected database to nothing (to have an empty
520.1.22 by Brian Aker
Second pass of thd cleanup
921
    SELECT DATABASE() in the future). For this we free() session->db and set
1 by brian
clean slate
922
    it to 0.
923
  */
520.1.22 by Brian Aker
Second pass of thd cleanup
924
  if (session->db && !strcmp(session->db, db))
925
    mysql_change_db_impl(session, NULL, session->variables.collation_server);
575.4.4 by Yoshinori Sano
Rename mysql to drizzle.
926
  pthread_mutex_unlock(&LOCK_drizzle_create_db);
520.1.22 by Brian Aker
Second pass of thd cleanup
927
  start_waiting_global_read_lock(session);
1 by brian
clean slate
928
exit2:
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
929
  return(error);
1 by brian
clean slate
930
}
931
932
/*
255 by Brian Aker
Removed RAID table delete point.
933
  Removes files with known extensions plus.
520.1.22 by Brian Aker
Second pass of thd cleanup
934
  session MUST be set when calling this function!
1 by brian
clean slate
935
*/
936
520.1.22 by Brian Aker
Second pass of thd cleanup
937
static long mysql_rm_known_files(Session *session, MY_DIR *dirp, const char *db,
482 by Brian Aker
Remove uint.
938
				 const char *org_path, uint32_t level,
327.2.4 by Brian Aker
Refactoring table.h
939
                                 TableList **dropped_tables)
1 by brian
clean slate
940
{
941
  long deleted=0;
308 by Brian Aker
ulong conversion
942
  uint32_t found_other_files=0;
1 by brian
clean slate
943
  char filePath[FN_REFLEN];
327.2.4 by Brian Aker
Refactoring table.h
944
  TableList *tot_list=0, **tot_list_next;
1 by brian
clean slate
945
946
  tot_list_next= &tot_list;
947
482 by Brian Aker
Remove uint.
948
  for (uint32_t idx=0 ;
520.1.22 by Brian Aker
Second pass of thd cleanup
949
       idx < (uint) dirp->number_off_files && !session->killed ;
1 by brian
clean slate
950
       idx++)
951
  {
952
    FILEINFO *file=dirp->dir_entry+idx;
953
    char *extension;
954
955
    /* skiping . and .. */
956
    if (file->name[0] == '.' && (!file->name[1] ||
957
       (file->name[1] == '.' &&  !file->name[2])))
958
      continue;
959
960
    if (!(extension= strrchr(file->name, '.')))
376 by Brian Aker
strend remove
961
      extension= strchr(file->name, '\0');
1 by brian
clean slate
962
    if (find_type(extension, &deletable_extentions,1+2) <= 0)
963
    {
584.2.7 by Stewart Smith
rename and delete tabledefinition protobuf file
964
      /*
965
        ass ass ass.
966
967
        strange checking for magic extensions that are then deleted if
968
        not reg_ext (i.e. .frm).
969
970
        and (previously) we'd err out on drop database if files not matching
971
        engine ha_known_exts() or deletable_extensions were present.
972
973
        presumably this was to avoid deleting other user data... except if that
974
        data happened to be in files ending in .BAK, .opt or .TMD. *fun*
975
       */
976
      find_type(extension, ha_known_exts(),1+2);
1 by brian
clean slate
977
      continue;
978
    }
979
    /* just for safety we use files_charset_info */
980
    if (db && !my_strcasecmp(files_charset_info,
981
                             extension, reg_ext))
982
    {
983
      /* Drop the table nicely */
984
      *extension= 0;			// Remove extension
327.2.4 by Brian Aker
Refactoring table.h
985
      TableList *table_list=(TableList*)
520.1.22 by Brian Aker
Second pass of thd cleanup
986
                              session->calloc(sizeof(*table_list) + 
1 by brian
clean slate
987
                                          strlen(db) + 1 +
988
                                          MYSQL50_TABLE_NAME_PREFIX_LENGTH + 
989
                                          strlen(file->name) + 1);
990
991
      if (!table_list)
992
        goto err;
993
      table_list->db= (char*) (table_list+1);
411.1.1 by Brian Aker
Work on removing GNU specific calls.
994
      table_list->table_name= my_stpcpy(table_list->db, db) + 1;
398.1.10 by Monty Taylor
Actually removed VOID() this time.
995
      filename_to_tablename(file->name, table_list->table_name,
996
                            MYSQL50_TABLE_NAME_PREFIX_LENGTH +
997
                            strlen(file->name) + 1);
1 by brian
clean slate
998
      table_list->alias= table_list->table_name;	// If lower_case_table_names=2
584.2.1 by Stewart Smith
convert tmp_file_prefix to TMP_FILE_PREFIX to indicate it's a macro
999
      table_list->internal_tmp_table= is_prefix(file->name, TMP_FILE_PREFIX);
1 by brian
clean slate
1000
      /* Link into list */
1001
      (*tot_list_next)= table_list;
1002
      tot_list_next= &table_list->next_local;
1003
      deleted++;
1004
    }
1005
    else
1006
    {
461 by Monty Taylor
Removed NullS. bu-bye.
1007
      strxmov(filePath, org_path, "/", file->name, NULL);
1 by brian
clean slate
1008
      if (my_delete_with_symlink(filePath,MYF(MY_WME)))
1009
      {
1010
	goto err;
1011
      }
1012
    }
1013
  }
520.1.22 by Brian Aker
Second pass of thd cleanup
1014
  if (session->killed ||
590.1.7 by Stewart Smith
remove mysql_frm_type
1015
      (tot_list && mysql_rm_table_part2(session, tot_list, 1, 0, 1)))
1 by brian
clean slate
1016
    goto err;
1017
1018
  my_dirend(dirp);  
1019
  
1020
  if (dropped_tables)
1021
    *dropped_tables= tot_list;
1022
  
1023
  /*
1024
    If the directory is a symbolic link, remove the link first, then
1025
    remove the directory the symbolic link pointed at
1026
  */
1027
  if (found_other_files)
1028
  {
1029
    my_error(ER_DB_DROP_RMDIR, MYF(0), org_path, EEXIST);
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1030
    return(-1);
1 by brian
clean slate
1031
  }
1032
  else
1033
  {
1034
    /* Don't give errors if we can't delete 'RAID' directory */
1035
    if (rm_dir_w_symlink(org_path, level == 0))
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1036
      return(-1);
1 by brian
clean slate
1037
  }
1038
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1039
  return(deleted);
1 by brian
clean slate
1040
1041
err:
1042
  my_dirend(dirp);
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1043
  return(-1);
1 by brian
clean slate
1044
}
1045
1046
1047
/*
1048
  Remove directory with symlink
1049
1050
  SYNOPSIS
1051
    rm_dir_w_symlink()
1052
    org_path    path of derictory
1053
    send_error  send errors
1054
  RETURN
1055
    0 OK
1056
    1 ERROR
1057
*/
1058
253 by Brian Aker
Removed final my_bool from sql_db.
1059
static bool rm_dir_w_symlink(const char *org_path, bool send_error)
1 by brian
clean slate
1060
{
1061
  char tmp_path[FN_REFLEN], *pos;
1062
  char *path= tmp_path;
1063
  unpack_filename(tmp_path, org_path);
1064
#ifdef HAVE_READLINK
1065
  int error;
1066
  char tmp2_path[FN_REFLEN];
1067
1068
  /* Remove end FN_LIBCHAR as this causes problem on Linux in readlink */
376 by Brian Aker
strend remove
1069
  pos= strchr(path, '\0');
1 by brian
clean slate
1070
  if (pos > path && pos[-1] == FN_LIBCHAR)
1071
    *--pos=0;
1072
1073
  if ((error= my_readlink(tmp2_path, path, MYF(MY_WME))) < 0)
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1074
    return(1);
1 by brian
clean slate
1075
  if (!error)
1076
  {
1077
    if (my_delete(path, MYF(send_error ? MY_WME : 0)))
1078
    {
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1079
      return(send_error);
1 by brian
clean slate
1080
    }
1081
    /* Delete directory symbolic link pointed at */
1082
    path= tmp2_path;
1083
  }
1084
#endif
1085
  /* Remove last FN_LIBCHAR to not cause a problem on OS/2 */
376 by Brian Aker
strend remove
1086
  pos= strchr(path, '\0');
1 by brian
clean slate
1087
1088
  if (pos > path && pos[-1] == FN_LIBCHAR)
1089
    *--pos=0;
1090
  if (rmdir(path) < 0 && send_error)
1091
  {
1092
    my_error(ER_DB_DROP_RMDIR, MYF(0), path, errno);
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1093
    return(1);
1 by brian
clean slate
1094
  }
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1095
  return(0);
1 by brian
clean slate
1096
}
1097
1098
1099
/**
1100
  @brief Internal implementation: switch current database to a valid one.
1101
520.1.22 by Brian Aker
Second pass of thd cleanup
1102
  @param session            Thread context.
1 by brian
clean slate
1103
  @param new_db_name    Name of the database to switch to. The function will
1104
                        take ownership of the name (the caller must not free
1105
                        the allocated memory). If the name is NULL, we're
1106
                        going to switch to NULL db.
1107
  @param new_db_charset Character set of the new database.
1108
*/
1109
520.1.22 by Brian Aker
Second pass of thd cleanup
1110
static void mysql_change_db_impl(Session *session,
1 by brian
clean slate
1111
                                 LEX_STRING *new_db_name,
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
1112
                                 const CHARSET_INFO * const new_db_charset)
1 by brian
clean slate
1113
{
520.1.21 by Brian Aker
THD -> Session rename
1114
  /* 1. Change current database in Session. */
1 by brian
clean slate
1115
1116
  if (new_db_name == NULL)
1117
  {
1118
    /*
520.1.21 by Brian Aker
THD -> Session rename
1119
      Session::set_db() does all the job -- it frees previous database name and
1 by brian
clean slate
1120
      sets the new one.
1121
    */
1122
520.1.22 by Brian Aker
Second pass of thd cleanup
1123
    session->set_db(NULL, 0);
1 by brian
clean slate
1124
  }
575.4.7 by Monty Taylor
More header cleanup.
1125
  else if (my_strcasecmp(system_charset_info, new_db_name->str,
1126
                         INFORMATION_SCHEMA_NAME.c_str()) == 0)
1 by brian
clean slate
1127
  {
1128
    /*
520.1.21 by Brian Aker
THD -> Session rename
1129
      Here we must use Session::set_db(), because we want to copy
1 by brian
clean slate
1130
      INFORMATION_SCHEMA_NAME constant.
1131
    */
1132
575.4.7 by Monty Taylor
More header cleanup.
1133
    session->set_db(INFORMATION_SCHEMA_NAME.c_str(),
1134
                    INFORMATION_SCHEMA_NAME.length());
1 by brian
clean slate
1135
  }
1136
  else
1137
  {
1138
    /*
520.1.21 by Brian Aker
THD -> Session rename
1139
      Here we already have a copy of database name to be used in Session. So,
1140
      we just call Session::reset_db(). Since Session::reset_db() does not releases
1 by brian
clean slate
1141
      the previous database name, we should do it explicitly.
1142
    */
1143
520.1.22 by Brian Aker
Second pass of thd cleanup
1144
    if (session->db)
1145
      free(session->db);
1 by brian
clean slate
1146
520.1.22 by Brian Aker
Second pass of thd cleanup
1147
    session->reset_db(new_db_name->str, new_db_name->length);
1 by brian
clean slate
1148
  }
1149
1150
  /* 3. Update db-charset environment variables. */
1151
520.1.22 by Brian Aker
Second pass of thd cleanup
1152
  session->db_charset= new_db_charset;
1153
  session->variables.collation_database= new_db_charset;
1 by brian
clean slate
1154
}
1155
1156
1157
1158
/**
1159
  Backup the current database name before switch.
1160
520.1.22 by Brian Aker
Second pass of thd cleanup
1161
  @param[in]      session             thread handle
1 by brian
clean slate
1162
  @param[in, out] saved_db_name   IN: "str" points to a buffer where to store
1163
                                  the old database name, "length" contains the
1164
                                  buffer size
1165
                                  OUT: if the current (default) database is
1166
                                  not NULL, its name is copied to the
1167
                                  buffer pointed at by "str"
1168
                                  and "length" is updated accordingly.
1169
                                  Otherwise "str" is set to NULL and
1170
                                  "length" is set to 0.
1171
*/
1172
520.1.22 by Brian Aker
Second pass of thd cleanup
1173
static void backup_current_db_name(Session *session,
1 by brian
clean slate
1174
                                   LEX_STRING *saved_db_name)
1175
{
520.1.22 by Brian Aker
Second pass of thd cleanup
1176
  if (!session->db)
1 by brian
clean slate
1177
  {
1178
    /* No current (default) database selected. */
1179
1180
    saved_db_name->str= NULL;
1181
    saved_db_name->length= 0;
1182
  }
1183
  else
1184
  {
520.1.22 by Brian Aker
Second pass of thd cleanup
1185
    strmake(saved_db_name->str, session->db, saved_db_name->length - 1);
1186
    saved_db_name->length= session->db_length;
1 by brian
clean slate
1187
  }
1188
}
1189
1190
1191
/**
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1192
  Return true if db1_name is equal to db2_name, false otherwise.
1 by brian
clean slate
1193
1194
  The function allows to compare database names according to the MySQL
1195
  rules. The database names db1 and db2 are equal if:
1196
     - db1 is NULL and db2 is NULL;
1197
     or
1198
     - db1 is not-NULL, db2 is not-NULL, db1 is equal (ignoring case) to
1199
       db2 in system character set (UTF8).
1200
*/
1201
1202
static inline bool
1203
cmp_db_names(const char *db1_name,
1204
             const char *db2_name)
1205
{
1206
  return
1207
         /* db1 is NULL and db2 is NULL */
1208
         (!db1_name && !db2_name) ||
1209
1210
         /* db1 is not-NULL, db2 is not-NULL, db1 == db2. */
1211
         (db1_name && db2_name && my_strcasecmp(system_charset_info, db1_name, db2_name) == 0);
1212
}
1213
1214
1215
/**
1216
  @brief Change the current database and its attributes unconditionally.
1217
520.1.22 by Brian Aker
Second pass of thd cleanup
1218
  @param session          thread handle
1 by brian
clean slate
1219
  @param new_db_name  database name
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1220
  @param force_switch if force_switch is false, then the operation will fail if
1 by brian
clean slate
1221
1222
                        - new_db_name is NULL or empty;
1223
1224
                        - OR new database name is invalid
1225
                          (check_db_name() failed);
1226
1227
                        - OR user has no privilege on the new database;
1228
1229
                        - OR new database does not exist;
1230
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1231
                      if force_switch is true, then
1 by brian
clean slate
1232
1233
                        - if new_db_name is NULL or empty, the current
1234
                          database will be NULL, @@collation_database will
1235
                          be set to @@collation_server, the operation will
1236
                          succeed.
1237
1238
                        - if new database name is invalid
1239
                          (check_db_name() failed), the current database
1240
                          will be NULL, @@collation_database will be set to
1241
                          @@collation_server, but the operation will fail;
1242
1243
                        - user privileges will not be checked
520.1.21 by Brian Aker
THD -> Session rename
1244
                          (Session::db_access however is updated);
1 by brian
clean slate
1245
1246
                          TODO: is this really the intention?
1247
                                (see sp-security.test).
1248
1249
                        - if new database does not exist,the current database
1250
                          will be NULL, @@collation_database will be set to
1251
                          @@collation_server, a warning will be thrown, the
1252
                          operation will succeed.
1253
1254
  @details The function checks that the database name corresponds to a
1255
  valid and existent database, checks access rights and changes the current
1256
  database with database attributes (@@collation_database session variable,
520.1.21 by Brian Aker
THD -> Session rename
1257
  Session::db_access).
1 by brian
clean slate
1258
1259
  This function is not the only way to switch the database that is
1260
  currently employed. When the replication slave thread switches the
520.1.22 by Brian Aker
Second pass of thd cleanup
1261
  database before executing a query, it calls session->set_db directly.
1 by brian
clean slate
1262
  However, if the query, in turn, uses a stored routine, the stored routine
1263
  will use this function, even if it's run on the slave.
1264
1265
  This function allocates the name of the database on the system heap: this
1266
  is necessary to be able to uniformly change the database from any module
1267
  of the server. Up to 5.0 different modules were using different memory to
1268
  store the name of the database, and this led to memory corruption:
1269
  a stack pointer set by Stored Procedures was used by replication after
1270
  the stack address was long gone.
1271
1272
  @return Operation status
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1273
    @retval false Success
1274
    @retval true  Error
1 by brian
clean slate
1275
*/
1276
520.1.22 by Brian Aker
Second pass of thd cleanup
1277
bool mysql_change_db(Session *session, const LEX_STRING *new_db_name, bool force_switch)
1 by brian
clean slate
1278
{
1279
  LEX_STRING new_db_file_name;
264.2.6 by Andrey Hristov
Constify the usage of CHARSET_INFO almost to the last place in the code.
1280
  const CHARSET_INFO *db_default_cl;
1 by brian
clean slate
1281
1282
  if (new_db_name == NULL ||
1283
      new_db_name->length == 0)
1284
  {
1285
    if (force_switch)
1286
    {
1287
      /*
1288
        This can happen only if we're switching the current database back
1289
        after loading stored program. The thing is that loading of stored
1290
        program can happen when there is no current database.
1291
1292
        TODO: actually, new_db_name and new_db_name->str seem to be always
1293
        non-NULL. In case of stored program, new_db_name->str == "" and
1294
        new_db_name->length == 0.
1295
      */
1296
520.1.22 by Brian Aker
Second pass of thd cleanup
1297
      mysql_change_db_impl(session, NULL, session->variables.collation_server);
1 by brian
clean slate
1298
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1299
      return(false);
1 by brian
clean slate
1300
    }
1301
    else
1302
    {
1303
      my_message(ER_NO_DB_ERROR, ER(ER_NO_DB_ERROR), MYF(0));
1304
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1305
      return(true);
1 by brian
clean slate
1306
    }
1307
  }
1308
1309
  if (my_strcasecmp(system_charset_info, new_db_name->str,
575.4.7 by Monty Taylor
More header cleanup.
1310
                    INFORMATION_SCHEMA_NAME.c_str()) == 0)
1 by brian
clean slate
1311
  {
1312
    /* Switch the current database to INFORMATION_SCHEMA. */
575.4.7 by Monty Taylor
More header cleanup.
1313
    /* const_cast<> is safe here: mysql_change_db_impl does a copy */
1314
    LEX_STRING is_name= { const_cast<char *>(INFORMATION_SCHEMA_NAME.c_str()),
1315
                          INFORMATION_SCHEMA_NAME.length() };
1316
    mysql_change_db_impl(session, &is_name, system_charset_info);
1 by brian
clean slate
1317
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1318
    return(false);
1 by brian
clean slate
1319
  }
1320
1321
  /*
1322
    Now we need to make a copy because check_db_name requires a
1323
    non-constant argument. Actually, it takes database file name.
1324
1325
    TODO: fix check_db_name().
1326
  */
1327
1328
  new_db_file_name.str= my_strndup(new_db_name->str, new_db_name->length,
1329
                                   MYF(MY_WME));
1330
  new_db_file_name.length= new_db_name->length;
1331
1332
  if (new_db_file_name.str == NULL)
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1333
    return(true);                             /* the error is set */
1 by brian
clean slate
1334
1335
  /*
1336
    NOTE: if check_db_name() fails, we should throw an error in any case,
1337
    even if we are called from sp_head::execute().
1338
1339
    It's next to impossible however to get this error when we are called
1340
    from sp_head::execute(). But let's switch the current database to NULL
1341
    in this case to be sure.
1342
  */
1343
1344
  if (check_db_name(&new_db_file_name))
1345
  {
1346
    my_error(ER_WRONG_DB_NAME, MYF(0), new_db_file_name.str);
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
1347
    free(new_db_file_name.str);
1 by brian
clean slate
1348
1349
    if (force_switch)
520.1.22 by Brian Aker
Second pass of thd cleanup
1350
      mysql_change_db_impl(session, NULL, session->variables.collation_server);
1 by brian
clean slate
1351
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1352
    return(true);
1 by brian
clean slate
1353
  }
1354
1355
  if (check_db_dir_existence(new_db_file_name.str))
1356
  {
1357
    if (force_switch)
1358
    {
1359
      /* Throw a warning and free new_db_file_name. */
1360
520.1.22 by Brian Aker
Second pass of thd cleanup
1361
      push_warning_printf(session, DRIZZLE_ERROR::WARN_LEVEL_NOTE,
1 by brian
clean slate
1362
                          ER_BAD_DB_ERROR, ER(ER_BAD_DB_ERROR),
1363
                          new_db_file_name.str);
1364
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
1365
      free(new_db_file_name.str);
1 by brian
clean slate
1366
1367
      /* Change db to NULL. */
1368
520.1.22 by Brian Aker
Second pass of thd cleanup
1369
      mysql_change_db_impl(session, NULL, session->variables.collation_server);
1 by brian
clean slate
1370
1371
      /* The operation succeed. */
1372
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1373
      return(false);
1 by brian
clean slate
1374
    }
1375
    else
1376
    {
1377
      /* Report an error and free new_db_file_name. */
1378
1379
      my_error(ER_BAD_DB_ERROR, MYF(0), new_db_file_name.str);
477 by Monty Taylor
Removed my_free(). It turns out that it had been def'd to ignore the flags passed to it in the second arg anyway. Gotta love that.
1380
      free(new_db_file_name.str);
1 by brian
clean slate
1381
1382
      /* The operation failed. */
1383
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1384
      return(true);
1 by brian
clean slate
1385
    }
1386
  }
1387
1388
  /*
520.1.21 by Brian Aker
THD -> Session rename
1389
    NOTE: in mysql_change_db_impl() new_db_file_name is assigned to Session
1390
    attributes and will be freed in Session::~Session().
1 by brian
clean slate
1391
  */
1392
520.1.22 by Brian Aker
Second pass of thd cleanup
1393
  db_default_cl= get_default_db_collation(session, new_db_file_name.str);
1 by brian
clean slate
1394
520.1.22 by Brian Aker
Second pass of thd cleanup
1395
  mysql_change_db_impl(session, &new_db_file_name, db_default_cl);
1 by brian
clean slate
1396
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1397
  return(false);
1 by brian
clean slate
1398
}
1399
1400
1401
/**
1402
  Change the current database and its attributes if needed.
1403
520.1.22 by Brian Aker
Second pass of thd cleanup
1404
  @param          session             thread handle
1 by brian
clean slate
1405
  @param          new_db_name     database name
1406
  @param[in, out] saved_db_name   IN: "str" points to a buffer where to store
1407
                                  the old database name, "length" contains the
1408
                                  buffer size
1409
                                  OUT: if the current (default) database is
1410
                                  not NULL, its name is copied to the
1411
                                  buffer pointed at by "str"
1412
                                  and "length" is updated accordingly.
1413
                                  Otherwise "str" is set to NULL and
1414
                                  "length" is set to 0.
1415
  @param          force_switch    @see mysql_change_db()
1416
  @param[out]     cur_db_changed  out-flag to indicate whether the current
1417
                                  database has been changed (valid only if
1418
                                  the function suceeded)
1419
*/
1420
520.1.22 by Brian Aker
Second pass of thd cleanup
1421
bool mysql_opt_change_db(Session *session,
1 by brian
clean slate
1422
                         const LEX_STRING *new_db_name,
1423
                         LEX_STRING *saved_db_name,
1424
                         bool force_switch,
1425
                         bool *cur_db_changed)
1426
{
520.1.22 by Brian Aker
Second pass of thd cleanup
1427
  *cur_db_changed= !cmp_db_names(session->db, new_db_name->str);
1 by brian
clean slate
1428
1429
  if (!*cur_db_changed)
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1430
    return false;
1 by brian
clean slate
1431
520.1.22 by Brian Aker
Second pass of thd cleanup
1432
  backup_current_db_name(session, saved_db_name);
1 by brian
clean slate
1433
520.1.22 by Brian Aker
Second pass of thd cleanup
1434
  return mysql_change_db(session, new_db_name, force_switch);
1 by brian
clean slate
1435
}
1436
1437
1438
/*
1439
  Check if there is directory for the database name.
1440
1441
  SYNOPSIS
1442
    check_db_dir_existence()
1443
    db_name   database name
1444
1445
  RETURN VALUES
51.1.51 by Jay Pipes
Removed/replaced DBUG symbols and standardized TRUE/FALSE
1446
    false   There is directory for the specified database name.
1447
    true    The directory does not exist.
1 by brian
clean slate
1448
*/
1449
1450
bool check_db_dir_existence(const char *db_name)
1451
{
1452
  char db_dir_path[FN_REFLEN];
482 by Brian Aker
Remove uint.
1453
  uint32_t db_dir_path_len;
1 by brian
clean slate
1454
1455
  db_dir_path_len= build_table_filename(db_dir_path, sizeof(db_dir_path),
1456
                                        db_name, "", "", 0);
1457
1458
  if (db_dir_path_len && db_dir_path[db_dir_path_len - 1] == FN_LIBCHAR)
1459
    db_dir_path[db_dir_path_len - 1]= 0;
1460
1461
  /* Check access. */
1462
1463
  return my_access(db_dir_path, F_OK);
1464
}