Move more config file code from krb5 to base

This commit is contained in:
Nicolas Williams
2020-03-02 22:26:17 -06:00
parent 5eade227ce
commit 78a21fdd95
4 changed files with 219 additions and 159 deletions

View File

@@ -90,7 +90,7 @@ libheimbase_OBJS = \
libheimbase_gen_OBJS= $(OBJ)\heim_err.obj libheimbase_gen_OBJS= $(OBJ)\heim_err.obj
$(LIBHEIMBASE): $(libheimbase_OBJS) $(libheimbase_gen_OBJS) $(LIBHEIMBASE): $(libheimbase_OBJS) $(libheimbase_gen_OBJS)
$(LIBCON_C) -OUT:$@ $(LIBROKEN) $(PTHREAD_LIB) Secur32.lib Shell32.lib Advapi32.lib @<< $(LIBCON_C) -OUT:$@ $(LIBROKEN) $(PTHREAD_LIB) Secur32.lib Shell32.lib Advapi32.lib Shlwapi.lib @<<
$(libheimbase_OBJS: = $(libheimbase_OBJS: =
) )
$(libheimbase_gen_OBJS: = $(libheimbase_gen_OBJS: =

View File

@@ -232,3 +232,207 @@ heim_set_debug_dest(heim_context context, heim_log_facility *fac)
context->debug_dest = fac; context->debug_dest = fac;
return 0; return 0;
} }
#define PATH_SEP ":"
static heim_error_code
add_file(char ***pfilenames, int *len, char *file)
{
char **pp = *pfilenames;
int i;
for(i = 0; i < *len; i++) {
if(strcmp(pp[i], file) == 0) {
free(file);
return 0;
}
}
pp = realloc(*pfilenames, (*len + 2) * sizeof(*pp));
if (pp == NULL) {
free(file);
return ENOMEM;
}
pp[*len] = file;
pp[*len + 1] = NULL;
*pfilenames = pp;
*len += 1;
return 0;
}
#ifdef WIN32
static char *
get_default_config_config_files_from_registry(void)
{
static const char *KeyName = "Software\\Heimdal"; /* XXX #define this */
char *config_file = NULL;
LONG rcode;
HKEY key;
rcode = RegOpenKeyEx(HKEY_CURRENT_USER, KeyName, 0, KEY_READ, &key);
if (rcode == ERROR_SUCCESS) {
config_file = heim_parse_reg_value_as_multi_string(NULL, key, "config",
REG_NONE, 0, PATH_SEP);
RegCloseKey(key);
}
if (config_file)
return config_file;
rcode = RegOpenKeyEx(HKEY_LOCAL_MACHINE, KeyName, 0, KEY_READ, &key);
if (rcode == ERROR_SUCCESS) {
config_file = heim_parse_reg_value_as_multi_string(NULL, key, "config",
REG_NONE, 0, PATH_SEP);
RegCloseKey(key);
}
return config_file;
}
#endif
heim_error_code
heim_prepend_config_files(const char *filelist,
char **pq,
char ***ret_pp)
{
heim_error_code ret;
const char *p, *q;
char **pp;
int len;
char *fn;
pp = NULL;
len = 0;
p = filelist;
while(1) {
ssize_t l;
q = p;
l = strsep_copy(&q, PATH_SEP, NULL, 0);
if(l == -1)
break;
fn = malloc(l + 1);
if(fn == NULL) {
heim_free_config_files(pp);
return ENOMEM;
}
(void) strsep_copy(&p, PATH_SEP, fn, l + 1);
ret = add_file(&pp, &len, fn);
if (ret) {
heim_free_config_files(pp);
return ret;
}
}
if (pq != NULL) {
int i;
for (i = 0; pq[i] != NULL; i++) {
fn = strdup(pq[i]);
if (fn == NULL) {
heim_free_config_files(pp);
return ENOMEM;
}
ret = add_file(&pp, &len, fn);
if (ret) {
heim_free_config_files(pp);
return ret;
}
}
}
*ret_pp = pp;
return 0;
}
heim_error_code
heim_prepend_config_files_default(const char *prepend,
const char *def,
const char *envvar,
char ***pfilenames)
{
heim_error_code ret;
char **defpp, **pp = NULL;
ret = heim_get_default_config_files(def, envvar, &defpp);
if (ret)
return ret;
ret = heim_prepend_config_files(prepend, defpp, &pp);
heim_free_config_files(defpp);
if (ret) {
return ret;
}
*pfilenames = pp;
return 0;
}
heim_error_code
heim_get_default_config_files(const char *def,
const char *envvar,
char ***pfilenames)
{
const char *files = NULL;
files = secure_getenv(envvar);
#ifdef _WIN32
if (files == NULL) {
char * reg_files;
reg_files = get_default_config_config_files_from_registry();
if (reg_files != NULL) {
heim_error_code code;
code = heim_prepend_config_files(reg_files, NULL, pfilenames);
free(reg_files);
return code;
}
}
#endif
if (files == NULL)
files = def;
return heim_prepend_config_files(files, NULL, pfilenames);
}
#ifdef _WIN32
#define REGPATH_KERBEROS "SOFTWARE\\Kerberos"
#define REGPATH_HEIMDAL "SOFTWARE\\Heimdal"
#endif
heim_error_code
heim_set_config_files(heim_context context, char **filenames,
heim_config_binding **res)
{
heim_error_code ret;
*res = NULL;
while (filenames != NULL && *filenames != NULL && **filenames != '\0') {
ret = heim_config_parse_file_multi(context, *filenames, res);
if (ret != 0 && ret != ENOENT && ret != EACCES && ret != EPERM
&& ret != HEIM_ERR_CONFIG_BADFORMAT) {
heim_config_file_free(context, *res);
*res = NULL;
return ret;
}
filenames++;
}
#ifdef _WIN32
heim_load_config_from_registry(context, REGPATH_KERBEROS,
REGPATH_HEIMDAL, res);
#endif
return ret;
}
void
heim_free_config_files(char **filenames)
{
char **p;
for (p = filenames; p && *p != NULL; p++)
free(*p);
free(filenames);
}

