GNUnet 0.22.2
plugin_reclaim_credential_jwt.c
Go to the documentation of this file.
1/*
2 This file is part of GNUnet
3 Copyright (C) 2013, 2014, 2016 GNUnet e.V.
4
5 GNUnet is free software: you can redistribute it and/or modify it
6 under the terms of the GNU Affero General Public License as published
7 by the Free Software Foundation, either version 3 of the License,
8 or (at your option) any later version.
9
10 GNUnet is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Affero General Public License for more details.
14
15 You should have received a copy of the GNU Affero General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>.
17
18 SPDX-License-Identifier: AGPL3.0-or-later
19 */
20
28#include "platform.h"
29#include "gnunet_util_lib.h"
31#include <inttypes.h>
32#include <jansson.h>
33
43static char *
45 uint32_t type,
46 const void *data,
47 size_t data_size)
48{
49 switch (type)
50 {
53
54 default:
55 return NULL;
56 }
57}
58
59
71static int
73 uint32_t type,
74 const char *s,
75 void **data,
76 size_t *data_size)
77{
78 if (NULL == s)
79 return GNUNET_SYSERR;
80 switch (type)
81 {
83 *data = GNUNET_strdup (s);
84 *data_size = strlen (s) + 1;
85 return GNUNET_OK;
86
87 default:
88 return GNUNET_SYSERR;
89 }
90}
91
92
97static struct
98{
99 const char *name;
100 uint32_t number;
102 { NULL, UINT32_MAX } };
103
111static uint32_t
112jwt_typename_to_number (void *cls, const char *jwt_typename)
113{
114 unsigned int i;
115
116 i = 0;
117 while ((NULL != jwt_cred_name_map[i].name) &&
118 (0 != strcasecmp (jwt_typename, jwt_cred_name_map[i].name)))
119 i++;
120 return jwt_cred_name_map[i].number;
121}
122
123
131static const char *
132jwt_number_to_typename (void *cls, uint32_t type)
133{
134 unsigned int i;
135
136 i = 0;
137 while ((NULL != jwt_cred_name_map[i].name) && (type !=
139 number))
140 i++;
141 return jwt_cred_name_map[i].name;
142}
143
144
152static struct GNUNET_RECLAIM_AttributeList *
154 const char *data,
155 size_t data_size)
156{
157 struct GNUNET_RECLAIM_AttributeList *attrs;
158 const char *jwt_body;
159 char delim[] = ".";
160 char *jwt_string;
161 char *val_str = NULL;
162 char *decoded_jwt;
163 char *tmp;
164 json_t *json_val;
165 json_error_t json_err;
166
168
169 jwt_string = GNUNET_strndup (data, data_size);
170 jwt_body = strtok (jwt_string, delim);
171 if (NULL == jwt_body)
172 {
174 "Failed to parse JSON %s\n", jwt_string);
175 return attrs;
176 }
177 jwt_body = strtok (NULL, delim);
178 if (NULL == jwt_body)
179 {
181 "Failed to parse JSON %s\n", jwt_string);
182 GNUNET_free (jwt_string);
183 return attrs;
184 }
185 GNUNET_STRINGS_base64url_decode (jwt_body, strlen (jwt_body),
186 (void **) &decoded_jwt);
187 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Decoded JWT: %s\n", decoded_jwt);
188 GNUNET_assert (NULL != decoded_jwt);
189 json_val = json_loads (decoded_jwt, JSON_DECODE_ANY, &json_err);
190 GNUNET_free (decoded_jwt);
191 {
192 const char *key;
193 const char *addr_key;
194 json_t *value;
195 json_t *addr_value;
196
197 json_object_foreach (json_val, key, value) {
198 if (0 == strcmp ("iss", key))
199 continue;
200 if (0 == strcmp ("jti", key))
201 continue;
202 if (0 == strcmp ("exp", key))
203 continue;
204 if (0 == strcmp ("iat", key))
205 continue;
206 if (0 == strcmp ("nbf", key))
207 continue;
208 if (0 == strcmp ("aud", key))
209 continue;
210 if (0 == strcmp ("address", key))
211 {
212 if (! json_is_object (value))
213 {
215 "address claim in wrong format!");
216 continue;
217 }
218 json_object_foreach (value, addr_key, addr_value) {
219 val_str = json_dumps (addr_value, JSON_ENCODE_ANY);
220 tmp = val_str;
221 // Remove leading " from jasson conversion
222 if (tmp[0] == '"')
223 tmp++;
224 // Remove trailing " from jansson conversion
225 if (tmp[strlen (tmp) - 1] == '"')
226 tmp[strlen (tmp) - 1] = '\0';
228 addr_key,
229 NULL,
231 tmp,
232 strlen (val_str));
233 GNUNET_free (val_str);
234 }
235 continue;
236 }
237 val_str = json_dumps (value, JSON_ENCODE_ANY);
238 tmp = val_str;
239 // Remove leading " from jasson conversion
240 if (tmp[0] == '"')
241 tmp++;
242 // Remove trailing " from jansson conversion
243 if (tmp[strlen (tmp) - 1] == '"')
244 tmp[strlen (tmp) - 1] = '\0';
246 key,
247 NULL,
249 tmp,
250 strlen (val_str));
251 GNUNET_free (val_str);
252 }
253 }
254 json_decref (json_val);
255 GNUNET_free (jwt_string);
256 return attrs;
257}
258
259
267static struct GNUNET_RECLAIM_AttributeList *
269 const struct GNUNET_RECLAIM_Credential *cred)
270{
272 return NULL;
273 return jwt_parse_attributes (cls, cred->data, cred->data_size);
274}
275
276
284static struct GNUNET_RECLAIM_AttributeList *
286 const struct GNUNET_RECLAIM_Presentation *cred)
287{
289 return NULL;
290 return jwt_parse_attributes (cls, cred->data, cred->data_size);
291}
292
293
301static char *
302jwt_get_issuer (void *cls,
303 const char *data,
304 size_t data_size)
305{
306 const char *jwt_body;
307 char *jwt_string;
308 char delim[] = ".";
309 char *issuer = NULL;
310 char *decoded_jwt;
311 json_t *issuer_json;
312 json_t *json_val;
313 json_error_t json_err;
314
315 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Parsing JWT attributes.\n");
316 jwt_string = GNUNET_strndup (data, data_size);
317 jwt_body = strtok (jwt_string, delim);
318 jwt_body = strtok (NULL, delim);
319 GNUNET_STRINGS_base64url_decode (jwt_body, strlen (jwt_body),
320 (void **) &decoded_jwt);
321 json_val = json_loads (decoded_jwt, JSON_DECODE_ANY, &json_err);
322 GNUNET_free (decoded_jwt);
323 GNUNET_free (jwt_string);
324 if (NULL == json_val)
325 return NULL;
326 issuer_json = json_object_get (json_val, "iss");
327 if ((NULL == issuer_json) || (! json_is_string (issuer_json)))
328 {
329 json_decref (json_val);
330 return NULL;
331 }
332 issuer = GNUNET_strdup (json_string_value (issuer_json));
333 json_decref (json_val);
334 return issuer;
335}
336
337
345static char *
347 const struct GNUNET_RECLAIM_Credential *cred)
348{
350 return NULL;
351 return jwt_get_issuer (cls, cred->data, cred->data_size);
352}
353
354
362static char *
364 const struct GNUNET_RECLAIM_Presentation *cred)
365{
367 return NULL;
368 return jwt_get_issuer (cls, cred->data, cred->data_size);
369}
370
371
381 const char *data,
382 size_t data_size,
383 struct GNUNET_TIME_Absolute *exp)
384{
385 const char *jwt_body;
386 char *jwt_string;
387 char delim[] = ".";
388 char *decoded_jwt;
389 json_t *exp_json;
390 json_t *json_val;
391 json_error_t json_err;
392
393 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Parsing JWT attributes.\n");
394 jwt_string = GNUNET_strndup (data, data_size);
395 jwt_body = strtok (jwt_string, delim);
396 jwt_body = strtok (NULL, delim);
397 GNUNET_STRINGS_base64url_decode (jwt_body, strlen (jwt_body),
398 (void **) &decoded_jwt);
399 json_val = json_loads (decoded_jwt, JSON_DECODE_ANY, &json_err);
400 GNUNET_free (decoded_jwt);
401 GNUNET_free (jwt_string);
402 if (NULL == json_val)
403 return GNUNET_SYSERR;
404 exp_json = json_object_get (json_val, "exp");
405 if ((NULL == exp_json) || (! json_is_integer (exp_json)))
406 {
407 json_decref (json_val);
408 return GNUNET_SYSERR;
409 }
410 exp->abs_value_us = json_integer_value (exp_json) * 1000 * 1000;
411 json_decref (json_val);
412 return GNUNET_OK;
413}
414
415
425 const struct GNUNET_RECLAIM_Credential *cred,
426 struct GNUNET_TIME_Absolute *exp)
427{
429 return GNUNET_NO;
430 return jwt_get_expiration (cls, cred->data, cred->data_size, exp);
431}
432
433
443 const struct GNUNET_RECLAIM_Presentation *cred,
444 struct GNUNET_TIME_Absolute *exp)
445{
447 return GNUNET_NO;
448 return jwt_get_expiration (cls, cred->data, cred->data_size, exp);
449}
450
451
454 const struct GNUNET_RECLAIM_Credential *cred,
455 const struct GNUNET_RECLAIM_AttributeList *attrs,
456 struct GNUNET_RECLAIM_Presentation **presentation)
457{
459 return GNUNET_NO;
460 *presentation = GNUNET_RECLAIM_presentation_new (
462 cred->data,
463 cred->data_size);
464 return GNUNET_OK;
465}
466
467
468void *
470
477void *
479{
481
498 return api;
499}
500
501
502void *
504
511void *
513{
515
516 GNUNET_free (api);
517 return NULL;
518}
519
520
521/* end of plugin_reclaim_credential_type_jwt.c */
static gnutls_certificate_credentials_t cred
The credential.
static char * data
The data to insert into the dht.
struct GNUNET_HashCode key
The key used in the DHT.
static char * value
Value of the record to add/remove.
static uint32_t type
Type string converted to DNS type value.
static size_t data_size
Number of bytes in data.
Plugin API for reclaim attribute types.
#define GNUNET_log(kind,...)
GNUNET_GenericReturnValue
Named constants for return values.
@ GNUNET_OK
@ GNUNET_NO
@ GNUNET_SYSERR
#define GNUNET_assert(cond)
Use this for fatal errors that cannot be handled.
@ GNUNET_ERROR_TYPE_WARNING
@ GNUNET_ERROR_TYPE_ERROR
@ GNUNET_ERROR_TYPE_DEBUG
#define GNUNET_strdup(a)
Wrapper around GNUNET_xstrdup_.
#define GNUNET_strndup(a, length)
Wrapper around GNUNET_xstrndup_.
#define GNUNET_new(type)
Allocate a struct or union of the given type.
#define GNUNET_free(ptr)
Wrapper around free.
struct GNUNET_RECLAIM_Presentation * GNUNET_RECLAIM_presentation_new(uint32_t type, const void *data, size_t data_size)
void GNUNET_RECLAIM_attribute_list_add(struct GNUNET_RECLAIM_AttributeList *attrs, const char *attr_name, const struct GNUNET_RECLAIM_Identifier *credential, uint32_t type, const void *data, size_t data_size)
Add a new attribute to a claim list.
@ GNUNET_RECLAIM_CREDENTIAL_TYPE_JWT
A JSON Web Token credential.
@ GNUNET_RECLAIM_ATTRIBUTE_TYPE_STRING
String attribute.
size_t GNUNET_STRINGS_base64url_decode(const char *data, size_t len, void **out)
Decode from Base64url.
Definition: strings.c:1759
static enum GNUNET_GenericReturnValue jwt_create_presentation(void *cls, const struct GNUNET_RECLAIM_Credential *cred, const struct GNUNET_RECLAIM_AttributeList *attrs, struct GNUNET_RECLAIM_Presentation **presentation)
static struct GNUNET_RECLAIM_AttributeList * jwt_parse_attributes(void *cls, const char *data, size_t data_size)
Parse a JWT and return the respective claim value as Attribute.
static char * jwt_get_issuer_c(void *cls, const struct GNUNET_RECLAIM_Credential *cred)
Parse a JWT and return the issuer.
void * libgnunet_plugin_reclaim_credential_jwt_done(void *cls)
Exit point from the plugin.
static char * jwt_get_issuer(void *cls, const char *data, size_t data_size)
Parse a JWT and return the issuer.
static struct GNUNET_RECLAIM_AttributeList * jwt_parse_attributes_c(void *cls, const struct GNUNET_RECLAIM_Credential *cred)
Parse a JWT and return the respective claim value as Attribute.
static enum GNUNET_GenericReturnValue jwt_get_expiration(void *cls, const char *data, size_t data_size, struct GNUNET_TIME_Absolute *exp)
Parse a JWT and return the expiration.
const char * name
static char * jwt_get_issuer_p(void *cls, const struct GNUNET_RECLAIM_Presentation *cred)
Parse a JWT and return the issuer.
static char * jwt_value_to_string(void *cls, uint32_t type, const void *data, size_t data_size)
Convert the 'value' of an credential to a string.
static uint32_t jwt_typename_to_number(void *cls, const char *jwt_typename)
Convert a type name to the corresponding number.
void * libgnunet_plugin_reclaim_credential_jwt_init(void *cls)
Entry point for the plugin.
static const char * jwt_number_to_typename(void *cls, uint32_t type)
Convert a type number to the corresponding type string (e.g.
static int jwt_string_to_value(void *cls, uint32_t type, const char *s, void **data, size_t *data_size)
Convert human-readable version of a 'value' of an credential to the binary representation.
static enum GNUNET_GenericReturnValue jwt_get_expiration_p(void *cls, const struct GNUNET_RECLAIM_Presentation *cred, struct GNUNET_TIME_Absolute *exp)
Parse a JWT and return the expiration.
static struct @42 jwt_cred_name_map[]
Mapping of credential type numbers to human-readable credential type names.
static struct GNUNET_RECLAIM_AttributeList * jwt_parse_attributes_p(void *cls, const struct GNUNET_RECLAIM_Presentation *cred)
Parse a JWT and return the respective claim value as Attribute.
static enum GNUNET_GenericReturnValue jwt_get_expiration_c(void *cls, const struct GNUNET_RECLAIM_Credential *cred, struct GNUNET_TIME_Absolute *exp)
Parse a JWT and return the expiration.
A list of GNUNET_RECLAIM_Attribute structures.
Each plugin is required to return a pointer to a struct of this type as the return value from its ent...
GNUNET_RECLAIM_CredentialValueToStringFunction value_to_string
Conversion to string.
GNUNET_RECLAIM_CredentialTypenameToNumberFunction typename_to_number
Typename to number.
GNUNET_RECLAIM_PresentationGetIssuerFunction get_issuer_p
Attesation issuer.
GNUNET_RECLAIM_CredentialNumberToTypenameFunction number_to_typename
Number to typename.
GNUNET_RECLAIM_CredentialGetIssuerFunction get_issuer
Attesation issuer.
GNUNET_RECLAIM_PresentationGetExpirationFunction get_expiration_p
Expiration.
GNUNET_RECLAIM_PresentationValueToStringFunction value_to_string_p
Conversion to string.
GNUNET_RECLAIM_CredentialStringToValueFunction string_to_value
Conversion to binary.
GNUNET_RECLAIM_CredentialGetExpirationFunction get_expiration
Expiration.
void * cls
Closure for all of the callbacks.
GNUNET_RECLAIM_CredentialGetAttributesFunction get_attributes
Attesation attributes.
GNUNET_RECLAIM_PresentationStringToValueFunction string_to_value_p
Conversion to binary.
GNUNET_RECLAIM_PresentationTypenameToNumberFunction typename_to_number_p
Typename to number.
GNUNET_RECLAIM_PresentationNumberToTypenameFunction number_to_typename_p
Number to typename.
GNUNET_RECLAIM_CredentialToPresentation create_presentation
Get presentation.
GNUNET_RECLAIM_PresentationGetAttributesFunction get_attributes_p
Attesation attributes.
A credential presentation.
Time for absolute times used by GNUnet, in microseconds.
uint64_t abs_value_us
The actual value.