codesonar fixes
[oweals/gnunet.git] / src / testing / test_testing_group_remote.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 testing/test_testing_group_remote.c
22  * @brief testcase for testing remote and local starting and connecting
23  *        of hosts from the testing library.  The test_testing_data_remote.conf
24  *        file should be modified if this testcase is intended to be used.
25  */
26 #include "platform.h"
27 #include "gnunet_testing_lib.h"
28
29 #define VERBOSE GNUNET_YES
30
31
32 /**
33  * How long until we give up on connecting the peers?
34  */
35 #define TIMEOUT GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_SECONDS, 300)
36
37 #define DEFAULT_NUM_PEERS 8;
38
39 static int ok;
40
41 static int peers_left;
42
43 static int peers_failed;
44
45 static struct GNUNET_TESTING_PeerGroup *pg;
46
47 static unsigned long long num_peers;
48
49
50 /**
51  * Check whether peers successfully shut down.
52  */
53 void shutdown_callback (void *cls,
54                         const char *emsg)
55 {
56   if (emsg != NULL)
57     {
58 #if VERBOSE
59       GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
60                   "Shutdown of peers failed!\n");
61 #endif
62       if (ok == 0)
63         ok = 666;
64     }
65   else
66     {
67 #if VERBOSE
68       GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
69                   "All peers successfully shut down!\n");
70 #endif
71     }
72 }
73
74
75 static void
76 my_cb (void *cls,
77        const struct GNUNET_PeerIdentity *id,
78        const struct GNUNET_CONFIGURATION_Handle *cfg,
79        struct GNUNET_TESTING_Daemon *d, const char *emsg)
80 {
81   if (emsg != NULL)
82     {
83       peers_failed++;
84     }
85
86   peers_left--;
87   if (peers_left == 0)
88     {
89       //GNUNET_TESTING_daemons_stop (pg, TIMEOUT, &shutdown_callback, NULL);
90       ok = 0;
91     }
92   else if (peers_failed == peers_left)
93     {
94       GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Too many peers failed, ending test!\n");
95       //GNUNET_TESTING_daemons_stop (pg, TIMEOUT, &shutdown_callback, NULL);
96     }
97 }
98
99
100 static void
101 run (void *cls,
102      char *const *args,
103      const char *cfgfile, const struct GNUNET_CONFIGURATION_Handle *cfg)
104 {
105   struct GNUNET_TESTING_Host *hosts;
106   struct GNUNET_TESTING_Host *hostpos;
107   struct GNUNET_TESTING_Host *temphost;
108   char *hostfile;
109   struct stat frstat;
110   char *buf;
111   char *data;
112   int count;
113   int ret;
114   ok = 1;
115 #if VERBOSE
116   GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Starting daemons.\n");
117 #endif
118
119   if (GNUNET_SYSERR ==
120       GNUNET_CONFIGURATION_get_value_number (cfg, "testing", "num_peers",
121                                              &num_peers))
122     num_peers = DEFAULT_NUM_PEERS;
123
124   GNUNET_assert(num_peers > 0 && num_peers < (unsigned long long)-1);
125   if (GNUNET_OK !=
126       GNUNET_CONFIGURATION_get_value_string (cfg, "testing", "hostfile",
127                                              &hostfile))
128     hostfile = NULL;
129
130   hosts = NULL;
131   data = NULL;
132   if (hostfile != NULL)
133     {
134       if (GNUNET_OK != GNUNET_DISK_file_test (hostfile))
135           GNUNET_DISK_fn_write (hostfile, NULL, 0, GNUNET_DISK_PERM_USER_READ
136             | GNUNET_DISK_PERM_USER_WRITE);
137       if ((0 != STAT (hostfile, &frstat)) || (frstat.st_size == 0))
138         {
139           GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
140                       "Could not open file specified for host list, ending test!");
141           ok = 1119;
142           GNUNET_free(hostfile);
143           return;
144         }
145
146     data = GNUNET_malloc_large (frstat.st_size);
147     GNUNET_assert(data != NULL);
148     if (frstat.st_size !=
149         GNUNET_DISK_fn_read (hostfile, data, frstat.st_size))
150       {
151         GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
152                   "Could not read file %s specified for host list, ending test!", hostfile);
153         GNUNET_free (hostfile);
154         GNUNET_free (data);
155         return;
156       }
157
158     GNUNET_free_non_null(hostfile);
159
160     buf = data;
161     count = 0;
162     while (count < frstat.st_size)
163       {
164         count++;
165         if (count >= frstat.st_size)
166           break;
167
168         /* if (((data[count] == '\n') || (data[count] == '\0')) && (buf != &data[count]))*/
169         if (((data[count] == '\n')) && (buf != &data[count]))
170           {
171             data[count] = '\0';
172             temphost = GNUNET_malloc(sizeof(struct GNUNET_TESTING_Host));
173             ret = sscanf(buf, "%a[a-zA-Z0-9]@%a[a-zA-Z0-9.]:%hd", &temphost->username, &temphost->hostname, &temphost->port);
174             if (3 == ret)
175               {
176                 GNUNET_log(GNUNET_ERROR_TYPE_WARNING, "Successfully read host %s, port %d and user %s from file\n", temphost->hostname, temphost->port, temphost->username);
177               }
178             else
179               {
180                 GNUNET_log(GNUNET_ERROR_TYPE_WARNING, "Error reading line `%s' in hostfile\n", buf);
181                 GNUNET_free(temphost);
182                 buf = &data[count + 1];
183                 continue;
184               }
185             /* temphost->hostname = buf; */
186             temphost->next = hosts;
187             hosts = temphost;
188             buf = &data[count + 1];
189           }
190         else if ((data[count] == '\n') || (data[count] == '\0'))
191           buf = &data[count + 1];
192       }
193     }
194
195   peers_left = num_peers;
196   pg = GNUNET_TESTING_daemons_start (cfg,
197                                      peers_left,
198                                      TIMEOUT,
199                                      NULL,
200                                      NULL,
201                                      &my_cb,
202                                      NULL,
203                                      NULL,
204                                      NULL,
205                                      hosts);
206   hostpos = hosts;
207   while (hostpos != NULL)
208     {
209       temphost = hostpos->next;
210       GNUNET_free(hostpos->hostname);
211       GNUNET_free(hostpos->username);
212       GNUNET_free(hostpos);
213       hostpos = temphost;
214     }
215   GNUNET_free_non_null(data);
216   GNUNET_assert (pg != NULL);
217
218 }
219
220 static int
221 check ()
222 {
223   char *const argv[] = { "test-testing",
224     "-c",
225     "test_testing_data_remote.conf",
226 #if VERBOSE
227     "-L", "DEBUG",
228 #endif
229     NULL
230   };
231   struct GNUNET_GETOPT_CommandLineOption options[] = {
232     GNUNET_GETOPT_OPTION_END
233   };
234   GNUNET_PROGRAM_run ((sizeof (argv) / sizeof (char *)) - 1,
235                       argv, "test-testing-group", "nohelp",
236                       options, &run, &ok);
237   return ok;
238 }
239
240 int
241 main (int argc, char *argv[])
242 {
243   int ret;
244
245   GNUNET_log_setup ("test-testing-group",
246 #if VERBOSE
247                     "DEBUG",
248 #else
249                     "WARNING",
250 #endif
251                     NULL);
252   ret = check ();
253   /**
254    * Still need to remove the base testing directory here,
255    * because group starts will create subdirectories under this
256    * main dir.  However, we no longer need to sleep, as the
257    * shutdown sequence won't return until everything is cleaned
258    * up.
259    */
260   GNUNET_DISK_directory_remove ("/tmp/test-gnunet-testing");
261   return ret;
262 }
263
264 /* end of test_testing_group.c */