1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
|
$NetBSD: patch-ca,v 1.2 2009/12/01 08:49:46 manu Exp $
--- lasso/saml-2.0/profile.c.orig 2009-03-27 17:12:48.000000000 +0100
+++ lasso/saml-2.0/profile.c 2009-11-30 20:19:47.000000000 +0100
@@ -49,5 +49,5 @@
static char* lasso_saml20_profile_build_artifact(LassoProvider *provider);
static void remove_all_signatures(LassoNode *node);
-static char * lasso_saml20_profile_export_to_query(LassoProfile *profile, LassoNode *msg, int sign);
+static int lasso_saml20_profile_export_to_query(LassoProfile *profile, LassoNode *msg, int sign, char **query);
/*
@@ -1014,36 +1014,41 @@
* Return value: a newly allocated string containing the query string if successfull, NULL otherwise.
*/
-static char *
-lasso_saml20_profile_export_to_query(LassoProfile *profile, LassoNode *msg, int sign) {
+static int
+lasso_saml20_profile_export_to_query(LassoProfile *profile, LassoNode *msg, int sign, char **query) {
char *unsigned_query = NULL;
char *result = NULL;
+ int rc = 0;
- g_return_val_if_fail(LASSO_IS_NODE(msg), NULL);
+ lasso_bad_param(PROFILE, profile);
+ lasso_bad_param(NODE, msg);
unsigned_query = lasso_node_build_query(msg);
if (profile->msg_relayState) {
- char *query = unsigned_query;
- xmlChar *encoded_relayState;
- if (strlen(profile->msg_relayState) < 81) {
- encoded_relayState = xmlURIEscape((xmlChar*)profile->msg_relayState);
- if (encoded_relayState != NULL) {
- unsigned_query = g_strdup_printf("%s&RelayState=%s", query,
- (char*)encoded_relayState);
- lasso_release_string(query);
- lasso_release_xml_string(encoded_relayState);
- }
- } else {
- g_warning("Refused to encode a RelayState of more than 80 bytes, #3.4.3 of"
- " saml-bindings-2.0-os");
+ unsigned_query = lasso_url_add_parameters(unsigned_query, 1, "RelayState", profile->msg_relayState, NULL);
+
+ if (strlen(profile->msg_relayState) > 80) {
+ g_warning("Encoded a RelayState of more than 80 bytes, see #3.4.3 of saml-bindings-2.0-os");
}
}
if (sign && lasso_flag_add_signature) {
+ LassoServer *server = profile->server;
+ goto_cleanup_if_fail_with_rc (LASSO_IS_SERVER(server),
+ LASSO_PROFILE_ERROR_MISSING_SERVER);
+ goto_cleanup_if_fail_with_rc (
+ profile->server->signature_method != LASSO_SIGNATURE_TYPE_NONE &&
+ profile->server->private_key,
+ LASSO_DS_ERROR_PRIVATE_KEY_LOAD_FAILED);
+
result = lasso_query_sign(unsigned_query, profile->server->signature_method,
profile->server->private_key);
- lasso_release_string(unsigned_query);
+ lasso_transfer_string(*query, result);
} else {
- result = unsigned_query;
+ lasso_transfer_string(*query, unsigned_query);
}
- return result;
+cleanup:
+ lasso_release_string(unsigned_query);
+ lasso_release_string(result);
+ return rc;
+
}
@@ -1054,8 +1059,13 @@
if (node == NULL)
return;
+
klass = LASSO_NODE_GET_CLASS(node);
- if (klass->node_data->sign_type_offset != 0) {
- G_STRUCT_MEMBER(LassoSignatureType, node,klass->node_data->sign_type_offset) =
- LASSO_SIGNATURE_TYPE_NONE;
+ /* follow the class parenting chain */
+ while (klass && LASSO_IS_NODE_CLASS(klass)) {
+ if (klass && klass->node_data && klass->node_data->sign_type_offset != 0) {
+ G_STRUCT_MEMBER(LassoSignatureType, node, klass->node_data->sign_type_offset) =
+ LASSO_SIGNATURE_TYPE_NONE;
+ }
+ klass = g_type_class_peek_parent(klass);
}
}
@@ -1097,4 +1107,17 @@
}
+/**
+ * lasso_saml20_profile_build_http_redirect:
+ * @profile: a #LassoProfile object
+ * @msg: a #LassoNode object representing a SAML 2.0 message
+ * @must_sign: wheter to sign the query message using query signatures
+ * @url: the URL where the query is targeted
+ *
+ * Build an HTTP URL with a query-string following the SAML 2.0 HTTP-Redirect binding rules,
+ * eventually sign it. Any signature at the message level is removed.
+ *
+ * Return value: 0 if successful, an error code otherwise.
+ */
+
gint
lasso_saml20_profile_build_http_redirect(LassoProfile *profile,
@@ -1103,12 +1126,17 @@
const char *url)
{
- char *query;
+ char *query = NULL;
+ int rc = 0;
if (url == NULL) {
return critical_error(LASSO_PROFILE_ERROR_UNKNOWN_PROFILE_URL);
}
+ /* remove XML signature */
+ remove_signature(msg);
/* No signature on the XML message */
- remove_all_signatures(msg);
- query = lasso_saml20_profile_export_to_query(profile, msg, must_sign);
+ rc = lasso_saml20_profile_export_to_query(profile, msg, must_sign, &query);
+ if (rc)
+ return rc;
+
lasso_assign_new_string(profile->msg_url, lasso_concat_url_query(url, query));
lasso_release(profile->msg_body);
@@ -1317,4 +1345,5 @@
lasso_release(idx);
}
+ /* remove signature at the message level */
rc = lasso_saml20_profile_build_http_redirect(profile, msg, must_sign, url);
lasso_release(url);
|