mirror of
https://github.com/matrix-construct/construct
synced 2025-01-16 09:36:54 +01:00
authd: use a list for auth_providers
We only need to iterate over this small fixed size list, so dictionary iteration will be less efficient.
This commit is contained in:
parent
f4d828ef96
commit
d955cd9f97
2 changed files with 43 additions and 24 deletions
|
@ -48,6 +48,7 @@
|
||||||
|
|
||||||
#include "stdinc.h"
|
#include "stdinc.h"
|
||||||
#include "rb_dictionary.h"
|
#include "rb_dictionary.h"
|
||||||
|
#include "rb_lib.h"
|
||||||
#include "authd.h"
|
#include "authd.h"
|
||||||
#include "provider.h"
|
#include "provider.h"
|
||||||
#include "notice.h"
|
#include "notice.h"
|
||||||
|
@ -55,7 +56,7 @@
|
||||||
static EVH provider_timeout_event;
|
static EVH provider_timeout_event;
|
||||||
|
|
||||||
rb_dictionary *auth_clients;
|
rb_dictionary *auth_clients;
|
||||||
rb_dictionary *auth_providers; /* Referenced by name */
|
rb_dlink_list auth_providers;
|
||||||
|
|
||||||
static rb_dlink_list free_pids;
|
static rb_dlink_list free_pids;
|
||||||
static uint32_t pid;
|
static uint32_t pid;
|
||||||
|
@ -66,7 +67,6 @@ void
|
||||||
init_providers(void)
|
init_providers(void)
|
||||||
{
|
{
|
||||||
auth_clients = rb_dictionary_create("pending auth clients", rb_uint32cmp);
|
auth_clients = rb_dictionary_create("pending auth clients", rb_uint32cmp);
|
||||||
auth_providers = rb_dictionary_create("auth providers", strcmp);
|
|
||||||
timeout_ev = rb_event_addish("provider_timeout_event", provider_timeout_event, NULL, 1);
|
timeout_ev = rb_event_addish("provider_timeout_event", provider_timeout_event, NULL, 1);
|
||||||
|
|
||||||
load_provider(&rdns_provider);
|
load_provider(&rdns_provider);
|
||||||
|
@ -79,7 +79,7 @@ init_providers(void)
|
||||||
void
|
void
|
||||||
destroy_providers(void)
|
destroy_providers(void)
|
||||||
{
|
{
|
||||||
rb_dlink_node *ptr;
|
rb_dlink_node *ptr, *nptr;
|
||||||
rb_dictionary_iter iter;
|
rb_dictionary_iter iter;
|
||||||
struct auth_client *auth;
|
struct auth_client *auth;
|
||||||
struct auth_provider *provider;
|
struct auth_provider *provider;
|
||||||
|
@ -92,14 +92,17 @@ destroy_providers(void)
|
||||||
"Authentication system is down... try reconnecting in a few seconds");
|
"Authentication system is down... try reconnecting in a few seconds");
|
||||||
}
|
}
|
||||||
|
|
||||||
RB_DICTIONARY_FOREACH(provider, &iter, auth_providers)
|
RB_DLINK_FOREACH_SAFE(ptr, nptr, auth_providers.head)
|
||||||
{
|
{
|
||||||
|
struct auth_provider *provider = ptr->data;
|
||||||
|
|
||||||
if(provider->destroy)
|
if(provider->destroy)
|
||||||
provider->destroy();
|
provider->destroy();
|
||||||
|
|
||||||
|
rb_dlinkDelete(ptr, &auth_providers);
|
||||||
}
|
}
|
||||||
|
|
||||||
rb_dictionary_destroy(auth_clients, NULL, NULL);
|
rb_dictionary_destroy(auth_clients, NULL, NULL);
|
||||||
rb_dictionary_destroy(auth_providers, NULL, NULL);
|
|
||||||
rb_event_delete(timeout_ev);
|
rb_event_delete(timeout_ev);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -141,7 +144,7 @@ load_provider(struct auth_provider *provider)
|
||||||
if(provider->init != NULL)
|
if(provider->init != NULL)
|
||||||
provider->init();
|
provider->init();
|
||||||
|
|
||||||
rb_dictionary_add(auth_providers, provider->name, provider);
|
rb_dlinkAdd(provider, &provider->node, &auth_providers);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
@ -161,7 +164,7 @@ unload_provider(struct auth_provider *provider)
|
||||||
if(provider->destroy != NULL)
|
if(provider->destroy != NULL)
|
||||||
provider->destroy();
|
provider->destroy();
|
||||||
|
|
||||||
rb_dictionary_delete(auth_providers, provider->name);
|
rb_dlinkDelete(&provider->node, &auth_providers);
|
||||||
|
|
||||||
/* Reclaim ID */
|
/* Reclaim ID */
|
||||||
rb_dlinkAddAlloc(RB_UINT_TO_POINTER(provider->id), &free_pids);
|
rb_dlinkAddAlloc(RB_UINT_TO_POINTER(provider->id), &free_pids);
|
||||||
|
@ -181,11 +184,12 @@ cancel_providers(struct auth_client *auth)
|
||||||
{
|
{
|
||||||
if(auth->refcount > 0)
|
if(auth->refcount > 0)
|
||||||
{
|
{
|
||||||
rb_dictionary_iter iter;
|
rb_dlink_node *ptr;
|
||||||
struct auth_provider *provider;
|
|
||||||
|
|
||||||
RB_DICTIONARY_FOREACH(provider, &iter, auth_providers)
|
RB_DLINK_FOREACH(ptr, auth_providers.head)
|
||||||
{
|
{
|
||||||
|
struct auth_provider *provider = ptr->data;
|
||||||
|
|
||||||
if(provider->cancel != NULL && is_provider_running(auth, provider->id))
|
if(provider->cancel != NULL && is_provider_running(auth, provider->id))
|
||||||
/* Cancel if required */
|
/* Cancel if required */
|
||||||
provider->cancel(auth);
|
provider->cancel(auth);
|
||||||
|
@ -198,8 +202,7 @@ cancel_providers(struct auth_client *auth)
|
||||||
void
|
void
|
||||||
provider_done(struct auth_client *auth, uint32_t id)
|
provider_done(struct auth_client *auth, uint32_t id)
|
||||||
{
|
{
|
||||||
rb_dictionary_iter iter;
|
rb_dlink_node *ptr;
|
||||||
struct auth_provider *provider;
|
|
||||||
|
|
||||||
lrb_assert(is_provider_running(auth, id));
|
lrb_assert(is_provider_running(auth, id));
|
||||||
lrb_assert(id != UINT32_MAX);
|
lrb_assert(id != UINT32_MAX);
|
||||||
|
@ -214,8 +217,10 @@ provider_done(struct auth_client *auth, uint32_t id)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
RB_DICTIONARY_FOREACH(provider, &iter, auth_providers)
|
RB_DLINK_FOREACH(ptr, auth_providers.head)
|
||||||
{
|
{
|
||||||
|
struct auth_provider *provider = ptr->data;
|
||||||
|
|
||||||
if(provider->completed != NULL && is_provider_running(auth, provider->id))
|
if(provider->completed != NULL && is_provider_running(auth, provider->id))
|
||||||
/* Notify pending clients who asked for it */
|
/* Notify pending clients who asked for it */
|
||||||
provider->completed(auth, id);
|
provider->completed(auth, id);
|
||||||
|
@ -269,10 +274,9 @@ accept_client(struct auth_client *auth, uint32_t id)
|
||||||
static void
|
static void
|
||||||
start_auth(const char *cid, const char *l_ip, const char *l_port, const char *c_ip, const char *c_port)
|
start_auth(const char *cid, const char *l_ip, const char *l_port, const char *c_ip, const char *c_port)
|
||||||
{
|
{
|
||||||
struct auth_provider *provider;
|
|
||||||
struct auth_client *auth = rb_malloc(sizeof(struct auth_client));
|
struct auth_client *auth = rb_malloc(sizeof(struct auth_client));
|
||||||
long lcid = strtol(cid, NULL, 16);
|
long lcid = strtol(cid, NULL, 16);
|
||||||
rb_dictionary_iter iter;
|
rb_dlink_node *ptr;
|
||||||
|
|
||||||
if(lcid >= UINT32_MAX)
|
if(lcid >= UINT32_MAX)
|
||||||
return;
|
return;
|
||||||
|
@ -300,12 +304,17 @@ start_auth(const char *cid, const char *l_ip, const char *l_port, const char *c_
|
||||||
rb_strlcpy(auth->hostname, "*", sizeof(auth->hostname));
|
rb_strlcpy(auth->hostname, "*", sizeof(auth->hostname));
|
||||||
rb_strlcpy(auth->username, "*", sizeof(auth->username));
|
rb_strlcpy(auth->username, "*", sizeof(auth->username));
|
||||||
|
|
||||||
auth->data = rb_malloc(rb_dictionary_size(auth_providers) *
|
/* FIXME this is unsafe, the list being allocated needs to
|
||||||
|
* support the maximum PID allocated, not just the current
|
||||||
|
* length of auth_providers */
|
||||||
|
auth->data = rb_malloc(rb_dlink_list_length(&auth_providers) *
|
||||||
sizeof(struct auth_client_data));
|
sizeof(struct auth_client_data));
|
||||||
|
|
||||||
auth->providers_starting = true;
|
auth->providers_starting = true;
|
||||||
RB_DICTIONARY_FOREACH(provider, &iter, auth_providers)
|
RB_DLINK_FOREACH(ptr, auth_providers.head)
|
||||||
{
|
{
|
||||||
|
struct auth_provider *provider = ptr->data;
|
||||||
|
|
||||||
auth->data[provider->id].provider = provider;
|
auth->data[provider->id].provider = provider;
|
||||||
|
|
||||||
lrb_assert(provider->start != NULL);
|
lrb_assert(provider->start != NULL);
|
||||||
|
@ -372,11 +381,11 @@ provider_timeout_event(void *notused __unused)
|
||||||
|
|
||||||
RB_DICTIONARY_FOREACH(auth, &iter, auth_clients)
|
RB_DICTIONARY_FOREACH(auth, &iter, auth_clients)
|
||||||
{
|
{
|
||||||
rb_dictionary_iter iter2;
|
rb_dlink_node *ptr;
|
||||||
struct auth_provider *provider;
|
|
||||||
|
|
||||||
RB_DICTIONARY_FOREACH(provider, &iter2, auth_providers)
|
RB_DLINK_FOREACH(ptr, auth_providers.head)
|
||||||
{
|
{
|
||||||
|
struct auth_provider *provider = ptr->data;
|
||||||
const time_t timeout = get_provider_timeout(auth, provider->id);
|
const time_t timeout = get_provider_timeout(auth, provider->id);
|
||||||
|
|
||||||
if(is_provider_running(auth, provider->id) && provider->timeout != NULL &&
|
if(is_provider_running(auth, provider->id) && provider->timeout != NULL &&
|
||||||
|
|
|
@ -104,7 +104,7 @@ extern struct auth_provider ident_provider;
|
||||||
extern struct auth_provider blacklist_provider;
|
extern struct auth_provider blacklist_provider;
|
||||||
extern struct auth_provider opm_provider;
|
extern struct auth_provider opm_provider;
|
||||||
|
|
||||||
extern rb_dictionary *auth_providers;
|
extern rb_dlink_list auth_providers;
|
||||||
extern rb_dictionary *auth_clients;
|
extern rb_dictionary *auth_clients;
|
||||||
|
|
||||||
void load_provider(struct auth_provider *provider);
|
void load_provider(struct auth_provider *provider);
|
||||||
|
@ -138,16 +138,26 @@ auth_client_unref(struct auth_client *auth)
|
||||||
|
|
||||||
/* Get a provider by name */
|
/* Get a provider by name */
|
||||||
static inline struct auth_provider *
|
static inline struct auth_provider *
|
||||||
get_provider(const char *name)
|
find_provider(const char *name)
|
||||||
{
|
{
|
||||||
return rb_dictionary_retrieve(auth_providers, name);
|
rb_dlink_node *ptr;
|
||||||
|
|
||||||
|
RB_DLINK_FOREACH(ptr, auth_providers.head)
|
||||||
|
{
|
||||||
|
struct auth_provider *provider = ptr->data;
|
||||||
|
|
||||||
|
if(strcasecmp(provider->name, name) == 0)
|
||||||
|
return provider;
|
||||||
|
}
|
||||||
|
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Get a provider's id by name */
|
/* Get a provider's id by name */
|
||||||
static inline bool
|
static inline bool
|
||||||
get_provider_id(const char *name, uint32_t *id)
|
get_provider_id(const char *name, uint32_t *id)
|
||||||
{
|
{
|
||||||
struct auth_provider *provider = get_provider(name);
|
struct auth_provider *provider = find_provider(name);
|
||||||
|
|
||||||
if(provider != NULL)
|
if(provider != NULL)
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue