2
* Drizzle Client & Protocol Library
4
* Copyright (C) 2008 Eric Day (eday@oddments.org)
7
* Redistribution and use in source and binary forms, with or without
8
* modification, are permitted provided that the following conditions are
11
* * Redistributions of source code must retain the above copyright
12
* notice, this list of conditions and the following disclaimer.
14
* * Redistributions in binary form must reproduce the above
15
* copyright notice, this list of conditions and the following disclaimer
16
* in the documentation and/or other materials provided with the
19
* * The names of its contributors may not be used to endorse or
20
* promote products derived from this software without specific prior
23
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
24
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
25
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
26
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
27
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
28
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
29
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
30
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
31
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
32
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
33
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
48
#include <libdrizzle/drizzle_client.h>
68
drizzle_result_st result;
69
drizzle_column_st column;
78
client_con_st *client_con_list;
79
uint32_t client_con_count;
85
char client_process(client_st *client, client_con_st *client_con);
86
void con_info(drizzle_con_st *con);
87
void result_info(drizzle_result_st *result);
88
void column_info(drizzle_column_st *column);
90
#define CLIENT_ERROR(__function, __ret, __client) { \
91
printf(__function ":%d:%s\n", __ret, \
92
drizzle_error(&((__client)->drizzle))); \
95
int main(int argc, char *argv[])
100
drizzle_return_t ret;
102
int wait_for_connections= 0;
104
client_con_st *client_con;
108
char *password= NULL;
111
memset(&client, 0, sizeof(client_st));
113
/* Use one connection by default. */
114
client.client_con_count= 1;
116
while ((c = getopt(argc, argv, "bB:c:d:h:Hmp:P:u:")) != -1)
125
if (!strcasecmp(optarg, "none"))
126
client.level= BUFFER_NONE;
127
else if (!strcasecmp(optarg, "field"))
128
client.level= BUFFER_FIELD;
129
else if (!strcasecmp(optarg, "row"))
130
client.level= BUFFER_ROW;
131
else if (!strcasecmp(optarg, "all"))
132
client.level= BUFFER_ALL;
135
printf("Invalid buffer level: %s\n", optarg);
141
client.client_con_count= (uint32_t)atoi(optarg);
153
client.mysql_protocol= true;
161
port= (in_port_t)atoi(optarg);
170
printf("\nUsage: %s [options] [query]\n", argv[0]);
171
printf("\t-b - Use blocking sockets\n");
172
printf("\t-B <level> - Use buffer <level>, options are:\n");
173
printf("\t none - Don't buffer anything (default)\n");
174
printf("\t field - Only buffer individual fields\n");
175
printf("\t row - Only buffer individual rows\n");
176
printf("\t all - Buffer entire result\n");
177
printf("\t-c <cons> - Create <cons> connections\n");
178
printf("\t-d <db> - Use <db> for the connection\n");
179
printf("\t-h <host> - Connect to <host>\n");
180
printf("\t-H - Print this help menu\n");
181
printf("\t-m - Use MySQL protocol\n");
182
printf("\t-p <password> - Use <password> for authentication\n");
183
printf("\t-P <port> - Connect to <port>\n");
184
printf("\t-u <user> - Use <user> for authentication\n");
191
client.query= argv[optind];
192
client.query_len= strlen(client.query);
195
if (client.client_con_count > 0)
197
client.client_con_list= calloc(client.client_con_count,
198
sizeof(client_con_st));
199
if (client.client_con_list == NULL)
201
printf("calloc:%d\n", errno);
206
/* This may fail if there is other initialization that fails. See docs. */
207
if (drizzle_create(&(client.drizzle)) == NULL)
209
printf("drizzle_create failed\n");
214
drizzle_add_options(&(client.drizzle), DRIZZLE_NON_BLOCKING);
216
/* Start all connections, and if in non-blocking mode, return as soon as the
217
connection would block. In blocking mode, this completes the entire
218
connection/query/result. */
219
for (x= 0; x < client.client_con_count; x++)
221
/* This may fail if there is other initialization that fails. See docs. */
222
con= drizzle_con_add_tcp(&(client.drizzle),
223
&(client.client_con_list[x].con),
224
host, port, user, password, db,
225
client.mysql_protocol ? DRIZZLE_CON_MYSQL : 0);
227
CLIENT_ERROR("drizzle_con_add_tcp", 0, &client);
228
drizzle_con_set_context(&(client.client_con_list[x].con),
229
&(client.client_con_list[x]));
231
if (client_process(&client, &(client.client_con_list[x])) == 1)
232
wait_for_connections++;
235
/* If in non-blocking mode, continue to process connections as they become
236
ready. Loop exits when all connections have completed. */
237
while (wait_for_connections != 0)
239
ret= drizzle_con_wait(&(client.drizzle));
240
if (ret != DRIZZLE_RETURN_OK)
241
CLIENT_ERROR("drizzle_con_wait", ret, &client);
243
while ((con= drizzle_con_ready(&(client.drizzle))) != NULL)
245
client_con= (client_con_st *)drizzle_con_context(con);
247
if (client_process(&client, client_con) == 0)
248
wait_for_connections--;
252
for (x= 0; x < client.client_con_count; x++)
253
drizzle_con_free(&(client.client_con_list[x].con));
255
drizzle_free(&(client.drizzle));
257
if (client.client_con_list != NULL)
258
free(client.client_con_list);
263
char client_process(client_st *client, client_con_st *client_con)
265
drizzle_return_t ret;
266
drizzle_column_st *column;
270
drizzle_field_t field;
275
switch (client_con->state)
278
if (client->query == NULL)
281
/* This may fail if some allocation fails, but it will set ret. */
282
(void)drizzle_query(&(client_con->con), &(client_con->result),
283
client->query, client->query_len, &ret);
284
if (ret == DRIZZLE_RETURN_IO_WAIT)
286
else if (ret != DRIZZLE_RETURN_OK)
287
CLIENT_ERROR("drizzle_query", ret, client);
289
result_info(&(client_con->result));
291
if (drizzle_result_column_count(&(client_con->result)) == 0)
294
client_con->state= CLIENT_FIELDS;
297
if (client->level == BUFFER_ALL)
299
ret= drizzle_result_buffer(&(client_con->result));
300
if (ret == DRIZZLE_RETURN_IO_WAIT)
302
else if (ret != DRIZZLE_RETURN_OK)
303
CLIENT_ERROR("drizzle_result_buffer", ret, client);
305
while ((column= drizzle_column_next(&(client_con->result))) != NULL)
312
column= drizzle_column_read(&(client_con->result),
313
&(client_con->column), &ret);
314
if (ret == DRIZZLE_RETURN_IO_WAIT)
316
else if (ret != DRIZZLE_RETURN_OK)
317
CLIENT_ERROR("drizzle_column_read", ret, client);
323
drizzle_column_free(column);
327
client_con->state= CLIENT_ROWS;
330
if (client->level == BUFFER_ALL)
332
/* Everything has been buffered, just loop through and print. */
333
while ((row= drizzle_row_next(&(client_con->result))) != NULL)
335
field_sizes= drizzle_row_field_sizes(&(client_con->result));
337
printf("Row: %" PRId64 "\n",
338
drizzle_row_current(&(client_con->result)));
340
for (x= 0; x < drizzle_result_column_count(&(client_con->result)); x++)
346
printf(" (%zd) %.*s\n", field_sizes[x], (int32_t)field_sizes[x],
354
drizzle_result_free(&(client_con->result));
360
if (client->level == BUFFER_NONE || client->level == BUFFER_FIELD)
362
/* Still need to read a row at a time, and then each field. */
363
if (client_con->row == 0)
365
client_con->row= drizzle_row_read(&(client_con->result), &ret);
366
if (ret == DRIZZLE_RETURN_IO_WAIT)
371
else if (ret != DRIZZLE_RETURN_OK)
372
CLIENT_ERROR("drizzle_row", ret, client);
374
if (client_con->row == 0)
376
drizzle_result_free(&(client_con->result));
380
printf("Row: %" PRId64 "\n", client_con->row);
385
if (client->level == BUFFER_FIELD)
387
/* Since an entire field is buffered, we don't need to worry about
389
field= drizzle_field_buffer(&(client_con->result), &total, &ret);
394
field= drizzle_field_read(&(client_con->result), &offset, &length,
398
if (ret == DRIZZLE_RETURN_IO_WAIT)
400
else if (ret == DRIZZLE_RETURN_ROW_END)
402
else if (ret != DRIZZLE_RETURN_OK)
403
CLIENT_ERROR("drizzle_field_read", ret, client);
408
printf("%.*s", (int32_t)length, field);
410
printf(" (%zd) %.*s", total, (int32_t)length, field);
412
if (offset + length == total)
415
/* If we buffered the entire field, be sure to free it. */
416
if (client->level == BUFFER_FIELD)
417
drizzle_field_free(field);
423
else if (client->level == BUFFER_ROW)
425
/* The entire row will be buffered here, so no need to worry about
427
row = drizzle_row_buffer(&(client_con->result), &ret);
428
if (ret == DRIZZLE_RETURN_IO_WAIT)
430
else if (ret != DRIZZLE_RETURN_OK)
431
CLIENT_ERROR("drizzle_row", ret, client);
433
/* This marks the end of rows. */
437
field_sizes= drizzle_row_field_sizes(&(client_con->result));
439
printf("Row: %" PRId64 "\n",
440
drizzle_row_current(&(client_con->result)));
442
for (x= 0; x < drizzle_result_column_count(&(client_con->result)); x++)
448
printf(" (%zd) %.*s\n", field_sizes[x], (int32_t)field_sizes[x],
453
drizzle_row_free(&(client_con->result), row);
458
drizzle_result_free(&(client_con->result));
462
/* This should be impossible. */
469
void con_info(drizzle_con_st *con)
471
printf("Connected: protocol_version=%u\n"
477
drizzle_con_protocol_version(con), drizzle_con_server_version(con),
478
drizzle_con_thread_id(con), drizzle_con_capabilities(con),
479
drizzle_con_charset(con), drizzle_con_status(con));
482
void result_info(drizzle_result_st *result)
484
printf("Result: row_count=%" PRId64 "\n"
485
" insert_id=%" PRId64 "\n"
486
" warning_count=%u\n"
488
" affected_rows=%" PRId64 "\n\n",
489
drizzle_result_row_count(result),
490
drizzle_result_insert_id(result),
491
drizzle_result_warning_count(result),
492
drizzle_result_column_count(result),
493
drizzle_result_affected_rows(result));
496
void column_info(drizzle_column_st *column)
498
printf("Field: catalog=%s\n"
509
drizzle_column_catalog(column), drizzle_column_db(column),
510
drizzle_column_table(column), drizzle_column_orig_table(column),
511
drizzle_column_name(column), drizzle_column_orig_name(column),
512
drizzle_column_charset(column), drizzle_column_size(column),
513
drizzle_column_max_size(column), drizzle_column_type(column),
514
drizzle_column_flags(column));