-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathJob.py
executable file
·101 lines (84 loc) · 3.91 KB
/
Job.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
"""
Exports the Job class.
"""
import argparse
import configparser
import logging
from abc import ABC, abstractmethod
from datetime import datetime
from itertools import product
from os import path, chdir
from typing import Sequence, Tuple
class Job(ABC):
"""
Run a simulation job (a range of simulations with some specified parameters)
"""
def __init__(self, description: str):
"""
Parse parameters from CLI and settings.ini
:param str description: Description for help text
"""
# Parse CLI arguments
parser = argparse.ArgumentParser(description=description, formatter_class=argparse.ArgumentDefaultsHelpFormatter)
parser.add_argument("--dry-run", action="store_true", help="Don't actually do anything productive.")
parser.add_argument("subdir", help="Subdir to use for this job")
parser.add_argument("--skip-data-file-check", help="Skip checking for presence of the initial data file.",
action="store_true")
parser.add_argument("--force", action="store_true", help="Always run the simulation even if a simulation subdirectory already exists, and overwrite it")
self.args = parser.parse_args()
# Assert that subdir exists and chdir in it
if not path.isdir(self.args.subdir):
raise RuntimeError("Subdir '{}' does not exist.".format(self.args.subdir))
chdir(self.args.subdir)
# Setup logging
logging.basicConfig(level=logging.INFO, handlers=[logging.FileHandler("wrapper.log"), logging.StreamHandler()])
# Setup parser
config = configparser.ConfigParser(converters={'list': lambda s: s.split()})
config.optionxform = str # Use case-sensitive keys
# Assert that settings file exist and parse it
if not path.isfile('settings.ini'):
raise RuntimeError("settings.ini does not exist in specified subdir {}".format(self.args.subdir))
config.read('settings.ini')
# Assert that input file exists
input_file = '../' + config['job']['input_file']
if not path.isfile(input_file):
raise RuntimeError("Missing input file: {}".format(input_file))
# Assert that data file(s) exist
static_data_file = config['static_vars'].get('initial_data_file')
dyn_data_files = config['dyn_vars'].getlist('initial_data_file')
data_files = dyn_data_files if dyn_data_files else [static_data_file]
if not self.args.skip_data_file_check:
for data_file in data_files:
if not path.isfile(path.expanduser(config['job'].get('initial_data_file_prefix') + data_file)):
raise RuntimeError("Missing initial data file: {}"
.format(config['job'].get('initial_data_file_prefix') + data_file))
# Assert all required vars are accounted for
for var in config.getlist('job', 'required_vars'):
if var not in config['static_vars'] and var not in config['dyn_vars']:
raise RuntimeError("Missing value for variable '{}'".format(var))
self.static_vars = dict(config['static_vars'])
# Get dict of dynamic vars
self.dyn_vars = {var: config.getlist('dyn_vars', var) for var in config['dyn_vars']}
self.lammps_command = ' '.join([
config['lammps'].get('MPI_path'),
config['lammps'].get('MPI_arguments', ''),
config['lammps']['LAMMPS_path'],
config['lammps'].get('LAMMPS_arguments', '')
])
self.slurm_sbatch_cmd = config['job']['slurm_sbatch_args']
self.input_file = input_file
self.log_file = config['job']['log_file']
self.initial_data_file_prefix = config['job'].get('initial_data_file_prefix')
self.jobscript_header_file = config['lammps'].get('jobscript_header_file')
# Create Cartesian product of dynamic var values. Returns flat list of tuples (*vars)
dyn_values_list = list(product(*self.dyn_vars.values()))
logging.info(datetime.now())
logging.info("Got {} simulations".format(len(dyn_values_list)))
self._spawn_simulations(dyn_values_list)
@abstractmethod
def _spawn_simulations(self, dyn_values_list: Sequence[Tuple]) -> None:
"""
Dispatch simulations
:param dyn_values_list: List of tuples of values for the dynamic variables
"""
pass