1
/* Copyright (C) 2000-2004 DRIZZLE AB
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.
7
There are special exceptions to the terms and conditions of the GPL as it
8
is applied to this software. View the full text of the exception in file
9
EXCEPTIONS-CLIENT in the directory of this software distribution.
11
This program is distributed in the hope that it will be useful,
12
but WITHOUT ANY WARRANTY; without even the implied warranty of
13
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14
GNU General Public License for more details.
16
You should have received a copy of the GNU General Public License
17
along with this program; if not, write to the Free Software
18
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
20
#include <drizzled/global.h>
21
#include <mysys/my_sys.h>
25
#include <vio/violite.h>
33
#include <sys/socket.h>
34
#include <netinet/in.h>
35
#include <arpa/inet.h>
40
#ifdef HAVE_SYS_SELECT_H
41
#include <sys/select.h>
51
#define INADDR_NONE -1
54
#include <sql_common.h>
55
#include "client_settings.h"
56
#include <drizzled/version.h>
58
#undef net_buffer_length
59
#undef max_allowed_packet
61
uint32_t net_buffer_length= 8192;
62
uint32_t max_allowed_packet= 1024L*1024L*1024L;
65
#define SOCKET_ERROR -1
68
If allowed through some configuration, then this needs to
71
#define MAX_LONG_DATA_LENGTH 8192
72
#define unsigned_field(A) ((A)->flags & UNSIGNED_FLAG)
75
static DRIZZLE_PARAMETERS drizzle_internal_parameters=
76
{&max_allowed_packet, &net_buffer_length, 0};
78
const DRIZZLE_PARAMETERS * drizzle_get_parameters(void)
80
return &drizzle_internal_parameters;
83
bool drizzle_thread_init()
85
return my_thread_init();
88
void drizzle_thread_end()
95
Expand wildcard to a sql string
99
append_wild(char *to, char *end, const char *wild)
101
end-=5; /* Some extra */
104
to= strcpy(to," like '");
105
to+= 7; /* strlen(" like '"); */
107
while (*wild && to < end)
109
if (*wild == '\\' || *wild == '\'')
113
if (*wild) /* Too small buffer */
114
*to++='%'; /* Nicer this way */
120
/**************************************************************************
121
Ignore SIGPIPE handler
123
**************************************************************************/
126
my_pipe_sig_handler(int sig __attribute__((unused)))
128
#ifdef DONT_REMEMBER_SIGNAL
129
(void) signal(SIGPIPE, my_pipe_sig_handler);
134
/**************************************************************************
135
Change user and database
136
**************************************************************************/
138
int cli_read_change_user_result(DRIZZLE *drizzle)
142
pkt_length= cli_safe_read(drizzle);
144
if (pkt_length == packet_error)
150
bool drizzle_change_user(DRIZZLE *drizzle, const char *user,
151
const char *passwd, const char *db)
153
char buff[USERNAME_LENGTH+SCRAMBLED_PASSWORD_CHAR_LENGTH+NAME_LEN+2];
156
const CHARSET_INFO *saved_cs= drizzle->charset;
158
/* Get the connection-default character set. */
160
if (drizzle_init_character_set(drizzle))
162
drizzle->charset= saved_cs;
166
/* Use an empty string instead of NULL. */
173
/* Store user into the buffer */
174
end= strncpy(end, user, USERNAME_LENGTH) + USERNAME_LENGTH + 1;
176
/* write scrambled password according to server capabilities */
180
*end++= SCRAMBLE_LENGTH;
181
end+= SCRAMBLE_LENGTH;
185
*end++= '\0'; /* empty password */
186
/* Add database if needed */
187
end= strncpy(end, db ? db : "", NAME_LEN) + NAME_LEN + 1;
189
/* Add character set number. */
191
if (drizzle->server_capabilities & CLIENT_SECURE_CONNECTION)
193
int2store(end, (ushort) drizzle->charset->number);
197
/* Write authentication package */
198
(void)simple_command(drizzle,COM_CHANGE_USER, (uchar*) buff, (uint32_t) (end-buff), 1);
200
rc= (*drizzle->methods->read_change_user_result)(drizzle);
204
/* Free old connect information */
208
free(drizzle->passwd);
212
/* alloc new connect information */
213
drizzle->user= strdup(user);
214
drizzle->passwd= strdup(passwd);
215
drizzle->db= db ? strdup(db) : 0;
219
drizzle->charset= saved_cs;
225
#if defined(HAVE_GETPWUID) && defined(NO_GETPWUID_DECL)
226
struct passwd *getpwuid(uid_t);
227
char* getlogin(void);
230
/**************************************************************************
231
Do a query. If query returned rows, free old rows.
232
Read data by drizzle_store_result or by repeat call of drizzle_fetch_row
233
**************************************************************************/
236
drizzle_query(DRIZZLE *drizzle, const char *query)
238
return drizzle_real_query(drizzle,query, (uint) strlen(query));
242
/**************************************************************************
243
Return next field of the query results
244
**************************************************************************/
247
drizzle_fetch_field(DRIZZLE_RES *result)
249
if (result->current_field >= result->field_count)
251
return &result->fields[result->current_field++];
255
/**************************************************************************
256
Move to a specific row and column
257
**************************************************************************/
260
drizzle_data_seek(DRIZZLE_RES *result, uint64_t row)
264
for (tmp=result->data->data; row-- && tmp ; tmp = tmp->next) ;
265
result->current_row=0;
266
result->data_cursor = tmp;
270
/*************************************************************************
271
put the row or field cursor one a position one got from DRIZZLE_ROW_tell()
272
This doesn't restore any data. The next drizzle_fetch_row or
273
drizzle_fetch_field will return the next row or field after the last used
274
*************************************************************************/
277
drizzle_row_seek(DRIZZLE_RES *result, DRIZZLE_ROW_OFFSET row)
279
DRIZZLE_ROW_OFFSET return_value=result->data_cursor;
280
result->current_row= 0;
281
result->data_cursor= row;
287
drizzle_field_seek(DRIZZLE_RES *result, DRIZZLE_FIELD_OFFSET field_offset)
289
DRIZZLE_FIELD_OFFSET return_value=result->current_field;
290
result->current_field=field_offset;
295
/*****************************************************************************
296
List all tables in a database
297
If wild is given then only the tables matching wild is returned
298
*****************************************************************************/
301
drizzle_list_tables(DRIZZLE *drizzle, const char *wild)
304
char *ptr= strcpy(buff, "show tables");
305
ptr+= 11; /* strlen("show tables"); */
307
append_wild(ptr,buff+sizeof(buff),wild);
308
if (drizzle_query(drizzle,buff))
310
return (drizzle_store_result(drizzle));
314
DRIZZLE_FIELD *cli_list_fields(DRIZZLE *drizzle)
317
if (!(query= cli_read_rows(drizzle,(DRIZZLE_FIELD*) 0, 8)))
320
drizzle->field_count= (uint) query->rows;
321
return unpack_fields(query,&drizzle->field_alloc, drizzle->field_count, 1);
325
/**************************************************************************
326
List all fields in a table
327
If wild is given then only the fields matching wild is returned
328
Instead of this use query:
329
show fields in 'table' like "wild"
330
**************************************************************************/
333
drizzle_list_fields(DRIZZLE *drizzle, const char *table, const char *wild)
336
DRIZZLE_FIELD *fields;
337
char buff[257], *end;
339
end= strncpy(buff, table, 128) + 128;
340
end= strncpy(end+1, wild ? wild : "", 128) + 128;
342
free_old_query(drizzle);
343
if (simple_command(drizzle, COM_FIELD_LIST, (uchar*) buff,
344
(uint32_t) (end-buff), 1) ||
345
!(fields= (*drizzle->methods->list_fields)(drizzle)))
348
if (!(result = (DRIZZLE_RES *) malloc(sizeof(DRIZZLE_RES))))
351
memset(result, 0, sizeof(DRIZZLE_RES));
353
result->methods= drizzle->methods;
354
result->field_alloc=drizzle->field_alloc;
356
result->field_count = drizzle->field_count;
357
result->fields= fields;
362
/* List all running processes (threads) in server */
365
drizzle_list_processes(DRIZZLE *drizzle)
367
DRIZZLE_DATA *fields;
371
if (simple_command(drizzle,COM_PROCESS_INFO,0,0,0))
373
free_old_query(drizzle);
374
pos=(uchar*) drizzle->net.read_pos;
375
field_count=(uint) net_field_length(&pos);
376
if (!(fields = (*drizzle->methods->read_rows)(drizzle,(DRIZZLE_FIELD*) 0, 7)))
378
if (!(drizzle->fields=unpack_fields(fields,&drizzle->field_alloc,field_count,0)))
380
drizzle->status=DRIZZLE_STATUS_GET_RESULT;
381
drizzle->field_count=field_count;
382
return(drizzle_store_result(drizzle));
387
drizzle_shutdown(DRIZZLE *drizzle, enum drizzle_enum_shutdown_level shutdown_level)
390
level[0]= (uchar) shutdown_level;
391
return(simple_command(drizzle, COM_SHUTDOWN, level, 1, 0));
396
drizzle_refresh(DRIZZLE *drizzle,uint options)
399
bits[0]= (uchar) options;
400
return(simple_command(drizzle, COM_REFRESH, bits, 1, 0));
405
drizzle_kill(DRIZZLE *drizzle, uint32_t pid)
409
return(simple_command(drizzle,COM_PROCESS_KILL,buff,sizeof(buff),0));
414
drizzle_set_server_option(DRIZZLE *drizzle, enum enum_drizzle_set_option option)
417
int2store(buff, (uint) option);
418
return(simple_command(drizzle, COM_SET_OPTION, buff, sizeof(buff), 0));
422
const char *cli_read_statistics(DRIZZLE *drizzle)
424
drizzle->net.read_pos[drizzle->packet_length]=0; /* End of stat string */
425
if (!drizzle->net.read_pos[0])
427
set_drizzle_error(drizzle, CR_WRONG_HOST_INFO, unknown_sqlstate);
428
return drizzle->net.last_error;
430
return (char*) drizzle->net.read_pos;
435
drizzle_ping(DRIZZLE *drizzle)
438
res= simple_command(drizzle,COM_PING,0,0,0);
439
if (res == CR_SERVER_LOST && drizzle->reconnect)
440
res= simple_command(drizzle,COM_PING,0,0,0);
446
drizzle_get_server_info(const DRIZZLE *drizzle)
448
return((char*) drizzle->server_version);
453
drizzle_get_host_info(const DRIZZLE *drizzle)
455
return(drizzle->host_info);
460
drizzle_get_proto_info(const DRIZZLE *drizzle)
462
return (drizzle->protocol_version);
466
drizzle_get_client_info(void)
468
return (char*) DRIZZLE_SERVER_VERSION;
471
uint32_t drizzle_get_client_version(void)
473
return DRIZZLE_VERSION_ID;
476
bool drizzle_eof(const DRIZZLE_RES *res)
481
const DRIZZLE_FIELD * drizzle_fetch_field_direct(const DRIZZLE_RES *res, unsigned int fieldnr)
483
return &(res)->fields[fieldnr];
486
const DRIZZLE_FIELD * drizzle_fetch_fields(const DRIZZLE_RES *res)
491
DRIZZLE_ROW_OFFSET drizzle_row_tell(const DRIZZLE_RES *res)
493
return res->data_cursor;
496
DRIZZLE_FIELD_OFFSET drizzle_field_tell(const DRIZZLE_RES *res)
498
return res->current_field;
503
unsigned int drizzle_field_count(const DRIZZLE *drizzle)
505
return drizzle->field_count;
508
uint64_t drizzle_affected_rows(const DRIZZLE *drizzle)
510
return drizzle->affected_rows;
513
uint64_t drizzle_insert_id(const DRIZZLE *drizzle)
515
return drizzle->insert_id;
518
const char * drizzle_sqlstate(const DRIZZLE *drizzle)
520
return drizzle ? drizzle->net.sqlstate : cant_connect_sqlstate;
523
uint32_t drizzle_warning_count(const DRIZZLE *drizzle)
525
return drizzle->warning_count;
528
const char * drizzle_info(const DRIZZLE *drizzle)
530
return drizzle->info;
533
uint32_t drizzle_thread_id(const DRIZZLE *drizzle)
535
return drizzle->thread_id;
538
const char * drizzle_character_set_name(const DRIZZLE *drizzle)
540
return drizzle->charset->csname;
543
void drizzle_get_character_set_info(const DRIZZLE *drizzle, MY_CHARSET_INFO *csinfo)
545
csinfo->number = drizzle->charset->number;
546
csinfo->state = drizzle->charset->state;
547
csinfo->csname = drizzle->charset->csname;
548
csinfo->name = drizzle->charset->name;
549
csinfo->comment = drizzle->charset->comment;
550
csinfo->mbminlen = drizzle->charset->mbminlen;
551
csinfo->mbmaxlen = drizzle->charset->mbmaxlen;
553
if (drizzle->options.charset_dir)
554
csinfo->dir = drizzle->options.charset_dir;
556
csinfo->dir = charsets_dir;
559
uint drizzle_thread_safe(void)
565
bool drizzle_embedded(void)
567
#ifdef EMBEDDED_LIBRARY
574
/****************************************************************************
575
Some support functions
576
****************************************************************************/
579
Functions called my my_net_init() to set some application specific variables
582
void my_net_local_init(NET *net)
584
net->max_packet= (uint) net_buffer_length;
585
my_net_set_read_timeout(net, CLIENT_NET_READ_TIMEOUT);
586
my_net_set_write_timeout(net, CLIENT_NET_WRITE_TIMEOUT);
588
net->max_packet_size= max(net_buffer_length, max_allowed_packet);
592
This function is used to create HEX string that you
593
can use in a SQL statement in of the either ways:
594
INSERT INTO blob_column VALUES (0xAABBCC); (any DRIZZLE version)
595
INSERT INTO blob_column VALUES (X'AABBCC');
597
The string in "from" is encoded to a HEX string.
598
The result is placed in "to" and a terminating null byte is appended.
600
The string pointed to by "from" must be "length" bytes long.
601
You must allocate the "to" buffer to be at least length*2+1 bytes long.
602
Each character needs two bytes, and you need room for the terminating
603
null byte. When drizzle_hex_string() returns, the contents of "to" will
604
be a null-terminated string. The return value is the length of the
605
encoded string, not including the terminating null character.
607
The return value does not contain any leading 0x or a leading X' and
608
trailing '. The caller must supply whichever of those is desired.
612
drizzle_hex_string(char *to, const char *from, uint32_t length)
617
for (end= from + length; from < end; from++)
619
*to++= _dig_vec_upper[((unsigned char) *from) >> 4];
620
*to++= _dig_vec_upper[((unsigned char) *from) & 0x0F];
623
return (uint32_t) (to-to0);
627
Add escape characters to a string (blob?) to make it suitable for a insert
628
to should at least have place for length*2+1 chars
629
Returns the length of the to string
633
drizzle_escape_string(char *to,const char *from, uint32_t length)
635
return escape_string_for_drizzle(default_charset_info, to, 0, from, length);
639
drizzle_real_escape_string(DRIZZLE *drizzle, char *to,const char *from,
642
if (drizzle->server_status & SERVER_STATUS_NO_BACKSLASH_ESCAPES)
643
return escape_quotes_for_drizzle(drizzle->charset, to, 0, from, length);
644
return escape_string_for_drizzle(drizzle->charset, to, 0, from, length);
648
myodbc_remove_escape(const DRIZZLE *drizzle, char *name)
652
bool use_mb_flag= use_mb(drizzle->charset);
655
for (end=name; *end ; end++) ;
658
for (to=name ; *name ; name++)
662
if (use_mb_flag && (l = my_ismbchar( drizzle->charset, name , end ) ) )
670
if (*name == '\\' && name[1])
677
int cli_unbuffered_fetch(DRIZZLE *drizzle, char **row)
679
if (packet_error == cli_safe_read(drizzle))
682
*row= ((drizzle->net.read_pos[0] == DRIZZLE_PROTOCOL_NO_MORE_DATA) ? NULL :
683
(char*) (drizzle->net.read_pos+1));
687
/********************************************************************
689
*********************************************************************/
692
Commit the current transaction
695
bool drizzle_commit(DRIZZLE *drizzle)
697
return((bool) drizzle_real_query(drizzle, "commit", 6));
701
Rollback the current transaction
704
bool drizzle_rollback(DRIZZLE *drizzle)
706
return((bool) drizzle_real_query(drizzle, "rollback", 8));
711
Set autocommit to either true or false
714
bool drizzle_autocommit(DRIZZLE *drizzle, bool auto_mode)
716
return((bool) drizzle_real_query(drizzle, auto_mode ?
717
"set autocommit=1":"set autocommit=0",
722
/********************************************************************
723
Multi query execution + SPs APIs
724
*********************************************************************/
727
Returns true/false to indicate whether any more query results exist
728
to be read using drizzle_next_result()
731
bool drizzle_more_results(const DRIZZLE *drizzle)
733
return (drizzle->server_status & SERVER_MORE_RESULTS_EXISTS) ? true:false;
738
Reads and returns the next query results
740
int drizzle_next_result(DRIZZLE *drizzle)
742
if (drizzle->status != DRIZZLE_STATUS_READY)
744
set_drizzle_error(drizzle, CR_COMMANDS_OUT_OF_SYNC, unknown_sqlstate);
748
net_clear_error(&drizzle->net);
749
drizzle->affected_rows= ~(uint64_t) 0;
751
if (drizzle->server_status & SERVER_MORE_RESULTS_EXISTS)
752
return((*drizzle->methods->next_result)(drizzle));
754
return(-1); /* No more results */
758
DRIZZLE_RES * drizzle_use_result(DRIZZLE *drizzle)
760
return (*drizzle->methods->use_result)(drizzle);
763
bool drizzle_read_query_result(DRIZZLE *drizzle)
765
return (*drizzle->methods->read_query_result)(drizzle);