1
/* - mode: c; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2008 Sun Microsystems, Inc.
6
* This program is free software; you can redistribute it and/or modify
7
* it under the terms of the GNU General Public License as published by
8
* the Free Software Foundation; version 2 of the License.
10
* This program is distributed in the hope that it will be useful,
11
* but WITHOUT ANY WARRANTY; without even the implied warranty of
12
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
* GNU General Public License for more details.
15
* You should have received a copy of the GNU General Public License
16
* along with this program; if not, write to the Free Software
17
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
21
This file is included by both libdrizzle.c (the DRIZZLE client C API)
22
and the drizzled server to connect to another DRIZZLE server.
24
The differences for the two cases are:
26
- Things that only works for the client:
27
- Trying to automaticly determinate user name if not supplied to
29
- Support for reading local file with LOAD DATA LOCAL
30
- SHARED memory handling
33
- Things that only works for the server
34
- Alarm handling on connect
36
In all other cases, the code should be idential for the client and
42
#include <libdrizzle/libdrizzle.h>
43
#include <libdrizzle/net_serv.h>
44
#include "libdrizzle_priv.h"
47
#include <sys/ioctl.h>
51
/* Remove client convenience wrappers */
52
#undef max_allowed_packet
53
#undef net_buffer_length
55
#include <libdrizzle/errmsg.h>
56
#include <vio/violite.h>
65
#include <sys/socket.h>
66
#include <netinet/in.h>
67
#include <arpa/inet.h>
72
#ifdef HAVE_SYS_SELECT_H
73
#include <sys/select.h>
79
#define SOCKET_ERROR -1
82
#include <drizzled/version.h>
83
#include <libdrizzle/sql_common.h>
84
#include <libdrizzle/gettext.h>
85
#include "local_infile.h"
94
#if defined(HAVE_GETPWUID) && defined(NO_GETPWUID_DECL)
95
struct passwd *getpwuid(uid_t);
101
/*****************************************************************************
102
Read a packet from server. Give error message if socket was down
103
or packet is an error message
104
*****************************************************************************/
106
uint32_t cli_safe_read(DRIZZLE *drizzle)
108
NET *net= &drizzle->net;
112
len=my_net_read(net);
114
if (len == packet_error || len == 0)
116
#ifdef DRIZZLE_SERVER
117
if (net->vio && vio_was_interrupted(net->vio))
118
return (packet_error);
119
#endif /*DRIZZLE_SERVER*/
120
drizzle_disconnect(drizzle);
121
drizzle_set_error(drizzle, net->last_errno == CR_NET_PACKET_TOO_LARGE ?
122
CR_NET_PACKET_TOO_LARGE : CR_SERVER_LOST,
123
sqlstate_get_unknown());
124
return (packet_error);
126
if (net->read_pos[0] == 255)
130
char *pos=(char*) net->read_pos+1;
131
net->last_errno=uint2korr(pos);
136
strncpy(net->sqlstate, pos+1, SQLSTATE_LENGTH);
137
pos+= SQLSTATE_LENGTH+1;
142
The SQL state hasn't been received -- it should be reset to HY000
143
(unknown error sql state).
146
strcpy(net->sqlstate, sqlstate_get_unknown());
149
strncpy(net->last_error,(char*) pos, min((uint32_t) len,
150
(uint32_t) sizeof(net->last_error)-1));
153
drizzle_set_error(drizzle, CR_UNKNOWN_ERROR, sqlstate_get_unknown());
155
Cover a protocol design error: error packet does not
156
contain the server status. Therefore, the client has no way
157
to find out whether there are more result sets of
158
a multiple-result-set statement pending. Luckily, in 5.0 an
159
error always aborts execution of a statement, wherever it is
160
a multi-statement or a stored procedure, so it should be
161
safe to unconditionally turn off the flag here.
163
drizzle->server_status&= ~SERVER_MORE_RESULTS_EXISTS;
165
return(packet_error);
171
cli_advanced_command(DRIZZLE *drizzle, enum enum_server_command command,
172
const unsigned char *header, uint32_t header_length,
173
const unsigned char *arg, uint32_t arg_length, bool skip_check)
175
NET *net= &drizzle->net;
177
bool stmt_skip= false;
179
if (drizzle->net.vio == 0)
180
{ /* Do reconnect if possible */
181
if (drizzle_reconnect(drizzle) || stmt_skip)
184
if (drizzle->status != DRIZZLE_STATUS_READY ||
185
drizzle->server_status & SERVER_MORE_RESULTS_EXISTS)
187
drizzle_set_error(drizzle, CR_COMMANDS_OUT_OF_SYNC,
188
sqlstate_get_unknown());
192
net_clear_error(net);
194
drizzle->affected_rows= ~(uint64_t) 0;
196
We don't want to clear the protocol buffer on COM_QUIT, because if
197
the previous command was a shutdown command, we may have the
198
response for the COM_QUIT already in the communication buffer
200
net_clear(&drizzle->net, (command != COM_QUIT));
202
if (net_write_command(net,(unsigned char) command, header, header_length,
205
if (net->last_errno == CR_NET_PACKET_TOO_LARGE)
207
drizzle_set_error(drizzle, CR_NET_PACKET_TOO_LARGE, sqlstate_get_unknown());
210
drizzle_disconnect(drizzle);
211
if (drizzle_reconnect(drizzle) || stmt_skip)
213
if (net_write_command(net,(unsigned char) command, header, header_length,
216
drizzle_set_error(drizzle, CR_SERVER_GONE_ERROR, sqlstate_get_unknown());
222
result= ((drizzle->packet_length=cli_safe_read(drizzle)) == packet_error ?
228
void free_old_query(DRIZZLE *drizzle)
232
/* TODO - we need to de-alloc field storage */
233
free(drizzle->fields->catalog);
234
free(drizzle->fields->db);
235
free(drizzle->fields->table);
236
free(drizzle->fields->org_table);
237
free(drizzle->fields->name);
238
free(drizzle->fields->org_name);
239
free(drizzle->fields->def);
240
free(drizzle->fields);
243
/* init_alloc_root(&drizzle->field_alloc,8192,0); */ /* Assume rowlength < 8192 */
245
drizzle->field_count= 0; /* For API */
246
drizzle->warning_count= 0;
256
drizzle_free_result(DRIZZLE_RES *result)
260
DRIZZLE *drizzle= result->handle;
263
if (drizzle->unbuffered_fetch_owner == &result->unbuffered_fetch_cancelled)
264
drizzle->unbuffered_fetch_owner= 0;
265
if (drizzle->status == DRIZZLE_STATUS_USE_RESULT)
267
(*drizzle->methods->flush_use_result)(drizzle);
268
drizzle->status=DRIZZLE_STATUS_READY;
269
if (drizzle->unbuffered_fetch_owner)
270
*drizzle->unbuffered_fetch_owner= true;
273
free_rows(result->data);
274
/* TODO: free result->fields */
276
free((unsigned char*) result->row);
277
free((unsigned char*) result);
284
/* Read all rows (fields or data) from server */
286
DRIZZLE_DATA *cli_read_rows(DRIZZLE *drizzle, DRIZZLE_FIELD *DRIZZLE_FIELDs, uint32_t fields)
293
DRIZZLE_DATA *result;
294
DRIZZLE_ROWS **prev_ptr,*cur;
295
NET *net = &drizzle->net;
297
if ((pkt_len= cli_safe_read(drizzle)) == packet_error)
299
if (!(result=(DRIZZLE_DATA*) malloc(sizeof(DRIZZLE_DATA))))
301
drizzle_set_error(drizzle, CR_OUT_OF_MEMORY,
302
sqlstate_get_unknown());
305
memset(result, 0, sizeof(DRIZZLE_DATA));
306
prev_ptr= &result->data;
308
result->fields=fields;
311
The last EOF packet is either a 254 (0xFE) character followed by 1-7 status bytes.
313
This doesn't conflict with normal usage of 254 which stands for a
314
string where the length of the string is 8 bytes. (see net_field_length())
317
while (*(cp=net->read_pos) != DRIZZLE_PROTOCOL_NO_MORE_DATA || pkt_len >= 8)
320
if (!(cur= (DRIZZLE_ROWS*) malloc(sizeof(DRIZZLE_ROWS))) ||
321
!(cur->data= ((DRIZZLE_ROW) malloc((fields+1)*sizeof(char *)+pkt_len))))
324
drizzle_set_error(drizzle, CR_OUT_OF_MEMORY, sqlstate_get_unknown());
328
prev_ptr= &cur->next;
329
to= (char*) (cur->data+fields+1);
331
for (field=0 ; field < fields ; field++)
333
if ((len= net_field_length(&cp)) == NULL_LENGTH)
335
cur->data[field] = 0;
339
cur->data[field] = to;
340
if (len > (uint32_t) (end_to - to))
343
drizzle_set_error(drizzle, CR_MALFORMED_PACKET,
344
sqlstate_get_unknown());
353
if (DRIZZLE_FIELDs[field].max_length < len)
354
DRIZZLE_FIELDs[field].max_length=len;
358
cur->data[field]=to; /* End of last field */
359
if ((pkt_len=cli_safe_read(drizzle)) == packet_error)
365
*prev_ptr=0; /* last pointer is null */
366
if (pkt_len > 1) /* DRIZZLE 4.1 protocol */
368
drizzle->warning_count= uint2korr(cp+1);
369
drizzle->server_status= uint2korr(cp+3);
375
Read one row. Uses packet buffer as storage for fields.
376
When next packet is read, the previous field values are destroyed
381
read_one_row(DRIZZLE *drizzle, uint32_t fields, DRIZZLE_ROW row, uint32_t *lengths)
384
uint32_t pkt_len,len;
385
unsigned char *pos, *prev_pos, *end_pos;
386
NET *net= &drizzle->net;
388
if ((pkt_len=cli_safe_read(drizzle)) == packet_error)
390
if (pkt_len <= 8 && net->read_pos[0] == DRIZZLE_PROTOCOL_NO_MORE_DATA)
392
if (pkt_len > 1) /* DRIZZLE 4.1 protocol */
394
drizzle->warning_count= uint2korr(net->read_pos+1);
395
drizzle->server_status= uint2korr(net->read_pos+3);
397
return 1; /* End of data */
399
prev_pos= 0; /* allowed to write at packet[-1] */
402
for (field=0 ; field < fields ; field++)
404
if ((len= net_field_length(&pos)) == NULL_LENGTH)
411
if (len > (uint32_t) (end_pos - pos))
413
drizzle_set_error(drizzle, CR_UNKNOWN_ERROR,
414
sqlstate_get_unknown());
417
row[field] = (char*) pos;
422
*prev_pos=0; /* Terminate prev field */
425
row[field]=(char*) prev_pos+1; /* End of last field */
426
*prev_pos=0; /* Terminate last field */
431
/**************************************************************************
432
Return next row of the query results
433
**************************************************************************/
436
drizzle_fetch_row(DRIZZLE_RES *res)
439
{ /* Unbufferred fetch */
442
DRIZZLE *drizzle= res->handle;
443
if (drizzle->status != DRIZZLE_STATUS_USE_RESULT)
445
drizzle_set_error(drizzle,
446
res->unbuffered_fetch_cancelled ?
447
CR_FETCH_CANCELED : CR_COMMANDS_OUT_OF_SYNC,
448
sqlstate_get_unknown());
450
else if (!(read_one_row(drizzle, res->field_count, res->row, res->lengths)))
453
return(res->current_row=res->row);
456
drizzle->status=DRIZZLE_STATUS_READY;
458
Reset only if owner points to us: there is a chance that somebody
459
started new query after drizzle_stmt_close():
461
if (drizzle->unbuffered_fetch_owner == &res->unbuffered_fetch_cancelled)
462
drizzle->unbuffered_fetch_owner= 0;
463
/* Don't clear handle in drizzle_free_result */
466
return((DRIZZLE_ROW) NULL);
470
if (!res->data_cursor)
472
return(res->current_row=(DRIZZLE_ROW) NULL);
474
tmp = res->data_cursor->data;
475
res->data_cursor = res->data_cursor->next;
476
return(res->current_row=tmp);
481
/**************************************************************************
482
Get column lengths of the current row
483
If one uses drizzle_use_result, res->lengths contains the length information,
484
else the lengths are calculated from the offset between pointers.
485
**************************************************************************/
488
drizzle_fetch_lengths(DRIZZLE_RES *res)
492
if (!(column=res->current_row))
493
return 0; /* Something is wrong */
495
(*res->methods->fetch_lengths)(res->lengths, column, res->field_count);
501
drizzle_options(DRIZZLE *drizzle,enum drizzle_option option, const void *arg)
504
case DRIZZLE_OPT_CONNECT_TIMEOUT:
505
drizzle->options.connect_timeout= *(uint32_t*) arg;
507
case DRIZZLE_OPT_READ_TIMEOUT:
508
drizzle->options.read_timeout= *(uint32_t*) arg;
510
case DRIZZLE_OPT_WRITE_TIMEOUT:
511
drizzle->options.write_timeout= *(uint32_t*) arg;
513
case DRIZZLE_OPT_COMPRESS:
514
drizzle->options.compress= 1; /* Remember for connect */
515
drizzle->options.client_flag|= CLIENT_COMPRESS;
517
case DRIZZLE_OPT_LOCAL_INFILE: /* Allow LOAD DATA LOCAL ?*/
518
if (!arg || (*(uint32_t*) arg) ? 1 : 0)
519
drizzle->options.client_flag|= CLIENT_LOCAL_FILES;
521
drizzle->options.client_flag&= ~CLIENT_LOCAL_FILES;
523
case DRIZZLE_READ_DEFAULT_FILE:
524
if (drizzle->options.my_cnf_file != NULL)
525
free(drizzle->options.my_cnf_file);
526
drizzle->options.my_cnf_file=strdup(arg);
528
case DRIZZLE_READ_DEFAULT_GROUP:
529
if (drizzle->options.my_cnf_group != NULL)
530
free(drizzle->options.my_cnf_group);
531
drizzle->options.my_cnf_group=strdup(arg);
533
case DRIZZLE_OPT_PROTOCOL:
535
case DRIZZLE_OPT_USE_REMOTE_CONNECTION:
536
case DRIZZLE_OPT_GUESS_CONNECTION:
537
drizzle->options.methods_to_use= option;
539
case DRIZZLE_SET_CLIENT_IP:
540
drizzle->options.client_ip= strdup(arg);
542
case DRIZZLE_SECURE_AUTH:
543
drizzle->options.secure_auth= *(const bool *) arg;
545
case DRIZZLE_REPORT_DATA_TRUNCATION:
546
drizzle->options.report_data_truncation= (*(const bool *) arg) ? 1 : 0;
548
case DRIZZLE_OPT_RECONNECT:
549
drizzle->reconnect= *(const bool *) arg;
551
case DRIZZLE_OPT_SSL_VERIFY_SERVER_CERT:
552
if (*(const bool*) arg)
553
drizzle->options.client_flag|= CLIENT_SSL_VERIFY_SERVER_CERT;
555
drizzle->options.client_flag&= ~CLIENT_SSL_VERIFY_SERVER_CERT;
564
/****************************************************************************
565
Functions to get information from the DRIZZLE structure
566
These are functions to make shared libraries more usable.
567
****************************************************************************/
570
uint64_t drizzle_num_rows(const DRIZZLE_RES *res)
572
return res->row_count;
575
unsigned int drizzle_num_fields(const DRIZZLE_RES *res)
577
return res->field_count;
582
Get version number for server in a form easy to test on
585
drizzle_get_server_version()
592
We will ensure that a newer server always has a bigger number.
595
Signed number > 323000
599
drizzle_get_server_version(const DRIZZLE *drizzle)
601
uint32_t major, minor, version;
602
char *pos= drizzle->server_version, *end_pos;
603
major= (uint32_t) strtoul(pos, &end_pos, 10); pos=end_pos+1;
604
minor= (uint32_t) strtoul(pos, &end_pos, 10); pos=end_pos+1;
605
version= (uint32_t) strtoul(pos, &end_pos, 10);
606
return (uint32_t) major*10000L+(uint32_t) (minor*100+version);