/**********************************************************************
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);
}