-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwindow.py
182 lines (167 loc) · 7.01 KB
/
window.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
"""
.. codeauthor:: Niklaus Johner <[email protected]>
This file contains the :class:`Window` object which represents a simulation window.
"""
import os
import subprocess
import logging
import numpy as npy
import scipy.interpolate
import matplotlib.pyplot as plt
import itertools
from phase import Phase
import pickle
class Window():
"""
This class is second in the hierarchical structure used in SiPMF. The :class:`System` contains
several (usually many) class:`Window` which correspond to the :class:`System` restrained to a small
region of the CV space using a quadratic restraining potential. The restraining potential has to
be of the form **1/2 spring_constant(x-cv_value)**. Notice the 1/2 in the definition used here, whereas some
MD codes do not have this factor. **Carefully check that the potential defined here matches
the potential defined in the MD code you are using!**
"""
def __repr__(self):
return "Window({0},{1},{2},{3})".format(self.system, self.cv_values, self.spring_constants, self.parent)
def __str__(self):
if self.parent:
return "{0} initialized form {1}".format(self.name, self.parent.name)
else:
return "{0}".format(self.name)
def __init__(self, system, cv_values, spring_constants, cv_shifts=None, parent=None, window_name=None):
"""
:param system: The system to which the window belongs
:param cv_values: The centers of the quadratic potentials restraining the CVs.
:param spring_constants: The spring constants of the potentials restraining the CVs
:param parent: The parent window
:type system: :class:`~system.System`
:type cv_values: :class:`list` (:class:`float`)
:type spring_constants: :class:`list` (:class:`float`)
:type parent: :class:`~window.Window`
"""
self.cv_names = [cv.name for cv in system.cv_list]
self.cv_values = cv_values
if not cv_shifts:
self.cv_shifts = [0.0 for cvv in self.cv_values]
else:
self.cv_shifts = cv_shifts
self.spring_constants = spring_constants
self.is_new = True
self.n_data = 0
self.n_run_phases = 0
self.phases = []
self.system = system
if not window_name:
window_name = "_".join(["".join([cvn, str(cvv)])
for cvn, cvv in zip(self.cv_names, self.cv_values)])
window_name += "_" + "_".join(["".join([cvn + "K", str(cvk)])
for cvn, cvk in zip(self.cv_names, self.spring_constants)])
self.name = window_name
self.subdir = os.path.join(system.simu_dir, self.name)
self.parent = parent
self.path_to_datafile = os.path.join(self.subdir, "data.txt")
self.datafile_n_data_tot = 0
self.datafile_n_data_skipped = 0
def Initialize(self):
logging.info("New window: {0}".format(self))
if os.path.isdir(self.subdir):
logging.error(
"Directory already exists, program stops to avoid overwriting {0}.".format(self.subdir))
raise IOError(
"Directory already exists, program stops to avoid overwriting {0}.".format(self.subdir))
r = subprocess.call(["mkdir", self.subdir])
if r != 0:
logging.error(
"Problem creating output directory {0}.".format(self.subdir))
raise IOError(
"Problem creating output directory {0}.".format(self.subdir))
self.last_phase_n_crashed = 0
d = {"cv_shifts": self.cv_shifts}
if self.parent:
d.update({"parent cv values": self.parent.cv_values})
d.update({"parent spring constants": self.parent.spring_constants})
else:
d.update({"init_restartdir": self.init_restartdir})
f = open(os.path.join(self.subdir, "info.pkl"), "w")
pickle.dump(d, f)
f.close()
def SubmitNextPhase(self, environment):
"""
Automatically creates the appropriate next :class:`Phase` and corresponding :class:`Job` and
submits it to the cluster. What the appropriate next phase is, is determined as follows:
- If the window does not have a parent and does not contain any phase yet, the new phase will be
a run phase using as restart its *init_restartdir*
- If the window has a parent phase does not contain any phase yet, the new phase will be
an initialization phase using as restart the last phase of the parent phase.
- If the window already contains one or several phases, the new phase will be
an run phase using as restart the last phase of this window.
:param environment: The environment used to submit the job to the cluster.
:type environment: :class:`~environment.Environment`
"""
if self.is_new:
if self.parent:
phase_name = "initialization"
phase_type = "initialization"
restart_phase = self.parent.phases[-1]
self.phases.append(Phase(self, phase_name, phase_type, restart_phase))
else:
phase_name = "phase1"
phase_type = "run"
self.phases.append(Phase(self, phase_name, phase_type))
self.is_new = False
else:
phase_name = "phase" + str(self.n_run_phases + 1)
phase_type = "run"
self.phases.append(Phase(self, phase_name, phase_type, self.phases[-1]))
next_phase = self.phases[-1]
next_phase.Initialize()
next_phase.job.Submit(environment)
def UpdateDataCount(self):
"""
Update the total number of data accumulated for this window (sum over the data in each phase of the window).
"""
self.n_data = 0
for phase in self.phases:
phase.UpdateDataCount()
self.n_data += phase.GetDataCount()
def UpdateDataFile(self, n_skip=0, n_tot=-1, new_only=True):
"""
Updates the window's datafile. It takes the data from all the run phases and
writes it into its own datafile, skipping the first n_skip data points
and adding a maximum of n_tot data points.
*n_tot=-1* means there is no maximal number of data points.
:param n_skip: The number of data points to skip.
:param n_tot: The total number of data points used to calculate the PMF.
:param new_only: Only add the data from phases that have not yet been added to the data files.
:type n_skip: :class:`int`
:type n_tot: :class:`int`
:type new_only: :class:`bool`
"""
if not new_only:
self.datafile_n_data_tot = 0
self.datafile_n_data_skipped = 0
if os.path.isfile(self.path_to_datafile):
subprocess.call(["rm", self.path_to_datafile])
for phase in self.phases:
phase.AddDataToWindow(n_skip, n_tot, new_only)
def ReadDataFile(self):
"""
Reads the datafile of the window and returns a tuple with
the list of times as the first element. The second element in
the tuple is a list containing one list of values for each CV.
"""
f = open(self.path_to_datafile, "r")
ll = f.readlines()
f.close()
t = []
cvs = [[] for i in range(self.system.dimensionality)]
for l in ll:
s = l.split()
t.append(float(s[0]))
for i in range(self.system.dimensionality):
cvs[i].append(float(s[i + 1]))
return (t, cvs)
def FindPhase(self, phase_name):
for p in self.phases:
if p.name == phase_name:
return p
return None