mirror of
https://github.com/rdkit/rdkit.git
synced 2026-06-05 22:04:27 +08:00
426 lines
12 KiB
C++
426 lines
12 KiB
C++
//
|
|
// Copyright (C) 2017 Greg Landrum
|
|
//
|
|
// @@ All Rights Reserved @@
|
|
// This file is part of the RDKit.
|
|
// The contents are covered by the terms of the BSD license
|
|
// which is included in the file license.txt, found at the root
|
|
// of the RDKit source tree.
|
|
//
|
|
#include <iostream>
|
|
|
|
#include <fstream>
|
|
#include <GraphMol/RDKitBase.h>
|
|
#include <GraphMol/SmilesParse/SmilesParse.h>
|
|
#include <GraphMol/FileParsers/FileParsers.h>
|
|
#include <GraphMol/Substruct/SubstructMatch.h>
|
|
#include <GraphMol/MolAlign/AlignMolecules.h>
|
|
|
|
#include <RDGeneral/RDLog.h>
|
|
|
|
#include "coordgen/sketcherMinimizer.h"
|
|
#include <CoordGen/CoordGen.h>
|
|
|
|
using namespace RDKit;
|
|
|
|
void test1() {
|
|
BOOST_LOG(rdInfoLog) << "-------------------------------------" << std::endl;
|
|
BOOST_LOG(rdInfoLog) << "test1: basics" << std::endl;
|
|
#if 1
|
|
{
|
|
ROMol* m = SmilesToMol("c1cc(CC)cnc1CC(=O)O");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "test1");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
delete m;
|
|
}
|
|
{
|
|
// ROMol* m = SmilesToMol("c1ccncc1");
|
|
|
|
ROMol* m = SmilesToMol("ClC(O)(F)C");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "test2");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
delete m;
|
|
}
|
|
|
|
{
|
|
ROMol* m = SmilesToMol(
|
|
"CC[C@H]1C(=O)N(CC(=O)N([C@H](C(=O)N[C@H](C(=O)N([C@H](C(=O)N[C@H](C(="
|
|
"O)N[C@@H](C(=O)N([C@H](C(=O)N([C@H](C(=O)N([C@H](C(=O)N([C@H](C(=O)N1)"
|
|
"[C@@H]([C@H](C)C/C=C/"
|
|
"C)O)C)C(C)C)C)CC(C)C)C)CC(C)C)C)C)C)CC(C)C)C)C(C)C)CC(C)C)C)C");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "cyclosporine a");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
delete m;
|
|
}
|
|
|
|
{
|
|
// ROMol* m = SmilesToMol("c1ccncc1");
|
|
|
|
ROMol* m = SmilesToMol("CCCNC=CNCOC=CC=CC=COC");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "single-double");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
delete m;
|
|
}
|
|
#endif
|
|
|
|
{
|
|
ROMol* m = SmilesToMol("O/C=C/C=C/C=C\\C=C/N");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "cis-trans");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
delete m;
|
|
}
|
|
|
|
{
|
|
ROMol* m = SmilesToMol("C1C3CC2CC(CC1C2)C3");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "admntn");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
delete m;
|
|
}
|
|
|
|
BOOST_LOG(rdInfoLog) << "done" << std::endl;
|
|
}
|
|
|
|
namespace {
|
|
bool compareConfs(const ROMol* m, ROMol* templ, const MatchVectType& mv,
|
|
bool alignFirst = false, int molConfId = -1,
|
|
int templateConfId = -1, double postol = 1e-2,
|
|
double rmstol = 0.1) {
|
|
PRECONDITION(m, "bad pointer");
|
|
PRECONDITION(templ, "bad pointer");
|
|
TEST_ASSERT(m->getNumAtoms() >= templ->getNumAtoms());
|
|
|
|
if (alignFirst) {
|
|
double rmsd =
|
|
MolAlign::alignMol(*templ, *m, molConfId, templateConfId, &mv);
|
|
if (rmsd > rmstol) return false;
|
|
}
|
|
|
|
const Conformer& conf1 = m->getConformer(molConfId);
|
|
const Conformer& conf2 = templ->getConformer(templateConfId);
|
|
for (unsigned int i = 0; i < templ->getNumAtoms(); i++) {
|
|
TEST_ASSERT(m->getAtomWithIdx(mv[i].second)->getAtomicNum() ==
|
|
templ->getAtomWithIdx(mv[i].first)->getAtomicNum());
|
|
|
|
RDGeom::Point3D pt1i = conf1.getAtomPos(mv[i].second);
|
|
RDGeom::Point3D pt2i = conf2.getAtomPos(mv[i].first);
|
|
if ((pt1i - pt2i).length() >= postol) return false;
|
|
}
|
|
return true;
|
|
}
|
|
} // namespace
|
|
|
|
void test2() {
|
|
BOOST_LOG(rdInfoLog) << "-------------------------------------" << std::endl;
|
|
BOOST_LOG(rdInfoLog) << "test2: using templates" << std::endl;
|
|
|
|
{
|
|
ROMol* core = SmilesToMol("C1CON1");
|
|
TEST_ASSERT(core);
|
|
core->setProp("_Name", "core");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*core) == 0);
|
|
TEST_ASSERT(core->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*core);
|
|
std::cerr << mb << std::endl;
|
|
|
|
ROMol* m = SmilesToMol("C1C(CCC)ON1");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "core+sidechain");
|
|
|
|
MatchVectType mv;
|
|
SubstructMatch(*m, *core, mv);
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
TEST_ASSERT(!compareConfs(m, core, mv));
|
|
|
|
{
|
|
auto coreConf = core->getConformer();
|
|
RDGeom::INT_POINT2D_MAP coordMap;
|
|
for (unsigned int i = 0; i < mv.size(); ++i) {
|
|
coordMap[mv[i].second] =
|
|
RDGeom::Point2D(coreConf.getAtomPos(mv[i].first).x,
|
|
coreConf.getAtomPos(mv[i].first).y);
|
|
}
|
|
CoordGen::CoordGenParams params;
|
|
params.coordMap = coordMap;
|
|
params.dbg_useFixed = true;
|
|
TEST_ASSERT(CoordGen::addCoords(*m, ¶ms) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
// m->setProp("_Name", "templated");
|
|
// mb = MolToMolBlock(*m);
|
|
// std::cerr << mb << std::endl;
|
|
TEST_ASSERT(compareConfs(m, core, mv));
|
|
}
|
|
{
|
|
CoordGen::CoordGenParams params;
|
|
params.templateMol = core;
|
|
params.dbg_useFixed = true;
|
|
TEST_ASSERT(CoordGen::addCoords(*m, ¶ms) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
m->setProp("_Name", "templated");
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
TEST_ASSERT(compareConfs(m, core, mv));
|
|
}
|
|
delete m;
|
|
delete core;
|
|
}
|
|
|
|
{
|
|
ROMol* core = SmilesToMol("C1CCCCCONCN1");
|
|
TEST_ASSERT(core);
|
|
core->setProp("_Name", "core");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*core) == 0);
|
|
TEST_ASSERT(core->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*core);
|
|
std::cerr << mb << std::endl;
|
|
|
|
ROMol* m = SmilesToMol("C1CCCCONC(CC)NC1");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "core+sidechain");
|
|
|
|
MatchVectType mv;
|
|
SubstructMatch(*m, *core, mv);
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
TEST_ASSERT(!compareConfs(m, core, mv));
|
|
|
|
{
|
|
auto coreConf = core->getConformer();
|
|
RDGeom::INT_POINT2D_MAP coordMap;
|
|
for (unsigned int i = 0; i < mv.size(); ++i) {
|
|
coordMap[mv[i].second] =
|
|
RDGeom::Point2D(coreConf.getAtomPos(mv[i].first).x,
|
|
coreConf.getAtomPos(mv[i].first).y);
|
|
}
|
|
CoordGen::CoordGenParams params;
|
|
params.coordMap = coordMap;
|
|
params.dbg_useFixed = true;
|
|
TEST_ASSERT(CoordGen::addCoords(*m, ¶ms) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
// m->setProp("_Name", "templated");
|
|
// mb = MolToMolBlock(*m);
|
|
// std::cerr << mb << std::endl;
|
|
TEST_ASSERT(compareConfs(m, core, mv));
|
|
}
|
|
{
|
|
CoordGen::CoordGenParams params;
|
|
params.templateMol = core;
|
|
params.dbg_useFixed = true;
|
|
TEST_ASSERT(CoordGen::addCoords(*m, ¶ms) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
m->setProp("_Name", "templated");
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
TEST_ASSERT(compareConfs(m, core, mv));
|
|
}
|
|
|
|
delete m;
|
|
delete core;
|
|
}
|
|
|
|
{
|
|
ROMol* core = SmilesToMol("C1CCCCCONCN1");
|
|
TEST_ASSERT(core);
|
|
core->setProp("_Name", "core");
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*core) == 0);
|
|
TEST_ASSERT(core->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*core);
|
|
std::cerr << mb << std::endl;
|
|
|
|
ROMol* m = SmilesToMol("C1CCCCONC(CCCCCC)NC1");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "core+sidechain");
|
|
|
|
MatchVectType mv;
|
|
SubstructMatch(*m, *core, mv);
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
TEST_ASSERT(!compareConfs(m, core, mv));
|
|
|
|
{
|
|
auto coreConf = core->getConformer();
|
|
RDGeom::INT_POINT2D_MAP coordMap;
|
|
for (unsigned int i = 0; i < mv.size(); ++i) {
|
|
coordMap[mv[i].second] =
|
|
RDGeom::Point2D(coreConf.getAtomPos(mv[i].first).x,
|
|
coreConf.getAtomPos(mv[i].first).y);
|
|
}
|
|
|
|
CoordGen::CoordGenParams params;
|
|
params.coordMap = coordMap;
|
|
params.dbg_useFixed = true;
|
|
TEST_ASSERT(CoordGen::addCoords(*m, ¶ms) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
// m->setProp("_Name", "templated");
|
|
// mb = MolToMolBlock(*m);
|
|
// std::cerr << mb << std::endl;
|
|
TEST_ASSERT(compareConfs(m, core, mv));
|
|
}
|
|
{
|
|
CoordGen::CoordGenParams params;
|
|
params.templateMol = core;
|
|
params.dbg_useFixed = true;
|
|
TEST_ASSERT(CoordGen::addCoords(*m, ¶ms) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
m->setProp("_Name", "templated");
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
TEST_ASSERT(compareConfs(m, core, mv));
|
|
}
|
|
delete m;
|
|
delete core;
|
|
}
|
|
|
|
{
|
|
ROMol* core = SmilesToMol("C1CCCC2C1NCC2");
|
|
TEST_ASSERT(core);
|
|
core->setProp("_Name", "core");
|
|
|
|
CoordGen::addCoords(*core);
|
|
TEST_ASSERT(core->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*core);
|
|
std::cerr << mb << std::endl;
|
|
|
|
ROMol* m = SmilesToMol("C1C(CCC)CC(CC3CC3)C2C1N(C(C)C)CC2");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "core+sidechain");
|
|
|
|
MatchVectType mv;
|
|
SubstructMatch(*m, *core, mv);
|
|
|
|
CoordGen::addCoords(*m);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
// This is a rigid core: if we provide the matching substructure,
|
|
// and do alignment, the conformations will still match.
|
|
TEST_ASSERT(!compareConfs(m, core, mv, false));
|
|
|
|
{
|
|
CoordGen::CoordGenParams params;
|
|
params.templateMol = core;
|
|
CoordGen::addCoords(*m, ¶ms);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
m->setProp("_Name", "templated");
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
TEST_ASSERT(compareConfs(m, core, mv, true, -1, -1, 0.3));
|
|
}
|
|
delete m;
|
|
delete core;
|
|
}
|
|
|
|
{
|
|
ROMol* core = SmilesToMol("CC(N)CC");
|
|
TEST_ASSERT(core);
|
|
core->setProp("_Name", "core");
|
|
|
|
CoordGen::addCoords(*core);
|
|
TEST_ASSERT(core->getNumConformers() == 1);
|
|
auto mb = MolToMolBlock(*core);
|
|
std::cerr << mb << std::endl;
|
|
|
|
ROMol* m = SmilesToMol("CC(N)CC(O)C");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "core+sidechain");
|
|
|
|
MatchVectType mv;
|
|
SubstructMatch(*m, *core, mv);
|
|
|
|
CoordGen::addCoords(*m);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
// This mol is just slightly bigger than the core, providing
|
|
// the matching substructure and doing alignment will cause
|
|
// the conformations to match.
|
|
TEST_ASSERT(!compareConfs(m, core, mv, false));
|
|
|
|
{
|
|
CoordGen::CoordGenParams params;
|
|
params.templateMol = core;
|
|
CoordGen::addCoords(*m, ¶ms);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
m->setProp("_Name", "templated");
|
|
mb = MolToMolBlock(*m);
|
|
std::cerr << mb << std::endl;
|
|
TEST_ASSERT(compareConfs(m, core, mv, true, -1, -1, 0.05));
|
|
}
|
|
delete m;
|
|
delete core;
|
|
}
|
|
|
|
BOOST_LOG(rdInfoLog) << "done" << std::endl;
|
|
}
|
|
|
|
void testGithub1929() {
|
|
BOOST_LOG(rdInfoLog) << "-------------------------------------" << std::endl;
|
|
BOOST_LOG(rdInfoLog)
|
|
<< "testing github1929: make sure coordgen works with bogus file names"
|
|
<< std::endl;
|
|
{
|
|
ROMol* m = SmilesToMol("c1cc(CC)cnc1CC(=O)O");
|
|
TEST_ASSERT(m);
|
|
m->setProp("_Name", "test1");
|
|
CoordGen::CoordGenParams params;
|
|
params.templateFileDir = "I_do_not_exist";
|
|
|
|
TEST_ASSERT(CoordGen::addCoords(*m, ¶ms) == 0);
|
|
TEST_ASSERT(m->getNumConformers() == 1);
|
|
delete m;
|
|
}
|
|
|
|
BOOST_LOG(rdInfoLog) << "done" << std::endl;
|
|
}
|
|
|
|
int main(int argc, char* argv[]) {
|
|
(void)argc;
|
|
(void)argv;
|
|
RDLog::InitLogs();
|
|
test2();
|
|
test1();
|
|
testGithub1929();
|
|
}
|