-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathlaunch_run.py
174 lines (148 loc) · 5.97 KB
/
launch_run.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
#!/usr/bin/env python
from argparse import ArgumentParser, RawDescriptionHelpFormatter
from contextlib import contextmanager
import datetime as dt
import os
import sys
import re
import shutil
import subprocess
@contextmanager
def cd(newdir):
'''Context-managed chdir
changes back to original directory on exit or failure
'''
prevdir = os.getcwd()
os.chdir(os.path.expanduser(newdir))
try:
yield
finally:
os.chdir(prevdir)
def runningInDocker():
'''Function to check whether script is running in a container
'''
with open('/proc/self/cgroup', 'r') as procfile:
for line in procfile:
fields = line.strip().split('/')
if 'docker' in fields:
return True
return False
def setup_parser():
# Create argument parser & set up arguments:
parser = ArgumentParser(description=__doc__,
formatter_class=RawDescriptionHelpFormatter)
parser.add_argument("-d", "--destdir", default=None, required=True,
help="Destination path for run directory")
parser.add_argument("configfile", nargs=1, help="Run config")
parser.add_argument("-o", "--overwrite", action="store_true",
help="Clobber an existing run directory" +
"with the same name/path.")
parser.add_argument("--debug", action="store_true",
help="Print debug information.")
# Handle arguments, noting that argparse expands linux wildcards.
args = parser.parse_args()
return args
def get_ramindices_end(fname='input/RamIndices.txt'):
# read file and get last date
with open(fname, 'r') as fh:
inds = fh.readlines()
lastline = inds[-1].strip()
lastdatestr = lastline.split()[0]
lastdate = dt.datetime.strptime(lastdatestr, '%Y%m%d')
return lastdate
def parse_block(config_file, srch='#STARTTIME'):
expr = re.compile(srch)
with open(config_file, 'r') as fh:
lines = fh.readlines()
lines = [line.rstrip() for line in lines]
for idx, line in enumerate(lines):
searchyn = re.match(expr, line)
if searchyn: # target line
useidx = idx
store = [ll.split()[0] for ll in lines[useidx:useidx+7]]
year = int(store[1])
month = int(store[2])
day = int(store[3])
hour = int(store[4])
minute = int(store[5])
second = int(store[6])
return dt.datetime(year, month, day, hour, minute, second)
def parse_config(config_file):
'''Placeholder, pending web interface
'''
st_date = parse_block(config_file, srch='#STARTTIME')
end_date = parse_block(config_file, srch='#STOPTIME')
print('LAUNCH_RUN: Requested {} to {}'.format(st_date, end_date))
lastramind = get_ramindices_end()
print('LAUNCH_RUN: RamIndices ends at {}'.format(lastramind))
sys.stdout.flush()
if end_date >= lastramind:
# if run ends after the last date in the Ramindices file,
# update it
with cd("Scripts"):
subprocess.run(['python', 'updateRamIndices.py'])
return st_date, end_date
def setup_rundir(args, st_date, end_date, gen_fluxbound=True):
'''Make, populate, and move RAM-SCB run directory
'''
# Now make rundir and copy in everything we need
stYYMMDD = '{:04d}-{:02d}-{:02d}'.format(st_date.year,
st_date.month,
st_date.day)
enYYMMDD = '{:04d}-{:02d}-{:02d}'.format(end_date.year,
end_date.month,
end_date.day)
if args.overwrite:
shutil.rmtree('run_ram_ror', ignore_errors=True)
compl = subprocess.run(['make', 'rundir', 'RUNDIR=run_ram_ror'],
check=True, capture_output=True)
# then make flux boundary files
if gen_fluxbound:
fmdir = '/SHIELDS/flux-model' if runningInDocker() else '../flux-model'
cmdline = ' '.join(['python', f'{fmdir}/makeGEOboundary.py',
f'-s {stYYMMDD}', f'-e {enYYMMDD} -m 0',
'-r input -o run_ram_ror/input_ram'])
compl = subprocess.run(cmdline, shell=True,
check=True, stdout=subprocess.PIPE)
sys.stdout.flush()
# add supplied PARAM file
shutil.copyfile(args.configfile[0], 'run_ram_ror/PARAM.in')
# and move rundir to final location
if args.overwrite:
shutil.rmtree(args.destdir, ignore_errors=True)
shutil.move('run_ram_ror', args.destdir)
def run_model(args):
'''Launch RAM-SCB and wait so the launch script won't exit
'''
with cd(args.destdir):
print(os.getcwd())
if not os.path.isfile('ram_scb.exe'):
raise RuntimeError(' '.join(['RAM-SCB not found in specified',
'directory, or directory not',
'created properly']))
# launch RAM-SCB, requires all relevant data in run dir
# and appropriate subdirectories
pram = subprocess.Popen(['./ram_scb.exe'])
# entrypoint process in docker must be kept in foreground
# to prevent container stopping
pram.wait()
def make_plots(args, st_date, en_date):
'''
'''
stdt = st_date.isoformat()[:19]
endt = end_date.isoformat()[:19]
with cd('Scripts'):
cmdline = ' '.join(['python', 'summaryPlots.py', f'-s {stdt}',
f'-e {endt}', f'-o {args.destdir}',
f'{args.destdir}'])
print(cmdline)
subprocess.run(cmdline, shell=True, check=True,
stdout=subprocess.PIPE)
if __name__ == '__main__':
# get arguments this program was called with
args = setup_parser()
# get date range and update Kp/F10.7 if req'd
st_date, end_date = parse_config(args.configfile[0])
setup_rundir(args, st_date, end_date)
run_model(args)
make_plots(args, st_date, end_date)