Files
rdkit/External/pubchem_shape/Wrap/test_rdshapealign.py
David Cosgrove 21b3c4e6f1 Shape overlay initial start aligned to principal axes (#8999)
* Add MolTransforms library.

* Transform to inertial frame of reference before overlay.

* Vital fix of typo in comment.

* Tidy up debugging cruft.

* Comment in CMakeLists.txt.

* Fix python wrappers.

* Extra tidy.

* Response to review.

* Tidy includes.

---------

Co-authored-by: David Cosgrove <david@cozchemix.co.uk>
2025-12-20 08:35:28 +01:00

149 lines
6.3 KiB
Python

import unittest
from rdkit import Chem
from rdkit.Chem import rdShapeAlign
from rdkit import RDConfig
from rdkit.Geometry import Point3D
datadir = RDConfig.RDBaseDir + '/External/pubchem_shape/test_data'
class TestCase(unittest.TestCase):
def setUp(self):
suppl = Chem.SDMolSupplier(datadir + '/test1.sdf')
self.ref = suppl[0]
self.probe = suppl[1]
def test1_Defaults(self):
tpl = rdShapeAlign.AlignMol(self.ref, self.probe, opt_param=0.5, max_preiters=3,
max_postiters=16)
self.assertAlmostEqual(tpl[0], 0.773, places=3)
self.assertAlmostEqual(tpl[1], 0.305, places=3)
def test2_NoColor(self):
tpl = rdShapeAlign.AlignMol(self.ref, self.probe, useColors=False, opt_param=0.5,
max_preiters=3, max_postiters=16)
self.assertAlmostEqual(tpl[0], 0.773, places=3)
self.assertAlmostEqual(tpl[1], 0.0, places=3)
def test3_FromShape(self):
shp = rdShapeAlign.PrepareConformer(self.ref)
self.assertTrue(type(shp) == rdShapeAlign.ShapeInput)
tpl = rdShapeAlign.AlignMol(shp, self.probe, opt_param=0.5, max_preiters=3, max_postiters=16)
self.assertAlmostEqual(tpl[0], 0.773, places=3)
self.assertAlmostEqual(tpl[1], 0.305, places=3)
def test4_ShapeInputOptions(self):
opts = rdShapeAlign.ShapeInputOptions()
opts.useColors = False
shp = rdShapeAlign.PrepareConformer(self.ref, -1, opts)
tpl = rdShapeAlign.AlignMol(shp, self.probe, opt_param=0.5, max_preiters=3, max_postiters=16)
self.assertAlmostEqual(tpl[0], 0.773, places=3)
self.assertAlmostEqual(tpl[1], 0.0, places=3)
opts.atomSubset = [4, 5, 6, 7, 8, 9]
shp = rdShapeAlign.PrepareConformer(self.ref, -1, opts)
self.assertAlmostEqual(shp.sov, 251.946, places=3)
self.assertAlmostEqual(shp.sof, 0.0, places=3)
opts.atomRadii = [(4, 1.9)]
shp = rdShapeAlign.PrepareConformer(self.ref, -1, opts)
self.assertAlmostEqual(shp.sov, 274.576, places=3)
with self.assertRaises(AttributeError):
opts.rhubarb = True
def test5_ShapeShapeOverlay(self):
refShp = rdShapeAlign.PrepareConformer(self.ref, -1)
probeShp = rdShapeAlign.PrepareConformer(self.probe, -1)
tpl = rdShapeAlign.AlignShapes(refShp, probeShp)
probeCp = Chem.Mol(self.probe)
rdShapeAlign.TransformConformer(refShp.shift, refShp.inertialRot, tpl[2], probeShp, probeCp.GetConformer(-1))
self.assertEqual(len(refShp.inertialRot), 9)
self.assertAlmostEqual(refShp.inertialRot[0], -0.926125, places=6)
self.assertAlmostEqual(refShp.inertialRot[8], -0.852890, places=6)
# Just show it did something. The full test is in the C++ layer.
self.assertNotEqual(self.probe.GetConformer().GetAtomPosition(0),
probeCp.GetConformer().GetAtomPosition(0))
matrix = tpl[2][:10]
with self.assertRaises(ValueError):
rdShapeAlign.TransformConformer(refShp.shift, refShp.inertialRot, matrix, probeShp, probeCp.GetConformer(-1))
def test6_notColorAtoms(self):
m1 = Chem.MolFromSmiles("Nc1ccccc1 |(0.392086,-2.22477,0.190651;"
"0.232269,-1.38667,0.118385;-1.06274,-0.918982,0.0342466;"
"-1.26098,0.446053,-0.0811879;-0.244035,1.36265,-0.11691;"
"1.05134,0.875929,-0.031248;1.28797,-0.499563,0.0864097)"
",atomProp:0.dummyLabel.*|")
opts = rdShapeAlign.ShapeInputOptions()
opts.notColorAtoms = [0]
shp = rdShapeAlign.PrepareConformer(m1, -1, opts)
self.assertAlmostEqual(shp.sof, 5.074, places=3)
def test7_customFeatures(self):
m1 = Chem.MolFromSmiles(
"O=CC=O |(-1.75978,0.148897,0;-0.621382,-0.394324,0;0.624061,0.3656,.1;1.7571,-0.120174,.1)|")
opts = rdShapeAlign.ShapeInputOptions()
opts.customFeatures = ((1, Point3D(-1.75978, 0.148897,
0), 1.0), (2, Point3D(1.7571, -0.120174, 0.1), 1.0))
shp = rdShapeAlign.PrepareConformer(m1, -1, opts)
self.assertEqual(len(shp.coord), 18)
m2 = Chem.Mol(m1)
opts2 = rdShapeAlign.ShapeInputOptions()
opts2.customFeatures = ((2, Point3D(-1.75978, 0.148897,
0), 1.0), (1, Point3D(1.7571, -0.120174, 0.1), 1.0))
shp2 = rdShapeAlign.PrepareConformer(m2, -1, opts2)
tpl = rdShapeAlign.AlignShapes(shp, shp2, opt_param=0.5)
self.assertAlmostEqual(tpl[0], 1.000, places=3)
self.assertAlmostEqual(tpl[1], 0.997, places=3)
tf = tpl[2]
self.assertGreater(0.0, tf[0])
self.assertLess(0.0, tf[3 * 3])
# check the getter:
cfs = opts2.customFeatures
self.assertEqual(len(cfs), 2)
self.assertEqual(cfs[0][0], 2)
self.assertEqual(cfs[1][0], 1)
def test8_customFeatures(self):
m1 = Chem.MolFromSmiles(
"O=CC=O |(-1.75978,0.148897,0;-0.621382,-0.394324,0;0.624061,0.3656,.1;1.7571,-0.120174,.1)|")
opts = rdShapeAlign.ShapeInputOptions()
opts.customFeatures = ((1, Point3D(-1.75978, 0.148897,
0), 1.0), (2, Point3D(1.7571, -0.120174, 0.1), 1.0))
m2 = Chem.Mol(m1)
opts2 = rdShapeAlign.ShapeInputOptions()
opts2.customFeatures = ((2, Point3D(-1.75978, 0.148897,
0), 1.0), (1, Point3D(1.7571, -0.120174, 0.1), 1.0))
tpl = rdShapeAlign.AlignMol(m1, m2, opts, opts2, opt_param=0.5)
self.assertAlmostEqual(tpl[0], 1.000, places=3)
self.assertAlmostEqual(tpl[1], 0.997, places=3)
def test9_FixedScore(self):
# Just to make sure it's there and returns a value.
opts = rdShapeAlign.ShapeInputOptions()
tpl = rdShapeAlign.ScoreMol(self.ref, self.ref, opts, opts)
self.assertAlmostEqual(tpl[0], 1.0, places=3)
self.assertAlmostEqual(tpl[1], 1.0, places=3)
opts = rdShapeAlign.ShapeInputOptions()
opts.useColors = False
opts.normalize = False
shp = rdShapeAlign.PrepareConformer(self.ref, -1, opts)
tpl = rdShapeAlign.ScoreMol(shp, self.probe, opts)
self.assertAlmostEqual(tpl[0], 0.0, places=3)
self.assertAlmostEqual(tpl[1], 0.0, places=3)
opts.useColors = True
shp1 = rdShapeAlign.PrepareConformer(self.probe, -1, opts)
shp2 = rdShapeAlign.PrepareConformer(self.probe, -1, opts)
tpl = rdShapeAlign.ScoreShape(shp1, shp2, True)
self.assertAlmostEqual(tpl[0], 1.0, places=3)
self.assertAlmostEqual(tpl[1], 1.0, places=3)
if __name__ == '__main__':
unittest.main()