Refactoring gnunet time
[oweals/gnunet.git] / src / nat / test_nat.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 /**
22  * @file nat/test_nat.c
23  * @brief Testcase for NAT library
24  * @author Milan Bouchet-Valat
25  */
26
27 /**
28  * Testcase for port redirection and public IP address retrieval.
29  * This test never fails, because there need to be a NAT box set up for that.
30  * So we only get IP address and open the 2086 port using any UPnP and NAT-PMP
31  * routers found, wait for 30s, close ports and return.
32  * Have a look at the logs and use NMAP to check that it works with your box.
33  */
34
35
36 #include "platform.h"
37 #include "gnunet_common.h"
38 #include "gnunet_util_lib.h"
39 #include "gnunet_program_lib.h"
40 #include "gnunet_scheduler_lib.h"
41 #include "gnunet_nat_lib.h"
42
43 /* Time to wait before stopping NAT, in seconds */
44 #define TIMEOUT 60
45
46 struct addr_cls
47 {
48   const struct sockaddr *addr;
49   socklen_t addrlen;
50 };
51
52 static void
53 addr_callback (void *cls, int add_remove,
54                const struct sockaddr *addr, socklen_t addrlen)
55 {
56   GNUNET_log (GNUNET_ERROR_TYPE_INFO, "External address changed: %s %s\n",
57               add_remove == GNUNET_YES ? "added" : "removed",
58               GNUNET_a2s (addr, addrlen));
59 }
60
61 static void
62 stop (void *cls, const struct GNUNET_SCHEDULER_TaskContext *tc)
63 {
64   struct GNUNET_NAT_Handle *nat = cls;
65
66   GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Stopping NAT and quitting...\n");
67   GNUNET_NAT_unregister (nat);
68 }
69
70 /* Return the address of the default interface,
71  * or any interface with a valid address if the default is not valid */
72 static int
73 process_if (void *cls,
74             const char *name,
75             int isDefault, const struct sockaddr *addr, socklen_t addrlen)
76 {
77   struct addr_cls *data = cls;
78
79   if (addr)
80     {
81       data->addr = addr;
82       data->addrlen = addrlen;
83     }
84
85   if (strcmp (name, "eth1") == 0 && addr->sa_family == AF_INET)
86     return GNUNET_SYSERR;
87
88   return GNUNET_OK;
89
90
91   if (isDefault && addr)
92     return GNUNET_SYSERR;
93   else
94     return GNUNET_OK;
95 }
96
97 static void
98 run (void *cls,
99      struct GNUNET_SCHEDULER_Handle *sched,
100      char *const *args,
101      const char *cfgfile, const struct GNUNET_CONFIGURATION_Handle *cfg)
102 {
103   struct GNUNET_NAT_Handle *nat;
104   struct addr_cls data;
105   struct sockaddr *addr;
106
107   GNUNET_log_setup ("test-nat", "DEBUG", NULL);
108
109   data.addr = NULL;
110   GNUNET_OS_network_interfaces_list (process_if, &data);
111   if (!data.addr)
112     {
113       GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
114                   "Could not find a valid interface address!\n");
115       exit (GNUNET_SYSERR);
116     }
117
118   addr = GNUNET_malloc (data.addrlen);
119   memcpy (addr, data.addr, data.addrlen);
120
121   GNUNET_assert (addr->sa_family == AF_INET || addr->sa_family == AF_INET6);
122   if (addr->sa_family == AF_INET)
123     ((struct sockaddr_in *) addr)->sin_port = htons (2086);
124   else
125     ((struct sockaddr_in6 *) addr)->sin6_port = htons (2086);
126
127   GNUNET_log (GNUNET_ERROR_TYPE_INFO,
128               "Requesting NAT redirection from address %s...\n",
129               GNUNET_a2s (addr, data.addrlen));
130
131   nat = GNUNET_NAT_register (sched, addr, data.addrlen, addr_callback, NULL);
132   GNUNET_free (addr);
133
134   GNUNET_SCHEDULER_add_delayed (sched,
135                                 GNUNET_TIME_relative_multiply
136                                 (GNUNET_TIME_UNIT_SECONDS, TIMEOUT), stop,
137                                 nat);
138 }
139
140 int
141 main (int argc, char *const argv[])
142 {
143   struct GNUNET_GETOPT_CommandLineOption options[] = {
144     GNUNET_GETOPT_OPTION_END
145   };
146
147   GNUNET_log_setup ("test-nat", "DEBUG", NULL);
148
149   GNUNET_log (GNUNET_ERROR_TYPE_INFO,
150               "Testing NAT library, timeout set to %d seconds\n", TIMEOUT);
151
152   GNUNET_PROGRAM_run (argc, argv, "test-nat", "nohelp", options, &run, NULL);
153
154   return 0;
155 }
156
157 /* end of test_nat.c */