source: utils/boxmaker.py@ 0c83d8

Action_Thermostats Add_AtomRandomPerturbation Add_FitFragmentPartialChargesAction Add_RotateAroundBondAction Add_SelectAtomByNameAction Added_ParseSaveFragmentResults AddingActions_SaveParseParticleParameters Adding_Graph_to_ChangeBondActions Adding_MD_integration_tests Adding_ParticleName_to_Atom Adding_StructOpt_integration_tests AtomFragments Automaking_mpqc_open AutomationFragmentation_failures Candidate_v1.5.4 Candidate_v1.6.0 Candidate_v1.6.1 ChangeBugEmailaddress ChangingTestPorts ChemicalSpaceEvaluator CombiningParticlePotentialParsing Combining_Subpackages Debian_Package_split Debian_package_split_molecuildergui_only Disabling_MemDebug Docu_Python_wait EmpiricalPotential_contain_HomologyGraph EmpiricalPotential_contain_HomologyGraph_documentation Enable_parallel_make_install Enhance_userguide Enhanced_StructuralOptimization Enhanced_StructuralOptimization_continued Example_ManyWaysToTranslateAtom Exclude_Hydrogens_annealWithBondGraph FitPartialCharges_GlobalError Fix_BoundInBox_CenterInBox_MoleculeActions Fix_ChargeSampling_PBC Fix_ChronosMutex Fix_FitPartialCharges Fix_FitPotential_needs_atomicnumbers Fix_ForceAnnealing Fix_IndependentFragmentGrids Fix_ParseParticles Fix_ParseParticles_split_forward_backward_Actions Fix_PopActions Fix_QtFragmentList_sorted_selection Fix_Restrictedkeyset_FragmentMolecule Fix_StatusMsg Fix_StepWorldTime_single_argument Fix_Verbose_Codepatterns Fix_fitting_potentials Fixes ForceAnnealing_goodresults ForceAnnealing_oldresults ForceAnnealing_tocheck ForceAnnealing_with_BondGraph ForceAnnealing_with_BondGraph_continued ForceAnnealing_with_BondGraph_continued_betteresults ForceAnnealing_with_BondGraph_contraction-expansion FragmentAction_writes_AtomFragments FragmentMolecule_checks_bonddegrees GeometryObjects Gui_Fixes Gui_displays_atomic_force_velocity ImplicitCharges IndependentFragmentGrids IndependentFragmentGrids_IndividualZeroInstances IndependentFragmentGrids_IntegrationTest IndependentFragmentGrids_Sole_NN_Calculation JobMarket_RobustOnKillsSegFaults JobMarket_StableWorkerPool JobMarket_unresolvable_hostname_fix MoreRobust_FragmentAutomation ODR_violation_mpqc_open PartialCharges_OrthogonalSummation PdbParser_setsAtomName PythonUI_with_named_parameters QtGui_reactivate_TimeChanged_changes Recreated_GuiChecks Rewrite_FitPartialCharges RotateToPrincipalAxisSystem_UndoRedo SaturateAtoms_findBestMatching SaturateAtoms_singleDegree StoppableMakroAction Subpackage_CodePatterns Subpackage_JobMarket Subpackage_LinearAlgebra Subpackage_levmar Subpackage_mpqc_open Subpackage_vmg Switchable_LogView ThirdParty_MPQC_rebuilt_buildsystem TrajectoryDependenant_MaxOrder TremoloParser_IncreasedPrecision TremoloParser_MultipleTimesteps TremoloParser_setsAtomName Ubuntu_1604_changes stable
Last change on this file since 0c83d8 was 0c83d8, checked in by Frederik Heber <heber@…>, 13 years ago

Less ugly advanced version

  • Class (struct) based global options
  • Minor bugfixes
  • Feature: Cubic cell
  • Feature: Skip unit conversion
  • Feature: Options via cmd line
  • Feature: More settings

TODO:

  • Output postprocessing (INPUTCONVs)
  • Auto-rotate to minimum BBox
  • Documentation
  • Property mode set to 100644
