- suggest backoff reset
[oweals/gnunet.git] / src / ats / test_ats_api_reset_backoff.c
1 /*
2      This file is part of GNUnet.
3      (C) 2010,2011 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 ats/test_ats_api_reset_backoff.c
22  * @brief test case for block reset api
23  * @author Christian Grothoff
24  * @author Matthias Wachs
25  */
26 #include "platform.h"
27 #include "gnunet_ats_service.h"
28 #include "ats.h"
29
30 #define TIMEOUT GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_SECONDS, 10)
31 #define ATS_TIMEOUT GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_SECONDS, 1)
32
33 static GNUNET_SCHEDULER_TaskIdentifier die_task;
34
35 static GNUNET_SCHEDULER_TaskIdentifier suggest_timeout_task;
36
37 static struct GNUNET_ATS_SchedulingHandle *ats;
38
39 struct GNUNET_OS_Process *arm_proc;
40
41 static int ret;
42
43 struct Address
44 {
45   char *plugin;
46   size_t plugin_len;
47
48   void *addr;
49   size_t addr_len;
50
51   struct GNUNET_ATS_Information *ats;
52   int ats_count;
53
54   void *session;
55 };
56
57 struct PeerContext
58 {
59   struct GNUNET_PeerIdentity id;
60
61   struct Address *addr;
62 };
63
64 struct GNUNET_HELLO_Address hello_addr;
65 struct Address address;
66 struct PeerContext peer;
67 struct GNUNET_ATS_Information atsi[2];
68
69 static void
70 stop_arm ()
71 {
72   if (0 != GNUNET_OS_process_kill (arm_proc, SIGTERM))
73     GNUNET_log_strerror (GNUNET_ERROR_TYPE_WARNING, "kill");
74   GNUNET_OS_process_wait (arm_proc);
75   GNUNET_OS_process_destroy (arm_proc);
76   arm_proc = NULL;
77 }
78
79
80 static void
81 end_badly (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc)
82 {
83   die_task = GNUNET_SCHEDULER_NO_TASK;
84
85   if (suggest_timeout_task != GNUNET_SCHEDULER_NO_TASK)
86   {
87     GNUNET_SCHEDULER_cancel (suggest_timeout_task);
88     suggest_timeout_task = GNUNET_SCHEDULER_NO_TASK;
89   }
90
91   if (ats != NULL)
92   {
93     GNUNET_ATS_scheduling_done (ats);
94     ats = NULL;
95   }
96
97   ret = GNUNET_SYSERR;
98
99   stop_arm ();
100 }
101
102
103 static void
104 end ()
105 {
106   GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Shutting down\n");
107   if (die_task != GNUNET_SCHEDULER_NO_TASK)
108   {
109     GNUNET_SCHEDULER_cancel (die_task);
110     die_task = GNUNET_SCHEDULER_NO_TASK;
111   }
112
113   if (suggest_timeout_task != GNUNET_SCHEDULER_NO_TASK)
114   {
115     GNUNET_SCHEDULER_cancel (suggest_timeout_task);
116     suggest_timeout_task = GNUNET_SCHEDULER_NO_TASK;
117   }
118
119   GNUNET_ATS_scheduling_done (ats);
120
121   ret = 0;
122
123   stop_arm ();
124 }
125
126
127 static void
128 suggest_timeout (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc)
129 {
130   suggest_timeout_task = GNUNET_SCHEDULER_NO_TASK;
131
132   GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Requesting address for peer timed out\n");
133 }
134
135 static void request_address ()
136 {
137
138   if (suggest_timeout_task != GNUNET_SCHEDULER_NO_TASK)
139   {
140     GNUNET_SCHEDULER_cancel (suggest_timeout_task);
141     suggest_timeout_task = GNUNET_SCHEDULER_NO_TASK;
142   }
143
144   suggest_timeout_task = GNUNET_SCHEDULER_add_delayed(GNUNET_TIME_UNIT_MINUTES, &suggest_timeout, NULL);
145
146   GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Requesting address for peer `%s'\n",
147               GNUNET_i2s (&peer.id));
148   GNUNET_ATS_suggest_address (ats, &peer.id);
149 }
150
151
152 static void
153 address_suggest_cb (void *cls, const struct GNUNET_HELLO_Address *a,
154                     struct Session *session,
155                     struct GNUNET_BANDWIDTH_Value32NBO bandwidth_out,
156                     struct GNUNET_BANDWIDTH_Value32NBO bandwidth_in,
157                     const struct GNUNET_ATS_Information *atsi,
158                     uint32_t ats_count)
159 {
160   static int suggestions;
161   GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "ATS suggests address `%s'\n",
162               GNUNET_i2s (&a->peer));
163
164   if (0 != memcmp (&a->peer, &peer.id,
165                          sizeof (struct GNUNET_PeerIdentity)))
166   {
167    GNUNET_break (0);
168    if (die_task != GNUNET_SCHEDULER_NO_TASK)
169    {
170      GNUNET_SCHEDULER_cancel (die_task);
171      die_task = GNUNET_SCHEDULER_NO_TASK;
172    }
173    GNUNET_SCHEDULER_add_now (&end_badly, NULL);
174    return;
175   }
176
177   if (0 != strcmp (a->transport_name, address.plugin))
178   {
179    GNUNET_break (0);
180    if (die_task != GNUNET_SCHEDULER_NO_TASK)
181    {
182      GNUNET_SCHEDULER_cancel (die_task);
183      die_task = GNUNET_SCHEDULER_NO_TASK;
184    }
185    GNUNET_SCHEDULER_add_now (&end_badly, NULL);
186    return;
187   }
188
189   if (a->address_length != address.addr_len)
190   {
191    GNUNET_break (0);
192    if (die_task != GNUNET_SCHEDULER_NO_TASK)
193    {
194      GNUNET_SCHEDULER_cancel (die_task);
195      die_task = GNUNET_SCHEDULER_NO_TASK;
196    }
197    GNUNET_SCHEDULER_add_now (&end_badly, NULL);
198    return;
199   }
200
201   if (0 != memcmp (a->address, address.addr,
202       a->address_length))
203   {
204    GNUNET_break (0);
205    if (die_task != GNUNET_SCHEDULER_NO_TASK)
206    {
207      GNUNET_SCHEDULER_cancel (die_task);
208      die_task = GNUNET_SCHEDULER_NO_TASK;
209    }
210    GNUNET_SCHEDULER_add_now (&end_badly, NULL);
211    return;
212   }
213
214   if (session != address.session)
215   {
216    GNUNET_break (0);
217    if (die_task != GNUNET_SCHEDULER_NO_TASK)
218    {
219      GNUNET_SCHEDULER_cancel (die_task);
220      die_task = GNUNET_SCHEDULER_NO_TASK;
221    }
222    GNUNET_SCHEDULER_add_now (&end_badly, NULL);
223    return;
224   }
225
226   suggestions ++;
227
228   if (2 == suggestions)
229   {
230     GNUNET_SCHEDULER_add_now(&end, NULL);
231     return;
232   }
233
234   if (suggest_timeout_task != GNUNET_SCHEDULER_NO_TASK)
235   {
236     GNUNET_SCHEDULER_cancel (suggest_timeout_task);
237     suggest_timeout_task = GNUNET_SCHEDULER_NO_TASK;
238   }
239
240   GNUNET_ATS_reset_backoff(ats, &a->peer);
241   request_address ();
242 }
243
244
245
246
247 void
248 start_arm (const char *cfgname)
249 {
250   arm_proc =
251     GNUNET_OS_start_process (GNUNET_YES, NULL, NULL, "gnunet-service-arm",
252                                "gnunet-service-arm",
253                                "-c", cfgname, NULL);
254 }
255
256 static void
257 check (void *cls, char *const *args, const char *cfgfile,
258        const struct GNUNET_CONFIGURATION_Handle *cfg)
259 {
260   ret = GNUNET_SYSERR;
261
262   die_task = GNUNET_SCHEDULER_add_delayed (TIMEOUT, &end_badly, NULL);
263   start_arm (cfgfile);
264
265   ats = GNUNET_ATS_scheduling_init (cfg, &address_suggest_cb, NULL);
266
267   if (ats == NULL)
268   {
269     ret = GNUNET_SYSERR;
270     end ();
271     return;
272   }
273
274   /* set up peer */
275   GNUNET_CRYPTO_hash_create_random (GNUNET_CRYPTO_QUALITY_WEAK,
276                                     &peer.id.hashPubKey);
277   GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Created peer `%s'\n", GNUNET_i2s (&peer.id));
278
279   address.plugin = "test";
280   address.session = NULL;
281   address.addr = GNUNET_strdup ("test");
282   address.addr_len = 4;
283
284   GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Adding address\n");
285
286   hello_addr.peer = peer.id;
287   hello_addr.transport_name = address.plugin;
288   hello_addr.address = address.addr;
289   hello_addr.address_length = address.addr_len;
290   GNUNET_ATS_address_update (ats, &hello_addr, address.session, NULL, 0);
291
292   request_address ();
293 }
294
295 int
296 main (int argc, char *argv[])
297 {
298   static char *const argv2[] = { "test_ats_api_scheduling",
299     "-c",
300     "test_ats_api.conf",
301     "-L", "WARNING",
302     NULL
303   };
304
305   static struct GNUNET_GETOPT_CommandLineOption options[] = {
306     GNUNET_GETOPT_OPTION_END
307   };
308
309   GNUNET_PROGRAM_run ((sizeof (argv2) / sizeof (char *)) - 1, argv2,
310                       "test_ats_api_scheduling", "nohelp", options, &check,
311                       NULL);
312
313
314   return ret;
315 }
316 /* end of file test_ats_api_reset_backoff.c */