2 This file is part of GNUnet.
3 (C) 2013 Christian Grothoff (and other contributing authors)
5 GNUnet is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published
7 by the Free Software Foundation; either version 3, or (at your
8 option) any later version.
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 General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with GNUnet; see the file COPYING. If not, write to the
17 Free Software Foundation, Inc., 59 Temple Place - Suite 330,
18 Boston, MA 02111-1307, USA.
22 * @file identity/gnunet-service-identity.c
23 * @brief identity management service
24 * @author Christian Grothoff
26 * The purpose of this service is to manage private keys that
27 * represent the various egos/pseudonyms/identities of a GNUnet user.
30 #include "gnunet_util_lib.h"
31 #include "gnunet_constants.h"
32 #include "gnunet_protocols.h"
33 #include "gnunet_statistics_service.h"
34 #include "gnunet_identity_service.h"
39 * Information we keep about each ego.
45 * We keep egos in a DLL.
50 * We keep egos in a DLL.
55 * Private key of the ego.
57 struct GNUNET_CRYPTO_EccPrivateKey *pk;
60 * String identifier for the ego.
68 * Handle to our current configuration.
70 static const struct GNUNET_CONFIGURATION_Handle *cfg;
73 * Handle to subsystem configuration which for each subsystem contains
74 * the name of the default ego.
76 static struct GNUNET_CONFIGURATION_Handle *subsystem_cfg;
79 * Handle to the statistics service.
81 static struct GNUNET_STATISTICS_Handle *stats;
84 * Notification context, simplifies client broadcasts.
86 static struct GNUNET_SERVER_NotificationContext *nc;
89 * Directory where we store the identities.
91 static char *ego_directory;
94 * Configuration file name where subsystem information is kept.
96 static char *subsystem_cfg_file;
99 * Head of DLL of all egos.
101 static struct Ego *ego_head;
104 * Tail of DLL of all egos.
106 static struct Ego *ego_tail;
110 * Task run during shutdown.
116 shutdown_task (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc)
122 GNUNET_SERVER_notification_context_destroy (nc);
127 GNUNET_STATISTICS_destroy (stats, GNUNET_NO);
130 GNUNET_CONFIGURATION_destroy (subsystem_cfg);
131 subsystem_cfg = NULL;
132 GNUNET_free (subsystem_cfg_file);
133 subsystem_cfg_file = NULL;
134 GNUNET_free (ego_directory);
135 ego_directory = NULL;
136 while (NULL != (e = ego_head))
138 GNUNET_CONTAINER_DLL_remove (ego_head, ego_tail, e);
139 GNUNET_CRYPTO_ecc_key_free (e->pk);
146 * Send a result code back to the client.
148 * @param client client that should receive the result code
149 * @param result_code code to transmit
150 * @param emsg error message to include (or NULL for none)
153 send_result_code (struct GNUNET_SERVER_Client *client,
154 uint32_t result_code,
157 struct GNUNET_IDENTITY_ResultCodeMessage *rcm;
163 elen = strlen (emsg) + 1;
164 rcm = GNUNET_malloc (sizeof (struct GNUNET_IDENTITY_ResultCodeMessage) + elen);
165 rcm->header.type = htons (GNUNET_MESSAGE_TYPE_IDENTITY_RESULT_CODE);
166 rcm->header.size = htons (sizeof (struct GNUNET_IDENTITY_ResultCodeMessage) + elen);
167 rcm->result_code = htonl (result_code);
168 memcpy (&rcm[1], emsg, elen);
169 GNUNET_SERVER_notification_context_unicast (nc, client, &rcm->header, GNUNET_YES);
175 * Create an update message with information about the current state of an ego.
177 * @param ego ego to create message for
178 * @return corresponding update message
180 static struct GNUNET_IDENTITY_UpdateMessage *
181 create_update_message (struct Ego *ego)
183 struct GNUNET_IDENTITY_UpdateMessage *um;
187 struct GNUNET_CRYPTO_EccPrivateKeyBinaryEncoded *enc;
189 name_len = (NULL == ego->identifier) ? 0 : (strlen (ego->identifier) + 1);
190 enc = GNUNET_CRYPTO_ecc_encode_key (ego->pk);
191 pk_len = ntohs (enc->size);
192 um = GNUNET_malloc (sizeof (struct GNUNET_IDENTITY_UpdateMessage) + pk_len + name_len);
193 um->header.type = htons (GNUNET_MESSAGE_TYPE_IDENTITY_UPDATE);
194 um->header.size = htons (sizeof (struct GNUNET_IDENTITY_UpdateMessage) + pk_len + name_len);
195 um->name_len = htons (name_len);
196 um->pk_len = htons (pk_len);
197 str = (char *) &um[1];
198 memcpy (str, enc, pk_len);
199 memcpy (&str[pk_len], ego->identifier, name_len);
206 * Handler for START message from client, sends information
207 * about all identities to the client immediately and
208 * adds the client to the notification context for future
212 * @param client who sent the message
213 * @param message the message received
216 handle_start_message (void *cls, struct GNUNET_SERVER_Client *client,
217 const struct GNUNET_MessageHeader *message)
219 struct GNUNET_IDENTITY_UpdateMessage *um;
222 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
223 "Received START message from client\n");
224 GNUNET_SERVER_notification_context_add (nc, client);
225 for (ego = ego_head; NULL != ego; ego = ego->next)
227 um = create_update_message (ego);
228 GNUNET_SERVER_notification_context_unicast (nc, client, &um->header, GNUNET_YES);
231 GNUNET_SERVER_receive_done (client, GNUNET_OK);
236 * Handler for GET_DEFAULT message from client, returns
237 * default identity for some service.
240 * @param client who sent the message
241 * @param message the message received
244 handle_get_default_message (void *cls, struct GNUNET_SERVER_Client *client,
245 const struct GNUNET_MessageHeader *message)
247 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
248 "Received GET_DEFAULT message from client\n");
249 // setup_estimate_message (&em);
250 // GNUNET_SERVER_notification_context_unicast (nc, client, &em.header, GNUNET_YES);
251 GNUNET_break (0); // not implemented!
252 GNUNET_SERVER_receive_done (client, GNUNET_SYSERR);
257 * Handler for SET_DEFAULT message from client, updates
258 * default identity for some service.
261 * @param client who sent the message
262 * @param message the message received
265 handle_set_default_message (void *cls, struct GNUNET_SERVER_Client *client,
266 const struct GNUNET_MessageHeader *message)
268 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
269 "Received SET_DEFAULT message from client\n");
270 // setup_estimate_message (&em);
271 // GNUNET_SERVER_notification_context_unicast (nc, client, &em.header, GNUNET_YES);
272 GNUNET_break (0); // not implemented!
273 GNUNET_SERVER_receive_done (client, GNUNET_SYSERR);
278 * Handler for CREATE message from client, creates
282 * @param client who sent the message
283 * @param message the message received
286 handle_create_message (void *cls, struct GNUNET_SERVER_Client *client,
287 const struct GNUNET_MessageHeader *message)
289 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
290 "Received CREATE message from client\n");
291 // setup_estimate_message (&em);
292 // GNUNET_SERVER_notification_context_unicast (nc, client, &em.header, GNUNET_YES);
293 GNUNET_break (0); // not implemented!
294 GNUNET_SERVER_receive_done (client, GNUNET_SYSERR);
300 * Handler for RENAME message from client, creates
304 * @param client who sent the message
305 * @param message the message received
308 handle_rename_message (void *cls, struct GNUNET_SERVER_Client *client,
309 const struct GNUNET_MessageHeader *message)
311 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
312 "Received RENAME message from client\n");
313 // setup_estimate_message (&em);
314 // GNUNET_SERVER_notification_context_unicast (nc, client, &em.header, GNUNET_YES);
315 GNUNET_break (0); // not implemented!
316 GNUNET_SERVER_receive_done (client, GNUNET_SYSERR);
321 * Handler for DELETE message from client, creates
325 * @param client who sent the message
326 * @param message the message received
329 handle_delete_message (void *cls, struct GNUNET_SERVER_Client *client,
330 const struct GNUNET_MessageHeader *message)
332 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
333 "Received DELETE message from client\n");
334 // setup_estimate_message (&em);
335 // GNUNET_SERVER_notification_context_unicast (nc, client, &em.header, GNUNET_YES);
336 GNUNET_break (0); // not implemented!
338 send_result_code (client, 1, gettext_noop ("no matching ego found"));
339 GNUNET_SERVER_receive_done (client, GNUNET_OK);
344 * Handle network size estimate clients.
347 * @param server the initialized server
348 * @param c configuration to use
352 struct GNUNET_SERVER_Handle *server,
353 const struct GNUNET_CONFIGURATION_Handle *c)
355 static const struct GNUNET_SERVER_MessageHandler handlers[] = {
356 {&handle_start_message, NULL,
357 GNUNET_MESSAGE_TYPE_IDENTITY_START, sizeof (struct GNUNET_MessageHeader)},
358 {&handle_get_default_message, NULL,
359 GNUNET_MESSAGE_TYPE_IDENTITY_GET_DEFAULT, 0},
360 {&handle_set_default_message, NULL,
361 GNUNET_MESSAGE_TYPE_IDENTITY_SET_DEFAULT, 0},
362 {&handle_create_message, NULL,
363 GNUNET_MESSAGE_TYPE_IDENTITY_CREATE, 0},
364 {&handle_rename_message, NULL,
365 GNUNET_MESSAGE_TYPE_IDENTITY_RENAME, 0},
366 {&handle_delete_message, NULL,
367 GNUNET_MESSAGE_TYPE_IDENTITY_DELETE, 0},
373 GNUNET_CONFIGURATION_get_value_filename (cfg, "identity",
377 GNUNET_log_config_missing (GNUNET_ERROR_TYPE_ERROR, "identity", "EGODIR");
378 GNUNET_SCHEDULER_shutdown ();
382 GNUNET_CONFIGURATION_get_value_filename (cfg, "identity",
384 &subsystem_cfg_file))
386 GNUNET_log_config_missing (GNUNET_ERROR_TYPE_ERROR, "identity", "SUBSYSTEM_CFG");
387 GNUNET_SCHEDULER_shutdown ();
390 subsystem_cfg = GNUNET_CONFIGURATION_create ();
392 GNUNET_DISK_file_test (subsystem_cfg_file)) &&
394 GNUNET_CONFIGURATION_parse (subsystem_cfg,
395 subsystem_cfg_file)) )
397 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
398 _("Failed to parse subsystem identity configuration file `%s'\n"),
400 GNUNET_SCHEDULER_shutdown ();
403 stats = GNUNET_STATISTICS_create ("identity", cfg);
404 GNUNET_SERVER_add_handlers (server, handlers);
405 nc = GNUNET_SERVER_notification_context_create (server, 1);
406 GNUNET_SCHEDULER_add_delayed (GNUNET_TIME_UNIT_FOREVER_REL, &shutdown_task,
412 * The main function for the network size estimation service.
414 * @param argc number of arguments from the command line
415 * @param argv command line arguments
416 * @return 0 ok, 1 on error
419 main (int argc, char *const *argv)
422 GNUNET_SERVICE_run (argc, argv, "identity",
423 GNUNET_SERVICE_OPTION_NONE,
424 &run, NULL)) ? 0 : 1;
428 /* end of gnunet-service-identity.c */