curly wars / auto-indentation
[oweals/gnunet.git] / src / util / test_server_with_client.c
1 /*
2      This file is part of GNUnet.
3      (C) 2009 Christian Grothoff (and other contributing authors)
4
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.
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      General Public License for more details.
14
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.
19 */
20 /**
21  * @file util/test_server_with_client.c
22  * @brief tests for server.c and client.c,
23  *       specifically disconnect_notify,
24  *       client_get_address and receive_done (resume processing)
25  */
26 #include "platform.h"
27 #include "gnunet_common.h"
28 #include "gnunet_scheduler_lib.h"
29 #include "gnunet_client_lib.h"
30 #include "gnunet_server_lib.h"
31 #include "gnunet_time_lib.h"
32
33 #define VERBOSE GNUNET_EXTRA_LOGGING
34
35 #define PORT 22335
36
37 #define MY_TYPE 128
38
39
40 static struct GNUNET_SERVER_Handle *server;
41
42 static struct GNUNET_CLIENT_Connection *client;
43
44 static struct GNUNET_CONFIGURATION_Handle *cfg;
45
46 static int ok;
47
48 static void
49 send_done (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc)
50 {
51   struct GNUNET_SERVER_Client *argclient = cls;
52
53   GNUNET_assert (ok == 3);
54   ok++;
55   GNUNET_SERVER_receive_done (argclient, GNUNET_OK);
56 }
57
58
59 static void
60 recv_cb (void *cls, struct GNUNET_SERVER_Client *argclient,
61          const struct GNUNET_MessageHeader *message)
62 {
63   void *addr;
64   size_t addrlen;
65   struct sockaddr_in sa;
66   struct sockaddr_in *have;
67
68   GNUNET_assert (GNUNET_OK ==
69                  GNUNET_SERVER_client_get_address (argclient, &addr, &addrlen));
70
71   GNUNET_assert (addrlen == sizeof (struct sockaddr_in));
72   have = addr;
73   memset (&sa, 0, sizeof (sa));
74 #if HAVE_SOCKADDR_IN_SIN_LEN
75   sa.sin_len = sizeof (sa);
76 #endif
77   sa.sin_family = AF_INET;
78   sa.sin_port = have->sin_port;
79   sa.sin_addr.s_addr = htonl (INADDR_LOOPBACK);
80   GNUNET_assert (0 == memcmp (&sa, addr, addrlen));
81   GNUNET_free (addr);
82   switch (ok)
83   {
84   case 2:
85     ok++;
86     GNUNET_SCHEDULER_add_delayed (GNUNET_TIME_relative_multiply
87                                   (GNUNET_TIME_UNIT_MILLISECONDS, 50),
88                                   &send_done, argclient);
89     break;
90   case 4:
91     ok++;
92     GNUNET_CLIENT_disconnect (client, GNUNET_YES);
93     GNUNET_SERVER_receive_done (argclient, GNUNET_OK);
94     break;
95   default:
96     GNUNET_assert (0);
97   }
98
99 }
100
101
102 static void
103 clean_up (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc)
104 {
105   GNUNET_SERVER_destroy (server);
106   server = NULL;
107   GNUNET_CONFIGURATION_destroy (cfg);
108   cfg = NULL;
109 }
110
111
112 /**
113  * Functions with this signature are called whenever a client
114  * is disconnected on the network level.
115  *
116  * @param cls closure
117  * @param client identification of the client
118  */
119 static void
120 notify_disconnect (void *cls, struct GNUNET_SERVER_Client *client)
121 {
122   if (client == NULL)
123     return;
124   GNUNET_assert (ok == 5);
125   ok = 0;
126   GNUNET_SCHEDULER_add_now (&clean_up, NULL);
127 }
128
129
130 static size_t
131 notify_ready (void *cls, size_t size, void *buf)
132 {
133   struct GNUNET_MessageHeader *msg;
134
135   GNUNET_assert (size >= 256);
136   GNUNET_assert (1 == ok);
137   ok++;
138   msg = buf;
139   msg->type = htons (MY_TYPE);
140   msg->size = htons (sizeof (struct GNUNET_MessageHeader));
141   msg++;
142   msg->type = htons (MY_TYPE);
143   msg->size = htons (sizeof (struct GNUNET_MessageHeader));
144   return 2 * sizeof (struct GNUNET_MessageHeader);
145 }
146
147
148 static struct GNUNET_SERVER_MessageHandler handlers[] = {
149   {&recv_cb, NULL, MY_TYPE, sizeof (struct GNUNET_MessageHeader)},
150   {NULL, NULL, 0, 0}
151 };
152
153
154 static void
155 task (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc)
156 {
157   struct sockaddr_in sa;
158   struct sockaddr *sap[2];
159   socklen_t slens[2];
160
161   sap[0] = (struct sockaddr *) &sa;
162   slens[0] = sizeof (sa);
163   sap[1] = NULL;
164   slens[1] = 0;
165   memset (&sa, 0, sizeof (sa));
166 #if HAVE_SOCKADDR_IN_SIN_LEN
167   sa.sin_len = sizeof (sa);
168 #endif
169   sa.sin_family = AF_INET;
170   sa.sin_port = htons (PORT);
171   server =
172       GNUNET_SERVER_create (NULL, NULL, sap, slens,
173                             GNUNET_TIME_relative_multiply
174                             (GNUNET_TIME_UNIT_MILLISECONDS, 250), GNUNET_NO);
175   GNUNET_assert (server != NULL);
176   handlers[0].callback_cls = cls;
177   GNUNET_SERVER_add_handlers (server, handlers);
178   GNUNET_SERVER_disconnect_notify (server, &notify_disconnect, cls);
179   cfg = GNUNET_CONFIGURATION_create ();
180   GNUNET_CONFIGURATION_set_value_number (cfg, "test", "PORT", PORT);
181   GNUNET_CONFIGURATION_set_value_string (cfg, "test", "HOSTNAME", "localhost");
182   GNUNET_CONFIGURATION_set_value_string (cfg, "resolver", "HOSTNAME",
183                                          "localhost");
184   client = GNUNET_CLIENT_connect ("test", cfg);
185   GNUNET_assert (client != NULL);
186   GNUNET_CLIENT_notify_transmit_ready (client, 256,
187                                        GNUNET_TIME_relative_multiply
188                                        (GNUNET_TIME_UNIT_MILLISECONDS, 250),
189                                        GNUNET_NO, &notify_ready, NULL);
190 }
191
192
193 /**
194  * Main method, starts scheduler with task1,
195  * checks that "ok" is correct at the end.
196  */
197 static int
198 check ()
199 {
200
201   ok = 1;
202   GNUNET_SCHEDULER_run (&task, NULL);
203   return ok;
204 }
205
206
207 int
208 main (int argc, char *argv[])
209 {
210   int ret = 0;
211
212   GNUNET_log_setup ("test_server_with_client",
213 #if VERBOSE
214                     "DEBUG",
215 #else
216                     "WARNING",
217 #endif
218                     NULL);
219   ret += check ();
220
221   return ret;
222 }
223
224 /* end of test_server_with_client.c */