From 4bdb7129ba187d6659354a2dab2907f2f1936ff8 Mon Sep 17 00:00:00 2001 From: James Le Cuirot Date: Fri, 11 Apr 2014 22:23:22 +0100 Subject: [PATCH] Ungetlib libmemcached Tested with several libmemcached versions between 0.31 and 1.0.18. Unfortunately the API is extremely volatile and awkward to use. Packaging scripts still need addressing. FS-353 --- configure.ac | 16 +++ src/mod/applications/mod_memcache/Makefile.am | 31 ++--- .../applications/mod_memcache/mod_memcache.c | 125 +++++++++++++----- 3 files changed, 119 insertions(+), 53 deletions(-) diff --git a/configure.ac b/configure.ac index 5914afe892..193041ee84 100644 --- a/configure.ac +++ b/configure.ac @@ -1129,6 +1129,22 @@ PKG_CHECK_MODULES([YAML], [yaml-0.1 >= 0.1.4],[ AM_CONDITIONAL([HAVE_YAML],[true])],[ AC_MSG_RESULT([no]); AM_CONDITIONAL([HAVE_YAML],[false])]) +PKG_CHECK_MODULES([MEMCACHED], [libmemcached >= 0.31],[ + AM_CONDITIONAL([HAVE_MEMCACHED],[true]) + MEMCACHED_LIBS="${MEMCACHED_LIBS} -lpthread" + save_LIBS="${LIBS}" + save_CPPFLAGS="${CPPFLAGS}" + LIBS="${MEMCACHED_LIBS}" + CPPFLAGS="${MEMCACHED_CFLAGS}" + AC_CHECK_FUNCS([memcached_server_name memcached_stat_execute]) + AC_CHECK_TYPES([memcached_instance_st*],,, [[#include ]]) + LIBS="${save_LIBS}" + CPPFLAGS="${save_CPPFLAGS}" +],[ + AC_MSG_RESULT([no]) + AM_CONDITIONAL([HAVE_MEMCACHED],[false]) +]) + AC_ARG_ENABLE(core-libedit-support, [AS_HELP_STRING([--disable-core-libedit-support], [Compile without libedit Support])]) diff --git a/src/mod/applications/mod_memcache/Makefile.am b/src/mod/applications/mod_memcache/Makefile.am index ece75c7af2..f489880890 100644 --- a/src/mod/applications/mod_memcache/Makefile.am +++ b/src/mod/applications/mod_memcache/Makefile.am @@ -1,30 +1,17 @@ include $(top_srcdir)/build/modmake.rulesam MODNAME=mod_memcache -MEMCACHED=libmemcached-0.32 -MEMCACHED_DIR=$(switch_srcdir)/libs/$(MEMCACHED) -MEMCACHED_BUILDDIR=$(switch_builddir)/libs/$(MEMCACHED) -MEMCACHEDLA=$(MEMCACHED_BUILDDIR)/libmemcached/libmemcached.la +if HAVE_MEMCACHED mod_LTLIBRARIES = mod_memcache.la mod_memcache_la_SOURCES = mod_memcache.c -mod_memcache_la_CFLAGS = $(AM_CFLAGS) -I$(MEMCACHED_DIR) -I$(MEMCACHED_BUILDDIR) -mod_memcache_la_LIBADD = $(switch_builddir)/libfreeswitch.la $(MEMCACHEDLA) +mod_memcache_la_CFLAGS = $(AM_CFLAGS) $(MEMCACHED_CFLAGS) +mod_memcache_la_LIBADD = $(switch_builddir)/libfreeswitch.la $(MEMCACHED_LIBS) mod_memcache_la_LDFLAGS = -avoid-version -module -no-undefined -shared -MY_DEFAULT_ARGS=--build=$(build) --host=$(host) --target=$(target) --disable-shared --with-pic - -BUILT_SOURCES=$(MEMCACHEDLA) - -$(MEMCACHED_DIR): - $(GETLIB) $(MEMCACHED).tar.gz - -$(MEMCACHED_BUILDDIR)/Makefile: $(MEMCACHED_DIR) - mkdir -p $(MEMCACHED_BUILDDIR) - echo $(MY_DEFAULT_ARGS) - cd $(MEMCACHED_BUILDDIR) && $(DEFAULT_VARS) $(MEMCACHED_DIR)/configure $(MY_DEFAULT_ARGS) --with-memcached=no --srcdir=$(MEMCACHED_DIR) CPPFLAGS= LDFLAGS= - $(TOUCH_TARGET) - -$(MEMCACHEDLA): $(MEMCACHED_BUILDDIR)/Makefile - cd $(MEMCACHED_BUILDDIR) && $(MAKE) - $(TOUCH_TARGET) +else +install: error +all: error +error: + $(error You must install libmemcached-dev to build mod_memcache) +endif diff --git a/src/mod/applications/mod_memcache/mod_memcache.c b/src/mod/applications/mod_memcache/mod_memcache.c index 44ea3cc554..10e32e93c1 100644 --- a/src/mod/applications/mod_memcache/mod_memcache.c +++ b/src/mod/applications/mod_memcache/mod_memcache.c @@ -29,6 +29,7 @@ * */ #include +#include #include /* Prototypes */ @@ -138,6 +139,81 @@ static void event_handler(switch_event_t *event) do_config(SWITCH_TRUE); } +#if HAVE_MEMCACHED_SERVER_NAME +#if HAVE_MEMCACHED_INSTANCE_STP +#define MCD_SERVER memcached_instance_st* +#else +#define MCD_SERVER memcached_server_instance_st +#endif +#define MCD_SERVER_NAME memcached_server_name(server) +#define MCD_SERVER_PORT memcached_server_port(server) +#else +#define MCD_SERVER memcached_server_st* +#define MCD_SERVER_NAME memcached_server_name(ptr, *server) +#define MCD_SERVER_PORT memcached_server_port(ptr, *server) +#endif + +struct stream_server +{ + switch_stream_handle_t *stream; + MCD_SERVER server; +}; + +static memcached_return server_show(const memcached_st *ptr, const MCD_SERVER server, void *context) +{ + struct stream_server *ss = (struct stream_server*) context; + ss->stream->write_function(ss->stream, " %s (%u)\n", MCD_SERVER_NAME, MCD_SERVER_PORT); + return MEMCACHED_SUCCESS; +} + +static memcached_return server_stat(const MCD_SERVER server, const char *key, size_t key_length, const char *value, size_t value_length, void *context) +{ + struct stream_server *ss = (struct stream_server*) context; + + if (ss->server != server) { + server_show(NULL, server, context); + ss->server = (MCD_SERVER) server; + } + + ss->stream->write_function(ss->stream, " %s: %s\n", key, value); + return MEMCACHED_SUCCESS; +} + +#if !HAVE_MEMCACHED_STAT_EXECUTE +static memcached_return server_stats(memcached_st *ptr, MCD_SERVER server, void *context) +{ + char **keys, **key, *value; + memcached_stat_st stat; + memcached_return rc; + + struct stream_server *ss = (struct stream_server*) context; + + rc = memcached_stat_servername(&stat, NULL, MCD_SERVER_NAME, MCD_SERVER_PORT); + if (rc != MEMCACHED_SUCCESS) goto mcache_error; + + keys = memcached_stat_get_keys(ptr, &stat, &rc); + if (rc != MEMCACHED_SUCCESS) goto mcache_error; + + for (key = keys; *key; key++) { + value = memcached_stat_get_value(ptr, &stat, *key, &rc); + + if (rc == MEMCACHED_SUCCESS && value) { + server_stat(server, *key, 0, value, 0, context); + switch_safe_free(value); + } else { + server_stat(server, *key, 0, "N/A", 0, context); + } + } + + switch_safe_free(keys); + return MEMCACHED_SUCCESS; + +mcache_error: + ss->stream->write_function(ss->stream, "-ERR %s\n", memcached_strerror(ptr, rc)); + return MEMCACHED_SUCCESS; +} +#endif + SWITCH_STANDARD_API(memcache_function) { switch_status_t status; @@ -157,7 +233,6 @@ SWITCH_STANDARD_API(memcache_function) memcached_return rc; memcached_st *memcached = NULL; memcached_stat_st *stat = NULL; - memcached_server_st *server_list; if (zstr(cmd)) { goto usage; @@ -303,41 +378,29 @@ SWITCH_STANDARD_API(memcache_function) switch_goto_status(SWITCH_STATUS_SUCCESS, mcache_error); } } else if (!strcasecmp(subcmd, "status")) { - switch_bool_t verbose = SWITCH_FALSE; - int x; - - if (argc > 1) { - if (!strcasecmp(argv[1], "verbose")) { - verbose = SWITCH_TRUE; - } - } + struct stream_server ss; + ss.stream = stream; + ss.server = NULL; stream->write_function(stream, "Lib version: %s\n", memcached_lib_version()); - stat = memcached_stat(memcached, NULL, &rc); - if (rc != MEMCACHED_SUCCESS && rc != MEMCACHED_SOME_ERRORS) { - stream->write_function(stream, "-ERR Error communicating with servers (%s)\n", memcached_strerror(memcached, rc)); - } - server_list = memcached_server_list(memcached); + server_count = memcached_server_count(memcached); stream->write_function(stream, "Servers: %d\n", server_count); - for (x = 0; x < server_count; x++) { - stream->write_function(stream, " %s (%u)\n", memcached_server_name(memcached, server_list[x]), - memcached_server_port(memcached, server_list[x])); - if (verbose == SWITCH_TRUE) { - char **list; - char **ptr; - char *value; - memcached_return rc2; - list = memcached_stat_get_keys(memcached, &stat[x], &rc); - for (ptr = list; *ptr; ptr++) { - value = memcached_stat_get_value(memcached, &stat[x], *ptr, &rc2); - stream->write_function(stream, " %s: %s\n", *ptr, value); - switch_safe_free(value); - } - switch_safe_free(list); - stream->write_function(stream, "\n"); - } + if (argc > 1 && !strcasecmp(argv[1], "verbose")) { +#if HAVE_MEMCACHED_STAT_EXECUTE + rc = memcached_stat_execute(memcached, NULL, server_stat, (void*) &ss); +#else + memcached_server_function callbacks[] = { (memcached_server_function) server_stats }; + rc = memcached_server_cursor(memcached, callbacks, (void*) &ss, 1); +#endif + } else { + memcached_server_function callbacks[] = { (memcached_server_function) server_show }; + rc = memcached_server_cursor(memcached, callbacks, (void*) &ss, 1); + } + + if (rc != MEMCACHED_SUCCESS) { + switch_goto_status(SWITCH_STATUS_SUCCESS, mcache_error); } } else { goto usage;