17
17
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
20
21
#ifndef DRIZZLED_SESSION_H
21
22
#define DRIZZLED_SESSION_H
23
#include "drizzled/cursor.h"
24
#include "drizzled/diagnostics_area.h"
25
#include "drizzled/file_exchange.h"
26
#include "drizzled/identifier.h"
27
#include "drizzled/internal_error_handler.h"
28
#include "drizzled/my_hash.h"
29
#include "drizzled/named_savepoint.h"
30
#include "drizzled/open_tables_state.h"
24
/* Classes in mysql */
31
26
#include "drizzled/plugin.h"
32
#include "drizzled/plugin/authorization.h"
33
#include "drizzled/pthread_globals.h"
27
#include <drizzled/sql_locale.h>
28
#include "drizzled/resource_context.h"
29
#include <drizzled/cursor.h>
30
#include <drizzled/current_session.h>
31
#include <drizzled/sql_error.h>
32
#include <drizzled/file_exchange.h>
33
#include <drizzled/select_result_interceptor.h>
34
#include <drizzled/statistics_variables.h>
35
#include <drizzled/xid.h>
34
36
#include "drizzled/query_id.h"
35
#include "drizzled/resource_context.h"
36
#include "drizzled/select_result_interceptor.h"
37
#include "drizzled/sql_error.h"
38
#include "drizzled/sql_locale.h"
39
#include "drizzled/statistics_variables.h"
37
#include "drizzled/named_savepoint.h"
40
38
#include "drizzled/transaction_context.h"
41
39
#include "drizzled/util/storable.h"
42
#include "drizzled/xid.h"
41
#include "drizzled/my_hash.h"
47
#include <sys/resource.h>
54
#include "drizzled/catalog/instance.h"
55
#include "drizzled/catalog/local.h"
57
#include <drizzled/session/property_map.h>
58
#include <drizzled/session/state.h>
59
#include <drizzled/session/table_messages.h>
60
#include <drizzled/session/transactions.h>
61
#include <drizzled/system_variables.h>
62
#include <drizzled/copy_info.h>
63
#include <drizzled/system_variables.h>
64
#include <drizzled/ha_data.h>
66
#include <boost/thread/thread.hpp>
49
#include "drizzled/internal/getrusage.h"
51
#include <drizzled/security_context.h>
52
#include <drizzled/open_tables_state.h>
54
#include <drizzled/internal_error_handler.h>
55
#include <drizzled/diagnostics_area.h>
57
#include <drizzled/plugin/authorization.h>
59
#include <boost/unordered_map.hpp>
67
60
#include <boost/thread/mutex.hpp>
68
#include <boost/thread/shared_mutex.hpp>
69
61
#include <boost/thread/condition_variable.hpp>
70
#include <boost/make_shared.hpp>
72
#include <drizzled/lex_column.h>
73
#include "drizzled/sql_lex.h"
75
#include "drizzled/visibility.h"
77
63
#define MIN_HANDSHAKE_SIZE 6
112
98
#define TC_HEURISTIC_RECOVER_ROLLBACK 2
113
99
extern uint32_t tc_heuristic_recover;
103
Local storage for proto that are tmp table. This should be enlarged
104
to hande the entire table-share for a local table. Once Hash is done,
105
we should consider exchanging the map for it.
107
typedef std::map <std::string, message::Table> ProtoCache;
110
The COPY_INFO structure is used by INSERT/REPLACE code.
111
The schema of the row counting by the INSERT/INSERT ... ON DUPLICATE KEY
113
If a row is inserted then the copied variable is incremented.
114
If a row is updated by the INSERT ... ON DUPLICATE KEY UPDATE and the
115
new data differs from the old one then the copied and the updated
116
variables are incremented.
117
The touched variable is incremented if a row was touched by the update part
118
of the INSERT ... ON DUPLICATE KEY UPDATE no matter whether the row
119
was actually changed or not.
123
ha_rows records; /**< Number of processed records */
124
ha_rows deleted; /**< Number of deleted records */
125
ha_rows updated; /**< Number of updated records */
126
ha_rows copied; /**< Number of copied records */
128
ha_rows touched; /* Number of touched records */
129
enum enum_duplicates handle_duplicates;
130
int escape_char, last_errno;
132
/* for INSERT ... UPDATE */
133
List<Item> *update_fields;
134
List<Item> *update_values;
135
/* for VIEW ... WITH CHECK OPTION */
156
uint32_t table_count;
158
THR_LOCK_DATA **locks;
169
} /* namespace drizzled */
171
/** @TODO why is this in the middle of the file */
172
#include <drizzled/lex_column.h>
115
177
class select_result;
118
180
#define Session_SENTRY_MAGIC 0xfeedd1ff
119
181
#define Session_SENTRY_GONE 0xdeadbeef
121
extern DRIZZLED_API struct drizzle_system_variables global_system_variables;
183
struct system_variables
185
system_variables() {};
187
How dynamically allocated system variables are handled:
189
The global_system_variables and max_system_variables are "authoritative"
190
They both should have the same 'version' and 'size'.
191
When attempting to access a dynamic variable, if the session version
192
is out of date, then the session version is updated and realloced if
193
neccessary and bytes copied from global to make up for missing data.
195
ulong dynamic_variables_version;
196
char * dynamic_variables_ptr;
197
uint32_t dynamic_variables_head; /* largest valid variable offset */
198
uint32_t dynamic_variables_size; /* how many bytes are in use */
200
uint64_t myisam_max_extra_sort_file_size;
201
uint64_t max_heap_table_size;
202
uint64_t tmp_table_size;
203
ha_rows select_limit;
204
ha_rows max_join_size;
205
uint64_t auto_increment_increment;
206
uint64_t auto_increment_offset;
207
uint64_t bulk_insert_buff_size;
208
uint64_t join_buff_size;
209
uint32_t max_allowed_packet;
210
uint64_t max_error_count;
211
uint64_t max_length_for_sort_data;
212
size_t max_sort_length;
213
uint64_t min_examined_row_limit;
214
bool optimizer_prune_level;
217
uint32_t optimizer_search_depth;
218
uint32_t div_precincrement;
219
uint64_t preload_buff_size;
220
uint32_t read_buff_size;
221
uint32_t read_rnd_buff_size;
222
size_t sortbuff_size;
223
uint32_t thread_handling;
224
uint32_t tx_isolation;
225
uint32_t completion_type;
226
/* Determines which non-standard SQL behaviour should be enabled */
228
uint64_t max_seeks_for_key;
229
size_t range_alloc_block_size;
230
uint32_t query_alloc_block_size;
231
uint32_t query_prealloc_size;
232
uint64_t group_concat_max_len;
233
uint64_t pseudo_thread_id;
235
plugin::StorageEngine *storage_engine;
237
/* Only charset part of these variables is sensible */
238
const CHARSET_INFO *character_set_filesystem;
240
/* Both charset and collation parts of these variables are important */
241
const CHARSET_INFO *collation_server;
243
inline const CHARSET_INFO *getCollation(void)
245
return collation_server;
249
MY_LOCALE *lc_time_names;
251
Time_zone *time_zone;
254
extern struct system_variables global_system_variables;
256
} /* namespace drizzled */
258
#include "drizzled/sql_lex.h"
263
void mark_transaction_to_rollback(Session *session, bool all);
266
Storage engine specific thread local data.
271
Storage engine specific thread local data.
272
Lifetime: one user connection.
276
* Resource contexts for both the "statement" and "normal"
279
* Resource context at index 0:
281
* Life time: one statement within a transaction. If @@autocommit is
282
* on, also represents the entire transaction.
284
* Resource context at index 1:
286
* Life time: one transaction within a connection.
290
* If the storage engine does not participate in a transaction,
291
* there will not be a resource context.
293
drizzled::ResourceContext resource_context[2];
295
Ha_data() :ha_ptr(NULL) {}
124
299
* Represents a client connection to the database server.
139
314
* all member variables that are not critical to non-internal operations of the
140
315
* session object.
143
class DRIZZLED_API Session : public Open_tables_state
317
class Session : public Open_tables_state
146
320
// Plugin storage in Session.
147
typedef boost::shared_ptr<Session> shared_ptr;
148
typedef Session& reference;
149
typedef const Session& const_reference;
150
typedef const Session* const_pointer;
151
typedef Session* pointer;
153
static shared_ptr make_shared(plugin::Client *client, catalog::Instance::shared_ptr instance_arg)
155
assert(instance_arg);
156
return boost::make_shared<Session>(client, instance_arg);
321
typedef boost::unordered_map<std::string, util::Storable *, util::insensitive_hash, util::insensitive_equal_to> PropertyMap;
160
324
MARK_COLUMNS_NONE: Means mark_used_colums is not set and no indicator to
232
enum_sql_command getSqlCommand() const
234
return lex->sql_command;
237
392
/** query associated with this statement */
238
typedef boost::shared_ptr<const std::string> QueryString;
241
boost::shared_ptr<std::string> query;
243
// Never allow for a modification of this outside of the class. c_str()
244
// requires under some setup non const, you must copy the QueryString in
247
QueryString getQueryString() const
252
void resetQueryString()
259
We need to copy the lock on the string in order to make sure we have a stable string.
260
Once this is done we can use it to build a const char* which can be handed off for
261
a method to use (Innodb is currently the only engine using this).
263
const char *getQueryStringCopy(size_t &length)
265
QueryString tmp_string(getQueryString());
273
length= tmp_string->length();
274
char *to_return= strmake(tmp_string->c_str(), tmp_string->length());
279
session::State::shared_ptr _state;
283
session::State::const_shared_ptr state()
289
396
Name of the current (default) database.
321
416
static const char * const DEFAULT_WHERE;
323
418
memory::Root warn_root; /**< Allocation area for warnings and errors */
325
419
plugin::Client *client; /**< Pointer to client object */
329
void setClient(plugin::Client *client_arg);
331
plugin::Client *getClient()
336
plugin::Client *getClient() const
341
420
plugin::Scheduler *scheduler; /**< Pointer to scheduler object */
342
421
void *scheduler_arg; /**< Pointer to the optional scheduler argument */
344
423
typedef boost::unordered_map< std::string, user_var_entry *, util::insensitive_hash, util::insensitive_equal_to> UserVars;
347
424
typedef std::pair< UserVars::iterator, UserVars::iterator > UserVarsRange;
348
425
UserVars user_vars; /**< Hash of user variables defined during the session's lifetime */
351
const UserVars &getUserVariables() const
356
drizzle_system_variables variables; /**< Mutable local variables local to the session */
358
enum_tx_isolation getTxIsolation()
360
return (enum_tx_isolation)variables.tx_isolation;
428
struct system_variables variables; /**< Mutable local variables local to the session */
363
429
struct system_status_var status_var; /**< Session-local status counters */
364
430
THR_LOCK_INFO lock_info; /**< Locking information for this session */
365
431
THR_LOCK_OWNER main_lock_id; /**< To use for conventional queries */
366
432
THR_LOCK_OWNER *lock_id; /**< If not main_lock_id, points to the lock_id of a cursor. */
434
boost::mutex LOCK_delete; /**< Locked before session is deleted */
442
void unlockForDelete()
444
LOCK_delete.unlock();
448
* A peek into the query string for the session. This is a best effort
449
* delivery, there is no guarantee whether the content is meaningful.
451
char process_list_info[PROCESS_LIST_WIDTH+1];
369
454
* A pointer to the stack frame of the scheduler thread
473
530
uint32_t file_id; /**< File ID for LOAD DATA INFILE */
474
531
/* @note the following three members should likely move to Client */
475
532
uint32_t max_client_packet_length; /**< Maximum number of bytes a client can send in a single packet */
478
boost::posix_time::ptime _epoch;
479
boost::posix_time::ptime _connect_time;
480
boost::posix_time::ptime _start_timer;
481
boost::posix_time::ptime _end_timer;
483
boost::posix_time::ptime _user_time;
485
uint64_t utime_after_lock; // This used by Innodb.
489
_user_time= boost::posix_time::not_a_date_time;
492
const boost::posix_time::ptime &start_timer() const
497
void getTimeDifference(boost::posix_time::time_duration &result_arg, const boost::posix_time::ptime &arg) const
499
result_arg= arg - _start_timer;
535
uint64_t thr_create_utime; /**< track down slow pthread_create */
536
uint64_t start_utime;
537
uint64_t utime_after_lock;
502
539
thr_lock_type update_lock_default;
519
556
query_id_t query_id;
520
557
query_id_t warn_query_id;
523
559
void **getEngineData(const plugin::MonitoredInTransaction *monitored);
524
560
ResourceContext *getResourceContext(const plugin::MonitoredInTransaction *monitored,
525
561
size_t index= 0);
527
session::Transactions transaction;
563
struct st_transactions {
564
std::deque<NamedSavepoint> savepoints;
565
TransactionContext all; ///< Trans since BEGIN WORK
566
TransactionContext stmt; ///< Trans for current statement
529
581
Field *dup_field;
530
582
sigset_t signals;
532
// As of right now we do not allow a concurrent execute to launch itself
534
bool concurrent_execute_allowed;
538
void setConcurrentExecute(bool arg)
540
concurrent_execute_allowed= arg;
543
bool isConcurrentExecuteAllowed() const
545
return concurrent_execute_allowed;
548
584
/* Tells if LAST_INSERT_ID(#) was called for the current statement */
549
585
bool arg_of_last_insert_id_function;
552
587
ALL OVER THIS FILE, "insert_id" means "*automatically generated* value for
553
588
insertion into an auto_increment column".
655
684
create_sort_index(); may differ from examined_row_count.
657
686
uint32_t row_count;
659
uint32_t getRowCount() const
664
session_id_t thread_id;
665
688
uint32_t tmp_table;
666
enum global_read_lock_t
669
GOT_GLOBAL_READ_LOCK= 1,
670
MADE_GLOBAL_READ_LOCK_BLOCK_COMMIT= 2
673
global_read_lock_t _global_read_lock;
677
global_read_lock_t isGlobalReadLock() const
679
return _global_read_lock;
682
void setGlobalReadLock(global_read_lock_t arg)
684
_global_read_lock= arg;
687
DrizzleLock *lockTables(Table **tables, uint32_t count, uint32_t flags);
688
bool lockGlobalReadLock();
689
bool lock_table_names(TableList *table_list);
690
bool lock_table_names_exclusively(TableList *table_list);
691
bool makeGlobalReadLockBlockCommit();
692
bool abortLockForThread(Table *table);
693
bool wait_if_global_read_lock(bool abort_on_refresh, bool is_not_commit);
694
int lock_table_name(TableList *table_list);
695
void abortLock(Table *table);
696
void removeLock(Table *table);
697
void unlockReadTables(DrizzleLock *sql_lock);
698
void unlockSomeTables(Table **table, uint32_t count);
699
void unlockTables(DrizzleLock *sql_lock);
700
void startWaitingGlobalReadLock();
701
void unlockGlobalReadLock();
704
int unlock_external(Table **table, uint32_t count);
705
int lock_external(Table **tables, uint32_t count);
706
bool wait_for_locked_table_names(TableList *table_list);
707
DrizzleLock *get_lock_data(Table **table_ptr, uint32_t count,
708
bool should_lock, Table **write_lock_used);
689
uint32_t global_read_lock;
711
690
uint32_t server_status;
712
691
uint32_t open_options;
713
692
uint32_t select_number; /**< number of select (used for EXPLAIN) */
1041
1001
* Schedule a session to be run on the default scheduler.
1043
static bool schedule(Session::shared_ptr&);
1045
static void unlink(session_id_t &session_id);
1046
static void unlink(Session::shared_ptr&);
1049
1006
For enter_cond() / exit_cond() to work the mutex must be got before
1050
1007
enter_cond(); this mutex is then released by exit_cond().
1051
1008
Usage must be: lock mutex; enter_cond(); your code; exit_cond().
1053
const char* enter_cond(boost::condition_variable_any &cond, boost::mutex &mutex, const char* msg);
1010
const char* enter_cond(boost::condition_variable &cond, boost::mutex &mutex, const char* msg);
1054
1011
void exit_cond(const char* old_msg);
1056
type::Time::epoch_t query_start()
1058
return getCurrentTimestampEpoch();
1063
_end_timer= _start_timer= boost::posix_time::microsec_clock::universal_time();
1064
utime_after_lock= (_start_timer - _epoch).total_microseconds();
1067
void set_time(time_t t) // This is done by a sys_var, as long as user_time is set, we will use that for all references to time
1069
_user_time= boost::posix_time::from_time_t(t);
1072
void set_time_after_lock()
1074
boost::posix_time::ptime mytime(boost::posix_time::microsec_clock::universal_time());
1075
utime_after_lock= (mytime - _epoch).total_microseconds();
1078
void set_end_timer()
1080
_end_timer= boost::posix_time::microsec_clock::universal_time();
1081
status_var.execution_time_nsec+=(_end_timer - _start_timer).total_microseconds();
1084
uint64_t getElapsedTime() const
1086
return (_end_timer - _start_timer).total_microseconds();
1013
inline time_t query_start() { return start_time; }
1014
inline void set_time()
1018
start_time= user_time;
1019
connect_microseconds= start_utime= utime_after_lock= my_micro_time();
1022
start_utime= utime_after_lock= my_micro_time_and_time(&start_time);
1024
inline void set_current_time() { start_time= time(NULL); }
1025
inline void set_time(time_t t)
1027
start_time= user_time= t;
1028
start_utime= utime_after_lock= my_micro_time();
1030
void set_time_after_lock() { utime_after_lock= my_micro_time(); }
1090
1032
* Returns the current micro-timestamp
1092
type::Time::epoch_t getCurrentTimestamp(bool actual= true) const
1094
type::Time::epoch_t t_mark;
1098
boost::posix_time::ptime mytime(boost::posix_time::microsec_clock::universal_time());
1099
t_mark= (mytime - _epoch).total_microseconds();
1103
t_mark= (_end_timer - _epoch).total_microseconds();
1109
// We may need to set user on this
1110
type::Time::epoch_t getCurrentTimestampEpoch() const
1112
if (not _user_time.is_not_a_date_time())
1113
return (_user_time - _epoch).total_seconds();
1115
return (_start_timer - _epoch).total_seconds();
1118
type::Time::epoch_t getCurrentTimestampEpoch(type::Time::usec_t &fraction_arg) const
1120
if (not _user_time.is_not_a_date_time())
1123
return (_user_time - _epoch).total_seconds();
1126
fraction_arg= _start_timer.time_of_day().fractional_seconds() % 1000000;
1127
return (_start_timer - _epoch).total_seconds();
1130
uint64_t found_rows(void) const
1034
inline uint64_t getCurrentTimestamp()
1036
return my_micro_time();
1038
inline uint64_t found_rows(void)
1132
1040
return limit_found_rows;
1135
1042
/** Returns whether the session is currently inside a transaction */
1136
bool inTransaction() const
1043
inline bool inTransaction()
1138
1045
return server_status & SERVER_STATUS_IN_TRANS;
1047
inline bool fill_derived_tables()
1049
return !lex->only_view_structure();
1141
1052
LEX_STRING *make_lex_string(LEX_STRING *lex_str,
1142
1053
const char* str, uint32_t length,
1143
1054
bool allocate_lex_string);
1145
1055
LEX_STRING *make_lex_string(LEX_STRING *lex_str,
1146
1056
const std::string &str,
1147
1057
bool allocate_lex_string);
1149
1059
int send_explain_fields(select_result *result);
1152
1061
Clear the current error, if any.
1153
1062
We do not clear is_fatal_error or is_fatal_sub_stmt_error since we
1312
1212
* Current implementation does not depend on that, but future changes
1313
1213
* should be done with this in mind;
1315
* @param passwd Scrambled password received from client
1316
* @param db Database name to connect to, may be NULL
1215
* @param Scrambled password received from client
1216
* @param Length of scrambled password
1217
* @param Database name to connect to, may be NULL
1318
bool checkUser(const std::string &passwd, const std::string &db);
1219
bool checkUser(const char *passwd, uint32_t passwd_len, const char *db);
1321
1222
* Returns the timestamp (in microseconds) of when the Session
1322
1223
* connected to the server.
1324
uint64_t getConnectMicroseconds() const
1326
return (_connect_time - _epoch).total_microseconds();
1329
uint64_t getConnectSeconds() const
1331
return (_connect_time - _epoch).total_seconds();
1225
inline uint64_t getConnectMicroseconds() const
1227
return connect_microseconds;
1574
1479
bool openTablesLock(TableList *tables);
1482
* Open all tables in list and process derived tables
1484
* @param Pointer to a list of tables for open
1485
* @param Bitmap of flags to modify how the tables will be open:
1486
* DRIZZLE_LOCK_IGNORE_FLUSH - open table even if someone has
1487
* done a flush or namelock on it.
1496
* This is to be used on prepare stage when you don't read any
1497
* data from the tables.
1499
bool openTables(TableList *tables, uint32_t flags= 0);
1576
1501
int open_tables_from_list(TableList **start, uint32_t *counter, uint32_t flags= 0);
1578
1503
Table *openTableLock(TableList *table_list, thr_lock_type lock_type);
1579
1504
Table *openTable(TableList *table_list, bool *refresh, uint32_t flags= 0);
1581
1506
void unlink_open_table(Table *find);
1582
void drop_open_table(Table *table, const identifier::Table &identifier);
1507
void drop_open_table(Table *table, TableIdentifier &identifier);
1583
1508
void close_cached_table(Table *table);
1585
1510
/* Create a lock in the cache */
1586
table::Placeholder *table_cache_insert_placeholder(const identifier::Table &identifier);
1587
bool lock_table_name_if_not_cached(const identifier::Table &identifier, Table **table);
1511
Table *table_cache_insert_placeholder(const char *db_name, const char *table_name);
1512
bool lock_table_name_if_not_cached(TableIdentifier &identifier, Table **table);
1514
typedef boost::unordered_map<std::string, message::Table, util::insensitive_hash, util::insensitive_equal_to> TableMessageCache;
1515
TableMessageCache table_message_cache;
1517
bool storeTableMessage(const TableIdentifier &identifier, message::Table &table_message);
1518
bool removeTableMessage(const TableIdentifier &identifier);
1519
bool getTableMessage(const TableIdentifier &identifier, message::Table &table_message);
1520
bool doesTableMessageExist(const TableIdentifier &identifier);
1521
bool renameTableMessage(const TableIdentifier &from, const TableIdentifier &to);
1523
/* Work with temporary tables */
1524
Table *find_temporary_table(TableList *table_list);
1525
Table *find_temporary_table(const char *db, const char *table_name);
1526
Table *find_temporary_table(TableIdentifier &identifier);
1528
void doGetTableNames(CachedDirectory &directory,
1529
const SchemaIdentifier &schema_identifier,
1530
std::set<std::string>& set_of_names);
1531
void doGetTableNames(const SchemaIdentifier &schema_identifier,
1532
std::set<std::string>& set_of_names);
1534
void doGetTableIdentifiers(CachedDirectory &directory,
1535
const SchemaIdentifier &schema_identifier,
1536
TableIdentifiers &set_of_identifiers);
1537
void doGetTableIdentifiers(const SchemaIdentifier &schema_identifier,
1538
TableIdentifiers &set_of_identifiers);
1540
int doGetTableDefinition(const drizzled::TableIdentifier &identifier,
1541
message::Table &table_proto);
1542
bool doDoesTableExist(const drizzled::TableIdentifier &identifier);
1544
void close_temporary_tables();
1545
void close_temporary_table(Table *table);
1546
// The method below just handles the de-allocation of the table. In
1547
// a better memory type world, this would not be needed.
1590
session::TableMessages _table_message_cache;
1549
void nukeTable(Table *table);
1593
session::TableMessages &getMessageCache()
1595
return _table_message_cache;
1552
void dumpTemporaryTableNames(const char *id);
1553
int drop_temporary_table(TableList *table_list);
1554
bool rm_temporary_table(plugin::StorageEngine *base, TableIdentifier &identifier);
1555
bool rm_temporary_table(TableIdentifier &identifier, bool best_effort= false);
1556
Table *open_temporary_table(TableIdentifier &identifier,
1557
bool link_in_list= true);
1598
1559
/* Reopen operations */
1599
bool reopen_tables();
1560
bool reopen_tables(bool get_locks, bool mark_share_as_old);
1561
bool reopen_name_locked_table(TableList* table_list, bool link_in);
1600
1562
bool close_cached_tables(TableList *tables, bool wait_for_refresh, bool wait_for_placeholders);
1602
void wait_for_condition(boost::mutex &mutex, boost::condition_variable_any &cond);
1564
void wait_for_condition(boost::mutex &mutex, boost::condition_variable &cond);
1603
1565
int setup_conds(TableList *leaves, COND **conds);
1604
1566
int lock_tables(TableList *tables, uint32_t count, bool *need_reopen);
1568
Table *create_virtual_tmp_table(List<CreateField> &field_list);
1606
1570
drizzled::util::Storable *getProperty(const std::string &arg)
1608
return life_properties.getProperty(arg);
1572
return life_properties[arg];
1611
1575
template<class T>
1612
1576
bool setProperty(const std::string &arg, T *value)
1614
life_properties.setProperty(arg, value);
1578
life_properties[arg]= value;