/* * Project: MoleCuilder * Description: creates and alters molecular systems * Copyright (C) 2010-2012 University of Bonn. All rights reserved. * Copyright (C) 2013 Frederik Heber. All rights reserved. * * * This file is part of MoleCuilder. * * MoleCuilder is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 2 of the License, or * (at your option) any later version. * * MoleCuilder is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with MoleCuilder. If not, see . */ /* * Psi3Parser.cpp * * Created on: Oct 04, 2011 * Author: heber */ // include config.h #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include "CodePatterns/MemDebug.hpp" #include "Psi3Parser.hpp" #include "Psi3Parser_Parameters.hpp" #include "Atom/atom.hpp" #include "Bond/bond.hpp" #include "CodePatterns/Log.hpp" #include "CodePatterns/toString.hpp" #include "Element/element.hpp" #include "Element/periodentafel.hpp" #include "LinearAlgebra/Vector.hpp" #include "molecule.hpp" #include "MoleculeListClass.hpp" #include "World.hpp" // declare specialized static variables const std::string FormatParserTrait::name = "psi3"; const std::string FormatParserTrait::suffix = "psi"; const ParserTypes FormatParserTrait::type = psi3; // a converter we often need ConvertTo FormatParser::Converter; /** Constructor of Psi3Parser. * */ FormatParser< psi3 >::FormatParser() : FormatParser_common(new Psi3Parser_Parameters()) {} /** Destructor of Psi3Parser. * */ FormatParser< psi3 >::~FormatParser() {} /** Load an PSI3 config file into the World. * \param *file input stream */ void FormatParser< psi3 >::load(istream *file) { bool Psi3Section = false; bool GeometrySection = false; char line[MAXSTRINGSIZE]; typedef boost::tokenizer > tokenizer; boost::char_separator sep("()"); boost::char_separator angularsep("<>"); boost::char_separator equalitysep(" ="); boost::char_separator whitesep(" \t"); ConvertTo toDouble; molecule *newmol = World::getInstance().createMolecule(); newmol->ActiveFlag = true; // TODO: Remove the insertion into molecule when saving does not depend on them anymore. Also, remove molecule.hpp include World::getInstance().getMolecules()->insert(newmol); while (file->good()) { file->getline(line, MAXSTRINGSIZE-1); std::string linestring(line); LOG(3, "INFO: Current line is: " << line); if ((linestring.find(")") != string::npos) && (linestring.find("(") == string::npos)) { LOG(3, "INFO: Line contains ')' and no '(' (end of section): " << line); // ends a section which do not overlap if (GeometrySection) GeometrySection = false; else Psi3Section = false; } if (GeometrySection) { // we have an atom tokenizer tokens(linestring, sep); // if (tokens.size() != 2) // throw Psi3ParseException; tokenizer::iterator tok_iter = tokens.begin(); ASSERT(tok_iter != tokens.end(), "FormatParser< psi3 >::load() - missing token for MoleculeSection in line "+linestring+"!"); std::stringstream whitespacefilter(*++tok_iter); std::string element; whitespacefilter >> ws >> element; LOG(2, "INFO: element of atom is " << element); ASSERT(tok_iter != tokens.end(), "FormatParser< psi3 >::load() - missing token for MoleculeSection in line "+linestring+"!"); std::string vector = *tok_iter; tokenizer vectorcomponents(vector, whitesep); Vector X; // if (vectorcomponents.size() != NDIM) // throw Psi3ParseException; tok_iter = vectorcomponents.begin(); ++tok_iter; for (int i=0; isetType(World::getInstance().getPeriode()->FindElement(element)); newAtom->setPosition(X); newmol->AddAtom(newAtom); LOG(1, "Adding atom " << *newAtom); } if ((Psi3Section) && (!GeometrySection)) { if (linestring.find("=") != string::npos) { // get param value tokenizer tokens(linestring, equalitysep); tokenizer::iterator tok_iter = tokens.begin(); ASSERT(tok_iter != tokens.end(), "FormatParser< psi3 >::load() - missing token before '=' for Psi3Section in line "+linestring+"!"); std::stringstream whitespacefilter(*tok_iter); std::string key; whitespacefilter >> ws >> key; //LOG(2, "INFO: key to check is: " << key); if (getParams().haveParameter(key)) { //LOG(2, "INFO: Line submitted to parameter is: " << linestring); std::stringstream linestream(linestring); linestream >> getParams(); } else { // unknown keys are simply ignored as long as parser is incomplete LOG(3, "INFO: '"+key+"' is unknown and ignored."); } } } if ((linestring.find("geometry") != string::npos) && (linestring.find("(") != string::npos)) { LOG(3, "INFO: Line contains geometry and '(': " << line); GeometrySection = true; } if ((linestring.find("psi:") != string::npos) && (linestring.find("(") != string::npos)) { LOG(3, "INFO: Line contains psi: and '(': " << line); Psi3Section = true; } } // refresh atom::nr and atom::name newmol->getAtomCount(); } void FormatParser< psi3 >::OutputPsi3Line(ostream * const out, const atom &_atom, const Vector *center) const { Vector recentered(_atom.getPosition()); recentered -= *center; *out << "\t( " << _atom.getType()->getSymbol() << "\t" << recentered[0] << "\t" << recentered[1] << "\t" << recentered[2] << " )" << endl; }; /** Saves all atoms and data into a PSI3 config file. * \param *file output stream * \param atoms atoms to store */ void FormatParser< psi3 >::save(ostream *file, const std::vector &atoms) { Vector center; // vector allatoms = World::getInstance().getAllAtoms(); // calculate center for (std::vector::const_iterator runner = atoms.begin();runner != atoms.end(); ++runner) center += (*runner)->getPosition(); center.Scale(1./(double)atoms.size()); // first without hessian if (file->fail()) { ELOG(1, "Cannot open psi3 output file."); } else { *file << "% Created by MoleCuilder" << std::endl; *file << "psi: (" << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::labelParam) << " = \"" << getParams().getParameter(Psi3Parser_Parameters::labelParam) << "\"" << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::jobtypeParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::jobtypeParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::wavefunctionParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::wavefunctionParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::maxiterParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::maxiterParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::referenceParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::referenceParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::basisParam) << " = \"" << getParams().getParameter(Psi3Parser_Parameters::basisParam) << "\"" << std::endl; const std::string reference = getParams().getParameter(Psi3Parser_Parameters::referenceParam); // if (reference == getParams().getReferenceName(Psi3Parser_Parameters::RHF)) { // } // if (reference == getParams().getReferenceName(Psi3Parser_Parameters::ROHF)) { // } if ((reference == getParams().getReferenceName(Psi3Parser_Parameters::UHF)) || (reference == getParams().getReferenceName(Psi3Parser_Parameters::TWOCON))) { const unsigned int multiplicity = calculateMultiplicity(atoms); *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::multiplicityParam) << " = " << multiplicity << std::endl; // << " = " << getParams().getParameter(Psi3Parser_Parameters::multiplicityParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::chargeParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::chargeParam) << std::endl; } if (reference == getParams().getReferenceName(Psi3Parser_Parameters::TWOCON)) { *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::soccParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::soccParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::doccParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::doccParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::subgroupParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::subgroupParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::unique_axisParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::unique_axisParam) << std::endl; } if ((reference != getParams().getReferenceName(Psi3Parser_Parameters::RHF)) && (reference != getParams().getReferenceName(Psi3Parser_Parameters::ROHF)) && (reference != getParams().getReferenceName(Psi3Parser_Parameters::UHF)) && (reference != getParams().getReferenceName(Psi3Parser_Parameters::TWOCON))) { ELOG(0, "Unknown level of reference requested for Psi3 output file."); } *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::freeze_coreParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::freeze_coreParam) << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::unitsParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::unitsParam) << std::endl; *file << "\tgeometry = (" << std::endl; // output of atoms BOOST_FOREACH(const atom *_atom, atoms) { OutputPsi3Line(file, *_atom, ¢er); } *file << "\t)" << std::endl; *file << "\t" << getParams().getParameterName(Psi3Parser_Parameters::originParam) << " = " << getParams().getParameter(Psi3Parser_Parameters::originParam) << std::endl; *file << ")" << std::endl; } } unsigned int FormatParser< psi3 >::calculateMultiplicity(const std::vector &atoms) const { // add up the number of electrons double valencies = 0.; BOOST_FOREACH(atom *_atom, atoms) { valencies += _atom->getType()->getAtomicNumber(); } // add doubly up all bond degrees (two electrons per bond) unsigned int degrees = 0; BOOST_FOREACH(atom *_atom, atoms) { BOOST_FOREACH(bond::ptr _bond, _atom->getListOfBonds()) { degrees += 2*_bond->getDegree(); } } // return difference+1 as multiplicity return (int)fabs(valencies-degrees)+1; }