View File

@@ -1,4 +1,3 @@
HEIMDAL_BASE_1.0 { HEIMDAL_BASE_1.0 {
global: global:
_bsearch_file; _bsearch_file;
@@ -117,8 +116,10 @@ HEIMDAL_BASE_1.0 {
heim_error_get_code; heim_error_get_code;
heim_expand_path_tokens; heim_expand_path_tokens;
heim_expand_path_tokensv; heim_expand_path_tokensv;
heim_free_config_files;
heim_free_error_message; heim_free_error_message;
heim_get_debug_dest; heim_get_debug_dest;
heim_get_default_config_files;
heim_get_error_message; heim_get_error_message;
heim_get_hash; heim_get_hash;
_heim_get_isa; _heim_get_isa;
@@ -149,9 +150,12 @@ HEIMDAL_BASE_1.0 {
heim_path_vget; heim_path_vget;
heim_plugin_register; heim_plugin_register;
heim_plugin_run_f; heim_plugin_run_f;
heim_prepend_config_files;
heim_prepend_config_files_default;
heim_prepend_error_message; heim_prepend_error_message;
heim_release; heim_release;
heim_retain; heim_retain;
heim_set_config_files;
heim_set_debug_dest; heim_set_debug_dest;
heim_set_error_message; heim_set_error_message;
heim_set_warn_dest; heim_set_warn_dest;

View File

@@ -712,11 +712,6 @@ krb5_free_context(krb5_context context)
free(context); free(context);
} }
#ifdef _WIN32
#define REGPATH_KERBEROS "SOFTWARE\\Kerberos"
#define REGPATH_HEIMDAL "SOFTWARE\\Heimdal"
#endif
/** /**
* Reinit the context from a new set of filenames. * Reinit the context from a new set of filenames.
* *
@@ -733,30 +728,9 @@ KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
krb5_set_config_files(krb5_context context, char **filenames) krb5_set_config_files(krb5_context context, char **filenames)
{ {
krb5_error_code ret; krb5_error_code ret;
krb5_config_binding *tmp = NULL;
while(filenames != NULL && *filenames != NULL && **filenames != '\0') {
ret = krb5_config_parse_file_multi(context, *filenames, &tmp);
if (ret != 0 && ret != ENOENT && ret != EACCES && ret != EPERM
&& ret != KRB5_CONFIG_BADFORMAT) {
krb5_config_file_free(context, tmp);
return ret;
}
filenames++;
}
#if 0
/* with this enabled and if there are no config files, Kerberos is
considererd disabled */
if(tmp == NULL)
return ENXIO;
#endif
#ifdef _WIN32 if ((ret = heim_set_config_files(context->hcontext, filenames,
heim_load_config_from_registry(context->hcontext, REGPATH_KERBEROS, &context->cf)) == 0)
REGPATH_HEIMDAL, &tmp);
#endif
krb5_config_file_free(context, context->cf);
context->cf = tmp;
ret = init_context_from_config_file(context); ret = init_context_from_config_file(context);
return ret; return ret;
} }
@@ -830,54 +804,7 @@ add_file(char ***pfilenames, int *len, char *file)
KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
krb5_prepend_config_files(const char *filelist, char **pq, char ***ret_pp) krb5_prepend_config_files(const char *filelist, char **pq, char ***ret_pp)
{ {
krb5_error_code ret; return heim_prepend_config_files(filelist, pq, ret_pp);
const char *p, *q;
char **pp;
int len;
char *fn;
pp = NULL;
len = 0;
p = filelist;
while(1) {
ssize_t l;
q = p;
l = strsep_copy(&q, PATH_SEP, NULL, 0);
if(l == -1)
break;
fn = malloc(l + 1);
if(fn == NULL) {
krb5_free_config_files(pp);
return ENOMEM;
}
(void)strsep_copy(&p, PATH_SEP, fn, l + 1);
ret = add_file(&pp, &len, fn);
if (ret) {
krb5_free_config_files(pp);
return ret;
}
}
if (pq != NULL) {
int i;
for (i = 0; pq[i] != NULL; i++) {
fn = strdup(pq[i]);
if (fn == NULL) {
krb5_free_config_files(pp);
return ENOMEM;
}
ret = add_file(&pp, &len, fn);
if (ret) {
krb5_free_config_files(pp);
return ret;
}
}
}
*ret_pp = pp;
return 0;
} }
/** /**
@@ -895,61 +822,10 @@ krb5_prepend_config_files(const char *filelist, char **pq, char ***ret_pp)
KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
krb5_prepend_config_files_default(const char *filelist, char ***pfilenames) krb5_prepend_config_files_default(const char *filelist, char ***pfilenames)
{ {
krb5_error_code ret; return heim_prepend_config_files_default(filelist, krb5_config_file,
char **defpp, **pp = NULL; "KRB5_CONFIG", pfilenames);
ret = krb5_get_default_config_files(&defpp);
if (ret)
return ret;
ret = krb5_prepend_config_files(filelist, defpp, &pp);
krb5_free_config_files(defpp);
if (ret) {
return ret;
}
*pfilenames = pp;
return 0;
} }
#ifdef _WIN32
/**
* Checks the registry for configuration file location
*
* Kerberos for Windows and other legacy Kerberos applications expect
* to find the configuration file location in the
* SOFTWARE\MIT\Kerberos registry key under the value "config".
*/
KRB5_LIB_FUNCTION char * KRB5_LIB_CALL
_krb5_get_default_config_config_files_from_registry()
{
static const char * KeyName = "Software\\MIT\\Kerberos";
char *config_file = NULL;
LONG rcode;
HKEY key;
rcode = RegOpenKeyEx(HKEY_CURRENT_USER, KeyName, 0, KEY_READ, &key);
if (rcode == ERROR_SUCCESS) {
config_file = heim_parse_reg_value_as_multi_string(NULL, key, "config",
REG_NONE, 0, PATH_SEP);
RegCloseKey(key);
}
if (config_file)
return config_file;
rcode = RegOpenKeyEx(HKEY_LOCAL_MACHINE, KeyName, 0, KEY_READ, &key);
if (rcode == ERROR_SUCCESS) {
config_file = heim_parse_reg_value_as_multi_string(NULL, key, "config",
REG_NONE, 0, PATH_SEP);
RegCloseKey(key);
}
return config_file;
}
#endif
/** /**
* Get the global configuration list. * Get the global configuration list.
* *
@@ -964,31 +840,10 @@ _krb5_get_default_config_config_files_from_registry()
KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL KRB5_LIB_FUNCTION krb5_error_code KRB5_LIB_CALL
krb5_get_default_config_files(char ***pfilenames) krb5_get_default_config_files(char ***pfilenames)
{ {
const char *files = NULL;
if (pfilenames == NULL) if (pfilenames == NULL)
return EINVAL; return EINVAL;
files = secure_getenv("KRB5_CONFIG"); return heim_get_default_config_files(krb5_config_file, "KRB5_CONFIG",
pfilenames);
#ifdef _WIN32
if (files == NULL) {
char * reg_files;
reg_files = _krb5_get_default_config_config_files_from_registry();
if (reg_files != NULL) {
krb5_error_code code;
code = krb5_prepend_config_files(reg_files, NULL, pfilenames);
free(reg_files);
return code;
}
}
#endif
if (files == NULL)
files = krb5_config_file;
return krb5_prepend_config_files(files, NULL, pfilenames);
} }
/** /**
@@ -1006,10 +861,7 @@ krb5_get_default_config_files(char ***pfilenames)
KRB5_LIB_FUNCTION void KRB5_LIB_CALL KRB5_LIB_FUNCTION void KRB5_LIB_CALL
krb5_free_config_files(char **filenames) krb5_free_config_files(char **filenames)
{ {
char **p; heim_free_config_files(filenames);
for(p = filenames; p && *p != NULL; p++)
free(*p);
free(filenames);
} }
/** /**