/* This file is part of the Project Athena Zephyr Notification System. * It contains source for the ZMakeAuthentication function. * * Created by: Robert French * * $Id: ZMkAuth.c,v 1.19 1999/01/22 23:19:16 ghudson Exp $ * * Copyright (c) 1987 by the Massachusetts Institute of Technology. * For copying and distribution information, see the file * "mit-copyright.h". */ #include #ifndef lint static const char rcsid_ZMakeAuthentication_c[] = "$Id: ZMkAuth.c,v 1.19 1999/01/22 23:19:16 ghudson Exp $"; #endif Code_t ZResetAuthentication () { #ifdef HAVE_KRB4 int i; for (i=0; i<__ngalaxies; i++) __galaxy_list[i].last_authent_time = 0; #endif return ZERR_NONE; } Code_t ZMakeAuthentication(notice, buffer, buffer_len, phdr_len) register ZNotice_t *notice; char *buffer; int buffer_len; int *phdr_len; { #ifdef HAVE_KRB4 int i; int cksum_len; int result; time_t now; KTEXT_ST authent; char *krealm, *cksum_start, *cstart, *cend; ZChecksum_t checksum; CREDENTIALS cred; if (notice->z_dest_galaxy && *notice->z_dest_galaxy) { for (i=0; i<__ngalaxies; i++) { if (strcasecmp(notice->z_dest_galaxy, __galaxy_list[i].galaxy_config.galaxy) == 0) break; } if (i == __ngalaxies) return(ZERR_GALAXYUNKNOWN); } else { i = __default_galaxy; } krealm = __galaxy_list[i].krealm; if (krealm[0] == '\0') #endif { notice->z_checksum = 0; notice->z_auth = 1; notice->z_authent_len = 0; notice->z_ascii_authent = ""; return (Z_FormatRawHeader(notice, buffer, buffer_len, phdr_len, NULL, NULL, NULL, NULL)); } #ifdef HAVE_KRB4 now = time(0); if ((__galaxy_list[i].last_authent_time == 0) || (now - __galaxy_list[i].last_authent_time > 120)) { result = krb_mk_req(&authent, SERVER_SERVICE, SERVER_INSTANCE, krealm, 0); if (result != MK_AP_OK) { __galaxy_list[i].last_authent_time = 0; return (result+krb_err_base); } __galaxy_list[i].last_authent_time = now; __galaxy_list[i].last_authent = authent; } else { authent = __galaxy_list[i].last_authent; } notice->z_auth = 1; notice->z_authent_len = authent.length; notice->z_ascii_authent = (char *)malloc((unsigned)authent.length*3); /* zero length authent is an error, so malloc(0) is not a problem */ if (!notice->z_ascii_authent) return (ENOMEM); if ((result = ZMakeAscii(notice->z_ascii_authent, authent.length*3, authent.dat, authent.length)) != ZERR_NONE) { free(notice->z_ascii_authent); return (result); } result = Z_FormatRawHeader(notice, buffer, buffer_len, phdr_len, &cksum_start, &cksum_len, &cstart, &cend); free(notice->z_ascii_authent); notice->z_authent_len = 0; if (result) return(result); /* Compute a checksum over the header and message. */ if ((result = krb_get_cred(SERVER_SERVICE, SERVER_INSTANCE, krealm, &cred)) != 0) return result; checksum = des_quad_cksum(cksum_start, NULL, cstart - cksum_start, 0, cred.session); checksum ^= des_quad_cksum(cend, NULL, (cksum_start + cksum_len) - cend, 0, cred.session); checksum ^= des_quad_cksum(notice->z_message, NULL, notice->z_message_len, 0, cred.session); notice->z_checksum = checksum; ZMakeAscii32(cstart, (buffer + buffer_len) - cstart, checksum); return (ZERR_NONE); #endif }