/*
* 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 .
*/
/*
* DepthFirstSearchAnalysis.cpp
*
* Created on: Feb 16, 2011
* Author: heber
*/
// include config.h
#ifdef HAVE_CONFIG_H
#include
#endif
#include "CodePatterns/MemDebug.hpp"
#include "DepthFirstSearchAnalysis.hpp"
#include
#include
#include "Atom/atom.hpp"
#include "Bond/bond.hpp"
#include "CodePatterns/Assert.hpp"
#include "CodePatterns/Info.hpp"
#include "CodePatterns/Log.hpp"
#include "CodePatterns/Verbose.hpp"
#include "Descriptors/AtomDescriptor.hpp"
#include "Descriptors/MoleculeDescriptor.hpp"
#include "Graph/ListOfLocalAtoms.hpp"
#include "molecule.hpp"
#include "MoleculeLeafClass.hpp"
#include "MoleculeListClass.hpp"
#include "World.hpp"
DepthFirstSearchAnalysis::DepthFirstSearchAnalysis() :
CurrentGraphNr(0),
ComponentNumber(0),
BackStepping(false)
{
ResetAllBondsToUnused();
}
DepthFirstSearchAnalysis::~DepthFirstSearchAnalysis()
{}
void DepthFirstSearchAnalysis::Init()
{
CurrentGraphNr = 0;
ComponentNumber = 0;
BackStepping = false;
std::for_each(World::getInstance().getAtomIter(),World::getInstance().atomEnd(),
std::mem_fun(&atom::resetGraphNr));
std::for_each(World::getInstance().getAtomIter(),World::getInstance().atomEnd(),
std::mem_fun(&atom::InitComponentNr));
}
bond::ptr DepthFirstSearchAnalysis::FindNextUnused(atom *vertex) const
{
const BondList& ListOfBonds = vertex->getListOfBonds();
for (BondList::const_iterator Runner = ListOfBonds.begin();
Runner != ListOfBonds.end();
++Runner)
if ((*Runner)->IsUsed() == GraphEdge::white)
return ((*Runner));
return bond::ptr();
}
void DepthFirstSearchAnalysis::ResetAllBondsToUnused() const
{
World::AtomComposite allatoms = World::getInstance().getAllAtoms();
for(World::AtomComposite::const_iterator AtomRunner = allatoms.begin();
AtomRunner != allatoms.end();
++AtomRunner) {
const BondList& ListOfBonds = (*AtomRunner)->getListOfBonds();
for(BondList::const_iterator BondRunner = ListOfBonds.begin();
BondRunner != ListOfBonds.end();
++BondRunner)
if ((*BondRunner)->leftatom == *AtomRunner)
(*BondRunner)->ResetUsed();
}
}
void DepthFirstSearchAnalysis::SetNextComponentNumber(atom *vertex, int nr) const
{
size_t i = 0;
ASSERT(vertex != NULL,
"DepthFirstSearchAnalysis::SetNextComponentNumber() - Given vertex is NULL!");
const BondList& ListOfBonds = vertex->getListOfBonds();
for (; i < ListOfBonds.size(); i++) {
if (vertex->ComponentNr[i] == -1) { // check if not yet used
vertex->ComponentNr[i] = nr;
break;
} else if (vertex->ComponentNr[i] == nr) // if number is already present, don't add another time
break; // breaking here will not cause error!
}
ASSERT(i < ListOfBonds.size(),
"DepthFirstSearchAnalysis::SetNextComponentNumber() - All Component entries are already occupied!");
}
bool DepthFirstSearchAnalysis::PickLocalBackEdges(const ListOfLocalAtoms_t &ListOfLocalAtoms, std::deque *&LocalStack) const
{
bool status = true;
if (BackEdgeStack.empty()) {
ELOG(1, "Reference BackEdgeStack is empty!");
return false;
}
std::deque MyBackEdgeStack = BackEdgeStack;
do { // go through all bonds and push local ones
const bond::ptr &Binder = MyBackEdgeStack.front(); // loop the stack for next item
MyBackEdgeStack.pop_front();
LOG(3, "INFO: Current candidate edge " << *Binder << ".");
const ListOfLocalAtoms_t::const_iterator leftiter = ListOfLocalAtoms.find(Binder->leftatom->getNr());
ASSERT( leftiter != ListOfLocalAtoms.end(),
"DepthFirstSearchAnalysis::PickLocalBackEdges() - could not find atom id "
+toString(Binder->leftatom->getNr())+" in ListOfLocalAtoms.");
atom * const Walker = leftiter->second; // get one atom in the reference molecule
if (Walker != NULL) { // if this Walker exists in the subgraph ...
const BondList& ListOfBonds = Walker->getListOfBonds();
for (BondList::const_iterator Runner = ListOfBonds.begin();
Runner != ListOfBonds.end();
++Runner) {
atom * const OtherAtom = (*Runner)->GetOtherAtom(Walker);
const ListOfLocalAtoms_t::const_iterator rightiter = ListOfLocalAtoms.find((*Runner)->rightatom->getNr());
if (OtherAtom == rightiter->second) { // found the bond
LocalStack->push_front((*Runner));
LOG(3, "INFO: Found local edge " << *(*Runner) << ".");
break;
}
}
}
} while (!MyBackEdgeStack.empty());
return status;
}
void DepthFirstSearchAnalysis::OutputGraphInfoPerAtom() const
{
LOG(1, "Final graph info for each atom is:");
World::ConstAtomComposite allatoms = const_cast(World::getInstance()).
getAllAtoms();
for_each(allatoms.begin(),allatoms.end(),mem_fun(&atom::OutputGraphInfo));
}
void DepthFirstSearchAnalysis::OutputGraphInfoPerBond() const
{
LOG(1, "Final graph info for each bond is:");
World::ConstAtomComposite allatoms = const_cast(World::getInstance()).
getAllAtoms();
for(World::ConstAtomComposite::const_iterator AtomRunner = allatoms.begin();
AtomRunner != allatoms.end();
++AtomRunner) {
const BondList& ListOfBonds = (*AtomRunner)->getListOfBonds();
for(BondList::const_iterator BondRunner = ListOfBonds.begin();
BondRunner != ListOfBonds.end();
++BondRunner)
if ((*BondRunner)->leftatom == *AtomRunner) {
const bond::ptr Binder = *BondRunner;
if (DoLog(2)) {
std::stringstream output;
output << ((Binder->Type == GraphEdge::TreeEdge) ? "TreeEdge " : "BackEdge ") << *Binder << ": <";
output << ((Binder->leftatom->SeparationVertex) ? "SP," : "") << "L" << Binder->leftatom->LowpointNr << " G" << Binder->leftatom->GraphNr << " Comp.";
Binder->leftatom->OutputComponentNumber(&output);
output << " === ";
output << ((Binder->rightatom->SeparationVertex) ? "SP," : "") << "L" << Binder->rightatom->LowpointNr << " G" << Binder->rightatom->GraphNr << " Comp.";
Binder->rightatom->OutputComponentNumber(&output);
output << ">.";
LOG(2, output.str());
}
if (Binder->Cyclic) // cyclic ??
LOG(3, "Lowpoint at each side are equal: CYCLIC!");
}
}
}
unsigned int DepthFirstSearchAnalysis::CyclicBondAnalysis() const
{
unsigned int NoCyclicBonds = 0;
World::ConstAtomComposite allatoms = const_cast(World::getInstance()).
getAllAtoms();
for(World::ConstAtomComposite::const_iterator AtomRunner = allatoms.begin();
AtomRunner != allatoms.end();
++AtomRunner) {
const BondList& ListOfBonds = (*AtomRunner)->getListOfBonds();
for(BondList::const_iterator BondRunner = ListOfBonds.begin();
BondRunner != ListOfBonds.end();
++BondRunner)
if ((*BondRunner)->leftatom == *AtomRunner)
if ((*BondRunner)->rightatom->LowpointNr == (*BondRunner)->leftatom->LowpointNr) { // cyclic ??
(*BondRunner)->Cyclic = true;
NoCyclicBonds++;
}
}
return NoCyclicBonds;
}
void DepthFirstSearchAnalysis::SetWalkersGraphNr(atom *&Walker)
{
if (!BackStepping) { // if we don't just return from (8)
Walker->GraphNr = CurrentGraphNr;
Walker->LowpointNr = CurrentGraphNr;
LOG(1, "Setting Walker[" << Walker->getName() << "]'s number to " << Walker->GraphNr << " with Lowpoint " << Walker->LowpointNr << ".");
AtomStack.push_front(Walker);
CurrentGraphNr++;
}
}
void DepthFirstSearchAnalysis::ProbeAlongUnusedBond(atom *&Walker, bond::ptr &Binder)
{
atom *OtherAtom = NULL;
do { // (3) if Walker has no unused egdes, go to (5)
BackStepping = false; // reset backstepping flag for (8)
if (Binder == NULL) // if we don't just return from (11), Binder is already set to next unused
Binder = FindNextUnused(Walker);
if (Binder == NULL)
break;
LOG(2, "Current Unused Bond is " << *Binder << ".");
// (4) Mark Binder used, ...
Binder->MarkUsed(GraphEdge::black);
OtherAtom = Binder->GetOtherAtom(Walker);
LOG(2, "(4) OtherAtom is " << OtherAtom->getName() << ".");
if (OtherAtom->GraphNr != -1) {
// (4a) ... if "other" atom has been visited (GraphNr != 0), set lowpoint to minimum of both, go to (3)
Binder->Type = GraphEdge::BackEdge;
BackEdgeStack.push_front(Binder);
Walker->LowpointNr = (Walker->LowpointNr < OtherAtom->GraphNr) ? Walker->LowpointNr : OtherAtom->GraphNr;
LOG(3, "(4a) Visited: Setting Lowpoint of Walker[" << Walker->getName() << "] to " << Walker->LowpointNr << ".");
} else {
// (4b) ... otherwise set OtherAtom as Ancestor of Walker and Walker as OtherAtom, go to (2)
Binder->Type = GraphEdge::TreeEdge;
OtherAtom->Ancestor = Walker;
Walker = OtherAtom;
LOG(3, "(4b) Not Visited: OtherAtom[" << OtherAtom->getName() << "]'s Ancestor is now " << OtherAtom->Ancestor->getName() << ", Walker is OtherAtom " << OtherAtom->getName() << ".");
break;
}
Binder.reset();
} while (1); // (3)
}
void DepthFirstSearchAnalysis::CheckForaNewComponent(atom *&Walker, ConnectedSubgraph &Subgraph)
{
atom *OtherAtom = NULL;
// (5) if Ancestor of Walker is ...
LOG(1, "(5) Number of Walker[" << Walker->getName() << "]'s Ancestor[" << Walker->Ancestor->getName() << "] is " << Walker->Ancestor->GraphNr << ".");
if (Walker->Ancestor->GraphNr != Root->GraphNr) {
// (6) (Ancestor of Walker is not Root)
if (Walker->LowpointNr < Walker->Ancestor->GraphNr) {
// (6a) set Ancestor's Lowpoint number to minimum of of its Ancestor and itself, go to Step(8)
Walker->Ancestor->LowpointNr = (Walker->Ancestor->LowpointNr < Walker->LowpointNr) ? Walker->Ancestor->LowpointNr : Walker->LowpointNr;
LOG(2, "(6) Setting Walker[" << Walker->getName() << "]'s Ancestor[" << Walker->Ancestor->getName() << "]'s Lowpoint to " << Walker->Ancestor->LowpointNr << ".");
} else {
// (7) (Ancestor of Walker is a separating vertex, remove all from stack till Walker (including), these and Ancestor form a component
Walker->Ancestor->SeparationVertex = true;
LOG(2, "(7) Walker[" << Walker->getName() << "]'s Ancestor[" << Walker->Ancestor->getName() << "]'s is a separating vertex, creating component.");
SetNextComponentNumber(Walker->Ancestor, ComponentNumber);
LOG(3, "(7) Walker[" << Walker->getName() << "]'s Ancestor's Compont is " << ComponentNumber << ".");
SetNextComponentNumber(Walker, ComponentNumber);
LOG(3, "(7) Walker[" << Walker->getName() << "]'s Compont is " << ComponentNumber << ".");
do {
ASSERT(!AtomStack.empty(), "DepthFirstSearchAnalysis_CheckForaNewComponent() - AtomStack is empty!");
OtherAtom = AtomStack.front();
AtomStack.pop_front();
Subgraph.push_back(OtherAtom);
SetNextComponentNumber(OtherAtom, ComponentNumber);
LOG(3, "(7) Other[" << OtherAtom->getName() << "]'s Compont is " << ComponentNumber << ".");
} while (OtherAtom != Walker);
ComponentNumber++;
}
// (8) Walker becomes its Ancestor, go to (3)
LOG(2, "(8) Walker[" << Walker->getName() << "] is now its Ancestor " << Walker->Ancestor->getName() << ", backstepping. ");
Walker = Walker->Ancestor;
BackStepping = true;
}
}
void DepthFirstSearchAnalysis::CleanRootStackDownTillWalker(atom *&Walker, bond::ptr &Binder, ConnectedSubgraph &Subgraph)
{
atom *OtherAtom = NULL;
if (!BackStepping) { // coming from (8) want to go to (3)
// (9) remove all from stack till Walker (including), these and Root form a component
//AtomStack.Output(out);
SetNextComponentNumber(Root, ComponentNumber);
LOG(3, "(9) Root[" << Root->getName() << "]'s Component is " << ComponentNumber << ".");
SetNextComponentNumber(Walker, ComponentNumber);
LOG(3, "(9) Walker[" << Walker->getName() << "]'s Component is " << ComponentNumber << ".");
do {
ASSERT(!AtomStack.empty(), "DepthFirstSearchAnalysis::CleanRootStackDownTillWalker() - AtomStack is empty!");
OtherAtom = AtomStack.front();
AtomStack.pop_front();
Subgraph.push_back(OtherAtom);
SetNextComponentNumber(OtherAtom, ComponentNumber);
LOG(3, "(7) Other[" << OtherAtom->getName() << "]'s Component is " << ComponentNumber << ".");
} while (OtherAtom != Walker);
ComponentNumber++;
// (11) Root is separation vertex, set Walker to Root and go to (4)
Walker = Root;
Binder = FindNextUnused(Walker);
if (Binder != NULL) { // Root is separation vertex
LOG(1, "(10) Walker is Root[" << Root->getName() << "], next Unused Bond is " << *Binder << ".");
LOG(1, "(11) Root is a separation vertex.");
Walker->SeparationVertex = true;
} else {
LOG(1, "(10) Walker is Root[" << Root->getName() << "], no next Unused Bond.");
}
}
}
const std::deque& DepthFirstSearchAnalysis::getBackEdgeStack() const
{
return BackEdgeStack;
}
void DepthFirstSearchAnalysis::operator()()
{
Info FunctionInfo("DepthFirstSearchAnalysis");
ListOfConnectedSubgraphs.clear();
int OldGraphNr = 0;
atom *Walker = NULL;
bond::ptr Binder;
if (World::getInstance().numAtoms() == 0)
return;
Init();
LOG(0, "STATUS: Start walking the bond graph.");
for(World::AtomIterator iter = World::getInstance().getAtomIter();
iter != World::getInstance().atomEnd();) { // don't advance, is done at the end
Root = *iter;
// (1) mark all edges unused, empty stack, set atom->GraphNr = -1 for all
AtomStack.clear();
// put into new subgraph molecule and add this to list of subgraphs
ConnectedSubgraph CurrentSubgraph;
CurrentSubgraph.push_back(Root);
OldGraphNr = CurrentGraphNr;
Walker = Root;
do { // (10)
do { // (2) set number and Lowpoint of Atom to i, increase i, push current atom
SetWalkersGraphNr(Walker);
ProbeAlongUnusedBond(Walker, Binder);
if (Binder == NULL) {
LOG(2, "No more Unused Bonds.");
break;
} else
Binder.reset();
} while (1); // (2)
// if we came from backstepping, yet there were no more unused bonds, we end up here with no Ancestor, because Walker is Root! Then we are finished!
if ((Walker == Root) && (Binder == NULL))
break;
CheckForaNewComponent( Walker, CurrentSubgraph);
CleanRootStackDownTillWalker(Walker, Binder, CurrentSubgraph);
} while ((BackStepping) || (Binder != NULL)); // (10) halt only if Root has no unused edges
ListOfConnectedSubgraphs.push_back(CurrentSubgraph);
// From OldGraphNr to CurrentGraphNr ranges an disconnected subgraph
std::stringstream output;
output << CurrentSubgraph;
LOG(1, "INFO: Disconnected subgraph ranges from " << OldGraphNr << " to "
<< CurrentGraphNr-1 << ": " << output.str());
// step on to next root
while (iter != World::getInstance().atomEnd()) {
if ((*iter)->GraphNr != -1) { // if already discovered, step on
iter++;
} else {
LOG(1,"Current next subgraph root candidate is " << (*iter)->getName()
<< " with GraphNr " << (*iter)->GraphNr << ".");
break;
}
}
}
LOG(0, "STATUS: Done walking the bond graph.");
// set cyclic bond criterium on "same LP" basis
CyclicBondAnalysis();
OutputGraphInfoPerAtom();
OutputGraphInfoPerBond();
}
void DepthFirstSearchAnalysis::UpdateMoleculeStructure() const
{
// remove all of World's molecules
for (World::MoleculeIterator iter = World::getInstance().getMoleculeIter();
World::getInstance().getMoleculeIter() != World::getInstance().moleculeEnd();
iter = World::getInstance().getMoleculeIter()) {
// TODO: remove when deprecated MoleculeListClass is gone
World::getInstance().getMolecules()->erase(*iter);
World::getInstance().destroyMolecule(*iter);
}
// instantiate new molecules
molecule *newmol = NULL;
for (ConnectedSubgraphList::const_iterator iter = ListOfConnectedSubgraphs.begin();
iter != ListOfConnectedSubgraphs.end();
++iter) {
newmol = (*iter).getMolecule();
if (DoLog(2)) {
LOG(2, "STATUS: Creating new molecule:");
std::stringstream output;
newmol->Output(&output);
std::stringstream outstream(output.str());
std::string line;
while (getline(outstream, line)) {
LOG(2, "\t"+line);
}
}
}
}
MoleculeLeafClass *DepthFirstSearchAnalysis::getMoleculeStructure() const
{
MoleculeLeafClass *Subgraphs = new MoleculeLeafClass(NULL);
MoleculeLeafClass *MolecularWalker = Subgraphs;
for (World::MoleculeIterator iter = World::getInstance().getMoleculeIter();
iter != World::getInstance().moleculeEnd();
++iter) {
// TODO: Remove the insertion into molecule when saving does not depend on them anymore. Also, remove molecule.hpp include
MolecularWalker = new MoleculeLeafClass(MolecularWalker);
MolecularWalker->Leaf = (*iter);
}
return Subgraphs;
}