1
/* $Id: session.c,v 1.24 2004/09/06 17:49:19 fpeters Exp $
3
* Lasso - A free implementation of the Liberty Alliance specifications.
5
* Copyright (C) 2004 Entr'ouvert
6
* http://lasso.entrouvert.org
8
* Authors: Nicolas Clapies <nclapies@entrouvert.com>
9
* Valery Febvre <vfebvre@easter-eggs.com>
11
* This program is free software; you can redistribute it and/or modify
12
* it under the terms of the GNU General Public License as published by
13
* the Free Software Foundation; either version 2 of the License, or
14
* (at your option) any later version.
16
* This program is distributed in the hope that it will be useful,
17
* but WITHOUT ANY WARRANTY; without even the implied warranty of
18
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19
* GNU General Public License for more details.
21
* You should have received a copy of the GNU General Public License
22
* along with this program; if not, write to the Free Software
23
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
26
#include <lasso/environs/session.h>
27
#include <lasso/lasso_config.h>
29
#define LASSO_SESSION_NODE "Session"
30
#define LASSO_SESSION_ASSERTIONS_NODE "Assertions"
31
#define LASSO_SESSION_ASSERTION_NODE "AuthnAssertion"
32
#define LASSO_SESSION_REMOTE_PROVIDERID_ATTR "RemoteProviderID"
34
struct _LassoSessionPrivate
36
gboolean dispose_has_run;
39
static GObjectClass *parent_class = NULL;
41
/*****************************************************************************/
42
/* private functions */
43
/*****************************************************************************/
46
lasso_session_copy_assertion(gpointer key,
50
g_hash_table_insert((GHashTable *)assertions, g_strdup((gchar *)key),
51
lasso_node_copy(LASSO_NODE(value)));
55
lasso_session_dump_assertion(gpointer key,
57
LassoNode *assertions)
59
LassoNode *assertion_node, *assertion_copy;
60
LassoNodeClass *assertion_class, *assertions_class;
62
/* new lasso assertion node */
63
assertion_node = lasso_node_new();
64
assertion_class = LASSO_NODE_GET_CLASS(assertion_node);
65
assertion_class->set_name(assertion_node, LASSO_SESSION_ASSERTION_NODE);
67
/* set the remote provider id */
68
assertion_class->set_prop(assertion_node, LASSO_SESSION_REMOTE_PROVIDERID_ATTR, key);
70
/* set assertion node */
71
assertion_copy = lasso_node_copy(LASSO_NODE(value));
72
assertion_class->add_child(assertion_node, assertion_copy, FALSE);
73
lasso_node_destroy(assertion_copy);
75
/* add lasso assertion node to lasso assertions node */
76
assertions_class = LASSO_NODE_GET_CLASS(assertions);
77
assertions_class->add_child(assertions, assertion_node, TRUE);
78
lasso_node_destroy(assertion_node);
81
/*****************************************************************************/
83
/*****************************************************************************/
86
lasso_session_add_assertion(LassoSession *session,
91
gboolean found = FALSE;
93
g_return_val_if_fail(session != NULL, -1);
94
g_return_val_if_fail(providerID != NULL, -2);
95
g_return_val_if_fail(assertion != NULL, -3);
97
/* add the remote provider id */
98
for(i = 0; i<session->providerIDs->len; i++) {
99
if(xmlStrEqual(providerID, g_ptr_array_index(session->providerIDs, i))) {
105
debug("An assertion existed already for this providerID, it was replaced by the new one.\n");
108
g_ptr_array_add(session->providerIDs, g_strdup(providerID));
111
/* add the assertion */
112
g_hash_table_insert(session->assertions, g_strdup(providerID),
113
lasso_node_copy(assertion));
115
session->is_dirty = TRUE;
121
lasso_session_copy(LassoSession *session)
126
if (session == NULL) {
130
copy = LASSO_SESSION(g_object_new(LASSO_TYPE_SESSION, NULL));
132
copy->providerIDs = g_ptr_array_new();
133
for(i=0; i<session->providerIDs->len; i++) {
134
g_ptr_array_add(copy->providerIDs,
135
g_strdup(g_ptr_array_index(session->providerIDs, i)));
137
copy->assertions = g_hash_table_new_full(g_str_hash, g_str_equal,
138
(GDestroyNotify)g_free,
139
(GDestroyNotify)lasso_node_destroy);
140
g_hash_table_foreach(session->assertions, (GHFunc)lasso_session_copy_assertion,
141
(gpointer)copy->assertions);
142
copy->is_dirty = session->is_dirty;
148
lasso_session_destroy(LassoSession *session)
150
if (LASSO_IS_SESSION(session)) {
151
g_object_unref(G_OBJECT(session));
156
lasso_session_dump(LassoSession *session)
158
LassoNode *session_node, *assertions_node;
159
LassoNodeClass *session_class, *assertions_class;
163
g_return_val_if_fail(session != NULL, NULL);
165
session_node = lasso_node_new();
166
session_class = LASSO_NODE_GET_CLASS(session_node);
167
session_class->set_name(session_node, LASSO_SESSION_NODE);
168
session_class->set_ns(session_node, lassoLassoHRef, NULL);
170
/* dump the assertions */
171
table_size = g_hash_table_size(session->assertions);
172
if (table_size > 0) {
173
assertions_node = lasso_node_new();
174
assertions_class = LASSO_NODE_GET_CLASS(assertions_node);
175
assertions_class->set_name(assertions_node, LASSO_SESSION_ASSERTIONS_NODE);
176
g_hash_table_foreach(session->assertions, (GHFunc)lasso_session_dump_assertion,
178
session_class->add_child(session_node, assertions_node, FALSE);
179
lasso_node_destroy(assertions_node);
182
/* Add lasso version in the xml node */
183
session_class->set_prop(LASSO_NODE(session_node), "version", PACKAGE_VERSION);
185
dump = lasso_node_export(session_node);
187
lasso_node_destroy(session_node);
193
lasso_session_get_assertion(LassoSession *session,
196
LassoNode *assertion;
198
g_return_val_if_fail(session != NULL, NULL);
199
g_return_val_if_fail(providerID != NULL, NULL);
201
assertion = (LassoNode *)g_hash_table_lookup(session->assertions,
203
if (assertion == NULL) {
207
return lasso_node_copy(assertion);
211
lasso_session_get_authentication_method(LassoSession *session,
212
gchar *remote_providerID)
214
LassoNode *assertion, *as;
215
gchar *providerID = remote_providerID;
216
gchar *authentication_method;
219
if (providerID == NULL) {
220
providerID = lasso_session_get_first_providerID(session);
222
assertion = lasso_session_get_assertion(session, providerID);
223
if (providerID == NULL) {
226
as = lasso_node_get_child(assertion, "AuthenticationStatement", NULL, NULL);
227
authentication_method = lasso_node_get_attr_value(as, "AuthenticationMethod", &err);
228
if (authentication_method == NULL) {
229
message(G_LOG_LEVEL_CRITICAL, err->message);
235
lasso_node_destroy(as);
236
lasso_node_destroy(assertion);
237
return authentication_method;
241
lasso_session_get_first_providerID(LassoSession *session)
245
g_return_val_if_fail(session != NULL, NULL);
247
if(session->providerIDs->len == 0) {
251
providerID = g_ptr_array_index(session->providerIDs, 0);
252
if (providerID == NULL) {
256
return g_strdup(providerID);
260
lasso_session_get_provider_index(LassoSession *session,
265
g_return_val_if_fail(session != NULL, NULL);
267
/* verify index is valid */
268
if ( (session->providerIDs == NULL) && (session->providerIDs->len < 0) ) {
271
if ( (index < 0) || (index >= session->providerIDs->len) ) {
275
/* get the provider id */
276
providerID = g_ptr_array_index(session->providerIDs, index);
277
if (providerID == NULL) {
281
return g_strdup(providerID);
285
lasso_session_remove_assertion(LassoSession *session,
288
LassoNode *assertion;
291
g_return_val_if_fail(session != NULL, -1);
292
g_return_val_if_fail(providerID != NULL, -1);
294
/* remove the assertion */
295
assertion = lasso_session_get_assertion(session, providerID);
296
if (assertion != NULL) {
297
g_hash_table_remove(session->assertions, providerID);
298
lasso_node_destroy(assertion);
301
/* remove the remote provider id */
302
for(i = 0; i<session->providerIDs->len; i++) {
303
if(xmlStrEqual(providerID, g_ptr_array_index(session->providerIDs, i))) {
304
g_ptr_array_remove_index(session->providerIDs, i);
309
session->is_dirty = TRUE;
314
/*****************************************************************************/
315
/* overrided parent class methods */
316
/*****************************************************************************/
319
lasso_session_dispose(LassoSession *session)
321
if (session->private->dispose_has_run == TRUE) {
324
session->private->dispose_has_run = TRUE;
326
debug("Session object 0x%x disposed ...\n", session);
328
g_hash_table_destroy(session->assertions);
329
session->assertions = NULL;
331
parent_class->dispose(G_OBJECT(session));
335
lasso_session_finalize(LassoSession *session)
339
debug("Session object 0x%x finalized ...\n", session);
341
/* free allocated memory for providerIDs array */
342
for (i=0; i<session->providerIDs->len; i++) {
343
g_free(session->providerIDs->pdata[i]);
344
session->providerIDs->pdata[i] = NULL;
346
g_ptr_array_free(session->providerIDs, TRUE);
347
session->providerIDs = NULL;
349
g_free(session->private);
350
session->private = NULL;
352
parent_class->finalize(G_OBJECT(session));
355
/*****************************************************************************/
356
/* instance and class init functions */
357
/*****************************************************************************/
360
lasso_session_instance_init(LassoSession *session)
362
session->private = g_new (LassoSessionPrivate, 1);
363
session->private->dispose_has_run = FALSE;
365
session->providerIDs = g_ptr_array_new();
366
session->assertions = g_hash_table_new_full(g_str_hash, g_str_equal,
367
(GDestroyNotify)g_free,
368
(GDestroyNotify)lasso_node_destroy);
369
session->is_dirty = TRUE;
373
lasso_session_class_init(LassoSessionClass *class)
375
GObjectClass *gobject_class = G_OBJECT_CLASS(class);
377
parent_class = g_type_class_peek_parent(class);
378
/* override parent class methods */
379
gobject_class->dispose = (void *)lasso_session_dispose;
380
gobject_class->finalize = (void *)lasso_session_finalize;
383
GType lasso_session_get_type() {
384
static GType this_type = 0;
387
static const GTypeInfo this_info = {
388
sizeof (LassoSessionClass),
391
(GClassInitFunc) lasso_session_class_init,
394
sizeof(LassoSession),
396
(GInstanceInitFunc) lasso_session_instance_init,
399
this_type = g_type_register_static(G_TYPE_OBJECT,
409
LassoSession *session;
411
session = LASSO_SESSION(g_object_new(LASSO_TYPE_SESSION, NULL));
417
lasso_session_new_from_dump(gchar *dump)
419
LassoSession *session;
420
LassoNode *session_node;
421
LassoNode *assertions_node, *assertion_node, *assertion;
422
xmlNodePtr assertions_xmlNode, assertion_xmlNode;
426
g_return_val_if_fail(dump != NULL, NULL);
428
session = LASSO_SESSION(g_object_new(LASSO_TYPE_SESSION, NULL));
431
session_node = lasso_node_new_from_dump(dump);
432
if (session_node == NULL) {
433
message(G_LOG_LEVEL_WARNING, "Can't create a session from dump\n");
438
assertions_node = lasso_node_get_child(session_node,
439
LASSO_SESSION_ASSERTIONS_NODE,
440
lassoLassoHRef, NULL);
441
if (assertions_node != NULL) {
442
assertions_xmlNode = LASSO_NODE_GET_CLASS(assertions_node)->get_xmlNode(assertions_node);
443
assertion_xmlNode = assertions_xmlNode->children;
445
while (assertion_xmlNode != NULL) {
446
/* assertion xmlNode */
447
if (assertion_xmlNode->type == XML_ELEMENT_NODE && \
448
xmlStrEqual(assertion_xmlNode->name, LASSO_SESSION_ASSERTION_NODE)) {
450
assertion_node = lasso_node_new_from_xmlNode(assertion_xmlNode);
451
providerID = lasso_node_get_attr_value(assertion_node,
452
LASSO_SESSION_REMOTE_PROVIDERID_ATTR,
454
if (providerID != NULL) {
455
assertion = lasso_node_get_child(assertion_node,
457
NULL, /* lassoLibHRef, FIXME changed for SourceID */
459
if (assertion != NULL) {
460
lasso_session_add_assertion(session, providerID, assertion);
461
lasso_node_destroy(assertion);
464
message(G_LOG_LEVEL_CRITICAL, err->message);
469
message(G_LOG_LEVEL_CRITICAL, err->message);
473
lasso_node_destroy(assertion_node);
475
assertion_xmlNode = assertion_xmlNode->next;
478
lasso_node_destroy(assertions_node);
479
lasso_node_destroy(session_node);