-
Notifications
You must be signed in to change notification settings - Fork 42
/
Copy pathschedules.py
149 lines (135 loc) · 5.57 KB
/
schedules.py
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
from typing import List, Tuple, Union, Generator, NoReturn
class Schedule:
"""
A Schedule class for storing information about server schedules and
generating the next shifts.
Parameters
----------
schedule : List[Tuple[int, float]]
A list of tuples representing shifts, where each tuple contains the
number of servers and the shift end date.
preemption : Union[bool, str], optional
Pre-emption option, should be either 'resume', 'restart', 'resample',
or False. Default is False.
Attributes
----------
schedule_type : str
Type of the schedule.
shift_end_dates : List[float]
List of shift end dates.
numbers_of_servers : List[int]
List of corresponding server numbers.
preemption : Union[bool, str]
Pre-emption option.
cyclelength : float
Length of the schedule cycle.
Methods
-------
initialise()
Initializes the generator object at the beginning of a simulation.
get_schedule_generator(boundaries, values)
A generator that yields the next time and number of servers according
to a given schedule.
get_next_shift()
Updates the next shifts from the generator.
"""
def __init__(self, numbers_of_servers: List[int], shift_end_dates: List[float], preemption: Union[bool, str] = False, offset: float = 0.0) -> NoReturn:
"""
Initializes the instance of the Schedule object.
Parameters
----------
numbers_of_servers : List[int]
A list containing the number of servers working at each shift
shift_end_dates : List[float]
A list containing the end dates of each shift.
preemption : Union[bool, str], optional
Pre-emption option, should be either 'resume', 'restart',
'resample', or False.
Default is False.
"""
if preemption not in [False, 'resume', 'restart', 'resample', 'reroute']:
raise ValueError("Pre-emption options should be either 'resume', 'restart', 'resample', 'reroute', or False.")
if not isinstance(offset, float):
raise ValueError("Offset should be a positive float.")
if offset < 0.0:
raise ValueError("Offset should be a positive float.")
self.schedule_type = 'schedule'
self.shift_end_dates = shift_end_dates
self.numbers_of_servers = numbers_of_servers
self.preemption = preemption
self.cyclelength = self.shift_end_dates[-1]
self.offset = offset
def initialise(self) -> NoReturn:
"""
Initializes the generator object at the beginning of a simulation.
"""
self.c = 0
self.next_shift_change_date = self.offset
self.next_c = self.numbers_of_servers[0]
self.schedule_generator = self.get_schedule_generator(self.shift_end_dates, self.numbers_of_servers, self.offset)
def get_schedule_generator(self, boundaries:List[float], values:List[int], offset:float) -> Generator[Tuple[float, int], None, None]:
"""
A generator that yields the next time and number of servers according
to a given schedule.
Parameters
----------
boundaries : List[float]
List of shift boundaries.
values : List[int]
List of corresponding server numbers.
Yields
------
Tuple[float, int]
A tuple representing the next shift date and the number of servers.
"""
num_boundaries = len(boundaries)
index = 0
date = 0
while True:
date = offset + boundaries[index % num_boundaries] + ((index) // num_boundaries * self.cyclelength)
index += 1
yield date, values[index % num_boundaries]
def get_next_shift(self) -> NoReturn:
"""
Updates the next shifts from the generator.
"""
self.c = self.next_c
date, c = next(self.schedule_generator)
self.next_shift_change_date = date
self.next_c = c
class Slotted(Schedule):
def __init__(self, slots, slot_sizes, capacitated=False, preemption=False, offset=0.0):
"""
Initialises the instance of the Slotted Schedule object
"""
if not capacitated:
if preemption is not False:
raise ValueError("Pre-emption options not availale for non-capacitated slots.")
if preemption not in [False, 'resume', 'restart', 'resample']:
raise ValueError("Pre-emption options should be either 'resume', 'restart', 'resample', or False.")
if not isinstance(offset, float):
raise ValueError("Offset should be a positive float.")
if offset < 0.0:
raise ValueError("Offset should be a positive float.")
self.schedule_type = 'slotted'
self.offset = offset
self.slots = slots
self.slot_sizes = slot_sizes
self.next_slot_sizes = [self.slot_sizes[-1]] + self.slot_sizes[:-1]
self.capacitated = capacitated
self.preemption = preemption
self.cyclelength = self.slots[-1]
self.c = 0
def initialise(self):
"""
Initialises the generator object at the beginning of a simulation
"""
self.schedule_generator = self.get_schedule_generator(self.slots, self.next_slot_sizes, self.offset)
self.get_next_slot()
def get_next_slot(self):
"""
Updates the next slot time and size from the generator
"""
date, size = next(self.schedule_generator)
self.next_slot_date = date
self.slot_size = size