desru/
lib.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
//! # Rust Simulation Framework
//!
//! This crate provides a flexible framework for simulating discrete-event systems (DES).
//! It allows users to schedule, manage, and execute events over time, making it suitable for
//! simulations of various systems such as queueing networks, resource allocation systems, and more.
//!
//! The core components of the framework are:
//! 
//! - [`Event`]: A struct representing a single event in the simulation, holding its scheduled time, an action, and context.
//! - [`EventScheduler`]: A struct that manages the execution of events, prioritizing those scheduled to run earlier.
//!
//! ## Key Features
//!
//! - **Event Scheduling:** Schedule events at specific times or after delays.
//! - **Event Logging:** Keep a log of all events executed and their outcomes for later analysis.
//! - **Flexible Execution:** Run the scheduler until a certain condition is met, such as reaching a max time.
//! - **Contextual Information:** Attach metadata (context) to each event for richer event processing.
//! 
//! ## Example: Scheduling an Event
//!
//! Below is a simple example demonstrating how to create an event, schedule it in the `EventScheduler`, and run the simulation.
//!
//! ```rust
//! use desru::{Event, EventScheduler};
//!
//! fn main() {
//!     let mut scheduler = EventScheduler::new();
//!     let mut event = Event::new(0.0,
//!                                Some(Box::new(|scheduler| Some("Executed".to_string()))),
//!                                None);
//!     scheduler.schedule(event);
//!     scheduler.run_until_max_time(10.0);
//! }
//! ```
//! ## Example: SimPy's Simple Car
//!
//! The SimPy Python package provides a [simple car example](https://simpy.readthedocs.io/en/latest/simpy_intro/basic_concepts.html#our-first-process ) as a first example. You can see from the following state diagram that a car immediately parks, and then alternates between parking and driving at fixed intervals of time. The duration of the car being parked is 5 units of time, while the duration of the car driving is only 2 units of time.
//!
//! <pre>
#![doc = mermaid!("car_state.mmd")]
//! </pre>
//!
//! In this example we implement the same process, but using `desru`.
//!
//!```rust
//!use desru::{Event, EventScheduler};
//!
//!fn car(scheduler: &mut EventScheduler) {
//!    // Start by parking, which repeats in a loop
//!    println!("Start parking at {}", scheduler.current_time);
//!
//!    let parking_duration = 5.0;
//!    let trip_duration = 2.0;
//!
//!    // Schedule event to start driving after parking
//!    scheduler.schedule(Event::new(
//!        scheduler.current_time + parking_duration,
//!        Some(Box::new(move |scheduler: &mut EventScheduler| {
//!            println!("Start driving at {}", scheduler.current_time);
//!
//!            // Schedule event to start parking after driving
//!            scheduler.schedule(Event::new(
//!                scheduler.current_time + trip_duration,
//!                Some(Box::new(move |scheduler: &mut EventScheduler| {
//!                    car(scheduler); // Recurse to repeat the cycle
//!                    None // No string context needed, returning None
//!                })),
//!                None,
//!            ));
//!            None // No string context needed, returning None
//!        })),
//!        None,
//!    ));
//!}
//!
//!fn main() {
//!    // Initialize the event scheduler
//!    let mut scheduler = EventScheduler::new();
//!
//!    // Start the car simulation
//!    car(&mut scheduler);
//!
//!    // Run the scheduler for a max time of 15 units
//!    scheduler.run_until_max_time(15.0);
//!}
//!```
//! You should expect to see this output:
//!
//!```bash
//! Start parking at 0
//! Start driving at 5
//! Start parking at 7
//! Start driving at 12
//! Start parking at 14
//! ```
//! The above implementation was intended to try to do what the SimPy example does: have multiple
//! events scheduled within a single function. However, we can refactor the code using multiple functions that schedule each other. Further, for this simple example at least, we don't need to keep defining the durations to the same constant values in every function call so we can also define those as global constants. I think it is easier to read this way:
//!
//!```rust
//! use desru::{Event, EventScheduler};
//!
//! const MAX_TIME: f64 = 15.0;
//! const PARK_DURATION: f64 =  5.0;
//! const DRIVE_DURATION: f64 =  2.0;
//!
//! fn car(scheduler: &mut EventScheduler) {
//!    park(scheduler);
//!}
//!
//!fn park(scheduler: &mut EventScheduler) {
//!    println!("Start parking at {}", scheduler.current_time);
//!    scheduler.schedule(Event::new(
//!        scheduler.current_time + PARK_DURATION,
//!        Some(Box::new(move |scheduler: &mut EventScheduler| {
//!            drive(scheduler);
//!            None
//!        })),
//!        None,
//!    ));
//!}
//!
//!fn drive(scheduler: &mut EventScheduler) {
//!    println!("Start driving at {}", scheduler.current_time);
//!    scheduler.schedule(Event::new(
//!        scheduler.current_time + DRIVE_DURATION,
//!        Some(Box::new(move |scheduler: &mut EventScheduler| {
//!            park(scheduler); // Return to parking
//!            None
//!        })),
//!        None,
//!    ));
//!}
//!
//!fn main() {
//!    // Initialize the event scheduler
//!    let mut scheduler = EventScheduler::new();
//!
//!    // Start the car simulation
//!    car(&mut scheduler);
//!
//!    // Run the scheduler for a max time
//!    scheduler.run_until_max_time(MAX_TIME);
//!}
//!```
//! For those coming from a SimPy background, what the above implementations in `desru` show us is
//! that it will sometimes be simpler to factorize out events into separate functions.
//!
//! ## Example: SimPy's Objected Oriented Car ("Process Interaction")
//!
//! This example is an implementation of SimPy's [*Waiting for a Process*](https://simpy.readthedocs.io/en/latest/simpy_intro/process_interaction.html#waiting-for-a-process) example. It it highly similar to the previous car example, however a class is used along with multiple methods.
//!
//! While Rust does not have classes in the usual sense, it does support its own flavor of [OOP](https://doc.rust-lang.org/beta/book/ch17-00-oop.html). While Rust's OOP does not have inheritance, for which Rust has some useful alternatives, we do not need it to implement an object-oriented implementation of SimPy's example.
//!
//! ```rust
//! use desru::{Event, EventScheduler};
//!
//! struct Car<'a> {
//!    scheduler: &'a mut EventScheduler,
//!}
//!
//!impl<'a> Car<'a> {
//!    fn new(scheduler: &'a mut EventScheduler) -> Self {
//!        let mut car = Car { scheduler };
//!        car.start();
//!        car
//!    }
//!
//!    fn start(&mut self) {
//!        // Start the car process
//!        self.scheduler.schedule(Event::new(
//!            self.scheduler.current_time,
//!            Some(Box::new(move |scheduler: &mut EventScheduler| {
//!                let mut car_instance = Car { scheduler };  // Create a car instance with a mutable reference
//!                car_instance.run(); // Call the run method to start the car process
//!                None
//!            })),
//!            None,
//!        ));
//!    }
//!
//!    fn run(&mut self) {
//!        // Car running process
//!        println!("Start parking and charging at {}", self.scheduler.current_time);
//!        let charge_duration = 5.0;
//!
//!        // Schedule the charge process
//!        self.scheduler.schedule(Event::new(
//!            self.scheduler.current_time + charge_duration,
//!            Some(Box::new(move |scheduler: &mut EventScheduler| {
//!                let mut car_instance = Car { scheduler };
//!                car_instance.drive(); // After charging, start driving
//!                None
//!            })),
//!            None,
//!        ));
//!    }
//!
//!    fn drive(&mut self) {
//!        // Start driving process
//!        println!("Start driving at {}", self.scheduler.current_time);
//!        let trip_duration = 2.0;
//!
//!        // Schedule the next run cycle (parking and charging) after the trip duration
//!        self.scheduler.schedule(Event::new(
//!            self.scheduler.current_time + trip_duration,
//!            Some(Box::new(move |scheduler: &mut EventScheduler| {
//!                let mut car_instance = Car { scheduler };
//!                car_instance.run(); // Repeat the cycle
//!                None
//!            })),
//!            None,
//!        ));
//!    }
//!}
//!
//!fn main() {
//!    // Initialize the event scheduler
//!    let mut scheduler = EventScheduler::new();
//!
//!    // Create a car instance
//!    let _car = Car::new(&mut scheduler);
//!
//!    // Run the scheduler for a max time of 15 units
//!    scheduler.run_until_max_time(15.0);
//!}
//!```
//!
//! ## Core Structs
//! - [`Event`]: Defines the core event object used to represent scheduled actions.
//! - [`EventScheduler`]: Manages the execution of events over simulated time.
//!
//! ## Customization
//! You can extend the framework by adding custom event types or adjusting how events are scheduled.
//!
//! ## Design Goals
//! This framework is designed to be:
//! 
//! - **Simple to use:** By providing straightforward methods to schedule and run events.
//! - **Flexible:** Allowing users to define custom event behaviors.
//! - **Efficient:** Using a priority queue to ensure events are executed in the correct order.
//!
//! ## Design Non-Goals
//! This framework is only for the very most core components for DES, and will not provide
//! implementations of simulation tools.
//!
//!
//! ## Future Directions
//!
//! Planned features include:
//! - **Advanced Scheduling Policies:** Adding support for different event scheduling strategies.
//! - **Performance Optimizations:** Improving efficiency for larger simulations.
//!
//! ## Crate Overview
//! This crate provides essential components for event-driven simulations in Rust. Starting
//! with events and a scheduler, and abstractions that provide weak coupling with state, this crate
//! can be used to implement most conceivable discrete event simulations.

