GNUnet 0.21.1
gnunet-service-messenger_peer_store.c
Go to the documentation of this file.
1/*
2 This file is part of GNUnet.
3 Copyright (C) 2023--2024 GNUnet e.V.
4
5 GNUnet is free software: you can redistribute it and/or modify it
6 under the terms of the GNU Affero General Public License as published
7 by the Free Software Foundation, either version 3 of the License,
8 or (at your 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 Affero General Public License for more details.
14
15 You should have received a copy of the GNU Affero General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>.
17
18 SPDX-License-Identifier: AGPL3.0-or-later
19 */
27
29#include "gnunet_common.h"
31#include "messenger_api_util.h"
32
34{
37};
38
39void
42{
43 GNUNET_assert ((store) && (service));
44
45 store->service = service;
47}
48
49
51iterate_destroy_peers (void *cls, const struct GNUNET_ShortHashCode *id,
52 void *value)
53{
55 GNUNET_free (entry);
56 return GNUNET_YES;
57}
58
59
60void
62{
63 GNUNET_assert ((store) && (store->peers));
64
65 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Clear peer store\n");
66
68 NULL);
70
71 store->peers = NULL;
72}
73
74
75void
77 const char *path)
78{
79 GNUNET_assert ((store) && (path));
80
82 return;
83
84 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Load peer store from path: %s\n",
85 path);
86
89
91 path, GNUNET_DISK_OPEN_READ, permission
92 );
93
94 if (! handle)
95 return;
96
98
101 struct GNUNET_PeerIdentity peer;
102 ssize_t len;
103
104 do {
105 len = GNUNET_DISK_file_read (handle, &peer, sizeof(peer));
106
107 if (len != sizeof(peer))
108 break;
109
110
112
113 if (! entry)
114 continue;
115
116 GNUNET_memcpy (&(entry->peer), &peer, sizeof(entry->peer));
117 entry->active = GNUNET_YES;
118
120
122 store->peers, &peer_id, entry,
124 continue;
125
126 GNUNET_free (entry);
127 } while (len == sizeof(peer));
128
130}
131
132
134iterate_save_peers (void *cls, const struct GNUNET_ShortHashCode *id,
135 void *value)
136{
137 struct GNUNET_DISK_FileHandle *handle = cls;
139
140 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Save peer store entry: %s\n",
141 GNUNET_sh2s (id));
142
143 if ((! entry) || (GNUNET_YES != entry->active))
144 return GNUNET_YES;
145
146 GNUNET_DISK_file_write (handle, &(entry->peer), sizeof(entry->peer));
147 return GNUNET_YES;
148}
149
150
151void
153 const char *path)
154{
155 GNUNET_assert ((store) && (path));
156
157 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Save peer store to path: %s\n",
158 path);
159
162
165 );
166
167 if (! handle)
168 return;
169
172 handle);
173
176}
177
178
180{
182 const struct GNUNET_HashCode *hash;
184};
185
187verify_store_peer (void *cls, const struct GNUNET_ShortHashCode *id,
188 void *value)
189{
192
193 if (! entry)
194 return GNUNET_YES;
195
196 if (GNUNET_OK == verify_message_by_peer (verify->message,
197 verify->hash, &(entry->peer)))
198 {
199 verify->sender = &(entry->peer);
200 return GNUNET_NO;
201 }
202
203 return GNUNET_YES;
204}
205
206
209 const struct GNUNET_PeerIdentity *peer,
210 const struct GNUNET_ShortHashCode *id,
212{
213 GNUNET_assert ((store) && (peer));
214
217
218 if (! entry)
219 return NULL;
220
221 GNUNET_memcpy (&(entry->peer), peer, sizeof(entry->peer));
222 entry->active = active;
223
225 store->peers, id, entry,
227 {
228 GNUNET_free (entry);
229 return NULL;
230 }
231
232 return entry;
233}
234
235
236static const struct GNUNET_PeerIdentity*
238{
239 static struct GNUNET_PeerIdentity peer;
240
241 if (GNUNET_OK != get_service_peer_identity (store->service, &peer))
242 return NULL;
243
244 return &peer;
245}
246
247
250 const struct GNUNET_MESSENGER_Message *message,
251 const struct GNUNET_HashCode *hash)
252{
253 GNUNET_assert ((store) && (store->peers) && (message) && (hash));
254
255 if (GNUNET_YES != is_peer_message (message))
256 return NULL;
257
259 verify.message = message;
260 verify.hash = hash;
261 verify.sender = NULL;
262
266
267 if (verify.sender)
268 return verify.sender;
269
270 const struct GNUNET_PeerIdentity *peer;
271 enum GNUNET_GenericReturnValue active;
272
273 if (GNUNET_MESSENGER_KIND_PEER == message->header.kind)
274 {
275 peer = &(message->body.peer.peer);
276 active = GNUNET_YES;
277 }
278 else if (GNUNET_MESSENGER_KIND_MISS == message->header.kind)
279 {
280 peer = &(message->body.miss.peer);
281 active = GNUNET_NO;
282 }
283 else
284 {
286 "Peer message does not contain a peer identity\n");
287
288 peer = get_store_service_peer_identity (store);
289 active = GNUNET_NO;
290
291 if (! peer)
292 return NULL;
293 }
294
297
298 if (0 != GNUNET_memcmp (&peer_id, &(message->header.sender_id)))
299 {
301 "Sender id does not match peer identity\n");
302 return NULL;
303 }
304
305 if (GNUNET_OK != verify_message_by_peer (message, hash, peer))
306 {
308 "Verification of message with peer identity failed!\n");
309 }
310
312 entry = add_peer_store_entry (store, peer, &peer_id, active);
313
314 if (! entry)
315 {
317 "Initialization of entry in peer store failed: %s\n",
319
320 return NULL;
321 }
322
323 return &(entry->peer);
324}
325
326
328{
331};
332
334find_store_peer (void *cls, const struct GNUNET_ShortHashCode *id, void *value)
335{
336 struct GNUNET_MESSENGER_ClosureFindPeer *find = cls;
338
339 if (! entry)
340 return GNUNET_YES;
341
342 if (0 == GNUNET_memcmp (find->requested, &(entry->peer)))
343 {
344 find->match = entry;
345 return GNUNET_NO;
346 }
347
348 return GNUNET_YES;
349}
350
351
352void
354 const struct GNUNET_PeerIdentity *peer,
356{
357 GNUNET_assert ((store) && (store->peers) && (peer));
358
361
362 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Update peer store entry: %s\n",
364
366 find.requested = peer;
367 find.match = NULL;
368
370 find_store_peer, &find);
371
372 if (find.match)
373 {
374 find.match->active = active;
375 return;
376 }
377
378 if (! add_peer_store_entry (store, peer, &peer_id, active))
380 "Initial update of entry in peer store failed: %s\n",
382}
static int verify
Verify mode.
Definition: gnunet-abd.c:127
static char * peer_id
Option –peer.
Definition: gnunet-cadet.c:42
static char * value
Value of the record to add/remove.
static struct GNUNET_SERVICE_Handle * service
Handle to our service instance.
void update_store_peer(struct GNUNET_MESSENGER_PeerStore *store, const struct GNUNET_PeerIdentity *peer, enum GNUNET_GenericReturnValue active)
Adds a peer identity to the store if necessary.
static enum GNUNET_GenericReturnValue find_store_peer(void *cls, const struct GNUNET_ShortHashCode *id, void *value)
struct GNUNET_PeerIdentity * get_store_peer_of(struct GNUNET_MESSENGER_PeerStore *store, const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash)
Returns the peer identity inside the store which verifies the signature of a given message as valid.
void load_peer_store(struct GNUNET_MESSENGER_PeerStore *store, const char *path)
Loads peer identities from a file into a peer store.
void clear_peer_store(struct GNUNET_MESSENGER_PeerStore *store)
Clears a peer store, wipes its content and deallocates its memory.
static const struct GNUNET_PeerIdentity * get_store_service_peer_identity(struct GNUNET_MESSENGER_PeerStore *store)
static enum GNUNET_GenericReturnValue iterate_destroy_peers(void *cls, const struct GNUNET_ShortHashCode *id, void *value)
static struct GNUNET_MESSENGER_PeerStoreEntry * add_peer_store_entry(struct GNUNET_MESSENGER_PeerStore *store, const struct GNUNET_PeerIdentity *peer, const struct GNUNET_ShortHashCode *id, enum GNUNET_GenericReturnValue active)
static enum GNUNET_GenericReturnValue verify_store_peer(void *cls, const struct GNUNET_ShortHashCode *id, void *value)
static enum GNUNET_GenericReturnValue iterate_save_peers(void *cls, const struct GNUNET_ShortHashCode *id, void *value)
void save_peer_store(const struct GNUNET_MESSENGER_PeerStore *store, const char *path)
Saves peer identities from a peer store into a file.
void init_peer_store(struct GNUNET_MESSENGER_PeerStore *store, struct GNUNET_MESSENGER_Service *service)
Initializes a peer store as fully empty.
enum GNUNET_GenericReturnValue get_service_peer_identity(struct GNUNET_MESSENGER_Service *service, struct GNUNET_PeerIdentity *peer)
Tries to write the peer identity of the peer running a service on to the peer parameter.
static struct GNUNET_VPN_Handle * handle
Handle to vpn service.
Definition: gnunet-vpn.c:35
commonly used definitions; globals in this file are exempt from the rule that the module name ("commo...
struct GNUNET_DISK_FileHandle * GNUNET_DISK_file_open(const char *fn, enum GNUNET_DISK_OpenFlags flags, enum GNUNET_DISK_AccessPermissions perm)
Open a file.
Definition: disk.c:1237
enum GNUNET_GenericReturnValue GNUNET_DISK_file_test(const char *fil)
Check that fil corresponds to a filename (of a file that exists and that is not a directory).
Definition: disk.c:482
ssize_t GNUNET_DISK_file_write(const struct GNUNET_DISK_FileHandle *h, const void *buffer, size_t n)
Write a buffer to a file.
Definition: disk.c:686
off_t GNUNET_DISK_file_seek(const struct GNUNET_DISK_FileHandle *h, off_t offset, enum GNUNET_DISK_Seek whence)
Move the read/write pointer in a file.
Definition: disk.c:205
GNUNET_DISK_AccessPermissions
File access permissions, UNIX-style.
enum GNUNET_GenericReturnValue GNUNET_DISK_file_sync(const struct GNUNET_DISK_FileHandle *h)
Write file changes to disk.
Definition: disk.c:1427
enum GNUNET_GenericReturnValue GNUNET_DISK_file_close(struct GNUNET_DISK_FileHandle *h)
Close an open file.
Definition: disk.c:1308
ssize_t GNUNET_DISK_file_read(const struct GNUNET_DISK_FileHandle *h, void *result, size_t len)
Read the contents of a binary file into a buffer.
Definition: disk.c:622
@ GNUNET_DISK_OPEN_READ
Open the file for reading.
@ GNUNET_DISK_OPEN_WRITE
Open the file for writing.
@ GNUNET_DISK_OPEN_CREATE
Create file if it doesn't exist.
@ GNUNET_DISK_PERM_USER_READ
Owner can read.
@ GNUNET_DISK_PERM_USER_WRITE
Owner can write.
@ GNUNET_DISK_SEEK_SET
Seek an absolute position (from the start of the file).
struct GNUNET_CONTAINER_MultiShortmap * GNUNET_CONTAINER_multishortmap_create(unsigned int len, int do_not_copy_keys)
Create a multi peer map (hash map for public keys of peers).
enum GNUNET_GenericReturnValue GNUNET_CONTAINER_multishortmap_put(struct GNUNET_CONTAINER_MultiShortmap *map, const struct GNUNET_ShortHashCode *key, void *value, enum GNUNET_CONTAINER_MultiHashMapOption opt)
Store a key-value pair in the map.
int GNUNET_CONTAINER_multishortmap_iterate(struct GNUNET_CONTAINER_MultiShortmap *map, GNUNET_CONTAINER_ShortmapIterator it, void *it_cls)
Iterate over all entries in the map.
void GNUNET_CONTAINER_multishortmap_destroy(struct GNUNET_CONTAINER_MultiShortmap *map)
Destroy a hash map.
int GNUNET_CONTAINER_multishortmap_get_multiple(struct GNUNET_CONTAINER_MultiShortmap *map, const struct GNUNET_ShortHashCode *key, GNUNET_CONTAINER_ShortmapIterator it, void *it_cls)
Iterate over all entries in the map that match a particular key.
@ GNUNET_CONTAINER_MULTIHASHMAPOPTION_MULTIPLE
Allow multiple values with the same key.
#define GNUNET_log(kind,...)
#define GNUNET_memcmp(a, b)
Compare memory in a and b, where both must be of the same pointer type.
#define GNUNET_memcpy(dst, src, n)
Call memcpy() but check for n being 0 first.
GNUNET_GenericReturnValue
Named constants for return values.
@ GNUNET_OK
@ GNUNET_YES
@ GNUNET_NO
#define GNUNET_assert(cond)
Use this for fatal errors that cannot be handled.
const char * GNUNET_sh2s(const struct GNUNET_ShortHashCode *shc)
Convert a short hash value to a string (for printing debug messages).
@ GNUNET_ERROR_TYPE_WARNING
@ GNUNET_ERROR_TYPE_ERROR
@ GNUNET_ERROR_TYPE_DEBUG
#define GNUNET_new(type)
Allocate a struct or union of the given type.
#define GNUNET_free(ptr)
Wrapper around free.
@ GNUNET_MESSENGER_KIND_MISS
The miss kind.
@ GNUNET_MESSENGER_KIND_PEER
The peer kind.
enum GNUNET_GenericReturnValue is_peer_message(const struct GNUNET_MESSENGER_Message *message)
Returns whether a specific kind of message can be sent by the service without usage of a clients priv...
enum GNUNET_GenericReturnValue verify_message_by_peer(const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash, const struct GNUNET_PeerIdentity *identity)
Verifies the signature of a given message and its hash with a specific peer's identity.
void convert_peer_identity_to_id(const struct GNUNET_PeerIdentity *identity, struct GNUNET_ShortHashCode *id)
Converts a peers identity to a short hash code which can be used as id to refer to a peer via sender ...
Handle used to access files (and pipes).
A 512-bit hashcode.
struct GNUNET_MESSENGER_PeerStoreEntry * match
const struct GNUNET_PeerIdentity * requested
const struct GNUNET_MESSENGER_Message * message
struct GNUNET_MESSENGER_MessageMiss miss
struct GNUNET_MESSENGER_MessagePeer peer
enum GNUNET_MESSENGER_MessageKind kind
The kind of the message.
struct GNUNET_ShortHashCode sender_id
The senders id inside of the room the message was sent in.
struct GNUNET_PeerIdentity peer
The peer identity of a disconnected door to a room.
struct GNUNET_PeerIdentity peer
The peer identity of the sender opening a room.
struct GNUNET_MESSENGER_MessageHeader header
Header.
struct GNUNET_MESSENGER_MessageBody body
Body.
struct GNUNET_MESSENGER_Service * service
struct GNUNET_CONTAINER_MultiShortmap * peers
The identity of the host (wraps the signing key of the peer).
A 256-bit hashcode.