~drizzle-trunk/drizzle/development

1415 by Brian Aker
Mass overhaul to use schema_identifier.
1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
 *  vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
3
 *
4
 *  Copyright (C) 2010 Brian Aker
5
 *
6
 *  This program is free software; you can redistribute it and/or modify
7
 *  it under the terms of the GNU General Public License as published by
8
 *  the Free Software Foundation; version 2 of the License.
9
 *
10
 *  This program is distributed in the hope that it will be useful,
11
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
12
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13
 *  GNU General Public License for more details.
14
 *
15
 *  You should have received a copy of the GNU General Public License
16
 *  along with this program; if not, write to the Free Software
17
 *  Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
18
 */
19
20
#include "config.h"
21
22
#include "drizzled/session.h"
23
24
#include "drizzled/global_charset_info.h"
25
#include "drizzled/charset.h"
1979.1.1 by Joseph Daly
move trx id assignment outside of schema engine and in plugin/schema_engine/schema.cc
26
#include "drizzled/transaction_services.h"
1415 by Brian Aker
Mass overhaul to use schema_identifier.
27
28
#include "drizzled/plugin/storage_engine.h"
29
#include "drizzled/plugin/authorization.h"
30
31
namespace drizzled
32
{
33
34
namespace plugin
35
{
36
37
class AddSchemaNames : 
1966.2.9 by Brian Aker
Remove the use of "using std" from the plugin interface .cc files.
38
  public std::unary_function<StorageEngine *, void>
1415 by Brian Aker
Mass overhaul to use schema_identifier.
39
{
1966.2.3 by Brian Aker
Fix another style issue.
40
  SchemaIdentifier::vector &schemas;
1415 by Brian Aker
Mass overhaul to use schema_identifier.
41
42
public:
43
1966.2.3 by Brian Aker
Fix another style issue.
44
  AddSchemaNames(SchemaIdentifier::vector &of_names) :
1415 by Brian Aker
Mass overhaul to use schema_identifier.
45
    schemas(of_names)
46
  {
47
  }
48
49
  result_type operator() (argument_type engine)
50
  {
51
    engine->doGetSchemaIdentifiers(schemas);
52
  }
53
};
54
1966.2.3 by Brian Aker
Fix another style issue.
55
void StorageEngine::getIdentifiers(Session &session, SchemaIdentifier::vector &schemas)
1415 by Brian Aker
Mass overhaul to use schema_identifier.
56
{
57
  // Add hook here for engines to register schema.
1966.2.14 by Brian Aker
Merge in some additional std namespace finds.
58
  std::for_each(StorageEngine::getSchemaEngines().begin(), StorageEngine::getSchemaEngines().end(),
1415 by Brian Aker
Mass overhaul to use schema_identifier.
59
           AddSchemaNames(schemas));
60
2008.1.1 by Brian Aker
Adding user identifier that makes use of a shared ptr to handle concurrency
61
  plugin::Authorization::pruneSchemaNames(session.user(), schemas);
1415 by Brian Aker
Mass overhaul to use schema_identifier.
62
}
63
1966.2.9 by Brian Aker
Remove the use of "using std" from the plugin interface .cc files.
64
class StorageEngineGetSchemaDefinition: public std::unary_function<StorageEngine *, bool>
1415 by Brian Aker
Mass overhaul to use schema_identifier.
65
{
1642 by Brian Aker
This adds const to SchemaIdentifier.
66
  const SchemaIdentifier &identifier;
1966.2.1 by Brian Aker
Clean up style for shared_ptr for messages around table/schema.
67
  message::schema::shared_ptr &schema_proto;
1415 by Brian Aker
Mass overhaul to use schema_identifier.
68
69
public:
1642 by Brian Aker
This adds const to SchemaIdentifier.
70
  StorageEngineGetSchemaDefinition(const SchemaIdentifier &identifier_arg,
1966.2.1 by Brian Aker
Clean up style for shared_ptr for messages around table/schema.
71
                                   message::schema::shared_ptr &schema_proto_arg) :
1415 by Brian Aker
Mass overhaul to use schema_identifier.
72
    identifier(identifier_arg),
73
    schema_proto(schema_proto_arg) 
74
  {
75
  }
76
77
  result_type operator() (argument_type engine)
78
  {
79
    return engine->doGetSchemaDefinition(identifier, schema_proto);
80
  }
81
};
82
83
/*
84
  Return value is "if parsed"
85
*/
1966.2.1 by Brian Aker
Clean up style for shared_ptr for messages around table/schema.
86
bool StorageEngine::getSchemaDefinition(const drizzled::TableIdentifier &identifier, message::schema::shared_ptr &proto)
1415 by Brian Aker
Mass overhaul to use schema_identifier.
87
{
88
  return StorageEngine::getSchemaDefinition(identifier, proto);
89
}
90
1966.2.1 by Brian Aker
Clean up style for shared_ptr for messages around table/schema.
91
bool StorageEngine::getSchemaDefinition(const SchemaIdentifier &identifier, message::schema::shared_ptr &proto)
1415 by Brian Aker
Mass overhaul to use schema_identifier.
92
{
93
  EngineVector::iterator iter=
1966.2.14 by Brian Aker
Merge in some additional std namespace finds.
94
    std::find_if(StorageEngine::getSchemaEngines().begin(), StorageEngine::getSchemaEngines().end(),
95
                 StorageEngineGetSchemaDefinition(identifier, proto));
1415 by Brian Aker
Mass overhaul to use schema_identifier.
96
97
  if (iter != StorageEngine::getSchemaEngines().end())
98
  {
99
    return true;
100
  }
101
102
  return false;
103
}
104
1642 by Brian Aker
This adds const to SchemaIdentifier.
105
bool StorageEngine::doesSchemaExist(const SchemaIdentifier &identifier)
1415 by Brian Aker
Mass overhaul to use schema_identifier.
106
{
1966.2.1 by Brian Aker
Clean up style for shared_ptr for messages around table/schema.
107
  message::schema::shared_ptr proto;
1415 by Brian Aker
Mass overhaul to use schema_identifier.
108
109
  return StorageEngine::getSchemaDefinition(identifier, proto);
110
}
111
112
1642 by Brian Aker
This adds const to SchemaIdentifier.
113
const CHARSET_INFO *StorageEngine::getSchemaCollation(const SchemaIdentifier &identifier)
1415 by Brian Aker
Mass overhaul to use schema_identifier.
114
{
1966.2.1 by Brian Aker
Clean up style for shared_ptr for messages around table/schema.
115
  message::schema::shared_ptr schmema_proto;
1415 by Brian Aker
Mass overhaul to use schema_identifier.
116
  bool found;
117
118
  found= StorageEngine::getSchemaDefinition(identifier, schmema_proto);
119
1812.3.8 by Brian Aker
This switches our schema system over to using a shared ptr. Lets see how
120
  if (found && schmema_proto->has_collation())
1415 by Brian Aker
Mass overhaul to use schema_identifier.
121
  {
1966.2.9 by Brian Aker
Remove the use of "using std" from the plugin interface .cc files.
122
    const std::string buffer= schmema_proto->collation();
1415 by Brian Aker
Mass overhaul to use schema_identifier.
123
    const CHARSET_INFO* cs= get_charset_by_name(buffer.c_str());
124
125
    if (not cs)
126
    {
1954.2.1 by Brian Aker
getSQLPath() modified to take a string so that we can const the table
127
      std::string path;
128
      identifier.getSQLPath(path);
129
1415 by Brian Aker
Mass overhaul to use schema_identifier.
130
      errmsg_printf(ERRMSG_LVL_ERROR,
1954.2.1 by Brian Aker
getSQLPath() modified to take a string so that we can const the table
131
                    _("Error while loading database options: '%s':"), path.c_str());
1415 by Brian Aker
Mass overhaul to use schema_identifier.
132
      errmsg_printf(ERRMSG_LVL_ERROR, ER(ER_UNKNOWN_COLLATION), buffer.c_str());
133
134
      return default_charset_info;
135
    }
136
137
    return cs;
138
  }
139
140
  return default_charset_info;
141
}
142
143
class CreateSchema : 
1966.2.9 by Brian Aker
Remove the use of "using std" from the plugin interface .cc files.
144
  public std::unary_function<StorageEngine *, void>
1415 by Brian Aker
Mass overhaul to use schema_identifier.
145
{
146
  const drizzled::message::Schema &schema_message;
147
148
public:
149
1856.2.8 by Joseph Daly
working alter, drop, create schema
150
  CreateSchema(const drizzled::message::Schema &arg) :
151
    schema_message(arg)
1415 by Brian Aker
Mass overhaul to use schema_identifier.
152
  {
153
  }
154
155
  result_type operator() (argument_type engine)
156
  {
157
    // @todo eomeday check that at least one engine said "true"
1979.1.1 by Joseph Daly
move trx id assignment outside of schema engine and in plugin/schema_engine/schema.cc
158
    bool success= engine->doCreateSchema(schema_message);
159
160
    if (success) 
161
    {
162
      TransactionServices &transaction_services= TransactionServices::singleton();
163
      transaction_services.allocateNewTransactionId();
164
    }
1415 by Brian Aker
Mass overhaul to use schema_identifier.
165
  }
166
};
167
1856.2.8 by Joseph Daly
working alter, drop, create schema
168
bool StorageEngine::createSchema(const drizzled::message::Schema &schema_message)
1415 by Brian Aker
Mass overhaul to use schema_identifier.
169
{
170
  // Add hook here for engines to register schema.
1966.2.14 by Brian Aker
Merge in some additional std namespace finds.
171
  std::for_each(StorageEngine::getSchemaEngines().begin(), StorageEngine::getSchemaEngines().end(),
172
                CreateSchema(schema_message));
1415 by Brian Aker
Mass overhaul to use schema_identifier.
173
174
  return true;
175
}
176
177
class DropSchema : 
1966.2.9 by Brian Aker
Remove the use of "using std" from the plugin interface .cc files.
178
  public std::unary_function<StorageEngine *, void>
1415 by Brian Aker
Mass overhaul to use schema_identifier.
179
{
180
  uint64_t &success_count;
1642 by Brian Aker
This adds const to SchemaIdentifier.
181
  const SchemaIdentifier &identifier;
1415 by Brian Aker
Mass overhaul to use schema_identifier.
182
183
public:
184
1642 by Brian Aker
This adds const to SchemaIdentifier.
185
  DropSchema(const SchemaIdentifier &arg, uint64_t &count_arg) :
1415 by Brian Aker
Mass overhaul to use schema_identifier.
186
    success_count(count_arg),
187
    identifier(arg)
188
  {
189
  }
190
191
  result_type operator() (argument_type engine)
192
  {
193
    // @todo someday check that at least one engine said "true"
194
    bool success= engine->doDropSchema(identifier);
195
196
    if (success)
1979.1.1 by Joseph Daly
move trx id assignment outside of schema engine and in plugin/schema_engine/schema.cc
197
    {
1415 by Brian Aker
Mass overhaul to use schema_identifier.
198
      success_count++;
1979.1.1 by Joseph Daly
move trx id assignment outside of schema engine and in plugin/schema_engine/schema.cc
199
      TransactionServices &transaction_services= TransactionServices::singleton();
200
      transaction_services.allocateNewTransactionId();
201
    }
1415 by Brian Aker
Mass overhaul to use schema_identifier.
202
  }
203
};
204
2041.3.10 by Brian Aker
Put more of drop schema behind interface.
205
static bool drop_all_tables_in_schema(Session& session,
206
                                      SchemaIdentifier::const_reference identifier,
207
                                      TableIdentifier::vector &dropped_tables,
208
                                      uint64_t &deleted)
209
{
210
  TransactionServices &transaction_services= TransactionServices::singleton();
211
212
  plugin::StorageEngine::getIdentifiers(session, identifier, dropped_tables);
213
214
  for (TableIdentifier::vector::iterator it= dropped_tables.begin();
215
       it != dropped_tables.end();
216
       it++)
217
  {
218
    boost::mutex::scoped_lock scopedLock(table::Cache::singleton().mutex());
219
    table::Cache::singleton().removeTable(&session, *it,
220
                                          RTFC_WAIT_OTHER_THREAD_FLAG |
221
                                          RTFC_CHECK_KILLED_FLAG);
222
    if (plugin::StorageEngine::dropTable(session, *it))
223
    {
2041.3.15 by Brian Aker
Cleanup error usage around identifier usage.
224
      my_error(ER_TABLE_DROP, *it);
2041.3.10 by Brian Aker
Put more of drop schema behind interface.
225
      return false;
226
    }
2051 by Brian Aker
Second patch from david which has the transaction services using identifier.
227
    transaction_services.dropTable(&session, *it, true);
2041.3.10 by Brian Aker
Put more of drop schema behind interface.
228
    deleted++;
229
  }
230
231
  return true;
232
}
233
234
bool StorageEngine::dropSchema(Session::reference session, SchemaIdentifier::const_reference identifier)
235
{
236
  uint64_t deleted= 0;
237
  bool error= false;
238
  TableIdentifier::vector dropped_tables;
239
  message::Schema schema_proto;
240
241
  do
242
  {
243
    // Remove all temp tables first, this prevents loss of table from
244
    // shadowing (ie temp over standard table)
245
    {
246
      // Lets delete the temporary tables first outside of locks.  
247
      TableIdentifier::vector set_of_identifiers;
248
      session.doGetTableIdentifiers(identifier, set_of_identifiers);
249
250
      for (TableIdentifier::vector::iterator iter= set_of_identifiers.begin(); iter != set_of_identifiers.end(); iter++)
251
      {
252
        if (session.drop_temporary_table(*iter))
253
        {
2041.3.15 by Brian Aker
Cleanup error usage around identifier usage.
254
          my_error(ER_TABLE_DROP, *iter);
2041.3.10 by Brian Aker
Put more of drop schema behind interface.
255
          error= true;
256
          break;
257
        }
258
      }
259
    }
260
261
    /* After deleting database, remove all cache entries related to schema */
262
    table::Cache::singleton().removeSchema(identifier);
263
264
    if (not drop_all_tables_in_schema(session, identifier, dropped_tables, deleted))
265
    {
266
      error= true;
2041.3.15 by Brian Aker
Cleanup error usage around identifier usage.
267
      my_error(ER_DROP_SCHEMA, identifier);
2041.3.10 by Brian Aker
Put more of drop schema behind interface.
268
      break;
269
    }
270
2041.3.12 by Brian Aker
Move to private.
271
    uint64_t counter= 0;
272
    // Add hook here for engines to register schema.
273
    std::for_each(StorageEngine::getSchemaEngines().begin(), StorageEngine::getSchemaEngines().end(),
274
                  DropSchema(identifier, counter));
275
276
    if (not counter)
2041.3.10 by Brian Aker
Put more of drop schema behind interface.
277
    {
2041.3.15 by Brian Aker
Cleanup error usage around identifier usage.
278
      my_error(ER_DROP_SCHEMA, identifier);
2041.3.10 by Brian Aker
Put more of drop schema behind interface.
279
      error= true;
280
281
      break;
282
    }
283
    else
284
    {
285
      /* We've already verified that the schema does exist, so safe to log it */
286
      TransactionServices &transaction_services= TransactionServices::singleton();
2052 by Brian Aker
Fix additional member of trans services to handle identifier.
287
      transaction_services.dropSchema(&session, identifier);
2041.3.10 by Brian Aker
Put more of drop schema behind interface.
288
    }
289
  } while (0);
290
291
  if (deleted > 0)
292
  {
293
    session.clear_error();
294
    session.server_status|= SERVER_STATUS_DB_DROPPED;
295
    session.my_ok((uint32_t) deleted);
296
    session.server_status&= ~SERVER_STATUS_DB_DROPPED;
297
  }
298
299
300
  return error;
301
}
302
1415 by Brian Aker
Mass overhaul to use schema_identifier.
303
class AlterSchema : 
1966.2.9 by Brian Aker
Remove the use of "using std" from the plugin interface .cc files.
304
  public std::unary_function<StorageEngine *, void>
1415 by Brian Aker
Mass overhaul to use schema_identifier.
305
{
306
  uint64_t &success_count;
307
  const drizzled::message::Schema &schema_message;
308
309
public:
310
311
  AlterSchema(const drizzled::message::Schema &arg, uint64_t &count_arg) :
312
    success_count(count_arg),
313
    schema_message(arg)
314
  {
315
  }
316
317
  result_type operator() (argument_type engine)
318
  {
319
    // @todo eomeday check that at least one engine said "true"
320
    bool success= engine->doAlterSchema(schema_message);
321
1802.12.1 by Brian Aker
This solves bug lp:654905
322
1415 by Brian Aker
Mass overhaul to use schema_identifier.
323
    if (success)
1979.1.1 by Joseph Daly
move trx id assignment outside of schema engine and in plugin/schema_engine/schema.cc
324
    {
1415 by Brian Aker
Mass overhaul to use schema_identifier.
325
      success_count++;
1979.1.1 by Joseph Daly
move trx id assignment outside of schema engine and in plugin/schema_engine/schema.cc
326
      TransactionServices &transaction_services= TransactionServices::singleton();
327
      transaction_services.allocateNewTransactionId();
328
    }
1415 by Brian Aker
Mass overhaul to use schema_identifier.
329
  }
330
};
331
332
bool StorageEngine::alterSchema(const drizzled::message::Schema &schema_message)
333
{
334
  uint64_t success_count= 0;
335
1966.2.14 by Brian Aker
Merge in some additional std namespace finds.
336
  std::for_each(StorageEngine::getSchemaEngines().begin(), StorageEngine::getSchemaEngines().end(),
337
                AlterSchema(schema_message, success_count));
1415 by Brian Aker
Mass overhaul to use schema_identifier.
338
339
  return success_count ? true : false;
340
}
341
342
} /* namespace plugin */
343
} /* namespace drizzled */