///////////////////////////////////
// CONTENTS:                    //
// 0. IMPORTS                  //
// 1. DEFINE EVENT STRUCT     //
// 2. DEFINE EVENT SCHEDULER //
// 3. STOP CONDITIONS       //
// 4. UNIT TESTS           //
////////////////////////////

/////////////////
// $0 IMPORTS //
///////////////

use simple_mermaid::mermaid;
use std::collections::{BinaryHeap, HashMap};
use std::cmp::Ordering;
use std::fmt;

/////////////////////////////
// $1 DEFINE EVENT STRUCT //
///////////////////////////

/// Represents an event in the simulation.
///
/// Each event has a scheduled time (`time`), an associated action (`action`) 
/// that will be executed when the event occurs, and a `context` for storing 
/// key-value pairs of additional information about the event.
///
/// # Fields
/// - `time`: The time at which the event is scheduled to run.
/// - `action`: A closure that represents the task to be performed when the event is triggered.
///   It returns an `Option<String>` to optionally pass a result when executed.
/// - `context`: A map containing any extra contextual information as key-value pairs (both as `String`).
/// - `active`: A boolean indicating if the event is active. If false, the event will not run.
pub struct Event {
    pub time: f64,
    pub action: Box<dyn FnMut(&mut EventScheduler) -> Option<String>>,
    pub context: HashMap<String, String>,
    pub active: bool,
    }

