/********************************************************************** cifspacegrouptest.cpp - Unit tests for to check if space group is being handled properly in .cif format. Copyright (C) 2016 by Schrodinger Inc. This file is part of the Open Babel project. For more information, see This program 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 version 2 of the License. This program 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. ***********************************************************************/ #include "obtest.h" #include #include #include #include #include #include using namespace std; using namespace OpenBabel; std::string static GetFilename(const std::string &filename) { string path = TESTDATADIR + filename; return path; } void testSpaceGroupUniqueTransformations() { // See https://github.com/openbabel/openbabel/pull/260 // also https://github.com/openbabel/openbabel/pull/255 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.ReadFile(&mol, GetFilename("test01.cif")); OBUnitCell* pUC = (OBUnitCell*)mol.GetData(OBGenericDataType::UnitCell); const SpaceGroup* pSG = pUC->GetSpaceGroup(); SpaceGroup* sg = new SpaceGroup(*pSG); pSG = SpaceGroup::Find(sg); OB_ASSERT(pSG != nullptr); // Check also for errors and warnings string summary = obErrorLog.GetMessageSummary(); // OB_ASSERT( summary.find("error") == string::npos); // OB_ASSERT( summary.find("warning") == string::npos); OB_ASSERT( pSG->GetId() == 64 ); } void testSpaceGroupClean() { // See https://github.com/openbabel/openbabel/pull/254 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.SetOutFormat("pdb"); conv.ReadFile(&mol, GetFilename("test02.cif")); OBUnitCell* pUC = (OBUnitCell*)mol.GetData(OBGenericDataType::UnitCell); const SpaceGroup* pSG = pUC->GetSpaceGroup(); SpaceGroup* sg = new SpaceGroup(*pSG); pSG = SpaceGroup::Find(sg); OB_ASSERT(pSG != nullptr); // Check also for errors and warnings string summary = obErrorLog.GetMessageSummary(); // OB_ASSERT( summary.find("error") == string::npos); // OB_ASSERT( summary.find("warning") == string::npos); OB_ASSERT( pSG->GetId() == 166 ); string pdb = conv.WriteString(&mol); pdb = conv.WriteString(&mol); OB_ASSERT(pdb.find("H -3 m") != string::npos); } void testSpaceGroupTransformations() { // See https://github.com/openbabel/openbabel/pull/254 SpaceGroup group; vector trans; vector trans_exp; vector trans_got; // Same transformation trans_exp.push_back("x,y,z"); trans.push_back("x,y,z"); trans.push_back(" x , y , z "); trans.push_back("x,y+1,z"); trans.push_back("x,y-1,z-1"); trans.push_back("x,y-1,z"); // Same transformation trans_exp.push_back("x,-y,z"); trans.push_back("x,-y,z"); trans.push_back("x,1-y,z"); trans.push_back("x,-y+1,z"); // Same transformation trans_exp.push_back("x,-y,-z"); trans.push_back("x,-y,-z"); trans.push_back("x,1-y,1-z"); trans.push_back("x,-y+1,-z+1"); // Same transformation trans_exp.push_back("x,-y,-y-z"); trans.push_back("x,-y,-z-y"); trans.push_back("x,1-y,-y+1-z"); trans.push_back("x,-y+1,-y-z+1"); // Same transformation trans_exp.push_back("x,-y,1/6-y-z"); trans.push_back("x,-y,-z-y+1/6"); trans.push_back("x,-y,-z-y+7/6"); trans.push_back("x,1-y,7/6-y+1-z"); trans.push_back("x,-y+1,-y-z+1+1/6"); trans.push_back("x,-y+1,-y+1/6-z+1"); // Same transformation trans_exp.push_back("x,3/4-y+z,5/6-y-z"); trans.push_back("x,3/4-y+z,-1/6-z-y"); trans.push_back("x,-y+3/4+z,-z-y-1/6"); trans.push_back("x,z-y+3/4,-z-y-7/6"); trans.push_back("x,1+z+3/4-y,-7/6-y-z"); trans.push_back("X , 3 / 4 - Y + 1 + z , - Y - Z - 1 / 6 "); trans.push_back("x,z-y+1+3/4,-y-1/6-z+1"); vector::const_iterator i, iend; iend = trans.end(); for (i = trans.begin(); i != iend; ++i) group.AddTransform(i->c_str()); // Loop over symmetry operators transform3dIterator ti; const transform3d *t = group.BeginTransform(ti); while(t){ trans_got.push_back(t->DescribeAsString()); //cout << t->DescribeAsString() << "\n"; t = group.NextTransform(ti); } OB_ASSERT( trans_exp.size() == trans_got.size() ); OB_ASSERT( equal(trans_exp.begin(), trans_exp.end(), trans_got.begin()) ); } void testDecayToP1() { // See https://github.com/openbabel/openbabel/pull/261 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.ReadFile(&mol, GetFilename("test03.cif")); OBUnitCell* pUC = (OBUnitCell*)mol.GetData(OBGenericDataType::UnitCell); const SpaceGroup* pSG = pUC->GetSpaceGroup(); SpaceGroup* sg = new SpaceGroup(*pSG); pSG = SpaceGroup::Find(sg); OB_ASSERT(pSG != nullptr); // Check also for errors and warnings string summary = obErrorLog.GetMessageSummary(); OB_ASSERT( summary.find("2 warnings") != string::npos); OB_ASSERT( pSG->GetId() == 1 ); } void testAlternativeOrigin() { // See https://github.com/openbabel/openbabel/pull/1558 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.ReadFile(&mol, GetFilename("test04.cif")); OBUnitCell* pUC = (OBUnitCell*)mol.GetData(OBGenericDataType::UnitCell); const SpaceGroup* pSG = pUC->GetSpaceGroup(); SpaceGroup* sg = new SpaceGroup(*pSG); pSG = SpaceGroup::Find(sg); string summary = obErrorLog.GetMessageSummary(); OB_ASSERT( summary.find("warning") == string::npos); OB_ASSERT( pSG != nullptr ); OB_ASSERT( pSG->GetOriginAlternative() == 1); } void testPdbOutAlternativeOrigin() { // See https://github.com/openbabel/openbabel/pull/1558 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.SetOutFormat("pdb"); conv.ReadFile(&mol, GetFilename("test04.cif")); string pdb = conv.WriteString(&mol); // ending space is needed to check that there is no origin set OB_ASSERT(pdb.find("P 4/n b m ") != string::npos); conv.AddOption("o", OBConversion::OUTOPTIONS); pdb = conv.WriteString(&mol); OB_ASSERT(pdb.find("P 4/n b m:1") != string::npos); } void testPdbOutHexagonalAlternativeOrigin() { // See https://github.com/openbabel/openbabel/pull/1558 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.SetOutFormat("pdb"); conv.ReadFile(&mol, GetFilename("test02.cif")); string pdb = conv.WriteString(&mol); conv.AddOption("o", OBConversion::OUTOPTIONS); pdb = conv.WriteString(&mol); OB_ASSERT(pdb.find("H -3 m") != string::npos); // Test with missing Hall name in the CIF // https://github.com/openbabel/openbabel/pull/1578 OBMol mol_nohall; conv.ReadFile(&mol_nohall, GetFilename("test02.nohall.cif")); pdb = conv.WriteString(&mol_nohall); OB_ASSERT(pdb.find("H -3 m") != string::npos); } void testPdbOutAlternativeOriginSilicon() { // See https://github.com/openbabel/openbabel/pull/1558 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.SetOutFormat("pdb"); conv.ReadFile(&mol, GetFilename("test05.cif")); string pdb = conv.WriteString(&mol); conv.AddOption("o", OBConversion::OUTOPTIONS); pdb = conv.WriteString(&mol); OB_ASSERT(pdb.find("F d 3 m:1") != string::npos); } void testPdbOutHexagonalAlternativeOrigin2() { // See https://github.com/openbabel/openbabel/pull/1558 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.SetOutFormat("pdb"); conv.ReadFile(&mol, GetFilename("test06.cif")); string pdb = conv.WriteString(&mol); conv.AddOption("o", OBConversion::OUTOPTIONS); pdb = conv.WriteString(&mol); OB_ASSERT(pdb.find("H -3 m") != string::npos); } void testPdbRemSpacesHMName() { // See https://github.com/openbabel/openbabel/pull/1558 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.SetOutFormat("pdb"); conv.ReadFile(&mol, GetFilename("test07.cif")); string pdb = conv.WriteString(&mol); conv.AddOption("o", OBConversion::OUTOPTIONS); pdb = conv.WriteString(&mol); OB_ASSERT(pdb.find("I41/amd:2") != string::npos); } void testPdbOccupancies() { // See https://github.com/openbabel/openbabel/pull/1558 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.SetOutFormat("pdb"); conv.ReadFile(&mol, GetFilename("test08.cif")); string pdb = conv.WriteString(&mol); conv.AddOption("o", OBConversion::OUTOPTIONS); pdb = conv.WriteString(&mol); OB_ASSERT(pdb.find("HETATM 1 NA UNL 1 0.325 0.000 4.425 0.36") != string::npos); OB_ASSERT(pdb.find("HETATM 17 O UNL 8 1.954 8.956 3.035 1.00") != string::npos); OBMol mol_pdb; conv.SetInFormat("pdb"); conv.ReadFile(&mol_pdb, GetFilename("test09.pdb")); pdb = conv.WriteString(&mol_pdb); OB_ASSERT(pdb.find("HETATM 1 NA UNL 1 0.325 0.000 4.425 0.36") != string::npos); OB_ASSERT(pdb.find("HETATM 2 NA UNL 1 0.002 8.956 1.393 0.10") != string::npos); OB_ASSERT(pdb.find("HETATM 17 O UNL 8 1.954 8.956 3.035 1.00") != string::npos); } void testCIFMolecules() { // See https://github.com/openbabel/openbabel/pull/1558 OBConversion conv; OBMol mol; conv.SetInFormat("cif"); conv.SetOutFormat("smi"); // check for disconnected fragments conv.ReadFile(&mol, GetFilename("1519159.cif")); string smi = conv.WriteString(&mol); // never, never disconnected fragments from a molecule OB_ASSERT(smi.find(".") == string::npos); } void testCIFOutputFormat() { // See https://github.com/openbabel/openbabel/pull/2170 OBConversion conv; OBMol mol; conv.SetInFormat("sdf"); conv.SetOutFormat("cif"); // check correct format conv.ReadFile(&mol, GetFilename("kevlar.sdf")); string cif = conv.WriteString(&mol); string ref = " H0 H -71.99400 -128.76240 56.30360 1.000"; OB_ASSERT(cif.find(ref) != string::npos); } int cifspacegrouptest(int argc, char* argv[]) { int defaultchoice = 1; int choice = defaultchoice; if (argc > 1) { if(sscanf(argv[1], "%d", &choice) != 1) { printf("Couldn't parse that input as a number\n"); return -1; } } // Define location of file formats for testing #ifdef FORMATDIR char env[BUFF_SIZE]; snprintf(env, BUFF_SIZE, "BABEL_LIBDIR=%s", FORMATDIR); putenv(env); #endif switch(choice) { case 1: testSpaceGroupUniqueTransformations(); break; case 2: testSpaceGroupClean(); break; case 3: testSpaceGroupTransformations(); break; case 4: testDecayToP1(); break; case 5: testAlternativeOrigin(); break; case 6: testPdbOutAlternativeOrigin(); break; case 7: testPdbOutHexagonalAlternativeOrigin(); break; case 8: testPdbOutAlternativeOriginSilicon(); break; case 9: testPdbOutHexagonalAlternativeOrigin2(); break; case 10: testPdbRemSpacesHMName(); break; case 11: testPdbOccupancies(); break; case 12: testCIFMolecules(); break; case 13: testCIFOutputFormat(); break; default: cout << "Test number " << choice << " does not exist!\n"; return -1; } return(0); }