File size: 7.6 KB
Line 
1import re, os, os.path, sys, operator
2
3class c_opt():
4 basename = None
5 tremofiledir = './'
6 potentialsfiledir = './'
7 outfilename = 'out'
8
9 source = None
10 molarmass = None
11 density = None
12 temp = None
13
14 number = '1000'
15
16 cubicdomain = 'on'
17 cubiccell = 'off'
18 autorotate = 'off' # Not implemented yet
19 autodim = 'on'
20
21 def update(self, name, value):
22 if name in dir(self):
23 exec('self.%s = "%s"' % (name, value))
24 else:
25 print 'Warning: Unknown option:', name
26
27
28def ReadSettings(opt):
29 # Obtain basename
30 if len(sys.argv) >= 2:
31 opt.basename = sys.argv[1]
32 else:
33 print 'Usage: boxmaker.py <basename> [options]'
34 exit()
35
36 # Read settings file
37 with open('boxmaker.' + opt.basename) as f:
38 for line in f:
39 if len(line) > 0 and line[0] != '#':
40 L, S, R = line.partition('=')
41 opt.update(L.strip(), R.strip())
42
43 # Parse parameters
44 i = 2
45 while i < len(sys.argv):
46 L = sys.argv[i]
47
48 if L[0] in '+-':
49 LN = L[1:]
50
51 if L[0] == '+':
52 R = 'on'
53 else:
54 R = 'off'
55 else:
56 LN = L
57 i += 1
58 R = sys.argv[i]
59
60 opt.update(LN, R)
61 i += 1
62
63
64def ReadUnits(opt):
65 lines = [] # The file needs to be processed twice, so we save the lines in the first run
66
67 with open(opt.tremofiledir + opt.basename + '.tremolo') as f:
68 for line in f:
69 if len(line) > 0 and line[0] != '#':
70 line = line.strip()
71 lines.append(line)
72
73 if 'systemofunits' in line:
74 L, S, SOU = line.partition('=')
75 SOU = SOU.strip()[:-1] # Remove semicolon
76
77 if SOU == 'custom':
78 units = {}
79 quantities = ['length', 'mass']
80
81 for quantity in quantities:
82 units[quantity] = [None, None] # Init with scaling factor and unit 'None'.
83
84 for line in lines:
85 for quantity in quantities:
86 if quantity in line:
87 L, S, R = line.partition('=')
88 R = R.strip()[:-1] # Remove semicolon
89
90 if 'scalingfactor' in line:
91 units[quantity][0] = R
92 else:
93 units[quantity][1] = R
94
95 elif SOU == 'kcalpermole':
96 units = {'length': ['1', 'angstrom'], 'mass': ['1', 'u']}
97
98 elif SOU == 'evolt':
99 units = {'length': ['1', 'angstrom'], 'mass': ['1', 'u']}
100
101 else: # SI
102 units = {'length': ['1', 'm'], 'mass': ['1', 'kg']}
103
104 return units
105
106
107def ConvertUnits(have, want):
108 # Redo with pipes?
109 ret = os.system("units '%s' '%s' > temp_units_output" % (have, want))
110
111 if ret == 0:
112 with open('temp_units_output') as f:
113 line = f.readline()
114
115 os.system('rm temp_units_output')
116
117 return float(line[3:-1])
118 else:
119 raise NameError('UnitError')
120
121
122def UpdateSettings(opt):
123 # Map boolean values
124 for name in ['cubicdomain', 'cubiccell', 'autorotate', 'autodim']:
125 value = eval('opt.' + name)
126
127 if value == 'on':
128 value = True
129 elif value == 'off':
130 value = False
131 else:
132 print 'Not a boolean value:', value
133 exit()
134
135 exec('opt.' + name + '= value')
136
137 # Convert dimensions
138 if opt.autodim:
139 units = ReadUnits(opt)
140
141 have = opt.molarmass
142 want = '%s*%s / mol' % tuple(units['mass'])
143 opt.molarmass = ConvertUnits(have, want)
144
145 have = opt.density
146 want = '(%s*%s) ' % tuple(units['mass']) + '/ (%s*%s)**3' % tuple(units['length'])
147 opt.density = ConvertUnits(have, want)
148 else:
149 opt.molarmass = float(opt.molarmass)
150 opt.density = float(opt.density)
151
152 # Number might be an integer or a 3-vector
153 nvec = opt.number.split()
154 if len(nvec) == 3:
155 opt.number = [0]*3
156
157 for i in range(3):
158 opt.number[i] = int(nvec[i])
159 else:
160 opt.number = int(opt.number)
161
162
163def FindBestCube(opt):
164 newroot = int( round(opt.number**(1./3)) )
165 newnumber = newroot**3
166
167 if newnumber != opt.number:
168 print 'Warning: Number changed to %d.' % newnumber
169
170 return [newroot] * 3
171
172
173def FindBestCuboid(opt):
174 n = opt.number
175
176 # Prime factors of n
177 factors = []
178
179 for i in [2, 3]:
180 while n % i == 0:
181 factors.append(i)
182 n /= 2
183
184 t = 5
185 diff = 2
186
187 while t*t <= n:
188 while n % t == 0:
189 factors.append(t)
190 n /= t
191
192 t = t + diff
193 diff = 6 - diff
194
195 if n > 1:
196 factors.append(n)
197
198 # Even distribution of current biggest prime to each vector -> similar sizes
199 if len(factors) < 3:
200 print 'Warning: Not enough prime factors - falling back to cubic placement'
201 return FindBestCube(opt)
202
203 factors.sort()
204 distri = [[],[],[]]
205 current = 0
206
207 for factor in factors:
208 distri[current].append(factor)
209 current += 1
210 if current == 3:
211 current = 0
212
213 result = [0]*3
214
215 print '======== CUBOID USED:',
216
217 for i in range(3):
218 result[i] = int( reduce(operator.mul, distri[i]) )
219
220 print result
221 return result
222
223
224def GetSourceBBabs(opt):
225 bbmax = [0.0]*3
226 bbmin = [float('inf')]*3
227
228 s_name_ext = os.path.basename(opt.source).rsplit('.', 1)
229 s_namepart = s_name_ext[0]
230
231 if len(s_name_ext) > 1:
232 s_ext = s_name_ext[1]
233 else:
234 s_ext = ''
235
236 # Convert from any format to xyz
237 os.system('molecuilder -o xyz --parse-tremolo-potentials %s -i temp_source.xyz -l %s' % (opt.potentialsfiledir+opt.basename+'.potentials', opt.source))
238
239 # Calculate bounding box from xyz-file
240 with open('temp_source.xyz') as f:
241 N = int(f.readline())
242 comment = f.readline()
243
244 for i in xrange(N):
245 buf = f.readline()
246 xyz = buf.split()[1:]
247
248 for i in range(3):
249 bbmax[i] = max(bbmax[i], float(xyz[i]))
250 bbmin[i] = min(bbmin[i], float(xyz[i]))
251
252 bb = [0.0]*3
253
254 for i in range(3):
255 bb[i] = abs(bbmax[i] - bbmin[i])
256
257 os.system('rm temp_source.*')
258 return bb
259
260# Global options with sensible default parameters
261opt = c_opt()
262
263ReadSettings(opt)
264UpdateSettings(opt)
265
266if type(opt.number) == type([]):
267 # Number is a vector - use it without any modification
268 nbox = opt.number
269else:
270 if opt.cubicdomain:
271 nbox = FindBestCube(opt)
272 else:
273 nbox = FindBestCuboid(opt)
274
275avogadro = 6.022143e23
276VolumePerMolecule = opt.molarmass / (avogadro * opt.density)
277cell = [VolumePerMolecule**(1./3)] * 3
278
279if not opt.cubiccell:
280 try:
281 bb = GetSourceBBabs(opt)
282 print '======== BBOX:', bb
283 # Scaling factor - the molecules bounding box is scaled to fit the volume suiting the density
284 s = (VolumePerMolecule / (bb[0]*bb[1]*bb[2])) ** (1./3)
285
286 if s < 1:
287 print 'Warning: Molecular cells will overlap.'
288
289 for i in range(3):
290 cell[i] = bb[i]*s
291 except ZeroDivisionError:
292 print 'Warning: Singularity in bounding box, falling back to cubic cell.'
293
294
295print '======== CELL: ', cell
296
297import pyMoleCuilder as mol
298
299mol.CommandVerbose('0')
300mol.ParserParseTremoloPotentials(opt.potentialsfiledir + opt.basename + '.potentials')
301mol.WorldInput(opt.source)
302mol.WorldCenterInBox('%f 0 0 %f 0 %f' % tuple(cell))
303mol.WorldRepeatBox('%d %d %d' % tuple(nbox))
304mol.WorldOutput(opt.outfilename + '.data')
305mol.WorldOutput(opt.outfilename + '.xyz')
306
307domain = [0.0]*3
308
309for i in range(3):
310 domain[i] = cell[i]*nbox[i]
311
312print '======== DOMAIN: ', domain
Note: See TracBrowser for help on using the repository browser.