Blame support/htdbm.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
 * htdbm.c: simple program for manipulating DBM
Packit 90a5c9
 * password databases for the Apache HTTP server
Packit 90a5c9
 *
Packit 90a5c9
 * Contributed by Mladen Turk <mturk mappingsoft.com>
Packit 90a5c9
 * 12 Oct 2001
Packit 90a5c9
 */
Packit 90a5c9
Packit 90a5c9
#include "passwd_common.h"
Packit 90a5c9
#include "apr_file_io.h"
Packit 90a5c9
#include "apr_file_info.h"
Packit 90a5c9
#include "apr_pools.h"
Packit 90a5c9
#include "apr_signal.h"
Packit 90a5c9
#include "apr_md5.h"
Packit 90a5c9
#include "apr_sha1.h"
Packit 90a5c9
#include "apr_dbm.h"
Packit 90a5c9
#include "apr_getopt.h"
Packit 90a5c9
Packit 90a5c9
#if APR_HAVE_STDLIB_H
Packit 90a5c9
#include <stdlib.h>
Packit 90a5c9
#endif
Packit 90a5c9
#if APR_HAVE_STRING_H
Packit 90a5c9
#include <string.h>
Packit 90a5c9
#endif
Packit 90a5c9
#if APR_HAVE_STRINGS_H
Packit 90a5c9
#include <strings.h>
Packit 90a5c9
#endif
Packit 90a5c9
#include <time.h>
Packit 90a5c9
Packit 90a5c9
#if APR_CHARSET_EBCDIC
Packit 90a5c9
#include "apr_xlate.h"
Packit 90a5c9
#endif /*APR_CHARSET_EBCDIC*/
Packit 90a5c9
Packit 90a5c9
#if APR_HAVE_UNISTD_H
Packit 90a5c9
#include <unistd.h>
Packit 90a5c9
#endif
Packit 90a5c9
#if APR_HAVE_CRYPT_H
Packit 90a5c9
#include <crypt.h>
Packit 90a5c9
#endif
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
typedef struct htdbm_t htdbm_t;
Packit 90a5c9
Packit 90a5c9
struct htdbm_t {
Packit 90a5c9
    apr_dbm_t               *dbm;
Packit 90a5c9
    struct passwd_ctx       ctx;
Packit 90a5c9
#if APR_CHARSET_EBCDIC
Packit 90a5c9
    apr_xlate_t             *to_ascii;
Packit 90a5c9
#endif
Packit 90a5c9
    char                    *filename;
Packit 90a5c9
    char                    *username;
Packit 90a5c9
    char                    *comment;
Packit 90a5c9
    char                    *type;
Packit 90a5c9
    int                     create;
Packit 90a5c9
    int                     rdonly;
Packit 90a5c9
};
Packit 90a5c9
Packit 90a5c9
Packit 90a5c9
#define HTDBM_MAKE   0
Packit 90a5c9
#define HTDBM_DELETE 1
Packit 90a5c9
#define HTDBM_VERIFY 2
Packit 90a5c9
#define HTDBM_LIST   3
Packit 90a5c9
#define HTDBM_NOFILE 4
Packit 90a5c9
Packit 90a5c9
static void terminate(void)
Packit 90a5c9
{
Packit 90a5c9
    apr_terminate();
Packit 90a5c9
#ifdef NETWARE
Packit 90a5c9
    pressanykey();
Packit 90a5c9
#endif
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static void htdbm_terminate(htdbm_t *htdbm)
Packit 90a5c9
{
Packit 90a5c9
    if (htdbm->dbm)
Packit 90a5c9
        apr_dbm_close(htdbm->dbm);
Packit 90a5c9
    htdbm->dbm = NULL;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static htdbm_t *h;
Packit 90a5c9
Packit 90a5c9
static void htdbm_interrupted(void)
Packit 90a5c9
{
Packit 90a5c9
    htdbm_terminate(h);
Packit 90a5c9
    fprintf(stderr, "htdbm Interrupted !\n");
Packit 90a5c9
    exit(ERR_INTERRUPTED);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static apr_status_t htdbm_init(apr_pool_t **pool, htdbm_t **hdbm)
Packit 90a5c9
{
Packit 90a5c9
Packit 90a5c9
#if APR_CHARSET_EBCDIC
Packit 90a5c9
    apr_status_t rv;
Packit 90a5c9
#endif
Packit 90a5c9
Packit 90a5c9
    apr_pool_create( pool, NULL);
Packit 90a5c9
    apr_pool_abort_set(abort_on_oom, *pool);
Packit 90a5c9
    apr_file_open_stderr(&errfile, *pool);
Packit 90a5c9
    apr_signal(SIGINT, (void (*)(int)) htdbm_interrupted);
Packit 90a5c9
Packit 90a5c9
    (*hdbm) = (htdbm_t *)apr_pcalloc(*pool, sizeof(htdbm_t));
Packit 90a5c9
    (*hdbm)->ctx.pool = *pool;
Packit 90a5c9
Packit 90a5c9
#if APR_CHARSET_EBCDIC
Packit 90a5c9
    rv = apr_xlate_open(&((*hdbm)->to_ascii), "ISO-8859-1", APR_DEFAULT_CHARSET, (*hdbm)->ctx.pool);
Packit 90a5c9
    if (rv) {
Packit 90a5c9
        fprintf(stderr, "apr_xlate_open(to ASCII)->%d\n", rv);
Packit 90a5c9
        return APR_EGENERAL;
Packit 90a5c9
    }
Packit 90a5c9
    rv = apr_SHA1InitEBCDIC((*hdbm)->to_ascii);
Packit 90a5c9
    if (rv) {
Packit 90a5c9
        fprintf(stderr, "apr_SHA1InitEBCDIC()->%d\n", rv);
Packit 90a5c9
        return APR_EGENERAL;
Packit 90a5c9
    }
Packit 90a5c9
    rv = apr_MD5InitEBCDIC((*hdbm)->to_ascii);
Packit 90a5c9
    if (rv) {
Packit 90a5c9
        fprintf(stderr, "apr_MD5InitEBCDIC()->%d\n", rv);
Packit 90a5c9
        return APR_EGENERAL;
Packit 90a5c9
    }
Packit 90a5c9
#endif /*APR_CHARSET_EBCDIC*/
Packit 90a5c9
Packit 90a5c9
    /* Set MD5 as default */
Packit 90a5c9
    (*hdbm)->ctx.alg = ALG_APMD5;
Packit 90a5c9
    (*hdbm)->type = "default";
Packit 90a5c9
    return APR_SUCCESS;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static apr_status_t htdbm_open(htdbm_t *htdbm)
Packit 90a5c9
{
Packit 90a5c9
    if (htdbm->create)
Packit 90a5c9
        return apr_dbm_open_ex(&htdbm->dbm, htdbm->type, htdbm->filename, APR_DBM_RWCREATE,
Packit 90a5c9
                            APR_OS_DEFAULT, htdbm->ctx.pool);
Packit 90a5c9
    else
Packit 90a5c9
        return apr_dbm_open_ex(&htdbm->dbm, htdbm->type, htdbm->filename,
Packit 90a5c9
                            htdbm->rdonly ? APR_DBM_READONLY : APR_DBM_READWRITE,
Packit 90a5c9
                            APR_OS_DEFAULT, htdbm->ctx.pool);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static apr_status_t htdbm_save(htdbm_t *htdbm, int *changed)
Packit 90a5c9
{
Packit 90a5c9
    apr_datum_t key, val;
Packit 90a5c9
Packit 90a5c9
    if (!htdbm->username)
Packit 90a5c9
        return APR_SUCCESS;
Packit 90a5c9
Packit 90a5c9
    key.dptr = htdbm->username;
Packit 90a5c9
    key.dsize = strlen(htdbm->username);
Packit 90a5c9
    if (apr_dbm_exists(htdbm->dbm, key))
Packit 90a5c9
        *changed = 1;
Packit 90a5c9
Packit 90a5c9
    val.dsize = strlen(htdbm->ctx.passwd);
Packit 90a5c9
    if (!htdbm->comment)
Packit 90a5c9
        val.dptr  = htdbm->ctx.passwd;
Packit 90a5c9
    else {
Packit 90a5c9
        val.dptr = apr_pstrcat(htdbm->ctx.pool, htdbm->ctx.passwd, ":",
Packit 90a5c9
                               htdbm->comment, NULL);
Packit 90a5c9
        val.dsize += (strlen(htdbm->comment) + 1);
Packit 90a5c9
    }
Packit 90a5c9
    return apr_dbm_store(htdbm->dbm, key, val);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static apr_status_t htdbm_del(htdbm_t *htdbm)
Packit 90a5c9
{
Packit 90a5c9
    apr_datum_t key;
Packit 90a5c9
Packit 90a5c9
    key.dptr = htdbm->username;
Packit 90a5c9
    key.dsize = strlen(htdbm->username);
Packit 90a5c9
    if (!apr_dbm_exists(htdbm->dbm, key))
Packit 90a5c9
        return APR_ENOENT;
Packit 90a5c9
Packit 90a5c9
    return apr_dbm_delete(htdbm->dbm, key);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static apr_status_t htdbm_verify(htdbm_t *htdbm)
Packit 90a5c9
{
Packit 90a5c9
    apr_datum_t key, val;
Packit 90a5c9
    char *pwd;
Packit 90a5c9
    char *rec, *cmnt;
Packit 90a5c9
Packit 90a5c9
    key.dptr = htdbm->username;
Packit 90a5c9
    key.dsize = strlen(htdbm->username);
Packit 90a5c9
    if (!apr_dbm_exists(htdbm->dbm, key))
Packit 90a5c9
        return APR_ENOENT;
Packit 90a5c9
    if (apr_dbm_fetch(htdbm->dbm, key, &val) != APR_SUCCESS)
Packit 90a5c9
        return APR_ENOENT;
Packit 90a5c9
    rec = apr_pstrndup(htdbm->ctx.pool, val.dptr, val.dsize);
Packit 90a5c9
    cmnt = strchr(rec, ':');
Packit 90a5c9
    if (cmnt)
Packit 90a5c9
        pwd = apr_pstrndup(htdbm->ctx.pool, rec, cmnt - rec);
Packit 90a5c9
    else
Packit 90a5c9
        pwd = apr_pstrdup(htdbm->ctx.pool, rec);
Packit 90a5c9
    return apr_password_validate(htdbm->ctx.passwd, pwd);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static apr_status_t htdbm_list(htdbm_t *htdbm)
Packit 90a5c9
{
Packit 90a5c9
    apr_status_t rv;
Packit 90a5c9
    apr_datum_t key, val;
Packit 90a5c9
    char *cmnt;
Packit 90a5c9
    int i = 0;
Packit 90a5c9
Packit 90a5c9
    rv = apr_dbm_firstkey(htdbm->dbm, &key);
Packit 90a5c9
    if (rv != APR_SUCCESS) {
Packit 90a5c9
        fprintf(stderr, "Empty database -- %s\n", htdbm->filename);
Packit 90a5c9
        return APR_ENOENT;
Packit 90a5c9
    }
Packit 90a5c9
    fprintf(stderr, "Dumping records from database -- %s\n", htdbm->filename);
Packit 90a5c9
    fprintf(stderr, "    %-32s Comment\n", "Username");
Packit 90a5c9
    while (key.dptr != NULL) {
Packit 90a5c9
        rv = apr_dbm_fetch(htdbm->dbm, key, &val;;
Packit 90a5c9
        if (rv != APR_SUCCESS) {
Packit 90a5c9
            fprintf(stderr, "Failed getting data from %s\n", htdbm->filename);
Packit 90a5c9
            return APR_EGENERAL;
Packit 90a5c9
        }
Packit 90a5c9
        /* Note: we don't store \0-terminators on our dbm data */
Packit 90a5c9
        fprintf(stderr, "    %-32.*s", (int)key.dsize, key.dptr);
Packit 90a5c9
        cmnt = memchr(val.dptr, ':', val.dsize);
Packit 90a5c9
        if (cmnt)
Packit 90a5c9
            fprintf(stderr, " %.*s", (int)(val.dptr+val.dsize - (cmnt+1)), cmnt + 1);
Packit 90a5c9
        fprintf(stderr, "\n");
Packit 90a5c9
        rv = apr_dbm_nextkey(htdbm->dbm, &key);
Packit 90a5c9
        if (rv != APR_SUCCESS)
Packit 90a5c9
            fprintf(stderr, "Failed getting NextKey\n");
Packit 90a5c9
        ++i;
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    fprintf(stderr, "Total #records : %d\n", i);
Packit 90a5c9
    return APR_SUCCESS;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static int htdbm_make(htdbm_t *htdbm)
Packit 90a5c9
{
Packit 90a5c9
    char cpw[MAX_STRING_LEN];
Packit 90a5c9
    int ret;
Packit 90a5c9
Packit 90a5c9
    htdbm->ctx.out = cpw;
Packit 90a5c9
    htdbm->ctx.out_len = sizeof(cpw);
Packit 90a5c9
    ret = mkhash(&htdbm->ctx);
Packit 90a5c9
    if (ret != 0) {
Packit 90a5c9
        fprintf(stderr, "Error: %s\n", htdbm->ctx.errstr);
Packit 90a5c9
        return ret;
Packit 90a5c9
    }
Packit 90a5c9
    htdbm->ctx.passwd = apr_pstrdup(htdbm->ctx.pool, cpw);
Packit 90a5c9
    return 0;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static apr_status_t htdbm_valid_username(htdbm_t *htdbm)
Packit 90a5c9
{
Packit 90a5c9
    if (!htdbm->username || (strlen(htdbm->username) > 64) || (strlen(htdbm->username) < 1)) {
Packit 90a5c9
        fprintf(stderr, "Invalid username length\n");
Packit 90a5c9
        return APR_EINVAL;
Packit 90a5c9
    }
Packit 90a5c9
    if (strchr(htdbm->username, ':')) {
Packit 90a5c9
        fprintf(stderr, "Username contains invalid characters\n");
Packit 90a5c9
        return APR_EINVAL;
Packit 90a5c9
    }
Packit 90a5c9
    return APR_SUCCESS;
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
static void htdbm_usage(void)
Packit 90a5c9
{
Packit 90a5c9
    fprintf(stderr,
Packit 90a5c9
        "htdbm -- program for manipulating DBM password databases.\n\n"
Packit 90a5c9
        "Usage: htdbm   [-cimBdpstvx] [-C cost] [-TDBTYPE] database username\n"
Packit 90a5c9
        "                -b[cmBdptsv] [-C cost] [-TDBTYPE] database username password\n"
Packit 90a5c9
        "                -n[imBdpst]  [-C cost] username\n"
Packit 90a5c9
        "                -nb[mBdpst]  [-C cost] username password\n"
Packit 90a5c9
        "                -v[imBdps]   [-C cost] [-TDBTYPE] database username\n"
Packit 90a5c9
        "                -vb[mBdps]   [-C cost] [-TDBTYPE] database username password\n"
Packit 90a5c9
        "                -x                     [-TDBTYPE] database username\n"
Packit 90a5c9
        "                -l                     [-TDBTYPE] database\n"
Packit 90a5c9
        "Options:\n"
Packit 90a5c9
        "   -c   Create a new database.\n"
Packit 90a5c9
        "   -n   Don't update database; display results on stdout.\n"
Packit 90a5c9
        "   -b   Use the password from the command line rather than prompting for it.\n"
Packit 90a5c9
        "   -i   Read password from stdin without verification (for script usage).\n"
Packit 90a5c9
        "   -m   Force MD5 encryption of the password (default).\n"
Packit 90a5c9
        "   -B   Force BCRYPT encryption of the password (very secure).\n"
Packit 90a5c9
        "   -C   Set the computing time used for the bcrypt algorithm\n"
Packit 90a5c9
        "        (higher is more secure but slower, default: %d, valid: 4 to 31).\n"
Packit 90a5c9
        "   -d   Force CRYPT encryption of the password (8 chars max, insecure).\n"
Packit 90a5c9
        "   -s   Force SHA encryption of the password (insecure).\n"
Packit 90a5c9
        "   -p   Do not encrypt the password (plaintext, insecure).\n"
Packit 90a5c9
        "   -T   DBM Type (SDBM|GDBM|DB|default).\n"
Packit 90a5c9
        "   -l   Display usernames from database on stdout.\n"
Packit 90a5c9
        "   -v   Verify the username/password.\n"
Packit 90a5c9
        "   -x   Remove the username record from database.\n"
Packit 90a5c9
        "   -t   The last param is username comment.\n"
Packit 90a5c9
        "The SHA algorithm does not use a salt and is less secure than the "
Packit 90a5c9
        "MD5 algorithm.\n",
Packit 90a5c9
        BCRYPT_DEFAULT_COST);
Packit 90a5c9
    exit(ERR_SYNTAX);
Packit 90a5c9
}
Packit 90a5c9
Packit 90a5c9
int main(int argc, const char * const argv[])
Packit 90a5c9
{
Packit 90a5c9
    apr_pool_t *pool;
Packit 90a5c9
    apr_status_t rv;
Packit 90a5c9
    char errbuf[MAX_STRING_LEN];
Packit 90a5c9
    int  need_file = 1;
Packit 90a5c9
    int  need_user = 1;
Packit 90a5c9
    int  need_pwd  = 1;
Packit 90a5c9
    int  need_cmnt = 0;
Packit 90a5c9
    int  changed = 0;
Packit 90a5c9
    int  cmd = HTDBM_MAKE;
Packit 90a5c9
    int  i, ret, args_left = 2;
Packit 90a5c9
    apr_getopt_t *state;
Packit 90a5c9
    char opt;
Packit 90a5c9
    const char *opt_arg;
Packit 90a5c9
Packit 90a5c9
    apr_app_initialize(&argc, &argv, NULL);
Packit 90a5c9
    atexit(terminate);
Packit 90a5c9
Packit 90a5c9
    if ((rv = htdbm_init(&pool, &h)) != APR_SUCCESS) {
Packit 90a5c9
        fprintf(stderr, "Unable to initialize htdbm terminating!\n");
Packit 90a5c9
        apr_strerror(rv, errbuf, sizeof(errbuf));
Packit 90a5c9
        exit(1);
Packit 90a5c9
    }
Packit 90a5c9
Packit 90a5c9
    rv = apr_getopt_init(&state, pool, argc, argv);
Packit 90a5c9
    if (rv != APR_SUCCESS)
Packit 90a5c9
        exit(ERR_SYNTAX);
Packit 90a5c9
Packit 90a5c9
    while ((rv = apr_getopt(state, "cnmspdBbtivxlC:T:", &opt, &opt_arg)) == APR_SUCCESS) {
Packit 90a5c9
        switch (opt) {
Packit 90a5c9
        case 'c':
Packit 90a5c9
            h->create = 1;
Packit 90a5c9
            break;
Packit 90a5c9
        case 'n':
Packit 90a5c9
            need_file = 0;
Packit 90a5c9
            cmd = HTDBM_NOFILE;
Packit 90a5c9
            args_left--;
Packit 90a5c9
            break;
Packit 90a5c9
        case 'l':
Packit 90a5c9
            need_pwd = 0;
Packit 90a5c9
            need_user = 0;
Packit 90a5c9
            cmd = HTDBM_LIST;
Packit 90a5c9
            h->rdonly = 1;
Packit 90a5c9
            args_left--;
Packit 90a5c9
            break;
Packit 90a5c9
        case 't':
Packit 90a5c9
            need_cmnt = 1;
Packit 90a5c9
            args_left++;
Packit 90a5c9
            break;
Packit 90a5c9
        case 'T':
Packit 90a5c9
            h->type = apr_pstrdup(h->ctx.pool, opt_arg);
Packit 90a5c9
            break;
Packit 90a5c9
        case 'v':
Packit 90a5c9
            h->rdonly = 1;
Packit 90a5c9
            cmd = HTDBM_VERIFY;
Packit 90a5c9
            break;
Packit 90a5c9
        case 'x':
Packit 90a5c9
            need_pwd = 0;
Packit 90a5c9
            cmd = HTDBM_DELETE;
Packit 90a5c9
            break;
Packit 90a5c9
        default:
Packit 90a5c9
            ret = parse_common_options(&h->ctx, opt, opt_arg);
Packit 90a5c9
            if (ret) {
Packit 90a5c9
                fprintf(stderr, "Error: %s\n", h->ctx.errstr);
Packit 90a5c9
                exit(ret);
Packit 90a5c9
            }
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
    if (h->ctx.passwd_src == PW_ARG) {
Packit 90a5c9
            need_pwd = 0;
Packit 90a5c9
            args_left++;
Packit 90a5c9
    }
Packit 90a5c9
    /*
Packit 90a5c9
     * Make sure we still have exactly the right number of arguments left
Packit 90a5c9
     * (the filename, the username, and possibly the password if -b was
Packit 90a5c9
     * specified).
Packit 90a5c9
     */
Packit 90a5c9
    i = state->ind;
Packit 90a5c9
    if (rv != APR_EOF || argc - i != args_left)
Packit 90a5c9
        htdbm_usage();
Packit 90a5c9
Packit 90a5c9
    if (need_file) {
Packit 90a5c9
        h->filename = apr_pstrdup(h->ctx.pool, argv[i++]);
Packit 90a5c9
        if ((rv = htdbm_open(h)) != APR_SUCCESS) {
Packit 90a5c9
            fprintf(stderr, "Error opening database %s\n", h->filename);
Packit 90a5c9
            apr_strerror(rv, errbuf, sizeof(errbuf));
Packit 90a5c9
            fprintf(stderr,"%s\n",errbuf);
Packit 90a5c9
            exit(ERR_FILEPERM);
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
    if (need_user) {
Packit 90a5c9
        h->username = apr_pstrdup(pool, argv[i++]);
Packit 90a5c9
        if (htdbm_valid_username(h) != APR_SUCCESS)
Packit 90a5c9
            exit(ERR_BADUSER);
Packit 90a5c9
    }
Packit 90a5c9
    if (h->ctx.passwd_src == PW_ARG)
Packit 90a5c9
        h->ctx.passwd = apr_pstrdup(pool, argv[i++]);
Packit 90a5c9
Packit 90a5c9
    if (need_pwd) {
Packit 90a5c9
        ret = get_password(&h->ctx);
Packit 90a5c9
        if (ret) {
Packit 90a5c9
            fprintf(stderr, "Error: %s\n", h->ctx.errstr);
Packit 90a5c9
            exit(ret);
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
    if (need_cmnt)
Packit 90a5c9
        h->comment = apr_pstrdup(pool, argv[i++]);
Packit 90a5c9
Packit 90a5c9
    switch (cmd) {
Packit 90a5c9
        case HTDBM_VERIFY:
Packit 90a5c9
            if ((rv = htdbm_verify(h)) != APR_SUCCESS) {
Packit 90a5c9
                if (APR_STATUS_IS_ENOENT(rv)) {
Packit 90a5c9
                    fprintf(stderr, "The user '%s' could not be found in database\n", h->username);
Packit 90a5c9
                    exit(ERR_BADUSER);
Packit 90a5c9
                }
Packit 90a5c9
                else {
Packit 90a5c9
                    fprintf(stderr, "Password mismatch for user '%s'\n", h->username);
Packit 90a5c9
                    exit(ERR_PWMISMATCH);
Packit 90a5c9
                }
Packit 90a5c9
            }
Packit 90a5c9
            else
Packit 90a5c9
                fprintf(stderr, "Password validated for user '%s'\n", h->username);
Packit 90a5c9
            break;
Packit 90a5c9
        case HTDBM_DELETE:
Packit 90a5c9
            if (htdbm_del(h) != APR_SUCCESS) {
Packit 90a5c9
                fprintf(stderr, "Cannot find user '%s' in database\n", h->username);
Packit 90a5c9
                exit(ERR_BADUSER);
Packit 90a5c9
            }
Packit 90a5c9
            h->username = NULL;
Packit 90a5c9
            changed = 1;
Packit 90a5c9
            break;
Packit 90a5c9
        case HTDBM_LIST:
Packit 90a5c9
            htdbm_list(h);
Packit 90a5c9
            break;
Packit 90a5c9
        default:
Packit 90a5c9
            ret = htdbm_make(h);
Packit 90a5c9
            if (ret)
Packit 90a5c9
                exit(ret);
Packit 90a5c9
            break;
Packit 90a5c9
    }
Packit 90a5c9
    if (need_file && !h->rdonly) {
Packit 90a5c9
        if ((rv = htdbm_save(h, &changed)) != APR_SUCCESS) {
Packit 90a5c9
            apr_strerror(rv, errbuf, sizeof(errbuf));
Packit 90a5c9
            exit(ERR_FILEPERM);
Packit 90a5c9
        }
Packit 90a5c9
        fprintf(stdout, "Database %s %s.\n", h->filename,
Packit 90a5c9
                h->create ? "created" : (changed ? "modified" : "updated"));
Packit 90a5c9
    }
Packit 90a5c9
    if (cmd == HTDBM_NOFILE) {
Packit 90a5c9
        if (!need_cmnt) {
Packit 90a5c9
            fprintf(stderr, "%s:%s\n", h->username, h->ctx.passwd);
Packit 90a5c9
        }
Packit 90a5c9
        else {
Packit 90a5c9
            fprintf(stderr, "%s:%s:%s\n", h->username, h->ctx.passwd,
Packit 90a5c9
                    h->comment);
Packit 90a5c9
        }
Packit 90a5c9
    }
Packit 90a5c9
    htdbm_terminate(h);
Packit 90a5c9
Packit 90a5c9
    return 0; /* Suppress compiler warning. */
Packit 90a5c9
}