Blame modules/ldap/util_ldap_cache.c

Packit 90a5c9
/* Licensed to the Apache Software Foundation (ASF) under one or more
Packit 90a5c9
 * contributor license agreements.  See the NOTICE file distributed with
Packit 90a5c9
 * this work for additional information regarding copyright ownership.
Packit 90a5c9
 * The ASF licenses this file to You under the Apache License, Version 2.0
Packit 90a5c9
 * (the "License"); you may not use this file except in compliance with
Packit 90a5c9
 * the License.  You may obtain a copy of the License at
Packit 90a5c9
 *
Packit 90a5c9
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 90a5c9
 *
Packit 90a5c9
 * Unless required by applicable law or agreed to in writing, software
Packit 90a5c9
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 90a5c9
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 90a5c9
 * See the License for the specific language governing permissions and
Packit 90a5c9
 * limitations under the License.
Packit 90a5c9
 */
Packit 90a5c9
Packit 90a5c9
/*
Packit 90a5c9
 * util_ldap_cache.c: LDAP cache things
Packit 90a5c9
 *
Packit 90a5c9
 * Original code from auth_ldap module for Apache v1.3:
Packit 90a5c9
 * Copyright 1998, 1999 Enbridge Pipelines Inc.
Packit 90a5c9
 * Copyright 1999-2001 Dave Carrigan
Packit 90a5c9
 */