// Implement debug for using {:?}
impl fmt::Debug for Event {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        f.debug_struct("Event")
         .field("time", &self.time)
         .field("active", &self.active)
         .field("context", &self.context)
         .finish()
    }
}

// Implement Clone manually for Event
impl Clone for Event {
    /// Creates a clone of the event.
    ///
    /// **Note**: The action closure is not cloned, since closures cannot be cloned. A placeholder
    /// action that returns `None` is used in the cloned event. The `context` and other fields are 
    /// copied as usual.
    fn clone(&self) -> Self {
        Event {
            time: self.time,
            action: Box::new(|_| None), // Placeholder action for clone.
            context: self.context.clone(),
            active: self.active,
            }
        }
    }

// Implement Event methods
impl Event {
    /// Creates a new `Event` with the given time, action, and context.
    ///
    /// # Parameters
    /// - `time`: The time when the event should be executed.
    /// - `action`: An optional closure representing the event's task. Defaults to a no-op (returns `None`).
    /// - `context`: An optional `HashMap` of context information. Defaults to an empty map.
    ///
    /// # Returns
    /// A new `Event` instance.
    ///
    /// # Example
    /// ```
    /// use desru::{Event};
    ///
    /// let event = Event::new(5.0, None, None);
    /// assert_eq!(event.time, 5.0);
    /// ```
    pub fn new(time: f64, action: Option<Box<dyn FnMut(&mut EventScheduler) -> Option<String>>>, context: Option<HashMap<String, String>>) -> Self {
        Event {
            time,
            action: action.unwrap_or_else(|| Box::new(|_| None)),
            context: context.unwrap_or_default(),
            active: true,
            }
    }

    /// Executes the action of the event if it is active.
    ///
    /// # Returns
    /// - `Some(String)`: The result of the action if the event is active and the action produces a result.
    /// - `None`: If the event is inactive or the action produces no result.
    ///
    /// # Example
    /// ```
    /// use desru::{Event, EventScheduler};
    ///
    /// let mut scheduler = EventScheduler::new();
    /// let mut event = Event::new(0.0,
    ///                            Some(Box::new(|scheduler| Some("Executed".to_string()))),
    ///                            None);
    /// assert_eq!(event.run(&mut scheduler), Some("Executed".to_string()));
    /// ```
    pub fn run(&mut self, scheduler: &mut EventScheduler) -> Option<String> {
        if self.active {
           (self.action)(scheduler)
        } else {
            None
        }
    }
}

// Implement ordering traits for Event to use in BinaryHeap
impl PartialEq for Event {
    /// Checks if two events are equal based on their scheduled time.
    fn eq(&self, other: &Self) -> bool {
        self.time == other.time
    }
}

