Skip to content

Commit 60b80f6

Browse files
committed
adds a new method for simulate_until_max_customers
1 parent 2db3522 commit 60b80f6

File tree

6 files changed

+59
-21
lines changed

6 files changed

+59
-21
lines changed

CHANGES.rst

+3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
11
History
22
-------
33

4+
+ **3.2.4 (2024-12-04)**
5+
+ Adds a new method for the `simulate_until_max_customers: "Complete" simulates until a specific number of completed customer journeys; while "Finish" simulates until a specific number of customers have reached the exit node (through bailking or reneging).
6+
47
+ **3.2.3 (2024-10-15)**
58
+ Allow some numerical imprecision in the PMF probability sums. This allows for very large arrays of probabilities and use of Pandas and Numpy to define probabilities.
69

ciw/node.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -86,7 +86,7 @@ def __repr__(self):
8686
"""
8787
return "Node %s" % self.id_number
8888

89-
def accept(self, next_individual):
89+
def accept(self, next_individual, completed=False):
9090
"""
9191
Accepts a new customer to the queue:
9292
- remove previous exit date and blockage status
@@ -654,7 +654,7 @@ def renege(self):
654654
self.write_reneging_record(reneging_individual)
655655
self.reset_individual_attributes(reneging_individual)
656656
self.simulation.statetracker.change_state_renege(self, next_node, reneging_individual, False)
657-
next_node.accept(reneging_individual)
657+
next_node.accept(reneging_individual, completed=False)
658658
self.release_blocked_individual()
659659

660660
def get_reneging_date(self, ind):

ciw/simulation.py

+10-3
Original file line numberDiff line numberDiff line change
@@ -273,13 +273,18 @@ def simulate_until_max_time(self, max_simulation_time, progress_bar=False):
273273
self.progress_bar.close()
274274

275275
def simulate_until_max_customers(
276-
self, max_customers, progress_bar=False, method="Finish"
276+
self, max_customers, progress_bar=False, method="Complete"
277277
):
278278
"""
279279
Runs the simulation until max_customers is reached:
280280
281+
- Method: Complete
282+
Simulates until max_customers has reached the Exit Node after
283+
completing their journey
281284
- Method: Finish
282-
Simulates until max_customers has reached the Exit Node
285+
Simulates until max_customers has reached the Exit Node whether
286+
they have completed their journey or not (included baulkers and
287+
renegers)
283288
- Method: Arrive
284289
Simulates until max_customers have spawned at the Arrival Node
285290
- Method: Accept
@@ -292,8 +297,10 @@ def simulate_until_max_customers(
292297
if progress_bar:
293298
self.progress_bar = tqdm.tqdm(total=max_customers)
294299

295-
if method == "Finish":
300+
if method == "Complete":
296301
check = lambda: self.nodes[-1].number_of_completed_individuals
302+
elif method == "Finish":
303+
check = lambda: self.nodes[-1].number_of_individuals
297304
elif method == "Arrive":
298305
check = lambda: self.nodes[0].number_of_individuals
299306
elif method == "Accept":

ciw/tests/test_simulation.py

+28-11
Original file line numberDiff line numberDiff line change
@@ -194,29 +194,43 @@ def test_simulate_until_max_time_with_pbar_method(self):
194194
self.assertEqual(Q.progress_bar.total, 150)
195195
self.assertEqual(Q.progress_bar.n, 150)
196196

197-
def test_simulate_until_max_customers_finish(self):
197+
def test_simulate_until_max_customers(self):
198198
N = ciw.create_network(
199199
arrival_distributions=[ciw.dists.Exponential(1.0)],
200200
service_distributions=[ciw.dists.Exponential(0.5)],
201201
number_of_servers=[1],
202202
routing=[[0.0]],
203203
queue_capacities=[3],
204204
)
205-
# Test default method, 'Finish'
205+
# Test default method, 'Complete'
206206
ciw.seed(2)
207-
Q1 = ciw.Simulation(N)
208-
Q1.simulate_until_max_customers(10, method="Finish")
209-
self.assertEqual(Q1.nodes[-1].number_of_completed_individuals, 10)
210-
completed_records = Q1.get_all_records(only=["service"])
207+
Q = ciw.Simulation(N)
208+
Q.simulate_until_max_customers(10)
209+
self.assertEqual(Q.nodes[-1].number_of_completed_individuals, 10)
210+
self.assertEqual(Q.nodes[-1].number_of_individuals, 34)
211+
completed_records = Q.get_all_records(only=["service"])
211212
self.assertEqual(len(completed_records), 10)
212213

214+
# Test 'Complete' method
215+
ciw.seed(2)
216+
Q0 = ciw.Simulation(N)
217+
Q0.simulate_until_max_customers(10, method="Complete")
218+
self.assertEqual(Q0.nodes[-1].number_of_completed_individuals, 10)
219+
self.assertEqual(Q0.nodes[-1].number_of_individuals, 34)
220+
completed_records = Q0.get_all_records(only=["service"])
221+
self.assertEqual(len(completed_records), 10)
222+
223+
next_active_node = Q0.find_next_active_node()
224+
end_time_complete = next_active_node.next_event_date
225+
213226
# Test 'Finish' method
214227
ciw.seed(2)
215228
Q2 = ciw.Simulation(N)
216-
Q2.simulate_until_max_customers(10)
217-
self.assertEqual(Q2.nodes[-1].number_of_completed_individuals, 10)
229+
Q2.simulate_until_max_customers(10, method="Finish")
230+
self.assertEqual(Q2.nodes[-1].number_of_completed_individuals, 3)
231+
self.assertEqual(Q2.nodes[-1].number_of_individuals, 10)
218232
completed_records = Q2.get_all_records(only=["service"])
219-
self.assertEqual(len(completed_records), 10)
233+
self.assertEqual(len(completed_records), 3)
220234

221235
next_active_node = Q2.find_next_active_node()
222236
end_time_finish = next_active_node.next_event_date
@@ -248,8 +262,11 @@ def test_simulate_until_max_customers_finish(self):
248262
next_active_node = Q4.find_next_active_node()
249263
end_time_accept = next_active_node.next_event_date
250264

251-
# Assert that finish time of finish > accept > arrive
252-
self.assertGreater(end_time_finish, end_time_accept)
265+
# Assert that finish time of complete > accept > finish > arrive
266+
self.assertGreater(end_time_complete, end_time_accept)
267+
self.assertGreater(end_time_complete, end_time_finish)
268+
self.assertGreater(end_time_complete, end_time_arrive)
269+
self.assertGreater(end_time_accept, end_time_finish)
253270
self.assertGreater(end_time_accept, end_time_arrive)
254271
self.assertGreater(end_time_finish, end_time_arrive)
255272

ciw/version.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -1 +1 @@
1-
__version__ = "3.2.3"
1+
__version__ = "3.2.4"

docs/Guides/Simulation/sim_numcusts.rst

+15-4
Original file line numberDiff line numberDiff line change
@@ -9,11 +9,12 @@ This can be done using the :code:`simulate_until_max_customers` method.
99
The method takes in a variable :code:`max_customers`.
1010
There are three methods of counting customers:
1111

12-
- :code:`'Finish'`: Simulates until :code:`max_customers` has reached the Exit Node.
12+
- :code:`'Complete'`: Simulates until :code:`max_customers` has reached the Exit Node due to completing their journey through the system.
13+
- :code:`'Finish'`: Simulates until :code:`max_customers` has reached the Exit Node, regardless if the customer reaches there without completing their journey, for example by :ref:`baulking <baulking-functions>` or :ref:`reneging <reneging-customers>`.
1314
- :code:`'Arrive'`: Simulates until :code:`max_customers` have spawned at the Arrival Node.
1415
- :code:`'Accept'`: Simulates until :code:`max_customers` have been spawned and accepted (not rejected) at the Arrival Node.
1516

16-
The method of counting customers is specified with the optional keyword argument :code:`method`. The default value is is :code:`'Finish'`.
17+
The method of counting customers is specified with the optional keyword argument :code:`method`. The default value is is :code:`'Complete'`.
1718

1819
Consider an :ref:`M/M/1/3 <kendall-notation>` queue::
1920

@@ -25,15 +26,25 @@ Consider an :ref:`M/M/1/3 <kendall-notation>` queue::
2526
... queue_capacities=[3]
2627
... )
2728

28-
To simulate until 30 customers have finished service::
29+
30+
To simulate until 30 customers have completed::
2931

3032
>>> ciw.seed(1)
3133
>>> Q = ciw.Simulation(N)
32-
>>> Q.simulate_until_max_customers(30, method='Finish')
34+
>>> Q.simulate_until_max_customers(30, method='Complete')
3335
>>> recs = Q.get_all_records()
3436
>>> len([r for r in recs if r.record_type=="service"])
3537
30
3638

39+
To simulate until 30 customers have finished::
40+
41+
>>> ciw.seed(1)
42+
>>> Q = ciw.Simulation(N)
43+
>>> Q.simulate_until_max_customers(30, method='Finish')
44+
>>> recs = Q.get_all_records()
45+
>>> len(recs)
46+
30
47+
3748
To simulate until 30 customers have arrived::
3849

3950
>>> ciw.seed(1)

0 commit comments

Comments
 (0)