source: src/Parser/MpqcParser.cpp@ 0b738b

Action_Thermostats Add_AtomRandomPerturbation Add_RotateAroundBondAction Add_SelectAtomByNameAction Adding_Graph_to_ChangeBondActions Adding_MD_integration_tests Adding_StructOpt_integration_tests AutomationFragmentation_failures Candidate_v1.6.0 Candidate_v1.6.1 Candidate_v1.7.0 ChangeBugEmailaddress ChangingTestPorts ChemicalSpaceEvaluator Disabling_MemDebug Docu_Python_wait EmpiricalPotential_contain_HomologyGraph_documentation Enhance_userguide Enhanced_StructuralOptimization Enhanced_StructuralOptimization_continued Example_ManyWaysToTranslateAtom Exclude_Hydrogens_annealWithBondGraph FitPartialCharges_GlobalError Fix_ChronosMutex Fix_StatusMsg Fix_StepWorldTime_single_argument Fix_Verbose_Codepatterns ForceAnnealing_goodresults ForceAnnealing_oldresults ForceAnnealing_tocheck ForceAnnealing_with_BondGraph ForceAnnealing_with_BondGraph_continued ForceAnnealing_with_BondGraph_continued_betteresults ForceAnnealing_with_BondGraph_contraction-expansion GeometryObjects Gui_displays_atomic_force_velocity IndependentFragmentGrids_IntegrationTest JobMarket_RobustOnKillsSegFaults JobMarket_StableWorkerPool PartialCharges_OrthogonalSummation PythonUI_with_named_parameters QtGui_reactivate_TimeChanged_changes Recreated_GuiChecks RotateToPrincipalAxisSystem_UndoRedo StoppableMakroAction TremoloParser_IncreasedPrecision TremoloParser_MultipleTimesteps Ubuntu_1604_changes stable
Last change on this file since 0b738b was 3fb9ab, checked in by Frederik Heber <heber@…>, 9 years ago

FIX: Preventing MPQC from using old temp files for restart.

  • TESTFIX: needed to add "checkstate=no" to file contents contained in unit test.
  • TESTFIX: needed to adapt all check .in files of MPQC and other regression tests.
  • Property mode set to 100644