impl Eq for Event {}

impl PartialOrd for Event {
    /// Compares two events based on their time, in reverse order, for use in a max-heap.
    ///
    /// This allows events with earlier times to be processed first.
    fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
        Some(self.cmp(other))
    }
}

impl Ord for Event {
    /// Defines the ordering between two events.
    ///
    /// The event with the earlier time has higher priority, enabling
    /// the `BinaryHeap` to act as a priority queue.
    fn cmp(&self, other: &Self) -> Ordering {
        other.time.partial_cmp(&self.time).unwrap()
    }
}

////////////////////////////////
// $2 DEFINE EVENT SCHEDULER //
//////////////////////////////

/// Manages and schedules events using a priority queue.
///
/// The `EventScheduler` executes events based on their scheduled time, maintaining an event log
/// and allowing for conditional execution (e.g., stop after a certain time or when certain criteria are met).
///
/// # Fields
/// - `current_time`: The current time in the simulation, updated as events are processed.
/// - `event_queue`: A binary heap used as a priority queue for storing scheduled events.
/// - `event_log`: A log that stores all events executed and their results.
pub struct EventScheduler {
    pub current_time: f64,
    pub event_queue: BinaryHeap<Event>,
    pub event_log: Vec<(Event, Option<String>)>,
}

// Implement EventScheduler methods
impl EventScheduler {
    /// Creates a new `EventScheduler` with an empty event queue.
    ///
    /// # Returns
    /// A new `EventScheduler` instance.
    ///
    /// # Example
    /// ```
    /// use desru::{EventScheduler};
    ///
    /// let scheduler = EventScheduler::new();
    /// assert_eq!(scheduler.current_time, 0.0);
    /// ```
    pub fn new() -> Self {
        EventScheduler {
            current_time: 0.0,
            event_queue: BinaryHeap::new(),
            event_log: Vec::new(),
        }
    }

    /// Schedules a new event by adding it to the event queue.
    ///
    /// # Parameters
    /// - `event`: The event to be scheduled.
    ///
    /// # Example
    /// ```
    /// use desru::{Event, EventScheduler};
    ///
    /// let mut scheduler = EventScheduler::new();
    /// let event = Event::new(5.0, None, None);
    /// scheduler.schedule(event);
    /// ```
    pub fn schedule(&mut self, event: Event) {
        self.event_queue.push(event);
    }

    /// Schedules a timeout event to be executed after a specified delay.
    ///
    /// # Parameters
    /// - `delay`: The amount of time after which the event should occur.
    /// - `action`: The action to be executed (optional).
    /// - `context`: Additional context for the event (optional).
    ///
    /// # Example
    /// ```
    /// use desru::EventScheduler;
    ///
    /// let mut scheduler = EventScheduler::new();
    /// scheduler.timeout(10.0,
    ///                   Some(Box::new(|_| Some("Timeout event".to_string()))),
    ///                   None);
    /// ```
    pub fn timeout(&mut self, delay: f64, action: Option<Box<dyn FnMut(&mut EventScheduler) -> Option<String>>>, context: Option<HashMap<String, String>>) {
        let event = Event::new(self.current_time + delay, action, context);
        self.schedule(event);
    }

    /// Runs the event scheduler until a stop condition is met.
    ///
    /// # Parameters
    /// - `stop`: A closure that takes a reference to the scheduler and returns `true` when the scheduler should stop.
    /// - `log_filter`: An optional closure that determines whether to log an event. Defaults to logging all events.
    ///
    /// # Returns
    /// A vector of executed events along with their results.
    ///
    /// # Example
    /// ```
    /// use desru::{Event, EventScheduler};
    ///
    /// let mut scheduler = EventScheduler::new();
    /// scheduler.timeout(5.0,
    ///                   Some(Box::new(|_| Some("Event executed".to_string()))),
    ///                   None);
    /// let stop_fn = Box::new(|s: &EventScheduler| s.current_time >= 10.0);
    /// scheduler.run(stop_fn, None);
    /// ```
    pub fn run(&mut self, stop: Box<dyn Fn(&Self) -> bool>, log_filter: Option<Box<dyn Fn(&Event, &Option<String>) -> bool>>)  -> Vec<(Event, Option<String>)> {
        let log_filter = log_filter.unwrap_or_else(|| Box::new(|_, _| true));
        while !stop(self) {
            if let Some(mut event) = self.event_queue.pop() {
                self.current_time = event.time;
                let event_result = event.run(self);
                if log_filter(&event, &event_result) {
                    self.event_log.push((event, event_result));
                }
            } else {
                break;
            }
        }
        self.event_log.clone()
    }

