source: src/Actions/FragmentationAction/FragmentationAutomationAction.cpp@ db6f7d

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 db6f7d was db6f7d, checked in by Frederik Heber <heber@…>, 12 years ago

FIX: We must not assert but check whether full molecule is loaded in FragmentationAutomationAction.

  • TESTFIX: Regression test Jobmarket molecuilderrun did not supply full molecule.
  • Property mode set to 100644
File size: 17.8 KB
Line 
1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
4 * Copyright (C) 2010-2012 University of Bonn. All rights reserved.
5 *
6 *
7 * This file is part of MoleCuilder.
8 *
9 * MoleCuilder is free software: you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation, either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * MoleCuilder is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with MoleCuilder. If not, see <http://www.gnu.org/licenses/>.
21 */
22
23/*
24 * FragmentationAutomationAction.cpp
25 *
26 * Created on: May 18, 2012
27 * Author: heber
28 */
29
30// include config.h
31#ifdef HAVE_CONFIG_H
32#include <config.h>
33#endif
34
35#include <boost/archive/text_iarchive.hpp>
36// boost asio needs specific operator new
37#include <boost/asio.hpp>
38
39#include "CodePatterns/MemDebug.hpp"
40
41#include <boost/mpl/remove.hpp>
42#include <boost/lambda/lambda.hpp>
43
44#include "CodePatterns/Assert.hpp"
45#include "CodePatterns/Info.hpp"
46#include "CodePatterns/Log.hpp"
47#include "JobMarket/Jobs/FragmentJob.hpp"
48
49#include "Fragmentation/Automation/createMatrixNrLookup.hpp"
50#include "Fragmentation/Automation/extractJobIds.hpp"
51#include "Fragmentation/Automation/FragmentationChargeDensity.hpp"
52#include "Fragmentation/Automation/FragmentationResults.hpp"
53#include "Fragmentation/Automation/MPQCFragmentController.hpp"
54#include "Fragmentation/Automation/VMGDebugGridFragmentController.hpp"
55#include "Fragmentation/Automation/VMGFragmentController.hpp"
56#include "Fragmentation/EnergyMatrix.hpp"
57#include "Fragmentation/ForceMatrix.hpp"
58#include "Fragmentation/Fragmentation.hpp"
59#include "Fragmentation/SetValues/Fragment.hpp"
60#include "Fragmentation/SetValues/Histogram.hpp"
61#include "Fragmentation/SetValues/IndexedVectors.hpp"
62#include "Fragmentation/HydrogenSaturation_enum.hpp"
63#include "Fragmentation/KeySet.hpp"
64#include "Fragmentation/KeySetsContainer.hpp"
65#include "Fragmentation/Summation/writeTable.hpp"
66#include "Graph/DepthFirstSearchAnalysis.hpp"
67#include "Helpers/defs.hpp"
68#include "Jobs/MPQCJob.hpp"
69#include "Jobs/MPQCData.hpp"
70#include "Jobs/MPQCData_printKeyNames.hpp"
71#ifdef HAVE_VMG
72#include "Jobs/VMGDebugGridJob.hpp"
73#include "Jobs/VMGJob.hpp"
74#include "Jobs/VMGData.hpp"
75#include "Jobs/VMGDataFused.hpp"
76#include "Jobs/VMGDataMap.hpp"
77#include "Jobs/VMGData_printKeyNames.hpp"
78#endif
79#include "World.hpp"
80
81#include <fstream>
82#include <iostream>
83#include <string>
84#include <vector>
85
86#include <boost/mpl/for_each.hpp>
87
88#include "Actions/FragmentationAction/FragmentationAutomationAction.hpp"
89
90using namespace MoleCuilder;
91
92// and construct the stuff
93#include "FragmentationAutomationAction.def"
94#include "Action_impl_pre.hpp"
95/** =========== define the function ====================== */
96
97class controller_AddOn;
98
99// needs to be defined for using the FragmentController
100controller_AddOn *getAddOn()
101{
102 return NULL;
103}
104
105/** Helper function to get number of atoms somehow.
106 *
107 * Here, we just parse the number of lines in the adjacency file as
108 * it should correspond to the number of atoms, except when some atoms
109 * are not bonded, but then fragmentation makes no sense.
110 *
111 * @param path path to the adjacency file
112 */
113size_t getNoAtomsFromAdjacencyFile(const std::string &path)
114{
115 size_t NoAtoms = 0;
116
117 // parse in special file to get atom count (from line count)
118 std::string filename(path);
119 filename += FRAGMENTPREFIX;
120 filename += ADJACENCYFILE;
121 std::ifstream adjacency(filename.c_str());
122 if (adjacency.fail()) {
123 LOG(0, endl << "getNoAtomsFromAdjacencyFile() - Unable to open " << filename << ", is the directory correct?");
124 return false;
125 }
126 std::string buffer;
127 while (getline(adjacency, buffer))
128 NoAtoms++;
129 LOG(1, "INFO: There are " << NoAtoms << " atoms.");
130
131 return NoAtoms;
132}
133
134
135
136/** Place results from FragmentResult into EnergyMatrix and ForceMatrix.
137 *
138 * @param fragmentData MPQCData resulting from the jobs
139 * @param MatrixNrLookup Lookup up-map from job id to fragment number
140 * @param FragmentCounter total number of fragments
141 * @param NoAtoms total number of atoms
142 * @param Energy energy matrix to be filled on return
143 * @param Force force matrix to be filled on return
144 * @return true - everything ok, false - else
145 */
146bool putResultsintoMatrices(
147 const std::map<JobId_t, MPQCData> &fragmentData,
148 const std::map< JobId_t, size_t > &MatrixNrLookup,
149 const size_t FragmentCounter,
150 const size_t NoAtoms,
151 EnergyMatrix &Energy,
152 ForceMatrix &Force)
153{
154 for (std::map<JobId_t, MPQCData>::const_iterator dataiter = fragmentData.begin();
155 dataiter != fragmentData.end(); ++dataiter) {
156 const MPQCData &extractedData = dataiter->second;
157 const JobId_t &jobid = dataiter->first;
158 std::map< JobId_t, size_t >::const_iterator nriter = MatrixNrLookup.find(jobid);
159 ASSERT( nriter != MatrixNrLookup.end(),
160 "putResultsintoMatrices() - MatrixNrLookup does not contain id "
161 +toString(jobid)+".");
162 // place results into EnergyMatrix ...
163 {
164 MatrixContainer::MatrixArray matrix;
165 matrix.resize(1);
166 matrix[0].resize(1, extractedData.energies.total);
167 if (!Energy.AddMatrix(
168 std::string("MPQCJob ")+toString(jobid),
169 matrix,
170 nriter->second)) {
171 ELOG(1, "Adding energy matrix failed.");
172 return false;
173 }
174 }
175 // ... and ForceMatrix (with two empty columns in front)
176 {
177 MatrixContainer::MatrixArray matrix;
178 const size_t rows = extractedData.forces.size();
179 matrix.resize(rows);
180 for (size_t i=0;i<rows;++i) {
181 const size_t columns = 2+extractedData.forces[i].size();
182 matrix[i].resize(columns, 0.);
183 // for (size_t j=0;j<2;++j)
184 // matrix[i][j] = 0.;
185 for (size_t j=2;j<columns;++j)
186 matrix[i][j] = extractedData.forces[i][j-2];
187 }
188 if (!Force.AddMatrix(
189 std::string("MPQCJob ")+toString(jobid),
190 matrix,
191 nriter->second)) {
192 ELOG(1, "Adding force matrix failed.");
193 return false;
194 }
195 }
196 }
197 // add one more matrix (not required for energy)
198 MatrixContainer::MatrixArray matrix;
199 matrix.resize(1);
200 matrix[0].resize(1, 0.);
201 if (!Energy.AddMatrix(std::string("MPQCJob total"), matrix, FragmentCounter))
202 return false;
203 // but for energy because we need to know total number of atoms
204 matrix.resize(NoAtoms);
205 for (size_t i = 0; i< NoAtoms; ++i)
206 matrix[i].resize(2+NDIM, 0.);
207 if (!Force.AddMatrix(std::string("MPQCJob total"), matrix, FragmentCounter))
208 return false;
209
210 return true;
211}
212
213/** Print MPQCData from received results.
214 *
215 * @param fragmentData MPQCData resulting from the jobs, associated to job id
216 * @param KeySetFilename filename with keysets to associate forces correctly
217 * @param NoAtoms total number of atoms
218 * @param full_sample summed up charge from fragments on return
219 */
220bool printReceivedMPQCResults(
221 const std::map<JobId_t, MPQCData> &fragmentData,
222 const std::string &KeySetFilename,
223 size_t NoAtoms)
224{
225 // create a vector of all job ids
226 std::vector<JobId_t> jobids;
227 std::transform(fragmentData.begin(),fragmentData.end(),
228 std::back_inserter(jobids),
229 boost::bind( &std::map<JobId_t,MPQCData>::value_type::first, boost::lambda::_1 )
230 );
231
232 // create lookup from job nr to fragment number
233 size_t FragmentCounter = 0;
234 const std::map< JobId_t, size_t > MatrixNrLookup=
235 createMatrixNrLookup(jobids, FragmentCounter);
236
237 // place results into maps
238 EnergyMatrix Energy;
239 ForceMatrix Force;
240 if (!putResultsintoMatrices(fragmentData, MatrixNrLookup, FragmentCounter, NoAtoms, Energy, Force))
241 return false;
242
243 // initialise keysets
244 KeySetsContainer KeySet;
245 KeySetsContainer ForceKeySet;
246 if (!Energy.InitialiseIndices()) return false;
247
248 if (!Force.ParseIndices(KeySetFilename.c_str())) return false;
249
250 {
251 // else needs keysets without hydrogens
252 std::stringstream filename;
253 filename << FRAGMENTPREFIX << KEYSETFILE;
254 if (!KeySet.ParseKeySets(KeySetFilename, filename.str(), FragmentCounter)) return false;
255 }
256
257 {
258 // forces need keysets including hydrogens
259 std::stringstream filename;
260 filename << FRAGMENTPREFIX << FORCESFILE;
261 if (!ForceKeySet.ParseKeySets(KeySetFilename, filename.str(), FragmentCounter)) return false;
262 }
263
264 // combine all found data
265 if (!KeySet.ParseManyBodyTerms()) return false;
266
267 EnergyMatrix EnergyFragments;
268 ForceMatrix ForceFragments;
269 if (!EnergyFragments.AllocateMatrix(Energy.Header, Energy.MatrixCounter, Energy.RowCounter, Energy.ColumnCounter)) return false;
270 if (!ForceFragments.AllocateMatrix(Force.Header, Force.MatrixCounter, Force.RowCounter, Force.ColumnCounter)) return false;
271
272 if(!Energy.SetLastMatrix(0., 0)) return false;
273 if(!Force.SetLastMatrix(0., 2)) return false;
274
275 for (int BondOrder=0;BondOrder<KeySet.Order;BondOrder++) {
276 // --------- sum up energy --------------------
277 LOG(1, "INFO: Summing energy of order " << BondOrder+1 << " ...");
278 if (!EnergyFragments.SumSubManyBodyTerms(Energy, KeySet, BondOrder)) return false;
279 if (!Energy.SumSubEnergy(EnergyFragments, NULL, KeySet, BondOrder, 1.)) return false;
280
281 // --------- sum up Forces --------------------
282 LOG(1, "INFO: Summing forces of order " << BondOrder+1 << " ...");
283 if (!ForceFragments.SumSubManyBodyTerms(Force, KeySet, BondOrder)) return false;
284 if (!Force.SumSubForces(ForceFragments, KeySet, BondOrder, 1.)) return false;
285 }
286
287 // for debugging print resulting energy and forces
288 LOG(1, "INFO: Resulting energy is " << Energy.Matrix[ FragmentCounter ][0][0]);
289 std::stringstream output;
290 for (int i=0; i< Force.RowCounter[FragmentCounter]; ++i) {
291 for (int j=0; j< Force.ColumnCounter[FragmentCounter]; ++j)
292 output << Force.Matrix[ FragmentCounter ][i][j] << " ";
293 output << "\n";
294 }
295 LOG(1, "INFO: Resulting forces are " << std::endl << output.str());
296
297 return true;
298}
299
300void writeToFile(const std::string &filename, const std::string contents)
301{
302 std::ofstream tablefile(filename.c_str());
303 tablefile << contents;
304 tablefile.close();
305}
306
307/** Print MPQCData from received results.
308 *
309 * @param results summed up results container
310 */
311void printReceivedFullResults(
312 const FragmentationResults &results)
313{
314 // print tables (without eigenvalues, they go extra)
315 {
316 typedef boost::mpl::remove<MPQCDataEnergyVector_t, MPQCDataFused::energy_eigenvalues>::type
317 MPQCDataEnergyVector_noeigenvalues_t;
318 const std::string energyresult =
319 writeTable<MPQCDataEnergyMap_t, MPQCDataEnergyVector_noeigenvalues_t >()(
320 results.Result_Energy_fused, results.getMaxLevel());
321 LOG(0, "Energy table is \n" << energyresult);
322 std::string filename;
323 filename += FRAGMENTPREFIX + std::string("_Energy.dat");
324 writeToFile(filename, energyresult);
325 }
326
327 {
328 const std::string gridresult =
329 writeTable<VMGDataMap_t, VMGDataVector_t >()(
330 results.Result_LongRange_fused, results.getMaxLevel(), 2);
331 LOG(0, "VMG table is \n" << gridresult);
332 std::string filename;
333 filename += FRAGMENTPREFIX + std::string("_VMGEnergy.dat");
334 writeToFile(filename, gridresult);
335 }
336
337 {
338 const std::string gridresult =
339 writeTable<VMGDataLongRangeMap_t, VMGDataLongRangeVector_t >()(
340 results.Result_LongRangeIntegrated_fused, results.getMaxLevel(), 2);
341 LOG(0, "LongRange table is \n" << gridresult);
342 std::string filename;
343 filename += FRAGMENTPREFIX + std::string("_LongRangeEnergy.dat");
344 writeToFile(filename, gridresult);
345 }
346
347 {
348 const std::string eigenvalueresult;
349 LOG(0, "Eigenvalue table is \n" << eigenvalueresult);
350 std::string filename;
351 filename += FRAGMENTPREFIX + std::string("_Eigenvalues.dat");
352 writeToFile(filename, eigenvalueresult);
353 }
354
355 {
356 const std::string forceresult =
357 writeTable<MPQCDataForceMap_t, MPQCDataForceVector_t>()(
358 results.Result_Force_fused, results.getMaxLevel());
359 LOG(0, "Force table is \n" << forceresult);
360 std::string filename;
361 filename += FRAGMENTPREFIX + std::string("_Forces.dat");
362 writeToFile(filename, forceresult);
363 }
364 // we don't want to print grid to a table
365 {
366 // print times (without flops for now)
367 typedef boost::mpl::remove<
368 boost::mpl::remove<MPQCDataTimeVector_t, MPQCDataFused::times_total_flops>::type,
369 MPQCDataFused::times_gather_flops>::type
370 MPQCDataTimeVector_noflops_t;
371 const std::string timesresult =
372 writeTable<MPQCDataTimeMap_t, MPQCDataTimeVector_noflops_t >()(
373 results.Result_Time_fused, results.getMaxLevel());
374 LOG(0, "Times table is \n" << timesresult);
375 std::string filename;
376 filename += FRAGMENTPREFIX + std::string("_Times.dat");
377 writeToFile(filename, timesresult);
378 }
379}
380
381
382Action::state_ptr FragmentationFragmentationAutomationAction::performCall() {
383 boost::asio::io_service io_service;
384
385 // TODO: Have io_service run in second thread and merge with current again eventually
386
387 size_t Exitflag = 0;
388 std::map<JobId_t, MPQCData> fragmentData;
389 {
390 MPQCFragmentController mpqccontroller(io_service);
391 mpqccontroller.setHost(params.host.get());
392 mpqccontroller.setPort(params.port.get());
393 mpqccontroller.setLevel(params.level.get());
394 // Phase One: obtain ids
395 std::vector< boost::filesystem::path > jobfiles = params.jobfiles.get();
396 mpqccontroller.requestIds(jobfiles.size());
397
398 // Phase Two: create and add MPQCJobs
399 if (!mpqccontroller.addJobsFromFiles(params.executable.get().string(), jobfiles))
400 return Action::failure;
401
402 // Phase Three: calculate result
403 mpqccontroller.waitforResults(jobfiles.size());
404 mpqccontroller.getResults(fragmentData);
405
406 Exitflag += mpqccontroller.getExitflag();
407 }
408
409#ifdef HAVE_VMG
410 if (params.DoLongrange.get()) {
411 if ( World::getInstance().getAllAtoms().size() == 0) {
412 ELOG(1, "Please load the full molecule into the world before starting this action.");
413 return Action::failure;
414 }
415
416 // obtain combined charge density
417 FragmentationChargeDensity summedChargeDensity(
418 fragmentData,
419 params.path.get());
420 const std::vector<SamplingGrid> full_sample = summedChargeDensity.getFullSampledGrid();
421
422 LOG(1, "INFO: There are " << fragmentData.size() << " short-range and "
423 << full_sample.size() << " level-wise long-range jobs.");
424
425 // Phase Four: obtain more ids
426 std::map<JobId_t, VMGData> longrangeData;
427 {
428 VMGFragmentController vmgcontroller(io_service);
429 vmgcontroller.setHost(params.host.get());
430 vmgcontroller.setPort(params.port.get());
431 const size_t NoJobs = fragmentData.size()+full_sample.size();
432 vmgcontroller.requestIds(NoJobs);
433
434 // Phase Five: create VMGJobs
435 const size_t near_field_cells = params.near_field_cells.get();
436 const size_t interpolation_degree = params.interpolation_degree.get();
437 if (!vmgcontroller.createLongRangeJobs(
438 fragmentData,
439 full_sample,
440 summedChargeDensity.getFragment(),
441 near_field_cells,
442 interpolation_degree))
443 return Action::failure;
444
445 // Phase Six: calculate result
446 vmgcontroller.waitforResults(NoJobs);
447 vmgcontroller.getResults(longrangeData);
448 ASSERT( NoJobs == longrangeData.size(),
449 "FragmentationFragmentationAutomationAction::performCall() - number of MPQCresults+"
450 +toString(full_sample.size())+"="+toString(NoJobs)
451 +" and VMGresults "+toString(longrangeData.size())+" don't match.");
452 Exitflag += vmgcontroller.getExitflag();
453 }
454
455 // remove full solution corresponding to full_sample from map (must be highest ids), has to be treated extra
456 std::map<JobId_t, VMGData>::iterator iter = longrangeData.end();
457 for (size_t i=0;i<full_sample.size();++i)
458 --iter;
459 std::map<JobId_t, VMGData>::iterator remove_iter = iter;
460 std::vector<VMGData> fullsolutionData;
461 for (; iter != longrangeData.end(); ++iter)
462 fullsolutionData.push_back(iter->second);
463 longrangeData.erase(remove_iter, longrangeData.end());
464
465 // Final phase: sum up and print result
466 {
467 FragmentationResults results(
468 fragmentData,
469 longrangeData,
470 fullsolutionData,
471 params.path.get(),
472 getNoAtomsFromAdjacencyFile(params.path.get()),
473 full_sample);
474
475 LOG(1, "INFO: Parsing fragment files from " << params.path.get() << ".");
476 printReceivedFullResults(results);
477 }
478
479 std::map<JobId_t, std::string> debugData;
480 {
481 if (!full_sample.empty()) {
482 // create debug jobs for each level to print the summed-up potential to vtk files
483 VMGDebugGridFragmentController debugcontroller(io_service);
484 debugcontroller.setHost(params.host.get());
485 debugcontroller.setPort(params.port.get());
486 debugcontroller.requestIds(full_sample.size());
487 if (!debugcontroller.createDebugJobs(full_sample))
488 return Action::failure;
489 debugcontroller.waitforResults(full_sample.size());
490 debugcontroller.getResults(debugData);
491 Exitflag += debugcontroller.getExitflag();
492 }
493 }
494 }
495#else
496 // Final phase: print result
497 {
498 LOG(1, "INFO: Parsing fragment files from " << params.path.get() << ".");
499 printReceivedMPQCResults(
500 fragmentData,
501 params.path.get(),
502 getNoAtomsFromAdjacencyFile(params.path.get()));
503 }
504#endif
505
506 return (Exitflag == 0) ? Action::success : Action::failure;
507}
508
509Action::state_ptr FragmentationFragmentationAutomationAction::performUndo(Action::state_ptr _state) {
510 return Action::success;
511}
512
513Action::state_ptr FragmentationFragmentationAutomationAction::performRedo(Action::state_ptr _state){
514 return Action::success;
515}
516
517bool FragmentationFragmentationAutomationAction::canUndo() {
518 return false;
519}
520
521bool FragmentationFragmentationAutomationAction::shouldUndo() {
522 return false;
523}
524/** =========== end of function ====================== */
Note: See TracBrowser for help on using the repository browser.