forked from CrayLabs/SmartSim
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathapplication.py
314 lines (257 loc) · 11.1 KB
/
application.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
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
# BSD 2-Clause License
#
# Copyright (c) 2021-2024, Hewlett Packard Enterprise
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions are met:
#
# 1. Redistributions of source code must retain the above copyright notice, this
# list of conditions and the following disclaimer.
#
# 2. Redistributions in binary form must reproduce the above copyright notice,
# this list of conditions and the following disclaimer in the documentation
# and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
from __future__ import annotations
import collections
import copy
import textwrap
import typing as t
from os import path as osp
from .._core.utils.helpers import expand_exe_path
from ..log import get_logger
from .entity import SmartSimEntity
from .files import EntityFiles
logger = get_logger(__name__)
# TODO: Remove this supression when we strip fileds/functionality
# (run-settings/batch_settings/params_as_args/etc)!
# pylint: disable-next=too-many-public-methods
class Application(SmartSimEntity):
"""The Application class enables users to execute computational tasks in an
Experiment workflow, such as launching compiled applications, running scripts,
or performing general computational operations.
Applications are designed to be added to Jobs, where LaunchSettings are also
provided to inject launcher-specific behavior into the Job.
"""
def __init__(
self,
name: str,
exe: str,
exe_args: t.Optional[t.Union[str, t.Sequence[str]]] = None,
files: t.Optional[EntityFiles] = None,
file_parameters: t.Mapping[str, str] | None = None,
) -> None:
"""Initialize an ``Application``
Applications require a name and an executable. Optionally, users may provide
executable arguments, files and file parameters. To create a simple Application
that echos `Hello World!`, consider the example below:
.. highlight:: python
.. code-block:: python
# Create an application that runs the 'echo' command
my_app = Application(name="my_app", exe="echo", exe_args="Hello World!")
:param name: name of the application
:param exe: executable to run
:param exe_args: executable arguments
:param files: files to be copied, symlinked, and/or configured prior to
execution
:param file_parameters: parameters and values to be used when configuring
files
"""
super().__init__(name)
"""The name of the application"""
self._exe = expand_exe_path(exe)
"""The executable to run"""
self._exe_args = self._build_exe_args(exe_args) or []
"""The executable arguments"""
self._files = copy.deepcopy(files) if files else None
"""Files to be copied, symlinked, and/or configured prior to execution"""
self._file_parameters = (
copy.deepcopy(file_parameters) if file_parameters else {}
)
"""Parameters and values to be used when configuring files"""
self._incoming_entities: t.List[SmartSimEntity] = []
"""Entities for which the prefix will have to be known by other entities"""
self._key_prefixing_enabled = False
"""Unique prefix to avoid key collisions"""
@property
def exe(self) -> str:
"""Return the executable.
:return: the executable
"""
return self._exe
@exe.setter
def exe(self, value: str) -> None:
"""Set the executable.
:param value: the executable
"""
self._exe = copy.deepcopy(value)
@property
def exe_args(self) -> t.MutableSequence[str]:
"""Return the executable arguments.
:return: the executable arguments
"""
return self._exe_args
@exe_args.setter
def exe_args(self, value: t.Union[str, t.Sequence[str], None]) -> None:
"""Set the executable arguments.
:param value: the executable arguments
"""
self._exe_args = self._build_exe_args(value)
def add_exe_args(self, args: t.Union[str, t.List[str], None]) -> None:
"""Add executable arguments to executable
:param args: executable arguments
"""
args = self._build_exe_args(args)
self._exe_args.extend(args)
@property
def files(self) -> t.Union[EntityFiles, None]:
"""Return attached EntityFiles object.
:return: the EntityFiles object of files to be copied, symlinked,
and/or configured prior to execution
"""
return self._files
@files.setter
def files(self, value: t.Optional[EntityFiles]) -> None:
"""Set the EntityFiles object.
:param value: the EntityFiles object of files to be copied, symlinked,
and/or configured prior to execution
"""
self._files = copy.deepcopy(value)
@property
def file_parameters(self) -> t.Mapping[str, str]:
"""Return file parameters.
:return: the file parameters
"""
return self._file_parameters
@file_parameters.setter
def file_parameters(self, value: t.Mapping[str, str]) -> None:
"""Set the file parameters.
:param value: the file parameters
"""
self._file_parameters = copy.deepcopy(value)
@property
def incoming_entities(self) -> t.List[SmartSimEntity]:
"""Return incoming entities.
:return: incoming entities
"""
return self._incoming_entities
@incoming_entities.setter
def incoming_entities(self, value: t.List[SmartSimEntity]) -> None:
"""Set the incoming entities.
:param value: incoming entities
"""
self._incoming_entities = copy.copy(value)
@property
def key_prefixing_enabled(self) -> bool:
"""Return whether key prefixing is enabled for the application.
:param value: key prefixing enabled
"""
return self._key_prefixing_enabled
@key_prefixing_enabled.setter
def key_prefixing_enabled(self, value: bool) -> None:
"""Set whether key prefixing is enabled for the application.
:param value: key prefixing enabled
"""
self.key_prefixing_enabled = copy.deepcopy(value)
def as_executable_sequence(self) -> t.Sequence[str]:
"""Converts the executable and its arguments into a sequence of program arguments.
:return: a sequence of strings representing the executable and its arguments
"""
return [self.exe, *self.exe_args]
def attach_generator_files(
self,
to_copy: t.Optional[t.List[str]] = None,
to_symlink: t.Optional[t.List[str]] = None,
to_configure: t.Optional[t.List[str]] = None,
) -> None:
"""Attach files to an entity for generation
Attach files needed for the entity that, upon generation,
will be located in the path of the entity. Invoking this method
after files have already been attached will overwrite
the previous list of entity files.
During generation, files "to_copy" are copied into
the path of the entity, and files "to_symlink" are
symlinked into the path of the entity.
Files "to_configure" are text based application input files where
parameters for the application are set. Note that only applications
support the "to_configure" field. These files must have
fields tagged that correspond to the values the user
would like to change. The tag is settable but defaults
to a semicolon e.g. THERMO = ;10;
:param to_copy: files to copy
:param to_symlink: files to symlink
:param to_configure: input files with tagged parameters
:raises ValueError: if the generator file already exists
"""
to_copy = to_copy or []
to_symlink = to_symlink or []
to_configure = to_configure or []
# Check that no file collides with the parameter file written
# by Generator. We check the basename, even though it is more
# restrictive than what we need (but it avoids relative path issues)
for strategy in [to_copy, to_symlink, to_configure]:
if strategy is not None and any(
osp.basename(filename) == "smartsim_params.txt" for filename in strategy
):
raise ValueError(
"`smartsim_params.txt` is a file automatically "
+ "generated by SmartSim and cannot be ovewritten."
)
self.files = EntityFiles(to_configure, to_copy, to_symlink)
@property
def attached_files_table(self) -> str:
"""Return a list of attached files as a plain text table
:return: String version of table
"""
if not self.files:
return "No file attached to this application."
return str(self.files)
@staticmethod
def _build_exe_args(exe_args: t.Union[str, t.Sequence[str], None]) -> t.List[str]:
"""Check and convert exe_args input to a desired collection format
:param exe_args:
:raises TypeError: if exe_args is not a list of str or str
"""
if not exe_args:
return []
if not (
isinstance(exe_args, str)
or (
isinstance(exe_args, collections.abc.Sequence)
and all(isinstance(arg, str) for arg in exe_args)
)
):
raise TypeError("Executable arguments were not a list of str or a str.")
if isinstance(exe_args, str):
return exe_args.split()
return list(exe_args)
def print_attached_files(self) -> None:
"""Print a table of the attached files on std out"""
print(self.attached_files_table)
def __str__(self) -> str: # pragma: no cover
exe_args_str = "\n".join(self.exe_args)
entities_str = "\n".join(str(entity) for entity in self.incoming_entities)
return textwrap.dedent(f"""\
Name: {self.name}
Type: {self.type}
Executable:
{self.exe}
Executable Arguments:
{exe_args_str}
Entity Files: {self.files}
File Parameters: {self.file_parameters}
Incoming Entities:
{entities_str}
Key Prefixing Enabled: {self.key_prefixing_enabled}
""")