From d0594ed3ce1c2e49d5047c52f2f392937d7e408f Mon Sep 17 00:00:00 2001 From: yeoncheol-kim Date: Mon, 30 Oct 2023 13:58:06 +0900 Subject: [PATCH] CLEANUP: Detach ASCII/Binary protocols from memcached --- Makefile.am | 6 +- memcached.c | 12289 ++---------------------------------------------- memcached.h | 36 + proto_ascii.c | 7368 +++++++++++++++++++++++++++++ proto_ascii.h | 8 + proto_bin.c | 4313 +++++++++++++++++ proto_bin.h | 9 + 7 files changed, 12066 insertions(+), 11963 deletions(-) create mode 100644 proto_ascii.c create mode 100644 proto_ascii.h create mode 100644 proto_bin.c create mode 100644 proto_bin.h diff --git a/Makefile.am b/Makefile.am index 268efcb7a..3c28771f6 100644 --- a/Makefile.am +++ b/Makefile.am @@ -90,7 +90,11 @@ memcached_SOURCES = \ cmdlog.h \ lqdetect.c \ lqdetect.h \ - trace.h + trace.h \ + proto_ascii.c \ + proto_ascii.h \ + proto_bin.c \ + proto_bin.h memcached_LDFLAGS =-R '$(libdir)' memcached_CFLAGS = @PROFILER_FLAGS@ ${AM_CFLAGS} memcached_DEPENDENCIES = libmcd_util.la diff --git a/memcached.c b/memcached.c index 2493d25fc..0d66b0e90 100644 --- a/memcached.c +++ b/memcached.c @@ -38,12 +38,6 @@ #include "arcus_hb.h" #endif -#if defined(ENABLE_SASL) || defined(ENABLE_ISASL) -#define SASL_ENABLED -#endif - -#define ZK_CONNECTIONS 1 - #include #include #include @@ -61,6 +55,8 @@ #include #include +#include "proto_ascii.h" +#include "proto_bin.h" /* Lock for global stats */ static pthread_mutex_t stats_lock = PTHREAD_MUTEX_INITIALIZER; @@ -96,18 +92,22 @@ volatile sig_atomic_t memcached_shutdown=0; * sizeof(time_t) > sizeof(unsigned int). */ volatile rel_time_t current_time; -static time_t process_started; /* when the process was started */ +time_t process_started; /* when the process was started */ /** exported globals **/ struct settings settings; struct mc_stats mc_stats; EXTENSION_LOGGER_DESCRIPTOR *mc_logger; -static union { +union { ENGINE_HANDLE *v0; ENGINE_HANDLE_V1 *v1; } mc_engine; +void ritem_set_first(conn *c, int rtype, int vleng); +void perform_callbacks(ENGINE_EVENT_TYPE type, + const void *data, const void *c); + /** file scope variables **/ static conn *listen_conn = NULL; static struct event_base *main_base; @@ -117,15 +117,15 @@ topkeys_t *default_topkeys = NULL; static struct engine_event_handler *engine_event_handlers[MAX_ENGINE_EVENT_TYPE + 1]; #ifdef ENABLE_ZK_INTEGRATION -static char *arcus_zk_cfg = NULL; +char *arcus_zk_cfg = NULL; #endif #ifdef COMMAND_LOGGING -static bool cmdlog_in_use = false; +bool cmdlog_in_use = false; #endif #ifdef DETECT_LONG_QUERY -static bool lqdetect_in_use = false; +bool lqdetect_in_use = false; #endif /* @@ -144,15 +144,6 @@ enum try_read_result { static enum try_read_result try_read_network(conn *c); static enum try_read_result try_read_udp(conn *c); -/* stats */ -static void stats_init(void); -static void server_stats(ADD_STAT add_stats, conn *c, bool aggregate); -static void process_stats_settings(ADD_STAT add_stats, void *c); -#ifdef ENABLE_ZK_INTEGRATION -static void process_stats_zookeeper(ADD_STAT add_stats, void *c); -#endif -static void update_stat_cas(conn *c, ENGINE_ERROR_CODE ret); - /* defaults */ static void settings_init(void); @@ -160,12 +151,7 @@ static void settings_init(void); static void event_handler(const int fd, const short which, void *arg); static bool update_event(conn *c, const int new_flags); static void complete_nread(conn *c); -static int try_read_command_binary(conn *c); -static int try_read_command_ascii(conn *c); -static void write_and_free(conn *c, char *buf, int bytes); static int ensure_iov_space(conn *c); -static int add_iov(conn *c, const void *buf, int len); -static int add_msghdr(conn *c); enum transmit_result { TRANSMIT_COMPLETE, /** All done writing. */ @@ -187,7 +173,7 @@ static void set_current_time(void) current_time = (rel_time_t) (timer.tv_sec - process_started); } -static rel_time_t get_current_time(void) +rel_time_t get_current_time(void) { return current_time; } @@ -198,7 +184,7 @@ static rel_time_t get_current_time(void) * return unix time. Use the fact that delta can't exceed one month * (and real time value can't be that low). */ -static rel_time_t realtime(const time_t exptime) +rel_time_t realtime(const time_t exptime) { if (exptime == 0) return 0; /* 0 means never expire */ if (exptime < 0) { @@ -227,7 +213,7 @@ static rel_time_t realtime(const time_t exptime) * given time value is relative to current time. * return delta from current unix time. */ -static rel_time_t human_readable_time(const rel_time_t exptime) +rel_time_t human_readable_time(const rel_time_t exptime) { if (exptime == 0) { return exptime; @@ -245,7 +231,7 @@ static rel_time_t human_readable_time(const rel_time_t exptime) } /* grow the dynamic buffer of the given connection */ -static bool grow_dynamic_buffer(conn *c, size_t needed) +bool grow_dynamic_buffer(conn *c, size_t needed) { size_t nsize = c->dynamic_buffer.size; size_t available = nsize - c->dynamic_buffer.offset; @@ -283,7 +269,7 @@ static void disable_stats_detail(void) "Detailed stats internally disabled.\n"); } -static void stats_init(void) +void stats_init(void) { mc_stats.daemon_conns = 0; mc_stats.rejected_conns = 0; @@ -298,7 +284,7 @@ static void stats_init(void) stats_prefix_init(settings.prefix_delimiter, disable_stats_detail); } -static void stats_reset(const void *cookie) +void stats_reset(const void *cookie) { LOCK_STATS(); mc_stats.rejected_conns = 0; @@ -371,7 +357,7 @@ static void settings_init(void) * * Returns 0 on success, -1 on out-of-memory. */ -static int add_msghdr(conn *c) +int add_msghdr(conn *c) { assert(c != NULL); struct msghdr *msg; @@ -453,7 +439,7 @@ static void register_callback(ENGINE_HANDLE *eh, } // Perform all callbacks of a given type for the given connection. -static void perform_callbacks(ENGINE_EVENT_TYPE type, +void perform_callbacks(ENGINE_EVENT_TYPE type, const void *data, const void *c) { for (struct engine_event_handler *h = engine_event_handlers[type]; @@ -1035,7 +1021,7 @@ static void conn_shrink(conn *c) } } -static void ritem_set_first(conn *c, int rtype, int vleng) +void ritem_set_first(conn *c, int rtype, int vleng) { c->rtype = rtype; @@ -1202,7 +1188,7 @@ static int ensure_iov_space(conn *c) * Returns 0 on success, -1 on out-of-memory. */ -static int add_iov(conn *c, const void *buf, int len) +int add_iov(conn *c, const void *buf, int len) { assert(c != NULL); struct msghdr *m; @@ -1251,186 +1237,10 @@ static int add_iov(conn *c, const void *buf, int len) return 0; } -static int add_iov_hinfo_value_all(conn *c, item_info *hinfo) -{ - if (hinfo->nvalue > 0) { - if (add_iov(c, hinfo->value, hinfo->nvalue) != 0) - return -1; - } - if (hinfo->naddnl > 0) { - value_item **addnl = hinfo->addnl; - for (int i = 0; i < hinfo->naddnl; i++) { - if (add_iov(c, addnl[i]->ptr, addnl[i]->len) != 0) - return -1; - } - } - return 0; -} - -static int add_iov_hinfo_value_some(conn *c, item_info *hinfo, int length) -{ - int i, iosize; - - if (hinfo->naddnl == 0) { - return add_iov(c, hinfo->value, length); - } - - /* hinfo->naddnl > 0 */ - if (hinfo->nvalue > 0) { - iosize = length < hinfo->nvalue - ? length : hinfo->nvalue; - if (add_iov(c, hinfo->value, iosize) != 0) - return -1; - length -= iosize; - } - for (i = 0; i < hinfo->naddnl && length > 0; i++) { - iosize = length < hinfo->addnl[i]->len - ? length : hinfo->addnl[i]->len; - if (add_iov(c, hinfo->addnl[i]->ptr, iosize) != 0) - return -1; - length -= iosize; - } - return 0; -} - -static int add_iov_einfo_value_all(conn *c, eitem_info *einfo) -{ - if (einfo->nvalue > 0) { - if (add_iov(c, einfo->value, einfo->nvalue) != 0) - return -1; - } - if (einfo->naddnl > 0) { - value_item **addnl = einfo->addnl; - for (int i = 0; i < einfo->naddnl; i++) { - if (add_iov(c, addnl[i]->ptr, addnl[i]->len) != 0) - return -1; - } - } - return 0; -} - -static int add_iov_einfo_value_some(conn *c, eitem_info *einfo, int length) -{ - int i, iosize; - - if (einfo->naddnl == 0) { - return add_iov(c, einfo->value, length); - } - - /* einfo->naddnl > 0 */ - if (einfo->nvalue > 0) { - iosize = length < einfo->nvalue - ? length : einfo->nvalue; - if (add_iov(c, einfo->value, iosize) != 0) - return -1; - length -= iosize; - } - for (i = 0; i < einfo->naddnl && length > 0; i++) { - iosize = length < einfo->addnl[i]->len - ? length : einfo->addnl[i]->len; - if (add_iov(c, einfo->addnl[i]->ptr, iosize) != 0) - return -1; - length -= iosize; - } - return 0; -} - -static int hinfo_check_ascii_tail_string(item_info *hinfo) -{ - if (hinfo->naddnl == 0) { - return memcmp((char*)hinfo->value + hinfo->nbytes - 2, "\r\n", 2); - } - - value_item *last = hinfo->addnl[hinfo->naddnl-1]; - if (last->len >= 2) { - return memcmp(last->ptr + last->len - 2, "\r\n", 2); - } - - /* last->len == 1 */ - if (memcmp(last->ptr + last->len - 1, "\n", 1) != 0) { - return -1; - } - if (hinfo->naddnl >= 2) { - last = hinfo->addnl[hinfo->naddnl-2]; - return memcmp(last->ptr + last->len - 1, "\r", 1); - } else { - return memcmp((char*)hinfo->value + hinfo->nvalue - 1, "\r", 1); - } -} - -static void hinfo_set_ascii_tail_string(item_info *hinfo) -{ - if (hinfo->naddnl == 0) { - memcpy((char*)hinfo->value + hinfo->nbytes - 2, "\r\n", 2); - return; - } - - value_item *last = hinfo->addnl[hinfo->naddnl-1]; - if (last->len >= 2) { - memcpy(last->ptr + last->len - 2, "\r\n", 2); - return; - } - - /* last->len == 1 */ - memcpy(last->ptr + last->len - 1, "\n", 1); - if (hinfo->naddnl >= 2) { - last = hinfo->addnl[hinfo->naddnl-2]; - memcpy(last->ptr + last->len - 1, "\r", 1); - } else { - memcpy((char*)hinfo->value + hinfo->nvalue - 1, "\r", 1); - } -} - -static int einfo_check_ascii_tail_string(eitem_info *einfo) -{ - if (einfo->naddnl == 0) { - return memcmp((char*)einfo->value + einfo->nbytes - 2, "\r\n", 2); - } - - value_item *last = einfo->addnl[einfo->naddnl-1]; - if (last->len >= 2) { - return memcmp(last->ptr + last->len - 2, "\r\n", 2); - } - - /* last->len == 1 */ - if (memcmp(last->ptr + last->len - 1, "\n", 1) != 0) { - return -1; - } - if (einfo->naddnl >= 2) { - last = einfo->addnl[einfo->naddnl-2]; - return memcmp(last->ptr + last->len - 1, "\r", 1); - } else { - return memcmp((char*)einfo->value + einfo->nvalue - 1, "\r", 1); - } -} - -static void einfo_set_ascii_tail_string(eitem_info *einfo) -{ - if (einfo->naddnl == 0) { - memcpy((char*)einfo->value + einfo->nbytes - 2, "\r\n", 2); - return; - } - - value_item *last = einfo->addnl[einfo->naddnl-1]; - if (last->len >= 2) { - memcpy(last->ptr + last->len - 2, "\r\n", 2); - return; - } - - /* last->len == 1 */ - memcpy(last->ptr + last->len - 1, "\n", 1); - if (einfo->naddnl >= 2) { - last = einfo->addnl[einfo->naddnl-2]; - memcpy(last->ptr + last->len - 1, "\r", 1); - } else { - memcpy((char*)einfo->value + einfo->nvalue - 1, "\r", 1); - } -} - /* * Constructs a set of UDP headers and attaches them to the outgoing messages. */ -static int build_udp_headers(conn *c) +int build_udp_headers(conn *c) { assert(c != NULL); unsigned char *hdr; @@ -1465,162 +1275,7 @@ static int build_udp_headers(conn *c) return 0; } -static void pipe_state_clear(conn *c) -{ - c->pipe_state = PIPE_STATE_OFF; - c->pipe_count = 0; -} - -static int pipe_response_save(conn *c, const char *str, size_t len) -{ - if (c->pipe_state == PIPE_STATE_ON) { - if (c->pipe_count == 0) { - /* skip the memory of response head string: "RESPONSE %d\r\n" */ - c->pipe_reslen = PIPE_RES_HEAD_SIZE; - c->pipe_resptr = &c->pipe_response[c->pipe_reslen]; - } - if ((c->pipe_reslen + (len+2)) < (PIPE_RES_MAX_SIZE-PIPE_RES_TAIL_SIZE)) { - sprintf(c->pipe_resptr, "%s\r\n", str); - c->pipe_reslen += (len+2); - c->pipe_resptr = &c->pipe_response[c->pipe_reslen]; - c->pipe_count++; - if (c->pipe_count >= PIPE_CMD_MAX_COUNT && c->noreply == true) { - /* c->noreply == true: There are remaining pipe operations. */ - c->pipe_state = PIPE_STATE_ERR_CFULL; /* pipe count overflow */ - return -1; - } - else if ((strncmp(str, "CLIENT_ERROR", 12) == 0) || - (strncmp(str, "SERVER_ERROR", 12) == 0) || - (strncmp(str, "ERROR", 5) == 0)) { /* severe error */ - c->pipe_state = PIPE_STATE_ERR_BAD; /* bad error in pipelining */ - return -1; - } - } else { - c->pipe_state = PIPE_STATE_ERR_MFULL; /* pipe memory overflow */ - return -1; - } - } else { - /* A response message has come here before pipe error is reset. - * Maybe, clients may not send all the commands of the pipelining. - * So, force to reset the current pipelining. - */ - mc_logger->log(EXTENSION_LOG_INFO, c, - "%d: response message before pipe error is reset. %s\n", - c->sfd, str); - pipe_state_clear(c); - } - return 0; -} - -static void pipe_response_done(conn *c, bool end_of_pipelining) -{ - char headbuf[PIPE_RES_HEAD_SIZE]; - int headlen; - int headidx; - - /* pipe response head string */ - headlen = sprintf(headbuf, "RESPONSE %d\r\n", c->pipe_count); - assert(headlen > 0 && headlen <= PIPE_RES_HEAD_SIZE); - headidx = PIPE_RES_HEAD_SIZE - headlen; - memcpy(&c->pipe_response[headidx], headbuf, headlen); - - /* pipe response tail string */ - if (c->pipe_state == PIPE_STATE_ON) { - sprintf(c->pipe_resptr, "END\r\n"); - c->pipe_reslen += 5; - - pipe_state_clear(c); - } else { - if (c->pipe_state == PIPE_STATE_ERR_CFULL) { - sprintf(c->pipe_resptr, "PIPE_ERROR command overflow\r\n"); - c->pipe_reslen += 29; - } else if (c->pipe_state == PIPE_STATE_ERR_MFULL) { - sprintf(c->pipe_resptr, "PIPE_ERROR memory overflow\r\n"); - c->pipe_reslen += 28; - } else { /* PIPE_STATE_ERR_BAD */ - sprintf(c->pipe_resptr, "PIPE_ERROR bad error\r\n"); - c->pipe_reslen += 22; - } - if (end_of_pipelining) { - pipe_state_clear(c); - } - /* The pipe_state will be cleared - * after swallowing the remaining data. - */ - } - - c->wbytes = c->pipe_reslen - headidx; - c->wcurr = &c->pipe_response[headidx]; - - conn_set_state(c, conn_write); - c->write_and_go = conn_new_cmd; -} - -static void out_string(conn *c, const char *str) -{ - assert(c != NULL); - size_t len = strlen(str); - bool original_noreply = c->noreply; - - if (settings.verbose > 1) { - if (c->noreply) - mc_logger->log(EXTENSION_LOG_DEBUG, c, ">%d NOREPLY %s\n", c->sfd, str); - else - mc_logger->log(EXTENSION_LOG_DEBUG, c, ">%d %s\n", c->sfd, str); - } - - if (c->pipe_state != PIPE_STATE_OFF) { - if (pipe_response_save(c, str, len) < 0) { /* PIPE_STATE_ERR.. */ - c->noreply = false; /* stop pipelining */ - } - } - - if (c->noreply) { - c->noreply = false; - /* Clear the ewouldblock so that the next read command from - * the same connection does not falsely block and time out. - * - * It's better not to set the ewouldblock if noreply exists - * when write operations are performed. - */ - if (c->ewouldblock) { - c->ewouldblock = false; -#ifdef MULTI_NOTIFY_IO_COMPLETE - mc_logger->log(EXTENSION_LOG_WARNING, c, - "[FATAL] Unexpected ewouldblock in noreply processing.\n"); -#endif - } - conn_set_state(c, conn_new_cmd); - return; - } - - /* Nuke a partial output... */ - c->msgcurr = 0; - c->msgused = 0; - c->iovused = 0; - add_msghdr(c); - - if (c->pipe_state != PIPE_STATE_OFF) { - pipe_response_done(c, !original_noreply); - return; - } - - if ((len + 2) > c->wsize) { - /* ought to be always enough. just fail for simplicity */ - str = "SERVER_ERROR output line too long"; - len = strlen(str); - } - - memcpy(c->wbuf, str, len); - memcpy(c->wbuf + len, "\r\n", 2); - c->wbytes = len + 2; - c->wcurr = c->wbuf; - - conn_set_state(c, conn_write); - c->write_and_go = conn_new_cmd; -} - -static inline char *get_item_type_str(uint8_t type) +inline char *get_item_type_str(uint8_t type) { if (type == ITEM_TYPE_KV) return "kv"; else if (type == ITEM_TYPE_LIST) return "list"; @@ -1630,7 +1285,7 @@ static inline char *get_item_type_str(uint8_t type) else return "unknown"; } -static inline char get_item_type_char(uint8_t type) +inline char get_item_type_char(uint8_t type) { if (type == ITEM_TYPE_KV) return 'K'; else if (type == ITEM_TYPE_LIST) return 'L'; @@ -1640,7 +1295,7 @@ static inline char get_item_type_char(uint8_t type) else return 'A'; } -static inline char *get_ovflaction_str(uint8_t ovflact) +inline char *get_ovflaction_str(uint8_t ovflact) { if (ovflact == OVFL_HEAD_TRIM) return "head_trim"; else if (ovflact == OVFL_TAIL_TRIM) return "tail_trim"; @@ -1652,11665 +1307,375 @@ static inline char *get_ovflaction_str(uint8_t ovflact) else return "unknown"; } -static void -handle_unexpected_errorcode_ascii(conn *c, const char *func_name, ENGINE_ERROR_CODE ret) +static void reset_cmd_handler(conn *c) { - if (ret == ENGINE_DISCONNECT) { - conn_set_state(c, conn_closing); - } else if (ret == ENGINE_ENOTSUP) { - out_string(c, "NOT_SUPPORTED"); - } else { - mc_logger->log(EXTENSION_LOG_WARNING, c, "[%s] Unexpected Error: %d\n", - func_name, (int)ret); - out_string(c, "SERVER_ERROR internal"); + c->ascii_cmd = NULL; + c->cmd = -1; + c->substate = bin_no_state; + if (c->item != NULL) { + mc_engine.v1->release(mc_engine.v0, c, c->item); + c->item = NULL; } -} - -/* - * we get here after reading the value in set/add/replace commands. The command - * has been stored in c->cmd, and the item is ready in c->item. - */ -static void process_lop_insert_complete(conn *c) -{ - assert(c->coll_op == OPERATION_LOP_INSERT); - assert(c->coll_eitem != NULL); - ENGINE_ERROR_CODE ret; - - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_LIST, c->coll_eitem, &c->einfo); - - if (einfo_check_ascii_tail_string(&c->einfo) != 0) { /* check "\r\n" */ - ret = ENGINE_EINVAL; - out_string(c, "CLIENT_ERROR bad data chunk"); - } else { - bool created; - ret = mc_engine.v1->list_elem_insert(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_index, c->coll_eitem, - c->coll_attrp, &created, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_lop_insert(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } #ifdef DETECT_LONG_QUERY - if (lqdetect_in_use && ret == ENGINE_SUCCESS) { - if (! lqdetect_lop_insert(c->client_ip, c->coll_key, c->coll_index)) { - lqdetect_in_use = false; - } - } + if (c->lq_result) { + lqdetect_result_release(c->lq_result); + c->lq_result = NULL; + } #endif - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, lop_insert, c->coll_key, c->coll_nkey); - if (created) out_string(c, "CREATED_STORED"); - else out_string(c, "STORED"); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, lop_insert, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND"); - break; - default: - STATS_CMD_NOKEY(c, lop_insert); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EOVERFLOW) out_string(c, "OVERFLOWED"); - else if (ret == ENGINE_EINDEXOOR) out_string(c, "OUT_OF_RANGE"); - else if (ret == ENGINE_PREFIX_ENAME) out_string(c, "CLIENT_ERROR invalid prefix name"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } + if (c->coll_eitem != NULL) { + conn_coll_eitem_free(c); + c->coll_eitem = NULL; } - - if (ret != ENGINE_SUCCESS) { - mc_engine.v1->list_elem_free(mc_engine.v0, c, c->coll_eitem); + if (c->coll_strkeys != NULL) { + assert(c->coll_strkeys == (void*)&c->memblist); + mblck_list_free(&c->thread->mblck_pool, &c->memblist); + c->coll_strkeys = NULL; } - c->coll_eitem = NULL; -} - -static void process_sop_insert_complete(conn *c) -{ - assert(c->coll_op == OPERATION_SOP_INSERT); - assert(c->coll_eitem != NULL); - ENGINE_ERROR_CODE ret; - - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_SET, c->coll_eitem, &c->einfo); - - if (einfo_check_ascii_tail_string(&c->einfo) != 0) { /* check "\r\n" */ - ret = ENGINE_EINVAL; - out_string(c, "CLIENT_ERROR bad data chunk"); + conn_shrink(c); + if (c->rbytes > 0) { + conn_set_state(c, conn_parse_cmd); } else { - bool created; - ret = mc_engine.v1->set_elem_insert(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_eitem, c->coll_attrp, &created, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_sop_insert(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, sop_insert, c->coll_key, c->coll_nkey); - if (created) out_string(c, "CREATED_STORED"); - else out_string(c, "STORED"); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, sop_insert, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND"); - break; - default: - STATS_CMD_NOKEY(c, sop_insert); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EOVERFLOW) out_string(c, "OVERFLOWED"); - else if (ret == ENGINE_ELEM_EEXISTS) out_string(c, "ELEMENT_EXISTS"); - else if (ret == ENGINE_PREFIX_ENAME) out_string(c, "CLIENT_ERROR invalid prefix name"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - } - - if (ret != ENGINE_SUCCESS) { - mc_engine.v1->set_elem_free(mc_engine.v0, c, c->coll_eitem); + conn_set_state(c, conn_waiting); } - c->coll_eitem = NULL; } -static void process_sop_delete_complete(conn *c) +static void complete_nread(conn *c) { - assert(c->coll_op == OPERATION_SOP_DELETE); - assert(c->coll_eitem != NULL); - value_item *value = (value_item *)c->coll_eitem; + assert(c != NULL); + assert(c->protocol == ascii_prot || c->protocol == binary_prot); - if (strncmp(&value->ptr[value->len-2], "\r\n", 2) != 0) { - out_string(c, "CLIENT_ERROR bad data chunk"); + if (c->protocol == ascii_prot) { + complete_nread_ascii(c); } else { - bool dropped; - ENGINE_ERROR_CODE ret; - - ret = mc_engine.v1->set_elem_delete(mc_engine.v0, c, c->coll_key, c->coll_nkey, - value->ptr, value->len, c->coll_drop, - &dropped, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_sop_delete(c->coll_key, c->coll_nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_ELEM_HITS(c, sop_delete, c->coll_key, c->coll_nkey); - if (dropped) out_string(c, "DELETED_DROPPED"); - else out_string(c, "DELETED"); - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, sop_delete, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND_ELEMENT"); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, sop_delete, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND"); - break; - default: - STATS_CMD_NOKEY(c, sop_delete); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } + complete_nread_binary(c); } - - free(c->coll_eitem); - c->coll_eitem = NULL; } -static void process_sop_exist_complete(conn *c) +/* set up a connection to write a buffer then free it, used for stats */ +void write_and_free(conn *c, char *buf, int bytes) { - assert(c->coll_op == OPERATION_SOP_EXIST); - assert(c->coll_eitem != NULL); - value_item *value = (value_item *)c->coll_eitem; - - if (strncmp(&value->ptr[value->len-2], "\r\n", 2) != 0) { - out_string(c, "CLIENT_ERROR bad data chunk"); + if (buf) { + c->write_and_free = buf; + c->wcurr = buf; + c->wbytes = bytes; + conn_set_state(c, conn_write); + c->write_and_go = conn_new_cmd; } else { - bool exist; - ENGINE_ERROR_CODE ret; - - ret = mc_engine.v1->set_elem_exist(mc_engine.v0, c, c->coll_key, c->coll_nkey, - value->ptr, value->len, &exist, 0); - if (settings.detail_enabled) { - stats_prefix_record_sop_exist(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, sop_exist, c->coll_key, c->coll_nkey); - if (exist) out_string(c, "EXIST"); - else out_string(c, "NOT_EXIST"); - break; - case ENGINE_KEY_ENOENT: - case ENGINE_UNREADABLE: - STATS_MISSES(c, sop_exist, c->coll_key, c->coll_nkey); - if (ret == ENGINE_KEY_ENOENT) out_string(c, "NOT_FOUND"); - else out_string(c, "UNREADABLE"); - break; - default: - STATS_CMD_NOKEY(c, sop_exist); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } + out_string(c, "SERVER_ERROR out of memory writing stats"); } - - free(c->coll_eitem); - c->coll_eitem = NULL; } -static int make_mop_elem_response(char *bufptr, eitem_info *einfo) +void append_stat(const char *name, ADD_STAT add_stats, conn *c, + const char *fmt, ...) { - char *tmpptr = bufptr; + char val_str[STAT_VAL_LEN]; + int vlen; + va_list ap; - /* field */ - assert(einfo->nscore > 0); - memcpy(tmpptr, einfo->score, einfo->nscore); - tmpptr += (int)einfo->nscore; + assert(name); + assert(add_stats); + assert(c); + assert(fmt); - /* nbytes */ - sprintf(tmpptr, " %u ", einfo->nbytes-2); - tmpptr += strlen(tmpptr); + va_start(ap, fmt); + vlen = vsnprintf(val_str, sizeof(val_str) - 1, fmt, ap); + va_end(ap); - return (int)(tmpptr - bufptr); + add_stats(name, strlen(name), val_str, vlen, c); } -static void process_mop_insert_complete(conn *c) +void aggregate_callback(void *in, void *out) { - assert(c->coll_op == OPERATION_MOP_INSERT); - assert(c->coll_eitem != NULL); - ENGINE_ERROR_CODE ret; - - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_MAP, c->coll_eitem, &c->einfo); - - /* copy the field string into the element item. */ - memcpy((void*)c->einfo.score, c->coll_field.value, c->coll_field.length); - - if (einfo_check_ascii_tail_string(&c->einfo) != 0) { /* check "\r\n" */ - ret = ENGINE_EINVAL; - out_string(c, "CLIENT_ERROR bad data chunk"); - } else { - bool created; - ret = mc_engine.v1->map_elem_insert(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_eitem, c->coll_attrp, &created, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_mop_insert(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, mop_insert, c->coll_key, c->coll_nkey); - if (created) out_string(c, "CREATED_STORED"); - else out_string(c, "STORED"); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, mop_insert, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND"); - break; - default: - STATS_CMD_NOKEY(c, mop_insert); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EOVERFLOW) out_string(c, "OVERFLOWED"); - else if (ret == ENGINE_ELEM_EEXISTS) out_string(c, "ELEMENT_EXISTS"); - else if (ret == ENGINE_PREFIX_ENAME) out_string(c, "CLIENT_ERROR invalid prefix name"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - } - - if (ret != ENGINE_SUCCESS) { - mc_engine.v1->map_elem_free(mc_engine.v0, c, c->coll_eitem); - } - c->coll_eitem = NULL; -} - -static void process_mop_update_complete(conn *c) -{ - assert(c->coll_op == OPERATION_MOP_UPDATE); - assert(c->coll_eitem != NULL); - value_item *value = (value_item *)c->coll_eitem; - - if (strncmp(&value->ptr[value->len-2], "\r\n", 2) != 0) { - out_string(c, "CLIENT_ERROR bad data chunk"); - } else { - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->map_elem_update(mc_engine.v0, c, c->coll_key, c->coll_nkey, - &c->coll_field, value->ptr, value->len, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_mop_update(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_ELEM_HITS(c, mop_update, c->coll_key, c->coll_nkey); - out_string(c, "UPDATED"); - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, mop_update, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND_ELEMENT"); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, mop_update, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND"); - break; - default: - STATS_CMD_NOKEY(c, mop_update); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - } - - free((void*)c->coll_eitem); - c->coll_eitem = NULL; + struct thread_stats *out_thread_stats = out; + struct thread_stats *in_thread_stats = in; + threadlocal_stats_aggregate(in_thread_stats, out_thread_stats); } -static void process_mop_delete_complete(conn *c) +/* return server specific stats only */ +void server_stats(ADD_STAT add_stats, conn *c, bool aggregate) { - assert(c->coll_op == OPERATION_MOP_DELETE); + pid_t pid = getpid(); + rel_time_t now = get_current_time(); - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - field_t *fld_tokens = NULL; - uint32_t del_count = 0; - bool dropped; + struct thread_stats thread_stats; + threadlocal_stats_clear(&thread_stats); - if (c->coll_strkeys != NULL) { - fld_tokens = (field_t*)token_buff_get(&c->thread->token_buff, c->coll_numkeys); - if (fld_tokens != NULL) { - bool must_backward_compatible = true; /* Must be backward compatible */ - ret = tokenize_sblocks(&c->memblist, c->coll_lenkeys, c->coll_numkeys, - MAX_FIELD_LENG, must_backward_compatible, (token_t*)fld_tokens); - /* ret : ENGINE_SUCCESS | ENGINE_EBADVALUE | ENGINE_ENOMEM */ - } else { - ret = ENGINE_ENOMEM; - } + if (aggregate && mc_engine.v1->aggregate_stats != NULL) { + mc_engine.v1->aggregate_stats(mc_engine.v0, (const void *)c, + aggregate_callback, &thread_stats); + } else { + threadlocal_stats_aggregate(default_thread_stats, &thread_stats); } - if (ret == ENGINE_SUCCESS) { - ret = mc_engine.v1->map_elem_delete(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_numkeys, fld_tokens, c->coll_drop, - &del_count, &dropped, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_mop_delete(c->coll_key, c->coll_nkey, is_hit); - } -#ifdef DETECT_LONG_QUERY - if (lqdetect_in_use && ret == ENGINE_SUCCESS) { - if (! lqdetect_mop_delete(c->client_ip, c->coll_key, del_count, - c->coll_numkeys, c->coll_drop)) { - lqdetect_in_use = false; - } - } +#ifndef __WIN32__ + struct rusage usage; + getrusage(RUSAGE_SELF, &usage); #endif - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_ELEM_HITS(c, mop_delete, c->coll_key, c->coll_nkey); - if (dropped) out_string(c, "DELETED_DROPPED"); - else out_string(c, "DELETED"); - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, mop_delete, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND_ELEMENT"); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, mop_delete, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND"); - break; - default: - STATS_CMD_NOKEY(c, mop_delete); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EBADVALUE) out_string(c, "CLIENT_ERROR bad data chunk"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - - /* free key strings and tokens buffer */ - if (c->coll_strkeys != NULL) { - /* free token buffer */ - if (fld_tokens != NULL) { - token_buff_release(&c->thread->token_buff, fld_tokens); - } - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } -} - -static ENGINE_ERROR_CODE -out_mop_get_response(conn *c, bool delete, struct elems_result *eresultp) -{ - eitem **elem_array = eresultp->elem_array; - uint32_t elem_count = eresultp->elem_count; - int bufsize; - int resplen; - char *respbuf; /* response string buffer */ - char *respptr; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - do { - /* allocate response string buffer */ - bufsize = ((2*UINT32_STR_LENG) + 30) /* response head and tail size */ - + (elem_count * ((MAX_FIELD_LENG+2) + (UINT32_STR_LENG+2))); /* response body size */ - respbuf = (char*)malloc(bufsize); - if (respbuf == NULL) { - ret = ENGINE_ENOMEM; break; - } - respptr = respbuf; - - /* make response head */ - sprintf(respptr, "VALUE %u %u\r\n", htonl(eresultp->flags), elem_count); - if (add_iov(c, respptr, strlen(respptr)) != 0) { - ret = ENGINE_ENOMEM; break; - } - respptr += strlen(respptr); - - /* make response body */ - for (int i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_MAP, - elem_array[i], &c->einfo); - resplen = make_mop_elem_response(respptr, &c->einfo); - if ((add_iov(c, respptr, resplen) != 0) || - (add_iov_einfo_value_all(c, &c->einfo) != 0)) - { - ret = ENGINE_ENOMEM; break; - } - respptr += resplen; - } - if (ret == ENGINE_ENOMEM) break; - - /* make response tail */ - if (delete) { - sprintf(respptr, "%s\r\n", (eresultp->dropped ? "DELETED_DROPPED" : "DELETED")); - } else { - sprintf(respptr, "END\r\n"); - } - if ((add_iov(c, respptr, strlen(respptr)) != 0) || - (IS_UDP(c->transport) && build_udp_headers(c) != 0)) { - ret = ENGINE_ENOMEM; break; - } - } while(0); - if (ret == ENGINE_SUCCESS) { - c->coll_eitem = (void *)elem_array; - c->coll_ecount = elem_count; - c->coll_resps = respbuf; - c->coll_op = OPERATION_MOP_GET; - conn_set_state(c, conn_mwrite); - c->msgcurr = 0; - } else { /* ENGINE_ENOMEM */ - mc_engine.v1->map_elem_release(mc_engine.v0, c, elem_array, elem_count); - if (elem_array) - free(elem_array); - if (respbuf) - free(respbuf); - } - return ret; -} - -static void process_mop_get_complete(conn *c) -{ - assert(c->coll_op == OPERATION_MOP_GET); - assert(c->ewouldblock == false); - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - struct elems_result eresult; - field_t *fld_tokens = NULL; - bool delete = c->coll_delete; - bool drop_if_empty = c->coll_drop; - - if (c->coll_strkeys != NULL) { - fld_tokens = (field_t*)token_buff_get(&c->thread->token_buff, c->coll_numkeys); - if (fld_tokens != NULL) { - bool must_backward_compatible = true; /* Must be backward compatible */ - ret = tokenize_sblocks(&c->memblist, c->coll_lenkeys, c->coll_numkeys, - MAX_FIELD_LENG, must_backward_compatible, (token_t*)fld_tokens); - /* ret : ENGINE_SUCCESS | ENGINE_EBADVALUE | ENGINE_ENOMEM */ - } else { - ret = ENGINE_ENOMEM; - } - } - - if (ret == ENGINE_SUCCESS) { - ret = mc_engine.v1->map_elem_get(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_numkeys, fld_tokens, delete, drop_if_empty, - &eresult, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_mop_get(c->coll_key, c->coll_nkey, is_hit); - } -#ifdef DETECT_LONG_QUERY - if (lqdetect_in_use && ret == ENGINE_SUCCESS) { - if (! lqdetect_mop_get(c->client_ip, c->coll_key, eresult.elem_count, - c->coll_numkeys, delete, drop_if_empty)) { - lqdetect_in_use = false; - } - } +#ifdef ENABLE_ZK_INTEGRATION + arcus_hb_stats hb_stats; + arcus_hb_get_stats(&hb_stats); #endif - } - - switch (ret) { - case ENGINE_SUCCESS: - ret = out_mop_get_response(c, delete, &eresult); - if (ret == ENGINE_SUCCESS) { - STATS_ELEM_HITS(c, mop_get, c->coll_key, c->coll_nkey); - } else { - STATS_CMD_NOKEY(c, mop_get); - out_string(c, "SERVER_ERROR out of memory writing get response"); - } - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, mop_get, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND_ELEMENT"); - break; - case ENGINE_KEY_ENOENT: - case ENGINE_UNREADABLE: - STATS_MISSES(c, mop_get, c->coll_key, c->coll_nkey); - if (ret == ENGINE_KEY_ENOENT) out_string(c, "NOT_FOUND"); - else out_string(c, "UNREADABLE"); - break; - default: - STATS_CMD_NOKEY(c, mop_get); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EBADVALUE) out_string(c, "CLIENT_ERROR bad data chunk"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - - /* free key strings and tokens buffer */ - if (c->coll_strkeys != NULL) { - /* free token buffer */ - if (fld_tokens != NULL) { - token_buff_release(&c->thread->token_buff, fld_tokens); - } - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } -} - -static int make_bop_elem_response(char *bufptr, eitem_info *einfo) -{ - char *tmpptr = bufptr; - - /* bkey */ - if (einfo->nscore > 0) { - memcpy(tmpptr, "0x", 2); tmpptr += 2; - safe_hexatostr(einfo->score, einfo->nscore, tmpptr); - tmpptr += strlen(tmpptr); - } else { - sprintf(tmpptr, "%"PRIu64"", *(uint64_t*)einfo->score); - tmpptr += strlen(tmpptr); - } - /* eflag */ - if (einfo->neflag > 0) { - memcpy(tmpptr, " 0x", 3); tmpptr += 3; - safe_hexatostr(einfo->eflag, einfo->neflag, tmpptr); - tmpptr += strlen(tmpptr); - } - /* nbytes */ - sprintf(tmpptr, " %u ", einfo->nbytes-2); - tmpptr += strlen(tmpptr); - - return (int)(tmpptr - bufptr); -} - -static ENGINE_ERROR_CODE -out_bop_trim_response(conn *c, void *elem, uint32_t flags) -{ - eitem **elem_array; - int bufsize; - int resplen; - char *respbuf; /* response string buffer */ - char *respptr; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - do { - /* allocate response string buffer */ - bufsize = sizeof(void*) /* an element pointer */ - + ((2*UINT32_STR_LENG) + 30) /* response head and tail size */ - + ((MAX_BKEY_LENG*2+2) + (MAX_EFLAG_LENG*2+2) + UINT32_STR_LENG+3); /* response body size */ - respbuf = (char*)malloc(bufsize); - if (respbuf == NULL) { - ret = ENGINE_ENOMEM; break; - } - respptr = respbuf; - - /* save the trimmed element */ - elem_array = (eitem **)respptr; - elem_array[0] = elem; - respptr = (char*)&elem_array[1]; - - /* make response */ - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, elem, &c->einfo); - sprintf(respptr, "VALUE %u 1\r\n", htonl(flags)); - resplen = strlen(respptr); - resplen += make_bop_elem_response(respptr + resplen, &c->einfo); - - if ((add_iov(c, respptr, resplen) != 0) || - (add_iov_einfo_value_all(c, &c->einfo) != 0) || - (add_iov(c, "TRIMMED\r\n", 9) != 0) || - (IS_UDP(c->transport) && build_udp_headers(c) != 0)) - { - ret = ENGINE_ENOMEM; break; - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - c->coll_eitem = elem_array; - c->coll_ecount = 1; - c->coll_resps = respbuf; - conn_set_state(c, conn_mwrite); - c->msgcurr = 0; - } else { /* ENGINE_ENOMEM */ - mc_engine.v1->btree_elem_release(mc_engine.v0, c, &elem, 1); - if (respbuf) - free(respbuf); - out_string(c, "SERVER_ERROR out of memory writing trim response"); - } - return ret; -} - -static void process_bop_insert_complete(conn *c) -{ - assert(c->coll_op == OPERATION_BOP_INSERT || - c->coll_op == OPERATION_BOP_UPSERT); - assert(c->coll_eitem != NULL); - - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, c->coll_eitem, &c->einfo); - - if (einfo_check_ascii_tail_string(&c->einfo) != 0) { /* check "\r\n" */ - /* release the btree element */ - mc_engine.v1->btree_elem_free(mc_engine.v0, c, c->coll_eitem); - c->coll_eitem = NULL; - out_string(c, "CLIENT_ERROR bad data chunk"); - } else { - eitem_result trim_result; // contain the info of an element trimmed by maxcount overflow - bool created; - bool replaced; - bool replace_if_exist = (c->coll_op == OPERATION_BOP_UPSERT ? true : false); - ENGINE_ERROR_CODE ret; - - ret = mc_engine.v1->btree_elem_insert(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_eitem, replace_if_exist, - c->coll_attrp, &replaced, &created, - (c->coll_getrim ? &trim_result : NULL), 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_bop_insert(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - /* release the btree element in advance since coll_eitem field is to be used, soon. */ - if (ret != ENGINE_SUCCESS) { - mc_engine.v1->btree_elem_free(mc_engine.v0, c, c->coll_eitem); - } - c->coll_eitem = NULL; - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, bop_insert, c->coll_key, c->coll_nkey); - if (c->coll_getrim && trim_result.elems != NULL) { /* getrim flag */ - assert(trim_result.count == 1); - (void)out_bop_trim_response(c, trim_result.elems, trim_result.flags); - } else { - /* no getrim flag or no trimmed element */ - if (replaced) { - out_string(c, "REPLACED"); - } else { - if (created) out_string(c, "CREATED_STORED"); - else out_string(c, "STORED"); - } - } - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, bop_insert, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND"); - break; - default: - STATS_CMD_NOKEY(c, bop_insert); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EBADBKEY) out_string(c, "BKEY_MISMATCH"); - else if (ret == ENGINE_EOVERFLOW) out_string(c, "OVERFLOWED"); - else if (ret == ENGINE_EBKEYOOR) out_string(c, "OUT_OF_RANGE"); - else if (ret == ENGINE_ELEM_EEXISTS) out_string(c, "ELEMENT_EXISTS"); - else if (ret == ENGINE_PREFIX_ENAME) out_string(c, "CLIENT_ERROR invalid prefix name"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - } -} - -static void process_bop_update_complete(conn *c) -{ - assert(c->coll_op == OPERATION_BOP_UPDATE); - assert(c->ewouldblock == false); - char *new_value = NULL; - int new_nbytes = 0; - - if (c->coll_eitem != NULL) { - value_item *value = (value_item *)c->coll_eitem; - if (strncmp(&value->ptr[value->len-2], "\r\n", 2) != 0) { - out_string(c, "CLIENT_ERROR bad data chunk"); - free((void*)c->coll_eitem); - c->coll_eitem = NULL; - return; - } - new_value = value->ptr; - new_nbytes = value->len; - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->btree_elem_update(mc_engine.v0, c, c->coll_key, c->coll_nkey, - &c->coll_bkrange, - (c->coll_eupdate.neflag == EFLAG_NULL ? NULL : &c->coll_eupdate), - new_value, new_nbytes, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_bop_update(c->coll_key, c->coll_nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_ELEM_HITS(c, bop_update, c->coll_key, c->coll_nkey); - out_string(c, "UPDATED"); - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, bop_update, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND_ELEMENT"); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, bop_update, c->coll_key, c->coll_nkey); - out_string(c, "NOT_FOUND"); - break; - default: - STATS_CMD_NOKEY(c, bop_update); - if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EBADBKEY) out_string(c, "BKEY_MISMATCH"); - else if (ret == ENGINE_EBADEFLAG) out_string(c, "EFLAG_MISMATCH"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - - if (c->coll_eitem != NULL) { - free((void*)c->coll_eitem); - c->coll_eitem = NULL; - } -} - -#ifdef SUPPORT_BOP_MGET -static ENGINE_ERROR_CODE -process_bop_mget_single(conn *c, const char *key, size_t nkey, - struct elems_result *eresultp, - char *resultbuf, int *resultlen) -{ - char *respptr = resultbuf; - int resplen; - ENGINE_ERROR_CODE ret; - - ret = mc_engine.v1->btree_elem_get(mc_engine.v0, c, key, nkey, - &c->coll_bkrange, - (c->coll_efilter.ncompval==0 ? NULL : &c->coll_efilter), - c->coll_roffset, c->coll_rcount, false, false, - eresultp, 0); - /* The read-only operation do not return ENGINE_EWOULDBLOCK */ - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_bop_get(key, nkey, is_hit); - } - - if (ret == ENGINE_SUCCESS) { - do { - /* format : VALUE \r\n */ - sprintf(respptr, " %s %u %u\r\n", - (eresultp->trimmed ? "TRIMMED" : "OK"), - htonl(eresultp->flags), eresultp->elem_count); - resplen = strlen(respptr); - - if ((add_iov(c, "VALUE ", 6) != 0) || - (add_iov(c, key, nkey) != 0) || - (add_iov(c, respptr, resplen) != 0)) { - ret = ENGINE_ENOMEM; break; - } - respptr += resplen; - - for (int e = 0; e < eresultp->elem_count; e++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - eresultp->elem_array[e], &c->einfo); - sprintf(respptr, "ELEMENT "); - resplen = strlen(respptr); - resplen += make_bop_elem_response(respptr + resplen, &c->einfo); - - if ((add_iov(c, respptr, resplen) != 0) || - (add_iov_einfo_value_all(c, &c->einfo) != 0)) { - ret = ENGINE_ENOMEM; break; - } - respptr += resplen; - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - STATS_ELEM_HITS(c, bop_get, key, nkey); - } else { /* ret == ENGINE_ENOMEM */ - STATS_CMD_NOKEY(c, bop_get); - mc_engine.v1->btree_elem_release(mc_engine.v0, c, - eresultp->elem_array, - eresultp->elem_count); - free(eresultp->elem_array); - eresultp->elem_array = NULL; - eresultp->elem_count = 0; - } - } else { - if (ret == ENGINE_ELEM_ENOENT) { - STATS_NONE_HITS(c, bop_get, key, nkey); - sprintf(respptr, " %s\r\n", "NOT_FOUND_ELEMENT"); - ret = ENGINE_SUCCESS; - } - else if (ret == ENGINE_KEY_ENOENT || ret == ENGINE_EBKEYOOR || ret == ENGINE_UNREADABLE) { - STATS_MISSES(c, bop_get, key, nkey); - if (ret == ENGINE_KEY_ENOENT) sprintf(respptr, " %s\r\n", "NOT_FOUND"); - else if (ret == ENGINE_EBKEYOOR) sprintf(respptr, " %s\r\n", "OUT_OF_RANGE"); - else sprintf(respptr, " %s\r\n", "UNREADABLE"); - ret = ENGINE_SUCCESS; - } - else if (ret == ENGINE_EBADTYPE || ret == ENGINE_EBADBKEY) { - STATS_CMD_NOKEY(c, bop_get); - if (ret == ENGINE_EBADTYPE) sprintf(respptr, " %s\r\n", "TYPE_MISMATCH"); - else sprintf(respptr, " %s\r\n", "BKEY_MISMATCH"); - ret = ENGINE_SUCCESS; - } - else { - /* ENGINE_ENOMEM or ENGINE_DISCONNECT or SERVER error */ - STATS_CMD_NOKEY(c, bop_get); - } - if (ret == ENGINE_SUCCESS) { - resplen = strlen(respptr); - if ((add_iov(c, "VALUE ", 6) != 0) || - (add_iov(c, key, nkey) != 0) || - (add_iov(c, respptr, resplen) != 0)) { - ret = ENGINE_ENOMEM; - } - respptr += resplen; - } - } - if (ret == ENGINE_SUCCESS) { - *resultlen = (respptr - resultbuf); - } - return ret; -} -static void process_bop_mget_complete(conn *c) -{ - assert(c->coll_op == OPERATION_BOP_MGET); - assert(c->coll_eitem != NULL); - - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - struct elems_result *eresult = (struct elems_result *)c->coll_eitem; - char *resultptr = (char*)eresult + (c->coll_numkeys * sizeof(struct elems_result)); - int k,resultlen; - token_t *key_tokens; - - key_tokens = (token_t*)token_buff_get(&c->thread->token_buff, c->coll_numkeys); - if (key_tokens != NULL) { - bool must_backward_compatible = true; /* Must be backward compatible */ - ret = tokenize_sblocks(&c->memblist, c->coll_lenkeys, c->coll_numkeys, - KEY_MAX_LENGTH, must_backward_compatible, key_tokens); - /* ret : ENGINE_SUCCESS | ENGINE_EBADVALUE | ENGINE_ENOMEM */ - } else { - ret = ENGINE_ENOMEM; - } - if (ret == ENGINE_SUCCESS) { - if (c->coll_bkrange.to_nbkey == BKEY_NULL) { - memcpy(c->coll_bkrange.to_bkey, c->coll_bkrange.from_bkey, - (c->coll_bkrange.from_nbkey==0 ? sizeof(uint64_t) : c->coll_bkrange.from_nbkey)); - c->coll_bkrange.to_nbkey = c->coll_bkrange.from_nbkey; - } - for (k = 0; k < c->coll_numkeys; k++) { - ret = process_bop_mget_single(c, key_tokens[k].value, key_tokens[k].length, - &eresult[k], resultptr, &resultlen); - if (ret != ENGINE_SUCCESS) { - break; /* ENGINE_ENOMEM or ENGINE_DISCONNECT or SEVERE error */ - } - resultptr += resultlen; - } - if (k == c->coll_numkeys) { - sprintf(resultptr, "END\r\n"); - if ((add_iov(c, resultptr, strlen(resultptr)) != 0) || - (IS_UDP(c->transport) && build_udp_headers(c) != 0)) { - ret = ENGINE_ENOMEM; - } - } - if (ret != ENGINE_SUCCESS) { /* release elements */ - int bop_get_count = k; - for (k = 0; k < bop_get_count; k++) { - /* Individual bop get might not get elements. - * So, we check the element array is not empty. - */ - if (eresult[k].elem_array != NULL) { - mc_engine.v1->btree_elem_release(mc_engine.v0, c, - eresult[k].elem_array, - eresult[k].elem_count); - free(eresult[k].elem_array); - eresult[k].elem_array = NULL; - eresult[k].elem_count = 0; - } - } - } - } + LOCK_STATS(); - switch (ret) { - case ENGINE_SUCCESS: - STATS_OKS_NOKEY(c, bop_mget); - /* Remember this command so we can garbage collect it later */ - /* c->coll_eitem = (void *)elem_array; */ - c->coll_ecount = 0; - for (k = 0; k < c->coll_numkeys; k++) { - if (eresult[k].elem_array != NULL) { - c->coll_ecount += eresult[k].elem_count; - } - } - c->coll_op = OPERATION_BOP_MGET; - conn_set_state(c, conn_mwrite); - c->msgcurr = 0; - break; - default: - STATS_CMD_NOKEY(c, bop_mget); - if (ret == ENGINE_EBADVALUE) out_string(c, "CLIENT_ERROR bad data chunk"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - - /* free token buffer */ - if (key_tokens != NULL) { - token_buff_release(&c->thread->token_buff, key_tokens); - } - if (ret != ENGINE_SUCCESS) { - if (c->coll_strkeys != NULL) { - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } - if (c->coll_eitem != NULL) { - free((void *)c->coll_eitem); - c->coll_eitem = NULL; - } - } -} + APPEND_STAT("pid", "%lu", (long)pid); + APPEND_STAT("uptime", "%u", now); + APPEND_STAT("time", "%ld", now + (long)process_started); + APPEND_STAT("version", "%s", VERSION); + APPEND_STAT("libevent", "%s", event_get_version()); + APPEND_STAT("pointer_size", "%d", (int)(8 * sizeof(void *))); +#ifdef ENABLE_ZK_INTEGRATION + APPEND_STAT("hb_count", "%"PRIu64, hb_stats.count); + APPEND_STAT("hb_latency", "%"PRIu64, hb_stats.latency); #endif -#ifdef SUPPORT_BOP_SMGET -#ifdef JHPARK_OLD_SMGET_INTERFACE -static ENGINE_ERROR_CODE -out_bop_smget_old_response(conn *c, token_t *key_tokens, - eitem **elem_array, uint32_t *kfnd_array, - uint32_t *flag_array, uint32_t *kmis_array, - uint32_t elem_count, uint32_t kmis_count, - bool trimmed, bool duplicated) -{ - char *respptr = ((char*)kmis_array + (c->coll_numkeys * sizeof(uint32_t))); - int resplen; - int i, kidx; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - do { - sprintf(respptr, "VALUE %u\r\n", elem_count); - if (add_iov(c, respptr, strlen(respptr)) != 0) { - ret = ENGINE_ENOMEM; break; - } - respptr += strlen(respptr); - - for (i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - elem_array[i], &c->einfo); - sprintf(respptr, " %u ", htonl(flag_array[i])); /* flags */ - resplen = strlen(respptr); - resplen += make_bop_elem_response(respptr + resplen, &c->einfo); - kidx = kfnd_array[i]; - if ((add_iov(c, key_tokens[kidx].value, key_tokens[kidx].length) != 0) || - (add_iov(c, respptr, resplen) != 0) || - (add_iov_einfo_value_all(c, &c->einfo) != 0)) - { - ret = ENGINE_ENOMEM; break; - } - respptr += resplen; - } - if (ret == ENGINE_ENOMEM) break; +#ifndef __WIN32__ + append_stat("rusage_user", add_stats, c, "%ld.%06ld", + (long)usage.ru_utime.tv_sec, + (long)usage.ru_utime.tv_usec); + append_stat("rusage_system", add_stats, c, "%ld.%06ld", + (long)usage.ru_stime.tv_sec, + (long)usage.ru_stime.tv_usec); +#endif - sprintf(respptr, "MISSED_KEYS %u\r\n", kmis_count); - if (add_iov(c, respptr, strlen(respptr)) != 0) { - ret = ENGINE_ENOMEM; break; - } - respptr += strlen(respptr); - - if (kmis_count > 0) { - for (i = 0; i < kmis_count; i++) { - /* the last key string does not have delimiter character */ - kidx = kmis_array[i]; - if ((add_iov(c, key_tokens[kidx].value, key_tokens[kidx].length) != 0) || - (add_iov(c, "\r\n", 2) != 0)) { - ret = ENGINE_ENOMEM; break; - } - } - if (ret == ENGINE_ENOMEM) break; - } - - if (trimmed == true) { - sprintf(respptr, (duplicated ? "DUPLICATED_TRIMMED\r\n" : "TRIMMED\r\n")); - } else { - sprintf(respptr, (duplicated ? "DUPLICATED\r\n" : "END\r\n")); - } - if ((add_iov(c, respptr, strlen(respptr)) != 0) || - (IS_UDP(c->transport) && build_udp_headers(c) != 0)) { - ret = ENGINE_ENOMEM; break; - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - /* Remember this command so we can garbage collect it later */ - /* c->coll_eitem = (void *)elem_array; */ - c->coll_ecount = elem_count; - c->coll_op = OPERATION_BOP_SMGET; - conn_set_state(c, conn_mwrite); - c->msgcurr = 0; - } else { - mc_engine.v1->btree_elem_release(mc_engine.v0, c, elem_array, elem_count); - out_string(c, "SERVER_ERROR out of memory writing get response"); - } - return ret; -} - -static void process_bop_smget_complete_old(conn *c) -{ - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - int smget_count = c->coll_roffset + c->coll_rcount; - token_t *key_tokens; - - eitem **elem_array = (eitem **)c->coll_eitem; - uint32_t *kfnd_array = (uint32_t*)((char*)elem_array + (smget_count*sizeof(eitem*))); - uint32_t *flag_array = (uint32_t*)((char*)kfnd_array + (smget_count*sizeof(uint32_t))); - uint32_t *kmis_array = (uint32_t*)((char*)flag_array + (smget_count*sizeof(uint32_t))); - uint32_t elem_count = 0; - uint32_t kmis_count = 0; - bool trimmed; - bool duplicated; - - key_tokens = (token_t*)token_buff_get(&c->thread->token_buff, c->coll_numkeys); - if (key_tokens != NULL) { - bool must_backward_compatible = true; /* Must be backward compatible */ - ret = tokenize_sblocks(&c->memblist, c->coll_lenkeys, c->coll_numkeys, - KEY_MAX_LENGTH, must_backward_compatible, key_tokens); - /* ret : ENGINE_SUCCESS | ENGINE_EBADVALUE | ENGINE_ENOMEM */ - } else { - ret = ENGINE_ENOMEM; - } - if (ret == ENGINE_SUCCESS) { - if (c->coll_bkrange.to_nbkey == BKEY_NULL) { - memcpy(c->coll_bkrange.to_bkey, c->coll_bkrange.from_bkey, - (c->coll_bkrange.from_nbkey==0 ? sizeof(uint64_t) : c->coll_bkrange.from_nbkey)); - c->coll_bkrange.to_nbkey = c->coll_bkrange.from_nbkey; - } - assert(c->coll_numkeys > 0); - assert(c->coll_rcount > 0); - assert((c->coll_roffset + c->coll_rcount) <= MAX_SMGET_REQ_COUNT); - ret = mc_engine.v1->btree_elem_smget_old(mc_engine.v0, c, key_tokens, c->coll_numkeys, - &c->coll_bkrange, - (c->coll_efilter.ncompval==0 ? NULL : &c->coll_efilter), - c->coll_roffset, c->coll_rcount, - elem_array, kfnd_array, flag_array, &elem_count, - kmis_array, &kmis_count, &trimmed, &duplicated, 0); - } - - switch (ret) { - case ENGINE_SUCCESS: - if (out_bop_smget_old_response(c, key_tokens, elem_array, - kfnd_array, flag_array, kmis_array, - elem_count, kmis_count, - trimmed, duplicated) == ENGINE_SUCCESS) { - STATS_OKS_NOKEY(c, bop_smget); - } else { - STATS_CMD_NOKEY(c, bop_smget); - } - break; - default: - STATS_CMD_NOKEY(c, bop_smget); - if (ret == ENGINE_EBADVALUE) out_string(c, "CLIENT_ERROR bad data chunk"); - else if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EBADBKEY) out_string(c, "BKEY_MISMATCH"); - else if (ret == ENGINE_EBKEYOOR) out_string(c, "OUT_OF_RANGE"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - - /* free token buffer */ - if (key_tokens != NULL) { - token_buff_release(&c->thread->token_buff, key_tokens); - } - - if (ret != ENGINE_SUCCESS) { - if (c->coll_strkeys != NULL) { - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } - if (c->coll_eitem != NULL) { - free((void *)c->coll_eitem); - c->coll_eitem = NULL; - } - } -} -#endif - -static char *get_smget_miss_response(int res) -{ - if (res == ENGINE_KEY_ENOENT) return " NOT_FOUND\r\n"; - else if (res == ENGINE_UNREADABLE) return " UNREADABLE\r\n"; - else if (res == ENGINE_EBKEYOOR) return " OUT_OF_RANGE\r\n"; - else return " UNKNOWN\r\n"; -} - -static int make_smget_trim_response(char *bufptr, eitem_info *einfo) -{ - char *tmpptr = bufptr; - - /* bkey */ - if (einfo->nscore > 0) { - memcpy(tmpptr, " 0x", 3); tmpptr += 3; - safe_hexatostr(einfo->score, einfo->nscore, tmpptr); - tmpptr += strlen(tmpptr); - } else { - sprintf(tmpptr, " %"PRIu64"", *(uint64_t*)einfo->score); - tmpptr += strlen(tmpptr); - } - sprintf(tmpptr, "\r\n"); - tmpptr += 2; - return (int)(tmpptr - bufptr); -} - -static ENGINE_ERROR_CODE -out_bop_smget_response(conn *c, token_t *key_tokens, smget_result_t *smresp) -{ - char *respptr = (char *)&smresp->miss_kinfo[c->coll_numkeys]; - int resplen; - int i, kidx; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - do { - /* Change smget response head string: VALUE => ELEMENTS. - * It makes incompatible with the clients of lower version. - */ - sprintf(respptr, "ELEMENTS %u\r\n", smresp->elem_count); - if (add_iov(c, respptr, strlen(respptr)) != 0) { - ret = ENGINE_ENOMEM; break; - } - respptr += strlen(respptr); - - for (i = 0; i < smresp->elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - smresp->elem_array[i], &c->einfo); - sprintf(respptr, " %u ", htonl(smresp->elem_kinfo[i].flag)); - resplen = strlen(respptr); - resplen += make_bop_elem_response(respptr + resplen, &c->einfo); - kidx = smresp->elem_kinfo[i].kidx; - if ((add_iov(c, key_tokens[kidx].value, key_tokens[kidx].length) != 0) || - (add_iov(c, respptr, resplen) != 0) || - (add_iov_einfo_value_all(c, &c->einfo) != 0)) - { - ret = ENGINE_ENOMEM; break; - } - respptr += resplen; - } - if (ret == ENGINE_ENOMEM) break; - - sprintf(respptr, "MISSED_KEYS %u\r\n", smresp->miss_count); - if (add_iov(c, respptr, strlen(respptr)) != 0) { - ret = ENGINE_ENOMEM; break; - } - respptr += strlen(respptr); - - if (smresp->miss_count > 0) { - char *str = NULL; - for (i = 0; i < smresp->miss_count; i++) { - /* the last key string does not have delimiter character */ - kidx = smresp->miss_kinfo[i].kidx; - str = get_smget_miss_response(smresp->miss_kinfo[i].code); - if ((add_iov(c, key_tokens[kidx].value, key_tokens[kidx].length) != 0) || - (add_iov(c, str, strlen(str)) != 0)) { - ret = ENGINE_ENOMEM; break; - } - } - if (ret == ENGINE_ENOMEM) break; - } - - sprintf(respptr, "TRIMMED_KEYS %u\r\n", smresp->trim_count); - if (add_iov(c, respptr, strlen(respptr)) != 0) { - ret = ENGINE_ENOMEM; break; - } - respptr += strlen(respptr); - - if (smresp->trim_count > 0) { - for (i = 0; i < smresp->trim_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - smresp->trim_elems[i], &c->einfo); - resplen = make_smget_trim_response(respptr, &c->einfo); - kidx = smresp->trim_kinfo[i].kidx; - if ((add_iov(c, key_tokens[kidx].value, key_tokens[kidx].length) != 0) || - (add_iov(c, respptr, resplen) != 0)) { - ret = ENGINE_ENOMEM; break; - } - respptr += resplen; - } - if (ret == ENGINE_ENOMEM) break; - } - - sprintf(respptr, (smresp->duplicated ? "DUPLICATED\r\n" : "END\r\n")); - if ((add_iov(c, respptr, strlen(respptr)) != 0) || - (IS_UDP(c->transport) && build_udp_headers(c) != 0)) { - ret = ENGINE_ENOMEM; break; - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - /* Remember this command so we can garbage collect it later */ - /* c->coll_eitem = (void *)elem_array; */ - c->coll_ecount = smresp->elem_count+smresp->trim_count; - c->coll_op = OPERATION_BOP_SMGET; - conn_set_state(c, conn_mwrite); - c->msgcurr = 0; - } else { - mc_engine.v1->btree_elem_release(mc_engine.v0, c, smresp->elem_array, - smresp->elem_count+smresp->trim_count); - out_string(c, "SERVER_ERROR out of memory writing get response"); - } - return ret; -} - -static void process_bop_smget_complete(conn *c) -{ - assert(c->coll_op == OPERATION_BOP_SMGET); - assert(c->coll_eitem != NULL); -#ifdef JHPARK_OLD_SMGET_INTERFACE - if (c->coll_smgmode == 0) { - process_bop_smget_complete_old(c); - return; - } -#endif - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - token_t *key_tokens; - smget_result_t smres; - - smres.elem_array = (eitem **)c->coll_eitem; - smres.elem_kinfo = (smget_ehit_t *)&smres.elem_array[c->coll_rcount+c->coll_numkeys]; - smres.miss_kinfo = (smget_emis_t *)&smres.elem_kinfo[c->coll_rcount]; - - key_tokens = (token_t*)token_buff_get(&c->thread->token_buff, c->coll_numkeys); - if (key_tokens != NULL) { - bool must_backward_compatible = true; /* Must be backward compatible */ - ret = tokenize_sblocks(&c->memblist, c->coll_lenkeys, c->coll_numkeys, - KEY_MAX_LENGTH, must_backward_compatible, key_tokens); - /* ret : ENGINE_SUCCESS | ENGINE_EBADVALUE | ENGINE_ENOMEM */ - } else { - ret = ENGINE_ENOMEM; - } - if (ret == ENGINE_SUCCESS) { - if (c->coll_bkrange.to_nbkey == BKEY_NULL) { - memcpy(c->coll_bkrange.to_bkey, c->coll_bkrange.from_bkey, - (c->coll_bkrange.from_nbkey==0 ? sizeof(uint64_t) : c->coll_bkrange.from_nbkey)); - c->coll_bkrange.to_nbkey = c->coll_bkrange.from_nbkey; - } - assert(c->coll_numkeys > 0); - assert(c->coll_rcount > 0); - assert((c->coll_roffset + c->coll_rcount) <= MAX_SMGET_REQ_COUNT); - ret = mc_engine.v1->btree_elem_smget(mc_engine.v0, c, key_tokens, c->coll_numkeys, - &c->coll_bkrange, - (c->coll_efilter.ncompval==0 ? NULL : &c->coll_efilter), - c->coll_roffset, c->coll_rcount, -#ifdef JHPARK_OLD_SMGET_INTERFACE - (c->coll_smgmode == 2 ? true : false), -#else - c->coll_unique, -#endif - &smres, 0); - } - - switch (ret) { - case ENGINE_SUCCESS: - if (out_bop_smget_response(c, key_tokens, &smres) == ENGINE_SUCCESS) { - STATS_OKS_NOKEY(c, bop_smget); - } else { - STATS_CMD_NOKEY(c, bop_smget); - } - break; - default: - STATS_CMD_NOKEY(c, bop_smget); - if (ret == ENGINE_EBADVALUE) out_string(c, "CLIENT_ERROR bad data chunk"); - else if (ret == ENGINE_EBADTYPE) out_string(c, "TYPE_MISMATCH"); - else if (ret == ENGINE_EBADBKEY) out_string(c, "BKEY_MISMATCH"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory"); -#if 0 // JHPARK_SMGET_OFFSET_HANDLING - else if (ret == ENGINE_EBKEYOOR) out_string(c, "OUT_OF_RANGE"); -#endif - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - - /* free token buffer */ - if (key_tokens != NULL) { - token_buff_release(&c->thread->token_buff, key_tokens); - } - - if (ret != ENGINE_SUCCESS) { - if (c->coll_strkeys != NULL) { - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } - if (c->coll_eitem != NULL) { - free((void *)c->coll_eitem); - c->coll_eitem = NULL; - } - } -} -#endif - -/** - * Get a suffix buffer and insert it into the list of used suffix buffers - * @param c the connection object - * @return a pointer to a new suffix buffer or NULL if allocation failed - */ -static char *get_suffix_buffer(conn *c) -{ - if (c->suffixleft == c->suffixsize) { - char **new_suffix_list; - size_t sz = sizeof(char*) * c->suffixsize * 2; - - new_suffix_list = realloc(c->suffixlist, sz); - if (new_suffix_list) { - c->suffixsize *= 2; - c->suffixlist = new_suffix_list; - } else { - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, - "=%d Failed to resize suffix buffer\n", c->sfd); - } - return NULL; - } - } - - char *suffix = cache_alloc(c->thread->suffix_cache); - if (suffix != NULL) { - *(c->suffixlist + c->suffixleft) = suffix; - ++c->suffixleft; - } - return suffix; -} - -static ENGINE_ERROR_CODE -process_get_single(conn *c, char *key, size_t nkey, bool return_cas) -{ - item *it; - char *cas_val = NULL; - int cas_len = 0; - ENGINE_ERROR_CODE ret; - - ret = mc_engine.v1->get(mc_engine.v0, c, &it, key, nkey, 0); - if (ret != ENGINE_SUCCESS) { - it = NULL; - } - if (settings.detail_enabled) { - stats_prefix_record_get(key, nkey, (it != NULL)); - } - - if (it) { - if (!mc_engine.v1->get_item_info(mc_engine.v0, c, it, &c->hinfo)) { - mc_engine.v1->release(mc_engine.v0, c, it); - return ENGINE_ENOMEM; - } - assert(hinfo_check_ascii_tail_string(&c->hinfo) == 0); /* check "\r\n" */ - - /* Rebuild the suffix */ - char *suffix = get_suffix_buffer(c); - if (suffix == NULL) { - mc_engine.v1->release(mc_engine.v0, c, it); - return ENGINE_ENOMEM; - } - int suffix_len = snprintf(suffix, SUFFIX_SIZE, " %u %u\r\n", - htonl(c->hinfo.flags), c->hinfo.nbytes - 2); - /* suffix_len < SUFFIX_SIZE because the length of nbytes string is smaller than 10. */ - - /* rebuild cas value */ - if (return_cas) { - suffix_len -= 2; /* remove "\r\n" from suffix string */ - cas_val = get_suffix_buffer(c); - if (cas_val == NULL) { - mc_engine.v1->release(mc_engine.v0, c, it); - return ENGINE_ENOMEM; - } - cas_len = snprintf(cas_val, SUFFIX_SIZE, " %"PRIu64"\r\n", c->hinfo.cas); - } - - /* Construct the response. Each hit adds three elements to the - * outgoing data list: - * VALUE [ ]\r\n" + "\r\n" - */ - if (add_iov(c, "VALUE ", 6) != 0 || - add_iov(c, c->hinfo.key, c->hinfo.nkey) != 0 || - add_iov(c, suffix, suffix_len) != 0 || - (return_cas && add_iov(c, cas_val, cas_len) != 0) || - add_iov_hinfo_value_all(c, &c->hinfo) != 0) - { - mc_engine.v1->release(mc_engine.v0, c, it); - return ENGINE_ENOMEM; - } - - /* save the item */ - if (c->ileft >= c->isize) { - item **new_list = realloc(c->ilist, sizeof(item *) * c->isize * 2); - if (new_list) { - c->isize *= 2; - c->ilist = new_list; - } else { - mc_engine.v1->release(mc_engine.v0, c, it); - return ENGINE_ENOMEM;/* out of memory */ - } - } - *(c->ilist + c->ileft) = it; - c->ileft++; - - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, ">%d sending key %s\n", - c->sfd, key); - } - /* item_get() has incremented it->refcount for us */ - STATS_HITS(c, get, key, nkey); - MEMCACHED_COMMAND_GET(c->sfd, key, nkey, c->hinfo.nbytes, c->hinfo.cas); - } else { - STATS_MISSES(c, get, key, nkey); - MEMCACHED_COMMAND_GET(c->sfd, key, nkey, -1, 0); - } - /* Even if the key is not found, return ENGINE_SUCCESS. */ - return ENGINE_SUCCESS; -} - -static void process_mget_complete(conn *c, bool return_cas) -{ - assert(return_cas ? (c->coll_op == OPERATION_MGETS) : (c->coll_op == OPERATION_MGET)); - assert(c->coll_strkeys != NULL); - - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - token_t *key_tokens; - - do { - key_tokens = (token_t*)token_buff_get(&c->thread->token_buff, c->coll_numkeys); - if (key_tokens != NULL) { - bool must_backward_compatible = false; - ret = tokenize_sblocks(&c->memblist, c->coll_lenkeys, c->coll_numkeys, - KEY_MAX_LENGTH, must_backward_compatible, key_tokens); - if (ret != ENGINE_SUCCESS) { - break; /* ENGINE_EBADVALUE | ENGINE_ENOMEM */ - } - } else { - ret = ENGINE_ENOMEM; break; - } - - /* do get operation for each key */ - for (int k = 0; k < c->coll_numkeys; k++) { - ret = process_get_single(c, key_tokens[k].value, key_tokens[k].length, - return_cas); - if (ret != ENGINE_SUCCESS) { - break; /* ret == ENGINE_ENOMEM*/ - } - } - - /* Some items and suffixes might have saved in the above execution. - * To release the items and free the suffixes, the below code is needed. - */ - c->icurr = c->ilist; - c->suffixcurr = c->suffixlist; - - if (ret != ENGINE_SUCCESS) { - /* Releasing items on ilist and freeing suffixes will be - * performed later by calling out_string() function. - * See conn_write() and conn_mwrite() state. - */ - break; - } - - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, ">%d END\n", c->sfd); - } - - /* If the loop was terminated because of out-of-memory, it is not - * reliable to add END\r\n to the buffer, because it might not end - * in \r\n. So we send SERVER_ERROR instead. - */ - if ((add_iov(c, "END\r\n", 5) != 0) || - (IS_UDP(c->transport) && build_udp_headers(c) != 0)) { - /* Releasing items on ilist and freeing suffixes will be - * performed later by calling out_string() function. - * See conn_write() and conn_mwrite() state. - */ - ret = ENGINE_ENOMEM; - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - conn_set_state(c, conn_mwrite); - c->msgcurr = 0; - } else { - if (ret == ENGINE_EBADVALUE) out_string(c, "CLIENT_ERROR bad data chunk"); - else if (ret == ENGINE_ENOMEM) out_string(c, "SERVER_ERROR out of memory writing get response"); - else handle_unexpected_errorcode_ascii(c, __func__, ret); - } - - /* free token buffer */ - if (key_tokens != NULL) { - token_buff_release(&c->thread->token_buff, key_tokens); - } - if (ret != ENGINE_SUCCESS) { - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } -} - -static void update_stat_cas(conn *c, ENGINE_ERROR_CODE ret) -{ - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, cas, c->hinfo.key, c->hinfo.nkey); - break; - case ENGINE_KEY_EEXISTS: - STATS_BADVAL(c, cas, c->hinfo.key, c->hinfo.nkey); - break; - case ENGINE_KEY_ENOENT: - case ENGINE_EBADTYPE: - STATS_MISSES(c, cas, c->hinfo.key, c->hinfo.nkey); - break; - default: - STATS_CMD_NOKEY(c, cas); - } -} - -static void complete_update_ascii(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - - /* The condition of 'c->coll_strkeys != NULL' is given for map collection. - * See process_mop_delete_complete() and process_mop_get_complete(). - */ - if (c->coll_eitem != NULL || c->coll_strkeys != NULL) { - if (c->coll_op == OPERATION_LOP_INSERT) process_lop_insert_complete(c); - else if (c->coll_op == OPERATION_SOP_INSERT) process_sop_insert_complete(c); - else if (c->coll_op == OPERATION_SOP_DELETE) process_sop_delete_complete(c); - else if (c->coll_op == OPERATION_SOP_EXIST) process_sop_exist_complete(c); - else if (c->coll_op == OPERATION_MOP_INSERT) process_mop_insert_complete(c); - else if (c->coll_op == OPERATION_MOP_UPDATE) process_mop_update_complete(c); - else if (c->coll_op == OPERATION_MOP_DELETE) process_mop_delete_complete(c); - else if (c->coll_op == OPERATION_MOP_GET) process_mop_get_complete(c); - else if (c->coll_op == OPERATION_BOP_INSERT || - c->coll_op == OPERATION_BOP_UPSERT) process_bop_insert_complete(c); - else if (c->coll_op == OPERATION_BOP_UPDATE) process_bop_update_complete(c); + APPEND_STAT("daemon_connections", "%u", mc_stats.daemon_conns); + APPEND_STAT("curr_connections", "%u", mc_stats.curr_conns); + APPEND_STAT("quit_connections", "%u", mc_stats.quit_conns); + APPEND_STAT("reject_connections", "%u", mc_stats.rejected_conns); + APPEND_STAT("total_connections", "%u", mc_stats.total_conns); + APPEND_STAT("connection_structures", "%u", mc_stats.conn_structs); + APPEND_STAT("cmd_get", "%"PRIu64, thread_stats.cmd_get); + APPEND_STAT("cmd_set", "%"PRIu64, thread_stats.cmd_set); + APPEND_STAT("cmd_incr", "%"PRIu64, thread_stats.cmd_incr); + APPEND_STAT("cmd_decr", "%"PRIu64, thread_stats.cmd_decr); + APPEND_STAT("cmd_delete", "%"PRIu64, thread_stats.cmd_delete); + APPEND_STAT("cmd_cas", "%"PRIu64, thread_stats.cmd_cas); + APPEND_STAT("cmd_flush", "%"PRIu64, thread_stats.cmd_flush); + APPEND_STAT("cmd_flush_prefix", "%"PRIu64, thread_stats.cmd_flush_prefix); + APPEND_STAT("cmd_auth", "%"PRIu64, thread_stats.cmd_auth); + APPEND_STAT("cmd_lop_create", "%"PRIu64, thread_stats.cmd_lop_create); + APPEND_STAT("cmd_lop_insert", "%"PRIu64, thread_stats.cmd_lop_insert); + APPEND_STAT("cmd_lop_delete", "%"PRIu64, thread_stats.cmd_lop_delete); + APPEND_STAT("cmd_lop_get", "%"PRIu64, thread_stats.cmd_lop_get); + APPEND_STAT("cmd_sop_create", "%"PRIu64, thread_stats.cmd_sop_create); + APPEND_STAT("cmd_sop_insert", "%"PRIu64, thread_stats.cmd_sop_insert); + APPEND_STAT("cmd_sop_delete", "%"PRIu64, thread_stats.cmd_sop_delete); + APPEND_STAT("cmd_sop_get", "%"PRIu64, thread_stats.cmd_sop_get); + APPEND_STAT("cmd_sop_exist", "%"PRIu64, thread_stats.cmd_sop_exist); + APPEND_STAT("cmd_mop_create", "%"PRIu64, thread_stats.cmd_mop_create); + APPEND_STAT("cmd_mop_insert", "%"PRIu64, thread_stats.cmd_mop_insert); + APPEND_STAT("cmd_mop_update", "%"PRIu64, thread_stats.cmd_mop_update); + APPEND_STAT("cmd_mop_delete", "%"PRIu64, thread_stats.cmd_mop_delete); + APPEND_STAT("cmd_mop_get", "%"PRIu64, thread_stats.cmd_mop_get); + APPEND_STAT("cmd_bop_create", "%"PRIu64, thread_stats.cmd_bop_create); + APPEND_STAT("cmd_bop_insert", "%"PRIu64, thread_stats.cmd_bop_insert); + APPEND_STAT("cmd_bop_update", "%"PRIu64, thread_stats.cmd_bop_update); + APPEND_STAT("cmd_bop_delete", "%"PRIu64, thread_stats.cmd_bop_delete); + APPEND_STAT("cmd_bop_get", "%"PRIu64, thread_stats.cmd_bop_get); + APPEND_STAT("cmd_bop_count", "%"PRIu64, thread_stats.cmd_bop_count); + APPEND_STAT("cmd_bop_position", "%"PRIu64, thread_stats.cmd_bop_position); + APPEND_STAT("cmd_bop_pwg", "%"PRIu64, thread_stats.cmd_bop_pwg); + APPEND_STAT("cmd_bop_gbp", "%"PRIu64, thread_stats.cmd_bop_gbp); #ifdef SUPPORT_BOP_MGET - else if (c->coll_op == OPERATION_BOP_MGET) process_bop_mget_complete(c); + APPEND_STAT("cmd_bop_mget", "%"PRIu64, thread_stats.cmd_bop_mget); #endif #ifdef SUPPORT_BOP_SMGET - else if (c->coll_op == OPERATION_BOP_SMGET) process_bop_smget_complete(c); -#endif - else if (c->coll_op == OPERATION_MGET) process_mget_complete(c, false); - else if (c->coll_op == OPERATION_MGETS) process_mget_complete(c, true); - return; - } - - item *it = c->item; - if (!mc_engine.v1->get_item_info(mc_engine.v0, c, it, &c->hinfo)) { - mc_engine.v1->release(mc_engine.v0, c, it); - mc_logger->log(EXTENSION_LOG_WARNING, c, - "%d: Failed to get item info\n", c->sfd); - out_string(c, "SERVER_ERROR failed to get item details"); - return; - } - - ENGINE_ERROR_CODE ret; - if (hinfo_check_ascii_tail_string(&c->hinfo) != 0) { /* check "\r\n" */ - out_string(c, "CLIENT_ERROR bad data chunk"); - ret = ENGINE_EBADVALUE; - } else { - ret = mc_engine.v1->store(mc_engine.v0, c, it, &c->cas, c->store_op, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); -#ifdef ENABLE_DTRACE - switch (c->store_op) { - case OPERATION_ADD: - MEMCACHED_COMMAND_ADD(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_REPLACE: - MEMCACHED_COMMAND_REPLACE(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_APPEND: - MEMCACHED_COMMAND_APPEND(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_PREPEND: - MEMCACHED_COMMAND_PREPEND(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_SET: - MEMCACHED_COMMAND_SET(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_CAS: - MEMCACHED_COMMAND_CAS(c->sfd, c->hinfo.key, c->hinfo.nkey, c->hinfo.nbytes, c->cas); - break; - } + APPEND_STAT("cmd_bop_smget", "%"PRIu64, thread_stats.cmd_bop_smget); #endif - - switch (ret) { - case ENGINE_SUCCESS: - out_string(c, "STORED"); - break; - case ENGINE_KEY_EEXISTS: - out_string(c, "EXISTS"); - break; - case ENGINE_KEY_ENOENT: - out_string(c, "NOT_FOUND"); - break; - case ENGINE_NOT_STORED: - out_string(c, "NOT_STORED"); - break; - case ENGINE_PREFIX_ENAME: - out_string(c, "CLIENT_ERROR invalid prefix name"); - break; - case ENGINE_ENOMEM: - out_string(c, "SERVER_ERROR out of memory"); - break; - case ENGINE_EINVAL: - out_string(c, "CLIENT_ERROR invalid arguments"); - break; - case ENGINE_E2BIG: - out_string(c, "CLIENT_ERROR value too big"); - break; - case ENGINE_EACCESS: - out_string(c, "CLIENT_ERROR access control violation"); - break; - case ENGINE_NOT_MY_VBUCKET: - out_string(c, "SERVER_ERROR not my vbucket"); - break; - case ENGINE_EBADTYPE: - out_string(c, "TYPE_MISMATCH"); - break; - case ENGINE_FAILED: - out_string(c, "SERVER_ERROR failure"); - break; - default: - handle_unexpected_errorcode_ascii(c, __func__, ret); - } - } - - if (c->store_op == OPERATION_CAS) { - update_stat_cas(c, ret); - } else { - STATS_CMD(c, set, c->hinfo.key, c->hinfo.nkey); - } - - mc_engine.v1->release(mc_engine.v0, c, c->item); - c->item = 0; -} - -/** - * get a pointer to the start of the request struct for the current command - */ -static void* binary_get_request(conn *c) -{ - char *ret = c->rcurr; - ret -= (sizeof(c->binary_header) + c->binary_header.request.keylen + - c->binary_header.request.extlen); - - assert(ret >= c->rbuf); - return ret; -} - -/** - * get a pointer to the key in this request - */ -static char* binary_get_key(conn *c) -{ - return c->rcurr - (c->binary_header.request.keylen); -} - -/** - * Insert a key into a buffer, but replace all non-printable characters - * with a '.'. - * - * @param dest where to store the output - * @param destsz size of destination buffer - * @param prefix string to insert before the data - * @param client the client we are serving - * @param from_client set to true if this data is from the client - * @param key the key to add to the buffer - * @param nkey the number of bytes in the key - * @return number of bytes in dest if success, -1 otherwise - */ -static ssize_t key_to_printable_buffer(char *dest, size_t destsz, - int client, bool from_client, - const char *prefix, - const char *key, size_t nkey) -{ - ssize_t nw = snprintf(dest, destsz, "%c%d %s ", from_client ? '>' : '<', - client, prefix); - if (nw == -1) { - return -1; - } - - char *ptr = dest + nw; - destsz -= nw; - if (nkey > destsz) { - nkey = destsz; - } - - for (ssize_t ii = 0; ii < nkey; ++ii, ++key, ++ptr) { - if (isgraph(*key)) { - *ptr = *key; - } else { - *ptr = '.'; - } - } - - *ptr = '\0'; - return ptr - dest; -} - -/** - * Convert a byte array to a text string - * - * @param dest where to store the output - * @param destsz size of destination buffer - * @param prefix string to insert before the data - * @param client the client we are serving - * @param from_client set to true if this data is from the client - * @param data the data to add to the buffer - * @param size the number of bytes in data to print - * @return number of bytes in dest if success, -1 otherwise - */ -static ssize_t bytes_to_output_string(char *dest, size_t destsz, - int client, bool from_client, - const char *prefix, - const char *data, size_t size) -{ - ssize_t nw = snprintf(dest, destsz, "%c%d %s", from_client ? '>' : '<', - client, prefix); - if (nw == -1) { - return -1; - } - ssize_t offset = nw; - - for (ssize_t ii = 0; ii < size; ++ii) { - if (ii % 4 == 0) { - if ((nw = snprintf(dest + offset, destsz - offset, "\n%c%d ", - from_client ? '>' : '<', client)) == -1) { - return -1; - } - offset += nw; - } - if ((nw = snprintf(dest + offset, destsz - offset, - " 0x%02x", (unsigned char)data[ii])) == -1) { - return -1; - } - offset += nw; - } - - if ((nw = snprintf(dest + offset, destsz - offset, "\n")) == -1) { - return -1; - } - - return offset + nw; -} - -static void add_bin_header(conn *c, uint16_t err, uint8_t hdr_len, uint16_t key_len, uint32_t body_len) -{ - protocol_binary_response_header* header; - assert(c); - - c->msgcurr = 0; - c->msgused = 0; - c->iovused = 0; - int ret = add_msghdr(c); - assert(ret == 0); - - header = (protocol_binary_response_header *)c->wbuf; - - header->response.magic = (uint8_t)PROTOCOL_BINARY_RES; - header->response.opcode = c->binary_header.request.opcode; - header->response.keylen = (uint16_t)htons(key_len); - - header->response.extlen = (uint8_t)hdr_len; - header->response.datatype = (uint8_t)PROTOCOL_BINARY_RAW_BYTES; - header->response.status = (uint16_t)htons(err); - - header->response.bodylen = htonl(body_len); - header->response.opaque = c->opaque; - header->response.cas = htonll(c->cas); - - if (settings.verbose > 1) { - char buffer[1024]; - if (bytes_to_output_string(buffer, sizeof(buffer), c->sfd, false, - "Writing bin response:", - (const char*)header->bytes, - sizeof(header->bytes)) != -1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, "%s", buffer); - } - } - - add_iov(c, c->wbuf, sizeof(header->response)); -} - -static void write_bin_packet(conn *c, protocol_binary_response_status err, int swallow) -{ - ssize_t len; - char *buffer = c->wbuf + sizeof(protocol_binary_response_header); - const int MAX_BUF_SIZE = c->wsize - sizeof(protocol_binary_response_header); - - switch (err) { - case PROTOCOL_BINARY_RESPONSE_SUCCESS: - len = 0; - break; - case PROTOCOL_BINARY_RESPONSE_ENOMEM: - len = snprintf(buffer, MAX_BUF_SIZE, "Out of memory"); - break; - case PROTOCOL_BINARY_RESPONSE_UNKNOWN_COMMAND: - len = snprintf(buffer, MAX_BUF_SIZE, "Unknown command"); - break; - case PROTOCOL_BINARY_RESPONSE_KEY_ENOENT: - len = snprintf(buffer, MAX_BUF_SIZE, "Not found"); - break; - case PROTOCOL_BINARY_RESPONSE_EINVAL: - len = snprintf(buffer, MAX_BUF_SIZE, "Invalid arguments"); - break; - case PROTOCOL_BINARY_RESPONSE_KEY_EEXISTS: - len = snprintf(buffer, MAX_BUF_SIZE, "Data exists for key"); - break; - case PROTOCOL_BINARY_RESPONSE_E2BIG: - len = snprintf(buffer, MAX_BUF_SIZE, "Too large"); - break; - case PROTOCOL_BINARY_RESPONSE_DELTA_BADVAL: - len = snprintf(buffer, MAX_BUF_SIZE, - "Non-numeric server-side value for incr or decr"); - break; - case PROTOCOL_BINARY_RESPONSE_NOT_STORED: - len = snprintf(buffer, MAX_BUF_SIZE, "Not stored"); - break; - case PROTOCOL_BINARY_RESPONSE_EBADTYPE: - len = snprintf(buffer, MAX_BUF_SIZE, "Not supported operation, bad type"); - break; - case PROTOCOL_BINARY_RESPONSE_EOVERFLOW: - len = snprintf(buffer, MAX_BUF_SIZE, "Data structure full"); - break; - case PROTOCOL_BINARY_RESPONSE_EBADVALUE: - len = snprintf(buffer, MAX_BUF_SIZE, "Bad value"); - break; - case PROTOCOL_BINARY_RESPONSE_EINDEXOOR: - len = snprintf(buffer, MAX_BUF_SIZE, "Index out of range"); - break; - case PROTOCOL_BINARY_RESPONSE_EBKEYOOR: - len = snprintf(buffer, MAX_BUF_SIZE, "Bkey out of range"); - break; - case PROTOCOL_BINARY_RESPONSE_ELEM_ENOENT: - len = snprintf(buffer, MAX_BUF_SIZE, "Not found element"); - break; - case PROTOCOL_BINARY_RESPONSE_ELEM_EEXISTS: - len = snprintf(buffer, MAX_BUF_SIZE, "Element already exists"); - break; - case PROTOCOL_BINARY_RESPONSE_EBADATTR: - len = snprintf(buffer, MAX_BUF_SIZE, "Attribute not found"); - break; - case PROTOCOL_BINARY_RESPONSE_EBADBKEY: - len = snprintf(buffer, MAX_BUF_SIZE, "Bkey mismatch"); - break; - case PROTOCOL_BINARY_RESPONSE_EBADEFLAG: - len = snprintf(buffer, MAX_BUF_SIZE, "Eflag mismatch"); - break; - case PROTOCOL_BINARY_RESPONSE_UNREADABLE: - len = snprintf(buffer, MAX_BUF_SIZE, "Unreadable item"); - break; - case PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME: - len = snprintf(buffer, MAX_BUF_SIZE, "Invalid prefix name"); - break; - case PROTOCOL_BINARY_RESPONSE_PREFIX_ENOENT: - len = snprintf(buffer, MAX_BUF_SIZE, "Prefix not found"); - break; - case PROTOCOL_BINARY_RESPONSE_AUTH_ERROR: - len = snprintf(buffer, MAX_BUF_SIZE, "Auth failure"); - break; - case PROTOCOL_BINARY_RESPONSE_NOT_SUPPORTED: - len = snprintf(buffer, MAX_BUF_SIZE, "Not supported"); - break; - case PROTOCOL_BINARY_RESPONSE_NOT_MY_VBUCKET: - len = snprintf(buffer, MAX_BUF_SIZE, - "I'm not responsible for this vbucket"); - break; - default: - len = snprintf(buffer, MAX_BUF_SIZE, "UNHANDLED ERROR (%d)", err); - mc_logger->log(EXTENSION_LOG_WARNING, c, - ">%d UNHANDLED ERROR: %d\n", c->sfd, err); - } - - /* Allow the engine to pass extra error information */ - if (mc_engine.v1->errinfo != NULL) { - size_t elen = mc_engine.v1->errinfo(mc_engine.v0, c, buffer + len + 2, - MAX_BUF_SIZE - len - 3); - if (elen > 0) { - memcpy(buffer + len, ": ", 2); - len += elen + 2; - } - } - - if (err != PROTOCOL_BINARY_RESPONSE_SUCCESS && settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, - ">%d Writing an error: %s\n", c->sfd, buffer); - } - - add_bin_header(c, err, 0, 0, len); - if (len > 0) { - add_iov(c, buffer, len); - } - conn_set_state(c, conn_mwrite); - if (swallow > 0) { - c->sbytes = swallow; - c->write_and_go = conn_swallow; - } else { - c->write_and_go = conn_new_cmd; - } -} - -/* Form and send a response to a command over the binary protocol */ -static void write_bin_response(conn *c, void *d, int hlen, int keylen, int dlen) -{ - if (!c->noreply || c->cmd == PROTOCOL_BINARY_CMD_GET || - c->cmd == PROTOCOL_BINARY_CMD_GETK) { - add_bin_header(c, 0, hlen, keylen, dlen); - if (dlen > 0) { - add_iov(c, d, dlen); - } - conn_set_state(c, conn_mwrite); - c->write_and_go = conn_new_cmd; - } else { - conn_set_state(c, conn_new_cmd); - } -} - -static void -handle_unexpected_errorcode_bin(conn *c, const char *func_name, ENGINE_ERROR_CODE ret, int swallow) -{ - if (ret == ENGINE_DISCONNECT) { - conn_set_state(c, conn_closing); - } else if (ret == ENGINE_ENOTSUP) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_SUPPORTED, 0); - } else { - mc_logger->log(EXTENSION_LOG_WARNING, c, "[%s] Unexpected Error: %d\n", - func_name, (int)ret); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINTERNAL, swallow); - } -} - -static void complete_incr_bin(conn *c) -{ - protocol_binary_response_incr* rsp = (protocol_binary_response_incr*)c->wbuf; - protocol_binary_request_incr* req = binary_get_request(c); - - assert(c != NULL); - assert(c->wsize >= sizeof(*rsp)); - - /* fix byteorder in the request */ - req->message.body.delta = ntohll(req->message.body.delta); - req->message.body.initial = ntohll(req->message.body.initial); - req->message.body.expiration = ntohl(req->message.body.expiration); - char *key = binary_get_key(c); - size_t nkey = c->binary_header.request.keylen; - bool incr = (c->cmd == PROTOCOL_BINARY_CMD_INCREMENT || - c->cmd == PROTOCOL_BINARY_CMD_INCREMENTQ); - - if (settings.verbose > 1) { - char buffer[1024]; - ssize_t nw; - nw = key_to_printable_buffer(buffer, sizeof(buffer), c->sfd, true, - incr ? "INCR" : "DECR", key, nkey); - if (nw != -1) { - if (snprintf(buffer + nw, sizeof(buffer) - nw, - " %" PRIu64 ", %" PRIu64 ", %" PRIu64 "\n", - (uint64_t)req->message.body.delta, - (uint64_t)req->message.body.initial, - (uint64_t)req->message.body.expiration) != -1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, "%s", buffer); - } - } - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->arithmetic(mc_engine.v0, - c, key, nkey, incr, - req->message.body.expiration != 0xffffffff, - req->message.body.delta, - req->message.body.initial, - 0, /* flags */ - realtime(req->message.body.expiration), - &c->cas, - &rsp->message.body.value, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - if (incr) stats_prefix_record_incr(key, nkey); - else stats_prefix_record_decr(key, nkey); - } - - switch (ret) { - case ENGINE_SUCCESS: - rsp->message.body.value = htonll(rsp->message.body.value); - write_bin_response(c, &rsp->message.body, 0, 0, - sizeof (rsp->message.body.value)); - if (incr) { - STATS_HITS(c, incr, key, nkey); - } else { - STATS_HITS(c, decr, key, nkey); - } - break; - case ENGINE_KEY_EEXISTS: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_EEXISTS, 0); - break; - case ENGINE_KEY_ENOENT: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - if (incr) { - STATS_MISSES(c, incr, key, nkey); - } else { - STATS_MISSES(c, decr, key, nkey); - } - break; - case ENGINE_PREFIX_ENAME: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME, 0); - break; - case ENGINE_ENOMEM: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - break; - case ENGINE_EINVAL: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_DELTA_BADVAL, 0); - break; - case ENGINE_NOT_STORED: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_STORED, 0); - break; - case ENGINE_NOT_MY_VBUCKET: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_MY_VBUCKET, 0); - break; - case ENGINE_EBADTYPE: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - break; - default: - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - if (ret != ENGINE_DISCONNECT) { - abort(); - } - } -} - -static void complete_update_bin(conn *c) -{ - assert(c != NULL); - - item *it = c->item; - if (!mc_engine.v1->get_item_info(mc_engine.v0, c, it, &c->hinfo)) { - mc_engine.v1->release(mc_engine.v0, c, it); - mc_logger->log(EXTENSION_LOG_WARNING, c, - "%d: Failed to get item info\n", c->sfd); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINTERNAL, 0); - return; - } - /* We don't actually receive the trailing two characters in the bin - * protocol, so we're going to just set them here */ - hinfo_set_ascii_tail_string(&c->hinfo); /* set "\r\n" */ - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->store(mc_engine.v0, c, it, &c->cas, c->store_op, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); -#ifdef ENABLE_DTRACE - switch (c->cmd) { - case OPERATION_ADD: - MEMCACHED_COMMAND_ADD(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_REPLACE: - MEMCACHED_COMMAND_REPLACE(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_APPEND: - MEMCACHED_COMMAND_APPEND(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_PREPEND: - MEMCACHED_COMMAND_PREPEND(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - case OPERATION_SET: - MEMCACHED_COMMAND_SET(c->sfd, c->hinfo.key, c->hinfo.nkey, - (ret == ENGINE_SUCCESS) ? c->hinfo.nbytes : -1, c->cas); - break; - } -#endif - - switch (ret) { - case ENGINE_SUCCESS: - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_KEY_EEXISTS: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_EEXISTS, 0); - break; - case ENGINE_KEY_ENOENT: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - case ENGINE_NOT_STORED: - /* FIXME: check below code, later. */ - if (c->store_op == OPERATION_ADD) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_EEXISTS, 0); - } else if (c->store_op == OPERATION_REPLACE) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - } else { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_STORED, 0); - } - //write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_STORED, 0); - break; - case ENGINE_PREFIX_ENAME: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME, 0); - break; - case ENGINE_ENOMEM: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - break; - case ENGINE_EINVAL: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINVAL, 0); - break; - case ENGINE_E2BIG: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_E2BIG, 0); - break; - case ENGINE_NOT_MY_VBUCKET: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_MY_VBUCKET, 0); - break; - case ENGINE_EBADTYPE: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - break; - default: - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - if (c->store_op == OPERATION_CAS) { - update_stat_cas(c, ret); - } else { - STATS_CMD(c, set, c->hinfo.key, c->hinfo.nkey); - } - - mc_engine.v1->release(mc_engine.v0, c, c->item); - c->item = 0; -} - -static void process_bin_get(conn *c) -{ - item *it; - protocol_binary_response_get* rsp = (protocol_binary_response_get*)c->wbuf; - char* key = binary_get_key(c); - size_t nkey = c->binary_header.request.keylen; - - if (settings.verbose > 1) { - char buffer[1024]; - if (key_to_printable_buffer(buffer, sizeof(buffer), c->sfd, true, - "GET", key, nkey) != -1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, "%s\n", buffer); - } - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->get(mc_engine.v0, c, &it, key, nkey, - c->binary_header.request.vbucket); - if (settings.detail_enabled) { - stats_prefix_record_get(key, nkey, ret == ENGINE_SUCCESS); - } - - uint16_t keylen; - uint32_t bodylen; - - switch (ret) { - case ENGINE_SUCCESS: - if (!mc_engine.v1->get_item_info(mc_engine.v0, c, it, &c->hinfo)) { - mc_engine.v1->release(mc_engine.v0, c, it); - mc_logger->log(EXTENSION_LOG_WARNING, c, - "%d: Failed to get item info\n", c->sfd); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINTERNAL, 0); - break; - } - - /* the length has two unnecessary bytes ("\r\n") */ - keylen = 0; - bodylen = sizeof(rsp->message.body) + (c->hinfo.nbytes - 2); - - STATS_HITS(c, get, key, nkey); - - if (c->cmd == PROTOCOL_BINARY_CMD_GETK) { - bodylen += nkey; - keylen = nkey; - } - add_bin_header(c, 0, sizeof(rsp->message.body), keylen, bodylen); - rsp->message.header.response.cas = htonll(c->hinfo.cas); - - // add the flags - rsp->message.body.flags = c->hinfo.flags; - add_iov(c, &rsp->message.body, sizeof(rsp->message.body)); - - if (c->cmd == PROTOCOL_BINARY_CMD_GETK) { - add_iov(c, c->hinfo.key, nkey); - } - - /* Add the data minus the CRLF */ - add_iov_hinfo_value_some(c, &c->hinfo, c->hinfo.nbytes - 2); - conn_set_state(c, conn_mwrite); - /* Remember this command so we can garbage collect it later */ - c->item = it; - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, get, key, nkey); - - MEMCACHED_COMMAND_GET(c->sfd, key, nkey, -1, 0); - - if (c->noreply) { - conn_set_state(c, conn_new_cmd); - } else { - if (c->cmd == PROTOCOL_BINARY_CMD_GETK) { - char *ofs = c->wbuf + sizeof(protocol_binary_response_header); - add_bin_header(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, - 0, nkey, nkey); - memcpy(ofs, key, nkey); - add_iov(c, ofs, nkey); - conn_set_state(c, conn_mwrite); - } else { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - } - } - break; - case ENGINE_NOT_MY_VBUCKET: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_MY_VBUCKET, 0); - break; - case ENGINE_EBADTYPE: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - break; - default: - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - - if (ret != ENGINE_DISCONNECT) { - /* @todo add proper error handling! */ - mc_logger->log(EXTENSION_LOG_WARNING, c, - "Unknown error code: %d\n", ret); - abort(); - } - } -} - -static void append_bin_stats(const char *key, const uint16_t klen, - const char *val, const uint32_t vlen, - const void *cookie) -{ - /* value without a key is invalid */ - if (klen == 0 && vlen > 0) { - return; - } - - conn *c = (conn*)cookie; - size_t needed = vlen + klen + sizeof(protocol_binary_response_header); - if (!grow_dynamic_buffer(c, needed)) { - return; - } - - char *buf = c->dynamic_buffer.buffer + c->dynamic_buffer.offset; - uint32_t bodylen = klen + vlen; - protocol_binary_response_header header = { - .response.magic = (uint8_t)PROTOCOL_BINARY_RES, - .response.opcode = PROTOCOL_BINARY_CMD_STAT, - .response.keylen = (uint16_t)htons(klen), - .response.datatype = (uint8_t)PROTOCOL_BINARY_RAW_BYTES, - .response.bodylen = htonl(bodylen), - .response.opaque = c->opaque - }; - - memcpy(buf, header.bytes, sizeof(header.response)); - buf += sizeof(header.response); - - if (klen > 0) { - memcpy(buf, key, klen); - buf += klen; - - if (vlen > 0) { - memcpy(buf, val, vlen); - } - } - - c->dynamic_buffer.offset += sizeof(header.response) + bodylen; - assert(c->dynamic_buffer.offset <= c->dynamic_buffer.size); -} - -/** - * Append a key-value pair to the stats output buffer. This function assumes - * that the output buffer is big enough. - */ -static void append_ascii_stats(const char *key, const uint16_t klen, - const char *val, const uint32_t vlen, - const void *cookie) -{ - /* value without a key is invalid */ - if (klen == 0 && vlen > 0) { - return; - } - - conn *c = (conn*)cookie; - size_t needed = vlen + klen + 10; // 10 == "STAT = \r\n" - if (!grow_dynamic_buffer(c, needed)) { - return; - } - - char *pos = c->dynamic_buffer.buffer + c->dynamic_buffer.offset; - uint32_t nbytes = 5; /* "END\r\n" or "STAT " */ - - if (klen == 0 && vlen == 0) { - memcpy(pos, "END\r\n", 5); - } else { - memcpy(pos, "STAT ", 5); - memcpy(pos + nbytes, key, klen); - nbytes += klen; - if (vlen != 0) { - pos[nbytes] = ' '; - ++nbytes; - memcpy(pos + nbytes, val, vlen); - nbytes += vlen; - } - memcpy(pos + nbytes, "\r\n", 2); - nbytes += 2; - } - - c->dynamic_buffer.offset += nbytes; - assert(c->dynamic_buffer.offset <= c->dynamic_buffer.size); -} - -static void process_bin_stat(conn *c) -{ - char *subcommand = binary_get_key(c); - size_t nkey = c->binary_header.request.keylen; - - if (settings.verbose > 1) { - char buffer[1024]; - if (key_to_printable_buffer(buffer, sizeof(buffer), c->sfd, true, - "STATS", subcommand, nkey) != -1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, "%s\n", buffer); - } - } - - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - if (nkey == 0) { - /* request all statistics */ - ret = mc_engine.v1->get_stats(mc_engine.v0, c, NULL, 0, append_bin_stats); - if (ret == ENGINE_SUCCESS) { - server_stats(&append_bin_stats, c, false); - } - } else if (strncmp(subcommand, "reset", 5) == 0) { - stats_reset(c); - mc_engine.v1->reset_stats(mc_engine.v0, c); - } else if (strncmp(subcommand, "settings", 8) == 0) { - process_stats_settings(&append_bin_stats, c); -#ifdef ENABLE_ZK_INTEGRATION - } else if (strncmp(subcommand, "zookeeper", 9) == 0) { - process_stats_zookeeper(&append_bin_stats, c); -#endif - } else if (strncmp(subcommand, "detail", 6) == 0) { - char *subcmd_pos = subcommand + 6; - if (settings.allow_detailed) { - if (strncmp(subcmd_pos, " dump", 5) == 0) { - int len; - char *dump_buf = stats_prefix_dump(NULL, 0, &len); - if (dump_buf == NULL) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - return; - } - append_bin_stats("detailed", strlen("detailed"), dump_buf, len, c); - free(dump_buf); - } else if (strncmp(subcmd_pos, " on", 3) == 0) { - settings.detail_enabled = 1; - } else if (strncmp(subcmd_pos, " off", 4) == 0) { - settings.detail_enabled = 0; - } else { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - return; - } - } else { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - return; - } - } else if (strncmp(subcommand, "aggregate", 9) == 0) { - server_stats(&append_bin_stats, c, true); - } else if (strncmp(subcommand, "topkeys", 7) == 0) { - if (default_topkeys) { - topkeys_stats(default_topkeys, c, get_current_time(), append_bin_stats); - } else { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - return; - } - } else { - ret = mc_engine.v1->get_stats(mc_engine.v0, c, subcommand, nkey, append_bin_stats); - } - - switch (ret) { - case ENGINE_SUCCESS: - append_bin_stats(NULL, 0, NULL, 0, c); - write_and_free(c, c->dynamic_buffer.buffer, c->dynamic_buffer.offset); - c->dynamic_buffer.buffer = NULL; - break; - case ENGINE_ENOMEM: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - break; - case ENGINE_KEY_ENOENT: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - case ENGINE_PREFIX_ENOENT: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENOENT, 0); - break; - case ENGINE_EINVAL: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINVAL, 0); - break; - default: - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static void bin_read_chunk(conn *c, enum bin_substates next_substate, uint32_t chunk) -{ - assert(c); - c->substate = next_substate; - - /* Ok... do we have room for everything in our buffer? */ - ptrdiff_t offset = c->rcurr + sizeof(protocol_binary_request_header) - c->rbuf; - if (chunk > c->rsize - offset) { - size_t nsize = c->rsize; - size_t size = chunk + sizeof(protocol_binary_request_header); - - while (size > nsize) { - nsize *= 2; - } - - if (nsize != c->rsize) { - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, - "%d: Need to grow buffer from %lu to %lu\n", - c->sfd, (unsigned long)c->rsize, (unsigned long)nsize); - } - char *newm = realloc(c->rbuf, nsize); - if (newm == NULL) { - if (settings.verbose) { - mc_logger->log(EXTENSION_LOG_WARNING, c, - "%d: Failed to grow buffer. closing connection\n", c->sfd); - } - conn_set_state(c, conn_closing); - return; - } - - c->rbuf= newm; - /* rcurr should point to the same offset in the packet */ - c->rcurr = c->rbuf + offset - sizeof(protocol_binary_request_header); - c->rsize = nsize; - } - if (c->rbuf != c->rcurr) { - memmove(c->rbuf, c->rcurr, c->rbytes); - c->rcurr = c->rbuf; - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, - "%d: Repack input buffer\n", c->sfd); - } - } - } - - /* preserve the header in the buffer.. */ - c->ritem = c->rcurr + sizeof(protocol_binary_request_header); - c->rlbytes = chunk; - c->rltotal = 0; - conn_set_state(c, conn_nread); -} - -static void bin_read_key(conn *c, enum bin_substates next_substate, int extra) -{ - bin_read_chunk(c, next_substate, c->keylen + extra); -} - -/* Just write an error message and disconnect the client */ -static void handle_binary_protocol_error(conn *c) -{ - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINVAL, 0); - if (settings.verbose) { - mc_logger->log(EXTENSION_LOG_INFO, c, - "%d: Protocol error (opcode %02x), close connection\n", - c->sfd, c->binary_header.request.opcode); - } - c->write_and_go = conn_closing; -} - -static void init_sasl_conn(conn *c) -{ - assert(c); - - if (!c->sasl_conn) { - int result=sasl_server_new("memcached", - NULL, NULL, NULL, NULL, - NULL, 0, &c->sasl_conn); - if (result != SASL_OK) { - if (settings.verbose) { - mc_logger->log(EXTENSION_LOG_INFO, c, - "%d: Failed to initialize SASL conn.\n", c->sfd); - } - c->sasl_conn = NULL; - } - } -} - -static void get_auth_data(const void *cookie, auth_data_t *data) -{ - conn *c = (conn*)cookie; - if (c->sasl_conn) { - sasl_getprop(c->sasl_conn, SASL_USERNAME, (void*)&data->username); -#ifdef ENABLE_ISASL - sasl_getprop(c->sasl_conn, ISASL_CONFIG, (void*)&data->config); -#endif - } -} - -#ifdef SASL_ENABLED -static void bin_list_sasl_mechs(conn *c) -{ - init_sasl_conn(c); - const char *result_string = NULL; - unsigned int string_length = 0; - int result=sasl_listmech(c->sasl_conn, NULL, - "", /* What to prepend the string with */ - " ", /* What to separate mechanisms with */ - "", /* What to append to the string */ - &result_string, &string_length, - NULL); - if (result != SASL_OK) { - /* Perhaps there's a better error for this... */ - if (settings.verbose) { - mc_logger->log(EXTENSION_LOG_INFO, c, - "%d: Failed to list SASL mechanisms.\n", c->sfd); - } - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_AUTH_ERROR, 0); - return; - } - write_bin_response(c, (char*)result_string, 0, 0, string_length); -} -#endif - -struct sasl_tmp { - int ksize; - int vsize; - char data[]; /* data + ksize == value */ -}; - -static void process_bin_sasl_auth(conn *c) -{ - assert(c->binary_header.request.extlen == 0); - - uint32_t nkey = c->binary_header.request.keylen; - uint32_t vlen = 0; - - if (nkey <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - nkey; - } else { - handle_binary_protocol_error(c); - return; - } - - if (nkey > MAX_SASL_MECH_LEN) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINVAL, vlen); - return; - } - - char *key = binary_get_key(c); - assert(key); - - size_t buffer_size = sizeof(struct sasl_tmp) + nkey + vlen + 2; - struct sasl_tmp *data = calloc(sizeof(struct sasl_tmp) + buffer_size, 1); - if (!data) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, vlen); - return; - } - - data->ksize = nkey; - data->vsize = vlen; - memcpy(data->data, key, nkey); - - c->item = data; - c->ritem = data->data + nkey; - c->rlbytes = vlen; - c->rltotal = 0; - conn_set_state(c, conn_nread); - c->substate = bin_reading_sasl_auth_data; -} - -static void process_bin_complete_sasl_auth(conn *c) -{ - const char *out = NULL; - unsigned int outlen = 0; - - assert(c->item); - init_sasl_conn(c); - - uint32_t nkey = c->binary_header.request.keylen; - uint32_t vlen = 0; - - if (nkey <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - nkey; - } else { - handle_binary_protocol_error(c); - return; - } - - struct sasl_tmp *stmp = c->item; - char mech[nkey+1]; - memcpy(mech, stmp->data, nkey); - mech[nkey] = 0x00; - - if (settings.verbose) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, - "%d: mech: ``%s'' with %d bytes of data\n", c->sfd, mech, vlen); - } - - const char *challenge = vlen == 0 ? NULL : (stmp->data + nkey); - - int result=-1; - - switch (c->cmd) { - case PROTOCOL_BINARY_CMD_SASL_AUTH: - result = sasl_server_start(c->sasl_conn, mech, challenge, vlen, - &out, &outlen); - break; - case PROTOCOL_BINARY_CMD_SASL_STEP: - result = sasl_server_step(c->sasl_conn, challenge, vlen, - &out, &outlen); - break; - default: - assert(false); /* CMD should be one of the above */ - /* This code is pretty much impossible, but makes the compiler happier */ - if (settings.verbose) { - mc_logger->log(EXTENSION_LOG_WARNING, c, - "%d: Unhandled command %d with challenge %s\n", - c->sfd, c->cmd, challenge); - } - break; - } - - free(c->item); - c->item = NULL; - c->ritem = NULL; - - if (settings.verbose) { - mc_logger->log(EXTENSION_LOG_INFO, c, - "%d: sasl result code: %d\n", c->sfd, result); - } - - switch (result) { - case SASL_OK: - write_bin_response(c, "Authenticated", 0, 0, strlen("Authenticated")); - auth_data_t data; - get_auth_data(c, &data); - perform_callbacks(ON_AUTH, (const void*)&data, c); - STATS_CMD_NOKEY(c, auth); - break; - case SASL_CONTINUE: - add_bin_header(c, PROTOCOL_BINARY_RESPONSE_AUTH_CONTINUE, 0, 0, outlen); - if (outlen > 0) { - add_iov(c, out, outlen); - } - conn_set_state(c, conn_mwrite); - c->write_and_go = conn_new_cmd; - break; - default: - if (settings.verbose) { - mc_logger->log(EXTENSION_LOG_INFO, c, - "%d: Unknown sasl response: %d\n", c->sfd, result); - } - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_AUTH_ERROR, 0); - STATS_ERRORS_NOKEY(c, auth); - } -} - -static bool authenticated(conn *c) -{ - bool rv = false; - - switch (c->cmd) { - case PROTOCOL_BINARY_CMD_SASL_LIST_MECHS: /* FALLTHROUGH */ - case PROTOCOL_BINARY_CMD_SASL_AUTH: /* FALLTHROUGH */ - case PROTOCOL_BINARY_CMD_SASL_STEP: /* FALLTHROUGH */ - case PROTOCOL_BINARY_CMD_VERSION: /* FALLTHROUGH */ - rv = true; - break; - default: - if (c->sasl_conn) { - const void *uname = NULL; - sasl_getprop(c->sasl_conn, SASL_USERNAME, &uname); - rv = uname != NULL; - } - } - - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, - "%d: authenticated() in cmd 0x%02x is %s\n", - c->sfd, c->cmd, rv ? "true" : "false"); - } - return rv; -} - -static void process_bin_lop_create(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - - /* fix byteorder in the request */ - protocol_binary_request_lop_create* req = binary_get_request(c); - req->message.body.exptime = ntohl(req->message.body.exptime); - req->message.body.maxcount = ntohl(req->message.body.maxcount); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d LOP CREATE ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " flags(%u) exptime(%d) maxcount(%d) ovflaction(%s) readable(%d)\n", - req->message.body.flags, req->message.body.exptime, req->message.body.maxcount, - get_ovflaction_str(req->message.body.ovflaction), req->message.body.readable); - } - - item_attr attr_data; - attr_data.flags = req->message.body.flags; - attr_data.exptime = realtime(req->message.body.exptime); - attr_data.maxcount = req->message.body.maxcount; - - if ((req->message.body.ovflaction > 0) && - (req->message.body.ovflaction == OVFL_ERROR || - req->message.body.ovflaction == OVFL_HEAD_TRIM || - req->message.body.ovflaction == OVFL_TAIL_TRIM)) { - attr_data.ovflaction = req->message.body.ovflaction; - } else { - attr_data.ovflaction = 0; - } - attr_data.readable = req->message.body.readable; - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->list_struct_create(mc_engine.v0, c, key, nkey, &attr_data, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_lop_create(key, nkey); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_OKS(c, lop_create, key, nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - default: - STATS_CMD_NOKEY(c, lop_create); - if (ret == ENGINE_KEY_EEXISTS) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_EEXISTS, 0); - else if (ret == ENGINE_PREFIX_ENAME) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME, 0); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static void process_bin_lop_prepare_nread(conn *c) -{ - assert(c != NULL); - assert(c->cmd == PROTOCOL_BINARY_CMD_LOP_INSERT); - - char *key = binary_get_key(c); - uint32_t nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - uint32_t vlen = 0; - - if (nkey + c->binary_header.request.extlen <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - (nkey + c->binary_header.request.extlen); - } else { - handle_binary_protocol_error(c); - return; - } - - /* fix byteorder in the request */ - protocol_binary_request_lop_insert* req = binary_get_request(c); - req->message.body.index = ntohl(req->message.body.index); - if (req->message.body.create) { - req->message.body.exptime = ntohl(req->message.body.exptime); - req->message.body.maxcount = ntohl(req->message.body.maxcount); - } - - if (settings.verbose > 1) { - fprintf(stderr, "<%d LOP INSERT ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " Index(%d) NBytes(%d)", req->message.body.index, vlen); - if (req->message.body.create) { - fprintf(stderr, " %s", "Create"); - } - fprintf(stderr, "\n"); - } - - eitem *elem; - ENGINE_ERROR_CODE ret; - - if ((vlen + 2) > settings.max_element_bytes) { - ret = ENGINE_E2BIG; - } else { - ret = mc_engine.v1->list_elem_alloc(mc_engine.v0, c, key, nkey, vlen+2, &elem); - } - if (ret == ENGINE_SUCCESS) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_LIST, elem, &c->einfo); - ritem_set_first(c, CONN_RTYPE_EINFO, vlen); - c->coll_eitem = (void *)elem; - c->coll_ecount = 1; - c->coll_op = OPERATION_LOP_INSERT; - c->coll_index = req->message.body.index; - if (req->message.body.create) { - c->coll_attrp = &c->coll_attr_space; /* create if not exist */ - c->coll_attrp->flags = req->message.body.flags; - c->coll_attrp->exptime = realtime(req->message.body.exptime); - c->coll_attrp->maxcount = req->message.body.maxcount; - c->coll_attrp->readable = 1; - } else { - c->coll_attrp = NULL; - } - conn_set_state(c, conn_nread); - c->substate = bin_reading_lop_nread_complete; - } else { - if (settings.detail_enabled) { - stats_prefix_record_lop_insert(key, nkey, false); - } - STATS_CMD_NOKEY(c, lop_insert); - if (ret == ENGINE_E2BIG) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_E2BIG, vlen); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, vlen); - else - handle_unexpected_errorcode_bin(c, __func__, ret, vlen); - } -} - -static void process_bin_lop_insert_complete(conn *c) -{ - assert(c->coll_op == OPERATION_LOP_INSERT); - assert(c->coll_eitem != NULL); - - /* We don't actually receive the trailing two characters in the bin - * protocol, so we're going to just set them here */ - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_LIST, c->coll_eitem, &c->einfo); - einfo_set_ascii_tail_string(&c->einfo); /* set "\r\n" */ - - bool created; - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->list_elem_insert(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_index, c->coll_eitem, - c->coll_attrp, &created, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_lop_insert(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, lop_insert, c->coll_key, c->coll_nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, lop_insert, c->coll_key, c->coll_nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, lop_insert); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EOVERFLOW) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EOVERFLOW, 0); - else if (ret == ENGINE_EINDEXOOR) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINDEXOOR, 0); - else if (ret == ENGINE_PREFIX_ENAME) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME, 0); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - if (ret != ENGINE_SUCCESS) { - mc_engine.v1->list_elem_free(mc_engine.v0, c, c->coll_eitem); - } - c->coll_eitem = NULL; -} - -static void process_bin_lop_nread_complete(conn *c) -{ - //protocol_binary_response_status eno = PROTOCOL_BINARY_RESPONSE_EINVAL; - assert(c != NULL); - process_bin_lop_insert_complete(c); -} - -static void process_bin_lop_delete(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - - /* fix byteorder in the request */ - protocol_binary_request_lop_delete* req = binary_get_request(c); - req->message.body.from_index = ntohl(req->message.body.from_index); - req->message.body.to_index = ntohl(req->message.body.to_index); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d LOP DELETE ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " From_Index(%d) To_Index(%d)\n", - req->message.body.from_index, req->message.body.to_index); - } - - uint32_t del_count; - bool dropped; - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->list_elem_delete(mc_engine.v0, c, key, nkey, - req->message.body.from_index, - req->message.body.to_index, - (bool)req->message.body.drop, - &del_count, &dropped, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_lop_delete(key, nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_ELEM_HITS(c, lop_delete, key, nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, lop_delete, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINDEXOOR, 0); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, lop_delete, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, lop_delete); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static ENGINE_ERROR_CODE -out_bin_lop_get_response(conn *c, bool delete, struct elems_result *eresultp) -{ - protocol_binary_response_lop_get* rsp = (protocol_binary_response_lop_get*)c->wbuf; - eitem **elem_array = eresultp->elem_array; - uint32_t elem_count = eresultp->elem_count; - uint32_t bodylen; - uint32_t *vlenptr; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - do { - bodylen = sizeof(rsp->message.body) + (elem_count * sizeof(uint32_t)); - if ((vlenptr = (uint32_t *)malloc(elem_count * sizeof(uint32_t))) == NULL) { - ret = ENGINE_ENOMEM; break; - } - for (int i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_LIST, - elem_array[i], &c->einfo); - bodylen += (c->einfo.nbytes - 2); - vlenptr[i] = htonl(c->einfo.nbytes - 2); - } - add_bin_header(c, 0, sizeof(rsp->message.body), 0, bodylen); - - // add the flags and count - rsp->message.body.flags = eresultp->flags; - rsp->message.body.count = htonl(elem_count); - add_iov(c, &rsp->message.body, sizeof(rsp->message.body)); - - // add value lengths - add_iov(c, (char*)vlenptr, elem_count*sizeof(uint32_t)); - - /* Add the data without CRLF */ - for (int i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_LIST, - elem_array[i], &c->einfo); - if (add_iov_einfo_value_some(c, &c->einfo, c->einfo.nbytes - 2) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - /* Remember this command so we can garbage collect it later */ - c->coll_eitem = (void *)elem_array; - c->coll_ecount = elem_count; - c->coll_resps = (char *)vlenptr; - c->coll_op = OPERATION_LOP_GET; - conn_set_state(c, conn_mwrite); - } else { - mc_engine.v1->list_elem_release(mc_engine.v0, c, elem_array, elem_count); - if (elem_array != NULL) { - free(elem_array); - elem_array = NULL; - } - if (vlenptr != NULL) { - free(vlenptr); - vlenptr = NULL; - } - } - return ret; -} - -static void process_bin_lop_get(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - assert(c->cmd == PROTOCOL_BINARY_CMD_LOP_GET); - struct elems_result eresult; - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - ENGINE_ERROR_CODE ret; - - /* fix byteorder in the request */ - protocol_binary_request_lop_get* req = binary_get_request(c); - int from_index = ntohl(req->message.body.from_index); - int to_index = ntohl(req->message.body.to_index); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d LOP GET ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " From_Index(%d) To_Index(%d) Delete(%s)\n", - from_index, to_index, - (req->message.body.delete ? "true" : "false")); - } - - ret = mc_engine.v1->list_elem_get(mc_engine.v0, c, key, nkey, - from_index, to_index, - (bool)req->message.body.delete, - (bool)req->message.body.drop, - &eresult, c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_lop_get(key, nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - ret = out_bin_lop_get_response(c, (bool)req->message.body.delete, &eresult); - if (ret == ENGINE_SUCCESS) { - STATS_ELEM_HITS(c, lop_get, key, nkey); - } else { - STATS_CMD_NOKEY(c, lop_get); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - } - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, lop_get, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINDEXOOR, 0); - break; - case ENGINE_KEY_ENOENT: - case ENGINE_UNREADABLE: - STATS_MISSES(c, lop_get, key, nkey); - if (ret == ENGINE_KEY_ENOENT) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - else - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_UNREADABLE, 0); - break; - default: - STATS_CMD_NOKEY(c, lop_get); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static void process_bin_sop_create(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - - /* fix byteorder in the request */ - protocol_binary_request_sop_create* req = binary_get_request(c); - req->message.body.exptime = ntohl(req->message.body.exptime); - req->message.body.maxcount = ntohl(req->message.body.maxcount); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d SOP CREATE ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " flags(%u) exptime(%d) maxcount(%d) ovflaction(%s) readable(%d)\n", - req->message.body.flags, req->message.body.exptime, req->message.body.maxcount, - "error", req->message.body.readable); - } - - item_attr attr_data; - attr_data.flags = req->message.body.flags; - attr_data.exptime = realtime(req->message.body.exptime); - attr_data.maxcount = req->message.body.maxcount; - attr_data.readable = req->message.body.readable; - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->set_struct_create(mc_engine.v0, c, key, nkey, &attr_data, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_sop_create(key, nkey); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_OKS(c, sop_create, key, nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - default: - STATS_CMD_NOKEY(c, sop_create); - if (ret == ENGINE_KEY_EEXISTS) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_EEXISTS, 0); - else if (ret == ENGINE_PREFIX_ENAME) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME, 0); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static void process_bin_sop_prepare_nread(conn *c) -{ - assert(c != NULL); - assert(c->cmd == PROTOCOL_BINARY_CMD_SOP_INSERT || - c->cmd == PROTOCOL_BINARY_CMD_SOP_DELETE || - c->cmd == PROTOCOL_BINARY_CMD_SOP_EXIST); - char *key = binary_get_key(c); - uint32_t nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - uint32_t vlen = 0; - - if (nkey + c->binary_header.request.extlen <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - (nkey + c->binary_header.request.extlen); - } else { - handle_binary_protocol_error(c); - return; - } - - if (settings.verbose > 1) { - if (c->cmd == PROTOCOL_BINARY_CMD_SOP_INSERT) { - fprintf(stderr, "<%d SOP INSERT ", c->sfd); - } else if (c->cmd == PROTOCOL_BINARY_CMD_SOP_DELETE) { - fprintf(stderr, "<%d SOP DELETE ", c->sfd); - } else { - fprintf(stderr, "<%d SOP EXIST ", c->sfd); - } - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " NBytes(%d)", vlen); - fprintf(stderr, "\n"); - } - - eitem *elem = NULL; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - if ((vlen + 2) > settings.max_element_bytes) { - ret = ENGINE_E2BIG; - } else { - if (c->cmd == PROTOCOL_BINARY_CMD_SOP_INSERT) { - ret = mc_engine.v1->set_elem_alloc(mc_engine.v0, c, key, nkey, vlen+2, &elem); - } else { /* PROTOCOL_BINARY_CMD_SOP_DELETE or PROTOCOL_BINARY_CMD_SOP_EXIST */ - if ((elem = (eitem *)malloc(sizeof(value_item) + vlen + 2)) == NULL) - ret = ENGINE_ENOMEM; - else - ((value_item*)elem)->len = vlen + 2; - } - } - if (ret == ENGINE_SUCCESS) { - if (c->cmd == PROTOCOL_BINARY_CMD_SOP_INSERT) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_SET, elem, &c->einfo); - ritem_set_first(c, CONN_RTYPE_EINFO, vlen); - } else { - c->ritem = ((value_item *)elem)->ptr; - c->rlbytes = vlen; - c->rltotal = 0; - } - c->coll_eitem = (void *)elem; - c->coll_ecount = 1; - if (c->cmd == PROTOCOL_BINARY_CMD_SOP_INSERT) { - protocol_binary_request_sop_insert* req = binary_get_request(c); - if (req->message.body.create) { - req->message.body.exptime = ntohl(req->message.body.exptime); - req->message.body.maxcount = ntohl(req->message.body.maxcount); - } - c->coll_op = OPERATION_SOP_INSERT; - if (req->message.body.create) { - c->coll_attrp = &c->coll_attr_space; /* create if not exist */ - c->coll_attrp->flags = req->message.body.flags; - c->coll_attrp->exptime = realtime(req->message.body.exptime); - c->coll_attrp->maxcount = req->message.body.maxcount; - c->coll_attrp->readable = 1; - } else { - c->coll_attrp = NULL; - } - } else if (c->cmd == PROTOCOL_BINARY_CMD_SOP_DELETE) { - protocol_binary_request_sop_delete* req = binary_get_request(c); - c->coll_op = OPERATION_SOP_DELETE; - c->coll_drop = (req->message.body.drop ? true : false); - } else { /* PROTOCOL_BINARY_CMD_SOP_EXIST */ - c->coll_op = OPERATION_SOP_EXIST; - } - conn_set_state(c, conn_nread); - c->substate = bin_reading_sop_nread_complete; - } else { - if (c->cmd == PROTOCOL_BINARY_CMD_SOP_INSERT) { - if (settings.detail_enabled) - stats_prefix_record_sop_insert(key, nkey, false); - STATS_CMD_NOKEY(c, sop_insert); - } else if (c->cmd == PROTOCOL_BINARY_CMD_SOP_DELETE) { - if (settings.detail_enabled) - stats_prefix_record_sop_delete(key, nkey, false); - STATS_CMD_NOKEY(c, sop_delete); - } else { - if (settings.detail_enabled) - stats_prefix_record_sop_exist(key, nkey, false); - STATS_CMD_NOKEY(c, sop_exist); - } - if (ret == ENGINE_E2BIG) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_E2BIG, vlen); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, vlen); - else - handle_unexpected_errorcode_bin(c, __func__, ret, vlen); - } -} - -static void process_bin_sop_insert_complete(conn *c) -{ - assert(c->coll_eitem != NULL); - - /* We don't actually receive the trailing two characters in the bin - * protocol, so we're going to just set them here */ - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_SET, c->coll_eitem, &c->einfo); - einfo_set_ascii_tail_string(&c->einfo); /* set "\r\n" */ - - bool created; - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->set_elem_insert(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_eitem, c->coll_attrp, &created, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_sop_insert(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, sop_insert, c->coll_key, c->coll_nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, sop_insert, c->coll_key, c->coll_nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, sop_insert); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EOVERFLOW) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EOVERFLOW, 0); - else if (ret == ENGINE_ELEM_EEXISTS) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ELEM_EEXISTS, 0); - else if (ret == ENGINE_PREFIX_ENAME) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME, 0); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - /* release the c->coll_eitem reference */ - if (ret != ENGINE_SUCCESS) { - mc_engine.v1->set_elem_free(mc_engine.v0, c, c->coll_eitem); - } - c->coll_eitem = NULL; -} - -static void process_bin_sop_delete_complete(conn *c) -{ - assert(c->coll_eitem != NULL); - - /* We don't actually receive the trailing two characters in the bin - * protocol, so we're going to just set them here */ - value_item *value = (value_item *)c->coll_eitem; - memcpy(value->ptr + value->len - 2, "\r\n", 2); - - bool dropped; - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->set_elem_delete(mc_engine.v0, c, c->coll_key, c->coll_nkey, - value->ptr, value->len, c->coll_drop, - &dropped, c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_sop_delete(c->coll_key, c->coll_nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_ELEM_HITS(c, sop_delete, c->coll_key, c->coll_nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, sop_delete, c->coll_key, c->coll_nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ELEM_ENOENT, 0); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, sop_delete, c->coll_key, c->coll_nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, sop_delete); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - /* release the c->coll_eitem reference */ - free(c->coll_eitem); - c->coll_eitem = NULL; -} - -static void process_bin_sop_exist_complete(conn *c) -{ - assert(c->coll_eitem != NULL); - - /* We don't actually receive the trailing two characters in the bin - * protocol, so we're going to just set them here */ - value_item *value = (value_item *)c->coll_eitem; - memcpy(value->ptr + value->len - 2, "\r\n", 2); - - bool exist; - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->set_elem_exist(mc_engine.v0, c, c->coll_key, c->coll_nkey, - value->ptr, value->len, &exist, - c->binary_header.request.vbucket); - - if (settings.detail_enabled) { - stats_prefix_record_sop_exist(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - { - STATS_HITS(c, sop_exist, c->coll_key, c->coll_nkey); - - protocol_binary_response_sop_exist* rsp = (protocol_binary_response_sop_exist*)c->wbuf; - rsp->message.body.exist = htonl((exist == true) ? 1 : 0); - write_bin_response(c, &rsp->message.body, 0, 0, sizeof(rsp->message.body)); - } - break; - case ENGINE_KEY_ENOENT: - case ENGINE_UNREADABLE: - STATS_MISSES(c, sop_exist, c->coll_key, c->coll_nkey); - if (ret == ENGINE_KEY_ENOENT) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - else - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_UNREADABLE, 0); - break; - default: - STATS_CMD_NOKEY(c, sop_exist); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - /* release the c->coll_eitem reference */ - free(c->coll_eitem); - c->coll_eitem = NULL; -} - -static void process_bin_sop_nread_complete(conn *c) -{ - //protocol_binary_response_status eno = PROTOCOL_BINARY_RESPONSE_EINVAL; - assert(c != NULL); - - if (c->cmd == PROTOCOL_BINARY_CMD_SOP_INSERT) - process_bin_sop_insert_complete(c); - else if (c->cmd == PROTOCOL_BINARY_CMD_SOP_DELETE) - process_bin_sop_delete_complete(c); - else if (c->cmd == PROTOCOL_BINARY_CMD_SOP_EXIST) - process_bin_sop_exist_complete(c); -} - -static ENGINE_ERROR_CODE -out_bin_sop_get_response(conn *c, bool delete, struct elems_result *eresultp) -{ - protocol_binary_response_sop_get* rsp = (protocol_binary_response_sop_get*)c->wbuf; - eitem **elem_array = eresultp->elem_array; - uint32_t elem_count = eresultp->elem_count; - uint32_t bodylen; - uint32_t *vlenptr; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - do { - bodylen = sizeof(rsp->message.body) + (elem_count * sizeof(uint32_t)); - if ((vlenptr = (uint32_t*)malloc(elem_count * sizeof(uint32_t))) == NULL) { - ret = ENGINE_ENOMEM; break; - } - for (int i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_SET, - elem_array[i], &c->einfo); - bodylen += (c->einfo.nbytes - 2); - vlenptr[i] = htonl(c->einfo.nbytes - 2); - } - add_bin_header(c, 0, sizeof(rsp->message.body), 0, bodylen); - - // add the flags and count - rsp->message.body.flags = eresultp->flags; - rsp->message.body.count = htonl(elem_count); - add_iov(c, &rsp->message.body, sizeof(rsp->message.body)); - - // add value lengths - add_iov(c, (char*)vlenptr, elem_count*sizeof(uint32_t)); - - /* Add the data without CRLF */ - for (int i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_SET, - elem_array[i], &c->einfo); - if (add_iov_einfo_value_some(c, &c->einfo, c->einfo.nbytes - 2) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - /* Remember this command so we can garbage collect it later */ - c->coll_eitem = (void *)elem_array; - c->coll_ecount = elem_count; - c->coll_resps = (char *)vlenptr; - c->coll_op = OPERATION_SOP_GET; - conn_set_state(c, conn_mwrite); - } else { - mc_engine.v1->set_elem_release(mc_engine.v0, c, elem_array, elem_count); - if (elem_array != NULL) { - free(elem_array); - elem_array = NULL; - } - if (vlenptr != NULL) { - free(vlenptr); - vlenptr = NULL; - } - } - return ret; -} - -static void process_bin_sop_get(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - assert(c->cmd == PROTOCOL_BINARY_CMD_SOP_GET); - struct elems_result eresult; - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - ENGINE_ERROR_CODE ret; - - /* fix byteorder in the request */ - protocol_binary_request_sop_get* req = binary_get_request(c); - uint32_t req_count = ntohl(req->message.body.count); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d SOP GET ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " Count(%d) Delete(%s)\n", req_count, - (req->message.body.delete ? "true" : "false")); - } - - ret = mc_engine.v1->set_elem_get(mc_engine.v0, c, key, nkey, req_count, - (bool)req->message.body.delete, - (bool)req->message.body.drop, - &eresult, c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_sop_get(key, nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - ret = out_bin_sop_get_response(c, (bool)req->message.body.delete, &eresult); - if (ret == ENGINE_SUCCESS) { - STATS_ELEM_HITS(c, sop_get, key, nkey); - } else { - STATS_CMD_NOKEY(c, sop_get); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - } - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, sop_get, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ELEM_ENOENT, 0); - break; - case ENGINE_KEY_ENOENT: - case ENGINE_UNREADABLE: - STATS_MISSES(c, sop_get, key, nkey); - if (ret == ENGINE_KEY_ENOENT) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - else - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_UNREADABLE, 0); - break; - default: - STATS_CMD_NOKEY(c, sop_get); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static void process_bin_bop_create(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - - /* fix byteorder in the request */ - protocol_binary_request_bop_create* req = binary_get_request(c); - req->message.body.exptime = ntohl(req->message.body.exptime); - req->message.body.maxcount = ntohl(req->message.body.maxcount); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d BOP CREATE ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " flags(%u) exptime(%d) maxcount(%d) ovflaction(%s) readable(%d)\n", - req->message.body.flags, req->message.body.exptime, req->message.body.maxcount, - get_ovflaction_str(req->message.body.ovflaction), req->message.body.readable); - } - - item_attr attr_data; - attr_data.flags = req->message.body.flags; - attr_data.exptime = realtime(req->message.body.exptime); - attr_data.maxcount = req->message.body.maxcount; - - if ((req->message.body.ovflaction > 0) && - (req->message.body.ovflaction == OVFL_ERROR || - req->message.body.ovflaction == OVFL_SMALLEST_TRIM || - req->message.body.ovflaction == OVFL_LARGEST_TRIM || - req->message.body.ovflaction == OVFL_SMALLEST_SILENT_TRIM || - req->message.body.ovflaction == OVFL_LARGEST_SILENT_TRIM)) { - attr_data.ovflaction = req->message.body.ovflaction; - } else { - attr_data.ovflaction = 0; - } - attr_data.readable = req->message.body.readable; - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->btree_struct_create(mc_engine.v0, c, key, nkey, &attr_data, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_bop_create(key, nkey); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_OKS(c, bop_create, key, nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - default: - STATS_CMD_NOKEY(c, bop_create); - if (ret == ENGINE_KEY_EEXISTS) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_EEXISTS, 0); - else if (ret == ENGINE_PREFIX_ENAME) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME, 0); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static void process_bin_bop_prepare_nread(conn *c) -{ - assert(c != NULL); - assert(c->cmd == PROTOCOL_BINARY_CMD_BOP_INSERT || - c->cmd == PROTOCOL_BINARY_CMD_BOP_UPSERT); - char *key = binary_get_key(c); - uint32_t nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - uint32_t vlen = 0; - - if (nkey + c->binary_header.request.extlen <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - (nkey + c->binary_header.request.extlen); - } else { - handle_binary_protocol_error(c); - return; - } - - /* fix byteorder in the request */ - protocol_binary_request_bop_insert* req = binary_get_request(c); - if (req->message.body.nbkey == 0) { - uint64_t bkey_temp; - memcpy((uint8_t*)&bkey_temp, req->message.body.bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(req->message.body.bkey, (uint8_t*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)req->message.body.bkey = ntohll(*(uint64_t*)req->message.body.bkey); - } - if (req->message.body.create) { - req->message.body.exptime = ntohl(req->message.body.exptime); - req->message.body.maxcount = ntohl(req->message.body.maxcount); - } - - if (settings.verbose > 1) { - fprintf(stderr, "<%d BOP %s ", - c->sfd, (c->cmd == PROTOCOL_BINARY_CMD_BOP_INSERT ? "INSERT" : "UPSERT")); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " NBKey(%d) NEFlag(%d) NBytes(%d)", - req->message.body.nbkey, req->message.body.neflag, vlen); - if (req->message.body.create) { - fprintf(stderr, " %s", "Create"); - } - fprintf(stderr, "\n"); - } - - eitem *elem; - ENGINE_ERROR_CODE ret; - - if ((vlen + 2) > settings.max_element_bytes) { - ret = ENGINE_E2BIG; - } else { - ret = mc_engine.v1->btree_elem_alloc(mc_engine.v0, c, key, nkey, - req->message.body.nbkey, - req->message.body.neflag, vlen+2, &elem); - } - if (ret == ENGINE_SUCCESS) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, elem, &c->einfo); - if (c->einfo.nscore == 0) { - memcpy((void*)c->einfo.score, req->message.body.bkey, sizeof(uint64_t)); - } else { - memcpy((void*)c->einfo.score, req->message.body.bkey, c->einfo.nscore); - } - if (c->einfo.neflag > 0) { - memcpy((void*)c->einfo.eflag, req->message.body.eflag, c->einfo.neflag); - } - - ritem_set_first(c, CONN_RTYPE_EINFO, vlen); - c->coll_eitem = (void *)elem; - c->coll_ecount = 1; - c->coll_op = (c->cmd == PROTOCOL_BINARY_CMD_BOP_INSERT ? OPERATION_BOP_INSERT - : OPERATION_BOP_UPSERT); - if (req->message.body.create) { - c->coll_attrp = &c->coll_attr_space; /* create if not exist */ - c->coll_attrp->flags = req->message.body.flags; - c->coll_attrp->exptime = realtime(req->message.body.exptime); - c->coll_attrp->maxcount = req->message.body.maxcount; - c->coll_attrp->readable = 1; - } else { - c->coll_attrp = NULL; - } - conn_set_state(c, conn_nread); - c->substate = bin_reading_bop_nread_complete; - } else { - if (settings.detail_enabled) { - stats_prefix_record_bop_insert(key, nkey, false); - } - STATS_CMD_NOKEY(c, bop_insert); - if (ret == ENGINE_E2BIG) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_E2BIG, vlen); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, vlen); - else - handle_unexpected_errorcode_bin(c, __func__, ret, vlen); - } -} - -static void process_bin_bop_insert_complete(conn *c) -{ - assert(c->coll_op == OPERATION_BOP_INSERT || - c->coll_op == OPERATION_BOP_UPSERT); - assert(c->coll_eitem != NULL); - - /* We don't actually receive the trailing two characters in the bin - * protocol, so we're going to just set them here */ - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, c->coll_eitem, &c->einfo); - einfo_set_ascii_tail_string(&c->einfo); /* set "\r\n" */ - - bool created; - bool replaced; - bool replace_if_exist = (c->coll_op == OPERATION_BOP_UPSERT ? true : false); - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->btree_elem_insert(mc_engine.v0, c, c->coll_key, c->coll_nkey, - c->coll_eitem, replace_if_exist, - c->coll_attrp, &replaced, &created, NULL, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_bop_insert(c->coll_key, c->coll_nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, bop_insert, c->coll_key, c->coll_nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, bop_insert, c->coll_key, c->coll_nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, bop_insert); - - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EBADBKEY) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADBKEY, 0); - else if (ret == ENGINE_EOVERFLOW) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EOVERFLOW, 0); - else if (ret == ENGINE_EBKEYOOR) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBKEYOOR, 0); - else if (ret == ENGINE_ELEM_EEXISTS) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ELEM_EEXISTS, 0); - else if (ret == ENGINE_PREFIX_ENAME) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENAME, 0); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - /* release the c->coll_eitem reference */ - if (ret != ENGINE_SUCCESS) { - mc_engine.v1->btree_elem_free(mc_engine.v0, c, c->coll_eitem); - } - c->coll_eitem = NULL; -} - -static void process_bin_bop_nread_complete(conn *c) -{ - //protocol_binary_response_status eno = PROTOCOL_BINARY_RESPONSE_EINVAL; - assert(c != NULL); - process_bin_bop_insert_complete(c); -} - -static void process_bin_bop_update_complete(conn *c) -{ - assert(c->coll_op == OPERATION_BOP_UPDATE); - char *new_value = NULL; - int new_nbytes = 0; - - if (c->coll_eitem != NULL) { - value_item *value = (value_item *)c->coll_eitem; - memcpy(value->ptr + value->len - 2, "\r\n", 2); - new_value = value->ptr; - new_nbytes = value->len; - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->btree_elem_update(mc_engine.v0, c, c->coll_key, c->coll_nkey, - &c->coll_bkrange, - (c->coll_eupdate.neflag == EFLAG_NULL ? NULL : &c->coll_eupdate), - new_value, new_nbytes, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_bop_update(c->coll_key, c->coll_nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_ELEM_HITS(c, bop_update, c->coll_key, c->coll_nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, bop_update, c->coll_key, c->coll_nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ELEM_ENOENT, 0); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, bop_update, c->coll_key, c->coll_nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, bop_update); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EBADBKEY) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADBKEY, 0); - else if (ret == ENGINE_EBADEFLAG) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADEFLAG, 0); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - if (c->coll_eitem != NULL) { - free((void*)c->coll_eitem); - c->coll_eitem = NULL; - } -} - -static void process_bin_bop_update_prepare_nread(conn *c) -{ - assert(c != NULL); - assert(c->cmd == PROTOCOL_BINARY_CMD_BOP_UPDATE); - char *key = binary_get_key(c); - uint32_t nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - uint32_t vlen = 0; - int real_nbkey; - - if (nkey + c->binary_header.request.extlen <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - (nkey + c->binary_header.request.extlen); - } else { - handle_binary_protocol_error(c); - return; - } - - /* fix byteorder in the request */ - protocol_binary_request_bop_update* req = binary_get_request(c); - if (req->message.body.nbkey == 0) { - uint64_t bkey_temp; - memcpy((uint8_t*)&bkey_temp, req->message.body.bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(req->message.body.bkey, (uint8_t*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)req->message.body.bkey = ntohll(*(uint64_t*)req->message.body.bkey); - real_nbkey = 8; - } else { - real_nbkey = req->message.body.nbkey; - } - /* build bkey range */ - memcpy(c->coll_bkrange.from_bkey, req->message.body.bkey, real_nbkey); - c->coll_bkrange.from_nbkey = req->message.body.nbkey; - c->coll_bkrange.to_nbkey = BKEY_NULL; - /* build elem update */ - c->coll_eupdate.fwhere = req->message.body.fwhere; - c->coll_eupdate.bitwop = req->message.body.bitwop; - c->coll_eupdate.neflag = req->message.body.neflag; - if (req->message.body.neflag > 0) { - memcpy(c->coll_eupdate.eflag, req->message.body.eflag, req->message.body.neflag); - } - - if (settings.verbose > 1) { - fprintf(stderr, "<%d BOP UPDATE", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " NBKey(%d) NEFlag(%d) NBytes(%d)", - req->message.body.nbkey, req->message.body.neflag, vlen); - fprintf(stderr, "\n"); - } - - if (req->message.body.novalue) { - c->coll_eitem = NULL; - c->coll_ecount = 0; - c->coll_op = OPERATION_BOP_UPDATE; - process_bin_bop_update_complete(c); - return; - } - - eitem *elem = NULL; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - if ((vlen + 2) > settings.max_element_bytes) { - ret = ENGINE_E2BIG; - } else { - if ((elem = (eitem *)malloc(sizeof(value_item) + vlen + 2)) == NULL) - ret = ENGINE_ENOMEM; - else - ((value_item*)elem)->len = vlen + 2; - } - if (ret == ENGINE_SUCCESS) { - c->ritem = ((value_item *)elem)->ptr; - c->rlbytes = vlen; - c->rltotal = 0; - c->coll_eitem = (void *)elem; - c->coll_ecount = 1; - c->coll_op = OPERATION_BOP_UPDATE; - conn_set_state(c, conn_nread); - c->substate = bin_reading_bop_update_nread_complete; - } else { - if (settings.detail_enabled) { - stats_prefix_record_bop_update(key, nkey, false); - } - STATS_CMD_NOKEY(c, bop_update); - /* ret == ENGINE_E2BIG || ret == ENGINE_ENOMEM */ - if (ret == ENGINE_E2BIG) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_E2BIG, vlen); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, vlen); - else - handle_unexpected_errorcode_bin(c, __func__, ret, vlen); - } -} - -static void process_bin_bop_delete(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - - /* fix byteorder in the request */ - protocol_binary_request_bop_delete* req = binary_get_request(c); - bkey_range *bkrange = &req->message.body.bkrange; - eflag_filter *efilter = &req->message.body.efilter; - if (bkrange->from_nbkey == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, bkrange->from_bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(bkrange->from_bkey, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)bkrange->from_bkey = ntohll(*(uint64_t*)bkrange->from_bkey); - } - if (bkrange->to_nbkey == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, bkrange->to_bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(bkrange->to_bkey, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)bkrange->to_bkey = ntohll(*(uint64_t*)bkrange->to_bkey); - } - req->message.body.count = ntohl(req->message.body.count); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d BOP DELETE ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - } - - uint32_t del_count; - uint32_t acc_count; /* access count */ - bool dropped; - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->btree_elem_delete(mc_engine.v0, c, key, nkey, - bkrange, efilter, req->message.body.count, - (bool)req->message.body.drop, - &del_count, &acc_count, &dropped, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_bop_delete(key, nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_ELEM_HITS(c, bop_delete, key, nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, bop_delete, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ELEM_ENOENT, 0); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, bop_delete, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, bop_delete); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EBADBKEY) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADBKEY, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static ENGINE_ERROR_CODE -out_bin_bop_get_response(conn *c, bool delete, struct elems_result *eresultp) -{ - protocol_binary_response_bop_get* rsp = (protocol_binary_response_bop_get*)c->wbuf; - eitem **elem_array = eresultp->elem_array; - uint32_t elem_count = eresultp->elem_count; - uint32_t bodylen; - uint32_t *bkeyptr; - uint32_t *vlenptr; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - do { - bodylen = sizeof(rsp->message.body) + (elem_count * (sizeof(uint64_t)+sizeof(uint32_t))); - if ((bkeyptr = (uint32_t *)malloc(elem_count * sizeof(uint64_t)+sizeof(uint32_t))) == NULL) { - ret = ENGINE_ENOMEM; break; - } - vlenptr = (uint32_t *)((char*)bkeyptr + (sizeof(uint64_t) * elem_count)); - for (int i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - elem_array[i], &c->einfo); - bodylen += (c->einfo.nbytes - 2); - bkeyptr[i] = htonll(*(uint64_t*)c->einfo.score); - vlenptr[i] = htonl(c->einfo.nbytes - 2); - } - add_bin_header(c, 0, sizeof(rsp->message.body), 0, bodylen); - - // add the flags and count - rsp->message.body.flags = eresultp->flags; - rsp->message.body.count = htonl(elem_count); - add_iov(c, &rsp->message.body, sizeof(rsp->message.body)); - - // add bkey data and value lengths - add_iov(c, (char*)bkeyptr, elem_count*(sizeof(uint64_t)+sizeof(uint32_t))); - - /* Add the data without CRLF */ - for (int i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - elem_array[i], &c->einfo); - if (add_iov_einfo_value_some(c, &c->einfo, c->einfo.nbytes - 2) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - /* Remember this command so we can garbage collect it later */ - c->coll_eitem = (void *)elem_array; - c->coll_ecount = elem_count; - c->coll_resps = (char *)bkeyptr; - c->coll_op = OPERATION_BOP_GET; - conn_set_state(c, conn_mwrite); - } else { - mc_engine.v1->btree_elem_release(mc_engine.v0, c, elem_array, elem_count); - if (elem_array != NULL) { - free(elem_array); - elem_array = NULL; - } - if (bkeyptr != NULL) { - free(bkeyptr); - bkeyptr = NULL; - } - } - return ret; -} - -static void process_bin_bop_get(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - assert(c->cmd == PROTOCOL_BINARY_CMD_BOP_GET); - struct elems_result eresult; - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - ENGINE_ERROR_CODE ret; - - /* fix byteorder in the request */ - protocol_binary_request_bop_get* req = binary_get_request(c); - bkey_range *bkrange = &req->message.body.bkrange; - eflag_filter *efilter = &req->message.body.efilter; - if (bkrange->from_nbkey == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, bkrange->from_bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(bkrange->from_bkey, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)bkrange->from_bkey = ntohll(*(uint64_t*)bkrange->from_bkey); - } - if (bkrange->to_nbkey == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, bkrange->to_bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(bkrange->to_bkey, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)bkrange->to_bkey = ntohll(*(uint64_t*)bkrange->to_bkey); - } - req->message.body.offset = ntohl(req->message.body.offset); - req->message.body.count = ntohl(req->message.body.count); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d BOP GET ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " Offset(%u) Count(%u) Delete(%s)\n", - req->message.body.offset, req->message.body.count, - (req->message.body.delete ? "true" : "false")); - } - - ret = mc_engine.v1->btree_elem_get(mc_engine.v0, c, key, nkey, - bkrange, efilter, - req->message.body.offset, - req->message.body.count, - (bool)req->message.body.delete, - (bool)req->message.body.drop, - &eresult, c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - bool is_hit = (ret==ENGINE_SUCCESS || ret==ENGINE_ELEM_ENOENT); - stats_prefix_record_bop_get(key, nkey, is_hit); - } - - switch (ret) { - case ENGINE_SUCCESS: - ret = out_bin_bop_get_response(c, (bool)req->message.body.delete, &eresult); - if (ret == ENGINE_SUCCESS) { - STATS_ELEM_HITS(c, bop_get, key, nkey); - } else { - STATS_CMD_NOKEY(c, bop_get); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - } - break; - case ENGINE_ELEM_ENOENT: - STATS_NONE_HITS(c, bop_get, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ELEM_ENOENT, 0); - break; - case ENGINE_KEY_ENOENT: - case ENGINE_EBKEYOOR: - case ENGINE_UNREADABLE: - STATS_MISSES(c, bop_get, key, nkey); - if (ret == ENGINE_KEY_ENOENT) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - else if (ret == ENGINE_EBKEYOOR) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBKEYOOR, 0); - else - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_UNREADABLE, 0); - break; - default: - STATS_CMD_NOKEY(c, bop_get); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EBADBKEY) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADBKEY, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static void process_bin_bop_count(conn *c) -{ - assert(c != NULL); - assert(c->cmd == PROTOCOL_BINARY_CMD_BOP_COUNT); - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - - /* fix byteorder in the request */ - protocol_binary_request_bop_count* req = binary_get_request(c); - bkey_range *bkrange = &req->message.body.bkrange; - eflag_filter *efilter = &req->message.body.efilter; - if (bkrange->from_nbkey == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, bkrange->from_bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(bkrange->from_bkey, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)bkrange->from_bkey = ntohll(*(uint64_t*)bkrange->from_bkey); - } - if (bkrange->to_nbkey == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, bkrange->to_bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(bkrange->to_bkey, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)bkrange->to_bkey = ntohll(*(uint64_t*)bkrange->to_bkey); - } - - if (settings.verbose > 1) { - fprintf(stderr, "<%d BOP COUNT", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, "\n"); - } - - uint32_t elem_count; - uint32_t opcost; /* operation cost */ - uint32_t flags = 0; - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->btree_elem_count(mc_engine.v0, c, key, nkey, - bkrange, efilter, &elem_count, &opcost, - c->binary_header.request.vbucket); - if (settings.detail_enabled) { - stats_prefix_record_bop_count(key, nkey, (ret==ENGINE_SUCCESS)); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, bop_count, key, nkey); - protocol_binary_response_bop_count* rsp = (protocol_binary_response_bop_count*)c->wbuf; - // add the flags and count - rsp->message.body.flags = flags; - rsp->message.body.count = htonl(elem_count); - write_bin_response(c, &rsp->message.body, 0, 0, sizeof(rsp->message.body)); - break; - case ENGINE_KEY_ENOENT: - case ENGINE_UNREADABLE: - STATS_MISSES(c, bop_count, key, nkey); - if (ret == ENGINE_KEY_ENOENT) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - else - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_UNREADABLE, 0); - break; - default: - STATS_CMD_NOKEY(c, bop_count); - if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EBADBKEY) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADBKEY, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -#if defined(SUPPORT_BOP_MGET) || defined(SUPPORT_BOP_SMGET) -static void process_bin_bop_prepare_nread_keys(conn *c) -{ - assert(c != NULL); - assert(c->cmd == PROTOCOL_BINARY_CMD_BOP_MGET || c->cmd == PROTOCOL_BINARY_CMD_BOP_SMGET); - - char *key = binary_get_key(c); - uint32_t nkey = c->binary_header.request.keylen; - c->coll_key = key; - c->coll_nkey = nkey; - size_t vlen = 0; - - if (nkey + c->binary_header.request.extlen <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - (nkey + c->binary_header.request.extlen); - } else { - handle_binary_protocol_error(c); - return; - } - - /* fix byteorder in the request */ - protocol_binary_request_bop_mkeys* req = binary_get_request(c); - bkey_range *bkrange = &req->message.body.bkrange; - if (bkrange->from_nbkey == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, bkrange->from_bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(bkrange->from_bkey, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)bkrange->from_bkey = ntohll(*(uint64_t*)bkrange->from_bkey); - } - if (bkrange->to_nbkey == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, bkrange->to_bkey, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(bkrange->to_bkey, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)bkrange->to_bkey = ntohll(*(uint64_t*)bkrange->to_bkey); - } - req->message.body.req_offset = ntohl(req->message.body.req_offset); - req->message.body.req_count = ntohl(req->message.body.req_count); - req->message.body.key_count = ntohl(req->message.body.key_count); - - if (settings.verbose > 1) { - fprintf(stderr, "<%d BOP %s ", c->sfd, (c->cmd==PROTOCOL_BINARY_CMD_BOP_MGET ? "MGET" : "SMGET")); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, " ReqOffset(%u) ReqCount(%u) KeyCount(%u)\n", - req->message.body.req_offset, req->message.body.req_count, req->message.body.key_count); - } - - eitem *elem=NULL; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - int need_size = 0; - do { - /* validation checking on arguments */ - if (req->message.body.key_count < 1 || vlen < 1 || req->message.body.req_count < 1) { - ret = ENGINE_EBADVALUE; break; - } - if (req->message.body.key_count > ((vlen/2)+1)) { - ret = ENGINE_EBADVALUE; break; - } -#ifdef SUPPORT_BOP_MGET - if (c->cmd == PROTOCOL_BINARY_CMD_BOP_MGET) { - int bmget_count; - int elem_array_size; - - if (req->message.body.key_count > MAX_BMGET_KEY_COUNT || - req->message.body.req_count > MAX_BMGET_ELM_COUNT) { - ret = ENGINE_EBADVALUE; break; - } - bmget_count = req->message.body.key_count * req->message.body.req_count; - elem_array_size = bmget_count * sizeof(eitem*); - - need_size = elem_array_size; - } -#endif -#ifdef SUPPORT_BOP_SMGET - if (c->cmd == PROTOCOL_BINARY_CMD_BOP_SMGET) { -#ifdef JHPARK_OLD_SMGET_INTERFACE - if (c->coll_smgmode == 0) { - int smget_count; - int elem_array_size; /* elem pointer array where the found elements will be saved */ - int kmis_array_size; /* key index array where the missed key indexes are to be saved */ - int elem_rshdr_size; /* the size of result header about the found elems */ - int kmis_rshdr_size; /* the size of result header about the missed keys */ - - if (req->message.body.key_count > MAX_SMGET_KEY_COUNT || - (req->message.body.req_offset + req->message.body.req_count) > MAX_SMGET_REQ_COUNT) { - ret = ENGINE_EBADVALUE; break; - } - smget_count = req->message.body.req_offset + req->message.body.req_count; - elem_array_size = smget_count * (sizeof(eitem*) + (2*sizeof(uint32_t))); - kmis_array_size = req->message.body.key_count * sizeof(uint32_t); - elem_rshdr_size = smget_count * (sizeof(uint64_t) + (3*sizeof(uint32_t))); - kmis_rshdr_size = req->message.body.key_count * sizeof(uint32_t); - need_size = elem_array_size + kmis_array_size + elem_rshdr_size + kmis_rshdr_size; - } else -#endif - { - int elem_array_size; /* smget element array size */ - int ehit_array_size; /* smget hitted elem array size */ - int emis_array_size; /* element missed keys array size */ - int elem_rshdr_size; /* the size of result header about the found elems */ - int emis_rshdr_size; /* the size of result header about the missed keys */ - - if (req->message.body.key_count > MAX_SMGET_KEY_COUNT || - (req->message.body.req_offset + req->message.body.req_count) > MAX_SMGET_REQ_COUNT) { - ret = ENGINE_EBADVALUE; break; - } - elem_array_size = (req->message.body.req_count + req->message.body.key_count) * sizeof(eitem*); - ehit_array_size = req->message.body.req_count * sizeof(smget_ehit_t); - emis_array_size = req->message.body.key_count * sizeof(smget_emis_t); - elem_rshdr_size = req->message.body.req_count * (sizeof(uint64_t) + (3*sizeof(uint32_t))); - emis_rshdr_size = req->message.body.key_count * sizeof(uint32_t); - need_size = elem_array_size + ehit_array_size + emis_array_size - + elem_rshdr_size + emis_rshdr_size; - } - } -#endif - assert(need_size > 0); - - if ((elem = (eitem *)malloc(need_size)) == NULL) { - ret = ENGINE_ENOMEM; - } else { - /* allocate memory blocks needed */ - if (mblck_list_alloc(&c->thread->mblck_pool, 1, vlen, &c->memblist) < 0) { - free((void*)elem); - ret = ENGINE_ENOMEM; - } else { - c->coll_bkrange = req->message.body.bkrange; - c->coll_efilter = req->message.body.efilter; - c->coll_roffset = req->message.body.req_offset; - c->coll_rcount = req->message.body.req_count; - c->coll_numkeys = req->message.body.key_count; - c->coll_lenkeys = vlen + 2; - } - } - } while(0); - - if (ret == ENGINE_SUCCESS) { - c->coll_strkeys = (void*)&c->memblist; - ritem_set_first(c, CONN_RTYPE_MBLCK, vlen); - c->coll_eitem = (void *)elem; - c->coll_ecount = 0; - c->coll_op = (c->cmd==PROTOCOL_BINARY_CMD_BOP_MGET ? OPERATION_BOP_MGET : OPERATION_BOP_SMGET); - conn_set_state(c, conn_nread); - c->substate = bin_reading_bop_nread_keys_complete; - } else { - /* ret == ENGINE_EBADVALUE || ret == ENGINE_ENOMEM */ - if (ret == ENGINE_EBADVALUE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADVALUE, vlen); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, vlen); - else - handle_unexpected_errorcode_bin(c, __func__, ret, vlen); - } -} -#endif - -#ifdef SUPPORT_BOP_MGET -static void process_bin_bop_mget_complete(conn *c) -{ - assert(c->coll_op == OPERATION_BOP_MGET); - assert(c->coll_eitem != NULL); - - ENGINE_ERROR_CODE ret = ENGINE_ENOTSUP; - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_SUPPORTED, 0); - - if (ret != ENGINE_SUCCESS) { - if (c->coll_strkeys != NULL) { - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } - if (c->coll_eitem != NULL) { - free((void *)c->coll_eitem); - c->coll_eitem = NULL; - } - } -} -#endif - -#ifdef SUPPORT_BOP_SMGET -#ifdef JHPARK_OLD_SMGET_INTERFACE -static void process_bin_bop_smget_complete_old(conn *c) -{ - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - int smget_count = c->coll_roffset + c->coll_rcount; - token_t *key_tokens; - - eitem **elem_array = (eitem **)c->coll_eitem; - uint32_t *kfnd_array = (uint32_t*)((char*)elem_array + (smget_count*sizeof(eitem*))); - uint32_t *flag_array = (uint32_t*)((char*)kfnd_array + (smget_count*sizeof(uint32_t))); - uint32_t *kmis_array = (uint32_t*)((char*)flag_array + (smget_count*sizeof(uint32_t))); - uint32_t elem_count = 0; - uint32_t kmis_count = 0; - bool trimmed; - bool duplicated; - - /* We don't actually receive the trailing two("\r\n") characters in binary protocol. - * We should consider this when tokenizing key strings. - */ - key_tokens = (token_t*)token_buff_get(&c->thread->token_buff, c->coll_numkeys); - if (key_tokens != NULL) { - bool must_backward_compatible = true; /* Must be backward compatible */ - ret = tokenize_mblocks(&c->memblist, c->coll_lenkeys-2, c->coll_numkeys, - KEY_MAX_LENGTH, must_backward_compatible, key_tokens); - /* ret : ENGINE_SUCCESS | ENGINE_EBADVALUE | ENGINE_ENOMEM */ - } else { - ret = ENGINE_ENOMEM; - } - if (ret == ENGINE_SUCCESS) { - if (c->coll_bkrange.to_nbkey == BKEY_NULL) { - memcpy(c->coll_bkrange.to_bkey, c->coll_bkrange.from_bkey, - (c->coll_bkrange.from_nbkey==0 ? sizeof(uint64_t) : c->coll_bkrange.from_nbkey)); - c->coll_bkrange.to_nbkey = c->coll_bkrange.from_nbkey; - } - assert(c->coll_numkeys > 0); - assert(c->coll_rcount > 0); - assert((c->coll_roffset + c->coll_rcount) <= MAX_SMGET_REQ_COUNT); - ret = mc_engine.v1->btree_elem_smget_old(mc_engine.v0, c, key_tokens, c->coll_numkeys, - &c->coll_bkrange, - (c->coll_efilter.ncompval==0 ? NULL : &c->coll_efilter), - c->coll_roffset, c->coll_rcount, - elem_array, kfnd_array, flag_array, &elem_count, - kmis_array, &kmis_count, &trimmed, &duplicated, - c->binary_header.request.vbucket); - } - - switch (ret) { - case ENGINE_SUCCESS: - { - protocol_binary_response_bop_smget* rsp = (protocol_binary_response_bop_smget*)c->wbuf; - uint32_t real_elem_hdr_size = elem_count * (sizeof(uint64_t) + (3*sizeof(uint32_t))); - uint32_t real_kmis_hdr_size = kmis_count * sizeof(uint32_t); - uint32_t bodylen, i; - uint64_t *bkeyptr; - uint32_t *vlenptr; - uint32_t *flagptr; - uint32_t *klenptr; - - char *resultptr = ((char *)kmis_array + (c->coll_numkeys * sizeof(uint32_t))); - if (((long)resultptr % 8) != 0) { /* NOT aligned */ - resultptr += (8 - ((long)resultptr % 8)); - } - bkeyptr = (uint64_t *)resultptr; - vlenptr = (uint32_t *)((char*)bkeyptr + (sizeof(uint64_t) * elem_count)); - flagptr = (uint32_t *)((char*)vlenptr + (sizeof(uint32_t) * elem_count)); - klenptr = (uint32_t *)((char*)flagptr + (sizeof(uint32_t) * elem_count)); - - bodylen = sizeof(rsp->message.body) + (real_elem_hdr_size + real_kmis_hdr_size); - for (i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - elem_array[i], &c->einfo); - bodylen += ((c->einfo.nbytes - 2) + key_tokens[kfnd_array[i]].length); - bkeyptr[i] = htonll(*(uint64_t*)c->einfo.score); - vlenptr[i] = htonl(c->einfo.nbytes - 2); - flagptr[i] = flag_array[i]; - klenptr[i] = htonl(key_tokens[kfnd_array[i]].length); - } - for (i = 0; i < kmis_count; i++) { - bodylen += key_tokens[kmis_array[i]].length; - klenptr[elem_count+i] = htonl(key_tokens[kmis_array[i]].length); - } - add_bin_header(c, 0, sizeof(rsp->message.body), 0, bodylen); - - // add the flags and count - rsp->message.body.elem_count = htonl(elem_count); - rsp->message.body.miss_count = htonl(kmis_count); - add_iov(c, &rsp->message.body, sizeof(rsp->message.body)); - - // add value lengths - if (add_iov(c, (char*)bkeyptr, real_elem_hdr_size+real_kmis_hdr_size) != 0) { - ret = ENGINE_ENOMEM; - } - - /* Add the data without CRLF */ - if (ret == ENGINE_SUCCESS) { - for (i = 0; i < elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - elem_array[i], &c->einfo); - if (add_iov_einfo_value_some(c, &c->einfo, c->einfo.nbytes - 2) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } - /* Add the found key */ - if (ret == ENGINE_SUCCESS) { - for (i = 0; i < elem_count; i++) { - if (add_iov(c, key_tokens[kfnd_array[i]].value, - key_tokens[kfnd_array[i]].length) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } - /* Add the missed key */ - if (ret == ENGINE_SUCCESS) { - for (i = 0; i < kmis_count; i++) { - if (add_iov(c, key_tokens[kmis_array[i]].value, - key_tokens[kmis_array[i]].length) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } - - if (ret == ENGINE_SUCCESS) { - /* Remember this command so we can garbage collect it later */ - /* c->coll_eitem = (void *)elem_array; */ - STATS_OKS_NOKEY(c, bop_smget); - c->coll_ecount = elem_count; - c->coll_op = OPERATION_BOP_SMGET; - conn_set_state(c, conn_mwrite); - } else { - STATS_CMD_NOKEY(c, bop_smget); - mc_engine.v1->btree_elem_release(mc_engine.v0, c, elem_array, elem_count); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - } - } - break; - default: - STATS_CMD_NOKEY(c, bop_smget); - if (ret == ENGINE_EBADVALUE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADVALUE, 0); - else if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EBADBKEY) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADBKEY, 0); - else if (ret == ENGINE_EBKEYOOR) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBKEYOOR, 0); - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - /* free token buffer */ - if (key_tokens != NULL) { - token_buff_release(&c->thread->token_buff, key_tokens); - } - - if (ret != ENGINE_SUCCESS) { - if (c->coll_strkeys != NULL) { - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } - if (c->coll_eitem != NULL) { - free((void *)c->coll_eitem); - c->coll_eitem = NULL; - } - } -} -#endif - -static void process_bin_bop_smget_complete(conn *c) -{ - assert(c->coll_eitem != NULL); -#ifdef JHPARK_OLD_SMGET_INTERFACE - if (c->coll_smgmode == 0) { - process_bin_bop_smget_complete_old(c); - return; - } -#endif - - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - token_t *key_tokens; - smget_result_t smres; - - smres.elem_array = (eitem **)c->coll_eitem; - smres.elem_kinfo = (smget_ehit_t *)&smres.elem_array[c->coll_rcount+c->coll_numkeys]; - smres.miss_kinfo = (smget_emis_t *)&smres.elem_kinfo[c->coll_rcount]; - - /* We don't actually receive the trailing two("\r\n") characters in binary protocol. - * We should consider this when tokenizing key strings. - */ - key_tokens = (token_t*)token_buff_get(&c->thread->token_buff, c->coll_numkeys); - if (key_tokens != NULL) { - bool must_backward_compatible = true; /* Must be backward compatible */ - ret = tokenize_mblocks(&c->memblist, c->coll_lenkeys-2, c->coll_numkeys, - KEY_MAX_LENGTH, must_backward_compatible, key_tokens); - /* ret : ENGINE_SUCCESS | ENGINE_EBADVALUE | ENGINE_ENOMEM */ - } else { - ret = ENGINE_ENOMEM; - } - if (ret == ENGINE_SUCCESS) { - if (c->coll_bkrange.to_nbkey == BKEY_NULL) { - memcpy(c->coll_bkrange.to_bkey, c->coll_bkrange.from_bkey, - (c->coll_bkrange.from_nbkey==0 ? sizeof(uint64_t) : c->coll_bkrange.from_nbkey)); - c->coll_bkrange.to_nbkey = c->coll_bkrange.from_nbkey; - } - assert(c->coll_numkeys > 0); - assert(c->coll_rcount > 0); - assert((c->coll_roffset + c->coll_rcount) <= MAX_SMGET_REQ_COUNT); - ret = mc_engine.v1->btree_elem_smget(mc_engine.v0, c, key_tokens, c->coll_numkeys, - &c->coll_bkrange, - (c->coll_efilter.ncompval==0 ? NULL : &c->coll_efilter), - c->coll_roffset, c->coll_rcount, -#ifdef JHPARK_OLD_SMGET_INTERFACE - (c->coll_smgmode == 2 ? true : false), &smres, -#else - c->coll_unique, &smres, -#endif - c->binary_header.request.vbucket); - } - - switch (ret) { - case ENGINE_SUCCESS: - { - protocol_binary_response_bop_smget* rsp = (protocol_binary_response_bop_smget*)c->wbuf; - uint32_t real_elem_hdr_size = smres.elem_count * (sizeof(uint64_t) + (3*sizeof(uint32_t))); - uint32_t real_emis_hdr_size = (smres.miss_count + smres.trim_count) * sizeof(uint32_t); - uint32_t bodylen, i; - uint64_t *bkeyptr; - uint32_t *vlenptr; - uint32_t *flagptr; - uint32_t *klenptr; - - char *resultptr = (char *)&smres.miss_kinfo[c->coll_numkeys]; - if (((long)resultptr % 8) != 0) { /* NOT aligned */ - resultptr += (8 - ((long)resultptr % 8)); - } - bkeyptr = (uint64_t *)resultptr; - vlenptr = (uint32_t *)((char*)bkeyptr + (sizeof(uint64_t) * smres.elem_count)); - flagptr = (uint32_t *)((char*)vlenptr + (sizeof(uint32_t) * smres.elem_count)); - klenptr = (uint32_t *)((char*)flagptr + (sizeof(uint32_t) * smres.elem_count)); - - bodylen = sizeof(rsp->message.body) + (real_elem_hdr_size + real_emis_hdr_size); - for (i = 0; i < smres.elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - smres.elem_array[i], &c->einfo); - bodylen += ((c->einfo.nbytes - 2) + key_tokens[smres.elem_kinfo[i].kidx].length); - bkeyptr[i] = htonll(*(uint64_t*)c->einfo.score); - vlenptr[i] = htonl(c->einfo.nbytes - 2); - flagptr[i] = smres.elem_kinfo[i].flag; - klenptr[i] = htonl(key_tokens[smres.elem_kinfo[i].kidx].length); - } - for (i = 0; i < smres.miss_count; i++) { - bodylen += key_tokens[smres.miss_kinfo[i].kidx].length; - klenptr[smres.elem_count+i] = htonl(key_tokens[smres.miss_kinfo[i].kidx].length); - } - for (i = 0; i < smres.trim_count; i++) { - bodylen += key_tokens[smres.trim_kinfo[i].kidx].length; - klenptr[smres.elem_count+smres.miss_count+i] = htonl(key_tokens[smres.trim_kinfo[i].kidx].length); - } - add_bin_header(c, 0, sizeof(rsp->message.body), 0, bodylen); - - // add the flags and count - rsp->message.body.elem_count = htonl(smres.elem_count); - rsp->message.body.miss_count = htonl(smres.miss_count); - rsp->message.body.trim_count = htonl(smres.trim_count); - add_iov(c, &rsp->message.body, sizeof(rsp->message.body)); - - // add value lengths - if (add_iov(c, (char*)bkeyptr, real_elem_hdr_size+real_emis_hdr_size) != 0) { - ret = ENGINE_ENOMEM; - } - - /* Add the data without CRLF */ - if (ret == ENGINE_SUCCESS) { - for (i = 0; i < smres.elem_count; i++) { - mc_engine.v1->get_elem_info(mc_engine.v0, c, ITEM_TYPE_BTREE, - smres.elem_array[i], &c->einfo); - if (add_iov_einfo_value_some(c, &c->einfo, c->einfo.nbytes - 2) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } - /* Add the found key */ - if (ret == ENGINE_SUCCESS) { - for (i = 0; i < smres.elem_count; i++) { - if (add_iov(c, key_tokens[smres.elem_kinfo[i].kidx].value, - key_tokens[smres.elem_kinfo[i].kidx].length) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } - /* Add the missed key */ - if (ret == ENGINE_SUCCESS) { - for (i = 0; i < smres.miss_count; i++) { - if (add_iov(c, key_tokens[smres.miss_kinfo[i].kidx].value, - key_tokens[smres.miss_kinfo[i].kidx].length) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } - /* Add the trimmed key */ - if (ret == ENGINE_SUCCESS) { - for (i = 0; i < smres.trim_count; i++) { - if (add_iov(c, key_tokens[smres.trim_kinfo[i].kidx].value, - key_tokens[smres.trim_kinfo[i].kidx].length) != 0) { - ret = ENGINE_ENOMEM; break; - } - } - } - - if (ret == ENGINE_SUCCESS) { - /* Remember this command so we can garbage collect it later */ - /* c->coll_eitem = (void *)elem_array; */ - STATS_OKS_NOKEY(c, bop_smget); - c->coll_ecount = smres.elem_count+smres.trim_count; - c->coll_op = OPERATION_BOP_SMGET; - conn_set_state(c, conn_mwrite); - } else { - STATS_CMD_NOKEY(c, bop_smget); - mc_engine.v1->btree_elem_release(mc_engine.v0, c, smres.elem_array, - smres.elem_count+smres.trim_count); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - } - } - break; - default: - STATS_CMD_NOKEY(c, bop_smget); - if (ret == ENGINE_EBADVALUE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADVALUE, 0); - else if (ret == ENGINE_EBADTYPE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADTYPE, 0); - else if (ret == ENGINE_EBADBKEY) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADBKEY, 0); -#if 0 // JHPARK_SMGET_OFFSET_HANDLING - else if (ret == ENGINE_EBKEYOOR) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBKEYOOR, 0); -#endif - else if (ret == ENGINE_ENOMEM) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - - /* free token buffer */ - if (key_tokens != NULL) { - token_buff_release(&c->thread->token_buff, key_tokens); - } - - if (ret != ENGINE_SUCCESS) { - if (c->coll_strkeys != NULL) { - /* free key string memory blocks */ - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } - if (c->coll_eitem != NULL) { - free((void *)c->coll_eitem); - c->coll_eitem = NULL; - } - } -} -#endif - -#if defined(SUPPORT_BOP_MGET) || defined(SUPPORT_BOP_SMGET) -static void process_bin_bop_nread_keys_complete(conn *c) -{ - assert(c != NULL); -#ifdef SUPPORT_BOP_MGET - if (c->coll_op == OPERATION_BOP_MGET) - process_bin_bop_mget_complete(c); -#endif -#ifdef SUPPORT_BOP_SMGET - if (c->coll_op == OPERATION_BOP_SMGET) - process_bin_bop_smget_complete(c); -#endif -} -#endif - -static void process_bin_getattr(conn *c) -{ - assert(c != NULL); - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - - if (settings.verbose > 1) { - fprintf(stderr, "<%d GETATTR ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - fprintf(stderr, "\n"); - } - - item_attr attr_data; - ENGINE_ITEM_ATTR attr_ids[ATTR_END]; - uint32_t attr_count = 0; - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->getattr(mc_engine.v0, c, key, nkey, - attr_ids, attr_count, &attr_data, - c->binary_header.request.vbucket); - if (settings.detail_enabled) { - stats_prefix_record_getattr(key, nkey); - } - - switch (ret) { - case ENGINE_SUCCESS: - { - STATS_HITS(c, getattr, key, nkey); - - protocol_binary_response_getattr* rsp = (protocol_binary_response_getattr*)c->wbuf; - rsp->message.body.flags = attr_data.flags; - rsp->message.body.expiretime = htonl(human_readable_time(attr_data.exptime)); - rsp->message.body.type = attr_data.type; - if (attr_data.type == ITEM_TYPE_LIST || attr_data.type == ITEM_TYPE_SET || - attr_data.type == ITEM_TYPE_BTREE) { - rsp->message.body.count = htonl(attr_data.count); - rsp->message.body.maxcount = htonl(attr_data.maxcount); - rsp->message.body.ovflaction = attr_data.ovflaction; - rsp->message.body.readable = attr_data.readable; - } - if (attr_data.type == ITEM_TYPE_BTREE) { - rsp->message.body.maxbkeyrange.len = attr_data.maxbkeyrange.len; - if (attr_data.maxbkeyrange.len != BKEY_NULL) { - if (attr_data.maxbkeyrange.len == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, attr_data.maxbkeyrange.val, sizeof(uint64_t)); - bkey_temp = htonll(bkey_temp); - memcpy(rsp->message.body.maxbkeyrange.val, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)rsp->message.body.maxbkeyrange.val = htonll(*(uint64_t*)attr_data.maxbkeyrange.val); - } else { - memcpy(rsp->message.body.maxbkeyrange.val, attr_data.maxbkeyrange.val, attr_data.maxbkeyrange.len); - } - } - } - write_bin_response(c, &rsp->message.body, 0, 0, sizeof(rsp->message.body)); - } - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, getattr, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, getattr); - if (ret == ENGINE_EBADATTR) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADATTR, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static void process_bin_setattr(conn *c) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - char *key = binary_get_key(c); - int nkey = c->binary_header.request.keylen; - - item_attr attr_data; - ENGINE_ITEM_ATTR attr_ids[ATTR_END]; - uint32_t attr_count = 0; - - /* fix byteorder in the request */ - protocol_binary_request_setattr* req = binary_get_request(c); - - if (req->message.body.expiretime_f != 0) { - attr_ids[attr_count++] = ATTR_EXPIRETIME; - req->message.body.expiretime = ntohl(req->message.body.expiretime); - attr_data.exptime = realtime(req->message.body.expiretime); - } - if (req->message.body.maxcount_f != 0) { - attr_ids[attr_count++] = ATTR_MAXCOUNT; - req->message.body.maxcount = ntohl(req->message.body.maxcount); - attr_data.maxcount = req->message.body.maxcount; - } - if (req->message.body.maxbkeyrange.len != BKEY_NULL) { - attr_ids[attr_count++] = ATTR_MAXBKEYRANGE; - if (req->message.body.maxbkeyrange.len == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, req->message.body.maxbkeyrange.val, sizeof(uint64_t)); - bkey_temp = ntohll(bkey_temp); - memcpy(attr_data.maxbkeyrange.val, (unsigned char*)&bkey_temp, sizeof(uint64_t)); - //*(uint64_t*)attr_data.maxbkeyrange.val = ntohll(*(uint64_t*)req->message.body.maxbkeyrange.val); - } else { - memcpy(attr_data.maxbkeyrange.val, req->message.body.maxbkeyrange.val, req->message.body.maxbkeyrange.len); - } - attr_data.maxbkeyrange.len = req->message.body.maxbkeyrange.len; - } - if (req->message.body.ovflaction != 0) { - attr_ids[attr_count++] = ATTR_OVFLACTION; - attr_data.ovflaction = req->message.body.ovflaction; - } - if (req->message.body.readable != 0) { - attr_ids[attr_count++] = ATTR_READABLE; - attr_data.readable = req->message.body.readable; - } - - if (settings.verbose > 1) { - fprintf(stderr, "<%d SETATTR ", c->sfd); - for (int ii = 0; ii < nkey; ++ii) { - fprintf(stderr, "%c", key[ii]); - } - for (int ii = 0; ii < attr_count; ii++) { - if (attr_ids[ii] == ATTR_EXPIRETIME) - fprintf(stderr, " expiretime=%d", (int32_t)attr_data.exptime); - else if (attr_ids[ii] == ATTR_MAXCOUNT) - fprintf(stderr, " maxcount=%d", attr_data.maxcount); - else if (attr_ids[ii] == ATTR_OVFLACTION) - fprintf(stderr, " ovflaction=%s", get_ovflaction_str(attr_data.ovflaction)); - else if (attr_ids[ii] == ATTR_READABLE) - fprintf(stderr, " readable=%s", (attr_data.readable ? "on" : "off")); - else if (attr_ids[ii] == ATTR_MAXBKEYRANGE) { - if (attr_data.maxbkeyrange.len == 0) { - uint64_t bkey_temp; - memcpy((unsigned char*)&bkey_temp, attr_data.maxbkeyrange.val, sizeof(uint64_t)); - fprintf(stderr, "maxbkeyrange=%"PRIu64, bkey_temp); - //fprintf(stderr, "maxbkeyrange=%"PRIu64, *(uint64_t*)attr_data.maxbkeyrange.val); - } else { - char buffer[MAX_BKEY_LENG*2+4]; - safe_hexatostr(attr_data.maxbkeyrange.val, attr_data.maxbkeyrange.len, buffer); - fprintf(stderr, "maxbkeyrange=0x%s", buffer); - } - } - } - fprintf(stderr, "\n"); - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->setattr(mc_engine.v0, c, key, nkey, - attr_ids, attr_count, &attr_data, - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_setattr(key, nkey); - } - - switch (ret) { - case ENGINE_SUCCESS: - STATS_HITS(c, setattr, key, nkey); - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_KEY_ENOENT: - STATS_MISSES(c, setattr, key, nkey); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - default: - STATS_CMD_NOKEY(c, setattr); - if (ret == ENGINE_EBADATTR) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADATTR, 0); - else if (ret == ENGINE_EBADVALUE) - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EBADVALUE, 0); - else - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } -} - -static bool binary_response_handler(const void *key, uint16_t keylen, - const void *ext, uint8_t extlen, - const void *body, uint32_t bodylen, - uint8_t datatype, uint16_t status, - uint64_t cas, const void *cookie) -{ - conn *c = (conn*)cookie; - /* Look at append_bin_stats */ - size_t needed = keylen + extlen + bodylen + sizeof(protocol_binary_response_header); - if (!grow_dynamic_buffer(c, needed)) { - if (settings.verbose > 0) { - mc_logger->log(EXTENSION_LOG_INFO, c, - "<%d ERROR: Failed to allocate memory for response\n", c->sfd); - } - return false; - } - - char *buf = c->dynamic_buffer.buffer + c->dynamic_buffer.offset; - protocol_binary_response_header header = { - .response.magic = (uint8_t)PROTOCOL_BINARY_RES, - .response.opcode = c->binary_header.request.opcode, - .response.keylen = (uint16_t)htons(keylen), - .response.extlen = extlen, - .response.datatype = datatype, - .response.status = (uint16_t)htons(status), - .response.bodylen = htonl(bodylen + keylen + extlen), - .response.opaque = c->opaque, - .response.cas = htonll(cas), - }; - - memcpy(buf, header.bytes, sizeof(header.response)); - buf += sizeof(header.response); - - if (extlen > 0) { - memcpy(buf, ext, extlen); - buf += extlen; - } - - if (keylen > 0) { - memcpy(buf, key, keylen); - buf += keylen; - } - - if (bodylen > 0) { - memcpy(buf, body, bodylen); - } - - c->dynamic_buffer.offset += needed; - - return true; -} - -static void process_bin_unknown_packet(conn *c) -{ - void *packet = c->rcurr - (c->binary_header.request.bodylen + - sizeof(c->binary_header)); - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->unknown_command(mc_engine.v0, c, packet, - binary_response_handler); - - switch (ret) { - case ENGINE_SUCCESS: - write_and_free(c, c->dynamic_buffer.buffer, c->dynamic_buffer.offset); - c->dynamic_buffer.buffer = NULL; - break; - case ENGINE_ENOTSUP: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_UNKNOWN_COMMAND, 0); - break; - default: - /* FATAL ERROR, shut down connection */ - conn_set_state(c, conn_closing); - } -} - -static void dispatch_bin_command(conn *c) -{ - int protocol_error = 0; - int extlen = c->binary_header.request.extlen; - int keylen = c->binary_header.request.keylen; - uint32_t bodylen = c->binary_header.request.bodylen; - - if (settings.require_sasl && !authenticated(c)) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_AUTH_ERROR, 0); - c->write_and_go = conn_closing; - return; - } - - MEMCACHED_PROCESS_COMMAND_START(c->sfd, c->rcurr, c->rbytes); - c->noreply = true; - - /* binprot supports 16bit keys, but internals are still 8bit */ - if (keylen > KEY_MAX_LENGTH) { - handle_binary_protocol_error(c); - return; - } - - switch (c->cmd) { - case PROTOCOL_BINARY_CMD_SETQ: - c->cmd = PROTOCOL_BINARY_CMD_SET; - break; - case PROTOCOL_BINARY_CMD_ADDQ: - c->cmd = PROTOCOL_BINARY_CMD_ADD; - break; - case PROTOCOL_BINARY_CMD_REPLACEQ: - c->cmd = PROTOCOL_BINARY_CMD_REPLACE; - break; - case PROTOCOL_BINARY_CMD_DELETEQ: - c->cmd = PROTOCOL_BINARY_CMD_DELETE; - break; - case PROTOCOL_BINARY_CMD_INCREMENTQ: - c->cmd = PROTOCOL_BINARY_CMD_INCREMENT; - break; - case PROTOCOL_BINARY_CMD_DECREMENTQ: - c->cmd = PROTOCOL_BINARY_CMD_DECREMENT; - break; - case PROTOCOL_BINARY_CMD_QUITQ: - c->cmd = PROTOCOL_BINARY_CMD_QUIT; - break; - case PROTOCOL_BINARY_CMD_FLUSHQ: - c->cmd = PROTOCOL_BINARY_CMD_FLUSH; - break; - case PROTOCOL_BINARY_CMD_APPENDQ: - c->cmd = PROTOCOL_BINARY_CMD_APPEND; - break; - case PROTOCOL_BINARY_CMD_PREPENDQ: - c->cmd = PROTOCOL_BINARY_CMD_PREPEND; - break; - case PROTOCOL_BINARY_CMD_GETQ: - c->cmd = PROTOCOL_BINARY_CMD_GET; - break; - case PROTOCOL_BINARY_CMD_GETKQ: - c->cmd = PROTOCOL_BINARY_CMD_GETK; - break; - case PROTOCOL_BINARY_CMD_LOP_INSERTQ: - c->cmd = PROTOCOL_BINARY_CMD_LOP_INSERT; - break; - case PROTOCOL_BINARY_CMD_LOP_DELETEQ: - c->cmd = PROTOCOL_BINARY_CMD_LOP_DELETE; - break; - case PROTOCOL_BINARY_CMD_SOP_INSERTQ: - c->cmd = PROTOCOL_BINARY_CMD_SOP_INSERT; - break; - case PROTOCOL_BINARY_CMD_SOP_DELETEQ: - c->cmd = PROTOCOL_BINARY_CMD_SOP_DELETE; - break; - case PROTOCOL_BINARY_CMD_BOP_INSERTQ: - c->cmd = PROTOCOL_BINARY_CMD_BOP_INSERT; - break; - case PROTOCOL_BINARY_CMD_BOP_UPSERTQ: - c->cmd = PROTOCOL_BINARY_CMD_BOP_UPSERT; - break; - case PROTOCOL_BINARY_CMD_BOP_UPDATEQ: - c->cmd = PROTOCOL_BINARY_CMD_BOP_UPDATE; - break; - case PROTOCOL_BINARY_CMD_BOP_DELETEQ: - c->cmd = PROTOCOL_BINARY_CMD_BOP_DELETE; - break; - default: - c->noreply = false; - } - - switch (c->cmd) { - case PROTOCOL_BINARY_CMD_VERSION: - if (extlen == 0 && keylen == 0 && bodylen == 0) { - write_bin_response(c, VERSION, 0, 0, strlen(VERSION)); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_FLUSH: - if (keylen == 0 && bodylen == extlen && (extlen == 0 || extlen == 4)) { - bin_read_key(c, bin_read_flush_exptime, extlen); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_FLUSH_PREFIX: - if (keylen > 0 && bodylen == extlen && (extlen == 0 || extlen == 4)) { - bin_read_key(c, bin_read_flush_prefix_exptime, extlen); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_NOOP: - if (extlen == 0 && keylen == 0 && bodylen == 0) { - write_bin_response(c, NULL, 0, 0, 0); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_SET: /* FALLTHROUGH */ - case PROTOCOL_BINARY_CMD_ADD: /* FALLTHROUGH */ - case PROTOCOL_BINARY_CMD_REPLACE: - if (extlen == 8 && keylen != 0 && bodylen >= (keylen + 8)) { - bin_read_key(c, bin_reading_set_header, 8); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_GETQ: /* FALLTHROUGH */ - case PROTOCOL_BINARY_CMD_GET: /* FALLTHROUGH */ - case PROTOCOL_BINARY_CMD_GETKQ: /* FALLTHROUGH */ - case PROTOCOL_BINARY_CMD_GETK: - if (extlen == 0 && bodylen == keylen && keylen > 0) { - bin_read_key(c, bin_reading_get_key, 0); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_DELETE: - if (keylen > 0 && extlen == 0 && bodylen == keylen) { - bin_read_key(c, bin_reading_del_header, extlen); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_INCREMENT: - case PROTOCOL_BINARY_CMD_DECREMENT: - if (keylen > 0 && extlen == 20 && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_incr_header, 20); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_APPEND: - case PROTOCOL_BINARY_CMD_PREPEND: - if (keylen > 0 && extlen == 0) { - bin_read_key(c, bin_reading_set_header, 0); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_STAT: - if (extlen == 0) { - bin_read_key(c, bin_reading_stat, 0); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_QUIT: - if (keylen == 0 && extlen == 0 && bodylen == 0) { - write_bin_response(c, NULL, 0, 0, 0); - c->write_and_go = conn_closing; - if (c->noreply) { - conn_set_state(c, conn_closing); - } - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_GETATTR: - if (keylen > 0 && extlen == (sizeof(bkey_t)+24) && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_getattr, (sizeof(bkey_t)+24)); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_SETATTR: - if (keylen > 0 && extlen == (sizeof(bkey_t)+12) && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_setattr, (sizeof(bkey_t)+12)); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_LOP_CREATE: - if (keylen > 0 && extlen == 16 && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_lop_create, 16); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_LOP_INSERT: - if (keylen > 0 && extlen == 20 && bodylen > (keylen + extlen)) { - bin_read_key(c, bin_reading_lop_prepare_nread, 20); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_LOP_DELETE: - if (keylen > 0 && extlen == 12 && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_lop_delete, 12); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_LOP_GET: - if (keylen > 0 && extlen == 12 && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_lop_get, 12); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_SOP_CREATE: - if (keylen > 0 && extlen == 16 && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_sop_create, 16); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_SOP_INSERT: - if (keylen > 0 && extlen == 16 && bodylen > (keylen + extlen)) { - bin_read_key(c, bin_reading_sop_prepare_nread, 16); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_SOP_DELETE: - if (keylen > 0 && extlen == 4 && bodylen > (keylen + extlen)) { - bin_read_key(c, bin_reading_sop_prepare_nread, 4); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_SOP_EXIST: - if (keylen > 0 && extlen == 0 && bodylen > (keylen + extlen)) { - bin_read_key(c, bin_reading_sop_prepare_nread, 0); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_SOP_GET: - if (keylen > 0 && extlen == 8 && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_sop_get, 8); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_BOP_CREATE: - if (keylen > 0 && extlen == 16 && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_bop_create, 16); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_BOP_INSERT: - case PROTOCOL_BINARY_CMD_BOP_UPSERT: - if (keylen > 0 && extlen == (MAX_BKEY_LENG+1+MAX_EFLAG_LENG+1+16) && bodylen > (keylen + extlen)) { - bin_read_key(c, bin_reading_bop_prepare_nread, (MAX_BKEY_LENG+1+MAX_EFLAG_LENG+1+16)); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_BOP_UPDATE: - if (keylen > 0 && extlen == (MAX_BKEY_LENG+1+MAX_EFLAG_LENG+1+8) && bodylen > (keylen + extlen)) { - bin_read_key(c, bin_reading_bop_update_prepare_nread, (MAX_BKEY_LENG+1+MAX_EFLAG_LENG+1+8)); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_BOP_DELETE: - if (keylen > 0 && extlen == (sizeof(bkey_range)+sizeof(eflag_filter)+8) && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_bop_delete, (sizeof(bkey_range)+sizeof(eflag_filter)+8)); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_BOP_GET: - if (keylen > 0 && extlen == (sizeof(bkey_range)+sizeof(eflag_filter)+12) && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_bop_get, (sizeof(bkey_range)+sizeof(eflag_filter)+12)); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_BOP_COUNT: - if (keylen > 0 && extlen == (sizeof(bkey_range)+sizeof(eflag_filter)) && bodylen == (keylen + extlen)) { - bin_read_key(c, bin_reading_bop_count, (sizeof(bkey_range)+sizeof(eflag_filter))); - } else { - protocol_error = 1; - } - break; -#if defined(SUPPORT_BOP_MGET) || defined(SUPPORT_BOP_SMGET) -#ifdef SUPPORT_BOP_MGET - case PROTOCOL_BINARY_CMD_BOP_MGET: -#endif -#ifdef SUPPORT_BOP_SMGET - case PROTOCOL_BINARY_CMD_BOP_SMGET: -#endif - if (keylen > 0 && extlen == (sizeof(bkey_range)+sizeof(eflag_filter)+12) && bodylen > (keylen + extlen)) { - bin_read_key(c, bin_reading_bop_prepare_nread_keys, (sizeof(bkey_range)+sizeof(eflag_filter)+12)); - } else { - protocol_error = 1; - } - break; -#endif -#ifdef SASL_ENABLED - case PROTOCOL_BINARY_CMD_SASL_LIST_MECHS: - if (extlen == 0 && keylen == 0 && bodylen == 0) { - bin_list_sasl_mechs(c); - } else { - protocol_error = 1; - } - break; - case PROTOCOL_BINARY_CMD_SASL_AUTH: - case PROTOCOL_BINARY_CMD_SASL_STEP: - if (extlen == 0 && keylen != 0) { - bin_read_key(c, bin_reading_sasl_auth, 0); - } else { - protocol_error = 1; - } - break; -#endif - default: - if (mc_engine.v1->unknown_command == NULL) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_UNKNOWN_COMMAND, bodylen); - } else { - bin_read_chunk(c, bin_reading_packet, c->binary_header.request.bodylen); - } - } - - if (protocol_error) - handle_binary_protocol_error(c); -} - -static void process_bin_update(conn *c) -{ - assert(c != NULL); - protocol_binary_request_set* req = binary_get_request(c); - item *it; - char *key = binary_get_key(c); - uint32_t nkey = c->binary_header.request.keylen; - uint32_t vlen = 0; - - /* fix byteorder in the request */ - req->message.body.flags = req->message.body.flags; - req->message.body.expiration = ntohl(req->message.body.expiration); - - if (nkey + c->binary_header.request.extlen <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - (nkey + c->binary_header.request.extlen); - } else { - handle_binary_protocol_error(c); - return; - } - - if (settings.verbose > 1) { - char buffer[1024]; - const char *prefix; - if (c->cmd == PROTOCOL_BINARY_CMD_ADD) { - prefix = "ADD"; - } else if (c->cmd == PROTOCOL_BINARY_CMD_SET) { - prefix = "SET"; - } else { - prefix = "REPLACE"; - } - - size_t nw; - nw = key_to_printable_buffer(buffer, sizeof(buffer), c->sfd, true, - prefix, key, nkey); - - if (nw != -1) { - if (snprintf(buffer + nw, sizeof(buffer) - nw, - " Value len is %d\n", vlen)) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, "%s", buffer); - } - } - } - - if (settings.detail_enabled) { - stats_prefix_record_set(key, nkey); - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->allocate(mc_engine.v0, c, &it, key, nkey, vlen+2, - req->message.body.flags, - realtime(req->message.body.expiration), - c->binary_header.request.cas); - if (ret == ENGINE_SUCCESS && !mc_engine.v1->get_item_info(mc_engine.v0, - c, it, &c->hinfo)) { - mc_engine.v1->release(mc_engine.v0, c, it); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINTERNAL, 0); - return; - } - - switch (ret) { - case ENGINE_SUCCESS: - if (c->cmd == PROTOCOL_BINARY_CMD_ADD) - c->store_op = OPERATION_ADD; - else if (c->cmd == PROTOCOL_BINARY_CMD_SET) - c->store_op = OPERATION_SET; - else if (c->cmd == PROTOCOL_BINARY_CMD_REPLACE) - c->store_op = OPERATION_REPLACE; - else - assert(0); - - if (c->binary_header.request.cas != 0) { - c->store_op = OPERATION_CAS; - } - - c->item = it; - ritem_set_first(c, CONN_RTYPE_HINFO, vlen); - conn_set_state(c, conn_nread); - c->substate = bin_read_set_value; - break; - case ENGINE_E2BIG: - case ENGINE_ENOMEM: - if (ret == ENGINE_E2BIG) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_E2BIG, vlen); - } else { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, vlen); - } - - /* - * Avoid stale data persisting in cache because we failed alloc. - * Unacceptable for SET (but only if cas matches). - * Anywhere else too? - */ - if (c->cmd == PROTOCOL_BINARY_CMD_SET) { - /* set temporarily noreply for the ASYNC interface if it isn't set. */ - bool set_temporary_noreply = false; - if (!c->noreply) { - c->noreply = true; - set_temporary_noreply = true; - } - mc_engine.v1->remove(mc_engine.v0, c, key, nkey, - ntohll(req->message.header.request.cas), - c->binary_header.request.vbucket); - if (set_temporary_noreply) { - c->noreply = false; - } - } - break; - default: - handle_unexpected_errorcode_bin(c, __func__, ret, vlen); - } -} - -static void process_bin_append_prepend(conn *c) -{ - assert(c != NULL); - item *it; - char *key = binary_get_key(c); - uint32_t nkey = c->binary_header.request.keylen; - uint32_t vlen = 0; - - if (nkey <= c->binary_header.request.bodylen) { - vlen = c->binary_header.request.bodylen - nkey; - } else { - handle_binary_protocol_error(c); - return; - } - - if (settings.detail_enabled) { - stats_prefix_record_set(key, nkey); - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->allocate(mc_engine.v0, c, &it, key, nkey, vlen+2, - 0, 0, c->binary_header.request.cas); - if (ret == ENGINE_SUCCESS && !mc_engine.v1->get_item_info(mc_engine.v0, - c, it, &c->hinfo)) { - mc_engine.v1->release(mc_engine.v0, c, it); - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINTERNAL, 0); - return; - } - - switch (ret) { - case ENGINE_SUCCESS: - if (c->cmd == PROTOCOL_BINARY_CMD_APPEND) - c->store_op = OPERATION_APPEND; - else if (c->cmd == PROTOCOL_BINARY_CMD_PREPEND) - c->store_op = OPERATION_PREPEND; - else - assert(0); - - c->item = it; - ritem_set_first(c, CONN_RTYPE_HINFO, vlen); - conn_set_state(c, conn_nread); - c->substate = bin_read_set_value; - break; - case ENGINE_E2BIG: - case ENGINE_ENOMEM: - if (ret == ENGINE_E2BIG) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_E2BIG, vlen); - } else { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_ENOMEM, vlen); - } - break; - default: - handle_unexpected_errorcode_bin(c, __func__, ret, vlen); - } -} - -static void process_bin_flush(conn *c) -{ - protocol_binary_request_flush* req = binary_get_request(c); - time_t exptime = 0; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - if (c->binary_header.request.extlen == sizeof(req->message.body)) { - exptime = ntohl(req->message.body.expiration); - if (exptime < 0) { - ret = ENGINE_EINVAL; - } - } - - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, - "%d: flush %ld", c->sfd, (long)exptime); - } - - if (ret == ENGINE_SUCCESS) { - ret = mc_engine.v1->flush(mc_engine.v0, c, NULL, -1, realtime(exptime)); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - } - - if (ret == ENGINE_SUCCESS) { - write_bin_response(c, NULL, 0, 0, 0); - } else { - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - STATS_CMD_NOKEY(c, flush); -} - -static void process_bin_flush_prefix(conn *c) -{ - protocol_binary_request_flush_prefix *req = binary_get_request(c); - char *prefix = binary_get_key(c); - size_t nprefix = c->binary_header.request.keylen; - time_t exptime = 0; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - if (c->binary_header.request.extlen == sizeof(req->message.body)) { - exptime = ntohl(req->message.body.expiration); - if (exptime < 0) { - ret = ENGINE_EINVAL; - } - } - - if (settings.verbose > 1) { - char buffer[1024]; - if (key_to_printable_buffer(buffer, sizeof(buffer), c->sfd, true, "FLUSH_PREFIX", prefix, nprefix) != -1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, "%s %ld\n", buffer, (long)exptime); - } - } - - if (nprefix == 6 && strncmp(prefix, "", 6) == 0) { - /* flush null prefix */ - prefix = NULL; - nprefix = 0; - } - - if (ret == ENGINE_SUCCESS) { - ret = mc_engine.v1->flush(mc_engine.v0, c, prefix, nprefix, realtime(exptime)); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - } - - if (settings.detail_enabled) { - if (ret == ENGINE_SUCCESS || ret == ENGINE_PREFIX_ENOENT) { - if (stats_prefix_delete(prefix, nprefix) == 0) { /* found */ - ret = ENGINE_SUCCESS; - } - } - } - - if (ret == ENGINE_SUCCESS) { - write_bin_response(c, NULL, 0, 0, 0); - } else if (ret == ENGINE_PREFIX_ENOENT) { - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_PREFIX_ENOENT, 0); - } else { - handle_unexpected_errorcode_bin(c, __func__, ret, 0); - } - STATS_CMD_NOKEY(c, flush_prefix); -} - -static void process_bin_delete(conn *c) -{ - assert(c != NULL); - protocol_binary_request_delete* req = binary_get_request(c); - char* key = binary_get_key(c); - size_t nkey = c->binary_header.request.keylen; - - if (settings.verbose > 1) { - char buffer[1024]; - if (key_to_printable_buffer(buffer, sizeof(buffer), c->sfd, true, - "DELETE", key, nkey) != -1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, "%s\n", buffer); - } - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->remove(mc_engine.v0, c, key, nkey, - ntohll(req->message.header.request.cas), - c->binary_header.request.vbucket); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - if (settings.detail_enabled) { - stats_prefix_record_delete(key, nkey); - } - - switch (ret) { - case ENGINE_SUCCESS: - write_bin_response(c, NULL, 0, 0, 0); - break; - case ENGINE_KEY_EEXISTS: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_EEXISTS, 0); - break; - case ENGINE_KEY_ENOENT: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_KEY_ENOENT, 0); - break; - case ENGINE_NOT_MY_VBUCKET: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_NOT_MY_VBUCKET, 0); - break; - default: - write_bin_packet(c, PROTOCOL_BINARY_RESPONSE_EINVAL, 0); - } -} - -static void complete_nread_binary(conn *c) -{ - assert(c != NULL); - assert(c->cmd >= 0); - - switch (c->substate) { - case bin_reading_set_header: - if (c->cmd == PROTOCOL_BINARY_CMD_APPEND || - c->cmd == PROTOCOL_BINARY_CMD_PREPEND) { - process_bin_append_prepend(c); - } else { - process_bin_update(c); - } - break; - case bin_read_set_value: - complete_update_bin(c); - break; - case bin_reading_get_key: - process_bin_get(c); - break; - case bin_reading_stat: - process_bin_stat(c); - break; - case bin_reading_del_header: - process_bin_delete(c); - break; - case bin_reading_incr_header: - complete_incr_bin(c); - break; - case bin_read_flush_exptime: - process_bin_flush(c); - break; - case bin_read_flush_prefix_exptime: - process_bin_flush_prefix(c); - break; - case bin_reading_sasl_auth: - process_bin_sasl_auth(c); - break; - case bin_reading_sasl_auth_data: - process_bin_complete_sasl_auth(c); - break; - case bin_reading_getattr: - process_bin_getattr(c); - break; - case bin_reading_setattr: - process_bin_setattr(c); - break; - case bin_reading_lop_create: - process_bin_lop_create(c); - break; - case bin_reading_lop_prepare_nread: - process_bin_lop_prepare_nread(c); - break; - case bin_reading_lop_nread_complete: - process_bin_lop_nread_complete(c); - break; - case bin_reading_lop_delete: - process_bin_lop_delete(c); - break; - case bin_reading_lop_get: - process_bin_lop_get(c); - break; - case bin_reading_sop_create: - process_bin_sop_create(c); - break; - case bin_reading_sop_prepare_nread: - process_bin_sop_prepare_nread(c); - break; - case bin_reading_sop_nread_complete: - process_bin_sop_nread_complete(c); - break; - case bin_reading_sop_get: - process_bin_sop_get(c); - break; - case bin_reading_bop_create: - process_bin_bop_create(c); - break; - case bin_reading_bop_prepare_nread: - process_bin_bop_prepare_nread(c); - break; - case bin_reading_bop_nread_complete: - process_bin_bop_nread_complete(c); - break; - case bin_reading_bop_update_prepare_nread: - process_bin_bop_update_prepare_nread(c); - break; - case bin_reading_bop_update_nread_complete: - process_bin_bop_update_complete(c); - break; - case bin_reading_bop_delete: - process_bin_bop_delete(c); - break; - case bin_reading_bop_get: - process_bin_bop_get(c); - break; - case bin_reading_bop_count: - process_bin_bop_count(c); - break; -#if defined(SUPPORT_BOP_MGET) || defined(SUPPORT_BOP_SMGET) - case bin_reading_bop_prepare_nread_keys: - process_bin_bop_prepare_nread_keys(c); - break; - case bin_reading_bop_nread_keys_complete: - process_bin_bop_nread_keys_complete(c); - break; -#endif - case bin_reading_packet: - process_bin_unknown_packet(c); - break; - default: - mc_logger->log(EXTENSION_LOG_WARNING, c, - "Not handling substate %d\n", c->substate); - abort(); - } -} - -static int try_read_command_binary(conn *c) -{ - /* Do we have the complete packet header? */ - if (c->rbytes < sizeof(c->binary_header)) { - /* need more data! */ - return 0; - } - -#ifdef NEED_ALIGN - if (((long)(c->rcurr)) % 8 != 0) { - /* must realign input buffer */ - memmove(c->rbuf, c->rcurr, c->rbytes); - c->rcurr = c->rbuf; - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, - "%d: Realign input buffer\n", c->sfd); - } - } -#endif - - protocol_binary_request_header* req; - req = (protocol_binary_request_header*)c->rcurr; - - if (settings.verbose > 1) { - /* Dump the packet before we convert it to host order */ - char buffer[1024]; - size_t nw; - nw = bytes_to_output_string(buffer, sizeof(buffer), c->sfd, - true, "Read binary protocol data:", - (const char*)req->bytes, - sizeof(req->bytes)); - if (nw != -1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, "%s", buffer); - } - } - - c->binary_header = *req; - c->binary_header.request.keylen = ntohs(req->request.keylen); - c->binary_header.request.bodylen = ntohl(req->request.bodylen); - c->binary_header.request.vbucket = ntohs(req->request.vbucket); - c->binary_header.request.cas = ntohll(req->request.cas); - - if (c->binary_header.request.magic != PROTOCOL_BINARY_REQ) { - if (settings.verbose) { - if (c->binary_header.request.magic != PROTOCOL_BINARY_RES) { - mc_logger->log(EXTENSION_LOG_INFO, c, - "%d: Invalid magic: %x\n", c->sfd, - c->binary_header.request.magic); - } else { - mc_logger->log(EXTENSION_LOG_INFO, c, - "%d: ERROR: Unsupported response packet received: %u\n", - c->sfd, (unsigned int)c->binary_header.request.opcode); - } - } - conn_set_state(c, conn_closing); - return 1; - } - - c->msgcurr = 0; - c->msgused = 0; - c->iovused = 0; - int ret = add_msghdr(c); - assert(ret == 0); - - c->cmd = c->binary_header.request.opcode; - c->keylen = c->binary_header.request.keylen; - c->opaque = c->binary_header.request.opaque; - /* clear the returned cas value */ - c->cas = 0; - - dispatch_bin_command(c); - - c->rbytes -= sizeof(c->binary_header); - c->rcurr += sizeof(c->binary_header); - - assert(c->rcurr <= (c->rbuf + c->rsize)); - - return 1; -} - -static void reset_cmd_handler(conn *c) -{ - c->ascii_cmd = NULL; - c->cmd = -1; - c->substate = bin_no_state; - if (c->item != NULL) { - mc_engine.v1->release(mc_engine.v0, c, c->item); - c->item = NULL; - } -#ifdef DETECT_LONG_QUERY - if (c->lq_result) { - lqdetect_result_release(c->lq_result); - c->lq_result = NULL; - } -#endif - if (c->coll_eitem != NULL) { - conn_coll_eitem_free(c); - c->coll_eitem = NULL; - } - if (c->coll_strkeys != NULL) { - assert(c->coll_strkeys == (void*)&c->memblist); - mblck_list_free(&c->thread->mblck_pool, &c->memblist); - c->coll_strkeys = NULL; - } - conn_shrink(c); - if (c->rbytes > 0) { - conn_set_state(c, conn_parse_cmd); - } else { - conn_set_state(c, conn_waiting); - } -} - -static bool ascii_response_handler(const void *cookie, - int nbytes, const char *dta) -{ - conn *c = (conn*)cookie; - if (!grow_dynamic_buffer(c, nbytes)) { - if (settings.verbose > 0) { - mc_logger->log(EXTENSION_LOG_INFO, c, - "<%d ERROR: Failed to allocate memory for response\n", c->sfd); - } - return false; - } - - char *buf = c->dynamic_buffer.buffer + c->dynamic_buffer.offset; - memcpy(buf, dta, nbytes); - c->dynamic_buffer.offset += nbytes; - return true; -} - -static void complete_nread_ascii(conn *c) -{ - if (c->ascii_cmd != NULL) { - if (!c->ascii_cmd->execute(c->ascii_cmd->cookie, c, 0, NULL, - ascii_response_handler)) { - conn_set_state(c, conn_closing); - } else { - if (c->dynamic_buffer.buffer != NULL) { - write_and_free(c, c->dynamic_buffer.buffer, - c->dynamic_buffer.offset); - c->dynamic_buffer.buffer = NULL; - } else { - conn_set_state(c, conn_new_cmd); - } - } - } else { - complete_update_ascii(c); - } -} - -static void complete_nread(conn *c) -{ - assert(c != NULL); - assert(c->protocol == ascii_prot || c->protocol == binary_prot); - - if (c->protocol == ascii_prot) { - complete_nread_ascii(c); - } else { - complete_nread_binary(c); - } -} - -#define COMMAND_TOKEN 0 -#define SUBCOMMAND_TOKEN 1 -#define PREFIX_TOKEN 1 -#define KEY_TOKEN 1 -#define LOP_KEY_TOKEN 2 -#define SOP_KEY_TOKEN 2 -#define MOP_KEY_TOKEN 2 -#define BOP_KEY_TOKEN 2 - -#define MAX_TOKENS 30 - -static void -print_invalid_command(conn *c, token_t *tokens, const size_t ntokens) -{ - /* To understand this function's implementation, - * You must know how the command is tokenized. - * See tokenize_command(). - */ - if (ntokens >= 2) { - int i; - /* make single string */ - for (i = 0; i < (ntokens-2); i++) { - tokens[i].value[tokens[i].length] = ' '; - } - mc_logger->log(EXTENSION_LOG_INFO, c, "[%s] INVALID_COMMAND: %s\n", - c->client_ip, tokens[0].value); - /* restore the tokens */ - for (i = 0; i < (ntokens-2); i++) { - tokens[i].value[tokens[i].length] = '\0'; - } - } -} - -/* set up a connection to write a buffer then free it, used for stats */ -static void write_and_free(conn *c, char *buf, int bytes) -{ - if (buf) { - c->write_and_free = buf; - c->wcurr = buf; - c->wbytes = bytes; - conn_set_state(c, conn_write); - c->write_and_go = conn_new_cmd; - } else { - out_string(c, "SERVER_ERROR out of memory writing stats"); - } -} - -#ifdef JHPARK_OLD_SMGET_INTERFACE -static inline int set_smget_mode_maybe(conn *c, token_t *tokens, size_t ntokens) -{ - int mode_index = ntokens - 2; - int mode_value = 0; - - if (tokens[mode_index].value) { - if (strcmp(tokens[mode_index].value, "duplicate") == 0) - mode_value = 1; - else if (strcmp(tokens[mode_index].value, "unique") == 0) - mode_value = 2; - } - return mode_value; -} -#else -static inline bool set_unique_maybe(conn *c, token_t *tokens, size_t ntokens) -{ - int unique_index = ntokens - 2; - - if (tokens[unique_index].value - && strcmp(tokens[unique_index].value, "unique") == 0) - return true; - else - return false; -} -#endif - -static inline void set_noreply_maybe(conn *c, token_t *tokens, size_t ntokens) -{ - char *token_value = tokens[ntokens-2].value; - - /* - NOTE: this function is not the first place where we are going to - send the reply. We could send it instead from process_command() - if the request line has wrong number of tokens. However parsing - malformed line for "noreply" option is not reliable anyway, so - it can't be helped. - */ - if (token_value && strcmp(token_value, "noreply") == 0) { - c->noreply = true; - } else { - c->noreply = false; - } -} - -static inline void set_pipe_noreply_maybe(conn *c, token_t *tokens, size_t ntokens) -{ - char *token_value = tokens[ntokens-2].value; - - /* - NOTE: this function is not the first place where we are going to - send the reply. We could send it instead from process_command() - if the request line has wrong number of tokens. However parsing - malformed line for "noreply" or "pipe" option is not reliable anyway, - so it can't be helped. - */ - if (token_value) { - if (strcmp(token_value, "noreply") == 0) { - c->noreply = true; - } else if (strcmp(token_value, "pipe") == 0) { - c->noreply = true; - if (c->pipe_state == PIPE_STATE_OFF) /* first pipe */ - c->pipe_state = PIPE_STATE_ON; - } else { - c->noreply = false; - } - } else { - c->noreply = false; - } -} - -static inline void set_pipe_maybe(conn *c, token_t *tokens, size_t ntokens) -{ - char *token_value = tokens[ntokens-2].value; - - /* - NOTE: this function is not the first place where we are going to - send the reply. We could send it instead from process_command() - if the request line has wrong number of tokens. However parsing - malformed line for "pipe" option is not reliable anyway, - so it can't be helped. - */ - if (token_value && strcmp(token_value, "pipe") == 0) { - c->noreply = true; - if (c->pipe_state == PIPE_STATE_OFF) /* first pipe */ - c->pipe_state = PIPE_STATE_ON; - } else { - c->noreply = false; - } -} - -static bool check_and_handle_pipe_state(conn *c) -{ - if (c->pipe_state == PIPE_STATE_OFF || c->pipe_state == PIPE_STATE_ON) { - return true; - } else { - assert(c->pipe_state == PIPE_STATE_ERR_CFULL || - c->pipe_state == PIPE_STATE_ERR_MFULL || - c->pipe_state == PIPE_STATE_ERR_BAD); - if (c->noreply) { - c->noreply = false; /* reset noreply */ - } else { - /* The last command of pipelining has come. */ - pipe_state_clear(c); - } - return false; - } -} - -void append_stat(const char *name, ADD_STAT add_stats, conn *c, - const char *fmt, ...) -{ - char val_str[STAT_VAL_LEN]; - int vlen; - va_list ap; - - assert(name); - assert(add_stats); - assert(c); - assert(fmt); - - va_start(ap, fmt); - vlen = vsnprintf(val_str, sizeof(val_str) - 1, fmt, ap); - va_end(ap); - - add_stats(name, strlen(name), val_str, vlen, c); -} - -inline static void process_stats_detail(conn *c, const char *command) -{ - assert(c != NULL); - - if (settings.allow_detailed) { - if (strcmp(command, "on") == 0) { - settings.detail_enabled = 1; - out_string(c, "OK"); - } - else if (strcmp(command, "off") == 0) { - settings.detail_enabled = 0; - out_string(c, "OK"); - } - else if (strcmp(command, "dump") == 0) { - int len; - char *stats = stats_prefix_dump(NULL, 0, &len); - if (stats == NULL) { - out_string(c, "SERVER_ERROR no more memory"); - return; - } - write_and_free(c, stats, len); - } - else { - out_string(c, "CLIENT_ERROR usage: stats detail on|off|dump"); - } - } else { - out_string(c, "CLIENT_ERROR detailed stats disabled"); - } -} - -static void process_stats_cachedump(conn *c, token_t *tokens, const size_t ntokens) -{ - char *buf = NULL; - unsigned int bytes = 0; - unsigned int id; - unsigned int limit = 0; - bool forward = true; - bool sticky = false; - bool valid = false; - - do { - if (ntokens < 5 || ntokens > 7) break; - if (!safe_strtoul(tokens[2].value, &id)) break; - if (!safe_strtoul(tokens[3].value, &limit)) break; - if (ntokens >= 6) { - if (strcmp(tokens[4].value, "forward")==0) forward = true; - else if (strcmp(tokens[4].value, "backward")==0) forward = false; - else break; - } - if (ntokens == 7) { - if (strcmp(tokens[5].value, "sticky")==0) sticky = true; - else break; - } - valid = true; - } while(0); - - if (valid == false) { - print_invalid_command(c, tokens, ntokens); - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - if (id > POWER_LARGEST) { - out_string(c, "CLIENT_ERROR Illegal slab id"); - return; - } - - if (limit == 0) limit = 50; - if (limit > 200) limit = 200; - - buf = mc_engine.v1->cachedump(mc_engine.v0, c, id, limit, - forward, sticky, &bytes); - write_and_free(c, buf, bytes); -} - -static void aggregate_callback(void *in, void *out) -{ - struct thread_stats *out_thread_stats = out; - struct thread_stats *in_thread_stats = in; - threadlocal_stats_aggregate(in_thread_stats, out_thread_stats); -} - -/* return server specific stats only */ -static void server_stats(ADD_STAT add_stats, conn *c, bool aggregate) -{ - pid_t pid = getpid(); - rel_time_t now = get_current_time(); - - struct thread_stats thread_stats; - threadlocal_stats_clear(&thread_stats); - - if (aggregate && mc_engine.v1->aggregate_stats != NULL) { - mc_engine.v1->aggregate_stats(mc_engine.v0, (const void *)c, - aggregate_callback, &thread_stats); - } else { - threadlocal_stats_aggregate(default_thread_stats, &thread_stats); - } - -#ifndef __WIN32__ - struct rusage usage; - getrusage(RUSAGE_SELF, &usage); -#endif - -#ifdef ENABLE_ZK_INTEGRATION - arcus_hb_stats hb_stats; - arcus_hb_get_stats(&hb_stats); -#endif - - LOCK_STATS(); - - APPEND_STAT("pid", "%lu", (long)pid); - APPEND_STAT("uptime", "%u", now); - APPEND_STAT("time", "%ld", now + (long)process_started); - APPEND_STAT("version", "%s", VERSION); - APPEND_STAT("libevent", "%s", event_get_version()); - APPEND_STAT("pointer_size", "%d", (int)(8 * sizeof(void *))); -#ifdef ENABLE_ZK_INTEGRATION - APPEND_STAT("hb_count", "%"PRIu64, hb_stats.count); - APPEND_STAT("hb_latency", "%"PRIu64, hb_stats.latency); -#endif - -#ifndef __WIN32__ - append_stat("rusage_user", add_stats, c, "%ld.%06ld", - (long)usage.ru_utime.tv_sec, - (long)usage.ru_utime.tv_usec); - append_stat("rusage_system", add_stats, c, "%ld.%06ld", - (long)usage.ru_stime.tv_sec, - (long)usage.ru_stime.tv_usec); -#endif - - APPEND_STAT("daemon_connections", "%u", mc_stats.daemon_conns); - APPEND_STAT("curr_connections", "%u", mc_stats.curr_conns); - APPEND_STAT("quit_connections", "%u", mc_stats.quit_conns); - APPEND_STAT("reject_connections", "%u", mc_stats.rejected_conns); - APPEND_STAT("total_connections", "%u", mc_stats.total_conns); - APPEND_STAT("connection_structures", "%u", mc_stats.conn_structs); - APPEND_STAT("cmd_get", "%"PRIu64, thread_stats.cmd_get); - APPEND_STAT("cmd_set", "%"PRIu64, thread_stats.cmd_set); - APPEND_STAT("cmd_incr", "%"PRIu64, thread_stats.cmd_incr); - APPEND_STAT("cmd_decr", "%"PRIu64, thread_stats.cmd_decr); - APPEND_STAT("cmd_delete", "%"PRIu64, thread_stats.cmd_delete); - APPEND_STAT("cmd_cas", "%"PRIu64, thread_stats.cmd_cas); - APPEND_STAT("cmd_flush", "%"PRIu64, thread_stats.cmd_flush); - APPEND_STAT("cmd_flush_prefix", "%"PRIu64, thread_stats.cmd_flush_prefix); - APPEND_STAT("cmd_auth", "%"PRIu64, thread_stats.cmd_auth); - APPEND_STAT("cmd_lop_create", "%"PRIu64, thread_stats.cmd_lop_create); - APPEND_STAT("cmd_lop_insert", "%"PRIu64, thread_stats.cmd_lop_insert); - APPEND_STAT("cmd_lop_delete", "%"PRIu64, thread_stats.cmd_lop_delete); - APPEND_STAT("cmd_lop_get", "%"PRIu64, thread_stats.cmd_lop_get); - APPEND_STAT("cmd_sop_create", "%"PRIu64, thread_stats.cmd_sop_create); - APPEND_STAT("cmd_sop_insert", "%"PRIu64, thread_stats.cmd_sop_insert); - APPEND_STAT("cmd_sop_delete", "%"PRIu64, thread_stats.cmd_sop_delete); - APPEND_STAT("cmd_sop_get", "%"PRIu64, thread_stats.cmd_sop_get); - APPEND_STAT("cmd_sop_exist", "%"PRIu64, thread_stats.cmd_sop_exist); - APPEND_STAT("cmd_mop_create", "%"PRIu64, thread_stats.cmd_mop_create); - APPEND_STAT("cmd_mop_insert", "%"PRIu64, thread_stats.cmd_mop_insert); - APPEND_STAT("cmd_mop_update", "%"PRIu64, thread_stats.cmd_mop_update); - APPEND_STAT("cmd_mop_delete", "%"PRIu64, thread_stats.cmd_mop_delete); - APPEND_STAT("cmd_mop_get", "%"PRIu64, thread_stats.cmd_mop_get); - APPEND_STAT("cmd_bop_create", "%"PRIu64, thread_stats.cmd_bop_create); - APPEND_STAT("cmd_bop_insert", "%"PRIu64, thread_stats.cmd_bop_insert); - APPEND_STAT("cmd_bop_update", "%"PRIu64, thread_stats.cmd_bop_update); - APPEND_STAT("cmd_bop_delete", "%"PRIu64, thread_stats.cmd_bop_delete); - APPEND_STAT("cmd_bop_get", "%"PRIu64, thread_stats.cmd_bop_get); - APPEND_STAT("cmd_bop_count", "%"PRIu64, thread_stats.cmd_bop_count); - APPEND_STAT("cmd_bop_position", "%"PRIu64, thread_stats.cmd_bop_position); - APPEND_STAT("cmd_bop_pwg", "%"PRIu64, thread_stats.cmd_bop_pwg); - APPEND_STAT("cmd_bop_gbp", "%"PRIu64, thread_stats.cmd_bop_gbp); -#ifdef SUPPORT_BOP_MGET - APPEND_STAT("cmd_bop_mget", "%"PRIu64, thread_stats.cmd_bop_mget); -#endif -#ifdef SUPPORT_BOP_SMGET - APPEND_STAT("cmd_bop_smget", "%"PRIu64, thread_stats.cmd_bop_smget); -#endif - APPEND_STAT("cmd_bop_incr", "%"PRIu64, thread_stats.cmd_bop_incr); - APPEND_STAT("cmd_bop_decr", "%"PRIu64, thread_stats.cmd_bop_decr); - APPEND_STAT("cmd_getattr", "%"PRIu64, thread_stats.cmd_getattr); - APPEND_STAT("cmd_setattr", "%"PRIu64, thread_stats.cmd_setattr); - APPEND_STAT("get_hits", "%"PRIu64, thread_stats.get_hits); - APPEND_STAT("get_misses", "%"PRIu64, thread_stats.get_misses); - APPEND_STAT("incr_hits", "%"PRIu64, thread_stats.incr_hits); - APPEND_STAT("incr_misses", "%"PRIu64, thread_stats.incr_misses); - APPEND_STAT("decr_hits", "%"PRIu64, thread_stats.decr_hits); - APPEND_STAT("decr_misses", "%"PRIu64, thread_stats.decr_misses); - APPEND_STAT("delete_hits", "%"PRIu64, thread_stats.delete_hits); - APPEND_STAT("delete_misses", "%"PRIu64, thread_stats.delete_misses); - APPEND_STAT("cas_hits", "%"PRIu64, thread_stats.cas_hits); - APPEND_STAT("cas_badval", "%"PRIu64, thread_stats.cas_badval); - APPEND_STAT("cas_misses", "%"PRIu64, thread_stats.cas_misses); - APPEND_STAT("auth_errors", "%"PRIu64, thread_stats.auth_errors); - APPEND_STAT("lop_create_oks", "%"PRIu64, thread_stats.lop_create_oks); - APPEND_STAT("lop_insert_misses", "%"PRIu64, thread_stats.lop_insert_misses); - APPEND_STAT("lop_insert_hits", "%"PRIu64, thread_stats.lop_insert_hits); - APPEND_STAT("lop_delete_misses", "%"PRIu64, thread_stats.lop_delete_misses); - APPEND_STAT("lop_delete_elem_hits", "%"PRIu64, thread_stats.lop_delete_elem_hits); - APPEND_STAT("lop_delete_none_hits", "%"PRIu64, thread_stats.lop_delete_none_hits); - APPEND_STAT("lop_get_misses", "%"PRIu64, thread_stats.lop_get_misses); - APPEND_STAT("lop_get_elem_hits", "%"PRIu64, thread_stats.lop_get_elem_hits); - APPEND_STAT("lop_get_none_hits", "%"PRIu64, thread_stats.lop_get_none_hits); - APPEND_STAT("sop_create_oks", "%"PRIu64, thread_stats.sop_create_oks); - APPEND_STAT("sop_insert_misses", "%"PRIu64, thread_stats.sop_insert_misses); - APPEND_STAT("sop_insert_hits", "%"PRIu64, thread_stats.sop_insert_hits); - APPEND_STAT("sop_delete_misses", "%"PRIu64, thread_stats.sop_delete_misses); - APPEND_STAT("sop_delete_elem_hits", "%"PRIu64, thread_stats.sop_delete_elem_hits); - APPEND_STAT("sop_delete_none_hits", "%"PRIu64, thread_stats.sop_delete_none_hits); - APPEND_STAT("sop_get_misses", "%"PRIu64, thread_stats.sop_get_misses); - APPEND_STAT("sop_get_elem_hits", "%"PRIu64, thread_stats.sop_get_elem_hits); - APPEND_STAT("sop_get_none_hits", "%"PRIu64, thread_stats.sop_get_none_hits); - APPEND_STAT("sop_exist_misses", "%"PRIu64, thread_stats.sop_exist_misses); - APPEND_STAT("sop_exist_hits", "%"PRIu64, thread_stats.sop_exist_hits); - APPEND_STAT("mop_create_oks", "%"PRIu64, thread_stats.mop_create_oks); - APPEND_STAT("mop_insert_misses", "%"PRIu64, thread_stats.mop_insert_misses); - APPEND_STAT("mop_insert_hits", "%"PRIu64, thread_stats.mop_insert_hits); - APPEND_STAT("mop_update_misses", "%"PRIu64, thread_stats.mop_update_misses); - APPEND_STAT("mop_update_elem_hits", "%"PRIu64, thread_stats.mop_update_elem_hits); - APPEND_STAT("mop_update_none_hits", "%"PRIu64, thread_stats.mop_update_none_hits); - APPEND_STAT("mop_delete_misses", "%"PRIu64, thread_stats.mop_delete_misses); - APPEND_STAT("mop_delete_elem_hits", "%"PRIu64, thread_stats.mop_delete_elem_hits); - APPEND_STAT("mop_delete_none_hits", "%"PRIu64, thread_stats.mop_delete_none_hits); - APPEND_STAT("mop_get_misses", "%"PRIu64, thread_stats.mop_get_misses); - APPEND_STAT("mop_get_elem_hits", "%"PRIu64, thread_stats.mop_get_elem_hits); - APPEND_STAT("mop_get_none_hits", "%"PRIu64, thread_stats.mop_get_none_hits); - APPEND_STAT("bop_create_oks", "%"PRIu64, thread_stats.bop_create_oks); - APPEND_STAT("bop_insert_misses", "%"PRIu64, thread_stats.bop_insert_misses); - APPEND_STAT("bop_insert_hits", "%"PRIu64, thread_stats.bop_insert_hits); - APPEND_STAT("bop_update_misses", "%"PRIu64, thread_stats.bop_update_misses); - APPEND_STAT("bop_update_elem_hits", "%"PRIu64, thread_stats.bop_update_elem_hits); - APPEND_STAT("bop_update_none_hits", "%"PRIu64, thread_stats.bop_update_none_hits); - APPEND_STAT("bop_delete_misses", "%"PRIu64, thread_stats.bop_delete_misses); - APPEND_STAT("bop_delete_elem_hits", "%"PRIu64, thread_stats.bop_delete_elem_hits); - APPEND_STAT("bop_delete_none_hits", "%"PRIu64, thread_stats.bop_delete_none_hits); - APPEND_STAT("bop_get_misses", "%"PRIu64, thread_stats.bop_get_misses); - APPEND_STAT("bop_get_elem_hits", "%"PRIu64, thread_stats.bop_get_elem_hits); - APPEND_STAT("bop_get_none_hits", "%"PRIu64, thread_stats.bop_get_none_hits); - APPEND_STAT("bop_count_misses", "%"PRIu64, thread_stats.bop_count_misses); - APPEND_STAT("bop_count_hits", "%"PRIu64, thread_stats.bop_count_hits); - APPEND_STAT("bop_position_misses", "%"PRIu64, thread_stats.bop_position_misses); - APPEND_STAT("bop_position_elem_hits", "%"PRIu64, thread_stats.bop_position_elem_hits); - APPEND_STAT("bop_position_none_hits", "%"PRIu64, thread_stats.bop_position_none_hits); - APPEND_STAT("bop_pwg_misses", "%"PRIu64, thread_stats.bop_pwg_misses); - APPEND_STAT("bop_pwg_elem_hits", "%"PRIu64, thread_stats.bop_pwg_elem_hits); - APPEND_STAT("bop_pwg_none_hits", "%"PRIu64, thread_stats.bop_pwg_none_hits); - APPEND_STAT("bop_gbp_misses", "%"PRIu64, thread_stats.bop_gbp_misses); - APPEND_STAT("bop_gbp_elem_hits", "%"PRIu64, thread_stats.bop_gbp_elem_hits); - APPEND_STAT("bop_gbp_none_hits", "%"PRIu64, thread_stats.bop_gbp_none_hits); -#ifdef SUPPORT_BOP_MGET - APPEND_STAT("bop_mget_oks", "%"PRIu64, thread_stats.bop_mget_oks); -#endif -#ifdef SUPPORT_BOP_SMGET - APPEND_STAT("bop_smget_oks", "%"PRIu64, thread_stats.bop_smget_oks); -#endif - APPEND_STAT("bop_incr_elem_hits", "%"PRIu64, thread_stats.bop_incr_elem_hits); - APPEND_STAT("bop_incr_none_hits", "%"PRIu64, thread_stats.bop_incr_none_hits); - APPEND_STAT("bop_incr_misses", "%"PRIu64, thread_stats.bop_incr_misses); - APPEND_STAT("bop_decr_elem_hits", "%"PRIu64, thread_stats.bop_decr_elem_hits); - APPEND_STAT("bop_decr_none_hits", "%"PRIu64, thread_stats.bop_decr_none_hits); - APPEND_STAT("bop_decr_misses", "%"PRIu64, thread_stats.bop_decr_misses); - APPEND_STAT("getattr_misses", "%"PRIu64, thread_stats.getattr_misses); - APPEND_STAT("getattr_hits", "%"PRIu64, thread_stats.getattr_hits); - APPEND_STAT("setattr_misses", "%"PRIu64, thread_stats.setattr_misses); - APPEND_STAT("setattr_hits", "%"PRIu64, thread_stats.setattr_hits); - APPEND_STAT("stat_prefixes", "%"PRIu64, stats_prefix_count()); - APPEND_STAT("bytes_read", "%"PRIu64, thread_stats.bytes_read); - APPEND_STAT("bytes_written", "%"PRIu64, thread_stats.bytes_written); - APPEND_STAT("limit_maxbytes", "%"PRIu64, settings.maxbytes); - APPEND_STAT("limit_maxconns", "%d", settings.maxconns); - APPEND_STAT("threads", "%d", settings.num_threads); - APPEND_STAT("conn_yields", "%"PRIu64, thread_stats.conn_yields); - UNLOCK_STATS(); -} - -static void process_stats_settings(ADD_STAT add_stats, void *c) -{ - assert(add_stats); -#ifdef ENABLE_ZK_INTEGRATION - arcus_hb_confs hb_confs; - arcus_hb_get_confs(&hb_confs); -#endif - APPEND_STAT("maxbytes", "%llu", (unsigned long long)settings.maxbytes); - APPEND_STAT("maxconns", "%d", settings.maxconns); - APPEND_STAT("tcpport", "%d", settings.port); - APPEND_STAT("udpport", "%d", settings.udpport); - APPEND_STAT("sticky_limit", "%llu", (unsigned long long)settings.sticky_limit); - APPEND_STAT("inter", "%s", settings.inter ? settings.inter : "NULL"); - APPEND_STAT("verbosity", "%d", settings.verbose); - APPEND_STAT("oldest", "%lu", (unsigned long)settings.oldest_live); - APPEND_STAT("evictions", "%s", settings.evict_to_free ? "on" : "off"); - APPEND_STAT("domain_socket", "%s", - settings.socketpath ? settings.socketpath : "NULL"); - APPEND_STAT("umask", "%o", settings.access); - APPEND_STAT("growth_factor", "%.2f", settings.factor); - APPEND_STAT("chunk_size", "%d", settings.chunk_size); - APPEND_STAT("num_threads", "%d", settings.num_threads); - APPEND_STAT("stat_key_prefix", "%c", settings.prefix_delimiter); - APPEND_STAT("detail_enabled", "%s", - settings.detail_enabled ? "yes" : "no"); - APPEND_STAT("allow_detailed", "%s", - settings.allow_detailed ? "yes" : "no"); - APPEND_STAT("reqs_per_event", "%d", settings.reqs_per_event); - APPEND_STAT("cas_enabled", "%s", settings.use_cas ? "yes" : "no"); - APPEND_STAT("tcp_backlog", "%d", settings.backlog); - APPEND_STAT("binding_protocol", "%s", - prot_text(settings.binding_protocol)); -#ifdef SASL_ENABLED - APPEND_STAT("auth_enabled_sasl", "%s", "yes"); -#else - APPEND_STAT("auth_enabled_sasl", "%s", "no"); -#endif - -#ifdef ENABLE_ISASL - APPEND_STAT("auth_sasl_engine", "%s", "isasl"); -#elif defined(ENABLE_SASL) - APPEND_STAT("auth_sasl_engine", "%s", "cyrus"); -#else - APPEND_STAT("auth_sasl_engine", "%s", "none"); -#endif - APPEND_STAT("auth_required_sasl", "%s", settings.require_sasl ? "yes" : "no"); - APPEND_STAT("item_size_max", "%llu", settings.item_size_max); - APPEND_STAT("max_list_size", "%u", settings.max_list_size); - APPEND_STAT("max_set_size", "%u", settings.max_set_size); - APPEND_STAT("max_map_size", "%u", settings.max_map_size); - APPEND_STAT("max_btree_size", "%u", settings.max_btree_size); - APPEND_STAT("max_element_bytes", "%u", settings.max_element_bytes); - APPEND_STAT("scrub_count", "%u", settings.scrub_count); - APPEND_STAT("topkeys", "%d", settings.topkeys); -#ifdef ENABLE_ZK_INTEGRATION - APPEND_STAT("hb_timeout", "%u", hb_confs.timeout); - APPEND_STAT("hb_failstop", "%u", hb_confs.failstop); -#endif - - for (EXTENSION_DAEMON_DESCRIPTOR *ptr = settings.extensions.daemons; - ptr != NULL; - ptr = ptr->next) { - APPEND_STAT("extension", "%s", ptr->get_name()); - } - - APPEND_STAT("logger", "%s", mc_logger->get_name()); - - for (EXTENSION_ASCII_PROTOCOL_DESCRIPTOR *ptr = settings.extensions.ascii; - ptr != NULL; - ptr = ptr->next) { - APPEND_STAT("ascii_extension", "%s", ptr->get_name(ptr->cookie)); - } -} - -#ifdef ENABLE_ZK_INTEGRATION -static void process_stats_zookeeper(ADD_STAT add_stats, void *c) -{ - assert(add_stats); - arcus_zk_confs zk_confs; - arcus_zk_stats zk_stats; - arcus_zk_get_confs(&zk_confs); - arcus_zk_get_stats(&zk_stats); - - APPEND_STAT("zk_libversion", "%s", zk_confs.zk_libversion); - APPEND_STAT("zk_timeout", "%u", zk_confs.zk_timeout); - APPEND_STAT("zk_failstop", "%s", zk_confs.zk_failstop ? "on" : "off"); - APPEND_STAT("zk_connected", "%s", zk_stats.zk_connected ? "true" : "false"); -#ifdef ENABLE_ZK_RECONFIG - APPEND_STAT("zk_reconfig_needed", "%s", zk_stats.zk_reconfig_needed ? "on" : "off"); - if (zk_stats.zk_reconfig_needed) { - APPEND_STAT("zk_reconfig_enabled", "%s", zk_stats.zk_reconfig_enabled ? "on" : "off"); - APPEND_STAT("zk_reconfig_version", "%" PRIx64, zk_stats.zk_reconfig_version); - } -#endif -} -#endif - -static void process_stats_command(conn *c, token_t *tokens, const size_t ntokens) -{ - assert(c != NULL && ntokens >= 2); - const char *subcommand = tokens[SUBCOMMAND_TOKEN].value; - - if (ntokens == 2) { - server_stats(&append_ascii_stats, c, false); - (void)mc_engine.v1->get_stats(mc_engine.v0, c, NULL, 0, - &append_ascii_stats); - } else if (strcmp(subcommand, "reset") == 0) { - stats_reset(c); - out_string(c, "RESET"); - return; - } else if (strcmp(subcommand, "detail") == 0) { - /* NOTE: how to tackle detail with binary? */ - if (ntokens < 4) - process_stats_detail(c, ""); /* outputs the error message */ - else - process_stats_detail(c, tokens[2].value); - /* Output already generated */ - return; - } else if (strcmp(subcommand, "settings") == 0) { - process_stats_settings(&append_ascii_stats, c); -#ifdef ENABLE_ZK_INTEGRATION - } else if (strcmp(subcommand, "zookeeper") == 0) { - process_stats_zookeeper(&append_ascii_stats, c); -#endif - } else if (strcmp(subcommand, "cachedump") == 0) { - process_stats_cachedump(c, tokens, ntokens); - return; - } else if (strcmp(subcommand, "aggregate") == 0) { - server_stats(&append_ascii_stats, c, true); - } else if (strcmp(subcommand, "topkeys") == 0) { - if (default_topkeys == NULL) { - out_string(c, "NOT_SUPPORTED"); - return; - } - topkeys_stats(default_topkeys, c, get_current_time(), append_ascii_stats); - } else if (strcmp(subcommand, "prefixes") == 0) { - int len; - char *stats = mc_engine.v1->prefix_dump_stats(mc_engine.v0, c, NULL, 0, &len); - if (stats == NULL) { - if (len == -1) - out_string(c, "NOT_SUPPORTED"); - else - out_string(c, "SERVER_ERROR no more memory"); - return; - } - write_and_free(c, stats, len); - return; /* Output already generated */ - } else if (strcmp(subcommand, "prefixlist") == 0) { - int len; - char *stats; - token_t *prefixes = ntokens > 4 ? &tokens[3] : NULL; - size_t nprefixes = ntokens > 4 ? ntokens-4 : 0; - - if (ntokens < 4) { - out_string(c, "CLIENT_ERROR subcommand(item|operation) is required"); - return; - } else if (strcmp(tokens[2].value, "item") == 0) { - stats = mc_engine.v1->prefix_dump_stats(mc_engine.v0, c, prefixes, nprefixes, &len); - } else if (strcmp(tokens[2].value, "operation") == 0) { - stats = stats_prefix_dump(prefixes, nprefixes, &len); - } else { - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - if (stats == NULL) { - if (len == -1) - out_string(c, "NOT_SUPPORTED"); - else - out_string(c, "SERVER_ERROR no more memory"); - return; - } - write_and_free(c, stats, len); - return; /* Output already generated */ - } else if (strcmp(subcommand, "prefix") == 0) { - /* command: stats prefix \r\n */ - if (ntokens != 4) { - print_invalid_command(c, tokens, ntokens); - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - if (tokens[2].length > PREFIX_MAX_LENGTH) { - out_string(c, "CLIENT_ERROR too long prefix name"); - return; - } - if (strcmp(tokens[2].value, "") == 0) { /* reserved keyword */ - (void)mc_engine.v1->prefix_get_stats(mc_engine.v0, c, NULL, 0, - append_ascii_stats); - stats_prefix_get(NULL, 0, append_ascii_stats, c); - } else { - (void)mc_engine.v1->prefix_get_stats(mc_engine.v0, c, - tokens[2].value, tokens[2].length, - append_ascii_stats); - stats_prefix_get(tokens[2].value, tokens[2].length, append_ascii_stats, c); - } - } else { - /* getting here means that the subcommand is either engine specific or - is invalid. query the engine and see. */ - ENGINE_ERROR_CODE ret; - int nb; - char buf[1024]; - - nb = detokenize(&tokens[1], ntokens - 2, buf, 1024); - if (nb <= 0) { - /* no matching stat */ - ret = ENGINE_KEY_ENOENT; - } else { - ret = mc_engine.v1->get_stats(mc_engine.v0, c, buf, nb, - append_ascii_stats); - } - - switch (ret) { - case ENGINE_SUCCESS: - append_ascii_stats(NULL, 0, NULL, 0, c); - write_and_free(c, c->dynamic_buffer.buffer, c->dynamic_buffer.offset); - c->dynamic_buffer.buffer = NULL; - break; - case ENGINE_ENOMEM: - out_string(c, "SERVER_ERROR out of memory writing stats"); - break; - case ENGINE_KEY_ENOENT: - out_string(c, "ERROR no matching stat"); - break; - default: - handle_unexpected_errorcode_ascii(c, __func__, ret); - break; - } - return; - } - - /* append terminator and start the transfer */ - append_ascii_stats(NULL, 0, NULL, 0, c); - - if (c->dynamic_buffer.buffer == NULL) { - out_string(c, "SERVER_ERROR out of memory writing stats"); - } else { - write_and_free(c, c->dynamic_buffer.buffer, c->dynamic_buffer.offset); - c->dynamic_buffer.buffer = NULL; - } -} - -/* ntokens is overwritten here... shrug.. */ -static inline void process_get_command(conn *c, token_t *tokens, size_t ntokens, bool return_cas) -{ - assert(c != NULL); - token_t *key_token = &tokens[KEY_TOKEN]; - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - do { - while (key_token->length != 0) { - if (key_token->length > KEY_MAX_LENGTH) { - ret = ENGINE_EINVAL; break; - } - /* do get operation for each key */ - ret = process_get_single(c, key_token->value, key_token->length, - return_cas); - if (ret != ENGINE_SUCCESS) { - break; /* ret == ENGINE_ENOMEM */ - } - key_token++; - } - if (ret != ENGINE_SUCCESS) break; - - /* If the command string hasn't been fully processed, get the next set of tokens. */ - if (key_token->value != NULL) { - /* The next reserved token has the length of untokenized command. */ - ntokens = tokenize_command(key_token->value, (key_token+1)->length, - tokens, MAX_TOKENS); - key_token = tokens; - } - } while(key_token->value != NULL); - - /* Some items and suffixes might have saved in the above execution. - * To release the items and free the suffixes, the below code is needed. - */ - c->icurr = c->ilist; - c->suffixcurr = c->suffixlist; - - if (ret != ENGINE_SUCCESS) { - /* Releasing items on ilist and freeing suffixes will be - * performed later by calling out_string() function. - * See conn_write() and conn_mwrite() state. - */ - if (ret == ENGINE_EINVAL) - out_string(c, "CLIENT_ERROR bad command line format"); - else /* ret == ENGINE_ENOMEM */ - out_string(c, "SERVER_ERROR out of memory writing get response"); - return; - } - - if (settings.verbose > 1) { - mc_logger->log(EXTENSION_LOG_DEBUG, c, ">%d END\n", c->sfd); - } - - /* If the loop was terminated because of out-of-memory, it is not - * reliable to add END\r\n to the buffer, because it might not end - * in \r\n. So we send SERVER_ERROR instead. - */ - if ((add_iov(c, "END\r\n", 5) != 0) || - (IS_UDP(c->transport) && build_udp_headers(c) != 0)) { - out_string(c, "SERVER_ERROR out of memory writing get response"); - } else { - conn_set_state(c, conn_mwrite); - c->msgcurr = 0; - } -} - -static void process_prepare_nread_keys(conn *c, uint32_t vlen, uint32_t kcnt, bool return_cas) -{ - ENGINE_ERROR_CODE ret = ENGINE_SUCCESS; - - /* allocate memory blocks needed */ - if (mblck_list_alloc(&c->thread->mblck_pool, 1, vlen, &c->memblist) < 0) { - ret = ENGINE_ENOMEM; - } - if (ret == ENGINE_SUCCESS) { - c->coll_strkeys = (void*)&c->memblist; - ritem_set_first(c, CONN_RTYPE_MBLCK, vlen); - c->coll_op = (return_cas ? OPERATION_MGETS : OPERATION_MGET); - conn_set_state(c, conn_nread); - } else { - out_string(c, "SERVER_ERROR out of memory"); - c->sbytes = vlen; - c->write_and_go = conn_swallow; - } -} - -static inline void process_mget_command(conn *c, token_t *tokens, const size_t ntokens, bool return_cas) -{ - uint32_t lenkeys, numkeys; - - if ((! safe_strtoul(tokens[COMMAND_TOKEN+1].value, &lenkeys)) || - (! safe_strtoul(tokens[COMMAND_TOKEN+2].value, &numkeys)) || - (lenkeys > (UINT_MAX-2)) || (lenkeys == 0) || (numkeys == 0)) { - print_invalid_command(c, tokens, ntokens); - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - - if (numkeys > MAX_MGET_KEY_COUNT || - numkeys > ((lenkeys/2) + 1) || - lenkeys > ((numkeys*KEY_MAX_LENGTH) + numkeys-1)) { - /* ENGINE_EBADVALUE */ - out_string(c, "CLIENT_ERROR bad value"); - c->sbytes = lenkeys + 2; - c->write_and_go = conn_swallow; - return; - } - lenkeys += 2; - - c->coll_numkeys = numkeys; - c->coll_lenkeys = lenkeys; - - process_prepare_nread_keys(c, lenkeys, numkeys, return_cas); -} - -static void process_update_command(conn *c, token_t *tokens, const size_t ntokens, - ENGINE_STORE_OPERATION store_op, bool handle_cas) -{ - assert(c != NULL); - char *key; - size_t nkey; - unsigned int flags; - int32_t exptime_int=0; - time_t exptime; - int vlen; - uint64_t req_cas_id=0; - item *it; - - set_noreply_maybe(c, tokens, ntokens); - - key = tokens[KEY_TOKEN].value; - nkey = tokens[KEY_TOKEN].length; - if (nkey > KEY_MAX_LENGTH) { - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - - if ((! safe_strtoul(tokens[2].value, (uint32_t *)&flags)) || - (! safe_strtol(tokens[3].value, &exptime_int)) || - (! safe_strtol(tokens[4].value, (int32_t *)&vlen)) || - (vlen < 0 || vlen > (INT_MAX-2))) - { - print_invalid_command(c, tokens, ntokens); - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - vlen += 2; - - /* Ubuntu 8.04 breaks when I pass exptime to safe_strtol */ - exptime = exptime_int; - - // does cas value exist? - if (handle_cas) { - if (!safe_strtoull(tokens[5].value, &req_cas_id)) { - print_invalid_command(c, tokens, ntokens); - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - } - - if (settings.detail_enabled) { - stats_prefix_record_set(key, nkey); - } - - ENGINE_ERROR_CODE ret; - ret = mc_engine.v1->allocate(mc_engine.v0, c, &it, key, nkey, vlen, - htonl(flags), realtime(exptime), req_cas_id); - if (ret == ENGINE_SUCCESS) { - if (!mc_engine.v1->get_item_info(mc_engine.v0, c, it, &c->hinfo)) { - mc_engine.v1->release(mc_engine.v0, c, it); - out_string(c, "SERVER_ERROR error getting item data"); - ret = ENGINE_FAILED; /* FIXME: error type */ - } else { - c->item = it; - ritem_set_first(c, CONN_RTYPE_HINFO, vlen); - c->store_op = store_op; - conn_set_state(c, conn_nread); - } - } - if (ret != ENGINE_SUCCESS) { - if (ret == ENGINE_E2BIG) { - out_string(c, "CLIENT_ERROR object too large for cache"); - } else if (ret == ENGINE_ENOMEM) { - out_string(c, "SERVER_ERROR out of memory storing object"); - } else if (ret == ENGINE_FAILED) { - /* out_string() was called above. so, do nothing */ - } else { - handle_unexpected_errorcode_ascii(c, __func__, ret); - } - - /* Avoid stale data persisting in cache because we failed alloc. - * Unacceptable for SET. Anywhere else too? - */ - if (store_op == OPERATION_SET) { - /* set temporarily noreply for the ASYNC interface */ - /* noreply flag is cleared in out_string() if it was set */ - assert(c->noreply == false); - c->noreply = true; - mc_engine.v1->remove(mc_engine.v0, c, key, nkey, 0, 0); - c->noreply = false; - } - - if (ret != ENGINE_DISCONNECT) { - /* swallow the data line */ - c->sbytes = vlen; - if (c->state == conn_write) { - c->write_and_go = conn_swallow; - } else { /* conn_new_cmd (by noreply) */ - conn_set_state(c, conn_swallow); - } - } - } -} - -static void process_arithmetic_command(conn *c, token_t *tokens, const size_t ntokens, - const bool incr) -{ - assert(c != NULL); - assert(c->ewouldblock == false); - uint64_t delta; - char *key; - size_t nkey; - - set_noreply_maybe(c, tokens, ntokens); - - key = tokens[KEY_TOKEN].value; - nkey = tokens[KEY_TOKEN].length; - if (nkey > KEY_MAX_LENGTH) { - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - if (!safe_strtoull(tokens[2].value, &delta)) { - out_string(c, "CLIENT_ERROR invalid numeric delta argument"); - return; - } - - bool create = false; - unsigned int flags = 0; - int32_t exptime_int = 0; - uint64_t init_value = 0; - - if (ntokens >= 7) { - if (! (safe_strtoul(tokens[3].value, (uint32_t *)&flags) - && safe_strtol(tokens[4].value, &exptime_int) - && safe_strtoull(tokens[5].value, &init_value))) { - print_invalid_command(c, tokens, ntokens); - out_string(c, "CLIENT_ERROR bad command line format"); - return; - } - create = true; - } - - if (settings.detail_enabled) { - if (incr) stats_prefix_record_incr(key, nkey); - else stats_prefix_record_decr(key, nkey); - } - - ENGINE_ERROR_CODE ret; - uint64_t cas; - uint64_t result; - - ret = mc_engine.v1->arithmetic(mc_engine.v0, c, key, nkey, - incr, create, delta, init_value, - htonl(flags), realtime(exptime_int), - &cas, &result, 0); - CONN_CHECK_AND_SET_EWOULDBLOCK(ret, c); - - char temp[INCR_MAX_STORAGE_LEN]; - switch (ret) { - case ENGINE_SUCCESS: - if (incr) { - STATS_HITS(c, incr, key, nkey); - } else { - STATS_HITS(c, decr, key, nkey); - } - snprintf(temp, sizeof(temp), "%"PRIu64, result); - out_string(c, temp); - break; - case ENGINE_KEY_ENOENT: - if (incr) { - STATS_MISSES(c, incr, key, nkey); - } else { - STATS_MISSES(c, decr, key, nkey); - } - out_string(c, "NOT_FOUND"); - break; - default: - if (incr) { - STATS_CMD_NOKEY(c, incr); - } else { - STATS_CMD_NOKEY(c, decr); - } - if (ret == ENGINE_EINVAL) - out_string(c, "CLIENT_ERROR cannot increment or decrement non-numeric value"); - else if (ret == ENGINE_PREFIX_ENAME) - out_string(c, "CLIENT_ERROR invalid prefix name"); - else if (ret == ENGINE_ENOMEM) - out_string(c, "SERVER_ERROR out of memory"); - else if (ret == ENGINE_NOT_STORED) - out_string(c, "SERVER_ERROR failed to store item"); - else if (ret == ENGINE_EBADTYPE) - out_string(c, "TYPE_MISMATCH"); - else - handle_unexpected_errorcode_ascii(c, __func__, ret); - } -} - -static void process_delete_command(conn *c, token_t *tokens, const size_t ntokens) -{ - assert(c->ewouldblock == false); - char *key; - size_t nkey; - - if (ntokens > 3) { - /* See "delete [