X-Git-Url: https://git.m6w6.name/?a=blobdiff_plain;f=lib%2Fmemcached_delete.c;h=dea5c92ff367c2df7631cab331e9037f54a814d9;hb=65c53b0994e903cceda1adc7f6df0c8ea72f12c4;hp=b50aaef0f9be5c7fdb4c459d0ee79b31f0c3f22d;hpb=452f5b545eec58ac1bddd32ea131b79e2b1a52a9;p=m6w6%2Flibmemcached diff --git a/lib/memcached_delete.c b/lib/memcached_delete.c index b50aaef0..dea5c92f 100644 --- a/lib/memcached_delete.c +++ b/lib/memcached_delete.c @@ -1,19 +1,31 @@ -#include +#include "common.h" memcached_return memcached_delete(memcached_st *ptr, char *key, size_t key_length, time_t expiration) { - size_t send_length, sent_length; - memcached_return rc; + return memcached_delete_by_key(ptr, key, key_length, + key, key_length, expiration); +} + +memcached_return memcached_delete_by_key(memcached_st *ptr, + char *master_key, size_t master_key_length, + char *key, size_t key_length, + time_t expiration) +{ + char to_write= (ptr->flags & MEM_BUFFER_REQUESTS) ? 0 : 1; + size_t send_length; char buffer[MEMCACHED_DEFAULT_COMMAND_SIZE]; unsigned int server_key; + uint8_t replicas= 0; + memcached_return rc[MEMCACHED_MAX_REPLICAS]; - rc= memcached_connect(ptr); + unlikely (key_length == 0) + return MEMCACHED_NO_KEY_PROVIDED; - if (rc != MEMCACHED_SUCCESS) - return rc; + unlikely (ptr->hosts == NULL || ptr->number_of_hosts == 0) + return MEMCACHED_NO_SERVERS; - server_key= memcached_generate_hash(key, key_length) % ptr->number_of_hosts; + server_key= memcached_generate_hash(ptr, master_key, master_key_length); if (expiration) send_length= snprintf(buffer, MEMCACHED_DEFAULT_COMMAND_SIZE, @@ -24,12 +36,47 @@ memcached_return memcached_delete(memcached_st *ptr, char *key, size_t key_lengt "delete %.*s\r\n", (int)key_length, key); if (send_length >= MEMCACHED_DEFAULT_COMMAND_SIZE) - return MEMCACHED_WRITE_FAILURE; + { + rc[replicas]= MEMCACHED_WRITE_FAILURE; + goto error; + } + + do + { + rc[replicas]= memcached_do(&ptr->hosts[server_key], buffer, send_length, to_write); + if (rc[replicas] != MEMCACHED_SUCCESS) + goto error; + + if ((ptr->flags & MEM_BUFFER_REQUESTS)) + { + rc[replicas]= MEMCACHED_BUFFERED; + } + else + { + rc[replicas]= memcached_response(&ptr->hosts[server_key], buffer, MEMCACHED_DEFAULT_COMMAND_SIZE, NULL); + if (rc[replicas] == MEMCACHED_DELETED) + rc[replicas]= MEMCACHED_SUCCESS; + } - sent_length= write(ptr->hosts[server_key].fd, buffer, send_length); + /* On error we just jump to the next potential server */ +error: + if (ptr->number_of_replicas > 1) + { + if (server_key == (ptr->number_of_hosts - 1)) + server_key= 0; + else + server_key++; + } + } while ((++replicas) < ptr->number_of_replicas); - if (sent_length == -1 || sent_length != send_length) - return MEMCACHED_WRITE_FAILURE; + /* As long as one object gets stored, we count this as a success */ + while (replicas--) + { + if (rc[replicas] == MEMCACHED_DELETED) + return MEMCACHED_SUCCESS; + else if (rc[replicas] == MEMCACHED_DELETED) + rc[replicas]= MEMCACHED_BUFFERED; + } - return memcached_response(ptr, buffer, MEMCACHED_DEFAULT_COMMAND_SIZE, server_key); + return rc[0]; }