stuff
[oweals/gnunet.git] / src / transport / gnunet-service-transport_clients.c
index 97939bbf7a290c068d788a5481448463a809fbcd..5136e03f1f89e540bef3d773205e45a7cd8f036f 100644 (file)
@@ -27,6 +27,8 @@
 #include "gnunet-service-transport_clients.h"
 #include "gnunet-service-transport_hello.h"
 #include "gnunet-service-transport_neighbours.h"
+#include "gnunet-service-transport_plugins.h"
+#include "gnunet-service-transport_validation.h"
 #include "gnunet-service-transport.h"
 #include "transport.h"
 
@@ -425,6 +427,23 @@ GST_clients_handle_start (void *cls,
 }
 
 
+/**
+ * Client sent us a HELLO.  Process the request.
+ *
+ * @param cls unused
+ * @param client the client
+ * @param message the HELLO message
+ */
+void
+GST_clients_handle_hello (void *cls,
+                         struct GNUNET_SERVER_Client *client,
+                         const struct GNUNET_MessageHeader *message)
+{
+  GST_validation_handle_hello (message);
+  GNUNET_SERVER_receive_done (client, GNUNET_OK);
+}
+
+
 /**
  * Client asked for transmission to a peer.  Process the request.
  *
@@ -437,6 +456,8 @@ GST_clients_handle_send (void *cls,
                         struct GNUNET_SERVER_Client *client,
                         const struct GNUNET_MessageHeader *message)
 {
+  /* FIXME */
+  GNUNET_SERVER_receive_done (client, GNUNET_SYSERR);
 }
 
 
@@ -452,6 +473,50 @@ GST_clients_handle_set_quota (void *cls,
                              struct GNUNET_SERVER_Client *client,
                              const struct GNUNET_MessageHeader *message)
 {
+  const struct QuotaSetMessage *qsm;
+
+  qsm = (const struct QuotaSetMessage *) message;
+  GNUNET_STATISTICS_update (GST_stats,
+                           gettext_noop ("# SET QUOTA messages received"),
+                           1,
+                           GNUNET_NO);
+#if DEBUG_TRANSPORT 
+  GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
+              "Received `%s' request (new quota %u) from client for peer `%4s'\n",
+              "SET_QUOTA",
+             (unsigned int) ntohl (qsm->quota.value__),
+             GNUNET_i2s (&qsm->peer));
+#endif
+  GST_neighbours_set_incoming_quota (&qsm->peer,
+                                    qsm->quota);
+  GNUNET_SERVER_receive_done (client, GNUNET_OK);
+}
+
+
+/**
+ * Take the given address and append it to the set of results sent back to
+ * the client.
+ *
+ * @param cls the transmission context used ('struct GNUNET_SERVER_TransmitContext*')
+ * @param address the resolved name, NULL to indicate the last response
+ */
+static void
+transmit_address_to_client (void *cls, 
+                           const char *address)
+{
+  struct GNUNET_SERVER_TransmitContext *tc = cls;
+
+  if (NULL == address)
+    {
+      GNUNET_SERVER_transmit_context_append_data (tc, NULL, 0,
+                                                 GNUNET_MESSAGE_TYPE_TRANSPORT_ADDRESS_REPLY);
+      GNUNET_SERVER_transmit_context_run (tc,
+                                         GNUNET_TIME_UNIT_FOREVER_REL);
+      return;
+    }
+  GNUNET_SERVER_transmit_context_append_data (tc, 
+                                             address, strlen (address) + 1,
+                                             GNUNET_MESSAGE_TYPE_TRANSPORT_ADDRESS_REPLY);
 }
 
 
