mirror of
https://salsa.debian.org/freeipa-team/freeipa.git
synced 2024-12-28 18:01:23 -06:00
c5e055ae00
Currently only the group SIDs from a PAC are used to find out about the membership in local groups. This patch adds the user SID to the list. Fixes https://fedorahosted.org/freeipa/ticket/3257
1732 lines
49 KiB
C
1732 lines
49 KiB
C
/*
|
|
* MIT Kerberos KDC database backend for FreeIPA
|
|
*
|
|
* Authors: Simo Sorce <ssorce@redhat.com>
|
|
*
|
|
* Copyright (C) 2011 Simo Sorce, Red Hat
|
|
* see file 'COPYING' for use and warranty information
|
|
*
|
|
* This program is free software you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "ipa_kdb.h"
|
|
#include <talloc.h>
|
|
#include <syslog.h>
|
|
#include "util/time.h"
|
|
#include "gen_ndr/ndr_krb5pac.h"
|
|
|
|
struct ipadb_adtrusts {
|
|
char *domain_name;
|
|
char *flat_name;
|
|
char *domain_sid;
|
|
};
|
|
|
|
struct ipadb_mspac {
|
|
char *flat_domain_name;
|
|
char *flat_server_name;
|
|
char *fallback_group;
|
|
uint32_t fallback_rid;
|
|
|
|
int num_trusts;
|
|
struct ipadb_adtrusts *trusts;
|
|
time_t last_update;
|
|
};
|
|
|
|
|
|
int krb5_klog_syslog(int, const char *, ...);
|
|
|
|
static char *user_pac_attrs[] = {
|
|
"objectClass",
|
|
"uid",
|
|
"cn",
|
|
"gidNumber",
|
|
"krbPrincipalName",
|
|
"krbCanonicalName",
|
|
"krbTicketPolicyReference",
|
|
"krbPrincipalExpiration",
|
|
"krbPasswordExpiration",
|
|
"krbPwdPolicyReference",
|
|
"krbPrincipalType",
|
|
"krbLastPwdChange",
|
|
"krbPrincipalAliases",
|
|
"krbLastSuccessfulAuth",
|
|
"krbLastFailedAuth",
|
|
"krbLoginFailedCount",
|
|
"krbLastAdminUnlock",
|
|
"krbTicketFlags",
|
|
"ipaNTSecurityIdentifier",
|
|
"ipaNTLogonScript",
|
|
"ipaNTProfilePath",
|
|
"ipaNTHomeDirectory",
|
|
"ipaNTHomeDirectoryDrive",
|
|
NULL
|
|
};
|
|
|
|
char *deref_search_attrs[] = {
|
|
"memberOf",
|
|
NULL
|
|
};
|
|
|
|
static char *memberof_pac_attrs[] = {
|
|
"gidNumber",
|
|
"ipaNTSecurityIdentifier",
|
|
NULL
|
|
};
|
|
|
|
#define SID_ID_AUTHS 6
|
|
#define SID_SUB_AUTHS 15
|
|
#define MAX(a,b) (((a)>(b))?(a):(b))
|
|
|
|
static int string_to_sid(char *str, struct dom_sid *sid)
|
|
{
|
|
unsigned long val;
|
|
char *s, *t;
|
|
int i;
|
|
|
|
memset(sid, '\0', sizeof(struct dom_sid));
|
|
|
|
s = str;
|
|
|
|
if (strncasecmp(s, "S-", 2) != 0) {
|
|
return EINVAL;
|
|
}
|
|
s += 2;
|
|
|
|
val = strtoul(s, &t, 10);
|
|
if (s == t || !t || *t != '-') {
|
|
return EINVAL;
|
|
}
|
|
s = t + 1;
|
|
sid->sid_rev_num = val;
|
|
|
|
val = strtoul(s, &t, 10);
|
|
if (s == t || !t) {
|
|
return EINVAL;
|
|
}
|
|
sid->id_auth[2] = (val & 0xff000000) >> 24;
|
|
sid->id_auth[3] = (val & 0x00ff0000) >> 16;
|
|
sid->id_auth[4] = (val & 0x0000ff00) >> 8;
|
|
sid->id_auth[5] = (val & 0x000000ff);
|
|
|
|
for (i = 0; i < SID_SUB_AUTHS; i++) {
|
|
switch (*t) {
|
|
case '\0':
|
|
/* no (more) subauths, we are done with it */
|
|
sid->num_auths = i;
|
|
return 0;
|
|
case '-':
|
|
/* there are (more) subauths */
|
|
s = t + 1;;
|
|
break;
|
|
default:
|
|
/* garbage */
|
|
return EINVAL;
|
|
}
|
|
|
|
val = strtoul(s, &t, 10);
|
|
if (s == t || !t) {
|
|
return EINVAL;
|
|
}
|
|
sid->sub_auths[i] = val;
|
|
}
|
|
|
|
if (*t != '\0') {
|
|
return EINVAL;
|
|
}
|
|
|
|
sid->num_auths = i;
|
|
return 0;
|
|
}
|
|
|
|
static char *dom_sid_string(TALLOC_CTX *memctx, const struct dom_sid *dom_sid)
|
|
{
|
|
size_t c;
|
|
size_t len;
|
|
int ofs;
|
|
uint32_t ia;
|
|
char *buf;
|
|
|
|
if (dom_sid == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
len = 25 + dom_sid->num_auths * 11;
|
|
|
|
buf = talloc_zero_size(memctx, len);
|
|
|
|
ia = (dom_sid->id_auth[5]) +
|
|
(dom_sid->id_auth[4] << 8 ) +
|
|
(dom_sid->id_auth[3] << 16) +
|
|
(dom_sid->id_auth[2] << 24);
|
|
|
|
ofs = snprintf(buf, len, "S-%u-%lu", (unsigned int) dom_sid->sid_rev_num,
|
|
(unsigned long) ia);
|
|
|
|
for (c = 0; c < dom_sid->num_auths; c++) {
|
|
ofs += snprintf(buf + ofs, MAX(len - ofs, 0), "-%lu",
|
|
(unsigned long) dom_sid->sub_auths[c]);
|
|
}
|
|
|
|
if (ofs >= len) {
|
|
talloc_free(buf);
|
|
return NULL;
|
|
}
|
|
|
|
return buf;
|
|
}
|
|
|
|
static struct dom_sid *dom_sid_dup(TALLOC_CTX *memctx,
|
|
const struct dom_sid *dom_sid)
|
|
{
|
|
struct dom_sid *new_sid;
|
|
size_t c;
|
|
|
|
if (dom_sid == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
new_sid = talloc(memctx, struct dom_sid);
|
|
if (new_sid == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
new_sid->sid_rev_num = dom_sid->sid_rev_num;
|
|
for (c = 0; c < SID_ID_AUTHS; c++) {
|
|
new_sid->id_auth[c] = dom_sid->id_auth[c];
|
|
}
|
|
new_sid->num_auths = dom_sid->num_auths;
|
|
for (c = 0; c < SID_SUB_AUTHS; c++) {
|
|
new_sid->sub_auths[c] = dom_sid->sub_auths[c];
|
|
}
|
|
|
|
return new_sid;
|
|
}
|
|
|
|
static int sid_append_rid(struct dom_sid *sid, uint32_t rid)
|
|
{
|
|
if (sid->num_auths >= SID_SUB_AUTHS) {
|
|
return EINVAL;
|
|
}
|
|
|
|
sid->sub_auths[sid->num_auths++] = rid;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @brief Takes a user sid and removes the rid.
|
|
* The sid is changed by this function,
|
|
* the removed rid is returned too.
|
|
*
|
|
* @param sid A user/group SID
|
|
* @param rid The actual RID found.
|
|
*
|
|
* @return 0 on success, EINVAL otherwise.
|
|
*/
|
|
static int sid_split_rid(struct dom_sid *sid, uint32_t *rid)
|
|
{
|
|
if (sid->num_auths == 0) {
|
|
return EINVAL;
|
|
}
|
|
|
|
sid->num_auths--;
|
|
if (rid != NULL) {
|
|
*rid = sid->sub_auths[sid->num_auths];
|
|
}
|
|
sid->sub_auths[sid->num_auths] = 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static krb5_error_code ipadb_fill_info3(struct ipadb_context *ipactx,
|
|
LDAPMessage *lentry,
|
|
TALLOC_CTX *memctx,
|
|
struct netr_SamInfo3 *info3)
|
|
{
|
|
LDAP *lcontext = ipactx->lcontext;
|
|
LDAPDerefRes *deref_results = NULL;
|
|
struct dom_sid sid;
|
|
gid_t prigid = -1;
|
|
time_t timeres;
|
|
char *strres;
|
|
int intres;
|
|
int ret;
|
|
|
|
ret = ipadb_ldap_attr_to_int(lcontext, lentry, "gidNumber", &intres);
|
|
if (ret) {
|
|
/* gidNumber is mandatory */
|
|
return ret;
|
|
}
|
|
prigid = intres;
|
|
|
|
|
|
info3->base.logon_time = 0; /* do not have this info yet */
|
|
info3->base.logoff_time = -1; /* do not force logoff */
|
|
|
|
/* TODO: is krbPrinciplaExpiration what we want to use in kickoff_time ?
|
|
* Needs more investigation */
|
|
#if 0
|
|
ret = ipadb_ldap_attr_to_time_t(lcontext, lentry,
|
|
"krbPrincipalExpiration", &timeres);
|
|
switch (ret) {
|
|
case 0:
|
|
unix_to_nt_time(&info3->base.acct_expiry, timeres);
|
|
break;
|
|
case ENOENT:
|
|
info3->base.acct_expiry = -1;
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
#else
|
|
info3->base.kickoff_time = -1;
|
|
#endif
|
|
|
|
ret = ipadb_ldap_attr_to_time_t(lcontext, lentry,
|
|
"krbLastPwdChange", &timeres);
|
|
switch (ret) {
|
|
case 0:
|
|
unix_to_nt_time(&info3->base.last_password_change, timeres);
|
|
break;
|
|
case ENOENT:
|
|
info3->base.last_password_change = 0;
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
|
|
/* TODO: from pw policy (ied->pol) */
|
|
info3->base.allow_password_change = 0;
|
|
info3->base.force_password_change = -1;
|
|
|
|
/* FIXME: handle computer accounts they do not use 'uid' */
|
|
ret = ipadb_ldap_attr_to_str(lcontext, lentry, "uid", &strres);
|
|
if (ret) {
|
|
/* uid is mandatory */
|
|
return ret;
|
|
}
|
|
info3->base.account_name.string = talloc_strdup(memctx, strres);
|
|
free(strres);
|
|
|
|
ret = ipadb_ldap_attr_to_str(lcontext, lentry, "cn", &strres);
|
|
switch (ret) {
|
|
case 0:
|
|
info3->base.full_name.string = talloc_strdup(memctx, strres);
|
|
free(strres);
|
|
break;
|
|
case ENOENT:
|
|
info3->base.full_name.string = "";
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
|
|
ret = ipadb_ldap_attr_to_str(lcontext, lentry,
|
|
"ipaNTLogonScript", &strres);
|
|
switch (ret) {
|
|
case 0:
|
|
info3->base.logon_script.string = talloc_strdup(memctx, strres);
|
|
free(strres);
|
|
break;
|
|
case ENOENT:
|
|
info3->base.logon_script.string = "";
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
|
|
ret = ipadb_ldap_attr_to_str(lcontext, lentry,
|
|
"ipaNTProfilePath", &strres);
|
|
switch (ret) {
|
|
case 0:
|
|
info3->base.profile_path.string = talloc_strdup(memctx, strres);
|
|
free(strres);
|
|
break;
|
|
case ENOENT:
|
|
info3->base.profile_path.string = "";
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
|
|
ret = ipadb_ldap_attr_to_str(lcontext, lentry,
|
|
"ipaNTHomeDirectory", &strres);
|
|
switch (ret) {
|
|
case 0:
|
|
info3->base.home_directory.string = talloc_strdup(memctx, strres);
|
|
free(strres);
|
|
break;
|
|
case ENOENT:
|
|
info3->base.home_directory.string = "";
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
|
|
ret = ipadb_ldap_attr_to_str(lcontext, lentry,
|
|
"ipaNTHomeDirectoryDrive", &strres);
|
|
switch (ret) {
|
|
case 0:
|
|
info3->base.home_drive.string = talloc_strdup(memctx, strres);
|
|
free(strres);
|
|
break;
|
|
case ENOENT:
|
|
info3->base.home_drive.string = "";
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
|
|
info3->base.logon_count = 0; /* we do not have this info yet */
|
|
info3->base.bad_password_count = 0; /* we do not have this info yet */
|
|
|
|
ret = ipadb_ldap_attr_to_str(lcontext, lentry,
|
|
"ipaNTSecurityIdentifier", &strres);
|
|
if (ret) {
|
|
/* SID is mandatory */
|
|
return ret;
|
|
}
|
|
ret = string_to_sid(strres, &sid);
|
|
free(strres);
|
|
if (ret) {
|
|
return ret;
|
|
}
|
|
|
|
ret = sid_split_rid(&sid, &info3->base.rid);
|
|
if (ret) {
|
|
return ret;
|
|
}
|
|
|
|
ret = ipadb_ldap_deref_results(lcontext, lentry, &deref_results);
|
|
switch (ret) {
|
|
LDAPDerefRes *dres;
|
|
LDAPDerefVal *dval;
|
|
struct dom_sid gsid;
|
|
uint32_t trid;
|
|
gid_t tgid;
|
|
char *s;
|
|
int count;
|
|
case 0:
|
|
count = 0;
|
|
for (dres = deref_results; dres; dres = dres->next) {
|
|
count++; /* count*/
|
|
}
|
|
info3->base.groups.rids = talloc_array(memctx,
|
|
struct samr_RidWithAttribute, count);
|
|
if (!info3->base.groups.rids) {
|
|
ldap_derefresponse_free(deref_results);
|
|
return ENOMEM;
|
|
}
|
|
|
|
count = 0;
|
|
info3->base.primary_gid = 0;
|
|
for (dres = deref_results; dres; dres = dres->next) {
|
|
gsid.sid_rev_num = 0;
|
|
tgid = 0;
|
|
for (dval = dres->attrVals; dval; dval = dval->next) {
|
|
if (strcasecmp(dval->type, "gidNumber") == 0) {
|
|
tgid = strtoul((char *)dval->vals[0].bv_val, &s, 10);
|
|
if (tgid == 0) {
|
|
continue;
|
|
}
|
|
}
|
|
if (strcasecmp(dval->type, "ipaNTSecurityIdentifier") == 0) {
|
|
ret = string_to_sid((char *)dval->vals[0].bv_val, &gsid);
|
|
if (ret) {
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
if (tgid && gsid.sid_rev_num) {
|
|
ret = sid_split_rid(&gsid, &trid);
|
|
if (ret) {
|
|
continue;
|
|
}
|
|
if (tgid == prigid) {
|
|
info3->base.primary_gid = trid;
|
|
continue;
|
|
}
|
|
info3->base.groups.rids[count].rid = trid;
|
|
info3->base.groups.rids[count].attributes =
|
|
SE_GROUP_ENABLED |
|
|
SE_GROUP_MANDATORY |
|
|
SE_GROUP_ENABLED_BY_DEFAULT;
|
|
count++;
|
|
}
|
|
}
|
|
info3->base.groups.count = count;
|
|
|
|
ldap_derefresponse_free(deref_results);
|
|
break;
|
|
case ENOENT:
|
|
info3->base.groups.count = 0;
|
|
info3->base.groups.rids = NULL;
|
|
break;
|
|
default:
|
|
return ret;
|
|
}
|
|
|
|
if (info3->base.primary_gid == 0) {
|
|
if (ipactx->mspac->fallback_rid) {
|
|
info3->base.primary_gid = ipactx->mspac->fallback_rid;
|
|
} else {
|
|
/* can't give a pack without a primary group rid */
|
|
return ENOENT;
|
|
}
|
|
}
|
|
|
|
/* always zero out, only valid flags are for extra sids with Krb */
|
|
info3->base.user_flags = 0; /* netr_UserFlags */
|
|
|
|
/* always zero out, not used for Krb, only NTLM */
|
|
memset(&info3->base.key, '\0', sizeof(info3->base.key));
|
|
|
|
if (ipactx->mspac->flat_server_name) {
|
|
info3->base.logon_server.string =
|
|
talloc_strdup(memctx, ipactx->mspac->flat_server_name);
|
|
if (!info3->base.logon_server.string) {
|
|
return ENOMEM;
|
|
}
|
|
} else {
|
|
/* can't give a pack without Server NetBIOS Name :-| */
|
|
return ENOENT;
|
|
}
|
|
|
|
if (ipactx->mspac->flat_domain_name) {
|
|
info3->base.logon_domain.string =
|
|
talloc_strdup(memctx, ipactx->mspac->flat_domain_name);
|
|
if (!info3->base.logon_domain.string) {
|
|
return ENOMEM;
|
|
}
|
|
} else {
|
|
/* can't give a pack without Domain NetBIOS Name :-| */
|
|
return ENOENT;
|
|
}
|
|
|
|
/* we got the domain SID for the user sid */
|
|
info3->base.domain_sid = talloc_memdup(memctx, &sid, sizeof(sid));
|
|
|
|
/* always zero out, not used for Krb, only NTLM */
|
|
memset(&info3->base.LMSessKey, '\0', sizeof(info3->base.key));
|
|
|
|
/* TODO: fill based on objectclass, user vs computer, etc... */
|
|
info3->base.acct_flags = ACB_NORMAL; /* samr_AcctFlags */
|
|
|
|
info3->base.sub_auth_status = 0;
|
|
info3->base.last_successful_logon = 0;
|
|
info3->base.last_failed_logon = 0;
|
|
info3->base.failed_logon_count = 0; /* We do not have it */
|
|
info3->base.reserved = 0; /* Reserved */
|
|
|
|
return 0;
|
|
}
|
|
|
|
static krb5_error_code ipadb_get_pac(krb5_context kcontext,
|
|
krb5_db_entry *client,
|
|
krb5_pac *pac)
|
|
{
|
|
TALLOC_CTX *tmpctx;
|
|
struct ipadb_e_data *ied;
|
|
struct ipadb_context *ipactx;
|
|
LDAPMessage *results = NULL;
|
|
LDAPMessage *lentry;
|
|
DATA_BLOB pac_data;
|
|
krb5_data data;
|
|
union PAC_INFO pac_info;
|
|
krb5_error_code kerr;
|
|
enum ndr_err_code ndr_err;
|
|
|
|
ipactx = ipadb_get_context(kcontext);
|
|
if (!ipactx) {
|
|
return KRB5_KDB_DBNOTINITED;
|
|
}
|
|
|
|
ied = (struct ipadb_e_data *)client->e_data;
|
|
if (ied->magic != IPA_E_DATA_MAGIC) {
|
|
return EINVAL;
|
|
}
|
|
|
|
if (!ied->ipa_user) {
|
|
return 0;
|
|
}
|
|
|
|
tmpctx = talloc_new(NULL);
|
|
if (!tmpctx) {
|
|
return ENOMEM;
|
|
}
|
|
|
|
memset(&pac_info, 0, sizeof(pac_info));
|
|
pac_info.logon_info.info = talloc_zero(tmpctx, struct PAC_LOGON_INFO);
|
|
if (!pac_info.logon_info.info) {
|
|
kerr = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
|
|
/* == Search PAC info == */
|
|
kerr = ipadb_deref_search(ipactx, ied->entry_dn, LDAP_SCOPE_BASE,
|
|
"(objectclass=*)", user_pac_attrs,
|
|
deref_search_attrs, memberof_pac_attrs,
|
|
&results);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
lentry = ldap_first_entry(ipactx->lcontext, results);
|
|
if (!lentry) {
|
|
kerr = ENOENT;
|
|
goto done;
|
|
}
|
|
|
|
/* == Fill Info3 == */
|
|
kerr = ipadb_fill_info3(ipactx, lentry, tmpctx,
|
|
&pac_info.logon_info.info->info3);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
/* == Package PAC == */
|
|
ndr_err = ndr_push_union_blob(&pac_data, tmpctx, &pac_info,
|
|
PAC_TYPE_LOGON_INFO,
|
|
(ndr_push_flags_fn_t)ndr_push_PAC_INFO);
|
|
if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
|
|
kerr = KRB5_KDB_INTERNAL_ERROR;
|
|
goto done;
|
|
}
|
|
|
|
kerr = krb5_pac_init(kcontext, pac);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
data.magic = KV5M_DATA;
|
|
data.data = (char *)pac_data.data;
|
|
data.length = pac_data.length;
|
|
|
|
kerr = krb5_pac_add_buffer(kcontext, *pac, KRB5_PAC_LOGON_INFO, &data);
|
|
|
|
done:
|
|
ldap_msgfree(results);
|
|
talloc_free(tmpctx);
|
|
return kerr;
|
|
}
|
|
|
|
static bool is_cross_realm_krbtgt(krb5_const_principal princ)
|
|
{
|
|
if ((princ->length != 2) ||
|
|
(princ->data[0].length != 6) ||
|
|
(strncasecmp(princ->data[0].data, "krbtgt", 6) != 0)) {
|
|
return false;
|
|
}
|
|
if (princ->data[1].length == princ->realm.length &&
|
|
strncasecmp(princ->data[1].data,
|
|
princ->realm.data, princ->realm.length) == 0) {
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool is_service_of_type(krb5_const_principal princ, const char *type)
|
|
{
|
|
size_t len;
|
|
|
|
if (princ->length < 2) {
|
|
return false;
|
|
}
|
|
|
|
len = strlen(type);
|
|
|
|
if ((princ->data[0].length == len) ||
|
|
(strncasecmp(princ->data[0].data, type, len) == 0)) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static char *gen_sid_string(TALLOC_CTX *memctx, struct dom_sid *dom_sid,
|
|
uint32_t rid)
|
|
{
|
|
char *str = NULL;
|
|
int ret;
|
|
|
|
ret = sid_append_rid(dom_sid, rid);
|
|
if (ret != 0) {
|
|
krb5_klog_syslog(LOG_ERR, "sid_append_rid failed");
|
|
return NULL;
|
|
}
|
|
|
|
str = dom_sid_string(memctx, dom_sid);
|
|
ret = sid_split_rid(dom_sid, NULL);
|
|
if (ret != 0) {
|
|
krb5_klog_syslog(LOG_ERR, "sid_split_rid failed");
|
|
talloc_free(str);
|
|
return NULL;
|
|
}
|
|
|
|
return str;
|
|
}
|
|
|
|
static int get_user_and_group_sids(TALLOC_CTX *memctx,
|
|
struct PAC_LOGON_INFO_CTR *logon_info,
|
|
char ***_group_sids)
|
|
{
|
|
int ret;
|
|
size_t c;
|
|
size_t p = 0;
|
|
struct dom_sid *domain_sid = NULL;
|
|
char **group_sids = NULL;
|
|
|
|
domain_sid = dom_sid_dup(memctx, logon_info->info->info3.base.domain_sid);
|
|
if (domain_sid == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "dom_sid_dup failed");
|
|
ret = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
group_sids = talloc_array(memctx, char *,
|
|
3 +
|
|
logon_info->info->info3.base.groups.count +
|
|
logon_info->info->info3.sidcount);
|
|
if (group_sids == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "talloc_array failed");
|
|
ret = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
group_sids[p] = gen_sid_string(memctx, domain_sid,
|
|
logon_info->info->info3.base.rid);
|
|
if (group_sids[p] == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "gen_sid_string failed");
|
|
ret = EINVAL;
|
|
goto done;
|
|
}
|
|
p++;
|
|
|
|
group_sids[p] = gen_sid_string(memctx, domain_sid,
|
|
logon_info->info->info3.base.primary_gid);
|
|
if (group_sids[p] == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "gen_sid_string failed");
|
|
ret = EINVAL;
|
|
goto done;
|
|
}
|
|
p++;
|
|
|
|
for (c = 0; c < logon_info->info->info3.base.groups.count; c++) {
|
|
group_sids[p] = gen_sid_string(memctx, domain_sid,
|
|
logon_info->info->info3.base.groups.rids[c].rid);
|
|
if (group_sids[p] == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "gen_sid_string 2 failed");
|
|
ret = EINVAL;
|
|
goto done;
|
|
}
|
|
p++;
|
|
}
|
|
for (c = 0; c < logon_info->info->info3.sidcount; c++) {
|
|
group_sids[p] = dom_sid_string(memctx,
|
|
logon_info->info->info3.sids[c].sid);
|
|
if (group_sids[p] == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "dom_sid_string failed");
|
|
ret = EINVAL;
|
|
goto done;
|
|
}
|
|
p++;
|
|
}
|
|
|
|
group_sids[p] = NULL;
|
|
|
|
*_group_sids = group_sids;
|
|
|
|
ret = 0;
|
|
done:
|
|
talloc_free(domain_sid);
|
|
if (ret != 0) {
|
|
talloc_free(group_sids);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int add_groups(TALLOC_CTX *memctx,
|
|
struct PAC_LOGON_INFO_CTR *logon_info,
|
|
size_t ipa_group_sids_count,
|
|
struct dom_sid2 *ipa_group_sids)
|
|
{
|
|
size_t c;
|
|
struct netr_SidAttr *sids = NULL;
|
|
|
|
if (ipa_group_sids_count == 0) {
|
|
return 0;
|
|
}
|
|
|
|
sids = talloc_realloc(memctx, logon_info->info->info3.sids,
|
|
struct netr_SidAttr,
|
|
logon_info->info->info3.sidcount + ipa_group_sids_count);
|
|
if (sids == NULL) {
|
|
return ENOMEM;
|
|
}
|
|
|
|
|
|
for (c = 0; c < ipa_group_sids_count; c++) {
|
|
sids[c + logon_info->info->info3.sidcount].sid = &ipa_group_sids[c];
|
|
sids[c + logon_info->info->info3.sidcount].attributes =
|
|
SE_GROUP_ENABLED |
|
|
SE_GROUP_MANDATORY |
|
|
SE_GROUP_ENABLED_BY_DEFAULT;
|
|
}
|
|
|
|
logon_info->info->info3.sidcount += ipa_group_sids_count;
|
|
logon_info->info->info3.sids = sids;
|
|
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int map_groups(TALLOC_CTX *memctx, krb5_context kcontext,
|
|
char **group_sids, size_t *_ipa_group_sids_count,
|
|
struct dom_sid **_ipa_group_sids)
|
|
{
|
|
struct ipadb_context *ipactx;
|
|
krb5_error_code kerr;
|
|
int ret;
|
|
LDAPMessage *results = NULL;
|
|
LDAPMessage *lentry;
|
|
char *basedn = NULL;
|
|
char *filter = NULL;
|
|
LDAPDerefRes *deref_results = NULL;
|
|
LDAPDerefRes *dres;
|
|
LDAPDerefVal *dval;
|
|
size_t c;
|
|
size_t count = 0;
|
|
size_t sid_index = 0;
|
|
struct dom_sid *sids = NULL;
|
|
char *entry_attrs[] ={"1.1", NULL};
|
|
unsigned long gid;
|
|
struct dom_sid sid;
|
|
char *endptr;
|
|
|
|
ipactx = ipadb_get_context(kcontext);
|
|
if (ipactx == NULL) {
|
|
return KRB5_KDB_DBNOTINITED;
|
|
}
|
|
|
|
basedn = talloc_asprintf(memctx, "cn=groups,cn=accounts,%s", ipactx->base);
|
|
if (basedn == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "talloc_asprintf failed.");
|
|
kerr = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
for (c = 0; group_sids[c] != NULL; c++) {
|
|
talloc_free(filter);
|
|
filter = talloc_asprintf(memctx, "(&(objectclass=ipaExternalGroup)(ipaExternalMember=%s))",
|
|
group_sids[c]);
|
|
if (filter == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "talloc_asprintf failed.");
|
|
kerr = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
kerr = ipadb_deref_search(ipactx, basedn, LDAP_SCOPE_ONE, filter,
|
|
entry_attrs, deref_search_attrs,
|
|
memberof_pac_attrs, &results);
|
|
if (kerr != 0) {
|
|
krb5_klog_syslog(LOG_ERR, "ipadb_deref_search failed.");
|
|
goto done;
|
|
}
|
|
|
|
lentry = ldap_first_entry(ipactx->lcontext, results);
|
|
if (lentry == NULL) {
|
|
continue;
|
|
}
|
|
|
|
ldap_derefresponse_free(deref_results);
|
|
ret = ipadb_ldap_deref_results(ipactx->lcontext, lentry, &deref_results);
|
|
switch (ret) {
|
|
case ENOENT:
|
|
/* No entry found, try next SID */
|
|
break;
|
|
case 0:
|
|
if (deref_results == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "No results.");
|
|
break;
|
|
}
|
|
|
|
for (dres = deref_results; dres; dres = dres->next) {
|
|
count++;
|
|
}
|
|
|
|
sids = talloc_realloc(memctx, sids, struct dom_sid, count);
|
|
if (sids == NULL) {
|
|
krb5_klog_syslog(LOG_ERR, "talloc_realloc failed.");
|
|
kerr = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
for (dres = deref_results; dres; dres = dres->next) {
|
|
gid = 0;
|
|
memset(&sid, '\0', sizeof(struct dom_sid));
|
|
for (dval = dres->attrVals; dval; dval = dval->next) {
|
|
if (strcasecmp(dval->type, "gidNumber") == 0) {
|
|
errno = 0;
|
|
gid = strtoul((char *)dval->vals[0].bv_val,
|
|
&endptr,10);
|
|
if (gid == 0 || gid >= UINT32_MAX || errno != 0 ||
|
|
*endptr != '\0') {
|
|
continue;
|
|
}
|
|
}
|
|
if (strcasecmp(dval->type,
|
|
"ipaNTSecurityIdentifier") == 0) {
|
|
kerr = string_to_sid((char *)dval->vals[0].bv_val, &sid);
|
|
if (kerr != 0) {
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
if (gid != 0 && sid.sid_rev_num != 0) {
|
|
/* TODO: check if gid maps to sid */
|
|
if (sid_index >= count) {
|
|
krb5_klog_syslog(LOG_ERR, "Index larger than "
|
|
"array, this shoould "
|
|
"never happen.");
|
|
kerr = EFAULT;
|
|
goto done;
|
|
}
|
|
memcpy(&sids[sid_index], &sid, sizeof(struct dom_sid));
|
|
sid_index++;
|
|
}
|
|
}
|
|
|
|
break;
|
|
default:
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
*_ipa_group_sids_count = sid_index;
|
|
*_ipa_group_sids = sids;
|
|
|
|
kerr = 0;
|
|
|
|
done:
|
|
ldap_derefresponse_free(deref_results);
|
|
talloc_free(basedn);
|
|
talloc_free(filter);
|
|
ldap_msgfree(results);
|
|
return kerr;
|
|
}
|
|
|
|
static krb5_error_code get_logon_info(krb5_context context,
|
|
TALLOC_CTX *memctx,
|
|
krb5_data *pac_blob,
|
|
struct PAC_LOGON_INFO_CTR *info)
|
|
{
|
|
DATA_BLOB pac_data;
|
|
enum ndr_err_code ndr_err;
|
|
|
|
pac_data.length = pac_blob->length;
|
|
pac_data.data = (uint8_t *)pac_blob->data;
|
|
|
|
ndr_err = ndr_pull_union_blob(&pac_data, memctx, info,
|
|
PAC_TYPE_LOGON_INFO,
|
|
(ndr_pull_flags_fn_t)ndr_pull_PAC_INFO);
|
|
if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
|
|
return KRB5_KDB_INTERNAL_ERROR;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static krb5_error_code add_local_groups(krb5_context context,
|
|
TALLOC_CTX *memctx,
|
|
struct PAC_LOGON_INFO_CTR *info)
|
|
{
|
|
int ret;
|
|
char **group_sids = NULL;
|
|
size_t ipa_group_sids_count = 0;
|
|
struct dom_sid *ipa_group_sids = NULL;
|
|
|
|
ret = get_user_and_group_sids(memctx, info, &group_sids);
|
|
if (ret != 0) {
|
|
return KRB5_KDB_INTERNAL_ERROR;
|
|
}
|
|
|
|
ret = map_groups(memctx, context, group_sids, &ipa_group_sids_count,
|
|
&ipa_group_sids);
|
|
if (ret != 0) {
|
|
return KRB5_KDB_INTERNAL_ERROR;
|
|
}
|
|
|
|
ret = add_groups(memctx, info, ipa_group_sids_count, ipa_group_sids);
|
|
if (ret != 0) {
|
|
krb5_klog_syslog(LOG_ERR, "add_groups failed");
|
|
return KRB5_KDB_INTERNAL_ERROR;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static krb5_error_code save_logon_info(krb5_context context,
|
|
TALLOC_CTX *memctx,
|
|
struct PAC_LOGON_INFO_CTR *info,
|
|
krb5_data *pac_blob)
|
|
{
|
|
DATA_BLOB pac_data;
|
|
enum ndr_err_code ndr_err;
|
|
|
|
ndr_err = ndr_push_union_blob(&pac_data, memctx, info,
|
|
PAC_TYPE_LOGON_INFO,
|
|
(ndr_push_flags_fn_t)ndr_push_PAC_INFO);
|
|
if (!NDR_ERR_CODE_IS_SUCCESS(ndr_err)) {
|
|
return KRB5_KDB_INTERNAL_ERROR;
|
|
}
|
|
|
|
free(pac_blob->data);
|
|
pac_blob->data = malloc(pac_data.length);
|
|
if (pac_blob->data == NULL) {
|
|
pac_blob->length = 0;
|
|
return ENOMEM;
|
|
}
|
|
memcpy(pac_blob->data, pac_data.data, pac_data.length);
|
|
pac_blob->length = pac_data.length;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static struct ipadb_adtrusts *get_domain_from_realm(krb5_context context,
|
|
krb5_data realm)
|
|
{
|
|
struct ipadb_context *ipactx;
|
|
struct ipadb_adtrusts *domain;
|
|
int i;
|
|
|
|
ipactx = ipadb_get_context(context);
|
|
if (!ipactx) {
|
|
return NULL;
|
|
}
|
|
|
|
if (ipactx->mspac == NULL) {
|
|
return NULL;
|
|
}
|
|
|
|
for (i = 0; i < ipactx->mspac->num_trusts; i++) {
|
|
domain = &ipactx->mspac->trusts[i];
|
|
if (strlen(domain->domain_name) != realm.length) {
|
|
continue;
|
|
}
|
|
if (strncasecmp(domain->domain_name, realm.data, realm.length) == 0) {
|
|
return domain;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static struct ipadb_adtrusts *get_domain_from_realm_update(krb5_context context,
|
|
krb5_data realm)
|
|
{
|
|
struct ipadb_context *ipactx;
|
|
struct ipadb_adtrusts *domain;
|
|
krb5_error_code kerr;
|
|
|
|
domain = get_domain_from_realm(context, realm);
|
|
if (domain == NULL) {
|
|
ipactx = ipadb_get_context(context);
|
|
if (!ipactx) {
|
|
return NULL;
|
|
}
|
|
|
|
kerr = ipadb_reinit_mspac(ipactx);
|
|
if (kerr != 0) {
|
|
return NULL;
|
|
}
|
|
|
|
domain = get_domain_from_realm(context, realm);
|
|
}
|
|
|
|
return domain;
|
|
}
|
|
|
|
static krb5_error_code filter_logon_info(krb5_context context,
|
|
TALLOC_CTX *memctx,
|
|
krb5_data realm,
|
|
struct PAC_LOGON_INFO_CTR *info)
|
|
{
|
|
|
|
/* We must refuse a PAC that comes signed with a cross realm TGT
|
|
* where the client pretends to be from a different realm. It is an
|
|
* attempt at getting us to sign fake credentials with the help of a
|
|
* compromised trusted realm */
|
|
|
|
struct ipadb_adtrusts *domain;
|
|
char *domsid;
|
|
|
|
domain = get_domain_from_realm_update(context, realm);
|
|
if (!domain) {
|
|
return EINVAL;
|
|
}
|
|
|
|
/* check netbios/flat name */
|
|
if (strcasecmp(info->info->info3.base.logon_domain.string,
|
|
domain->flat_name) != 0) {
|
|
krb5_klog_syslog(LOG_ERR, "PAC Info mismatch: domain = %s, "
|
|
"expected flat name = %s, "
|
|
"found logon name = %s",
|
|
domain->domain_name, domain->flat_name,
|
|
info->info->info3.base.logon_domain.string);
|
|
return EINVAL;
|
|
}
|
|
|
|
/* check sid */
|
|
domsid = dom_sid_string(NULL, info->info->info3.base.domain_sid);
|
|
if (!domsid) {
|
|
return EINVAL;
|
|
}
|
|
|
|
if (strcmp(domsid, domain->domain_sid) != 0) {
|
|
krb5_klog_syslog(LOG_ERR, "PAC Info mismatch: domain = %s, "
|
|
"expected domain SID = %s, "
|
|
"found domain SID = %s",
|
|
domain->domain_name, domain->domain_sid,
|
|
domsid);
|
|
talloc_free(domsid);
|
|
return EINVAL;
|
|
}
|
|
talloc_free(domsid);
|
|
|
|
/* According to MS-KILE, info->info->info3.sids must be zero, so check
|
|
* that it is the case here */
|
|
if (info->info->info3.sidcount != 0) {
|
|
return EINVAL;
|
|
}
|
|
|
|
/* According to MS-KILE, ResourceGroups must be zero, so check
|
|
* that it is the case here */
|
|
if (info->info->res_group_dom_sid != NULL &&
|
|
info->info->res_groups.count != 0) {
|
|
return EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static krb5_error_code ipadb_check_logon_info(krb5_context context,
|
|
krb5_data origin_realm,
|
|
krb5_data *pac_blob)
|
|
{
|
|
struct PAC_LOGON_INFO_CTR info;
|
|
krb5_error_code kerr;
|
|
TALLOC_CTX *tmpctx;
|
|
|
|
tmpctx = talloc_new(NULL);
|
|
if (!tmpctx) {
|
|
return ENOMEM;
|
|
}
|
|
|
|
kerr = get_logon_info(context, tmpctx, pac_blob, &info);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
kerr = filter_logon_info(context, tmpctx, origin_realm, &info);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
kerr = add_local_groups(context, tmpctx, &info);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
kerr = save_logon_info(context, tmpctx, &info, pac_blob);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
done:
|
|
talloc_free(tmpctx);
|
|
return kerr;
|
|
}
|
|
|
|
static krb5_error_code ipadb_verify_pac(krb5_context context,
|
|
unsigned int flags,
|
|
krb5_const_principal client_princ,
|
|
krb5_db_entry *server,
|
|
krb5_db_entry *krbtgt,
|
|
krb5_keyblock *server_key,
|
|
krb5_keyblock *krbtgt_key,
|
|
krb5_timestamp authtime,
|
|
krb5_authdata **authdata,
|
|
krb5_pac *pac)
|
|
{
|
|
krb5_keyblock *srv_key = NULL;
|
|
krb5_keyblock *priv_key = NULL;
|
|
krb5_error_code kerr;
|
|
krb5_ui_4 *types = NULL;
|
|
size_t num_buffers;
|
|
krb5_pac old_pac = NULL;
|
|
krb5_pac new_pac = NULL;
|
|
krb5_data data;
|
|
krb5_data pac_blob = { 0 , 0, NULL};
|
|
bool is_cross_realm = false;
|
|
size_t i;
|
|
|
|
kerr = krb5_pac_parse(context,
|
|
authdata[0]->contents,
|
|
authdata[0]->length,
|
|
&old_pac);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
/* for cross realm trusts cases we need to check the right checksum.
|
|
* when the PAC is signed by our realm, we can always just check it
|
|
* passing our realm krbtgt key as the kdc checksum key (privsvr).
|
|
* But when a trusted realm passes us a PAC the kdc checksum is
|
|
* generated with that realm krbtgt key, so we need to use the cross
|
|
* realm krbtgt to check the 'server' checksum instead. */
|
|
if (is_cross_realm_krbtgt(krbtgt->princ)) {
|
|
/* krbtgt from a trusted realm */
|
|
is_cross_realm = true;
|
|
|
|
srv_key = krbtgt_key;
|
|
|
|
} else {
|
|
/* krbtgt from our own realm */
|
|
priv_key = krbtgt_key;
|
|
}
|
|
|
|
kerr = krb5_pac_verify(context, old_pac, authtime,
|
|
client_princ, srv_key, priv_key);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
/* Now that the PAc is verified augment it with additional info if
|
|
* it is coming from a different realm */
|
|
if (is_cross_realm) {
|
|
kerr = krb5_pac_get_buffer(context, old_pac,
|
|
KRB5_PAC_LOGON_INFO, &pac_blob);
|
|
if (kerr != 0) {
|
|
goto done;
|
|
}
|
|
|
|
kerr = ipadb_check_logon_info(context, client_princ->realm, &pac_blob);
|
|
if (kerr != 0) {
|
|
goto done;
|
|
}
|
|
}
|
|
/* extract buffers and rebuilt pac from scratch so that when re-signing
|
|
* with a different cksum type does not cause issues due to mismatching
|
|
* signature buffer lengths */
|
|
kerr = krb5_pac_init(context, &new_pac);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
kerr = krb5_pac_get_types(context, old_pac, &num_buffers, &types);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
for (i = 0; i < num_buffers; i++) {
|
|
if (types[i] == KRB5_PAC_SERVER_CHECKSUM ||
|
|
types[i] == KRB5_PAC_PRIVSVR_CHECKSUM) {
|
|
continue;
|
|
}
|
|
|
|
if (types[i] == KRB5_PAC_LOGON_INFO &&
|
|
pac_blob.length != 0) {
|
|
kerr = krb5_pac_add_buffer(context, new_pac, types[i], &pac_blob);
|
|
if (kerr) {
|
|
krb5_pac_free(context, new_pac);
|
|
goto done;
|
|
}
|
|
|
|
continue;
|
|
}
|
|
|
|
kerr = krb5_pac_get_buffer(context, old_pac, types[i], &data);
|
|
if (kerr == 0) {
|
|
kerr = krb5_pac_add_buffer(context, new_pac, types[i], &data);
|
|
krb5_free_data_contents(context, &data);
|
|
}
|
|
if (kerr) {
|
|
krb5_pac_free(context, new_pac);
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
*pac = new_pac;
|
|
|
|
done:
|
|
krb5_free_authdata(context, authdata);
|
|
krb5_pac_free(context, old_pac);
|
|
krb5_free_data_contents(context, &pac_blob);
|
|
free(types);
|
|
return kerr;
|
|
}
|
|
|
|
static krb5_error_code ipadb_sign_pac(krb5_context context,
|
|
krb5_const_principal client_princ,
|
|
krb5_db_entry *server,
|
|
krb5_db_entry *krbtgt,
|
|
krb5_keyblock *server_key,
|
|
krb5_keyblock *krbtgt_key,
|
|
krb5_timestamp authtime,
|
|
krb5_pac pac,
|
|
krb5_data *pac_data)
|
|
{
|
|
krb5_keyblock *right_krbtgt_signing_key = NULL;
|
|
krb5_key_data *right_krbtgt_key;
|
|
krb5_db_entry *right_krbtgt = NULL;
|
|
krb5_principal krbtgt_princ = NULL;
|
|
krb5_error_code kerr;
|
|
char *princ = NULL;
|
|
int ret;
|
|
|
|
/* for cross realm trusts cases we need to sign with the right key.
|
|
* we need to fetch the right key on our own until the DAL is fixed
|
|
* to pass us separate check tgt keys and sign tgt keys */
|
|
|
|
/* We can only ever create the kdc checksum with our realm tgt key.
|
|
* So, if we get a cross realm tgt we have to fetch our realm tgt
|
|
* instead. */
|
|
if (is_cross_realm_krbtgt(krbtgt->princ)) {
|
|
|
|
ret = asprintf(&princ, "krbtgt/%.*s@%.*s",
|
|
server->princ->realm.length,
|
|
server->princ->realm.data,
|
|
server->princ->realm.length,
|
|
server->princ->realm.data);
|
|
if (ret == -1) {
|
|
princ = NULL;
|
|
kerr = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
kerr = krb5_parse_name(context, princ, &krbtgt_princ);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
kerr = ipadb_get_principal(context, krbtgt_princ, 0, &right_krbtgt);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
kerr = krb5_dbe_find_enctype(context, right_krbtgt,
|
|
-1, -1, 0, &right_krbtgt_key);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
if (!right_krbtgt_key) {
|
|
kerr = KRB5KDC_ERR_S_PRINCIPAL_UNKNOWN;
|
|
goto done;
|
|
}
|
|
|
|
right_krbtgt_signing_key = malloc(sizeof(krb5_keyblock));
|
|
if (!right_krbtgt_signing_key) {
|
|
kerr = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
kerr = krb5_dbe_decrypt_key_data(context, NULL, right_krbtgt_key,
|
|
right_krbtgt_signing_key, NULL);
|
|
if (kerr) {
|
|
goto done;
|
|
}
|
|
|
|
} else {
|
|
right_krbtgt_signing_key = krbtgt_key;
|
|
}
|
|
|
|
kerr = krb5_pac_sign(context, pac, authtime, client_princ,
|
|
server_key, right_krbtgt_signing_key, pac_data);
|
|
|
|
done:
|
|
free(princ);
|
|
krb5_free_principal(context, krbtgt_princ);
|
|
ipadb_free_principal(context, right_krbtgt);
|
|
if (right_krbtgt_signing_key != krbtgt_key) {
|
|
krb5_free_keyblock(context, right_krbtgt_signing_key);
|
|
}
|
|
return kerr;
|
|
}
|
|
|
|
krb5_error_code ipadb_sign_authdata(krb5_context context,
|
|
unsigned int flags,
|
|
krb5_const_principal client_princ,
|
|
krb5_db_entry *client,
|
|
krb5_db_entry *server,
|
|
krb5_db_entry *krbtgt,
|
|
krb5_keyblock *client_key,
|
|
krb5_keyblock *server_key,
|
|
krb5_keyblock *krbtgt_key,
|
|
krb5_keyblock *session_key,
|
|
krb5_timestamp authtime,
|
|
krb5_authdata **tgt_auth_data,
|
|
krb5_authdata ***signed_auth_data)
|
|
{
|
|
krb5_const_principal ks_client_princ;
|
|
krb5_authdata **pac_auth_data = NULL;
|
|
krb5_authdata *authdata[2] = { NULL, NULL };
|
|
krb5_authdata ad;
|
|
krb5_boolean is_as_req;
|
|
krb5_error_code kerr;
|
|
krb5_pac pac = NULL;
|
|
krb5_data pac_data;
|
|
bool is_nfs = false;
|
|
|
|
/* When using s4u2proxy client_princ actually refers to the proxied user
|
|
* while client->princ to the proxy service asking for the TGS on behalf
|
|
* of the proxied user. So always use client_princ in preference */
|
|
if (client_princ != NULL) {
|
|
ks_client_princ = client_princ;
|
|
} else {
|
|
ks_client_princ = client->princ;
|
|
}
|
|
|
|
/* NFS Server on Linux is limited and will choke on big tickets.
|
|
* So avoid attachnig the PAC to nfs/ tickets for now.
|
|
* FIXME: remove this when we have interface to support disabling
|
|
* PACs on arbitrary services */
|
|
if (is_service_of_type(ks_client_princ, "nfs") ||
|
|
is_service_of_type(server->princ, "nfs")) {
|
|
is_nfs = true;
|
|
}
|
|
|
|
is_as_req = ((flags & KRB5_KDB_FLAG_CLIENT_REFERRALS_ONLY) != 0);
|
|
|
|
if (is_as_req && (flags & KRB5_KDB_FLAG_INCLUDE_PAC)) {
|
|
|
|
if (is_nfs) {
|
|
*signed_auth_data = NULL;
|
|
kerr = 0;
|
|
goto done;
|
|
}
|
|
|
|
kerr = ipadb_get_pac(context, client, &pac);
|
|
if (kerr != 0 && kerr != ENOENT) {
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
if (!is_as_req & !is_nfs) {
|
|
/* find the existing PAC, if present */
|
|
kerr = krb5_find_authdata(context, tgt_auth_data, NULL,
|
|
KRB5_AUTHDATA_WIN2K_PAC, &pac_auth_data);
|
|
if (kerr != 0) {
|
|
goto done;
|
|
}
|
|
/* check or generate pac data */
|
|
if ((pac_auth_data == NULL) || (pac_auth_data[0] == NULL)) {
|
|
if (flags & KRB5_KDB_FLAG_CONSTRAINED_DELEGATION) {
|
|
kerr = ipadb_get_pac(context, client, &pac);
|
|
if (kerr != 0 && kerr != ENOENT) {
|
|
goto done;
|
|
}
|
|
}
|
|
} else {
|
|
if (pac_auth_data[1] != NULL) {
|
|
kerr = KRB5KDC_ERR_BADOPTION; /* FIXME: right error ? */
|
|
goto done;
|
|
}
|
|
|
|
kerr = ipadb_verify_pac(context, flags, ks_client_princ,
|
|
server, krbtgt, server_key, krbtgt_key,
|
|
authtime, pac_auth_data, &pac);
|
|
if (kerr != 0) {
|
|
goto done;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pac == NULL) {
|
|
/* No PAC to deal with, proceed */
|
|
*signed_auth_data = NULL;
|
|
kerr = 0;
|
|
goto done;
|
|
}
|
|
|
|
kerr = ipadb_sign_pac(context, ks_client_princ, server, krbtgt,
|
|
server_key, krbtgt_key, authtime, pac, &pac_data);
|
|
if (kerr != 0) {
|
|
goto done;
|
|
}
|
|
|
|
/* put in signed data */
|
|
ad.magic = KV5M_AUTHDATA;
|
|
ad.ad_type = KRB5_AUTHDATA_WIN2K_PAC;
|
|
ad.contents = (krb5_octet *)pac_data.data;
|
|
ad.length = pac_data.length;
|
|
authdata[0] = &ad;
|
|
|
|
kerr = krb5_encode_authdata_container(context,
|
|
KRB5_AUTHDATA_IF_RELEVANT,
|
|
authdata,
|
|
signed_auth_data);
|
|
if (kerr != 0) {
|
|
goto done;
|
|
}
|
|
|
|
kerr = 0;
|
|
|
|
done:
|
|
krb5_pac_free(context, pac);
|
|
return kerr;
|
|
}
|
|
|
|
static char *get_server_netbios_name(void)
|
|
{
|
|
char hostname[MAXHOSTNAMELEN + 1]; /* NOTE: this is 64, too little ? */
|
|
char *p;
|
|
int ret;
|
|
|
|
ret = gethostname(hostname, MAXHOSTNAMELEN);
|
|
if (ret) {
|
|
return NULL;
|
|
}
|
|
/* May miss termination */
|
|
hostname[MAXHOSTNAMELEN] = '\0';
|
|
for (p = hostname; *p; p++) {
|
|
if (*p == '.') {
|
|
*p = 0;
|
|
break;
|
|
} else {
|
|
*p = toupper(*p);
|
|
}
|
|
}
|
|
|
|
return strdup(hostname);
|
|
}
|
|
|
|
void ipadb_mspac_struct_free(struct ipadb_mspac **mspac)
|
|
{
|
|
int i;
|
|
|
|
if (!*mspac) return;
|
|
|
|
free((*mspac)->flat_domain_name);
|
|
free((*mspac)->fallback_group);
|
|
|
|
if ((*mspac)->num_trusts) {
|
|
for (i = 0; i < (*mspac)->num_trusts; i++) {
|
|
free((*mspac)->trusts[i].domain_name);
|
|
free((*mspac)->trusts[i].flat_name);
|
|
free((*mspac)->trusts[i].domain_sid);
|
|
}
|
|
}
|
|
|
|
*mspac = NULL;
|
|
}
|
|
|
|
krb5_error_code ipadb_mspac_get_trusted_domains(struct ipadb_context *ipactx)
|
|
{
|
|
struct ipadb_adtrusts *t;
|
|
LDAP *lc = ipactx->lcontext;
|
|
char *attrs[] = { "ipaNTTrustPartner", "ipaNTFlatName",
|
|
"ipaNTTrustedDomainSID", NULL };
|
|
char *filter = "(objectclass=ipaNTTrustedDomain)";
|
|
krb5_error_code kerr;
|
|
LDAPMessage *res = NULL;
|
|
LDAPMessage *le;
|
|
char *base = NULL;
|
|
int ret, n;
|
|
|
|
ret = asprintf(&base, "cn=ad,cn=trusts,%s", ipactx->base);
|
|
if (ret == -1) {
|
|
ret = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
kerr = ipadb_simple_search(ipactx, base, LDAP_SCOPE_SUBTREE,
|
|
filter, attrs, &res);
|
|
if (kerr == KRB5_KDB_NOENTRY) {
|
|
/* nothing to do, there are no trusts */
|
|
ret = 0;
|
|
goto done;
|
|
} else if (kerr != 0) {
|
|
ret = EIO;
|
|
goto done;
|
|
}
|
|
|
|
for (le = ldap_first_entry(lc, res); le; le = ldap_next_entry(lc, le)) {
|
|
n = ipactx->mspac->num_trusts;
|
|
ipactx->mspac->num_trusts++;
|
|
t = realloc(ipactx->mspac->trusts,
|
|
sizeof(struct ipadb_adtrusts) * ipactx->mspac->num_trusts);
|
|
if (!t) {
|
|
ret = ENOMEM;
|
|
goto done;
|
|
}
|
|
ipactx->mspac->trusts = t;
|
|
|
|
ret = ipadb_ldap_attr_to_str(lc, le, "ipaNTTrustPartner",
|
|
&t[n].domain_name);
|
|
if (ret) {
|
|
ret = EINVAL;
|
|
goto done;
|
|
}
|
|
|
|
ret = ipadb_ldap_attr_to_str(lc, le, "ipaNTFlatName",
|
|
&t[n].flat_name);
|
|
if (ret) {
|
|
ret = EINVAL;
|
|
goto done;
|
|
}
|
|
|
|
ret = ipadb_ldap_attr_to_str(lc, le, "ipaNTTrustedDomainSID",
|
|
&t[n].domain_sid);
|
|
if (ret) {
|
|
ret = EINVAL;
|
|
goto done;
|
|
}
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
done:
|
|
if (ret != 0) {
|
|
krb5_klog_syslog(LOG_ERR, "Failed to read list of trusted domains");
|
|
}
|
|
free(base);
|
|
return ret;
|
|
}
|
|
|
|
krb5_error_code ipadb_reinit_mspac(struct ipadb_context *ipactx)
|
|
{
|
|
char *dom_attrs[] = { "ipaNTFlatName",
|
|
"ipaNTFallbackPrimaryGroup",
|
|
NULL };
|
|
char *grp_attrs[] = { "ipaNTSecurityIdentifier", NULL };
|
|
krb5_error_code kerr;
|
|
LDAPMessage *result = NULL;
|
|
LDAPMessage *lentry;
|
|
struct dom_sid gsid;
|
|
char *resstr;
|
|
int ret;
|
|
time_t now;
|
|
|
|
/* Do not update the mspac struct more than once a minute. This would
|
|
* avoid heavy load on the directory server if there are lots of requests
|
|
* from domains which we do not trust. */
|
|
now = time(NULL);
|
|
if (ipactx->mspac != NULL && now > ipactx->mspac->last_update &&
|
|
(now - ipactx->mspac->last_update) < 60) {
|
|
return 0;
|
|
}
|
|
|
|
/* clean up in case we had old values around */
|
|
ipadb_mspac_struct_free(&ipactx->mspac);
|
|
|
|
ipactx->mspac = calloc(1, sizeof(struct ipadb_mspac));
|
|
if (!ipactx->mspac) {
|
|
kerr = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
ipactx->mspac->last_update = now;
|
|
|
|
kerr = ipadb_simple_search(ipactx, ipactx->base, LDAP_SCOPE_SUBTREE,
|
|
"(objectclass=ipaNTDomainAttrs)", dom_attrs,
|
|
&result);
|
|
if (kerr == KRB5_KDB_NOENTRY) {
|
|
return ENOENT;
|
|
} else if (kerr != 0) {
|
|
return EIO;
|
|
}
|
|
|
|
lentry = ldap_first_entry(ipactx->lcontext, result);
|
|
if (!lentry) {
|
|
kerr = ENOENT;
|
|
goto done;
|
|
}
|
|
|
|
ret = ipadb_ldap_attr_to_str(ipactx->lcontext, lentry,
|
|
"ipaNTFlatName",
|
|
&ipactx->mspac->flat_domain_name);
|
|
if (ret) {
|
|
kerr = ret;
|
|
goto done;
|
|
}
|
|
|
|
free(ipactx->mspac->flat_server_name);
|
|
ipactx->mspac->flat_server_name = get_server_netbios_name();
|
|
if (!ipactx->mspac->flat_server_name) {
|
|
kerr = ENOMEM;
|
|
goto done;
|
|
}
|
|
|
|
ret = ipadb_ldap_attr_to_str(ipactx->lcontext, lentry,
|
|
"ipaNTFallbackPrimaryGroup",
|
|
&ipactx->mspac->fallback_group);
|
|
if (ret && ret != ENOENT) {
|
|
kerr = ret;
|
|
goto done;
|
|
}
|
|
|
|
/* result and lentry not valid any more from here on */
|
|
ldap_msgfree(result);
|
|
result = NULL;
|
|
lentry = NULL;
|
|
|
|
if (ret != ENOENT) {
|
|
kerr = ipadb_simple_search(ipactx, ipactx->mspac->fallback_group,
|
|
LDAP_SCOPE_BASE,
|
|
"(objectclass=posixGroup)",
|
|
grp_attrs, &result);
|
|
if (kerr && kerr != KRB5_KDB_NOENTRY) {
|
|
kerr = ret;
|
|
goto done;
|
|
}
|
|
|
|
lentry = ldap_first_entry(ipactx->lcontext, result);
|
|
if (!lentry) {
|
|
kerr = ENOENT;
|
|
goto done;
|
|
}
|
|
|
|
if (kerr == 0) {
|
|
ret = ipadb_ldap_attr_to_str(ipactx->lcontext, lentry,
|
|
"ipaNTSecurityIdentifier",
|
|
&resstr);
|
|
if (ret && ret != ENOENT) {
|
|
kerr = ret;
|
|
goto done;
|
|
}
|
|
if (ret == 0) {
|
|
ret = string_to_sid(resstr, &gsid);
|
|
if (ret) {
|
|
kerr = ret;
|
|
goto done;
|
|
}
|
|
ret = sid_split_rid(&gsid, &ipactx->mspac->fallback_rid);
|
|
if (ret) {
|
|
kerr = ret;
|
|
goto done;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
kerr = ipadb_mspac_get_trusted_domains(ipactx);
|
|
|
|
done:
|
|
ldap_msgfree(result);
|
|
return kerr;
|
|
}
|