File size: 18.0 KB
RevLine 
[bcf653]1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
[0aa122]4 * Copyright (C) 2010-2012 University of Bonn. All rights reserved.
[5aaa43]5 * Copyright (C) 2013 Frederik Heber. All rights reserved.
[94d5ac6]6 *
7 *
8 * This file is part of MoleCuilder.
9 *
10 * MoleCuilder is free software: you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License as published by
12 * the Free Software Foundation, either version 2 of the License, or
13 * (at your option) any later version.
14 *
15 * MoleCuilder is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with MoleCuilder. If not, see <http://www.gnu.org/licenses/>.
[bcf653]22 */
23
[43dad6]24/*
25 * MpqcParser.cpp
26 *
27 * Created on: 12.06.2010
28 * Author: heber
29 */
30
[bf3817]31// include config.h
32#ifdef HAVE_CONFIG_H
33#include <config.h>
34#endif
35
[52cac0]36#include <iomanip>
[aa8ef2]37#include <iostream>
[03a589]38#include <boost/foreach.hpp>
[aa8ef2]39#include <boost/tokenizer.hpp>
[52cac0]40#include <sstream>
[aa8ef2]41#include <string>
42
[ad011c]43#include "CodePatterns/MemDebug.hpp"
[bbbad5]44
[1b2d30]45#include "MpqcParser.hpp"
[c1db05]46#include "MpqcParser_Parameters.hpp"
[1b2d30]47
[6f0841]48#include "Atom/atom.hpp"
[41a467]49#include "CodePatterns/Log.hpp"
50#include "CodePatterns/toString.hpp"
[1b2d30]51#include "config.hpp"
[3bdb6d]52#include "Element/element.hpp"
[41a467]53#include "Element/periodentafel.hpp"
54#include "LinearAlgebra/Vector.hpp"
[aa8ef2]55#include "molecule.hpp"
[30f2815]56#include "Parser/Exceptions.hpp"
[1b2d30]57#include "World.hpp"
58
[765f16]59// declare specialized static variables
60const std::string FormatParserTrait<mpqc>::name = "mpqc";
61const std::string FormatParserTrait<mpqc>::suffix = "in";
62const ParserTypes FormatParserTrait<mpqc>::type = mpqc;
[1b2d30]63
[ee50c1]64// a converter we often need
65ConvertTo<bool> FormatParser<mpqc>::Converter;
[1b2d30]66
67/** Constructor of MpqcParser.
68 *
69 */
[765f16]70FormatParser< mpqc >::FormatParser() :
71 FormatParser_common(new MpqcParser_Parameters())
72{}
[1b2d30]73
74/** Destructor of MpqcParser.
75 *
76 */
[765f16]77FormatParser< mpqc >::~FormatParser()
78{}
[1b2d30]79
80/** Load an MPQC config file into the World.
81 * \param *file input stream
82 */
[765f16]83void FormatParser< mpqc >::load(istream *file)
[1b2d30]84{
[4cbca0]85 bool MpqcSection = false;
86 bool MoleculeSection = false;
87 bool GeometrySection = false;
[30f2815]88 bool GeometrySection_n = false;
[4cbca0]89 bool BasisSection = false;
90 bool AuxBasisSection = false;
[aa8ef2]91 char line[MAXSTRINGSIZE];
92 typedef boost::tokenizer<boost::char_separator<char> > tokenizer;
93 boost::char_separator<char> sep("[]");
[4cbca0]94 boost::char_separator<char> angularsep("<>");
[311da7b]95 boost::char_separator<char> equalitysep(" =");
[aa8ef2]96 boost::char_separator<char> whitesep(" \t");
97 ConvertTo<double> toDouble;
[30f2815]98 int old_n = -1; // note down the last parsed "n" to ascertain it's ascending
[aa8ef2]99
100 molecule *newmol = World::getInstance().createMolecule();
101 newmol->ActiveFlag = true;
102 while (file->good()) {
103 file->getline(line, MAXSTRINGSIZE-1);
104 std::string linestring(line);
[30f2815]105 if (((linestring.find("atoms") == string::npos)
106 || (linestring.find("geometry") == string::npos))
107 && (linestring.find("}") != string::npos)) {
[4cbca0]108 GeometrySection = false;
[aa8ef2]109 }
[4cbca0]110 if ((linestring.find(")") != string::npos)) { // ends a section which do not overlap
111 MpqcSection = false;
112 MoleculeSection = false;
113 BasisSection = false;
114 AuxBasisSection = false;
115 }
116 if (MoleculeSection) {
117 if (GeometrySection) { // we have an atom
[30f2815]118// LOG(2, "DEBUG: Full line is '" << linestring << "'.");
119 // separate off [..] part
[4cbca0]120 tokenizer tokens(linestring, sep);
[30f2815]121 // split part prior to [..] into tokens
122 std::string prior_part(*tokens.begin());
123 tokenizer prefixtokens(prior_part, whitesep);
124 tokenizer::iterator tok_iter = prefixtokens.begin();
125// LOG(2, "DEBUG: Current tok_iter is " << *tok_iter << ".");
126 if (GeometrySection_n) {
127 ASSERT(tok_iter != prefixtokens.end(),
128 "FormatParser< mpqc >::load() - missing n entry for MoleculeSection in line "
129 +linestring+"!");
130 // if additional n is given, parse and check but discard eventually
131 int n;
132 std::stringstream whitespacefilter(*tok_iter++);
133 whitespacefilter >> ws >> n;
134 if ((old_n != -1) && ((n-1) != old_n))
135 ELOG(2, "n index is not simply ascending by one but "
136 << n-old_n << ", specific ids are lost!");
137 if (old_n >= n) {
138 ELOG(1, "n index is not simply ascending, coordinates might get mixed!");
139 throw ParserException();
140 }
141 old_n = n;
142 }
143 ASSERT(tok_iter != prefixtokens.end(),
144 "FormatParser< mpqc >::load() - missing atoms entry for MoleculeSection in line "
145 +linestring+"!");
146// LOG(2, "DEBUG: Current tok_iter is " << *tok_iter << ".");
[4cbca0]147 std::string element;
[30f2815]148 {
149 std::stringstream whitespacefilter(*tok_iter++);
150 whitespacefilter >> ws >> element;
151 }
152 // split [..] part and parse
153 tok_iter = (++tokens.begin());
154 ASSERT (tok_iter != tokens.end(),
155 "FormatParser< mpqc >::load() - missing geometry entry for MoleculeSection in line "
156 +linestring+"!");
157// LOG(2, "DEBUG: Current tok_iter is " << *tok_iter << ".");
158 std::string vector(*tok_iter);
[4cbca0]159 tokenizer vectorcomponents(vector, whitesep);
160 Vector X;
[30f2815]161// if (vectorcomponents.size() != NDIM)
162// throw ParserException();
[4cbca0]163 tok_iter = vectorcomponents.begin();
164 for (int i=0; i<NDIM; ++i) {
[30f2815]165// LOG(2, "DEBUG: Current tok_iter is " << *tok_iter << ".");
[4cbca0]166 X[i] = toDouble(*tok_iter++);
167 }
168 // create atom
169 atom *newAtom = World::getInstance().createAtom();
170 newAtom->setType(World::getInstance().getPeriode()->FindElement(element));
171 newAtom->setPosition(X);
172 newmol->AddAtom(newAtom);
[47d041]173 LOG(1, "Adding atom " << *newAtom);
[4cbca0]174 }
175 }
176 if (MpqcSection) {
[311da7b]177 if (linestring.find("mole<") != string::npos) { // get theory
[4cbca0]178 tokenizer tokens(linestring, angularsep);
179 tokenizer::iterator tok_iter = tokens.begin();
[311da7b]180 ++tok_iter;
181 ASSERT(tok_iter != tokens.end(),
[765f16]182 "FormatParser< mpqc >::load() - missing token in brackets<> for mole< in line "+linestring+"!");
[311da7b]183 std::string value(*tok_iter);
[4cbca0]184 std::stringstream linestream("theory = "+value);
[c1db05]185 linestream >> getParams();
[4cbca0]186 } else if (linestring.find("integrals<") != string::npos) { // get theory
187 tokenizer tokens(linestring, angularsep);
188 tokenizer::iterator tok_iter = tokens.begin();
[311da7b]189 ++tok_iter;
190 ASSERT(tok_iter != tokens.end(),
[765f16]191 "FormatParser< mpqc >::load() - missing token in brackets<> for integrals< in line "+linestring+"!");
[311da7b]192 std::string value(*tok_iter);
[4cbca0]193 std::stringstream linestream("integration = "+value);
[c1db05]194 linestream >> getParams();
[311da7b]195 } else if ((linestring.find("molecule") == string::npos) && (linestring.find("basis") == string::npos)){
196 // molecule and basis must not be parsed in this section
197 tokenizer tokens(linestring, equalitysep);
198 tokenizer::iterator tok_iter = tokens.begin();
199 ASSERT(tok_iter != tokens.end(),
[765f16]200 "FormatParser< mpqc >::load() - missing token before '=' for MpqcSection in line "+linestring+"!");
[311da7b]201 std::stringstream whitespacefilter(*tok_iter);
202 std::string key;
203 whitespacefilter >> ws >> key;
[ee50c1]204 if (getParams().haveParameter(key)) {
[311da7b]205 std::stringstream linestream(linestring);
[c1db05]206 linestream >> getParams();
[311da7b]207 } else { // unknown keys are simply ignored as long as parser is incomplete
[47d041]208 LOG(2, "INFO: '"+key+"' is unknown and ignored.");
[311da7b]209 }
[4cbca0]210 }
211 }
212 if (BasisSection) {
213 tokenizer tokens(linestring, equalitysep);
214 tokenizer::iterator tok_iter = tokens.begin();
[311da7b]215 ASSERT(tok_iter != tokens.end(),
[765f16]216 "FormatParser< mpqc >::load() - missing token for BasisSection in line "+linestring+"!");
[4cbca0]217 std::string key(*tok_iter++);
[311da7b]218 ASSERT(tok_iter != tokens.end(),
[765f16]219 "FormatParser< mpqc >::load() - missing value for BasisSection after key "+key+" in line "+linestring+"!");
[4cbca0]220 std::string value(*tok_iter);
[311da7b]221 tok_iter++;
[4cbca0]222 // TODO: use exception instead of ASSERT
223 ASSERT(tok_iter == tokens.end(),
[765f16]224 "FormatParser< mpqc >::load() - more than (key = value) on line "+linestring+".");
[4cbca0]225 if (key == "name") {
226 std::stringstream linestream("basis = "+value);
[c1db05]227 linestream >> getParams();
[4cbca0]228 }
229 }
230 if (AuxBasisSection) {
231 tokenizer tokens(linestring, equalitysep);
232 tokenizer::iterator tok_iter = tokens.begin();
[311da7b]233 ASSERT(tok_iter != tokens.end(),
[765f16]234 "FormatParser< mpqc >::load() - missing token for AuxBasisSection in line "+linestring+"!");
[4cbca0]235 std::string key(*tok_iter++);
[311da7b]236 ASSERT(tok_iter != tokens.end(),
[765f16]237 "FormatParser< mpqc >::load() - missing value for BasisSection after key "+key+" in line "+linestring+"!");
[4cbca0]238 std::string value(*tok_iter);
[311da7b]239 tok_iter++;
[4cbca0]240 // TODO: use exception instead of ASSERT
241 ASSERT(tok_iter == tokens.end(),
[765f16]242 "FormatParser< mpqc >::load() - more than (key = value) on line "+linestring+".");
[4cbca0]243 if (key == "name") {
244 std::stringstream linestream("aux_basis = "+value);
[c1db05]245 linestream >> getParams();
[aa8ef2]246 }
247 }
248 // set some scan flags
[4cbca0]249 if (linestring.find("mpqc:") != string::npos) {
250 MpqcSection = true;
251 }
252 if (linestring.find("molecule<Molecule>:") != string::npos) {
253 MoleculeSection = true;
254 }
[30f2815]255 if ((linestring.find("atoms") != string::npos)
256 && (linestring.find("geometry") != string::npos)) {
[4cbca0]257 GeometrySection = true;
[30f2815]258 if (linestring.find("n") != string::npos) // neither atoms nor geometry contains a letter n
259 GeometrySection_n = true;
[4cbca0]260 }
261 if ((linestring.find("basis<GaussianBasisSet>:") != string::npos) && ((linestring.find("abasis<") == string::npos))) {
262 BasisSection = true;
263 }
264 if (linestring.find("abasis<") != string::npos) {
265 AuxBasisSection = true;
[aa8ef2]266 }
267 }
[4afa46]268 // refresh atom::nr and atom::name
269 newmol->getAtomCount();
[1b2d30]270}
271
[03a589]272void FormatParser< mpqc >::OutputMPQCLine(ostream * const out, const atom &_atom, const Vector *center) const
273{
274 Vector recentered(_atom.getPosition());
275 recentered -= *center;
[52cac0]276 *out << "\t\t" << _atom.getType()->getSymbol() << " [ ";
277 {
278 std::stringstream posstream;
279 posstream << std::setprecision(12) << recentered[0] << "\t" << recentered[1] << "\t" << recentered[2];
280 *out << posstream.str();
281 }
282 *out << " ]" << std::endl;
[03a589]283};
284
285
[f31edc]286/** Saves all atoms and data into a MPQC config file.
[1b2d30]287 * \param *file output stream
[73916f]288 * \param atoms atoms to store
[1b2d30]289 */
[fac58f]290void FormatParser< mpqc >::save(ostream *file, const std::vector<const atom *> &atoms)
[1b2d30]291{
292 Vector center;
[a58c16]293// vector<const atom *> allatoms = const_cast<const World &>(World::getInstance()).
294// getAllAtoms();
[1b2d30]295
[d6b8e1]296 // calculate center
[a58c16]297// for (std::vector<const atom *>::const_iterator runner = atoms.begin();runner != atoms.end(); ++runner)
[e9dc19]298// center += (*runner)->getPosition();
299// center.Scale(1./(double)atoms.size());
300 center.Zero();
[d6b8e1]301
[1b2d30]302 // first without hessian
303 if (file->fail()) {
[47d041]304 ELOG(1, "Cannot open mpqc output file.");
[1b2d30]305 } else {
306 *file << "% Created by MoleCuilder" << endl;
307 *file << "mpqc: (" << endl;
[e3c4c5]308 *file << "\tcheckpoint = no" << endl;
[3fb9ab]309 *file << "\trestart = no" << endl;
[ee50c1]310 *file << "\tsavestate = " << getParams().getParameter(MpqcParser_Parameters::savestateParam) << endl;
311 *file << "\tdo_gradient = " << getParams().getParameter(MpqcParser_Parameters::do_gradientParam) << endl;
312 if (Converter(getParams().getParameter(MpqcParser_Parameters::hessianParam))) {
[f31edc]313 *file << "\tfreq<MolecularFrequencies>: (" << endl;
314 *file << "\t\tmolecule=$:molecule" << endl;
315 *file << "\t)" << endl;
316 }
[ee50c1]317 const std::string theory = getParams().getParameter(MpqcParser_Parameters::theoryParam);
318 if (theory == getParams().getTheoryName(MpqcParser_Parameters::CLHF)) {
319 *file << "\tmole<" << getParams().getParameter(MpqcParser_Parameters::theoryParam) << ">: (" << endl;
320 *file << "\t\tmolecule = $:molecule" << endl;
321 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::basisParam) << " = $:basis" << endl;
322 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::maxiterParam)
323 << " = " << getParams().getParameter(MpqcParser_Parameters::maxiterParam)<< endl;
324 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::memoryParam)
325 << " = " << getParams().getParameter(MpqcParser_Parameters::memoryParam) << endl;
326 *file << "\t)" << endl;
327 } else if (theory == getParams().getTheoryName(MpqcParser_Parameters::CLKS)) {
328 *file << "\tmole<" << getParams().getParameter(MpqcParser_Parameters::theoryParam) << ">: (" << endl;
329 *file << "\t\tfunctional<StdDenFunctional>:(name=B3LYP)" << endl;
330 *file << "\t\tmolecule = $:molecule" << endl;
331 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::basisParam) << " = $:basis" << endl;
332 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::maxiterParam)
333 << " = " << getParams().getParameter(MpqcParser_Parameters::maxiterParam)<< endl;
334 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::memoryParam)
335 << " = " << getParams().getParameter(MpqcParser_Parameters::memoryParam) << endl;
336 *file << "\t)" << endl;
337 } else if (theory == getParams().getTheoryName(MpqcParser_Parameters::MBPT2)) {
338 *file << "\tmole<" << getParams().getParameter(MpqcParser_Parameters::theoryParam) << ">: (" << endl;
339 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::basisParam) << " = $:basis" << endl;
340 *file << "\t\tmolecule = $:molecule" << endl;
341 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::memoryParam)
342 << " = " << getParams().getParameter(MpqcParser_Parameters::memoryParam) << endl;
343 *file << "\t\treference<CLHF>: (" << endl;
344 *file << "\t\t\t" << getParams().getParameterName(MpqcParser_Parameters::maxiterParam)
345 << " = " << getParams().getParameter(MpqcParser_Parameters::maxiterParam)<< endl;
346 *file << "\t\t\t" << getParams().getParameterName(MpqcParser_Parameters::basisParam) << " = $:basis" << endl;
347 *file << "\t\t\tmolecule = $:molecule" << endl;
348 *file << "\t\t\t" << getParams().getParameterName(MpqcParser_Parameters::memoryParam)
349 << " = " << getParams().getParameter(MpqcParser_Parameters::memoryParam) << endl;
350 *file << "\t\t)" << endl;
351 *file << "\t)" << endl;
352 } else if (theory == getParams().getTheoryName(MpqcParser_Parameters::MBPT2_R12)) {
353 *file << "\tmole<" << getParams().getParameter(MpqcParser_Parameters::theoryParam) << ">: (" << endl;
354 *file << "\t\tmolecule = $:molecule" << endl;
355 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::basisParam) << " = $:basis" << endl;
356 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::aux_basisParam) << " = $:abasis" << endl;
357 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::stdapproxParam)
358 << " = \"" << getParams().getParameter(MpqcParser_Parameters::stdapproxParam) << "\"" << endl;
359 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::nfzcParam)
360 << " = " << getParams().getParameter(MpqcParser_Parameters::nfzcParam) << endl;
361 *file << "\t\t" << getParams().getParameterName(MpqcParser_Parameters::memoryParam)
362 << " = " << getParams().getParameter(MpqcParser_Parameters::memoryParam) << endl;
363 *file << "\t\tintegrals<IntegralCints>:()" << endl;
364 *file << "\t\treference<CLHF>: (" << endl;
365 *file << "\t\t\tmolecule = $:molecule" << endl;
366 *file << "\t\t\t" << getParams().getParameterName(MpqcParser_Parameters::basisParam) << " = $:basis" << endl;
367 *file << "\t\t\tmaxiter = " << getParams().getParameter(MpqcParser_Parameters::maxiterParam) << endl;
368 *file << "\t\t\tmemory = " << getParams().getParameter(MpqcParser_Parameters::memoryParam) << endl;
369 *file << "\t\t\tintegrals<" << getParams().getParameter(MpqcParser_Parameters::integrationParam) << ">:()" << endl;
370 *file << "\t\t)" << endl;
371 *file << "\t)" << endl;
372 } else {
[47d041]373 ELOG(0, "Unknown level of theory requested for MPQC output file.");
[1b2d30]374 }
[e3c4c5]375 const std::string jobtype = getParams().getParameter(MpqcParser_Parameters::jobtypeParam);
376 if (jobtype == getParams().getJobtypeName(MpqcParser_Parameters::Optimization)) {
377 *file << "\t% optimizer object for the molecular geometry" << endl;
378 *file << "\topt<QNewtonOpt>: (" << endl;
379 *file << "\t\tfunction = $..:mole" << endl;
380 *file << "\t\tupdate<BFGSUpdate>: ()" << endl;
381 *file << "\t\tconvergence<MolEnergyConvergence>: (" << endl;
382 *file << "\t\t\tcartesian = yes" << endl;
383 *file << "\t\t\tenergy = $..:..:mole" << endl;
384 *file << "\t\t)" << endl;
385 *file << "\t)" << endl;
386 }
[1b2d30]387 *file << ")" << endl;
388 *file << "molecule<Molecule>: (" << endl;
389 *file << "\tunit = " << (World::getInstance().getConfig()->GetIsAngstroem() ? "angstrom" : "bohr" ) << endl;
390 *file << "\t{ atoms geometry } = {" << endl;
391 // output of atoms
[03a589]392 BOOST_FOREACH(const atom *_atom, atoms) {
393 OutputMPQCLine(file, *_atom, &center);
[1b2d30]394 }
395 *file << "\t}" << endl;
396 *file << ")" << endl;
397 *file << "basis<GaussianBasisSet>: (" << endl;
[ee50c1]398 *file << "\tname = \"" << getParams().getParameter(MpqcParser_Parameters::basisParam) << "\"" << endl;
[1b2d30]399 *file << "\tmolecule = $:molecule" << endl;
400 *file << ")" << endl;
[ee50c1]401 if (theory == getParams().getTheoryName(MpqcParser_Parameters::MBPT2_R12)) {
[f31edc]402 *file << "% auxiliary basis set specification" << endl;
403 *file << "\tabasis<GaussianBasisSet>: (" << endl;
[ee50c1]404 *file << "\tname = \"" << getParams().getParameter(MpqcParser_Parameters::aux_basisParam) << "\"" << endl;
[f31edc]405 *file << "\tmolecule = $:molecule" << endl;
406 *file << ")" << endl;
407 }
[1b2d30]408 }
409}
410
[963321a]411
Note: See TracBrowser for help on using the repository browser.