1
/* - mode: c; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2008 MySQL
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; either version 2 of the License, or
9
* (at your option) any later version.
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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
23
This file is included by both libdrizzle.c (the DRIZZLE client C API)
24
and the drizzled server to connect to another DRIZZLE server.
26
The differences for the two cases are:
28
- Things that only works for the client:
29
- Trying to automaticly determinate user name if not supplied to
31
- Support for reading local file with LOAD DATA LOCAL
32
- SHARED memory handling
33
- Protection against sigpipe
36
- Things that only works for the server
37
- Alarm handling on connect
39
In all other cases, the code should be idential for the client and
43
#include <drizzled/global.h>
48
#include <sys/ioctl.h>
52
/* Remove client convenience wrappers */
53
#undef max_allowed_packet
54
#undef net_buffer_length
56
#define CLI_DRIZZLE_CONNECT drizzle_connect
58
#include <mysys/my_sys.h>
59
#include <mystrings/m_string.h>
60
#include <mystrings/m_ctype.h>
61
#include <drizzled/error.h>
63
#include <vio/violite.h>
72
#include <sys/socket.h>
73
#include <netinet/in.h>
74
#include <arpa/inet.h>
79
#ifdef HAVE_SYS_SELECT_H
80
#include <sys/select.h>
86
#define SOCKET_ERROR -1
88
#define CONNECT_TIMEOUT 0
90
#include "client_settings.h"
91
#include <drizzled/version.h>
92
#include <libdrizzle/sql_common.h>
93
#include <libdrizzle/gettext.h>
96
char *drizzle_unix_port= 0;
97
const char *unknown_sqlstate= "HY000";
98
const char *not_error_sqlstate= "00000";
99
const char *cant_connect_sqlstate= "08001";
101
static bool drizzle_client_init= false;
102
static bool org_my_init_done= false;
104
static void drizzle_close_free_options(DRIZZLE *drizzle);
105
static void drizzle_close_free(DRIZZLE *drizzle);
107
static int wait_for_data(int fd, int32_t timeout);
108
int connect_with_timeout(int fd, const struct sockaddr *name, uint namelen, int32_t timeout);
110
CHARSET_INFO *default_client_charset_info = &my_charset_utf8_general_ci;
112
/* Server error code and message */
113
unsigned int drizzle_server_last_errno;
114
char drizzle_server_last_error[DRIZZLE_ERRMSG_SIZE];
116
/****************************************************************************
117
A modified version of connect(). connect_with_timeout() allows you to specify
118
a timeout value, in seconds, that we should wait until we
119
derermine we can't connect to a particular host. If timeout is 0,
120
connect_with_timeout() will behave exactly like connect().
122
Base version coded by Steve Bernacki, Jr. <steve@navinet.net>
123
*****************************************************************************/
126
int connect_with_timeout(int fd, const struct sockaddr *name, uint namelen, int32_t timeout)
128
int flags, res, s_err;
131
If they passed us a timeout of zero, we should behave
132
exactly like the normal connect() call does.
136
return connect(fd, (struct sockaddr*) name, namelen);
138
flags = fcntl(fd, F_GETFL, 0); /* Set socket to not block */
140
fcntl(fd, F_SETFL, flags | O_NONBLOCK); /* and save the flags.. */
143
res= connect(fd, (struct sockaddr*) name, namelen);
144
s_err= errno; /* Save the error... */
145
fcntl(fd, F_SETFL, flags);
146
if ((res != 0) && (s_err != EINPROGRESS))
148
errno= s_err; /* Restore it */
151
if (res == 0) /* Connected quickly! */
154
return wait_for_data(fd, timeout);
159
Wait up to timeout seconds for a connection to be established.
161
We prefer to do this with poll() as there is no limitations with this.
162
If not, we will use select()
165
static int wait_for_data(int fd, int32_t timeout)
171
ufds.events= POLLIN | POLLPRI;
172
if (!(res= poll(&ufds, 1, (int) timeout*1000)))
177
if (res < 0 || !(ufds.revents & (POLLIN | POLLPRI)) || (ufds.revents & POLLHUP))
183
#if defined(HAVE_GETPWUID) && defined(NO_GETPWUID_DECL)
184
struct passwd *getpwuid(uid_t);
185
char* getlogin(void);
188
static void read_user_name(char *name)
191
strcpy(name,"root"); /* allow use of surun */
197
if ((str=getlogin()) == NULL)
199
if ((skr=getpwuid(geteuid())) != NULL)
201
else if (!(str=getenv("USER")) && !(str=getenv("LOGNAME")) &&
202
!(str=getenv("LOGIN")))
205
strncpy(name,str,USERNAME_LENGTH);
207
(void) cuserid(name);
209
strcpy(name,"UNKNOWN_USER");
217
Set the internal error message to DRIZZLE handler
219
@param drizzle connection handle (client side)
220
@param errcode CR_ error code, passed to ER macro to get
222
@parma sqlstate SQL standard sqlstate
225
void set_drizzle_error(DRIZZLE *drizzle, int errcode, const char *sqlstate)
228
assert(drizzle != 0);
233
net->last_errno= errcode;
234
strcpy(net->last_error, ER(errcode));
235
strcpy(net->sqlstate, sqlstate);
239
drizzle_server_last_errno= errcode;
240
strcpy(drizzle_server_last_error, ER(errcode));
246
Clear possible error state of struct NET
248
@param net clear the state of the argument
251
void net_clear_error(NET *net)
254
net->last_error[0]= '\0';
255
strcpy(net->sqlstate, not_error_sqlstate);
259
Set an error message on the client.
261
@param drizzle connection handle
262
@param errcode CR_* errcode, for client errors
263
@param sqlstate SQL standard sql state, unknown_sqlstate for the
264
majority of client errors.
265
@param format error message template, in sprintf format
266
@param ... variable number of arguments
269
static void set_drizzle_extended_error(DRIZZLE *drizzle, int errcode,
270
const char *sqlstate,
271
const char *format, ...)
275
assert(drizzle != 0);
278
net->last_errno= errcode;
279
va_start(args, format);
280
vsnprintf(net->last_error, sizeof(net->last_error)-1,
283
strcpy(net->sqlstate, sqlstate);
288
/*****************************************************************************
289
Read a packet from server. Give error message if socket was down
290
or packet is an error message
291
*****************************************************************************/
293
uint32_t cli_safe_read(DRIZZLE *drizzle)
295
NET *net= &drizzle->net;
297
init_sigpipe_variables
299
/* Don't give sigpipe errors if the client doesn't want them */
300
set_sigpipe(drizzle);
302
len=my_net_read(net);
303
reset_sigpipe(drizzle);
305
if (len == packet_error || len == 0)
307
#ifdef DRIZZLE_SERVER
308
if (net->vio && vio_was_interrupted(net->vio))
309
return (packet_error);
310
#endif /*DRIZZLE_SERVER*/
312
set_drizzle_error(drizzle, net->last_errno == ER_NET_PACKET_TOO_LARGE ?
313
CR_NET_PACKET_TOO_LARGE: CR_SERVER_LOST, unknown_sqlstate);
314
return (packet_error);
316
if (net->read_pos[0] == 255)
320
char *pos=(char*) net->read_pos+1;
321
net->last_errno=uint2korr(pos);
326
strncpy(net->sqlstate, pos+1, SQLSTATE_LENGTH);
327
pos+= SQLSTATE_LENGTH+1;
332
The SQL state hasn't been received -- it should be reset to HY000
333
(unknown error sql state).
336
strcpy(net->sqlstate, unknown_sqlstate);
339
strncpy(net->last_error,(char*) pos, min((uint) len,
340
(uint32_t) sizeof(net->last_error)-1));
343
set_drizzle_error(drizzle, CR_UNKNOWN_ERROR, unknown_sqlstate);
345
Cover a protocol design error: error packet does not
346
contain the server status. Therefore, the client has no way
347
to find out whether there are more result sets of
348
a multiple-result-set statement pending. Luckily, in 5.0 an
349
error always aborts execution of a statement, wherever it is
350
a multi-statement or a stored procedure, so it should be
351
safe to unconditionally turn off the flag here.
353
drizzle->server_status&= ~SERVER_MORE_RESULTS_EXISTS;
355
return(packet_error);
360
void free_rows(DRIZZLE_DATA *cur)
364
free_root(&cur->alloc,MYF(0));
365
my_free((uchar*) cur,MYF(0));
370
cli_advanced_command(DRIZZLE *drizzle, enum enum_server_command command,
371
const unsigned char *header, uint32_t header_length,
372
const unsigned char *arg, uint32_t arg_length, bool skip_check)
374
NET *net= &drizzle->net;
376
init_sigpipe_variables
377
bool stmt_skip= false;
379
/* Don't give sigpipe errors if the client doesn't want them */
380
set_sigpipe(drizzle);
382
if (drizzle->net.vio == 0)
383
{ /* Do reconnect if possible */
384
if (drizzle_reconnect(drizzle) || stmt_skip)
387
if (drizzle->status != DRIZZLE_STATUS_READY ||
388
drizzle->server_status & SERVER_MORE_RESULTS_EXISTS)
390
set_drizzle_error(drizzle, CR_COMMANDS_OUT_OF_SYNC, unknown_sqlstate);
394
net_clear_error(net);
396
drizzle->affected_rows= ~(uint64_t) 0;
398
We don't want to clear the protocol buffer on COM_QUIT, because if
399
the previous command was a shutdown command, we may have the
400
response for the COM_QUIT already in the communication buffer
402
net_clear(&drizzle->net, (command != COM_QUIT));
404
if (net_write_command(net,(uchar) command, header, header_length,
407
if (net->last_errno == ER_NET_PACKET_TOO_LARGE)
409
set_drizzle_error(drizzle, CR_NET_PACKET_TOO_LARGE, unknown_sqlstate);
413
if (drizzle_reconnect(drizzle) || stmt_skip)
415
if (net_write_command(net,(uchar) command, header, header_length,
418
set_drizzle_error(drizzle, CR_SERVER_GONE_ERROR, unknown_sqlstate);
424
result= ((drizzle->packet_length=cli_safe_read(drizzle)) == packet_error ?
427
reset_sigpipe(drizzle);
431
void free_old_query(DRIZZLE *drizzle)
434
free_root(&drizzle->field_alloc,MYF(0));
435
init_alloc_root(&drizzle->field_alloc,8192,0); /* Assume rowlength < 8192 */
437
drizzle->field_count= 0; /* For API */
438
drizzle->warning_count= 0;
444
Flush result set sent from server
447
static void cli_flush_use_result(DRIZZLE *drizzle)
449
/* Clear the current execution status */
453
if ((pkt_len=cli_safe_read(drizzle)) == packet_error)
455
if (pkt_len <= 8 && drizzle->net.read_pos[0] == DRIZZLE_PROTOCOL_NO_MORE_DATA)
457
char *pos= (char*) drizzle->net.read_pos + 1;
458
drizzle->warning_count=uint2korr(pos); pos+=2;
459
drizzle->server_status=uint2korr(pos); pos+=2;
461
break; /* End of data */
468
/**************************************************************************
470
**************************************************************************/
472
void end_server(DRIZZLE *drizzle)
474
int save_errno= errno;
475
if (drizzle->net.vio != 0)
477
init_sigpipe_variables
478
set_sigpipe(drizzle);
479
vio_delete(drizzle->net.vio);
480
reset_sigpipe(drizzle);
481
drizzle->net.vio= 0; /* Marker */
483
net_end(&drizzle->net);
484
free_old_query(drizzle);
490
drizzle_free_result(DRIZZLE_RES *result)
494
DRIZZLE *drizzle= result->handle;
497
if (drizzle->unbuffered_fetch_owner == &result->unbuffered_fetch_cancelled)
498
drizzle->unbuffered_fetch_owner= 0;
499
if (drizzle->status == DRIZZLE_STATUS_USE_RESULT)
501
(*drizzle->methods->flush_use_result)(drizzle);
502
drizzle->status=DRIZZLE_STATUS_READY;
503
if (drizzle->unbuffered_fetch_owner)
504
*drizzle->unbuffered_fetch_owner= true;
507
free_rows(result->data);
509
free_root(&result->field_alloc,MYF(0));
511
my_free((uchar*) result->row,MYF(0));
512
my_free((uchar*) result,MYF(0));
516
/****************************************************************************
517
Get options from my.cnf
518
****************************************************************************/
520
static const char *default_options[]=
522
"port","socket","compress","password","pipe", "timeout", "user",
523
"init-command", "host", "database", "return-found-rows",
524
"ssl-key" ,"ssl-cert" ,"ssl-ca" ,"ssl-capath",
525
"character-sets-dir", "default-character-set", "interactive-timeout",
526
"connect-timeout", "local-infile", "disable-local-infile",
527
"ssl-cipher", "max-allowed-packet", "protocol", "shared-memory-base-name",
528
"multi-results", "multi-statements", "multi-queries", "secure-auth",
529
"report-data-truncation",
533
static TYPELIB option_types={array_elements(default_options)-1,
534
"options",default_options, NULL};
536
const char *sql_protocol_names_lib[] =
537
{ "TCP", "SOCKET", "PIPE", "MEMORY", NullS };
538
TYPELIB sql_protocol_typelib = {array_elements(sql_protocol_names_lib)-1,"",
539
sql_protocol_names_lib, NULL};
541
static int add_init_command(struct st_drizzle_options *options, const char *cmd)
545
if (!options->init_commands)
547
options->init_commands= (DYNAMIC_ARRAY*)my_malloc(sizeof(DYNAMIC_ARRAY),
549
init_dynamic_array(options->init_commands,sizeof(char*),0,5 CALLER_INFO);
552
if (!(tmp= my_strdup(cmd,MYF(MY_WME))) ||
553
insert_dynamic(options->init_commands, (uchar*)&tmp))
555
my_free(tmp, MYF(MY_ALLOW_ZERO_PTR));
562
void drizzle_read_default_options(struct st_drizzle_options *options,
563
const char *filename,const char *group)
566
char *argv_buff[1],**argv;
567
const char *groups[3];
569
argc=1; argv=argv_buff; argv_buff[0]= (char*) "client";
570
groups[0]= (char*) "client"; groups[1]= (char*) group; groups[2]=0;
572
load_defaults(filename, groups, &argc, &argv);
573
if (argc != 1) /* If some default option */
578
if (option[0][0] == '-' && option[0][1] == '-')
580
char *end=strrchr(*option,'=');
585
*end=0; /* Remove '=' */
587
/* Change all '_' in variable name to '-' */
588
for (end= *option ; *(end= strrchr(end,'_')) ; )
590
switch (find_type(*option+2,&option_types,2)) {
593
options->port=atoi(opt_arg);
598
my_free(options->unix_socket,MYF(MY_ALLOW_ZERO_PTR));
599
options->unix_socket=my_strdup(opt_arg,MYF(MY_WME));
602
case 3: /* compress */
604
options->client_flag|= CLIENT_COMPRESS;
606
case 4: /* password */
609
my_free(options->password,MYF(MY_ALLOW_ZERO_PTR));
610
options->password=my_strdup(opt_arg,MYF(MY_WME));
613
case 20: /* connect_timeout */
614
case 6: /* timeout */
616
options->connect_timeout=atoi(opt_arg);
621
my_free(options->user,MYF(MY_ALLOW_ZERO_PTR));
622
options->user=my_strdup(opt_arg,MYF(MY_WME));
625
case 8: /* init-command */
626
add_init_command(options,opt_arg);
631
my_free(options->host,MYF(MY_ALLOW_ZERO_PTR));
632
options->host=my_strdup(opt_arg,MYF(MY_WME));
635
case 10: /* database */
638
my_free(options->db,MYF(MY_ALLOW_ZERO_PTR));
639
options->db=my_strdup(opt_arg,MYF(MY_WME));
642
case 12: /* return-found-rows */
643
options->client_flag|=CLIENT_FOUND_ROWS;
645
case 13: /* Ignore SSL options */
651
case 17: /* charset-lib */
652
my_free(options->charset_dir,MYF(MY_ALLOW_ZERO_PTR));
653
options->charset_dir = my_strdup(opt_arg, MYF(MY_WME));
656
my_free(options->charset_name,MYF(MY_ALLOW_ZERO_PTR));
657
options->charset_name = my_strdup(opt_arg, MYF(MY_WME));
659
case 19: /* Interactive-timeout */
660
options->client_flag|= CLIENT_INTERACTIVE;
663
if (!opt_arg || atoi(opt_arg) != 0)
664
options->client_flag|= CLIENT_LOCAL_FILES;
666
options->client_flag&= ~CLIENT_LOCAL_FILES;
669
options->client_flag&= ~CLIENT_LOCAL_FILES;
671
case 24: /* max-allowed-packet */
673
options->max_allowed_packet= atoi(opt_arg);
675
case 25: /* protocol */
676
if ((options->protocol= find_type(opt_arg,
677
&sql_protocol_typelib,0)) <= 0)
679
fprintf(stderr, _("Unknown option to protocol: %s\n"), opt_arg);
683
case 27: /* multi-results */
684
options->client_flag|= CLIENT_MULTI_RESULTS;
686
case 28: /* multi-statements */
687
case 29: /* multi-queries */
688
options->client_flag|= CLIENT_MULTI_STATEMENTS | CLIENT_MULTI_RESULTS;
690
case 30: /* secure-auth */
691
options->secure_auth= true;
693
case 31: /* report-data-truncation */
694
options->report_data_truncation= opt_arg ? test(atoi(opt_arg)) : 1;
707
/**************************************************************************
708
Get column lengths of the current row
709
If one uses drizzle_use_result, res->lengths contains the length information,
710
else the lengths are calculated from the offset between pointers.
711
**************************************************************************/
713
static void cli_fetch_lengths(uint32_t *to, DRIZZLE_ROW column, uint32_t field_count)
715
uint32_t *prev_length;
719
prev_length=0; /* Keep gcc happy */
720
for (end=column + field_count + 1 ; column != end ; column++, to++)
727
if (start) /* Found end of prev string */
728
*prev_length= (uint32_t) (*column-start-1);
734
/***************************************************************************
735
Change field rows to field structs
736
***************************************************************************/
739
unpack_fields(DRIZZLE_DATA *data, MEM_ROOT *alloc,uint fields,
743
DRIZZLE_FIELD *field,*result;
744
uint32_t lengths[9]; /* Max of fields */
746
field= result= (DRIZZLE_FIELD*) alloc_root(alloc,
747
(uint) sizeof(*field)*fields);
750
free_rows(data); /* Free old data */
753
memset((char*) field, 0, (uint) sizeof(DRIZZLE_FIELD)*fields);
755
for (row= data->data; row ; row = row->next,field++)
758
/* fields count may be wrong */
759
assert((uint) (field - result) < fields);
760
cli_fetch_lengths(&lengths[0], row->data, default_value ? 8 : 7);
761
field->catalog= strmake_root(alloc,(char*) row->data[0], lengths[0]);
762
field->db= strmake_root(alloc,(char*) row->data[1], lengths[1]);
763
field->table= strmake_root(alloc,(char*) row->data[2], lengths[2]);
764
field->org_table= strmake_root(alloc,(char*) row->data[3], lengths[3]);
765
field->name= strmake_root(alloc,(char*) row->data[4], lengths[4]);
766
field->org_name= strmake_root(alloc,(char*) row->data[5], lengths[5]);
768
field->catalog_length= lengths[0];
769
field->db_length= lengths[1];
770
field->table_length= lengths[2];
771
field->org_table_length= lengths[3];
772
field->name_length= lengths[4];
773
field->org_name_length= lengths[5];
775
/* Unpack fixed length parts */
776
pos= (uchar*) row->data[6];
777
field->charsetnr= uint2korr(pos);
778
field->length= (uint) uint4korr(pos+2);
779
field->type= (enum enum_field_types) pos[6];
780
field->flags= uint2korr(pos+7);
781
field->decimals= (uint) pos[9];
783
if (INTERNAL_NUM_FIELD(field))
784
field->flags|= NUM_FLAG;
785
if (default_value && row->data[7])
787
field->def=strmake_root(alloc,(char*) row->data[7], lengths[7]);
788
field->def_length= lengths[7];
792
field->max_length= 0;
795
free_rows(data); /* Free old data */
799
/* Read all rows (fields or data) from server */
801
DRIZZLE_DATA *cli_read_rows(DRIZZLE *drizzle, DRIZZLE_FIELD *DRIZZLE_FIELDs, uint32_t fields)
808
DRIZZLE_DATA *result;
809
DRIZZLE_ROWS **prev_ptr,*cur;
810
NET *net = &drizzle->net;
812
if ((pkt_len= cli_safe_read(drizzle)) == packet_error)
814
if (!(result=(DRIZZLE_DATA*) my_malloc(sizeof(DRIZZLE_DATA),
815
MYF(MY_WME | MY_ZEROFILL))))
817
set_drizzle_error(drizzle, CR_OUT_OF_MEMORY, unknown_sqlstate);
820
init_alloc_root(&result->alloc,8192,0); /* Assume rowlength < 8192 */
821
result->alloc.min_malloc=sizeof(DRIZZLE_ROWS);
822
prev_ptr= &result->data;
824
result->fields=fields;
827
The last EOF packet is either a 254 (0xFE) character followed by 1-7 status bytes.
829
This doesn't conflict with normal usage of 254 which stands for a
830
string where the length of the string is 8 bytes. (see net_field_length())
833
while (*(cp=net->read_pos) != DRIZZLE_PROTOCOL_NO_MORE_DATA || pkt_len >= 8)
836
if (!(cur= (DRIZZLE_ROWS*) alloc_root(&result->alloc,
837
sizeof(DRIZZLE_ROWS))) ||
838
!(cur->data= ((DRIZZLE_ROW)
839
alloc_root(&result->alloc,
840
(fields+1)*sizeof(char *)+pkt_len))))
843
set_drizzle_error(drizzle, CR_OUT_OF_MEMORY, unknown_sqlstate);
847
prev_ptr= &cur->next;
848
to= (char*) (cur->data+fields+1);
850
for (field=0 ; field < fields ; field++)
852
if ((len= net_field_length(&cp)) == NULL_LENGTH)
854
cur->data[field] = 0;
858
cur->data[field] = to;
859
if (len > (uint32_t) (end_to - to))
862
set_drizzle_error(drizzle, CR_MALFORMED_PACKET, unknown_sqlstate);
871
if (DRIZZLE_FIELDs[field].max_length < len)
872
DRIZZLE_FIELDs[field].max_length=len;
876
cur->data[field]=to; /* End of last field */
877
if ((pkt_len=cli_safe_read(drizzle)) == packet_error)
883
*prev_ptr=0; /* last pointer is null */
884
if (pkt_len > 1) /* DRIZZLE 4.1 protocol */
886
drizzle->warning_count= uint2korr(cp+1);
887
drizzle->server_status= uint2korr(cp+3);
893
Read one row. Uses packet buffer as storage for fields.
894
When next packet is read, the previous field values are destroyed
899
read_one_row(DRIZZLE *drizzle, uint32_t fields, DRIZZLE_ROW row, uint32_t *lengths)
902
uint32_t pkt_len,len;
903
uchar *pos, *prev_pos, *end_pos;
904
NET *net= &drizzle->net;
906
if ((pkt_len=cli_safe_read(drizzle)) == packet_error)
908
if (pkt_len <= 8 && net->read_pos[0] == DRIZZLE_PROTOCOL_NO_MORE_DATA)
910
if (pkt_len > 1) /* DRIZZLE 4.1 protocol */
912
drizzle->warning_count= uint2korr(net->read_pos+1);
913
drizzle->server_status= uint2korr(net->read_pos+3);
915
return 1; /* End of data */
917
prev_pos= 0; /* allowed to write at packet[-1] */
920
for (field=0 ; field < fields ; field++)
922
if ((len= net_field_length(&pos)) == NULL_LENGTH)
929
if (len > (uint32_t) (end_pos - pos))
931
set_drizzle_error(drizzle, CR_UNKNOWN_ERROR, unknown_sqlstate);
934
row[field] = (char*) pos;
939
*prev_pos=0; /* Terminate prev field */
942
row[field]=(char*) prev_pos+1; /* End of last field */
943
*prev_pos=0; /* Terminate last field */
948
/****************************************************************************
949
Init DRIZZLE structure or allocate one
950
****************************************************************************/
953
drizzle_create(DRIZZLE *ptr)
956
if (!drizzle_client_init)
958
drizzle_client_init=true;
959
org_my_init_done=my_init_done;
961
/* Will init threads */
967
drizzle_port = DRIZZLE_PORT;
969
struct servent *serv_ptr;
973
if builder specifically requested a default port, use that
974
(even if it coincides with our factory default).
975
only if they didn't do we check /etc/services (and, failing
976
on that, fall back to the factory default of 4427).
977
either default can be overridden by the environment variable
978
DRIZZLE_TCP_PORT, which in turn can be overridden with command
982
#if DRIZZLE_PORT_DEFAULT == 0
983
if ((serv_ptr = getservbyname("drizzle", "tcp")))
984
drizzle_port = (uint) ntohs((ushort) serv_ptr->s_port);
986
if ((env = getenv("DRIZZLE_TCP_PORT")))
987
drizzle_port =(uint) atoi(env);
991
(void) signal(SIGPIPE, SIG_IGN);
995
/* Init if new thread */
996
if (my_thread_init())
1001
ptr= (DRIZZLE *) malloc(sizeof(DRIZZLE));
1005
set_drizzle_error(NULL, CR_OUT_OF_MEMORY, unknown_sqlstate);
1008
memset(ptr, 0, sizeof(DRIZZLE));
1013
memset(ptr, 0, sizeof(DRIZZLE));
1016
ptr->options.connect_timeout= CONNECT_TIMEOUT;
1017
ptr->charset=default_client_charset_info;
1018
strcpy(ptr->net.sqlstate, not_error_sqlstate);
1021
Only enable LOAD DATA INFILE by default if configured with
1022
--enable-local-infile
1025
#if defined(ENABLED_LOCAL_INFILE) && !defined(DRIZZLE_SERVER)
1026
ptr->options.client_flag|= CLIENT_LOCAL_FILES;
1029
ptr->options.methods_to_use= DRIZZLE_OPT_GUESS_CONNECTION;
1030
ptr->options.report_data_truncation= true; /* default */
1033
By default we don't reconnect because it could silently corrupt data (after
1034
reconnection you potentially lose table locks, user variables, session
1035
variables (transactions but they are specifically dealt with in
1036
drizzle_reconnect()).
1037
This is a change: < 5.0.3 drizzle->reconnect was set to 1 by default.
1038
How this change impacts existing apps:
1039
- existing apps which relyed on the default will see a behaviour change;
1040
they will have to set reconnect=1 after drizzle_connect().
1041
- existing apps which explicitely asked for reconnection (the only way they
1042
could do it was by setting drizzle.reconnect to 1 after drizzle_connect())
1043
will not see a behaviour change.
1044
- existing apps which explicitely asked for no reconnection
1045
(drizzle.reconnect=0) will not see a behaviour change.
1054
Free all memory and resources used by the client library
1057
When calling this there should not be any other threads using
1060
To make things simpler when used with windows dll's (which calls this
1061
function automaticly), it's safe to call this function multiple times.
1065
void drizzle_server_end()
1067
if (!drizzle_client_init)
1072
/* If library called my_init(), free memory allocated by it */
1073
if (!org_my_init_done)
1080
drizzle_thread_end();
1083
drizzle_client_init= org_my_init_done= 0;
1084
#ifdef EMBEDDED_SERVER
1087
fclose(stderror_file);
1095
Fill in SSL part of DRIZZLE structure and set 'use_ssl' flag.
1096
NB! Errors are not reported until you do drizzle_connect.
1099
#define strdup_if_not_null(A) (A) == 0 ? 0 : my_strdup((A),MYF(MY_WME))
1102
Note that the drizzle argument must be initialized with drizzle_init()
1103
before calling drizzle_connect !
1106
static bool cli_read_query_result(DRIZZLE *drizzle);
1107
static DRIZZLE_RES *cli_use_result(DRIZZLE *drizzle);
1109
static DRIZZLE_METHODS client_methods=
1111
cli_read_query_result, /* read_query_result */
1112
cli_advanced_command, /* advanced_command */
1113
cli_read_rows, /* read_rows */
1114
cli_use_result, /* use_result */
1115
cli_fetch_lengths, /* fetch_lengths */
1116
cli_flush_use_result, /* flush_use_result */
1117
#ifndef DRIZZLE_SERVER
1118
cli_list_fields, /* list_fields */
1119
cli_unbuffered_fetch, /* unbuffered_fetch */
1120
cli_read_statistics, /* read_statistics */
1121
cli_read_query_result, /* next_result */
1122
cli_read_change_user_result, /* read_change_user_result */
1129
int drizzle_init_character_set(DRIZZLE *drizzle)
1131
const char *default_collation_name;
1133
/* Set character set */
1134
if (!drizzle->options.charset_name)
1136
default_collation_name= DRIZZLE_DEFAULT_COLLATION_NAME;
1137
if (!(drizzle->options.charset_name=
1138
my_strdup(DRIZZLE_DEFAULT_CHARSET_NAME,MYF(MY_WME))))
1142
default_collation_name= NULL;
1145
const char *save= charsets_dir;
1146
if (drizzle->options.charset_dir)
1147
charsets_dir=drizzle->options.charset_dir;
1148
drizzle->charset=get_charset_by_csname(drizzle->options.charset_name,
1149
MY_CS_PRIMARY, MYF(MY_WME));
1150
if (drizzle->charset && default_collation_name)
1152
const CHARSET_INFO *collation;
1154
get_charset_by_name(default_collation_name, MYF(MY_WME))))
1156
if (!my_charset_same(drizzle->charset, collation))
1158
my_printf_error(ER_UNKNOWN_ERROR,
1159
_("COLLATION %s is not valid for CHARACTER SET %s"),
1161
default_collation_name, drizzle->options.charset_name);
1162
drizzle->charset= NULL;
1166
drizzle->charset= collation;
1170
drizzle->charset= NULL;
1175
if (!drizzle->charset)
1177
if (drizzle->options.charset_dir)
1178
set_drizzle_extended_error(drizzle, CR_CANT_READ_CHARSET, unknown_sqlstate,
1179
ER(CR_CANT_READ_CHARSET),
1180
drizzle->options.charset_name,
1181
drizzle->options.charset_dir);
1184
char cs_dir_name[FN_REFLEN];
1185
get_charsets_dir(cs_dir_name);
1186
set_drizzle_extended_error(drizzle, CR_CANT_READ_CHARSET, unknown_sqlstate,
1187
ER(CR_CANT_READ_CHARSET),
1188
drizzle->options.charset_name,
1199
CLI_DRIZZLE_CONNECT(DRIZZLE *drizzle,const char *host, const char *user,
1200
const char *passwd, const char *db,
1201
uint32_t port, const char *unix_socket, uint32_t client_flag)
1203
char buff[NAME_LEN+USERNAME_LENGTH+100];
1204
char *end,*host_info=NULL;
1205
uint32_t pkt_length;
1206
NET *net= &drizzle->net;
1207
init_sigpipe_variables
1209
/* Don't give sigpipe errors if the client doesn't want them */
1210
set_sigpipe(drizzle);
1211
drizzle->methods= &client_methods;
1212
net->vio = 0; /* If something goes wrong */
1213
drizzle->client_flag=0; /* For handshake */
1215
/* use default options */
1216
if (drizzle->options.my_cnf_file || drizzle->options.my_cnf_group)
1218
drizzle_read_default_options(&drizzle->options,
1219
(drizzle->options.my_cnf_file ?
1220
drizzle->options.my_cnf_file : "my"),
1221
drizzle->options.my_cnf_group);
1222
my_free(drizzle->options.my_cnf_file,MYF(MY_ALLOW_ZERO_PTR));
1223
my_free(drizzle->options.my_cnf_group,MYF(MY_ALLOW_ZERO_PTR));
1224
drizzle->options.my_cnf_file=drizzle->options.my_cnf_group=0;
1227
/* Some empty-string-tests are done because of ODBC */
1228
if (!host || !host[0])
1229
host=drizzle->options.host;
1230
if (!user || !user[0])
1232
user=drizzle->options.user;
1238
passwd=drizzle->options.password;
1243
db=drizzle->options.db;
1245
port=drizzle->options.port;
1247
unix_socket=drizzle->options.unix_socket;
1249
drizzle->server_status=SERVER_STATUS_AUTOCOMMIT;
1252
Part 0: Grab a socket and connect it to the server
1255
(!drizzle->options.protocol ||
1256
drizzle->options.protocol == DRIZZLE_PROTOCOL_TCP))
1258
struct addrinfo *res_lst, hints, *t_res;
1260
char port_buf[NI_MAXSERV];
1262
unix_socket=0; /* This is not used */
1270
snprintf(host_info=buff, sizeof(buff)-1, ER(CR_TCP_CONNECTION), host);
1272
memset(&hints, 0, sizeof(hints));
1273
hints.ai_socktype= SOCK_STREAM;
1275
snprintf(port_buf, NI_MAXSERV, "%d", port);
1276
gai_errno= getaddrinfo(host, port_buf, &hints, &res_lst);
1280
set_drizzle_extended_error(drizzle, CR_UNKNOWN_HOST, unknown_sqlstate,
1281
ER(CR_UNKNOWN_HOST), host, errno);
1286
for (t_res= res_lst; t_res != NULL; t_res= t_res->ai_next)
1288
int sock= socket(t_res->ai_family, t_res->ai_socktype,
1289
t_res->ai_protocol);
1290
if (sock == SOCKET_ERROR)
1293
net->vio= vio_new(sock, VIO_TYPE_TCPIP, VIO_BUFFERED_READ);
1300
if (connect_with_timeout(sock, t_res->ai_addr, t_res->ai_addrlen, drizzle->options.connect_timeout))
1302
vio_delete(net->vio);
1309
freeaddrinfo(res_lst);
1314
set_drizzle_extended_error(drizzle, CR_CONN_HOST_ERROR, unknown_sqlstate,
1315
ER(CR_CONN_HOST_ERROR), host, socket_errno);
1319
if (my_net_init(net, net->vio))
1321
vio_delete(net->vio);
1323
set_drizzle_error(drizzle, CR_OUT_OF_MEMORY, unknown_sqlstate);
1326
vio_keepalive(net->vio,true);
1328
/* If user set read_timeout, let it override the default */
1329
if (drizzle->options.read_timeout)
1330
my_net_set_read_timeout(net, drizzle->options.read_timeout);
1332
/* If user set write_timeout, let it override the default */
1333
if (drizzle->options.write_timeout)
1334
my_net_set_write_timeout(net, drizzle->options.write_timeout);
1336
if (drizzle->options.max_allowed_packet)
1337
net->max_packet_size= drizzle->options.max_allowed_packet;
1339
/* Get version info */
1340
drizzle->protocol_version= PROTOCOL_VERSION; /* Assume this */
1341
if (drizzle->options.connect_timeout &&
1342
vio_poll_read(net->vio, drizzle->options.connect_timeout))
1344
set_drizzle_extended_error(drizzle, CR_SERVER_LOST, unknown_sqlstate,
1345
ER(CR_SERVER_LOST_INITIAL_COMM_WAIT),
1351
Part 1: Connection established, read and parse first packet
1354
if ((pkt_length=cli_safe_read(drizzle)) == packet_error)
1356
if (drizzle->net.last_errno == CR_SERVER_LOST)
1357
set_drizzle_extended_error(drizzle, CR_SERVER_LOST, unknown_sqlstate,
1358
ER(CR_SERVER_LOST_INITIAL_COMM_READ),
1362
/* Check if version of protocol matches current one */
1364
drizzle->protocol_version= net->read_pos[0];
1365
if (drizzle->protocol_version != PROTOCOL_VERSION)
1367
set_drizzle_extended_error(drizzle, CR_VERSION_ERROR, unknown_sqlstate,
1368
ER(CR_VERSION_ERROR), drizzle->protocol_version,
1372
end= strchr((char*) net->read_pos+1, '\0');
1373
drizzle->thread_id=uint4korr(end+1);
1376
Scramble is split into two parts because old clients does not understand
1377
long scrambles; here goes the first part.
1379
strncpy(drizzle->scramble, end, SCRAMBLE_LENGTH_323);
1380
end+= SCRAMBLE_LENGTH_323+1;
1382
if (pkt_length >= (uint) (end+1 - (char*) net->read_pos))
1383
drizzle->server_capabilities=uint2korr(end);
1384
if (pkt_length >= (uint) (end+18 - (char*) net->read_pos))
1386
/* New protocol with 16 bytes to describe server characteristics */
1387
drizzle->server_language=end[2];
1388
drizzle->server_status=uint2korr(end+3);
1391
if (pkt_length >= (uint) (end + SCRAMBLE_LENGTH - SCRAMBLE_LENGTH_323 + 1 -
1392
(char *) net->read_pos))
1393
strncpy(drizzle->scramble+SCRAMBLE_LENGTH_323, end,
1394
SCRAMBLE_LENGTH-SCRAMBLE_LENGTH_323);
1396
drizzle->server_capabilities&= ~CLIENT_SECURE_CONNECTION;
1398
if (drizzle->options.secure_auth && passwd[0] &&
1399
!(drizzle->server_capabilities & CLIENT_SECURE_CONNECTION))
1401
set_drizzle_error(drizzle, CR_SECURE_AUTH, unknown_sqlstate);
1405
if (drizzle_init_character_set(drizzle))
1408
/* Save connection information */
1409
if (!my_multi_malloc(MYF(0),
1410
&drizzle->host_info, (uint) strlen(host_info)+1,
1411
&drizzle->host, (uint) strlen(host)+1,
1412
&drizzle->unix_socket,unix_socket ?
1413
(uint) strlen(unix_socket)+1 : (uint) 1,
1414
&drizzle->server_version,
1415
(uint) (end - (char*) net->read_pos),
1417
!(drizzle->user=my_strdup(user,MYF(0))) ||
1418
!(drizzle->passwd=my_strdup(passwd,MYF(0))))
1420
set_drizzle_error(drizzle, CR_OUT_OF_MEMORY, unknown_sqlstate);
1423
strcpy(drizzle->host_info,host_info);
1424
strcpy(drizzle->host,host);
1426
strcpy(drizzle->unix_socket,unix_socket);
1428
drizzle->unix_socket=0;
1429
strcpy(drizzle->server_version,(char*) net->read_pos+1);
1433
Part 2: format and send client info to the server for access check
1436
client_flag|=drizzle->options.client_flag;
1437
client_flag|=CLIENT_CAPABILITIES;
1438
if (client_flag & CLIENT_MULTI_STATEMENTS)
1439
client_flag|= CLIENT_MULTI_RESULTS;
1442
client_flag|=CLIENT_CONNECT_WITH_DB;
1444
/* Remove options that server doesn't support */
1445
client_flag= ((client_flag &
1446
~(CLIENT_COMPRESS | CLIENT_SSL)) |
1447
(client_flag & drizzle->server_capabilities));
1448
client_flag&= ~CLIENT_COMPRESS;
1450
int4store(buff, client_flag);
1451
int4store(buff+4, net->max_packet_size);
1452
buff[8]= (char) drizzle->charset->number;
1453
memset(buff+9, 0, 32-9);
1456
drizzle->client_flag=client_flag;
1458
/* This needs to be changed as it's not useful with big packets */
1459
if (user && user[0])
1460
strncpy(end,user,USERNAME_LENGTH); /* Max user name */
1462
read_user_name((char*) end);
1464
/* We have to handle different version of handshake here */
1465
end= strchr(end, '\0') + 1;
1469
*end++= SCRAMBLE_LENGTH;
1470
memset(end, 0, SCRAMBLE_LENGTH-1);
1471
memcpy(end, passwd, strlen(passwd));
1472
end+= SCRAMBLE_LENGTH;
1476
*end++= '\0'; /* empty password */
1478
/* Add database if needed */
1479
if (db && (drizzle->server_capabilities & CLIENT_CONNECT_WITH_DB))
1481
end= strncpy(end, db, NAME_LEN) + NAME_LEN + 1;
1482
drizzle->db= my_strdup(db,MYF(MY_WME));
1485
/* Write authentication package */
1486
if (my_net_write(net, (uchar*) buff, (size_t) (end-buff)) || net_flush(net))
1488
set_drizzle_extended_error(drizzle, CR_SERVER_LOST, unknown_sqlstate,
1489
ER(CR_SERVER_LOST_SEND_AUTH),
1495
Part 3: Authorization data's been sent. Now server can reply with
1496
OK-packet, or re-request scrambled password.
1499
if ((pkt_length=cli_safe_read(drizzle)) == packet_error)
1501
if (drizzle->net.last_errno == CR_SERVER_LOST)
1502
set_drizzle_extended_error(drizzle, CR_SERVER_LOST, unknown_sqlstate,
1503
ER(CR_SERVER_LOST_READ_AUTH),
1508
if (client_flag & CLIENT_COMPRESS) /* We will use compression */
1512
if (db && drizzle_select_db(drizzle, db))
1514
if (drizzle->net.last_errno == CR_SERVER_LOST)
1515
set_drizzle_extended_error(drizzle, CR_SERVER_LOST, unknown_sqlstate,
1516
ER(CR_SERVER_LOST_SETTING_DB),
1521
if (drizzle->options.init_commands)
1523
DYNAMIC_ARRAY *init_commands= drizzle->options.init_commands;
1524
char **ptr= (char**)init_commands->buffer;
1525
char **end_command= ptr + init_commands->elements;
1527
bool reconnect=drizzle->reconnect;
1528
drizzle->reconnect=0;
1530
for (; ptr < end_command; ptr++)
1533
if (drizzle_real_query(drizzle,*ptr, (uint32_t) strlen(*ptr)))
1535
if (drizzle->fields)
1537
if (!(res= cli_use_result(drizzle)))
1539
drizzle_free_result(res);
1542
drizzle->reconnect=reconnect;
1545
reset_sigpipe(drizzle);
1549
reset_sigpipe(drizzle);
1551
/* Free alloced memory */
1552
end_server(drizzle);
1553
drizzle_close_free(drizzle);
1554
if (!(((uint32_t) client_flag) & CLIENT_REMEMBER_OPTIONS))
1555
drizzle_close_free_options(drizzle);
1561
bool drizzle_reconnect(DRIZZLE *drizzle)
1563
DRIZZLE tmp_drizzle;
1566
if (!drizzle->reconnect ||
1567
(drizzle->server_status & SERVER_STATUS_IN_TRANS) || !drizzle->host_info)
1569
/* Allow reconnect next time */
1570
drizzle->server_status&= ~SERVER_STATUS_IN_TRANS;
1571
set_drizzle_error(drizzle, CR_SERVER_GONE_ERROR, unknown_sqlstate);
1574
drizzle_create(&tmp_drizzle);
1575
tmp_drizzle.options= drizzle->options;
1576
tmp_drizzle.options.my_cnf_file= tmp_drizzle.options.my_cnf_group= 0;
1578
if (!drizzle_connect(&tmp_drizzle,drizzle->host,drizzle->user,drizzle->passwd,
1579
drizzle->db, drizzle->port, drizzle->unix_socket,
1580
drizzle->client_flag | CLIENT_REMEMBER_OPTIONS))
1582
drizzle->net.last_errno= tmp_drizzle.net.last_errno;
1583
strcpy(drizzle->net.last_error, tmp_drizzle.net.last_error);
1584
strcpy(drizzle->net.sqlstate, tmp_drizzle.net.sqlstate);
1587
if (drizzle_set_character_set(&tmp_drizzle, drizzle->charset->csname))
1589
memset(&tmp_drizzle.options, 0, sizeof(tmp_drizzle.options));
1590
drizzle_close(&tmp_drizzle);
1591
drizzle->net.last_errno= tmp_drizzle.net.last_errno;
1592
strcpy(drizzle->net.last_error, tmp_drizzle.net.last_error);
1593
strcpy(drizzle->net.sqlstate, tmp_drizzle.net.sqlstate);
1597
tmp_drizzle.reconnect= 1;
1598
tmp_drizzle.free_me= drizzle->free_me;
1600
/* Don't free options as these are now used in tmp_drizzle */
1601
memset(&drizzle->options, 0, sizeof(drizzle->options));
1603
drizzle_close(drizzle);
1604
*drizzle=tmp_drizzle;
1605
net_clear(&drizzle->net, 1);
1606
drizzle->affected_rows= ~(uint64_t) 0;
1611
/**************************************************************************
1612
Set current database
1613
**************************************************************************/
1616
drizzle_select_db(DRIZZLE *drizzle, const char *db)
1620
if ((error=simple_command(drizzle,COM_INIT_DB, (const uchar*) db,
1621
(uint32_t) strlen(db),0)))
1623
my_free(drizzle->db,MYF(MY_ALLOW_ZERO_PTR));
1624
drizzle->db=my_strdup(db,MYF(MY_WME));
1629
/*************************************************************************
1630
Send a QUIT to the server and close the connection
1631
If handle is alloced by DRIZZLE connect free it.
1632
*************************************************************************/
1634
static void drizzle_close_free_options(DRIZZLE *drizzle)
1636
my_free(drizzle->options.user,MYF(MY_ALLOW_ZERO_PTR));
1637
my_free(drizzle->options.host,MYF(MY_ALLOW_ZERO_PTR));
1638
my_free(drizzle->options.password,MYF(MY_ALLOW_ZERO_PTR));
1639
my_free(drizzle->options.unix_socket,MYF(MY_ALLOW_ZERO_PTR));
1640
my_free(drizzle->options.db,MYF(MY_ALLOW_ZERO_PTR));
1641
my_free(drizzle->options.my_cnf_file,MYF(MY_ALLOW_ZERO_PTR));
1642
my_free(drizzle->options.my_cnf_group,MYF(MY_ALLOW_ZERO_PTR));
1643
my_free(drizzle->options.charset_dir,MYF(MY_ALLOW_ZERO_PTR));
1644
my_free(drizzle->options.charset_name,MYF(MY_ALLOW_ZERO_PTR));
1645
my_free(drizzle->options.client_ip,MYF(MY_ALLOW_ZERO_PTR));
1646
if (drizzle->options.init_commands)
1648
DYNAMIC_ARRAY *init_commands= drizzle->options.init_commands;
1649
char **ptr= (char**)init_commands->buffer;
1650
char **end= ptr + init_commands->elements;
1651
for (; ptr<end; ptr++)
1652
my_free(*ptr,MYF(MY_WME));
1653
delete_dynamic(init_commands);
1654
my_free((char*)init_commands,MYF(MY_WME));
1656
memset(&drizzle->options, 0, sizeof(drizzle->options));
1661
static void drizzle_close_free(DRIZZLE *drizzle)
1663
my_free((uchar*) drizzle->host_info,MYF(MY_ALLOW_ZERO_PTR));
1664
my_free(drizzle->user,MYF(MY_ALLOW_ZERO_PTR));
1665
my_free(drizzle->passwd,MYF(MY_ALLOW_ZERO_PTR));
1666
my_free(drizzle->db,MYF(MY_ALLOW_ZERO_PTR));
1667
my_free(drizzle->info_buffer,MYF(MY_ALLOW_ZERO_PTR));
1668
drizzle->info_buffer= 0;
1670
/* Clear pointers for better safety */
1671
drizzle->host_info= drizzle->user= drizzle->passwd= drizzle->db= 0;
1675
void drizzle_close(DRIZZLE *drizzle)
1677
if (drizzle) /* Some simple safety */
1679
/* If connection is still up, send a QUIT message */
1680
if (drizzle->net.vio != 0)
1682
free_old_query(drizzle);
1683
drizzle->status=DRIZZLE_STATUS_READY; /* Force command */
1684
drizzle->reconnect=0;
1685
simple_command(drizzle,COM_QUIT,(uchar*) 0,0,1);
1686
end_server(drizzle); /* Sets drizzle->net.vio= 0 */
1688
drizzle_close_free_options(drizzle);
1689
drizzle_close_free(drizzle);
1690
if (drizzle->free_me)
1691
my_free((uchar*) drizzle,MYF(0));
1697
static bool cli_read_query_result(DRIZZLE *drizzle)
1700
uint32_t field_count;
1701
DRIZZLE_DATA *fields;
1704
if ((length = cli_safe_read(drizzle)) == packet_error)
1706
free_old_query(drizzle); /* Free old result */
1707
#ifdef DRIZZLE_CLIENT /* Avoid warn of unused labels*/
1710
pos=(uchar*) drizzle->net.read_pos;
1711
if ((field_count= net_field_length(&pos)) == 0)
1713
drizzle->affected_rows= net_field_length_ll(&pos);
1714
drizzle->insert_id= net_field_length_ll(&pos);
1716
drizzle->server_status= uint2korr(pos); pos+=2;
1717
drizzle->warning_count= uint2korr(pos); pos+=2;
1719
if (pos < drizzle->net.read_pos+length && net_field_length(&pos))
1720
drizzle->info=(char*) pos;
1723
#ifdef DRIZZLE_CLIENT
1724
if (field_count == NULL_LENGTH) /* LOAD DATA LOCAL INFILE */
1728
if (!(drizzle->options.client_flag & CLIENT_LOCAL_FILES))
1730
set_drizzle_error(drizzle, CR_MALFORMED_PACKET, unknown_sqlstate);
1734
error= handle_local_infile(drizzle,(char*) pos);
1735
if ((length= cli_safe_read(drizzle)) == packet_error || error)
1737
goto get_info; /* Get info packet */
1740
if (!(drizzle->server_status & SERVER_STATUS_AUTOCOMMIT))
1741
drizzle->server_status|= SERVER_STATUS_IN_TRANS;
1743
if (!(fields=cli_read_rows(drizzle,(DRIZZLE_FIELD*)0, 7)))
1745
if (!(drizzle->fields= unpack_fields(fields,&drizzle->field_alloc,
1746
(uint) field_count, 0)))
1748
drizzle->status= DRIZZLE_STATUS_GET_RESULT;
1749
drizzle->field_count= (uint) field_count;
1755
Send the query and return so we can do something else.
1756
Needs to be followed by drizzle_read_query_result() when we want to
1757
finish processing it.
1761
drizzle_send_query(DRIZZLE *drizzle, const char* query, uint32_t length)
1763
return(simple_command(drizzle, COM_QUERY, (uchar*) query, length, 1));
1768
drizzle_real_query(DRIZZLE *drizzle, const char *query, uint32_t length)
1770
if (drizzle_send_query(drizzle,query,length))
1772
return((int) (*drizzle->methods->read_query_result)(drizzle));
1776
/**************************************************************************
1777
Alloc result struct for buffered results. All rows are read to buffer.
1778
drizzle_data_seek may be used.
1779
**************************************************************************/
1781
DRIZZLE_RES * drizzle_store_result(DRIZZLE *drizzle)
1783
DRIZZLE_RES *result;
1785
if (!drizzle->fields)
1787
if (drizzle->status != DRIZZLE_STATUS_GET_RESULT)
1789
set_drizzle_error(drizzle, CR_COMMANDS_OUT_OF_SYNC, unknown_sqlstate);
1792
drizzle->status=DRIZZLE_STATUS_READY; /* server is ready */
1793
if (!(result=(DRIZZLE_RES*) my_malloc((uint) (sizeof(DRIZZLE_RES)+
1795
drizzle->field_count),
1796
MYF(MY_WME | MY_ZEROFILL))))
1798
set_drizzle_error(drizzle, CR_OUT_OF_MEMORY, unknown_sqlstate);
1801
result->methods= drizzle->methods;
1802
result->eof= 1; /* Marker for buffered */
1803
result->lengths= (uint32_t*) (result+1);
1805
(*drizzle->methods->read_rows)(drizzle,drizzle->fields,drizzle->field_count)))
1807
my_free((uchar*) result,MYF(0));
1810
drizzle->affected_rows= result->row_count= result->data->rows;
1811
result->data_cursor= result->data->data;
1812
result->fields= drizzle->fields;
1813
result->field_alloc= drizzle->field_alloc;
1814
result->field_count= drizzle->field_count;
1815
/* The rest of result members is zeroed in malloc */
1816
drizzle->fields=0; /* fields is now in result */
1817
clear_alloc_root(&drizzle->field_alloc);
1818
/* just in case this was mistakenly called after drizzle_stmt_execute() */
1819
drizzle->unbuffered_fetch_owner= 0;
1820
return(result); /* Data fetched */
1824
/**************************************************************************
1825
Alloc struct for use with unbuffered reads. Data is fetched by domand
1826
when calling to drizzle_fetch_row.
1827
DRIZZLE_DATA_seek is a noop.
1829
No other queries may be specified with the same DRIZZLE handle.
1830
There shouldn't be much processing per row because DRIZZLE server shouldn't
1831
have to wait for the client (and will not wait more than 30 sec/packet).
1832
**************************************************************************/
1834
static DRIZZLE_RES * cli_use_result(DRIZZLE *drizzle)
1836
DRIZZLE_RES *result;
1838
if (!drizzle->fields)
1840
if (drizzle->status != DRIZZLE_STATUS_GET_RESULT)
1842
set_drizzle_error(drizzle, CR_COMMANDS_OUT_OF_SYNC, unknown_sqlstate);
1845
if (!(result=(DRIZZLE_RES*) my_malloc(sizeof(*result)+
1846
sizeof(uint32_t)*drizzle->field_count,
1847
MYF(MY_WME | MY_ZEROFILL))))
1849
result->lengths=(uint32_t*) (result+1);
1850
result->methods= drizzle->methods;
1851
if (!(result->row=(DRIZZLE_ROW)
1852
my_malloc(sizeof(result->row[0])*(drizzle->field_count+1), MYF(MY_WME))))
1853
{ /* Ptrs: to one row */
1854
my_free((uchar*) result,MYF(0));
1857
result->fields= drizzle->fields;
1858
result->field_alloc= drizzle->field_alloc;
1859
result->field_count= drizzle->field_count;
1860
result->current_field=0;
1861
result->handle= drizzle;
1862
result->current_row= 0;
1863
drizzle->fields=0; /* fields is now in result */
1864
clear_alloc_root(&drizzle->field_alloc);
1865
drizzle->status=DRIZZLE_STATUS_USE_RESULT;
1866
drizzle->unbuffered_fetch_owner= &result->unbuffered_fetch_cancelled;
1867
return(result); /* Data is read to be fetched */
1871
/**************************************************************************
1872
Return next row of the query results
1873
**************************************************************************/
1876
drizzle_fetch_row(DRIZZLE_RES *res)
1879
{ /* Unbufferred fetch */
1882
DRIZZLE *drizzle= res->handle;
1883
if (drizzle->status != DRIZZLE_STATUS_USE_RESULT)
1885
set_drizzle_error(drizzle,
1886
res->unbuffered_fetch_cancelled ?
1887
CR_FETCH_CANCELED : CR_COMMANDS_OUT_OF_SYNC,
1890
else if (!(read_one_row(drizzle, res->field_count, res->row, res->lengths)))
1893
return(res->current_row=res->row);
1896
drizzle->status=DRIZZLE_STATUS_READY;
1898
Reset only if owner points to us: there is a chance that somebody
1899
started new query after drizzle_stmt_close():
1901
if (drizzle->unbuffered_fetch_owner == &res->unbuffered_fetch_cancelled)
1902
drizzle->unbuffered_fetch_owner= 0;
1903
/* Don't clear handle in drizzle_free_result */
1906
return((DRIZZLE_ROW) NULL);
1910
if (!res->data_cursor)
1912
return(res->current_row=(DRIZZLE_ROW) NULL);
1914
tmp = res->data_cursor->data;
1915
res->data_cursor = res->data_cursor->next;
1916
return(res->current_row=tmp);
1921
/**************************************************************************
1922
Get column lengths of the current row
1923
If one uses drizzle_use_result, res->lengths contains the length information,
1924
else the lengths are calculated from the offset between pointers.
1925
**************************************************************************/
1928
drizzle_fetch_lengths(DRIZZLE_RES *res)
1932
if (!(column=res->current_row))
1933
return 0; /* Something is wrong */
1935
(*res->methods->fetch_lengths)(res->lengths, column, res->field_count);
1936
return res->lengths;
1941
drizzle_options(DRIZZLE *drizzle,enum drizzle_option option, const void *arg)
1944
case DRIZZLE_OPT_CONNECT_TIMEOUT:
1945
drizzle->options.connect_timeout= *(uint*) arg;
1947
case DRIZZLE_OPT_READ_TIMEOUT:
1948
drizzle->options.read_timeout= *(uint*) arg;
1950
case DRIZZLE_OPT_WRITE_TIMEOUT:
1951
drizzle->options.write_timeout= *(uint*) arg;
1953
case DRIZZLE_OPT_COMPRESS:
1954
drizzle->options.compress= 1; /* Remember for connect */
1955
drizzle->options.client_flag|= CLIENT_COMPRESS;
1957
case DRIZZLE_OPT_LOCAL_INFILE: /* Allow LOAD DATA LOCAL ?*/
1958
if (!arg || test(*(uint*) arg))
1959
drizzle->options.client_flag|= CLIENT_LOCAL_FILES;
1961
drizzle->options.client_flag&= ~CLIENT_LOCAL_FILES;
1963
case DRIZZLE_INIT_COMMAND:
1964
add_init_command(&drizzle->options,arg);
1966
case DRIZZLE_READ_DEFAULT_FILE:
1967
my_free(drizzle->options.my_cnf_file,MYF(MY_ALLOW_ZERO_PTR));
1968
drizzle->options.my_cnf_file=my_strdup(arg,MYF(MY_WME));
1970
case DRIZZLE_READ_DEFAULT_GROUP:
1971
my_free(drizzle->options.my_cnf_group,MYF(MY_ALLOW_ZERO_PTR));
1972
drizzle->options.my_cnf_group=my_strdup(arg,MYF(MY_WME));
1974
case DRIZZLE_SET_CHARSET_DIR:
1975
my_free(drizzle->options.charset_dir,MYF(MY_ALLOW_ZERO_PTR));
1976
drizzle->options.charset_dir=my_strdup(arg,MYF(MY_WME));
1978
case DRIZZLE_SET_CHARSET_NAME:
1979
my_free(drizzle->options.charset_name,MYF(MY_ALLOW_ZERO_PTR));
1980
drizzle->options.charset_name=my_strdup(arg,MYF(MY_WME));
1982
case DRIZZLE_OPT_PROTOCOL:
1983
drizzle->options.protocol= *(const uint*) arg;
1985
case DRIZZLE_OPT_USE_REMOTE_CONNECTION:
1986
case DRIZZLE_OPT_GUESS_CONNECTION:
1987
drizzle->options.methods_to_use= option;
1989
case DRIZZLE_SET_CLIENT_IP:
1990
drizzle->options.client_ip= my_strdup(arg, MYF(MY_WME));
1992
case DRIZZLE_SECURE_AUTH:
1993
drizzle->options.secure_auth= *(const bool *) arg;
1995
case DRIZZLE_REPORT_DATA_TRUNCATION:
1996
drizzle->options.report_data_truncation= test(*(const bool *) arg);
1998
case DRIZZLE_OPT_RECONNECT:
1999
drizzle->reconnect= *(const bool *) arg;
2001
case DRIZZLE_OPT_SSL_VERIFY_SERVER_CERT:
2002
if (*(const bool*) arg)
2003
drizzle->options.client_flag|= CLIENT_SSL_VERIFY_SERVER_CERT;
2005
drizzle->options.client_flag&= ~CLIENT_SSL_VERIFY_SERVER_CERT;
2014
/****************************************************************************
2015
Functions to get information from the DRIZZLE structure
2016
These are functions to make shared libraries more usable.
2017
****************************************************************************/
2020
uint64_t drizzle_num_rows(const DRIZZLE_RES *res)
2022
return res->row_count;
2025
unsigned int drizzle_num_fields(const DRIZZLE_RES *res)
2027
return res->field_count;
2030
uint drizzle_errno(const DRIZZLE *drizzle)
2032
return drizzle ? drizzle->net.last_errno : drizzle_server_last_errno;
2036
const char * drizzle_error(const DRIZZLE *drizzle)
2038
return drizzle ? _(drizzle->net.last_error) : _(drizzle_server_last_error);
2043
Get version number for server in a form easy to test on
2046
drizzle_get_server_version()
2053
We will ensure that a newer server always has a bigger number.
2056
Signed number > 323000
2060
drizzle_get_server_version(const DRIZZLE *drizzle)
2062
uint major, minor, version;
2063
char *pos= drizzle->server_version, *end_pos;
2064
major= (uint) strtoul(pos, &end_pos, 10); pos=end_pos+1;
2065
minor= (uint) strtoul(pos, &end_pos, 10); pos=end_pos+1;
2066
version= (uint) strtoul(pos, &end_pos, 10);
2067
return (uint32_t) major*10000L+(uint32_t) (minor*100+version);
2072
drizzle_set_character_set function sends SET NAMES cs_name to
2073
the server (which changes character_set_client, character_set_result
2074
and character_set_connection) and updates drizzle->charset so other
2075
functions like drizzle_real_escape will work correctly.
2077
int drizzle_set_character_set(DRIZZLE *drizzle, const char *cs_name)
2079
const CHARSET_INFO *cs;
2080
const char *save_csdir= charsets_dir;
2082
if (drizzle->options.charset_dir)
2083
charsets_dir= drizzle->options.charset_dir;
2085
if (strlen(cs_name) < MY_CS_NAME_SIZE &&
2086
(cs= get_charset_by_csname(cs_name, MY_CS_PRIMARY, MYF(0))))
2088
char buff[MY_CS_NAME_SIZE + 10];
2089
charsets_dir= save_csdir;
2090
sprintf(buff, "SET NAMES %s", cs_name);
2091
if (!drizzle_real_query(drizzle, buff, strlen(buff)))
2093
drizzle->charset= cs;
2098
char cs_dir_name[FN_REFLEN];
2099
get_charsets_dir(cs_dir_name);
2100
set_drizzle_extended_error(drizzle, CR_CANT_READ_CHARSET, unknown_sqlstate,
2101
ER(CR_CANT_READ_CHARSET), cs_name, cs_dir_name);
2103
charsets_dir= save_csdir;
2104
return drizzle->net.last_errno;