    /// Runs the event scheduler until a specified maximum time is reached.
    ///
    /// This is a convenience method that calls `run` with a predefined stop condition based on `max_time`.
    ///
    /// # Parameters
    /// - `max_time`: The maximum simulation time.
    ///
    /// # Returns
    /// A vector of executed events along with their results.
    ///
    /// # Example
    /// ```
    /// use desru::{Event, EventScheduler};
    ///
    /// let mut scheduler = EventScheduler::new();
    /// scheduler.timeout(5.0,
    ///                   Some(Box::new(|_| Some("Timeout event".to_string()))),
    ///                   None);
    /// scheduler.run_until_max_time(10.0);
    /// ```
    pub fn run_until_max_time(&mut self, max_time: f64) -> Vec<(Event, Option<String>)> {
        self.run(Box::new(stop_at_max_time_factory(max_time)), None)
    }
}

/////////////////////////
// $3 STOP CONDITIONS //
///////////////////////

// Stop function to halt the simulation at a maximum time
/// A factory function to create a stop condition that halts the simulation after a maximum time.
///
/// # Parameters
/// - `max_time`: The maximum simulation time.
///
/// # Returns
/// A closure that returns `true` when the scheduler's current tim
fn stop_at_max_time_factory(max_time: f64) -> Box<dyn Fn(&EventScheduler) -> bool> {
    Box::new(move |scheduler: &EventScheduler| {
        scheduler.current_time >= max_time
        || scheduler.event_queue.peek().map_or(true, |event| event.time >= max_time)
    })
}

////////////////////
// $4 UNIT TESTS //
//////////////////

#[cfg(test)]
mod tests {
    use super::*;
    use std::collections::HashMap;

    #[test]
    fn test_event_run() {
        let mut _scheduler = EventScheduler::new();
        let mut event = Event::new(0.0, Some(Box::new(|_scheduler| Some("Executed".to_string()))), None);
        let result = event.run(&mut _scheduler);

        assert_eq!(result, Some("Executed".to_string()));
    }

    #[test]
    fn test_inactive_event_run() {
        let mut _scheduler = EventScheduler::new();
        let mut event = Event::new(0.0, Some(Box::new(|_scheduler| Some("Executed".to_string()))), None);
        event.active = false;  // Set the event to inactive
        let result = event.run(&mut _scheduler);

        assert_eq!(result, None);
    }

    #[test]
    fn test_event_cloning() {
        let mut _scheduler = EventScheduler::new();
        let mut context = HashMap::new();
        context.insert("key".to_string(), "value".to_string());
        let original_event = Event::new(5.0, Some(Box::new(|_scheduler| Some("Executed".to_string()))), Some(context));

        let mut cloned_event = original_event.clone();
        assert_eq!(cloned_event.time, original_event.time);
        assert_eq!(cloned_event.context.get("key"), Some(&"value".to_string()));
        assert!(cloned_event.run(&mut _scheduler).is_none());  // Run should return None due to placeholder action
    }

    #[test]
    fn test_event_scheduling() {
        let mut scheduler = EventScheduler::new();
        let event = Event::new(5.0, None, None);
        scheduler.schedule(event);

        assert_eq!(scheduler.event_queue.len(), 1);
    }

    #[test]
    fn test_timeout_functionality() {
        let mut scheduler = EventScheduler::new();
        scheduler.timeout(10.0, Some(Box::new(|_| Some("Timeout Event".to_string()))), None);

        assert_eq!(scheduler.event_queue.len(), 1);
    }

    #[test]
    fn test_run_until_max_time() {
        let mut scheduler = EventScheduler::new();
        scheduler.timeout(5.0, Some(Box::new(|_| Some("Event 1".to_string()))), None);
        scheduler.timeout(15.0, Some(Box::new(|_| Some("Event 2".to_string()))), None);

        let executed_events = scheduler.run_until_max_time(10.0);
        assert_eq!(executed_events.len(), 1); // Only Event 1 should execute
    }

    #[test]
    fn test_stop_condition_functionality() {
        let mut _scheduler = EventScheduler::new();
        _scheduler.timeout(5.0, Some(Box::new(|_scheduler| Some("Event A".to_string()))), None);
        
        let stop_fn = Box::new(|s: &EventScheduler| s.current_time >= 5.0);
        let executed_events = _scheduler.run(stop_fn, None);
        
        assert_eq!(executed_events.len(), 1); // Event A should execute
    }
}