Packit 90a5c9
Packit 90a5c9
#include "httpd.h"
Packit 90a5c9
#include "util_ldap.h"
Packit 90a5c9
#include "util_ldap_cache.h"
Packit 90a5c9
#include <apr_strings.h>
Packit 90a5c9
Packit 90a5c9
#if APR_HAS_LDAP
Packit 90a5c9
Packit 90a5c9
/* ------------------------------------------------------------------ */
Packit 90a5c9
Packit 90a5c9
unsigned long util_ldap_url_node_hash(void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_url_node_t *node = n;
Packit 90a5c9
    return util_ald_hash_string(1, node->url);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
int util_ldap_url_node_compare(void *a, void *b)
Packit 90a5c9
{
Packit 90a5c9
    util_url_node_t *na = a;
Packit 90a5c9
    util_url_node_t *nb = b;
Packit 90a5c9
Packit 90a5c9
    return (strcmp(na->url, nb->url) == 0);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void *util_ldap_url_node_copy(util_ald_cache_t *cache, void *c)
Packit 90a5c9
{
Packit 90a5c9
    util_url_node_t *n = c;
Packit 90a5c9
    util_url_node_t *node = util_ald_alloc(cache, sizeof *node);
Packit 90a5c9
Packit 90a5c9
    if (node) {
Packit 90a5c9
        if (!(node->url = util_ald_strdup(cache, n->url))) {
Packit 90a5c9
            util_ald_free(cache, node);
Packit 90a5c9
            return NULL;
Packit 90a5c9
        }
Packit 90a5c9
        node->search_cache = n->search_cache;
Packit 90a5c9
        node->compare_cache = n->compare_cache;
Packit 90a5c9
        node->dn_compare_cache = n->dn_compare_cache;
Packit 90a5c9
        return node;
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        return NULL;
Packit 90a5c9
    }
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void util_ldap_url_node_free(util_ald_cache_t *cache, void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_url_node_t *node = n;
Packit 90a5c9
Packit 90a5c9
    util_ald_free(cache, node->url);
Packit 90a5c9
    util_ald_destroy_cache(node->search_cache);
Packit 90a5c9
    util_ald_destroy_cache(node->compare_cache);
Packit 90a5c9
    util_ald_destroy_cache(node->dn_compare_cache);
Packit 90a5c9
    util_ald_free(cache, node);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void util_ldap_url_node_display(request_rec *r, util_ald_cache_t *cache, void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_url_node_t *node = n;
Packit 90a5c9
    char date_str[APR_CTIME_LEN];
Packit 90a5c9
    const char *type_str;
Packit 90a5c9
    util_ald_cache_t *cache_node;
Packit 90a5c9
    int x;
Packit 90a5c9
Packit 90a5c9
    for (x=0;x<3;x++) {
Packit 90a5c9
        switch (x) {
Packit 90a5c9
            case 0:
Packit 90a5c9
                cache_node = node->search_cache;
Packit 90a5c9
                type_str = "Searches";
Packit 90a5c9
                break;
Packit 90a5c9
            case 1:
Packit 90a5c9
                cache_node = node->compare_cache;
Packit 90a5c9
                type_str = "Compares";
Packit 90a5c9
                break;
Packit 90a5c9
            case 2:
Packit 90a5c9
            default:
Packit 90a5c9
                cache_node = node->dn_compare_cache;
Packit 90a5c9
                type_str = "DN Compares";
Packit 90a5c9
                break;
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        if (cache_node->marktime) {
Packit 90a5c9
            apr_ctime(date_str, cache_node->marktime);
Packit 90a5c9
        }
Packit 90a5c9
        else
Packit 90a5c9
            date_str[0] = 0;
Packit 90a5c9
Packit 90a5c9
        ap_rprintf(r,
Packit 90a5c9
                   ""
Packit 90a5c9
                   "%s (%s)"
Packit 90a5c9
                   "%ld"
Packit 90a5c9
                   "%ld"
Packit 90a5c9
                   "%ld"
Packit 90a5c9
                   "%" APR_TIME_T_FMT ""
Packit 90a5c9
                   "%ld"
Packit 90a5c9
                   "%s"
Packit 90a5c9
                   "",
Packit 90a5c9
                   node->url,
Packit 90a5c9
                   type_str,
Packit 90a5c9
                   cache_node->size,
Packit 90a5c9
                   cache_node->maxentries,
Packit 90a5c9
                   cache_node->numentries,
Packit 90a5c9
                   apr_time_sec(cache_node->ttl),
Packit 90a5c9
                   cache_node->fullmark,
Packit 90a5c9
                   date_str);
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
/* ------------------------------------------------------------------ */
Packit 90a5c9
Packit 90a5c9
/* Cache functions for search nodes */
Packit 90a5c9
unsigned long util_ldap_search_node_hash(void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_search_node_t *node = n;
Packit 90a5c9
    return util_ald_hash_string(1, node->username);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
int util_ldap_search_node_compare(void *a, void *b)
Packit 90a5c9
{
Packit 90a5c9
    util_search_node_t *na = a;
Packit 90a5c9
    util_search_node_t *nb = b;
Packit 90a5c9
Packit 90a5c9
    return (strcmp(na->username, nb->username) == 0);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void *util_ldap_search_node_copy(util_ald_cache_t *cache, void *c)
Packit 90a5c9
{
Packit 90a5c9
    util_search_node_t *node = c;
Packit 90a5c9
    util_search_node_t *newnode = util_ald_alloc(cache, sizeof *newnode);
Packit 90a5c9
Packit 90a5c9
    /* safety check */
Packit 90a5c9
    if (newnode) {
Packit 90a5c9
Packit 90a5c9
        /* copy vals */
Packit 90a5c9
        if (node->vals) {
Packit 90a5c9
            int k = node->numvals;
Packit 90a5c9
            int i = 0;
Packit 90a5c9
            if (!(newnode->vals = util_ald_alloc(cache, sizeof(char *) * (k+1)))) {
Packit 90a5c9
                util_ldap_search_node_free(cache, newnode);
Packit 90a5c9
                return NULL;
Packit 90a5c9
            }
Packit 90a5c9
            newnode->numvals = node->numvals;
Packit 90a5c9
            for (;k;k--) {
Packit 90a5c9
                if (node->vals[i]) {
Packit 90a5c9
                    if (!(newnode->vals[i] = util_ald_strdup(cache, node->vals[i]))) {
Packit 90a5c9
                        util_ldap_search_node_free(cache, newnode);
Packit 90a5c9
                        return NULL;
Packit 90a5c9
                    }
Packit 90a5c9
                }
Packit 90a5c9
                else
Packit 90a5c9
                    newnode->vals[i] = NULL;
Packit 90a5c9
                i++;
Packit 90a5c9
            }
Packit 90a5c9
        }
Packit 90a5c9
        else {
Packit 90a5c9
            newnode->vals = NULL;
Packit 90a5c9
        }
Packit 90a5c9
        if (!(newnode->username = util_ald_strdup(cache, node->username)) ||
Packit 90a5c9
            !(newnode->dn = util_ald_strdup(cache, node->dn)) ) {
Packit 90a5c9
            util_ldap_search_node_free(cache, newnode);
Packit 90a5c9
            return NULL;
Packit 90a5c9
        }
Packit 90a5c9
        if (node->bindpw) {
Packit 90a5c9
            if (!(newnode->bindpw = util_ald_strdup(cache, node->bindpw))) {
Packit 90a5c9
                util_ldap_search_node_free(cache, newnode);
Packit 90a5c9
                return NULL;
Packit 90a5c9
            }
Packit 90a5c9
        } else {
Packit 90a5c9
            newnode->bindpw = NULL;
Packit 90a5c9
        }
Packit 90a5c9
        newnode->lastbind = node->lastbind;
Packit 90a5c9
Packit 90a5c9
    }
Packit 90a5c9
    return (void *)newnode;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void util_ldap_search_node_free(util_ald_cache_t *cache, void *n)
Packit 90a5c9
{
Packit 90a5c9
    int i = 0;
Packit 90a5c9
    util_search_node_t *node = n;
Packit 90a5c9
    int k = node->numvals;
Packit 90a5c9
Packit 90a5c9
    if (node->vals) {
Packit 90a5c9
        for (;k;k--,i++) {
Packit 90a5c9
            if (node->vals[i]) {
Packit 90a5c9
                util_ald_free(cache, node->vals[i]);
Packit 90a5c9
            }
Packit 90a5c9
        }
Packit 90a5c9
        util_ald_free(cache, node->vals);
Packit 90a5c9
    }
Packit 90a5c9
    util_ald_free(cache, node->username);
Packit 90a5c9
    util_ald_free(cache, node->dn);
Packit 90a5c9
    util_ald_free(cache, node->bindpw);
Packit 90a5c9
    util_ald_free(cache, node);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void util_ldap_search_node_display(request_rec *r, util_ald_cache_t *cache, void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_search_node_t *node = n;
Packit 90a5c9
    char date_str[APR_CTIME_LEN];
Packit 90a5c9
Packit 90a5c9
    apr_ctime(date_str, node->lastbind);
Packit 90a5c9
Packit 90a5c9
    ap_rprintf(r,
Packit 90a5c9
               ""
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "",
Packit 90a5c9
               node->username,
Packit 90a5c9
               node->dn,
Packit 90a5c9
               date_str);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
/* ------------------------------------------------------------------ */
Packit 90a5c9
Packit 90a5c9
unsigned long util_ldap_compare_node_hash(void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_compare_node_t *node = n;
Packit 90a5c9
    return util_ald_hash_string(3, node->dn, node->attrib, node->value);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
int util_ldap_compare_node_compare(void *a, void *b)
Packit 90a5c9
{
Packit 90a5c9
    util_compare_node_t *na = a;
Packit 90a5c9
    util_compare_node_t *nb = b;
Packit 90a5c9
Packit 90a5c9
    return (strcmp(na->dn, nb->dn) == 0 &&
Packit 90a5c9
            strcmp(na->attrib, nb->attrib) == 0 &&
Packit 90a5c9
            strcmp(na->value, nb->value) == 0);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void *util_ldap_compare_node_copy(util_ald_cache_t *cache, void *c)
Packit 90a5c9
{
Packit 90a5c9
    util_compare_node_t *n = c;
Packit 90a5c9
    util_compare_node_t *node = util_ald_alloc(cache, sizeof *node);
Packit 90a5c9
Packit 90a5c9
    if (node) {
Packit 90a5c9
        if (!(node->dn = util_ald_strdup(cache, n->dn)) ||
Packit 90a5c9
            !(node->attrib = util_ald_strdup(cache, n->attrib)) ||
Packit 90a5c9
            !(node->value = util_ald_strdup(cache, n->value)) ||
Packit 90a5c9
            ((n->subgroupList) && !(node->subgroupList = util_ald_sgl_dup(cache, n->subgroupList)))) {
Packit 90a5c9
            util_ldap_compare_node_free(cache, node);
Packit 90a5c9
            return NULL;
Packit 90a5c9
        }
Packit 90a5c9
        node->lastcompare = n->lastcompare;
Packit 90a5c9
        node->result = n->result;
Packit 90a5c9
        node->sgl_processed = n->sgl_processed;
Packit 90a5c9
        return node;
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        return NULL;
Packit 90a5c9
    }
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void util_ldap_compare_node_free(util_ald_cache_t *cache, void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_compare_node_t *node = n;
Packit 90a5c9
Packit 90a5c9
    util_ald_sgl_free(cache, &(node->subgroupList));
Packit 90a5c9
    util_ald_free(cache, node->dn);
Packit 90a5c9
    util_ald_free(cache, node->attrib);
Packit 90a5c9
    util_ald_free(cache, node->value);
Packit 90a5c9
    util_ald_free(cache, node);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void util_ldap_compare_node_display(request_rec *r, util_ald_cache_t *cache, void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_compare_node_t *node = n;
Packit 90a5c9
    char date_str[APR_CTIME_LEN];
Packit 90a5c9
    char *cmp_result;
Packit 90a5c9
    char *sub_groups_val;
Packit 90a5c9
    char *sub_groups_checked;
Packit 90a5c9
Packit 90a5c9
    apr_ctime(date_str, node->lastcompare);
Packit 90a5c9
Packit 90a5c9
    if (node->result == LDAP_COMPARE_TRUE) {
Packit 90a5c9
        cmp_result = "LDAP_COMPARE_TRUE";
Packit 90a5c9
    }
Packit 90a5c9
    else if (node->result == LDAP_COMPARE_FALSE) {
Packit 90a5c9
        cmp_result = "LDAP_COMPARE_FALSE";
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        cmp_result = apr_itoa(r->pool, node->result);
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    if (node->subgroupList) {
Packit 90a5c9
        sub_groups_val = "Yes";
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        sub_groups_val = "No";
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    if (node->sgl_processed) {
Packit 90a5c9
        sub_groups_checked = "Yes";
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        sub_groups_checked = "No";
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    ap_rprintf(r,
Packit 90a5c9
               ""
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "",
Packit 90a5c9
               node->dn,
Packit 90a5c9
               node->attrib,
Packit 90a5c9
               node->value,
Packit 90a5c9
               date_str,
Packit 90a5c9
               cmp_result,
Packit 90a5c9
               sub_groups_val,
Packit 90a5c9
               sub_groups_checked);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
/* ------------------------------------------------------------------ */
Packit 90a5c9
Packit 90a5c9
unsigned long util_ldap_dn_compare_node_hash(void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_dn_compare_node_t *node = n;
Packit 90a5c9
    return util_ald_hash_string(1, node->reqdn);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
int util_ldap_dn_compare_node_compare(void *a, void *b)
Packit 90a5c9
{
Packit 90a5c9
    util_dn_compare_node_t *na = a;
Packit 90a5c9
    util_dn_compare_node_t *nb = b;
Packit 90a5c9
Packit 90a5c9
    return (strcmp(na->reqdn, nb->reqdn) == 0);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void *util_ldap_dn_compare_node_copy(util_ald_cache_t *cache, void *c)
Packit 90a5c9
{
Packit 90a5c9
    util_dn_compare_node_t *n = c;
Packit 90a5c9
    util_dn_compare_node_t *node = util_ald_alloc(cache, sizeof *node);
Packit 90a5c9
Packit 90a5c9
    if (node) {
Packit 90a5c9
        if (!(node->reqdn = util_ald_strdup(cache, n->reqdn)) ||
Packit 90a5c9
            !(node->dn = util_ald_strdup(cache, n->dn))) {
Packit 90a5c9
            util_ldap_dn_compare_node_free(cache, node);
Packit 90a5c9
            return NULL;
Packit 90a5c9
        }
Packit 90a5c9
        return node;
Packit 90a5c9
    }
Packit 90a5c9
    else {
Packit 90a5c9
        return NULL;
Packit 90a5c9
    }
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void util_ldap_dn_compare_node_free(util_ald_cache_t *cache, void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_dn_compare_node_t *node = n;
Packit 90a5c9
    util_ald_free(cache, node->reqdn);
Packit 90a5c9
    util_ald_free(cache, node->dn);
Packit 90a5c9
    util_ald_free(cache, node);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
void util_ldap_dn_compare_node_display(request_rec *r, util_ald_cache_t *cache, void *n)
Packit 90a5c9
{
Packit 90a5c9
    util_dn_compare_node_t *node = n;
Packit 90a5c9
Packit 90a5c9
    ap_rprintf(r,
Packit 90a5c9
               ""
Packit 90a5c9
               "%s"
Packit 90a5c9
               "%s"
Packit 90a5c9
               "",
Packit 90a5c9
               node->reqdn,
Packit 90a5c9
               node->dn);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
/* ------------------------------------------------------------------ */
Packit 90a5c9
static apr_status_t util_ldap_cache_module_kill(void *data)
Packit 90a5c9
{
Packit 90a5c9
    util_ldap_state_t *st = data;
Packit 90a5c9
Packit 90a5c9
    util_ald_destroy_cache(st->util_ldap_cache);
Packit 90a5c9
#if APR_HAS_SHARED_MEMORY
Packit 90a5c9
    if (st->cache_rmm != NULL) {
Packit 90a5c9
        apr_rmm_destroy (st->cache_rmm);
Packit 90a5c9
        st->cache_rmm = NULL;
Packit 90a5c9
    }
Packit 90a5c9
    if (st->cache_shm != NULL) {
Packit 90a5c9
        apr_status_t result = apr_shm_destroy(st->cache_shm);
Packit 90a5c9
        st->cache_shm = NULL;
Packit 90a5c9
        return result;
Packit 90a5c9
    }
Packit 90a5c9
#endif
Packit 90a5c9
    return APR_SUCCESS;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
apr_status_t util_ldap_cache_init(apr_pool_t *pool, util_ldap_state_t *st)
Packit 90a5c9
{
Packit 90a5c9
#if APR_HAS_SHARED_MEMORY
Packit 90a5c9
    apr_status_t result;
Packit 90a5c9
    apr_size_t size;
Packit 90a5c9
Packit 90a5c9
    if (st->cache_bytes > 0) {
Packit 90a5c9
        if (st->cache_file) {
Packit 90a5c9
            /* Remove any existing shm segment with this name. */
Packit 90a5c9
            apr_shm_remove(st->cache_file, st->pool);
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        size = APR_ALIGN_DEFAULT(st->cache_bytes);
Packit 90a5c9
Packit 90a5c9
        result = apr_shm_create(&st->cache_shm, size, st->cache_file, st->pool);
Packit 90a5c9
        if (result != APR_SUCCESS) {
Packit 90a5c9
            return result;
Packit 90a5c9
        }
Packit 90a5c9
Packit 90a5c9
        /* Determine the usable size of the shm segment. */
Packit 90a5c9
        size = apr_shm_size_get(st->cache_shm);
Packit 90a5c9
Packit 90a5c9
        /* This will create a rmm "handler" to get into the shared memory area */
Packit 90a5c9
        result = apr_rmm_init(&st->cache_rmm, NULL,
Packit 90a5c9
                              apr_shm_baseaddr_get(st->cache_shm), size,
Packit 90a5c9
                              st->pool);
Packit 90a5c9
        if (result != APR_SUCCESS) {
Packit 90a5c9
            return result;
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
#endif
Packit 90a5c9
Packit 90a5c9
    apr_pool_cleanup_register(st->pool, st , util_ldap_cache_module_kill, apr_pool_cleanup_null);
Packit 90a5c9
Packit 90a5c9
    st->util_ldap_cache =
Packit 90a5c9
        util_ald_create_cache(st,
Packit 90a5c9
                              st->search_cache_size,
Packit 90a5c9
                              st->search_cache_ttl,
Packit 90a5c9
                              util_ldap_url_node_hash,
Packit 90a5c9
                              util_ldap_url_node_compare,
Packit 90a5c9
                              util_ldap_url_node_copy,
Packit 90a5c9
                              util_ldap_url_node_free,
Packit 90a5c9
                              util_ldap_url_node_display);
Packit 90a5c9
    return APR_SUCCESS;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
#endif /* APR_HAS_LDAP */