18
18
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
23
23
#include <assert.h>
24
#include <boost/lexical_cast.hpp>
25
#include <drizzled/identifier.h>
26
#include <drizzled/internal/my_sys.h>
28
#include <drizzled/error.h>
29
#include <drizzled/errmsg_print.h>
30
#include <drizzled/gettext.h>
32
#include <drizzled/table.h>
34
#include <drizzled/util/string.h>
35
#include <drizzled/util/tablename_to_filename.h>
25
#include "drizzled/identifier.h"
26
#include "drizzled/session.h"
27
#include "drizzled/internal/my_sys.h"
28
#include "drizzled/data_home.h"
30
#include "drizzled/table.h"
32
#include "drizzled/util/string.h"
37
34
#include <algorithm>
210
204
path length on success, 0 on failure
213
size_t Table::build_table_filename(std::string &in_path, const std::string &in_db, const std::string &in_table_name, bool is_tmp)
207
size_t TableIdentifier::build_table_filename(std::string &path, const char *db, const char *table_name, bool is_tmp)
209
char dbbuff[FN_REFLEN];
210
char tbbuff[FN_REFLEN];
215
211
bool conversion_error= false;
217
conversion_error= util::tablename_to_filename(in_db, in_path);
218
if (conversion_error)
220
errmsg_printf(error::ERROR,
221
_("Schema name cannot be encoded and fit within filesystem "
222
"name length restrictions."));
226
in_path.append(FN_ROOTDIR);
213
memset(tbbuff, 0, sizeof(tbbuff));
228
214
if (is_tmp) // It a conversion tmp
230
in_path.append(in_table_name);
216
strncpy(tbbuff, table_name, sizeof(tbbuff));
234
conversion_error= util::tablename_to_filename(in_table_name, in_path);
220
conversion_error= tablename_to_filename(table_name, tbbuff, sizeof(tbbuff));
235
221
if (conversion_error)
237
errmsg_printf(error::ERROR,
223
errmsg_printf(ERRMSG_LVL_ERROR,
238
224
_("Table name cannot be encoded and fit within filesystem "
239
225
"name length restrictions."));
244
return in_path.length();
247
Table::Table(const drizzled::Table &table) :
248
identifier::Schema(table.getShare()->getSchemaName()),
229
memset(dbbuff, 0, sizeof(dbbuff));
230
conversion_error= tablename_to_filename(db, dbbuff, sizeof(dbbuff));
231
if (conversion_error)
233
errmsg_printf(ERRMSG_LVL_ERROR,
234
_("Schema name cannot be encoded and fit within filesystem "
235
"name length restrictions."));
240
int rootdir_len= strlen(FN_ROOTDIR);
241
path.append(data_home);
242
ssize_t without_rootdir= path.length() - rootdir_len;
244
/* Don't add FN_ROOTDIR if dirzzle_data_home already includes it */
245
if (without_rootdir >= 0)
247
const char *tmp= path.c_str() + without_rootdir;
249
if (memcmp(tmp, FN_ROOTDIR, rootdir_len) != 0)
250
path.append(FN_ROOTDIR);
254
path.append(FN_ROOTDIR);
257
return path.length();
262
Translate a table name to a cursor name (WL #1324).
265
tablename_to_filename()
267
to OUT The cursor name
268
to_length The size of the cursor name buffer.
271
true if errors happen. false on success.
273
static bool tablename_to_filename(const char *from, char *to, size_t to_length)
277
for (; *from && length < to_length; length++, from++)
279
if ((*from >= '0' && *from <= '9') ||
280
(*from >= 'A' && *from <= 'Z') ||
281
(*from >= 'a' && *from <= 'z') ||
282
/* OSX defines an extra set of high-bit and multi-byte characters
283
that cannot be used on the filesystem. Instead of trying to sort
284
those out, we'll just escape encode all high-bit-set chars on OSX.
285
It won't really hurt anything - it'll just make some filenames ugly. */
286
#if !defined(TARGET_OS_OSX)
287
((unsigned char)*from >= 128) ||
293
to[length]= tolower(*from);
297
if (length + 3 >= to_length)
300
/* We need to escape this char in a way that can be reversed */
302
to[length++]= hexchars[(*from >> 4) & 15];
303
to[length]= hexchars[(*from) & 15];
306
if (internal::check_if_legal_tablename(to) &&
307
length + 4 < to_length)
309
memcpy(to + length, "@@@", 4);
315
TableIdentifier::TableIdentifier(const drizzled::Table &table) :
316
SchemaIdentifier(table.getShare()->getSchemaName()),
249
317
type(table.getShare()->getTableType()),
250
318
table_name(table.getShare()->getTableName())
326
void TableIdentifier::init()
328
std::string lower_table_name(table_name);
329
std::transform(lower_table_name.begin(), lower_table_name.end(),
330
lower_table_name.begin(), ::tolower);
261
333
case message::Table::FUNCTION:
262
334
case message::Table::STANDARD:
263
335
assert(path.size() == 0);
264
build_table_filename(path, getSchemaName(), table_name, false);
336
build_table_filename(path, getLower().c_str(), lower_table_name.c_str(), false);
266
338
case message::Table::INTERNAL:
267
339
assert(path.size() == 0);
268
build_table_filename(path, getSchemaName(), table_name, true);
340
build_table_filename(path, getLower().c_str(), lower_table_name.c_str(), true);
270
342
case message::Table::TEMPORARY:
271
343
if (path.empty())
279
case message::Table::FUNCTION:
280
case message::Table::STANDARD:
281
case message::Table::INTERNAL:
283
case message::Table::TEMPORARY:
287
pos= path.find("tmp/#sql");
288
if (pos != std::string::npos)
290
key_path= path.substr(pos);
296
350
util::insensitive_hash hasher;
297
351
hash_value= hasher(path);
299
std::string tb_name(getTableName());
300
std::transform(tb_name.begin(), tb_name.end(), tb_name.begin(), ::tolower);
353
key.resize(getKeySize());
354
size_t key_length= TableIdentifier::createKey(&key[0], *this);
302
key.set(getKeySize(), getSchemaName(), tb_name);
356
assert(key_length == getKeySize()); // If this is off, then we have a memory issue.
306
const std::string &Table::getPath() const
360
const std::string &TableIdentifier::getPath() const
311
const std::string &Table::getKeyPath() const
313
if (key_path.empty())
319
void Table::getSQLPath(std::string &sql_path) const // @todo this is just used for errors, we should find a way to optimize it
322
case message::Table::FUNCTION:
323
case message::Table::STANDARD:
324
sql_path.append(getSchemaName());
325
sql_path.append(".");
326
sql_path.append(table_name);
328
case message::Table::INTERNAL:
329
sql_path.append("temporary.");
330
sql_path.append(table_name);
332
case message::Table::TEMPORARY:
333
sql_path.append(getSchemaName());
334
sql_path.append(".#");
335
sql_path.append(table_name);
340
bool Table::isValid() const
342
if (not identifier::Schema::isValid())
348
if (table_name.empty())
354
if (table_name.size() > NAME_LEN)
360
if (table_name.at(table_name.length() -1) == ' ')
366
if (table_name.at(0) == '.')
373
const CHARSET_INFO * const cs= &my_charset_utf8mb4_general_ci;
375
int well_formed_error;
376
uint32_t res= cs->cset->well_formed_len(cs, table_name.c_str(), table_name.c_str() + table_name.length(),
377
NAME_CHAR_LEN, &well_formed_error);
378
if (well_formed_error or table_name.length() != res)
391
my_error(ER_WRONG_TABLE_NAME, MYF(0), name.c_str());
400
void Table::copyToTableMessage(message::Table &message) const
365
const std::string &TableIdentifier::getSQLPath()
367
if (sql_path.empty())
370
case message::Table::FUNCTION:
371
case message::Table::STANDARD:
372
sql_path.append(getLower());
373
sql_path.append(".");
374
sql_path.append(table_name);
376
case message::Table::INTERNAL:
377
sql_path.append("temporary.");
378
sql_path.append(table_name);
380
case message::Table::TEMPORARY:
381
sql_path.append(getLower());
382
sql_path.append(".#");
383
sql_path.append(table_name);
392
void TableIdentifier::copyToTableMessage(message::Table &message) const
402
394
message.set_name(table_name);
403
395
message.set_schema(getSchemaName());
406
void Table::Key::set(size_t resize_arg, const std::string &a, const std::string &b)
408
key_buffer.resize(resize_arg);
410
std::copy(a.begin(), a.end(), key_buffer.begin());
411
std::copy(b.begin(), b.end(), key_buffer.begin() + a.length() + 1);
413
util::sensitive_hash hasher;
414
hash_value= hasher(key_buffer);
417
std::size_t hash_value(Table const& b)
419
return b.getHashValue();
422
std::size_t hash_value(Table::Key const& b)
424
return b.getHashValue();
428
std::ostream& operator<<(std::ostream& output, Table::const_reference identifier)
431
output << identifier.getSchemaName();
433
output << identifier.getTableName();
435
output << message::type(identifier.getType());
437
output << identifier.getPath();
439
output << identifier.getHashValue();
442
return output; // for multiple << operators.
445
} /* namespace identifier */
398
std::size_t hash_value(TableIdentifier const& b)
400
return b.getHashValue();
446
403
} /* namespace drizzled */