From: Jim Jagielski Date: Mon, 19 Sep 2005 12:34:32 +0000 (+0000) Subject: Backport 279750 from TRUNK. X-Git-Tag: 2.1.8~49 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=ebee95b9a9c830f0be1032cb531fcfff7e6d40e5;p=thirdparty%2Fapache%2Fhttpd.git Backport 279750 from TRUNK. New provider function to list all available provider names in a specific group and version (ap_list_provider_names). git-svn-id: https://svn.apache.org/repos/asf/httpd/httpd/branches/2.2.x@290137 13f79535-47bb-0310-9956-ffa450edef68 --- diff --git a/CHANGES b/CHANGES index 9113d33879f..2d312edb0c8 100644 --- a/CHANGES +++ b/CHANGES @@ -1,6 +1,9 @@ -*- coding: utf-8 -*- Changes with Apache 2.1.8 + *) New provider function to list all available provider names in a + specific group and version (ap_list_provider_names). [Jim Jagielski] + *) mod_cache: Enhance CacheEnable/CacheDisable to control caching on a per-protocol, per-host and per-path basis. Intended for proxy configurations. [Colm MacCarthaigh] diff --git a/include/ap_provider.h b/include/ap_provider.h index c05da554c5a..8ccb27d24c6 100644 --- a/include/ap_provider.h +++ b/include/ap_provider.h @@ -19,6 +19,11 @@ #include "ap_config.h" +typedef struct { + const char *provider_name; +} ap_list_provider_names_t; + + /** * @package Provider API */ @@ -51,4 +56,17 @@ AP_DECLARE(void *) ap_lookup_provider(const char *provider_group, const char *provider_name, const char *provider_version); +/** + * This function is used to retrieve a list (array) of provider + * names from the specified group with the specified version. + * @param pool The pool to create any storage from + * @param provider_group The group to look for this provider in + * @param provider_version The version for the provider + * @return pointer to array of ap_list_provider_names_t of provider names (could be empty) + */ + +AP_DECLARE(apr_array_header_t *) ap_list_provider_names(apr_pool_t *pool, + const char *provider_group, + const char *provider_version); + #endif diff --git a/server/provider.c b/server/provider.c index 97d2a0f9f45..00d6954e896 100644 --- a/server/provider.c +++ b/server/provider.c @@ -16,14 +16,19 @@ #include "apr_pools.h" #include "apr_hash.h" +#include "apr_tables.h" +#include "apr_strings.h" #include "ap_provider.h" static apr_hash_t *global_providers = NULL; +static apr_hash_t *global_providers_names = NULL; + static apr_status_t cleanup_global_providers(void *ctx) { global_providers = NULL; + global_providers_names = NULL; return APR_SUCCESS; } @@ -33,15 +38,16 @@ AP_DECLARE(apr_status_t) ap_register_provider(apr_pool_t *pool, const char *provider_version, const void *provider) { - apr_hash_t *provider_group_hash; - apr_hash_t *provider_version_hash; + apr_hash_t *provider_group_hash, *provider_version_hash; if (global_providers == NULL) { global_providers = apr_hash_make(pool); + global_providers_names = apr_hash_make(pool);; apr_pool_cleanup_register(pool, NULL, cleanup_global_providers, apr_pool_cleanup_null); } + /* First, deal with storing the provider away */ provider_group_hash = apr_hash_get(global_providers, provider_group, APR_HASH_KEY_STRING); @@ -66,6 +72,31 @@ AP_DECLARE(apr_status_t) ap_register_provider(apr_pool_t *pool, apr_hash_set(provider_version_hash, provider_version, APR_HASH_KEY_STRING, provider); + /* Now, tuck away the provider names in an easy-to-get format */ + provider_group_hash = apr_hash_get(global_providers_names, provider_group, + APR_HASH_KEY_STRING); + + if (!provider_group_hash) { + provider_group_hash = apr_hash_make(pool); + apr_hash_set(global_providers_names, provider_group, APR_HASH_KEY_STRING, + provider_group_hash); + + } + + provider_version_hash = apr_hash_get(provider_group_hash, provider_version, + APR_HASH_KEY_STRING); + + if (!provider_version_hash) { + provider_version_hash = apr_hash_make(pool); + apr_hash_set(provider_group_hash, provider_version, APR_HASH_KEY_STRING, + provider_version_hash); + + } + + /* just set it. no biggy if it was there before. */ + apr_hash_set(provider_version_hash, provider_name, APR_HASH_KEY_STRING, + provider_name); + return APR_SUCCESS; } @@ -96,3 +127,39 @@ AP_DECLARE(void *) ap_lookup_provider(const char *provider_group, return apr_hash_get(provider_name_hash, provider_version, APR_HASH_KEY_STRING); } + +AP_DECLARE(apr_array_header_t *) ap_list_provider_names(apr_pool_t *pool, + const char *provider_group, + const char *provider_version) +{ + apr_array_header_t *ret = apr_array_make(pool, 10, sizeof(ap_list_provider_names_t)); + ap_list_provider_names_t *entry; + apr_hash_t *provider_group_hash, *h; + apr_hash_index_t *hi; + char *val, *key; + + if (global_providers_names == NULL) { + return ret; + } + + provider_group_hash = apr_hash_get(global_providers_names, provider_group, + APR_HASH_KEY_STRING); + + if (provider_group_hash == NULL) { + return ret; + } + + h = apr_hash_get(provider_group_hash, provider_version, + APR_HASH_KEY_STRING); + + if (h == NULL) { + return ret; + } + + for (hi = apr_hash_first(pool, h); hi; hi = apr_hash_next(hi)) { + apr_hash_this(hi, (void *)&key, NULL, (void *)&val); + entry = apr_array_push(ret); + entry->provider_name = apr_pstrdup(pool, val); + } + return ret; +}