/* This file is part of GNUnet. (C) 2009, 2010 Christian Grothoff (and other contributing authors) GNUnet is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3, or (at your option) any later version. GNUnet is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with GNUnet; see the file COPYING. If not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /** * @file transport/transport_api_address_iterate.c * @brief api for asking transport service to iterate over all * known addresses * * This api provides a single function call to ask the transport * service to list all peers and their known addresses, as pretty * printed by the appropriate plugin. Reports whether or not the * address is connected as well. */ #include "platform.h" #include "gnunet_client_lib.h" #include "gnunet_arm_service.h" #include "gnunet_hello_lib.h" #include "gnunet_protocols.h" #include "gnunet_server_lib.h" #include "gnunet_time_lib.h" #include "gnunet_transport_service.h" #include "transport.h" /** * Context for the address lookup. */ struct AddressLookupCtx { /** * Function to call with the human-readable address. */ GNUNET_TRANSPORT_AddressLookUpBinaryCallback cb; /** * Closure for cb. */ void *cb_cls; /** * Connection to the service. */ struct GNUNET_CLIENT_Connection *client; /** * When should this operation time out? */ struct GNUNET_TIME_Absolute timeout; }; /** * Function called with responses from the service. * * @param cls our 'struct AddressLookupCtx*' * @param msg NULL on timeout or error, otherwise presumably a * message with the human-readable peer and address */ static void peer_address_response_processor (void *cls, const struct GNUNET_MessageHeader *msg) { struct AddressLookupCtx *alucb = cls; struct AddressIterateResponseMessage *address; uint16_t size; char * transport; //size_t transport_len; //void * addr; size_t addrlen; if (msg == NULL) { alucb->cb (alucb->cb_cls, NULL, NULL, NULL, 0); GNUNET_CLIENT_disconnect (alucb->client, GNUNET_NO); GNUNET_free (alucb); return; } GNUNET_break (ntohs (msg->type) == GNUNET_MESSAGE_TYPE_TRANSPORT_ADDRESS_REPLY); size = ntohs (msg->size); GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Received message type %u size %u\n", ntohs (msg->type), size); if (size == sizeof (struct GNUNET_MessageHeader)) { /* done! */ alucb->cb (alucb->cb_cls, NULL, NULL, NULL, 0); GNUNET_CLIENT_disconnect (alucb->client, GNUNET_NO); GNUNET_free (alucb); return; } if (size < sizeof (struct AddressIterateResponseMessage)) { /* invalid reply */ GNUNET_break (0); alucb->cb (alucb->cb_cls, NULL, NULL, NULL, 0); GNUNET_CLIENT_disconnect (alucb->client, GNUNET_NO); GNUNET_free (alucb); return; } address = (struct AddressIterateResponseMessage *) &msg[1]; transport = (char *) &address[0]; //transport_len = ntohs(address->pluginlen); addrlen = ntohs(address->addrlen); /* expect more replies */ GNUNET_CLIENT_receive (alucb->client, &peer_address_response_processor, alucb, GNUNET_TIME_absolute_get_remaining (alucb->timeout)); alucb->cb (alucb->cb_cls, &address->peer, transport, NULL, addrlen); } /** * Return all the known addresses for a peer. * * @param cfg configuration to use * @param timeout how long is the lookup allowed to take at most * @param peer_address_callback function to call with the results * @param peer_address_callback_cls closure for peer_address_callback */ void GNUNET_TRANSPORT_address_iterate (const struct GNUNET_CONFIGURATION_Handle *cfg, struct GNUNET_TIME_Relative timeout, GNUNET_TRANSPORT_AddressLookUpBinaryCallback peer_address_callback, void *peer_address_callback_cls) { struct AddressIterateMessage msg; struct GNUNET_TIME_Absolute abs_timeout; struct AddressLookupCtx *peer_address_lookup_cb; struct GNUNET_CLIENT_Connection *client; client = GNUNET_CLIENT_connect ("transport", cfg); if (client == NULL) { peer_address_callback (peer_address_callback_cls, NULL, NULL, NULL, 0); return; } abs_timeout = GNUNET_TIME_relative_to_absolute (timeout); msg.header.size = htons (sizeof (struct AddressIterateMessage)); msg.header.type = htons (GNUNET_MESSAGE_TYPE_TRANSPORT_ADDRESS_ITERATE); msg.timeout = GNUNET_TIME_absolute_hton (abs_timeout); peer_address_lookup_cb = GNUNET_malloc (sizeof (struct AddressLookupCtx)); peer_address_lookup_cb->cb = peer_address_callback; peer_address_lookup_cb->cb_cls = peer_address_callback_cls; peer_address_lookup_cb->timeout = abs_timeout; peer_address_lookup_cb->client = client; GNUNET_assert (GNUNET_OK == GNUNET_CLIENT_transmit_and_get_response (client, &msg.header, timeout, GNUNET_YES, &peer_address_response_processor, peer_address_lookup_cb)); } /* end of transport_api_address_iterate.c */