Skip to main content

rns_core/
constants.rs

1// =============================================================================
2// Reticulum protocol constants
3// Ported from Python RNS source
4// =============================================================================
5
6// --- From Reticulum.py ---
7
8/// Maximum transmission unit in bytes
9pub const MTU: usize = 500;
10
11/// Truncated hash length in bits
12pub const TRUNCATED_HASHLENGTH: usize = 128;
13
14/// Minimum header size: 2 (flags + hops) + 1 (context) + 16 (dest hash)
15pub const HEADER_MINSIZE: usize = 2 + 1 + (TRUNCATED_HASHLENGTH / 8);
16
17/// Maximum header size: 2 (flags + hops) + 1 (context) + 32 (transport_id + dest hash)
18pub const HEADER_MAXSIZE: usize = 2 + 1 + (TRUNCATED_HASHLENGTH / 8) * 2;
19
20/// Minimum IFAC size
21pub const IFAC_MIN_SIZE: usize = 1;
22
23/// Maximum data unit: MTU - HEADER_MAXSIZE - IFAC_MIN_SIZE
24pub const MDU: usize = MTU - HEADER_MAXSIZE - IFAC_MIN_SIZE;
25
26// --- From Identity.py ---
27
28/// Full key size in bits (256 X25519 + 256 Ed25519)
29pub const KEYSIZE: usize = 512;
30
31/// Ratchet key size in bits
32pub const RATCHETSIZE: usize = 256;
33
34/// Token overhead in bytes (16 IV + 32 HMAC)
35pub const TOKEN_OVERHEAD: usize = 48;
36
37/// AES-128 block size in bytes
38pub const AES128_BLOCKSIZE: usize = 16;
39
40/// Full hash length in bits (SHA-256)
41pub const HASHLENGTH: usize = 256;
42
43/// Signature length in bits (Ed25519)
44pub const SIGLENGTH: usize = KEYSIZE;
45
46/// Name hash length in bits
47pub const NAME_HASH_LENGTH: usize = 80;
48
49/// Derived key length in bytes
50pub const DERIVED_KEY_LENGTH: usize = 64;
51
52// --- From Packet.py ---
53
54/// Packet types
55pub const PACKET_TYPE_DATA: u8 = 0x00;
56pub const PACKET_TYPE_ANNOUNCE: u8 = 0x01;
57pub const PACKET_TYPE_LINKREQUEST: u8 = 0x02;
58pub const PACKET_TYPE_PROOF: u8 = 0x03;
59
60/// Header types
61pub const HEADER_1: u8 = 0x00;
62pub const HEADER_2: u8 = 0x01;
63
64/// Packet context types
65pub const CONTEXT_NONE: u8 = 0x00;
66pub const CONTEXT_RESOURCE: u8 = 0x01;
67pub const CONTEXT_RESOURCE_ADV: u8 = 0x02;
68pub const CONTEXT_RESOURCE_REQ: u8 = 0x03;
69pub const CONTEXT_RESOURCE_HMU: u8 = 0x04;
70pub const CONTEXT_RESOURCE_PRF: u8 = 0x05;
71pub const CONTEXT_RESOURCE_ICL: u8 = 0x06;
72pub const CONTEXT_RESOURCE_RCL: u8 = 0x07;
73pub const CONTEXT_CACHE_REQUEST: u8 = 0x08;
74pub const CONTEXT_REQUEST: u8 = 0x09;
75pub const CONTEXT_RESPONSE: u8 = 0x0A;
76pub const CONTEXT_PATH_RESPONSE: u8 = 0x0B;
77pub const CONTEXT_COMMAND: u8 = 0x0C;
78pub const CONTEXT_COMMAND_STATUS: u8 = 0x0D;
79pub const CONTEXT_CHANNEL: u8 = 0x0E;
80pub const CONTEXT_KEEPALIVE: u8 = 0xFA;
81pub const CONTEXT_LINKIDENTIFY: u8 = 0xFB;
82pub const CONTEXT_LINKCLOSE: u8 = 0xFC;
83pub const CONTEXT_LINKPROOF: u8 = 0xFD;
84pub const CONTEXT_LRRTT: u8 = 0xFE;
85pub const CONTEXT_LRPROOF: u8 = 0xFF;
86
87/// Context flag values
88pub const FLAG_SET: u8 = 0x01;
89pub const FLAG_UNSET: u8 = 0x00;
90
91/// Encrypted MDU: floor((MDU - TOKEN_OVERHEAD - KEYSIZE/16) / AES128_BLOCKSIZE) * AES128_BLOCKSIZE - 1
92pub const ENCRYPTED_MDU: usize = {
93    let numerator = MDU - TOKEN_OVERHEAD - KEYSIZE / 16;
94    (numerator / AES128_BLOCKSIZE) * AES128_BLOCKSIZE - 1
95};
96
97/// Plain MDU (same as MDU)
98pub const PLAIN_MDU: usize = MDU;
99
100/// Explicit proof length: HASHLENGTH/8 + SIGLENGTH/8 = 32 + 64 = 96
101pub const EXPL_LENGTH: usize = HASHLENGTH / 8 + SIGLENGTH / 8;
102
103/// Implicit proof length: SIGLENGTH/8 = 64
104pub const IMPL_LENGTH: usize = SIGLENGTH / 8;
105
106/// Receipt status constants
107pub const RECEIPT_FAILED: u8 = 0x00;
108pub const RECEIPT_SENT: u8 = 0x01;
109pub const RECEIPT_DELIVERED: u8 = 0x02;
110pub const RECEIPT_CULLED: u8 = 0xFF;
111
112// --- From Destination.py ---
113
114/// Destination types
115pub const DESTINATION_SINGLE: u8 = 0x00;
116pub const DESTINATION_GROUP: u8 = 0x01;
117pub const DESTINATION_PLAIN: u8 = 0x02;
118pub const DESTINATION_LINK: u8 = 0x03;
119
120/// Destination directions
121pub const DESTINATION_IN: u8 = 0x11;
122pub const DESTINATION_OUT: u8 = 0x12;
123
124// --- From Transport.py ---
125
126/// Transport types
127pub const TRANSPORT_BROADCAST: u8 = 0x00;
128pub const TRANSPORT_TRANSPORT: u8 = 0x01;
129pub const TRANSPORT_RELAY: u8 = 0x02;
130pub const TRANSPORT_TUNNEL: u8 = 0x03;
131
132/// Maximum hops
133pub const PATHFINDER_M: u8 = 128;
134
135// --- PATHFINDER algorithm ---
136
137/// Retransmit retries (total sends = PATHFINDER_R + 1)
138pub const PATHFINDER_R: u8 = 1;
139
140/// Grace period between retries (seconds)
141pub const PATHFINDER_G: f64 = 5.0;
142
143/// Random window for announce rebroadcast (seconds)
144pub const PATHFINDER_RW: f64 = 0.5;
145
146/// Path expiry = 7 days (seconds)
147pub const PATHFINDER_E: f64 = 604800.0;
148
149// --- Path expiry by interface mode ---
150
151/// Access Point path expiry = 1 day
152pub const AP_PATH_TIME: f64 = 86400.0;
153
154/// Roaming path expiry = 6 hours
155pub const ROAMING_PATH_TIME: f64 = 21600.0;
156
157// --- Announce bandwidth cap ---
158
159/// Default announce bandwidth cap (2% of interface bandwidth)
160pub const ANNOUNCE_CAP: f64 = 0.02;
161
162/// Maximum queued announces per interface
163pub const MAX_QUEUED_ANNOUNCES: usize = 16384;
164
165/// Queued announce lifetime (24 hours)
166pub const QUEUED_ANNOUNCE_LIFE: f64 = 86400.0;
167
168// --- Table limits ---
169
170/// How many local rebroadcasts of an announce is allowed
171pub const LOCAL_REBROADCASTS_MAX: u8 = 2;
172
173/// Maximum number of random blobs per destination to keep in memory
174pub const MAX_RANDOM_BLOBS: usize = 64;
175
176/// Maximum number of announce timestamps to keep per destination
177pub const MAX_RATE_TIMESTAMPS: usize = 16;
178
179/// Maximum packet hashlist size before rotation
180pub const HASHLIST_MAXSIZE: usize = 1_000_000;
181
182// --- Timeouts ---
183
184/// Reverse table entry timeout (8 minutes)
185pub const REVERSE_TIMEOUT: f64 = 480.0;
186
187/// Destination table entry timeout (7 days)
188pub const DESTINATION_TIMEOUT: f64 = 604800.0;
189
190/// Link stale time = 2 * KEEPALIVE(360) = 720 seconds
191pub const LINK_STALE_TIME: f64 = 720.0;
192
193/// Link timeout = STALE_TIME * 1.25 = 900 seconds
194pub const LINK_TIMEOUT: f64 = 900.0;
195
196/// Link establishment timeout per hop (seconds)
197pub const LINK_ESTABLISHMENT_TIMEOUT_PER_HOP: f64 = 6.0;
198
199// --- Path request ---
200
201/// Default timeout for path requests (seconds)
202pub const PATH_REQUEST_TIMEOUT: f64 = 15.0;
203
204/// Grace time before a path announcement is made (seconds)
205pub const PATH_REQUEST_GRACE: f64 = 0.4;
206
207/// Extra grace time for roaming-mode interfaces (seconds)
208pub const PATH_REQUEST_RG: f64 = 1.5;
209
210/// Minimum interval for automated path requests (seconds)
211pub const PATH_REQUEST_MI: f64 = 20.0;
212
213/// Maximum amount of unique path request tags to remember
214pub const MAX_PR_TAGS: usize = 32000;
215
216// --- Job intervals ---
217
218/// Announce check interval (seconds)
219pub const ANNOUNCES_CHECK_INTERVAL: f64 = 1.0;
220
221/// Table culling interval (seconds)
222pub const TABLES_CULL_INTERVAL: f64 = 5.0;
223
224/// Link check interval (seconds)
225pub const LINKS_CHECK_INTERVAL: f64 = 1.0;
226
227// --- Ingress Control (from Interface.py) ---
228
229/// Interface "new" period in seconds (2 hours).
230pub const IC_NEW_TIME: f64 = 7200.0;
231/// Announce frequency threshold for new interfaces (announces/sec).
232pub const IC_BURST_FREQ_NEW: f64 = 3.5;
233/// Announce frequency threshold for mature interfaces (announces/sec).
234pub const IC_BURST_FREQ: f64 = 12.0;
235/// Minimum burst active duration in seconds.
236pub const IC_BURST_HOLD: f64 = 60.0;
237/// Penalty delay before releasing held announces (seconds).
238pub const IC_BURST_PENALTY: f64 = 300.0;
239/// Interval between individual held announce releases (seconds).
240pub const IC_HELD_RELEASE_INTERVAL: f64 = 30.0;
241/// Maximum held announces per interface.
242pub const IC_MAX_HELD_ANNOUNCES: usize = 256;
243
244// --- Interface modes (from Interface.py) ---
245
246pub const MODE_FULL: u8 = 0x01;
247pub const MODE_POINT_TO_POINT: u8 = 0x02;
248pub const MODE_ACCESS_POINT: u8 = 0x03;
249pub const MODE_ROAMING: u8 = 0x04;
250pub const MODE_BOUNDARY: u8 = 0x05;
251pub const MODE_GATEWAY: u8 = 0x06;
252
253/// Interface modes that forward path requests for unknown destinations.
254/// Python: Interface.DISCOVER_PATHS_FOR = [MODE_ACCESS_POINT, MODE_GATEWAY, MODE_ROAMING]
255pub const DISCOVER_PATHS_FOR: [u8; 3] = [MODE_ACCESS_POINT, MODE_GATEWAY, MODE_ROAMING];
256
257/// Discovery path request expiry (seconds) — requests older than this are culled.
258pub const DISCOVERY_PATH_REQUEST_TIMEOUT: f64 = 15.0;
259
260// --- Path states ---
261
262pub const STATE_UNKNOWN: u8 = 0x00;
263pub const STATE_UNRESPONSIVE: u8 = 0x01;
264pub const STATE_RESPONSIVE: u8 = 0x02;
265
266// --- From Link.py ---
267
268/// Link ephemeral public key size: 32 (X25519) + 32 (Ed25519)
269pub const LINK_ECPUBSIZE: usize = 64;
270
271/// Link key size in bytes
272pub const LINK_KEYSIZE: usize = 32;
273
274/// Link MTU signalling bytes size
275pub const LINK_MTU_SIZE: usize = 3;
276
277/// Maximum keepalive interval in seconds
278pub const LINK_KEEPALIVE_MAX: f64 = 360.0;
279
280/// Minimum keepalive interval in seconds
281pub const LINK_KEEPALIVE_MIN: f64 = 5.0;
282
283/// Maximum RTT used for keepalive scaling
284pub const LINK_KEEPALIVE_MAX_RTT: f64 = 1.75;
285
286/// RTT timeout factor for stale→close transition
287pub const LINK_KEEPALIVE_TIMEOUT_FACTOR: f64 = 4.0;
288
289/// Grace period for stale→close transition
290pub const LINK_STALE_GRACE: f64 = 5.0;
291
292/// Factor to compute stale_time from keepalive
293pub const LINK_STALE_FACTOR: f64 = 2.0;
294
295/// Traffic timeout factor
296pub const LINK_TRAFFIC_TIMEOUT_FACTOR: f64 = 6.0;
297
298/// Link MDU: floor((MTU - IFAC_MIN_SIZE - HEADER_MINSIZE - TOKEN_OVERHEAD) / AES128_BLOCKSIZE) * AES128_BLOCKSIZE - 1
299pub const LINK_MDU: usize = {
300    let numerator = MTU - IFAC_MIN_SIZE - HEADER_MINSIZE - TOKEN_OVERHEAD;
301    (numerator / AES128_BLOCKSIZE) * AES128_BLOCKSIZE - 1
302};
303
304/// Link MTU bytemask (21-bit MTU field)
305pub const LINK_MTU_BYTEMASK: u32 = 0x1FFFFF;
306
307/// Link mode bytemask (3-bit mode field in upper byte)
308pub const LINK_MODE_BYTEMASK: u8 = 0xE0;
309
310// --- From Channel.py ---
311
312/// Initial window size at channel setup
313pub const CHANNEL_WINDOW: u16 = 2;
314
315/// Absolute minimum window size
316pub const CHANNEL_WINDOW_MIN: u16 = 2;
317
318/// Minimum window limit for slow links
319pub const CHANNEL_WINDOW_MIN_LIMIT_SLOW: u16 = 2;
320
321/// Minimum window limit for medium-speed links
322pub const CHANNEL_WINDOW_MIN_LIMIT_MEDIUM: u16 = 5;
323
324/// Minimum window limit for fast links
325pub const CHANNEL_WINDOW_MIN_LIMIT_FAST: u16 = 16;
326
327/// Maximum window size for slow links
328pub const CHANNEL_WINDOW_MAX_SLOW: u16 = 5;
329
330/// Maximum window size for medium-speed links
331pub const CHANNEL_WINDOW_MAX_MEDIUM: u16 = 12;
332
333/// Maximum window size for fast links
334pub const CHANNEL_WINDOW_MAX_FAST: u16 = 48;
335
336/// Minimum flexibility between window_max and window_min
337pub const CHANNEL_WINDOW_FLEXIBILITY: u16 = 4;
338
339/// Maximum sequence number
340pub const CHANNEL_SEQ_MAX: u16 = 0xFFFF;
341
342/// Sequence number modulus
343pub const CHANNEL_SEQ_MODULUS: u32 = 0x10000;
344
345/// Maximum number of send tries per envelope
346pub const CHANNEL_MAX_TRIES: u8 = 5;
347
348/// RTT threshold for fast links
349pub const CHANNEL_RTT_FAST: f64 = 0.18;
350
351/// RTT threshold for medium links
352pub const CHANNEL_RTT_MEDIUM: f64 = 0.75;
353
354/// RTT threshold for slow links
355pub const CHANNEL_RTT_SLOW: f64 = 1.45;
356
357/// Number of consecutive fast rounds to upgrade window
358pub const CHANNEL_FAST_RATE_THRESHOLD: u16 = 10;
359
360/// Channel envelope overhead: msgtype(2) + seq(2) + len(2)
361pub const CHANNEL_ENVELOPE_OVERHEAD: usize = 6;
362
363// --- From Buffer.py ---
364
365/// System message type for stream data
366pub const STREAM_DATA_MSGTYPE: u16 = 0xFF00;
367
368/// Maximum stream ID (14 bits)
369pub const STREAM_ID_MAX: u16 = 0x3FFF;
370
371/// Stream data overhead: 2 (stream header) + 6 (channel envelope)
372pub const STREAM_DATA_OVERHEAD: usize = 2 + CHANNEL_ENVELOPE_OVERHEAD;
373
374// --- From Resource.py ---
375
376/// Initial window size at beginning of transfer
377pub const RESOURCE_WINDOW: usize = 4;
378
379/// Absolute minimum window size during transfer
380pub const RESOURCE_WINDOW_MIN: usize = 2;
381
382/// Maximum window size for slow links
383pub const RESOURCE_WINDOW_MAX_SLOW: usize = 10;
384
385/// Maximum window size for very slow links
386pub const RESOURCE_WINDOW_MAX_VERY_SLOW: usize = 4;
387
388/// Maximum window size for fast links
389pub const RESOURCE_WINDOW_MAX_FAST: usize = 75;
390
391/// Global maximum window (for calculations)
392pub const RESOURCE_WINDOW_MAX: usize = RESOURCE_WINDOW_MAX_FAST;
393
394/// Minimum flexibility between window_max and window_min
395pub const RESOURCE_WINDOW_FLEXIBILITY: usize = 4;
396
397/// Sustained fast-rate rounds before enabling fast window
398/// = WINDOW_MAX_SLOW - WINDOW - 2 = 10 - 4 - 2 = 4
399pub const RESOURCE_FAST_RATE_THRESHOLD: usize = RESOURCE_WINDOW_MAX_SLOW - RESOURCE_WINDOW - 2;
400
401/// Sustained very-slow-rate rounds before capping to very slow
402pub const RESOURCE_VERY_SLOW_RATE_THRESHOLD: usize = 2;
403
404/// Fast rate threshold: 50 Kbps in bytes/sec = 50000 / 8 = 6250.0
405pub const RESOURCE_RATE_FAST: f64 = (50 * 1000) as f64 / 8.0;
406
407/// Very slow rate threshold: 2 Kbps in bytes/sec = 2000 / 8 = 250.0
408pub const RESOURCE_RATE_VERY_SLOW: f64 = (2 * 1000) as f64 / 8.0;
409
410/// Number of bytes in a map hash
411pub const RESOURCE_MAPHASH_LEN: usize = 4;
412
413/// Resource SDU = Packet.MDU (NOT ENCRYPTED_MDU)
414pub const RESOURCE_SDU: usize = MDU;
415
416/// Random hash size prepended to resource data
417pub const RESOURCE_RANDOM_HASH_SIZE: usize = 4;
418
419/// Maximum efficient resource size (1 MB - 1)
420pub const RESOURCE_MAX_EFFICIENT_SIZE: usize = 1 * 1024 * 1024 - 1;
421
422/// Maximum metadata size (16 MB - 1)
423pub const RESOURCE_METADATA_MAX_SIZE: usize = 16 * 1024 * 1024 - 1;
424
425/// Maximum auto-compress size (64 MB)
426pub const RESOURCE_AUTO_COMPRESS_MAX_SIZE: usize = 64 * 1024 * 1024;
427
428/// Part timeout factor (before RTT measured)
429pub const RESOURCE_PART_TIMEOUT_FACTOR: f64 = 4.0;
430
431/// Part timeout factor (after first RTT measured)
432pub const RESOURCE_PART_TIMEOUT_FACTOR_AFTER_RTT: f64 = 2.0;
433
434/// Proof timeout factor (reduced when awaiting proof)
435pub const RESOURCE_PROOF_TIMEOUT_FACTOR: f64 = 3.0;
436
437/// Maximum retries for part transfers
438pub const RESOURCE_MAX_RETRIES: usize = 16;
439
440/// Maximum retries for advertisement
441pub const RESOURCE_MAX_ADV_RETRIES: usize = 4;
442
443/// Sender grace time (seconds)
444pub const RESOURCE_SENDER_GRACE_TIME: f64 = 10.0;
445
446/// Processing grace for advertisement response (seconds)
447pub const RESOURCE_PROCESSING_GRACE: f64 = 1.0;
448
449/// Retry grace time (seconds)
450pub const RESOURCE_RETRY_GRACE_TIME: f64 = 0.25;
451
452/// Per-retry delay (seconds)
453pub const RESOURCE_PER_RETRY_DELAY: f64 = 0.5;
454
455/// Maximum watchdog sleep interval (seconds)
456pub const RESOURCE_WATCHDOG_MAX_SLEEP: f64 = 1.0;
457
458/// Response max grace time (seconds)
459pub const RESOURCE_RESPONSE_MAX_GRACE_TIME: f64 = 10.0;
460
461/// Advertisement overhead in bytes (fixed msgpack overhead)
462pub const RESOURCE_ADVERTISEMENT_OVERHEAD: usize = 134;
463
464/// Maximum hashmap entries per advertisement segment
465/// = floor((LINK_MDU - ADVERTISEMENT_OVERHEAD) / MAPHASH_LEN)
466pub const RESOURCE_HASHMAP_MAX_LEN: usize = (LINK_MDU - RESOURCE_ADVERTISEMENT_OVERHEAD) / RESOURCE_MAPHASH_LEN;
467
468/// Collision guard size = 2 * WINDOW_MAX + HASHMAP_MAX_LEN
469pub const RESOURCE_COLLISION_GUARD_SIZE: usize = 2 * RESOURCE_WINDOW_MAX + RESOURCE_HASHMAP_MAX_LEN;
470
471/// Hashmap not exhausted flag
472pub const RESOURCE_HASHMAP_IS_NOT_EXHAUSTED: u8 = 0x00;
473
474/// Hashmap exhausted flag
475pub const RESOURCE_HASHMAP_IS_EXHAUSTED: u8 = 0xFF;
476
477#[cfg(test)]
478mod tests {
479    use super::*;
480
481    #[test]
482    fn test_derived_constants() {
483        // MTU = 500
484        assert_eq!(MTU, 500);
485
486        // HEADER_MINSIZE = 2 + 1 + 16 = 19
487        assert_eq!(HEADER_MINSIZE, 19);
488
489        // HEADER_MAXSIZE = 2 + 1 + 32 = 35
490        assert_eq!(HEADER_MAXSIZE, 35);
491
492        // MDU = 500 - 35 - 1 = 464
493        assert_eq!(MDU, 464);
494
495        // ENCRYPTED_MDU = floor((464 - 48 - 32) / 16) * 16 - 1 = floor(384/16)*16 - 1 = 24*16 - 1 = 383
496        assert_eq!(ENCRYPTED_MDU, 383);
497
498        // PLAIN_MDU = MDU = 464
499        assert_eq!(PLAIN_MDU, 464);
500
501        // EXPL_LENGTH = 32 + 64 = 96
502        assert_eq!(EXPL_LENGTH, 96);
503
504        // IMPL_LENGTH = 64
505        assert_eq!(IMPL_LENGTH, 64);
506
507        // NAME_HASH_LENGTH = 80 bits = 10 bytes
508        assert_eq!(NAME_HASH_LENGTH / 8, 10);
509
510        // KEYSIZE = 512 bits = 64 bytes
511        assert_eq!(KEYSIZE / 8, 64);
512
513        // SIGLENGTH = 512 bits = 64 bytes
514        assert_eq!(SIGLENGTH / 8, 64);
515
516        // TRUNCATED_HASHLENGTH = 128 bits = 16 bytes
517        assert_eq!(TRUNCATED_HASHLENGTH / 8, 16);
518    }
519
520    #[test]
521    fn test_link_constants() {
522        assert_eq!(LINK_ECPUBSIZE, 64);
523        assert_eq!(LINK_MTU_SIZE, 3);
524        // LINK_MDU = floor((500 - 1 - 19 - 48) / 16) * 16 - 1 = floor(432/16)*16 - 1 = 27*16 - 1 = 431
525        assert_eq!(LINK_MDU, 431);
526        assert_eq!(CHANNEL_ENVELOPE_OVERHEAD, 6);
527        assert_eq!(STREAM_DATA_OVERHEAD, 8);
528        assert_eq!(STREAM_ID_MAX, 0x3FFF);
529    }
530
531    #[test]
532    fn test_transport_constants() {
533        // PATHFINDER_E = 7 days in seconds
534        assert_eq!(PATHFINDER_E, 60.0 * 60.0 * 24.0 * 7.0);
535
536        // AP_PATH_TIME = 1 day
537        assert_eq!(AP_PATH_TIME, 60.0 * 60.0 * 24.0);
538
539        // ROAMING_PATH_TIME = 6 hours
540        assert_eq!(ROAMING_PATH_TIME, 60.0 * 60.0 * 6.0);
541
542        // LINK_STALE_TIME = 2 * 360
543        assert_eq!(LINK_STALE_TIME, 720.0);
544
545        // LINK_TIMEOUT = STALE_TIME * 1.25
546        assert_eq!(LINK_TIMEOUT, LINK_STALE_TIME * 1.25);
547
548        // REVERSE_TIMEOUT = 8 minutes
549        assert_eq!(REVERSE_TIMEOUT, 8.0 * 60.0);
550
551        // DESTINATION_TIMEOUT = 7 days
552        assert_eq!(DESTINATION_TIMEOUT, 60.0 * 60.0 * 24.0 * 7.0);
553    }
554
555    #[test]
556    fn test_resource_constants() {
557        // SDU = MDU = 464 (NOT ENCRYPTED_MDU)
558        assert_eq!(RESOURCE_SDU, 464);
559        assert_eq!(RESOURCE_SDU, MDU);
560
561        // FAST_RATE_THRESHOLD = WINDOW_MAX_SLOW - WINDOW - 2 = 10 - 4 - 2 = 4
562        assert_eq!(RESOURCE_FAST_RATE_THRESHOLD, 4);
563
564        // RATE_FAST = 50000 / 8 = 6250.0
565        assert_eq!(RESOURCE_RATE_FAST, 6250.0);
566
567        // RATE_VERY_SLOW = 2000 / 8 = 250.0
568        assert_eq!(RESOURCE_RATE_VERY_SLOW, 250.0);
569
570        // HASHMAP_MAX_LEN = floor((431 - 134) / 4) = floor(297/4) = 74
571        assert_eq!(RESOURCE_HASHMAP_MAX_LEN, 74);
572
573        // COLLISION_GUARD_SIZE = 2 * 75 + 74 = 224
574        assert_eq!(RESOURCE_COLLISION_GUARD_SIZE, 224);
575
576        // Window constants
577        assert_eq!(RESOURCE_WINDOW, 4);
578        assert_eq!(RESOURCE_WINDOW_MIN, 2);
579        assert_eq!(RESOURCE_WINDOW_MAX_SLOW, 10);
580        assert_eq!(RESOURCE_WINDOW_MAX_VERY_SLOW, 4);
581        assert_eq!(RESOURCE_WINDOW_MAX_FAST, 75);
582        assert_eq!(RESOURCE_WINDOW_MAX, 75);
583        assert_eq!(RESOURCE_WINDOW_FLEXIBILITY, 4);
584    }
585}