f53aad544925c84d4a7d8df9d20ad5dccd331ff1
[oweals/gnunet.git] / src / peerstore / test_peerstore_api_sync.c
1 /*
2      This file is part of GNUnet.
3      Copyright (C) 2015 GNUnet e.V.
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., 51 Franklin Street, Fifth Floor,
18      Boston, MA 02110-1301, USA.
19 */
20 /**
21  * @file peerstore/test_peerstore_api_sync.c
22  * @brief testcase for peerstore sync-on-disconnect feature. Stores
23  *        a value just before disconnecting, and then checks that
24  *        this value is actually stored.
25  * @author Omar Tarabai
26  * @author Christian Grothoff (minor fix, comments)
27  */
28 #include "platform.h"
29 #include "gnunet_util_lib.h"
30 #include "gnunet_testing_lib.h"
31 #include "gnunet_peerstore_service.h"
32
33 /**
34  * Overall result, 0 for success.
35  */
36 static int ok = 404;
37
38 /**
39  * Configuration we use.
40  */
41 static const struct GNUNET_CONFIGURATION_Handle *cfg;
42
43 /**
44  * handle to talk to the peerstore.
45  */
46 static struct GNUNET_PEERSTORE_Handle *h;
47
48 /**
49  * Subsystem we store the value for.
50  */
51 static const char *subsystem = "test_peerstore_api_sync";
52
53 /**
54  * Fake PID under which we store the value.
55  */
56 static struct GNUNET_PeerIdentity pid;
57
58 /**
59  * Test key we're storing the test value under.
60  */
61 static const char *key = "test_peerstore_api_store_key";
62
63 /**
64  * Test value we are storing.
65  */
66 static const char *val = "test_peerstore_api_store_val";
67
68
69 /**
70  * Function that should be called with the result of the
71  * lookup, and finally once with NULL to signal the end
72  * of the iteration.
73  *
74  * Upon the first call, we set "ok" to success. On the
75  * second call (end of iteration) we terminate the test.
76  *
77  * @param cls NULL
78  * @param record the information stored in the peerstore
79  * @param emsg any error message
80  * @return #GNUNET_YES (all good, continue)
81  */
82 static int
83 iterate_cb (void *cls, 
84             const struct GNUNET_PEERSTORE_Record *record,
85             const char *emsg)
86 {
87   const char *rec_val;
88
89   GNUNET_break (NULL == emsg);
90   if (NULL == record)
91   {
92     GNUNET_PEERSTORE_disconnect (h, 
93                                  GNUNET_NO);
94     GNUNET_SCHEDULER_shutdown ();
95     return GNUNET_YES;
96   }
97   rec_val = record->value;
98   GNUNET_break (0 == strcmp (rec_val, val));
99   ok = 0;
100   return GNUNET_YES;
101 }
102
103
104 /**
105  * Run the 2nd stage of the test where we fetch the
106  * data that should have been stored.
107  *
108  * @param cls NULL
109  * @param tc unused
110  */
111 static void
112 test_cont (void *cls,
113            const struct GNUNET_SCHEDULER_TaskContext *tc)
114 {
115   h = GNUNET_PEERSTORE_connect (cfg);
116   GNUNET_PEERSTORE_iterate (h, 
117                             subsystem, 
118                             &pid, key,
119                             GNUNET_TIME_UNIT_FOREVER_REL, 
120                             &iterate_cb, NULL);
121 }
122
123
124 /**
125  * Actually run the test.
126  */
127 static void
128 test1 ()
129 {
130   h = GNUNET_PEERSTORE_connect (cfg);
131   GNUNET_PEERSTORE_store (h, 
132                           subsystem,
133                           &pid, 
134                           key, 
135                           val, strlen (val) + 1,
136                           GNUNET_TIME_UNIT_FOREVER_ABS,
137                           GNUNET_PEERSTORE_STOREOPTION_REPLACE, 
138                           NULL, NULL);
139   GNUNET_PEERSTORE_disconnect (h, 
140                                GNUNET_YES);
141   h = NULL;
142   /* We need to wait a little bit to give the disconnect
143      a chance to actually finish the operation; otherwise,
144      the test may fail non-deterministically if the new
145      connection is faster than the cleanup routine of the
146      old one. */
147   GNUNET_SCHEDULER_add_delayed (GNUNET_TIME_UNIT_SECONDS,
148                                 &test_cont,
149                                 NULL);
150 }
151
152
153 /**
154  * Initialize globals and launch the test.
155  *
156  * @param cls NULL
157  * @param c configuration to use
158  * @param peer handle to our peer (unused)
159  */
160 static void
161 run (void *cls, 
162      const struct GNUNET_CONFIGURATION_Handle *c,
163      struct GNUNET_TESTING_Peer *peer)
164 {
165   cfg = c;
166   GNUNET_assert (NULL != h);
167   memset (&pid, 1, sizeof (pid));
168   test1 ();
169 }
170
171
172 int
173 main (int argc, char *argv[])
174 {
175   if (0 !=
176       GNUNET_TESTING_service_run ("test-gnunet-peerstore-sync", 
177                                   "peerstore",
178                                   "test_peerstore_api_data.conf",
179                                   &run, NULL))
180     return 1;
181   if (0 != ok)
182     fprintf (stderr,
183              "Test failed: %d\n",
184              ok).
185   return ok;
186 }
187
188 /* end of test_peerstore_api_sync.c */