GNUnet 0.21.1
gnunet-service-messenger_message_recv.c
Go to the documentation of this file.
1/*
2 This file is part of GNUnet.
3 Copyright (C) 2020--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
31
32static void
34 struct GNUNET_MESSENGER_SrvTunnel *tunnel,
35 struct GNUNET_MESSENGER_MemberSession *session,
37{
38 if (session->prev)
39 forward_about_members (room, tunnel, session->prev, map);
40
41 struct GNUNET_MESSENGER_MessageStore *message_store =
43 struct GNUNET_MESSENGER_ListMessage *element;
44
45 for (element = session->messages.head; element; element = element->next)
46 {
48 &(element->hash)))
49 continue;
50
52 NULL,
55 "Forwarding of session message could be duplicated!\n");
56
57 const struct GNUNET_MESSENGER_Message *message = get_store_message (
58 message_store, &(element->hash));
59
60 if (message)
61 forward_tunnel_message (tunnel, message, &(element->hash));
62 }
63}
64
65
68 const struct GNUNET_CRYPTO_PublicKey *public_key,
69 struct GNUNET_MESSENGER_MemberSession *session)
70{
71 struct GNUNET_MESSENGER_SrvTunnel *tunnel = cls;
72
74 return GNUNET_YES;
75
78
79 forward_about_members (tunnel->room, tunnel, session, map);
80
82 return GNUNET_YES;
83}
84
85
88 struct GNUNET_MESSENGER_SrvTunnel *tunnel,
89 const struct GNUNET_MESSENGER_Message *message,
90 const struct GNUNET_HashCode *hash)
91{
92 const uint32_t version = get_tunnel_messenger_version (tunnel);
93
95 message->body.info.
96 messenger_version))
97 {
98 disconnect_tunnel (tunnel);
99 return GNUNET_NO;
100 }
101
102 if (version == get_tunnel_messenger_version (tunnel))
103 return GNUNET_NO;
104
105 if (room->host)
107 room->service));
108
109 struct GNUNET_PeerIdentity peer;
110 get_tunnel_peer_identity (tunnel, &peer);
111
112 if (GNUNET_YES != contains_list_tunnels (&(room->basement), &peer))
113 {
114 struct GNUNET_MESSENGER_MessageStore *message_store =
116
117 struct GNUNET_MESSENGER_ListTunnel *element;
118 for (element = room->basement.head; element; element = element->next)
119 {
120 if (! element->hash)
121 continue;
122
123 const struct GNUNET_MESSENGER_Message *message = get_store_message (
124 message_store, element->hash);
125
126 if (message)
127 forward_tunnel_message (tunnel, message, element->hash);
128 }
129 }
130
131 if (GNUNET_YES != contains_list_tunnels (&(room->basement), &peer))
132 {
133 struct GNUNET_MESSENGER_MemberStore *member_store =
135
136 iterate_store_members (member_store, iterate_forward_members, tunnel);
137 }
138
140 return GNUNET_NO;
141}
142
143
146 struct GNUNET_MESSENGER_SrvTunnel *tunnel,
147 const struct GNUNET_MESSENGER_Message *message,
148 const struct GNUNET_HashCode *hash)
149{
150 struct GNUNET_PeerIdentity peer;
151 GNUNET_PEER_resolve (tunnel->peer, &peer);
152
153 if (0 == GNUNET_memcmp (&peer, &(message->body.peer.peer)))
154 {
155 if (! tunnel->peer_message)
156 tunnel->peer_message = GNUNET_new (struct GNUNET_HashCode);
157
158 GNUNET_memcpy (tunnel->peer_message, &hash, sizeof(hash));
159 }
160
161 update_to_list_tunnels (&(room->basement), &(message->body.peer.peer), hash);
162 return GNUNET_YES;
163}
164
165
168 struct GNUNET_MESSENGER_SrvTunnel *tunnel,
169 const struct GNUNET_MESSENGER_Message *message,
170 const struct GNUNET_HashCode *hash)
171{
173
174 if ((GNUNET_YES == service->auto_routing) &&
175 (service->min_routers > count_of_tunnels (&(room->basement))))
176 open_srv_room (room, NULL);
177
178 return GNUNET_YES;
179}
180
181
182static void
184 struct GNUNET_MESSENGER_SrvRoom *room,
185 const struct GNUNET_MESSENGER_Message *message,
186 const struct GNUNET_HashCode *hash)
187{
188 struct GNUNET_MESSENGER_SrvTunnel *tunnel = cls;
189
190 if (! message)
191 {
192 struct GNUNET_MESSENGER_OperationStore *operation_store =
194
196 operation_store,
197 hash,
200 );
201 }
202 else
203 forward_tunnel_message (tunnel, message, hash);
204}
205
206
207/*
208 * Function returns GNUNET_NO to drop forwarding the request.
209 * It will only be forwarded if it can't be answered!
210 */
213 struct GNUNET_MESSENGER_SrvTunnel *tunnel,
214 const struct GNUNET_MESSENGER_Message *message,
215 const struct GNUNET_HashCode *hash)
216{
217 struct GNUNET_MESSENGER_MemberStore *member_store =
219 struct GNUNET_MESSENGER_Member *member = get_store_member_of (member_store,
220 message);
221
222 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Request for message (%s)\n",
223 GNUNET_h2s (hash));
224
225 if (! member)
226 return GNUNET_NO;
227
229 member, message, hash);
230
231 if ((! session) || (GNUNET_YES != check_member_session_history (session,
232 &(message->
233 body.request
234 .hash),
235 GNUNET_NO)))
236 return GNUNET_NO;
237
239 &(message->body.request.hash),
240 session, callback_found_message,
241 tunnel))
242 return GNUNET_YES;
243
244 return GNUNET_NO;
245}
static struct GNUNET_SERVICE_Handle * service
Handle to our service instance.
size_t count_of_tunnels(const struct GNUNET_MESSENGER_ListTunnels *tunnels)
Returns the count of peers in a list (typically from the basement of a room).
struct GNUNET_MESSENGER_MemberSession * get_member_session_of(struct GNUNET_MESSENGER_Member *member, const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash)
Returns the member session of a member using a public key which can verify the signature of a given m...
enum GNUNET_GenericReturnValue check_member_session_history(const struct GNUNET_MESSENGER_MemberSession *session, const struct GNUNET_HashCode *hash, enum GNUNET_GenericReturnValue ownership)
Checks the history of a session for a specific message which is identified by its hash and if the own...
enum GNUNET_GenericReturnValue is_member_session_completed(const struct GNUNET_MESSENGER_MemberSession *session)
Returns if the given member session has been completed.
struct GNUNET_MESSENGER_Member * get_store_member_of(struct GNUNET_MESSENGER_MemberStore *store, const struct GNUNET_MESSENGER_Message *message)
Returns the member of a store using a sender id of a given message.
int iterate_store_members(struct GNUNET_MESSENGER_MemberStore *store, GNUNET_MESSENGER_MemberIteratorCallback it, void *cls)
Iterate through all member sessions currently connected to the members of the given member store and ...
struct GNUNET_MESSENGER_Message * create_message_info(struct GNUNET_MESSENGER_Service *service)
Creates and allocates a new info message containing the hosts service peer identity and version.
enum GNUNET_GenericReturnValue recv_message_request(struct GNUNET_MESSENGER_SrvRoom *room, struct GNUNET_MESSENGER_SrvTunnel *tunnel, const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash)
Handles a received request message by checking for the requested message and forwarding it back if th...
enum GNUNET_GenericReturnValue recv_message_miss(struct GNUNET_MESSENGER_SrvRoom *room, struct GNUNET_MESSENGER_SrvTunnel *tunnel, const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash)
Handles a received miss message to react to activity in the basement of a room.
static void callback_found_message(void *cls, struct GNUNET_MESSENGER_SrvRoom *room, const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash)
static void forward_about_members(struct GNUNET_MESSENGER_SrvRoom *room, struct GNUNET_MESSENGER_SrvTunnel *tunnel, struct GNUNET_MESSENGER_MemberSession *session, struct GNUNET_CONTAINER_MultiHashMap *map)
enum GNUNET_GenericReturnValue recv_message_info(struct GNUNET_MESSENGER_SrvRoom *room, struct GNUNET_MESSENGER_SrvTunnel *tunnel, const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash)
Handles a received info message to change the current member id to the one generated by the host conn...
enum GNUNET_GenericReturnValue recv_message_peer(struct GNUNET_MESSENGER_SrvRoom *room, struct GNUNET_MESSENGER_SrvTunnel *tunnel, const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash)
Handles a received peer message to link it to its origin tunnel if the peer identity matches.
static enum GNUNET_GenericReturnValue iterate_forward_members(void *cls, const struct GNUNET_CRYPTO_PublicKey *public_key, struct GNUNET_MESSENGER_MemberSession *session)
const struct GNUNET_MESSENGER_Message * get_store_message(struct GNUNET_MESSENGER_MessageStore *store, const struct GNUNET_HashCode *hash)
Returns the message from a message store matching a given hash.
enum GNUNET_GenericReturnValue use_store_operation(struct GNUNET_MESSENGER_OperationStore *store, const struct GNUNET_HashCode *hash, enum GNUNET_MESSENGER_OperationType type, struct GNUNET_TIME_Relative delay)
Tries to use an operation under a given hash in a specific operation store.
struct GNUNET_MESSENGER_MessageStore * get_srv_room_message_store(struct GNUNET_MESSENGER_SrvRoom *room)
Returns the used message store of a given room.
void check_srv_room_peer_status(struct GNUNET_MESSENGER_SrvRoom *room, struct GNUNET_MESSENGER_SrvTunnel *tunnel)
Checks the current state of opening a given room from this peer and re-publishes it if necessary to a...
enum GNUNET_GenericReturnValue open_srv_room(struct GNUNET_MESSENGER_SrvRoom *room, struct GNUNET_MESSENGER_SrvHandle *handle)
Tries to open a room for a given handle.
enum GNUNET_GenericReturnValue request_srv_room_message(struct GNUNET_MESSENGER_SrvRoom *room, const struct GNUNET_HashCode *hash, const struct GNUNET_MESSENGER_MemberSession *session, GNUNET_MESSENGER_MessageRequestCallback callback, void *cls)
Requests a message from a room identified by a given hash.
struct GNUNET_MESSENGER_OperationStore * get_srv_room_operation_store(struct GNUNET_MESSENGER_SrvRoom *room)
Returns the used operation store of a given room.
struct GNUNET_MESSENGER_MemberStore * get_srv_room_member_store(struct GNUNET_MESSENGER_SrvRoom *room)
Returns the used member store of a given room.
#define GNUNET_MESSENGER_REQUEST_DELAY
void forward_tunnel_message(struct GNUNET_MESSENGER_SrvTunnel *tunnel, const struct GNUNET_MESSENGER_Message *message, const struct GNUNET_HashCode *hash)
Forwards a given message with a known hash through a tunnel.
enum GNUNET_GenericReturnValue update_tunnel_messenger_version(struct GNUNET_MESSENGER_SrvTunnel *tunnel, uint32_t version)
Updates the messenger version of the tunnel to a given version if it is compatible to the running pee...
uint32_t get_tunnel_messenger_version(const struct GNUNET_MESSENGER_SrvTunnel *tunnel)
Returns the current messenger version the peer connected via a given tunnel has reported to be using ...
void get_tunnel_peer_identity(const struct GNUNET_MESSENGER_SrvTunnel *tunnel, struct GNUNET_PeerIdentity *peer)
Writes the peer identity of the peer connected via tunnel to this peer into the peer parameter.
void disconnect_tunnel(struct GNUNET_MESSENGER_SrvTunnel *tunnel)
Disconnects and unbinds a channel from a tunnel.
enum GNUNET_GenericReturnValue send_tunnel_message(struct GNUNET_MESSENGER_SrvTunnel *tunnel, void *handle, struct GNUNET_MESSENGER_Message *message)
Sends a message by packing it automatically into an envelope and passing it through the tunnel.
enum GNUNET_GenericReturnValue GNUNET_CONTAINER_multihashmap_contains(const struct GNUNET_CONTAINER_MultiHashMap *map, const struct GNUNET_HashCode *key)
Check if the map contains any value under the given key (including values that are NULL).
enum GNUNET_GenericReturnValue GNUNET_CONTAINER_multihashmap_put(struct GNUNET_CONTAINER_MultiHashMap *map, const struct GNUNET_HashCode *key, void *value, enum GNUNET_CONTAINER_MultiHashMapOption opt)
Store a key-value pair in the map.
void GNUNET_CONTAINER_multihashmap_destroy(struct GNUNET_CONTAINER_MultiHashMap *map)
Destroy a hash map.
struct GNUNET_CONTAINER_MultiHashMap * GNUNET_CONTAINER_multihashmap_create(unsigned int len, int do_not_copy_keys)
Create a multi hash map.
@ GNUNET_CONTAINER_MULTIHASHMAPOPTION_UNIQUE_FAST
, ' bother checking if a value already exists (faster than GNUNET_CONTAINER_MULTIHASHMAPOPTION_UNIQUE...
#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
const char * GNUNET_h2s(const struct GNUNET_HashCode *hc)
Convert a hash value to a string (for printing debug messages).
@ GNUNET_ERROR_TYPE_WARNING
@ GNUNET_ERROR_TYPE_DEBUG
#define GNUNET_new(type)
Allocate a struct or union of the given type.
void GNUNET_PEER_resolve(GNUNET_PEER_Id id, struct GNUNET_PeerIdentity *pid)
Convert an interned PID to a normal peer identity.
Definition: peer.c:220
void update_to_list_tunnels(struct GNUNET_MESSENGER_ListTunnels *tunnels, const struct GNUNET_PeerIdentity *peer, const struct GNUNET_HashCode *hash)
Updates a specific peer from a tunnel in the list.
enum GNUNET_GenericReturnValue contains_list_tunnels(struct GNUNET_MESSENGER_ListTunnels *tunnels, const struct GNUNET_PeerIdentity *peer)
Tests linearly if the list of tunnels peer identities contains a specific peer identity and returns G...
static struct GNUNET_CONTAINER_MultiPeerMap * map
Peermap of PeerIdentities to "struct PeerEntry" (for fast lookup).
Definition: peer.c:63
Internal representation of the hash map.
An identity key as per LSD0001.
A 512-bit hashcode.
struct GNUNET_MESSENGER_ListMessage * next
struct GNUNET_MESSENGER_ListMessage * head
struct GNUNET_MESSENGER_ListTunnel * next
struct GNUNET_MESSENGER_ListTunnel * head
struct GNUNET_MESSENGER_MemberSession * prev
struct GNUNET_MESSENGER_ListMessages messages
struct GNUNET_MESSENGER_MessageRequest request
struct GNUNET_MESSENGER_MessagePeer peer
struct GNUNET_MESSENGER_MessageInfo info
struct GNUNET_PeerIdentity peer
The peer identity of the sender opening a room.
struct GNUNET_HashCode hash
The hash of the requested message.
struct GNUNET_MESSENGER_MessageBody body
Body.
struct GNUNET_MESSENGER_Service * service
struct GNUNET_MESSENGER_ListTunnels basement
struct GNUNET_MESSENGER_SrvHandle * host
struct GNUNET_MESSENGER_SrvRoom * room
The identity of the host (wraps the signing key of the peer).