--- /dev/null
+/*
+ * kadmin/v5server/admin.c
+ *
+ * Copyright 1995 by the Massachusetts Institute of Technology.
+ * All Rights Reserved.
+ *
+ * Export of this software from the United States of America may
+ * require a specific license from the United States Government.
+ * It is the responsibility of any person or organization contemplating
+ * export to obtain such a license before exporting.
+ *
+ * WITHIN THAT CONSTRAINT, permission to use, copy, modify, and
+ * distribute this software and its documentation for any purpose and
+ * without fee is hereby granted, provided that the above copyright
+ * notice appear in all copies and that both that copyright notice and
+ * this permission notice appear in supporting documentation, and that
+ * the name of M.I.T. not be used in advertising or publicity pertaining
+ * to distribution of the software without specific, written prior
+ * permission. M.I.T. makes no representations about the suitability of
+ * this software for any purpose. It is provided "as is" without express
+ * or implied warranty.
+ *
+ */
+
+/*
+ * admin.c - Handle administrative requests.
+ */
+
+#include "k5-int.h"
+#include "kadm5_defs.h"
+#include "adm_proto.h"
+#include "adm.h"
+\f
+/*
+ * Data structure used to pass information in and out of krb5_db_iterate.
+ */
+struct inq_context {
+ krb5_context ic_context; /* in */
+ int ic_level; /* in */
+ krb5_principal ic_who; /* in */
+ krb5_boolean ic_entry_found; /* out */
+ krb5_int32 *ic_ncomps; /* out */
+ krb5_data **ic_clist; /* out */
+ char *ic_next; /* out */
+};
+
+static krb5_db_entry admin_def_dbent;
+static krb5_boolean admin_def_dbent_inited = 0;
+
+static const char *admin_perm_denied_fmt = "ACL entry prevents %s operation by %s";
+static const char *admin_db_write_err_fmt = "database write failed during %s operation by %s";
+static const char *admin_db_success_fmt = "%s operation for %s successfully issued by %s";
+static const char *admin_db_read_err_fmt = "database read failed during %s operation by %s";
+static const char *admin_no_cl_ident_fmt = "cannot get client identity from ticket for %s operation";
+static const char *admin_db_rename_fmt = "%s operation from %s to %s successfully issued by %s";
+static const char *admin_db_del_err_fmt = "database delete entry(%s) failed during %s operation by %s";
+static const char *admin_key_dec_err_fmt = "key decode failed for %s's key during %s operation by %s";
+
+static const char *admin_add_principal_text = "Add Principal";
+static const char *admin_modify_principal_text = "Modify Principal";
+static const char *admin_delete_principal_text = "Delete Principal";
+static const char *admin_rename_principal_text = "Rename Principal";
+static const char *admin_change_pwd_text = "Change Password";
+static const char *admin_change_rpwd_text = "Change Random Password";
+static const char *admin_inquire_text = "Inquire";
+static const char *admin_extract_key_text = "Extract Key";
+
+extern char *programname;
+\f
+/*
+ * admin_init_def_dbent() - Initialize the default database entry.
+ */
+static void
+admin_init_def_dbent()
+{
+ /* Zero it all out, and fill in non-zero defaults */
+ memset((char *) &admin_def_dbent, 0, sizeof(admin_def_dbent));
+ admin_def_dbent.kvno = 1;
+ admin_def_dbent.max_life = KRB5_KDB_MAX_LIFE;
+ admin_def_dbent.max_renewable_life = KRB5_KDB_MAX_RLIFE;
+ admin_def_dbent.expiration = KRB5_KDB_EXPIRATION;
+ admin_def_dbent.attributes = KRB5_KDB_DEF_FLAGS;
+ admin_def_dbent_inited = 1;
+}
+\f
+/*
+ * admin_client_identity() - Determine client identity from ticket
+ */
+static krb5_error_code
+admin_client_identity(kcontext, debug_level, ticket, clientp, clnamep)
+ krb5_context kcontext;
+ int debug_level;
+ krb5_ticket *ticket;
+ krb5_principal *clientp;
+ char **clnamep;
+{
+ krb5_error_code kret = 0;
+
+ DPRINT(DEBUG_CALLS, debug_level, ("* admin_client_identity()\n"));
+ *clientp = (krb5_principal) NULL;
+ *clnamep = (char *) NULL;
+
+ /* Copy principal out of ticket */
+ if (!(kret = krb5_copy_principal(kcontext,
+ ticket->enc_part2->client,
+ clientp))) {
+ /* Flatten name */
+ if (kret = krb5_unparse_name(kcontext, *clientp, clnamep)) {
+ krb5_free_principal(kcontext, *clientp);
+ *clientp = (krb5_principal) NULL;
+ }
+ }
+
+ DPRINT(DEBUG_CALLS, debug_level, ("X admin_client_identity()=%d\n", kret));
+ return(kret);
+}
+\f
+/*
+ * admin_merge_dbentries() - Merge two database entries and a password.
+ */
+static krb5_error_code
+admin_merge_dbentries(kcontext, debug_level, who, defaultp,
+ valid, dbentp, password, is_pwchange)
+ krb5_context kcontext;
+ int debug_level;
+ krb5_principal who;
+ krb5_db_entry *defaultp;
+ krb5_ui_4 valid;
+ krb5_db_entry *dbentp;
+ char *password;
+ krb5_boolean is_pwchange;
+{
+ krb5_error_code kret = 0;
+ krb5_timestamp now;
+ DPRINT(DEBUG_CALLS, debug_level, ("* admin_merge_dbentries()\n"));
+
+ /*
+ * Ensure that there's a principal,
+ * we have the current t-o-d,
+ * and that we don't have a password and the random-password option.
+ */
+ kret = EINVAL;
+ if (dbentp->principal &&
+ !(kret = krb5_timeofday(kcontext, &now)) &&
+ (!password || ((valid & KRB5_ADM_M_RANDOMKEY) == 0))) {
+
+ /*
+ * Now fill in unsupplied values from defaults.
+ */
+ if ((valid & KRB5_ADM_M_KVNO) == 0)
+ dbentp->kvno = defaultp->kvno;
+ if ((valid & KRB5_ADM_M_MAXLIFE) == 0)
+ dbentp->max_life = defaultp->max_life;
+ if ((valid & KRB5_ADM_M_MAXRENEWLIFE) == 0)
+ dbentp->max_renewable_life = defaultp->max_renewable_life;
+ if ((valid & KRB5_ADM_M_EXPIRATION) == 0)
+ dbentp->expiration = defaultp->expiration;
+ if ((valid & KRB5_ADM_M_PWEXPIRATION) == 0)
+ dbentp->pw_expiration = defaultp->pw_expiration;
+ if ((valid & KRB5_ADM_M_FLAGS) == 0)
+ dbentp->attributes = defaultp->attributes;
+ if ((valid & KRB5_ADM_M_SALTTYPE) == 0) {
+ dbentp->salt_type = defaultp->salt_type;
+ dbentp->alt_salt_type = defaultp->alt_salt_type;
+ }
+
+ /*
+ * Now fill in un-suppliable values from our data.
+ */
+ if (password || is_pwchange)
+ dbentp->last_pwd_change = now; /* Changing password now */
+ else
+ dbentp->last_pwd_change = defaultp->last_pwd_change;
+ dbentp->last_success = defaultp->last_success;
+ dbentp->last_failed = defaultp->last_failed;
+ dbentp->fail_auth_count = defaultp->fail_auth_count;
+ dbentp->mod_name = who; /* Modified by caller */
+ dbentp->mod_date = now; /* Modified now */
+ dbentp->mkvno = key_master_entry()->kvno;
+
+ /* See if this is a random key or not */
+ if (password) {
+ krb5_data pwdata;
+ krb5_keyblock pkey, akey;
+ krb5_data psalt, asalt;
+
+ DPRINT(DEBUG_OPERATION, debug_level, ("> string-to-key\n"));
+ /*
+ * Now handle string-to-key and salt.
+ */
+ pwdata.data = password;
+ pwdata.length = strlen(password);
+ memset((char *) &pkey, 0, sizeof(pkey));
+ memset((char *) &akey, 0, sizeof(akey));
+ memset((char *) &psalt, 0, sizeof(psalt));
+ memset((char *) &asalt, 0, sizeof(asalt));
+
+ /* Convert password string to key */
+ if (!(kret = key_string_to_keys(kcontext,
+ dbentp->principal,
+ &pwdata,
+ dbentp->salt_type,
+ dbentp->alt_salt_type,
+ &pkey,
+ &akey,
+ &psalt,
+ &asalt))) {
+ /* Encrypt the keys */
+ DPRINT(DEBUG_OPERATION, debug_level, ("> encode\n"));
+ if (!(kret = key_encrypt_keys(kcontext,
+ dbentp->principal,
+ &pkey,
+ &akey,
+ &dbentp->key,
+ &dbentp->alt_key))) {
+ /* Squirrel away the salt. */
+ dbentp->salt_length = psalt.length;
+ dbentp->salt = (krb5_octet *) psalt.data;
+ dbentp->alt_salt_length = asalt.length;
+ dbentp->alt_salt = (krb5_octet *) asalt.data;
+ kret = 0;
+ }
+
+ /* Clean up the cleartext keys */
+ if (pkey.contents) {
+ memset((char *) pkey.contents, 0, (size_t) pkey.length);
+ krb5_xfree(pkey.contents);
+ }
+ if (akey.contents) {
+ memset((char *) akey.contents, 0, (size_t) akey.length);
+ krb5_xfree(akey.contents);
+ }
+ if (kret) {
+ /* If failed, cleanup the salt data too. */
+ if (psalt.data) {
+ memset((char *) psalt.data, 0, (size_t) psalt.length);
+ krb5_xfree(psalt.data);
+ }
+ if (asalt.data) {
+ memset((char *) asalt.data, 0, (size_t) asalt.length);
+ krb5_xfree(asalt.data);
+ }
+ }
+ }
+ }
+ else {
+ krb5_keyblock rkey, akey;
+
+ /* Random key */
+ DPRINT(DEBUG_OPERATION, debug_level, ("> random key\n"));
+ memset((char *) &rkey, 0, sizeof(rkey));
+ memset((char *) &akey, 0, sizeof(akey));
+ if ((dbentp->salt_type == KRB5_KDB_SALTTYPE_NORMAL) &&
+ (!(kret = key_random_key(kcontext, &rkey)))) {
+
+ /* Encrypt the keys */
+ DPRINT(DEBUG_OPERATION, debug_level, ("> encode\n"));
+ if (!(kret = key_encrypt_keys(kcontext,
+ dbentp->principal,
+ &rkey,
+ &akey,
+ &dbentp->key,
+ &dbentp->alt_key))) {
+ /* Squirrel away the salt. */
+ dbentp->salt_length = 0;
+ dbentp->salt = (krb5_octet *) NULL;
+ dbentp->alt_salt_length = 0;
+ dbentp->alt_salt = (krb5_octet *) NULL;
+ kret = 0;
+ }
+ /* Clean up the cleartext key */
+ if (rkey.contents) {
+ memset((char *) rkey.contents, 0, (size_t) rkey.length);
+ krb5_xfree(rkey.contents);
+ }
+ }
+ }
+ }
+ DPRINT(DEBUG_CALLS, debug_level, ("X admin_merge_dbentries()=%d\n", kret));
+ return(kret);
+}
+\f
+/*
+ * admin_add_modify() - Adds or modifies a database entry.
+ *
+ * Does all the work for add_principal, modify_principal, change_opwd,
+ * and change_orandpwd.
+ *
+ * Supplied argument list must contain at least one entry, the name of the
+ * target principal.
+ */
+static krb5_int32
+admin_add_modify(kcontext, debug_level, ticket, nargs, arglist,
+ should_exist, pwd_supplied, supp_pwd)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_int32 nargs; /* # rem. arguments */ /* In */
+ krb5_data *arglist; /* Remaining arguments */ /* In */
+ krb5_boolean should_exist; /* Should it exist? */ /* In */
+ krb5_boolean pwd_supplied; /* Password supplied? */ /* In */
+ char *supp_pwd; /* Supplied password */ /* In */
+{
+ krb5_int32 retval = KRB5_ADM_SUCCESS;
+ krb5_error_code kret = 0;
+ krb5_principal client;
+ char *client_name;
+ krb5_ui_4 valid_mask;
+ krb5_db_entry new_dbentry;
+ krb5_db_entry cur_dbentry;
+ krb5_principal principal;
+ char *new_password;
+ krb5_int32 operation;
+ const char *op_msg;
+#ifdef DEBUG
+ char *dbg_op_msg;
+#endif /* DEBUG */
+
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_add_modify(%s)\n", arglist[0].data));
+ /* Initialize */
+ client = (krb5_principal) NULL;
+ client_name = (char *) NULL;
+ valid_mask = 0;
+ memset((char *) &new_dbentry, 0, sizeof(new_dbentry));
+ memset((char *) &cur_dbentry, 0, sizeof(cur_dbentry));
+ principal = (krb5_principal) NULL;
+ new_password = (char *) NULL;
+
+ /* Determine what our operation is based on should_exist/pwd_supplied */
+ if (should_exist) {
+ if (pwd_supplied) {
+ operation = ACL_CHANGEPW;
+ op_msg = (supp_pwd) ? admin_change_pwd_text :
+ admin_change_rpwd_text;
+#ifdef DEBUG
+ dbg_op_msg = (supp_pwd) ? "CHANGE PASSWORD" :
+ "CHANGE RANDOM PASSWORD";
+#endif /* DEBUG */
+ }
+ else {
+ operation = ACL_MODIFY_PRINCIPAL;
+ op_msg = admin_modify_principal_text;
+#ifdef DEBUG
+ dbg_op_msg = "MODIFY PRINCIPAL";
+#endif /* DEBUG */
+ }
+ }
+ else {
+ if (pwd_supplied) {
+ return(KRB5_ADM_SYSTEM_ERROR);
+ }
+ else {
+ operation = ACL_ADD_PRINCIPAL;
+ op_msg = admin_add_principal_text;
+#ifdef DEBUG
+ dbg_op_msg = "ADD PRINCIPAL";
+#endif /* DEBUG */
+ }
+ }
+
+ /* Get the identity of our client */
+ if (!(kret = admin_client_identity(kcontext,
+ debug_level,
+ ticket,
+ &client,
+ &client_name))) {
+
+ /* See if this client can perform this operation. */
+ if (acl_op_permitted(kcontext, client, operation)) {
+
+ /* Parse the specified principal name */
+ if (!(kret = krb5_parse_name(kcontext,
+ arglist[0].data,
+ &principal))) {
+ int howmany;
+ krb5_boolean more;
+
+ howmany = 1;
+
+ /* Try to get the entry */
+ kret = krb5_db_get_principal(kcontext,
+ principal,
+ &cur_dbentry,
+ &howmany,
+ &more);
+
+ /*
+ * If we're modifying, there'd better be an entry.
+ * If we're adding, there'd better not be an entry.
+ */
+ if (!kret &&
+ ((should_exist && howmany) ||
+ (!should_exist && !howmany))) {
+
+ /* We need to have a principal */
+ new_dbentry.principal = principal;
+
+ /*
+ * Parse the argument list and make sure that only valid
+ * options are set.
+ */
+ if (!(kret = krb5_adm_proto_to_dbent(kcontext,
+ nargs-1,
+ &arglist[1],
+ &valid_mask,
+ &new_dbentry,
+ &new_password)) &&
+ ((valid_mask & ~KRB5_ADM_M_SET_VALID) == 0)) {
+ krb5_data pword_data;
+ krb5_int32 temp;
+
+ pword_data.data = (pwd_supplied) ? supp_pwd :
+ new_password;
+ pword_data.length = (pword_data.data) ?
+ strlen(pword_data.data) : 0;
+
+ /*
+ * Check viability of options specified. One
+ * of the following must be true:
+ * 1) randomkey was specified and no password.
+ * 2) randomkey is not specified and there
+ * is a password to change/set and it is
+ * is suitable.
+ * 3) randomkey is not specified and there is
+ * no password to change and this is
+ * is a modify entry request.
+ *
+ * Check the suitability of the new password, if
+ * one was supplied.
+ */
+ if (((valid_mask & KRB5_ADM_M_RANDOMKEY) &&
+ !pword_data.data)
+ ||
+ (!(valid_mask & KRB5_ADM_M_RANDOMKEY) &&
+ ((!pword_data.data && should_exist)||
+ passwd_check_npass_ok(kcontext,
+ debug_level,
+ new_dbentry.principal,
+ &new_dbentry,
+ &pword_data,
+ &temp)))) {
+ krb5_db_entry *merge;
+
+ /* Check if the default is initialized */
+ if (!admin_def_dbent_inited)
+ admin_init_def_dbent();
+
+ merge = (should_exist) ?
+ &cur_dbentry : &admin_def_dbent;
+
+ /* Merge the specified entries with the defaults */
+ if (!(kret = admin_merge_dbentries(kcontext,
+ debug_level,
+ client,
+ merge,
+ valid_mask,
+ &new_dbentry,
+ pword_data.data,
+ pwd_supplied)
+ )) {
+ int nument = 1;
+
+ /*
+ * Update the key version number if we're
+ * changing it.
+ */
+ if (should_exist && pwd_supplied)
+ new_dbentry.kvno++;
+
+ /* Write the entry. */
+ kret = krb5_db_put_principal(kcontext,
+ &new_dbentry,
+ &nument);
+ if (kret || (nument != 1)) {
+ /* Ultimate failure */
+ com_err(programname, kret,
+ admin_db_write_err_fmt,
+ op_msg,
+ client_name);
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> db write failed for %s\n",
+ dbg_op_msg));
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ else {
+ /* Ultimate success */
+ com_err(programname, 0,
+ admin_db_success_fmt,
+ op_msg,
+ arglist[0].data,
+ client_name);
+ }
+
+ /*
+ * Clean up droppings from
+ * admin_merge_dbentries
+ */
+ if (new_dbentry.alt_salt) {
+ memset((char *) new_dbentry.alt_salt,
+ 0,
+ (size_t) new_dbentry.alt_salt_length
+ );
+ krb5_xfree(new_dbentry.alt_salt);
+ }
+ if (new_dbentry.salt) {
+ memset((char *) new_dbentry.salt,
+ 0,
+ (size_t) new_dbentry.salt_length);
+ krb5_xfree(new_dbentry.salt);
+ }
+ if (new_dbentry.key.contents)
+ krb5_xfree(new_dbentry.key.contents);
+ if (new_dbentry.alt_key.contents)
+ krb5_xfree(new_dbentry.alt_key.contents);
+ }
+ else {
+ /* Option merge failed */
+ DPRINT(DEBUG_PROTO, debug_level,
+ ("= option merge failed for %s\n",
+ dbg_op_msg));
+ retval = KRB5_ADM_BAD_OPTION;
+ }
+ }
+ else {
+ /* Password was not suitable or conflicts */
+ if (valid_mask & KRB5_ADM_M_RANDOMKEY) {
+ DPRINT(DEBUG_PROTO, debug_level,
+ ("= conflicting options for %s\n",
+ dbg_op_msg));
+ retval = KRB5_ADM_BAD_OPTION;
+ }
+ else {
+ DPRINT(DEBUG_PROTO, debug_level,
+ ("= bad password for %s\n",
+ dbg_op_msg));
+ retval = KRB5_ADM_PW_UNACCEPT;
+ }
+ }
+
+ /* Clean up droppings from krb5_adm_proto_to_dbent */
+ if (new_password)
+ krb5_xfree(new_password);
+ if (new_dbentry.mod_name)
+ krb5_free_principal(kcontext,
+ new_dbentry.mod_name);
+ }
+ else {
+ /* Argument list parse failed or bad options */
+ DPRINT(DEBUG_PROTO, debug_level,
+ ("= argument list bad for %s\n", dbg_op_msg));
+ retval = KRB5_ADM_BAD_OPTION;
+ }
+ krb5_db_free_principal(kcontext,
+ ((should_exist) ?
+ &new_dbentry : &cur_dbentry),
+ 1);
+ }
+ else {
+ /* Database entry failed or yielded unexpected results */
+ if (kret) {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> database read error\n"));
+ com_err(programname, kret,
+ admin_db_read_err_fmt,
+ op_msg,
+ client_name);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ else {
+ if (should_exist) {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> principal %s not in database\n",
+ arglist[0].data));
+ retval = KRB5_ADM_P_DOES_NOT_EXIST;
+ }
+ else {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> principal %s already in database\n",
+ arglist[0].data));
+ retval = KRB5_ADM_P_ALREADY_EXISTS;
+ }
+ }
+ }
+
+ /* Clean up from krb5_parse_name */
+ krb5_free_principal(kcontext, principal);
+ }
+ else {
+ /* Principal name parse failed */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> bad principal string \"%s\"\n", arglist[0].data));
+ retval = (should_exist) ? KRB5_ADM_P_DOES_NOT_EXIST :
+ KRB5_ADM_BAD_OPTION;
+ }
+ }
+ else {
+ /* ACL check failed */
+ com_err(programname, 0, admin_perm_denied_fmt,
+ op_msg, client_name);
+ retval = KRB5_ADM_NOT_AUTHORIZED;
+ }
+
+ /* Clean up admin_client_identity droppings */
+ krb5_xfree(client_name);
+ krb5_free_principal(kcontext, client);
+ }
+ else {
+ /* We really choked here. */
+ com_err(programname, kret, admin_no_cl_ident_fmt, op_msg);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_add_modify() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_delete_rename() - Delete or rename a named principal.
+ */
+static krb5_int32
+admin_delete_rename(kcontext, debug_level, ticket, original, new)
+ krb5_context kcontext;
+ int debug_level;
+ krb5_ticket *ticket;
+ char *original;
+ char *new;
+{
+ krb5_int32 retval = 0;
+ krb5_error_code kret;
+ krb5_principal client;
+ char *client_name;
+ krb5_db_entry orig_entry;
+ krb5_principal orig_principal;
+ krb5_int32 operation;
+ const char * op_msg;
+
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_delete_rename(%s,%s)\n",
+ original,((new) ? new : "(null)")));
+
+ /* Initialize */
+ client = (krb5_principal) NULL;
+ client_name = (char *) NULL;
+ memset((char *) &orig_entry, 0, sizeof(orig_entry));
+ orig_principal = (krb5_principal) NULL;
+ operation = (new) ? ACL_RENAME_PRINCIPAL : ACL_DELETE_PRINCIPAL;
+ op_msg = (new) ? admin_rename_principal_text : admin_delete_principal_text;
+
+ /* Get the identity of our client */
+ if (!(kret = admin_client_identity(kcontext,
+ debug_level,
+ ticket,
+ &client,
+ &client_name))) {
+
+ /* See if this client can perform this operation. */
+ if (acl_op_permitted(kcontext, client, operation)) {
+
+ /* Parse the specified principal name */
+ if (!(kret = krb5_parse_name(kcontext,
+ original,
+ &orig_principal))) {
+ int howmany;
+ krb5_boolean more;
+
+ howmany = 1;
+
+ /* Try to get the entry */
+ if (!(kret = krb5_db_get_principal(kcontext,
+ orig_principal,
+ &orig_entry,
+ &howmany,
+ &more))
+ && howmany) {
+
+ /*
+ * We've got the entry. If we're renaming, we have
+ * to make sure that the new entry's not there. Then
+ * we can put the new entry. If we're deleting or
+ * renaming, we delete the entry last.
+ */
+ if (new) {
+ krb5_principal new_principal;
+
+ if (!(kret = krb5_parse_name(kcontext,
+ new,
+ &new_principal))) {
+ int n_howmany;
+ krb5_boolean n_more;
+ krb5_db_entry xxx_dbentry;
+ krb5_timestamp now;
+
+ n_howmany = 1;
+
+ /* Try to get the entry */
+ if (!(kret = krb5_db_get_principal(kcontext,
+ new_principal,
+ &xxx_dbentry,
+ &n_howmany,
+ &n_more))
+ && !(kret = krb5_timeofday(kcontext, &now))
+ && !n_howmany) {
+ /* Change our name */
+ krb5_free_principal(kcontext,
+ orig_entry.principal);
+ orig_entry.principal = new_principal;
+
+ /* Update our stats */
+ orig_entry.mod_name = client;
+ orig_entry.mod_date = now;
+
+ n_howmany = 1;
+ if ((kret = krb5_db_put_principal(kcontext,
+ &orig_entry,
+ &n_howmany))
+ || (n_howmany != 1)) {
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ else {
+ com_err(programname, 0,
+ admin_db_rename_fmt,
+ op_msg, original, new,
+ client_name);
+ }
+ orig_entry.principal = (krb5_principal) NULL;
+ }
+ else {
+ if (kret) {
+ com_err(programname, kret,
+ admin_db_read_err_fmt,
+ op_msg,
+ client_name);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ else {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> principal \"%s\" already exists\n",
+ new));
+ retval = KRB5_ADM_P_ALREADY_EXISTS;
+ }
+ if (!kret)
+ krb5_db_free_principal(kcontext,
+ &xxx_dbentry,
+ n_howmany);
+ }
+ /* Clean up from krb5_parse_name */
+ krb5_free_principal(kcontext, new_principal);
+ }
+ else {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> bad principal string \"%s\"\n",
+ new));
+ retval = KRB5_ADM_BAD_OPTION;
+ }
+ }
+ /*
+ * If we've fallen through, or if the new addition was
+ * successful, delete the original entry.
+ */
+ if (!kret && !retval) {
+ int num2do = 1;
+ /* Delete operation */
+ kret = krb5_db_delete_principal(kcontext,
+ orig_principal,
+ &num2do);
+ if ((kret != 0) || (num2do != 1)) {
+ if (kret) {
+ com_err(programname, kret,
+ admin_db_del_err_fmt,
+ original, op_msg, client_name);
+ }
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ else {
+ if (!new)
+ com_err(programname, 0,
+ admin_db_success_fmt,
+ op_msg, original, client_name);
+ }
+ }
+ krb5_db_free_principal(kcontext, &orig_entry, 1);
+ }
+ else {
+ /* Database lookup failed or returned unexpected result */
+ if (kret) {
+ com_err(programname, kret,
+ admin_db_read_err_fmt, op_msg, client_name);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ else {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> principal %s not in database\n",
+ original));
+ retval = KRB5_ADM_P_DOES_NOT_EXIST;
+ }
+ }
+
+ /* Clean up from krb5_parse_name */
+ krb5_free_principal(kcontext, orig_principal);
+ }
+ else {
+ /* Principal name parse failed */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> bad principal string \"%s\"\n", original));
+ retval = KRB5_ADM_P_DOES_NOT_EXIST;
+ }
+ }
+ else {
+ /* ACL check failed */
+ com_err(programname, 0, admin_perm_denied_fmt,
+ op_msg, client_name);
+ retval = KRB5_ADM_NOT_AUTHORIZED;
+ }
+
+ /* Clean up admin_client_identity droppings */
+ krb5_xfree(client_name);
+ krb5_free_principal(kcontext, client);
+ }
+ else {
+ /* We really choked here. */
+ com_err(programname, kret, admin_no_cl_ident_fmt, op_msg);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_delete_rename() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_inq_iterator() - Routine called by krb5_db_iterate to scan through the
+ * database for a particular entry and its next entry.
+ */
+static krb5_error_code
+admin_inq_iterator(pointer, dbentp)
+ krb5_pointer pointer;
+ krb5_db_entry *dbentp;
+{
+ krb5_error_code kret;
+ struct inq_context *iargp;
+
+ kret = 0;
+ iargp = (struct inq_context *) pointer;
+ /*
+ * See if we have found a target of our search.
+ */
+ if (!iargp->ic_entry_found) {
+ /*
+ * No, now we are looking for a particular entry.
+ */
+ if (!iargp->ic_who ||
+ krb5_principal_compare(iargp->ic_context,
+ iargp->ic_who,
+ dbentp->principal)) {
+ if (iargp->ic_who) {
+ DPRINT(DEBUG_OPERATION, iargp->ic_level,
+ ("> found entry\n"));
+ kret = krb5_adm_dbent_to_proto(iargp->ic_context,
+ KRB5_ADM_M_GET_VALID,
+ dbentp,
+ (char *) NULL,
+ iargp->ic_ncomps,
+ iargp->ic_clist);
+ if (!kret) {
+ iargp->ic_entry_found = 1;
+ DPRINT(DEBUG_OPERATION, iargp->ic_level,
+ ("> converted entry to protocol\n"));
+ }
+ else {
+ DPRINT(DEBUG_OPERATION, iargp->ic_level,
+ ("> convert entry to protocol failed\n"));
+ }
+ }
+ else {
+ iargp->ic_entry_found = 1;
+ *(iargp->ic_ncomps) = 0;
+ *(iargp->ic_clist) = (krb5_data *) NULL;
+ if (!iargp->ic_next) {
+ kret = krb5_unparse_name(iargp->ic_context,
+ dbentp->principal,
+ &iargp->ic_next);
+ if (kret) {
+ DPRINT(DEBUG_OPERATION, iargp->ic_level,
+ ("> unparse next entry failed\n"));
+ }
+ else {
+ DPRINT(DEBUG_OPERATION, iargp->ic_level,
+ ("> next entry is %s\n", iargp->ic_next));
+ }
+ }
+ }
+ }
+ }
+ else {
+ if (!iargp->ic_next) {
+ kret = krb5_unparse_name(iargp->ic_context,
+ dbentp->principal,
+ &iargp->ic_next);
+ if (kret) {
+ DPRINT(DEBUG_OPERATION, iargp->ic_level,
+ ("> unparse next entry failed\n"));
+ }
+ else {
+ DPRINT(DEBUG_OPERATION, iargp->ic_level,
+ ("> next entry is %s\n", iargp->ic_next));
+ }
+ }
+ }
+ return(kret);
+}
+\f
+/*
+ * admin_add_principal() - Add a principal with the specified attributes
+ */
+krb5_int32
+admin_add_principal(kcontext, debug_level, ticket, nargs, arglist)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_int32 nargs; /* # rem. arguments */ /* In */
+ krb5_data *arglist; /* Remaining arguments */ /* In */
+{
+ krb5_int32 retval;
+
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_add_principal(%s)\n", arglist[0].data));
+ retval = admin_add_modify(kcontext,
+ debug_level,
+ ticket,
+ nargs,
+ arglist,
+ 0,
+ 0,
+ (char *) NULL);
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_add_principal() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_delete_principal() - Delete the specified principal.
+ */
+krb5_int32
+admin_delete_principal(kcontext, debug_level, ticket, principal)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_data *principal; /* Principal to delete */ /* In */
+{
+ krb5_int32 retval;
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_delete_principal(%s)\n", principal->data));
+ retval = admin_delete_rename(kcontext,
+ debug_level,
+ ticket,
+ principal->data,
+ (char *) NULL);
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_delete_principal() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_rename_principal() - Rename the original principal to the
+ * specified principal.
+ */
+krb5_int32
+admin_rename_principal(kcontext, debug_level, ticket, original, new)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_data *original; /* Original principal */ /* In */
+ krb5_data *new; /* New Principal */ /* In */
+{
+ krb5_int32 retval;
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_rename_principal(%s)\n", original->data));
+ retval = admin_delete_rename(kcontext,
+ debug_level,
+ ticket,
+ original->data,
+ new->data);
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_rename_principal() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_modify_principal() - Modify the specified principal with the
+ * specifed attributes using the existing
+ * entry as a template.
+ */
+krb5_int32
+admin_modify_principal(kcontext, debug_level, ticket, nargs, arglist)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_int32 nargs; /* # rem. arguments */ /* In */
+ krb5_data *arglist; /* Remaining arguments */ /* In */
+{
+ krb5_int32 retval;
+
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_modify_principal(%s)\n", arglist[0].data));
+ retval = admin_add_modify(kcontext,
+ debug_level,
+ ticket,
+ nargs,
+ arglist,
+ 1,
+ 0,
+ (char *) NULL);
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_modify_principal() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_change_opwd() - Change the password of a principal.
+ */
+krb5_int32
+admin_change_opwd(kcontext, debug_level, ticket, principal, password)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_data *principal; /* Principal */ /* In */
+ krb5_data *password; /* New Password */ /* In */
+{
+ krb5_int32 retval;
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_change_opw(%s)\n", principal->data));
+ retval = admin_add_modify(kcontext,
+ debug_level,
+ ticket,
+ 1,
+ principal,
+ 1,
+ 1,
+ password->data);
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_change_opw() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_change_orandpwd() - Change the random key of a principal.
+ */
+krb5_int32
+admin_change_orandpwd(kcontext, debug_level, ticket, principal)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_data *principal; /* Principal */ /* In */
+{
+ krb5_int32 retval = 0;
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_change_orandpw(%s)\n", principal->data));
+ retval = admin_add_modify(kcontext,
+ debug_level,
+ ticket,
+ 1,
+ principal,
+ 1,
+ 1,
+ (char *) NULL);
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_change_orandpw() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_inquire() - Retrieve the attributes of a principal.
+ */
+krb5_int32
+admin_inquire(kcontext, debug_level, ticket, principal, ncompp, complistp)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_data *principal; /* Principal */ /* In */
+ krb5_int32 *ncompp; /* # reply components */ /* Out */
+ krb5_data **complistp; /* Reply component list */ /* Out */
+{
+ krb5_int32 retval = KRB5_ADM_SUCCESS;
+ krb5_error_code kret = 0;
+ krb5_principal client;
+ krb5_principal target;
+ char *client_name;
+
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_inquire(%s)\n", principal->data));
+ /* Initialize */
+ client = target = (krb5_principal) NULL;
+ client_name = (char *) NULL;
+
+ /* Get the identity of our client */
+ if (!(kret = admin_client_identity(kcontext,
+ debug_level,
+ ticket,
+ &client,
+ &client_name))) {
+
+ /* See if this client can perform this operation. */
+ if (acl_op_permitted(kcontext, client, ACL_INQUIRE)) {
+
+ /* Parse the specified principal name */
+ if (!principal->length ||
+ !(kret = krb5_parse_name(kcontext,
+ principal->data,
+ &target))) {
+ struct inq_context iargs;
+ krb5_data *tmplistp;
+ krb5_int32 tmpncomp;
+
+ /*
+ * Now - if we had a "find/findnext" set of database operators
+ * then we could do this easily, but here we're going to have
+ * to iterate through the whole database and find our entry.
+ * If we find it, continue to the next entry so that we can
+ * return it.
+ */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> Find entry %s\n",
+ ((principal->length) ? principal->data : "(null)")));
+ iargs.ic_context = kcontext;
+ iargs.ic_level = debug_level;
+ iargs.ic_who = (principal->length) ? target :
+ (krb5_principal) NULL;
+ iargs.ic_entry_found = 0;
+ iargs.ic_ncomps = &tmpncomp;
+ iargs.ic_clist = &tmplistp;
+ iargs.ic_next = (char *) NULL;
+ if (!(kret = krb5_db_iterate(kcontext,
+ admin_inq_iterator,
+ &iargs))) {
+
+ /* After scanning, did we find it? */
+ if (iargs.ic_entry_found) {
+
+ /*
+ * How inconvenient. We have the list already
+ * packaged up, but we have to insert our next
+ * element first.
+ */
+ if (*complistp = (krb5_data *)
+ malloc((size_t) (tmpncomp+1) * sizeof(krb5_data))
+ ) {
+ memset((char *) &(*complistp)[0], 0,
+ sizeof(krb5_data));
+ if (iargs.ic_next) {
+ (*complistp)[0].data = iargs.ic_next;
+ (*complistp)[0].length =
+ strlen((*complistp)[0].data);
+ }
+ memcpy(&(*complistp)[1], &tmplistp[0],
+ (size_t) tmpncomp * sizeof(krb5_data));
+ *ncompp = tmpncomp+1;
+ krb5_xfree(tmplistp);
+ }
+ else {
+ /* Could not get memory for new component list */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> could not get memory\n"));
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ }
+ else {
+ /* Could not find principal */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> cannot find principal \"%s\"\n",
+ principal->data));
+ retval = KRB5_ADM_P_DOES_NOT_EXIST;
+ }
+ }
+ else {
+ /* Could not iterate */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> could not iterate database\n"));
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ /* Cleanup */
+ krb5_free_principal(kcontext, target);
+ }
+ else {
+ /* Could not parse principal name */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> bad principal string \"%s\"\n", principal->data));
+ retval = KRB5_ADM_P_DOES_NOT_EXIST;
+ }
+ }
+ else {
+ /* Not authorized to perform this function */
+ com_err(programname, 0, admin_perm_denied_fmt,
+ admin_inquire_text, client_name);
+ retval = KRB5_ADM_NOT_AUTHORIZED;
+ }
+
+ /* Clean up */
+ krb5_xfree(client_name);
+ krb5_free_principal(kcontext, client);
+ }
+ else {
+ /* Something is really wrong here. */
+ com_err(programname, kret, admin_no_cl_ident_fmt, admin_inquire_text);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_inquire() = %d\n", retval));
+ return(retval);
+}
+\f
+/*
+ * admin_extract_key() - Extract the service key entry for this name/instance
+ */
+krb5_int32
+admin_extract_key(kcontext, debug_level, ticket,
+ instance, name, ncompp, complistp)
+ krb5_context kcontext; /* Kerberos context */ /* In */
+ int debug_level; /* Debug level */ /* In */
+ krb5_ticket *ticket; /* Kerberos ticket */ /* In */
+ krb5_data *instance; /* Instance of principal*/ /* In */
+ krb5_data *name; /* Name of principal */ /* In */
+ krb5_int32 *ncompp; /* # reply components */ /* Out */
+ krb5_data **complistp; /* Reply component list */ /* Out */
+{
+ krb5_int32 retval = KRB5_ADM_SUCCESS;
+ krb5_error_code kret = 0;
+ krb5_principal client;
+ char *client_name;
+ char *realm;
+ char *princname;
+ krb5_principal principal;
+ krb5_db_entry dbentry;
+ int nentries;
+ krb5_boolean more;
+ krb5_keytab_entry ktabentry;
+ krb5_keyblock xxxkey;
+
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("* admin_extract_key(%s/%s)\n", name->data, instance->data));
+
+ /* Get the identity of our client */
+ if (!(kret = admin_client_identity(kcontext,
+ debug_level,
+ ticket,
+ &client,
+ &client_name))) {
+
+ /* See if this client can perform this operation. */
+ if (acl_op_permitted(kcontext, client, ACL_EXTRACT)) {
+
+ realm = key_master_realm();
+ if (princname = (char *) malloc((size_t) name->length + 1 +
+ instance->length + 1 +
+ strlen(realm) + 1)) {
+
+ /* Formulate the name of our target */
+ sprintf(princname, "%s/%s@%s", name->data,
+ instance->data, realm);
+
+ /* See if it's a valid name */
+ if (!(kret = krb5_parse_name(kcontext,
+ princname,
+ &principal))) {
+
+ /* Get the database entry */
+ nentries = 1;
+ if (!(kret = krb5_db_get_principal(kcontext,
+ principal,
+ &dbentry,
+ &nentries,
+ &more)) &&
+ (nentries == 1) && (!more)) {
+
+ /* Decrypt the key entries. */
+ memset((char *) &ktabentry, 0, sizeof(ktabentry));
+ memset((char *) &xxxkey, 0, sizeof(xxxkey));
+ if (!(kret = key_decrypt_keys(kcontext,
+ principal,
+ &dbentry.key,
+ &dbentry.alt_key,
+ &ktabentry.key,
+ &xxxkey))) {
+ ktabentry.principal = principal;
+ ktabentry.vno = dbentry.kvno;
+
+ /* Pack the response */
+ if (kret = krb5_adm_ktent_to_proto(kcontext,
+ &ktabentry,
+ ncompp,
+ complistp)) {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> cannot package keytab protocol\n"));
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+
+ /* Cleanup from key_decrypt_keys */
+ if (xxxkey.contents) {
+ memset((char *) xxxkey.contents, 0,
+ (size_t) xxxkey.length);
+ krb5_xfree(xxxkey.contents);
+ }
+ if (ktabentry.key.contents) {
+ memset((char *) ktabentry.key.contents, 0,
+ (size_t) ktabentry.key.length);
+ krb5_xfree(ktabentry.key.contents);
+ }
+ }
+ else {
+ /* key_decrypt_keys failed */
+ com_err(programname, kret,
+ admin_key_dec_err_fmt,
+ princname, admin_extract_key_text,
+ client_name);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ krb5_db_free_principal(kcontext, &dbentry, nentries);
+ }
+ else {
+ /*
+ * Database lookup failed or produced unexpected
+ * results.
+ */
+ if (kret) {
+ com_err(programname, kret,
+ admin_db_read_err_fmt,
+ admin_extract_key_text,
+ client_name);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ else {
+ if (nentries == 0) {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> principal \"%s\" does not exist\n",
+ princname));
+ retval = KRB5_ADM_P_DOES_NOT_EXIST;
+ }
+ else {
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> principal \"%s\" not unique\n",
+ princname));
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ krb5_db_free_principal(kcontext,
+ &dbentry,
+ nentries);
+ }
+ }
+ }
+ krb5_free_principal(kcontext, principal);
+ }
+ else {
+ /* Name parse failed */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> bad principal string \"%s\"\n", princname));
+ retval = KRB5_ADM_P_DOES_NOT_EXIST;
+ }
+ free(princname);
+ }
+ else {
+ /* No memory. */
+ DPRINT(DEBUG_OPERATION, debug_level,
+ ("> no memory for principal name\n"));
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ }
+ else {
+ /* Not authorized to perform this operation */
+ com_err(programname, 0, admin_perm_denied_fmt,
+ admin_extract_key_text, client_name);
+ retval = KRB5_ADM_NOT_AUTHORIZED;
+ }
+
+ /* Clean up */
+ krb5_xfree(client_name);
+ krb5_free_principal(kcontext, client);
+ }
+ else {
+ /* Cannot get our identity */
+ com_err(programname, kret, admin_no_cl_ident_fmt,
+ admin_extract_key_text);
+ retval = KRB5_ADM_SYSTEM_ERROR;
+ }
+ DPRINT(DEBUG_CALLS, debug_level,
+ ("X admin_extract_key() = %d\n", retval));
+ return(retval);
+}
+