X-Git-Url: https://git.m6w6.name/?a=blobdiff_plain;f=lib%2Fmemcached_storage.c;h=8dc381c0b77ec4a19477be68992841e1aff9f752;hb=0d688cec9e53a2ed8e3f3817da9aecdd41d342ca;hp=43a96636130aae2455f1f7c2d7f558cc14f01a22;hpb=78dc18473677a28570ab29134060a6411d9504df;p=awesomized%2Flibmemcached diff --git a/lib/memcached_storage.c b/lib/memcached_storage.c index 43a96636..8dc381c0 100644 --- a/lib/memcached_storage.c +++ b/lib/memcached_storage.c @@ -53,9 +53,10 @@ static inline memcached_return memcached_send(memcached_st *ptr, char to_write; size_t write_length; ssize_t sent_length; - memcached_return rc; + memcached_return rc[MEMCACHED_MAX_REPLICAS]; char buffer[MEMCACHED_DEFAULT_COMMAND_SIZE]; unsigned int server_key; + uint8_t replicas= 0; WATCHPOINT_ASSERT(!(value == NULL && value_length > 0)); WATCHPOINT_ASSERT(!(value && value_length == 0)); @@ -66,6 +67,9 @@ static inline memcached_return memcached_send(memcached_st *ptr, if (ptr->number_of_hosts == 0) return MEMCACHED_NO_SERVERS; + if ((ptr->flags & MEM_VERIFY_KEY) && (memcachd_key_test(&key, &key_length, 1) == MEMCACHED_BAD_KEY_PROVIDED)) + return MEMCACHED_BAD_KEY_PROVIDED; + server_key= memcached_generate_hash(ptr, master_key, master_key_length); if (cas) @@ -81,46 +85,56 @@ static inline memcached_return memcached_send(memcached_st *ptr, (unsigned long long)expiration, value_length); if (write_length >= MEMCACHED_DEFAULT_COMMAND_SIZE) - { - rc= MEMCACHED_WRITE_FAILURE; - goto error; - } - - rc= memcached_do(ptr, server_key, buffer, write_length, 0); - if (rc != MEMCACHED_SUCCESS) - goto error; - - if ((sent_length= memcached_io_write(ptr, server_key, value, value_length, 0)) == -1) - { - rc= MEMCACHED_WRITE_FAILURE; - goto error; - } + return MEMCACHED_WRITE_FAILURE; - if ((ptr->flags & MEM_NO_BLOCK) && verb == SET_OP) + if ((ptr->flags & MEM_BUFFER_REQUESTS) && verb == SET_OP) to_write= 0; else to_write= 1; - if ((sent_length= memcached_io_write(ptr, server_key, "\r\n", 2, to_write)) == -1) + do { - rc= MEMCACHED_WRITE_FAILURE; - goto error; - } + rc[replicas]= memcached_do(&ptr->hosts[server_key], buffer, write_length, 0); - if (to_write == 0) - rc= MEMCACHED_SUCCESS; - else - rc= memcached_response(ptr, buffer, MEMCACHED_DEFAULT_COMMAND_SIZE, NULL, server_key); + if (rc[replicas] != MEMCACHED_SUCCESS) + goto error; + + if ((sent_length= memcached_io_write(&ptr->hosts[server_key], value, value_length, 0)) == -1) + { + rc[replicas]= MEMCACHED_WRITE_FAILURE; + goto error; + } + + if ((sent_length= memcached_io_write(&ptr->hosts[server_key], "\r\n", 2, to_write)) == -1) + { + rc[replicas]= MEMCACHED_WRITE_FAILURE; + goto error; + } - if (rc == MEMCACHED_STORED) - return MEMCACHED_SUCCESS; - else - return rc; + if (to_write == 0) + return MEMCACHED_BUFFERED; + else + rc[replicas]= memcached_response(&ptr->hosts[server_key], buffer, MEMCACHED_DEFAULT_COMMAND_SIZE, NULL); + /* On error we just jump to the next potential server */ error: - memcached_io_reset(ptr, server_key); + if (replicas > 1 && ptr->distribution == MEMCACHED_DISTRIBUTION_CONSISTENT) + { + if (server_key == (ptr->number_of_hosts - 1)) + server_key= 0; + else + server_key++; + } + } while ((++replicas) < ptr->number_of_replicas); + + /* As long as one object gets stored, we count this as a success */ + while (replicas--) + { + if (rc[replicas] == MEMCACHED_STORED) + return MEMCACHED_SUCCESS; + } - return rc; + return rc[0]; } memcached_return memcached_set(memcached_st *ptr, char *key, size_t key_length, @@ -203,7 +217,7 @@ memcached_return memcached_cas(memcached_st *ptr, memcached_return rc; rc= memcached_send(ptr, key, key_length, key, key_length, value, value_length, - expiration, flags, cas, APPEND_OP); + expiration, flags, cas, CAS_OP); return rc; } @@ -294,6 +308,6 @@ memcached_return memcached_cas_by_key(memcached_st *ptr, memcached_return rc; rc= memcached_send(ptr, key, key_length, key, key_length, value, value_length, - expiration, flags, cas, APPEND_OP); + expiration, flags, cas, CAS_OP); return rc; }