@@ -467,12 +532,108 @@ GST_clients_handle_address_lookup (void *cls,
                                   struct GNUNET_SERVER_Client *client,
                                   const struct GNUNET_MessageHeader *message)
 {
+  const struct AddressLookupMessage *alum;
+  struct GNUNET_TRANSPORT_PluginFunctions *papi;
+  const char *plugin_name;
+  const char *address;
+  uint32_t address_len;
+  uint16_t size;
+  struct GNUNET_SERVER_TransmitContext *tc;
+  struct GNUNET_TIME_Relative rtimeout;
+  int32_t numeric;
+
+  size = ntohs (message->size);
+  if (size < sizeof (struct AddressLookupMessage))
+    {
+      GNUNET_break (0);
+      GNUNET_SERVER_receive_done (client, GNUNET_SYSERR);
+      return;
+    }
+  alum = (const struct AddressLookupMessage *) message;
+  address_len = ntohl (alum->addrlen);
+  if (size <= sizeof (struct AddressLookupMessage) + address_len)
+    {
+      GNUNET_break (0);
+      GNUNET_SERVER_receive_done (client, GNUNET_SYSERR);
+      return;
+    }
+  address = (const char *) &alum[1];
+  plugin_name = (const char *) &address[address_len];
+  if (plugin_name
+      [size - sizeof (struct AddressLookupMessage) - address_len - 1] != '\0')
+    {
+      GNUNET_break (0);
+      GNUNET_SERVER_receive_done (client, GNUNET_SYSERR);
+      return;
+    }
+  rtimeout = GNUNET_TIME_relative_ntoh (alum->timeout);
+  numeric = ntohl (alum->numeric_only);
+  tc = GNUNET_SERVER_transmit_context_create (client);
+  papi = GST_plugins_find (plugin_name);
+  if (NULL == papi)
+    {
+      GNUNET_SERVER_transmit_context_append_data (tc, NULL, 0,
+                                                 GNUNET_MESSAGE_TYPE_TRANSPORT_ADDRESS_REPLY);
+      GNUNET_SERVER_transmit_context_run (tc, rtimeout);
+      return;
+    }
+  GNUNET_SERVER_disable_receive_done_warning (client);
+  papi->address_pretty_printer (papi->cls,
+                               plugin_name,
+                               address, address_len,
+                               numeric,
+                               rtimeout,
+                               &transmit_address_to_client, tc);
+}
+
+
+/**
+ * Send an address to the client.
+ *
+ * @param cls our 'struct GNUNET_SERVER_TransmitContext' (for sending)
+ * @param public_key public key for the peer, never NULL
+ * @param target peer this change is about, never NULL
+ * @param valid_until until what time do we consider the address valid?
+ * @param validation_block  is FOREVER if the address is for an unsupported plugin (from PEERINFO)
+ *                          is ZERO if the address is considered valid (no validation needed)
+ *                          is a time in the future if we're currently denying re-validation
+ * @param plugin_name name of the plugin
+ * @param plugin_address binary address
+ * @param plugin_address_len length of address
+ */
+static void
+send_address_to_client (void *cls,
+                       const struct GNUNET_CRYPTO_RsaPublicKeyBinaryEncoded *public_key,
+                       const struct GNUNET_PeerIdentity *target,
+                       struct GNUNET_TIME_Absolute valid_until,
+                       struct GNUNET_TIME_Absolute validation_block,
+                       const char *plugin_name,
+                       const void *plugin_address,
+                       size_t plugin_address_len)
+{
+  struct GNUNET_SERVER_TransmitContext *tc = cls;
+  char *addr_buf;
+
+  /* FIXME: move to a binary format!!! */
+  GNUNET_asprintf (&addr_buf, "%s --- %s, %s",
+                  GST_plugins_a2s (plugin_name,
+                                   plugin_address,
+                                   plugin_address_len),
+                  (GNUNET_YES == GST_neighbours_test_connected (target))
+                  ? "CONNECTED"
+                  : "DISCONNECTED",
+                  (GNUNET_TIME_absolute_get_remaining (valid_until).rel_value > 0)
+                  ? "VALIDATED"
+                  : "UNVALIDATED");
+  transmit_address_to_client (tc, addr_buf);
+  GNUNET_free (addr_buf);
 }
 
 
 /**
  * Client asked to obtain information about a peer's addresses.
  * Process the request.
+ * FIXME: use better name!
  *
  * @param cls unused
  * @param client the client
@@ -483,12 +644,54 @@ GST_clients_handle_peer_address_lookup (void *cls,
                                        struct GNUNET_SERVER_Client *client,
                                        const struct GNUNET_MessageHeader *message)
 {
+  const struct PeerAddressLookupMessage *peer_address_lookup;
+  struct GNUNET_SERVER_TransmitContext *tc;
+
+  peer_address_lookup = (const struct PeerAddressLookupMessage *) message;
+  GNUNET_break (ntohl (peer_address_lookup->reserved) == 0);
+  tc = GNUNET_SERVER_transmit_context_create (client);
+  (void) GST_validation_get_addresses (&peer_address_lookup->peer,
+                                      GNUNET_YES,
+                                      &send_address_to_client,
+                                      tc);
+  GNUNET_SERVER_transmit_context_append_data (tc,
+                                             NULL, 0,
+                                             GNUNET_MESSAGE_TYPE_TRANSPORT_ADDRESS_REPLY);
+  GNUNET_SERVER_transmit_context_run (tc, 
+                                     GNUNET_TIME_UNIT_FOREVER_REL);
 }
 
 
 /**
- * Client asked to obtain information about all addresses.
- * Process the request.
+ * Output the active address of connected neighbours to the given client.
+ *
+ * @param cls the 'struct GNUNET_SERVER_TransmitContext' for transmission to the client
+ * @param neighbour identity of the neighbour
+ * @param ats performance data
+ * @param ats_count number of entries in ats (excluding 0-termination)
+ */
+static void
+output_addresses (void *cls,
+                 const struct GNUNET_PeerIdentity *neighbour,
+                 const struct GNUNET_TRANSPORT_ATS_Information *ats,
+                 uint32_t ats_count)
+{
+  struct GNUNET_SERVER_TransmitContext *tc = cls;
+  char *addr_buf;
+
+  /* FIXME: move to a binary format!!! */
+  GNUNET_asprintf (&addr_buf, 
+                  "%s: %s",
+                  GNUNET_i2s(neighbour),
+                  GST_plugins_a2s ("FIXME", NULL, 0));
+  transmit_address_to_client (tc, addr_buf);
+  GNUNET_free (addr_buf);
+}
+
+
+/**
+ * Client asked to obtain information about all actively used addresses.
+ * Process the request.  FIXME: use better name!
  *
  * @param cls unused
  * @param client the client
@@ -498,7 +701,16 @@ void
 GST_clients_handle_address_iterate (void *cls,
                                    struct GNUNET_SERVER_Client *client,
                                    const struct GNUNET_MessageHeader *message)
-{
+{ 
+  struct GNUNET_SERVER_TransmitContext *tc;
+
+  GNUNET_SERVER_disable_receive_done_warning (client);
+  tc = GNUNET_SERVER_transmit_context_create (client);
+  GST_neighbours_iterate (&output_addresses,
+                         tc);
+  GNUNET_SERVER_transmit_context_append_data (tc, NULL, 0,
+                                              GNUNET_MESSAGE_TYPE_TRANSPORT_ADDRESS_REPLY);
+  GNUNET_SERVER_transmit_context_run (tc, GNUNET_TIME_UNIT_FOREVER_REL);
 }