Coverage for sparkle/CLI/help/reporting_scenario.py: 96%
201 statements
« prev ^ index » next coverage.py v7.8.0, created at 2025-04-03 10:42 +0000
« prev ^ index » next coverage.py v7.8.0, created at 2025-04-03 10:42 +0000
1"""Helper module to manage Sparkle scenarios."""
2# Keep in CLI help
4from __future__ import annotations
5import configparser
6from enum import Enum
7from pathlib import Path
8from pathlib import PurePath
10from sparkle.configurator.configurator import ConfigurationScenario
11from sparkle.solver import Solver
12from sparkle.instance import Instance_Set, InstanceSet
15class Scenario(str, Enum):
16 """Enum of possible execution scenarios for Sparkle."""
18 NONE = "NONE"
19 SELECTION = "SELECTION"
20 CONFIGURATION = "CONFIGURATION"
21 PARALLEL_PORTFOLIO = "PARALLEL_PORTFOLIO"
24class ReportingScenario:
25 """Class to manage scenarios executed with Sparkle."""
27 # ReportingScenario path names and defaults
28 __reporting_scenario_file = Path("latest_scenario.ini")
29 __reporting_scenario_dir = Path("Output")
30 DEFAULT_reporting_scenario_path = Path(
31 PurePath(__reporting_scenario_dir / __reporting_scenario_file))
33 # Constant default values
34 DEFAULT_latest_scenario = Scenario.NONE
36 DEFAULT_selection_portfolio_path = Path("")
37 DEFAULT_selection_test_case_directory = Path("")
39 DEFAULT_parallel_portfolio_path = Path("")
40 DEFAULT_parallel_portfolio_instance_list = []
42 DEFAULT_config_solver = Path("")
43 DEFAULT_config_instance_set_train = Path("")
44 DEFAULT_config_instance_set_test = Path("")
46 def __init__(self: ReportingScenario) -> None:
47 """Initialise a ReportingScenario object."""
48 # ReportingScenario 'dictionary' in configparser format
49 self.__scenario = configparser.ConfigParser()
51 # Initialise scenario in default file path
52 self.read_scenario_ini()
54 def read_scenario_ini(
55 self: ReportingScenario, file_path: Path = DEFAULT_reporting_scenario_path)\
56 -> None:
57 """Read the scenario from an INI file.
59 Args:
60 file_path: Path of the INI file for the scenario. Defaults to
61 DEFAULT_reporting_scenario_path.
62 """
63 # If the file does not exist or is empty set default values
64 if not Path(file_path).is_file() or Path(file_path).stat().st_size == 0:
65 self.set_latest_scenario()
66 self.set_selection_scenario_path()
67 self.set_selection_test_case_directory()
68 self.set_parallel_portfolio_path()
69 self.set_parallel_portfolio_instance_path()
70 self.set_config_solver()
71 self.set_config_instance_set_train()
72 self.set_config_instance_set_test()
74 # Read file
75 file_scenario = configparser.ConfigParser()
76 file_scenario.read(file_path)
78 # Set internal scenario based on data read from FILE if they were read
79 # successfully
80 if file_scenario.sections() != []:
81 section = "latest"
82 option_names = ("scenario",) # Comma to make it a tuple
83 for option in option_names:
84 if file_scenario.has_option(section, option):
85 value = Scenario(file_scenario.get(section, option))
86 self.set_latest_scenario(value)
87 file_scenario.remove_option(section, option)
89 section = "selection"
90 option_names = ("scenario_path",) # Comma to make it a tuple
91 for option in option_names:
92 if file_scenario.has_option(section, option):
93 value = Path(file_scenario.get(section, option))
94 self.set_selection_scenario_path(value)
95 file_scenario.remove_option(section, option)
97 option_names = ("test_case_directory",) # Comma to make it a tuple
98 for option in option_names:
99 if file_scenario.has_option(section, option):
100 value = Path(file_scenario.get(section, option))
101 self.set_selection_test_case_directory(value)
102 file_scenario.remove_option(section, option)
104 section = "configuration"
105 option_names = ("solver",) # Comma to make it a tuple
106 for option in option_names:
107 if file_scenario.has_option(section, option):
108 value = Path(file_scenario.get(section, option))
109 self.set_config_solver(value)
110 file_scenario.remove_option(section, option)
112 option_names = ("instance_set_train",) # Comma to make it a tuple
113 for option in option_names:
114 if file_scenario.has_option(section, option):
115 value = Path(file_scenario.get(section, option))
116 self.set_config_instance_set_train(value)
117 file_scenario.remove_option(section, option)
119 option_names = ("instance_set_test",) # Comma to make it a tuple
120 for option in option_names:
121 if file_scenario.has_option(section, option):
122 value = Path(file_scenario.get(section, option))
123 self.set_config_instance_set_test(value)
124 file_scenario.remove_option(section, option)
126 option_names = ("scenario_file_path", )
127 for option in option_names:
128 if file_scenario.has_option(section, option):
129 value = Path(file_scenario.get(section, option))
130 self.set_configuration_scenario(value)
131 file_scenario.remove_option(section, option)
133 section = "parallel_portfolio"
134 option_names = ("portfolio_path",) # Comma to make it a tuple
135 for option in option_names:
136 if file_scenario.has_option(section, option):
137 value = Path(file_scenario.get(section, option))
138 self.set_parallel_portfolio_path(value)
139 file_scenario.remove_option(section, option)
141 option_names = ("instance_path",) # Comma to make it a tuple
142 for option in option_names:
143 if file_scenario.has_option(section, option):
144 value = file_scenario.get(section, option)
145 self.set_parallel_portfolio_instance_path(value)
146 file_scenario.remove_option(section, option)
148 # Report on any unknown settings that were read
149 sections = file_scenario.sections()
151 for section in sections:
152 for option in file_scenario[section]:
153 print(f'Unrecognised section - option combination:"{section} '
154 f'{option}" in file {file_path} ignored')
156 # Print error if unable to read the scenario file
157 elif not file_path.exists():
158 print(f"WARNING: Failed to read latest scenario from {file_path}. "
159 "Default values will be used.")
161 def write_scenario_ini(
162 self: ReportingScenario, file_path: Path = DEFAULT_reporting_scenario_path)\
163 -> None:
164 """Write the scenario file in INI format.
166 Args:
167 file_path: Path of the INI file for the scenario. Defaults to
168 DEFAULT_reporting_scenario_path.
169 """
170 # Create needed directories if they don't exist
171 file_dir = file_path.parents[0]
172 file_dir.mkdir(parents=True, exist_ok=True)
174 # Write the scenario to file
175 with Path(str(file_path)).open("w") as scenario_file:
176 self.__scenario.write(scenario_file)
178 def __init_section(self: ReportingScenario, section: str) -> None:
179 """Initialise a section in the scenario file.
181 Args:
182 section: Name of the section.
183 """
184 if section not in self.__scenario:
185 self.__scenario[section] = {}
187 # Generic setters ###
189 def path_setter(self: ReportingScenario, section: str, name: str, value: Path)\
190 -> None:
191 """Set a generic Path for the scenario.
193 Args:
194 section: Name of the section.
195 name: Name of the path element.
196 value: Value of the path given.
197 """
198 if value is not None:
199 self.__init_section(section)
200 self.__scenario[section][name] = str(value)
202 # Generic getters ###
204 def none_if_empty_path(self: ReportingScenario, path: Path) -> Path:
205 """Return None if a path is empty or the Path otherwise.
207 Args:
208 path: Path value given.
210 Returns:
211 None if the given path is empty, the given Path value otherwise.
212 """
213 if str(path) == "" or str(path) == ".":
214 return None
215 return path
217 # Latest settings ###
219 def set_latest_scenario(self: ReportingScenario,
220 value: Scenario = DEFAULT_latest_scenario) -> None:
221 """Set the latest Scenario that was executed."""
222 section = "latest"
223 name = "scenario"
225 if value is not None:
226 self.__init_section(section)
227 self.__scenario[section][name] = value.name
229 def get_latest_scenario(self: ReportingScenario) -> Scenario:
230 """Return the latest Scenario that was executed."""
231 return Scenario(self.__scenario["latest"]["scenario"])
233 # Selection settings ###
235 def set_selection_scenario_path(
236 self: ReportingScenario, value: Path = DEFAULT_selection_portfolio_path)\
237 -> None:
238 """Set the path to portfolio selector used for algorithm selection."""
239 section = "selection"
240 name = "scenario_path"
241 self.path_setter(section, name, value)
243 def get_selection_scenario_path(self: ReportingScenario) -> Path:
244 """Return the path to portfolio selector used for algorithm selection."""
245 return Path(self.__scenario["selection"]["scenario_path"])
247 def set_selection_test_case_directory(
248 self: ReportingScenario,
249 value: Path = DEFAULT_selection_test_case_directory) -> None:
250 """Set the path to the testing set that was used for algorithm selection."""
251 section = "selection"
252 name = "test_case_directory"
253 self.path_setter(section, name, value)
255 def get_selection_test_case_directory(self: ReportingScenario) -> str:
256 """Return the path to the testing set that was used for algorithm selection."""
257 try:
258 path = self.__scenario["selection"]["test_case_directory"]
259 if Path(path) == Path("."):
260 path = None
261 except KeyError:
262 path = None
263 return path
265 # Parallel portfolio settings ###
267 def set_parallel_portfolio_path(
268 self: ReportingScenario,
269 value: Path = DEFAULT_parallel_portfolio_path) -> None:
270 """Set the path to the parallel portfolio."""
271 section = "parallel_portfolio"
272 name = "portfolio_path"
273 self.path_setter(section, name, value)
275 def get_parallel_portfolio_path(self: ReportingScenario) -> Path:
276 """Return the path to the parallel portfolio."""
277 return Path(self.__scenario["parallel_portfolio"]["portfolio_path"])
279 def set_parallel_portfolio_instance_path(
280 self: ReportingScenario,
281 value: Path = None) -> None:
282 """Set the instance path used with the parallel portfolio."""
283 section = "parallel_portfolio"
284 name = "instance_path"
285 self.path_setter(section, name, value)
287 def get_parallel_portfolio_instance_set(self: ReportingScenario) -> InstanceSet:
288 """Return the instance list used with the parallel portfolio.
290 If instance list is empty return None.
291 """
292 if self.__scenario["parallel_portfolio"]["instance_path"] is None:
293 return None
294 return Instance_Set(Path(self.__scenario["parallel_portfolio"]["instance_path"]))
296 # Configuration settings ###
298 def set_config_solver(self: ReportingScenario,
299 value: Solver | Path = DEFAULT_config_solver) -> None:
300 """Set the path to the solver that was configured."""
301 section = "configuration"
302 name = "solver"
303 if isinstance(value, Solver):
304 value = value.directory
305 self.path_setter(section, name, value)
307 def get_config_solver(self: ReportingScenario) -> Solver:
308 """Return the path to the solver that was configured."""
309 path = self.none_if_empty_path(Path(self.__scenario["configuration"]["solver"]))
310 if path is not None:
311 return Solver(path)
312 return None
314 def set_config_instance_set_train(
315 self: ReportingScenario, value: Path = DEFAULT_config_instance_set_train)\
316 -> None:
317 """Set the path to the training instance set used for configuration."""
318 section = "configuration"
319 name = "instance_set_train"
320 self.path_setter(section, name, value)
322 def get_config_instance_set_train(self: ReportingScenario) -> InstanceSet:
323 """Return the path to the training instance set used for configuration."""
324 path = self.none_if_empty_path(
325 Path(self.__scenario["configuration"]["instance_set_train"]))
326 if path is None:
327 return None
328 return Instance_Set(path)
330 def set_configuration_scenario(self: ReportingScenario,
331 value: Scenario | Path) -> None:
332 """Set the path to the scenario that was used for configuration."""
333 section = "configuration"
334 name = "scenario_file_path"
335 if isinstance(value, ConfigurationScenario):
336 value = value.scenario_file_path
337 self.path_setter(section, name, value)
339 def get_configuration_scenario(self: ReportingScenario,
340 scenario_class: ConfigurationScenario) -> Scenario:
341 """Return the path to the scenario that was used for configuration."""
342 path = self.none_if_empty_path(
343 Path(self.__scenario["configuration"]["scenario_file_path"]))
344 if path is None:
345 return None
346 return scenario_class.from_file(path)
348 def set_config_instance_set_test(
349 self: ReportingScenario, value: Path = DEFAULT_config_instance_set_test)\
350 -> None:
351 """Set the path to the testing instance set used for configuration."""
352 section = "configuration"
353 name = "instance_set_test"
354 self.path_setter(section, name, value)
356 def get_config_instance_set_test(self: ReportingScenario) -> InstanceSet:
357 """Return the path to the testing instance set used for configuration."""
358 path = self.none_if_empty_path(
359 Path(self.__scenario["configuration"]["instance_set_test"]))
360 if path is None:
361 return None
362 return Instance_Set(path)