Files
rdkit/Code/GraphMol/ChemReactions/ReactionPickler.cpp
Greg Landrum 86f197caac Fixes #7674 (#7681)
* Fixes $7674

* typo
2024-08-01 17:19:58 +02:00

313 lines
9.7 KiB
C++

//
// Copyright (C) 2009-2018 Greg Landrum
// Copyright (c) 2014, Novartis Institutes for BioMedical Research Inc.
//
// @@ 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 <sstream>
#include <GraphMol/RDKitBase.h>
#include <GraphMol/MolPickler.h>
#include <GraphMol/ChemReactions/Reaction.h>
#include <GraphMol/ChemReactions/ReactionPickler.h>
#include <RDGeneral/utils.h>
#include <RDGeneral/RDLog.h>
#include <RDGeneral/StreamOps.h>
#include <RDGeneral/types.h>
#include <cstdint>
using std::int32_t;
using std::uint32_t;
namespace RDKit {
const int32_t ReactionPickler::versionMajor = 3;
const int32_t ReactionPickler::versionMinor = 0;
const int32_t ReactionPickler::versionPatch = 0;
const int32_t ReactionPickler::endianId = 0xDEADBEEF;
void streamWrite(std::ostream &ss, ReactionPickler::Tags tag) {
int32_t tmp = tag;
streamWrite(ss, tmp);
}
template <typename T>
void streamWrite(std::ostream &ss, ReactionPickler::Tags tag, const T &what) {
streamWrite(ss, tag);
streamWrite(ss, what);
};
void streamRead(std::istream &ss, ReactionPickler::Tags &tag) {
int32_t tmp;
streamRead(ss, tmp);
tag = static_cast<ReactionPickler::Tags>(tmp);
}
void ReactionPickler::pickleReaction(const ChemicalReaction *rxn,
std::ostream &ss) {
pickleReaction(rxn, ss, MolPickler::getDefaultPickleProperties());
}
void ReactionPickler::pickleReaction(const ChemicalReaction *rxn,
std::ostream &ss,
unsigned int propertyFlags) {
PRECONDITION(rxn, "empty reaction");
streamWrite(ss, endianId);
streamWrite(ss, VERSION);
streamWrite(ss, versionMajor);
streamWrite(ss, versionMinor);
streamWrite(ss, versionPatch);
_pickle(rxn, ss, propertyFlags);
}
void ReactionPickler::pickleReaction(const ChemicalReaction *rxn,
std::string &res) {
pickleReaction(rxn, res, MolPickler::getDefaultPickleProperties());
}
void ReactionPickler::pickleReaction(const ChemicalReaction *rxn,
std::string &res,
unsigned int propertyFlags) {
PRECONDITION(rxn, "empty reaction");
std::stringstream ss(std::ios_base::binary | std::ios_base::out |
std::ios_base::in);
ReactionPickler::pickleReaction(rxn, ss, propertyFlags);
res = ss.str();
}
// NOTE: if the reaction passed in here already has reactants and products
// be left intact.
void ReactionPickler::reactionFromPickle(std::istream &ss,
ChemicalReaction *rxn) {
PRECONDITION(rxn, "empty reaction");
Tags tag;
int32_t tmpInt;
streamRead(ss, tmpInt);
if (tmpInt != endianId) {
throw ReactionPicklerException(
"Bad pickle format: bad endian ID or invalid file format");
}
streamRead(ss, tag);
if (tag != VERSION) {
throw ReactionPicklerException("Bad pickle format: no version tag");
}
int32_t majorVersion, minorVersion, patchVersion;
streamRead(ss, majorVersion);
streamRead(ss, minorVersion);
streamRead(ss, patchVersion);
if (majorVersion > versionMajor ||
(majorVersion == versionMajor && minorVersion > versionMinor)) {
BOOST_LOG(rdWarningLog)
<< "Depickling from a version number (" << majorVersion << "."
<< minorVersion << ")"
<< "that is higher than our version (" << versionMajor << "."
<< versionMinor << ").\nThis probably won't work." << std::endl;
}
majorVersion = 1000 * majorVersion + minorVersion * 10 + patchVersion;
_depickle(ss, rxn, majorVersion);
}
void ReactionPickler::reactionFromPickle(const std::string &pickle,
ChemicalReaction *rxn) {
PRECONDITION(rxn, "empty reaction");
std::stringstream ss(std::ios_base::binary | std::ios_base::out |
std::ios_base::in);
ss.write(pickle.c_str(), pickle.length());
ReactionPickler::reactionFromPickle(ss, rxn);
}
void ReactionPickler::_pickle(const ChemicalReaction *rxn, std::ostream &ss,
unsigned int propertyFlags) {
PRECONDITION(rxn, "empty reaction");
uint32_t tmpInt;
tmpInt = static_cast<int32_t>(rxn->getNumReactantTemplates());
streamWrite(ss, tmpInt);
tmpInt = static_cast<int32_t>(rxn->getNumProductTemplates());
streamWrite(ss, tmpInt);
tmpInt = static_cast<int32_t>(rxn->getNumAgentTemplates());
streamWrite(ss, tmpInt);
uint32_t flag = 0;
if (rxn->getImplicitPropertiesFlag()) {
flag |= 0x1;
}
if (rxn->df_needsInit) {
flag |= 0x2;
}
streamWrite(ss, flag);
// -------------------
//
// Write Reactants
//
// -------------------
streamWrite(ss, BEGINREACTANTS);
for (auto tmpl = rxn->beginReactantTemplates();
tmpl != rxn->endReactantTemplates(); ++tmpl) {
auto props = rxn->df_needsInit
? propertyFlags
: static_cast<unsigned int>(
PicklerOps::PropertyPickleOptions::AllProps);
MolPickler::pickleMol(tmpl->get(), ss, props);
}
streamWrite(ss, ENDREACTANTS);
streamWrite(ss, BEGINPRODUCTS);
for (auto tmpl = rxn->beginProductTemplates();
tmpl != rxn->endProductTemplates(); ++tmpl) {
auto props = rxn->df_needsInit
? propertyFlags
: static_cast<unsigned int>(
PicklerOps::PropertyPickleOptions::AllProps);
MolPickler::pickleMol(tmpl->get(), ss, props);
}
streamWrite(ss, ENDPRODUCTS);
if (rxn->getNumAgentTemplates()) {
streamWrite(ss, BEGINAGENTS);
for (auto tmpl = rxn->beginAgentTemplates();
tmpl != rxn->endAgentTemplates(); ++tmpl) {
// reagents don't have private properties set during initialization
MolPickler::pickleMol(
tmpl->get(), ss,
propertyFlags | PicklerOps::PropertyPickleOptions::AtomProps);
}
streamWrite(ss, ENDAGENTS);
}
// -------------------
//
// Write SSS parameters
//
// -------------------
streamWrite(ss, BEGINSSSPARAMS);
auto json = substructMatchParamsToJSON(rxn->getSubstructParams());
streamWrite(ss, json);
streamWrite(ss, ENDSSSPARAMS);
if (propertyFlags & PicklerOps::MolProps) {
streamWrite(ss, BEGINPROPS);
_pickleProperties(ss, *rxn, propertyFlags);
streamWrite(ss, ENDPROPS);
}
streamWrite(ss, ENDREACTION);
} // end of _pickle
void ReactionPickler::_pickleProperties(std::ostream &ss, const RDProps &props,
unsigned int pickleFlags) {
if (!pickleFlags) {
return;
}
streamWriteProps(ss, props, pickleFlags & PicklerOps::PrivateProps,
pickleFlags & PicklerOps::ComputedProps);
}
//! unpickle standard properties
void ReactionPickler::_unpickleProperties(std::istream &ss, RDProps &props) {
streamReadProps(ss, props);
}
void ReactionPickler::_depickle(std::istream &ss, ChemicalReaction *rxn,
int version) {
PRECONDITION(rxn, "empty reaction");
Tags tag;
uint32_t numReactants, numProducts, numAgents = 0;
streamRead(ss, numReactants);
streamRead(ss, numProducts);
if (version > 1000) {
streamRead(ss, numAgents);
}
// we use this here and below to set df_needsInit, so don't re-use the
// variable
uint32_t flag = 0;
streamRead(ss, flag);
rxn->setImplicitPropertiesFlag(flag & 0x1);
// -------------------
//
// Read Reactants
//
// -------------------
streamRead(ss, tag);
if (tag != BEGINREACTANTS) {
throw ReactionPicklerException(
"Bad pickle format: BEGINREACTANTS tag not found.");
}
for (unsigned int i = 0; i < numReactants; ++i) {
auto *mol = new RWMol();
MolPickler::molFromPickle(ss, mol);
rxn->addReactantTemplate(ROMOL_SPTR(mol));
}
streamRead(ss, tag);
if (tag != ENDREACTANTS) {
throw ReactionPicklerException(
"Bad pickle format: ENDREACTANTS tag not found.");
}
streamRead(ss, tag);
if (tag != BEGINPRODUCTS) {
throw ReactionPicklerException(
"Bad pickle format: BEGINPRODUCTS tag not found.");
}
for (unsigned int i = 0; i < numProducts; ++i) {
auto *mol = new RWMol();
MolPickler::molFromPickle(ss, mol);
rxn->addProductTemplate(ROMOL_SPTR(mol));
}
streamRead(ss, tag);
if (tag != ENDPRODUCTS) {
throw ReactionPicklerException(
"Bad pickle format: ENDPRODUCTS tag not found.");
}
if (numAgents != 0) {
streamRead(ss, tag);
if (tag != BEGINAGENTS) {
throw ReactionPicklerException(
"Bad pickle format: BEGINAGENTS tag not found.");
}
for (unsigned int i = 0; i < numAgents; ++i) {
auto *mol = new RWMol();
MolPickler::molFromPickle(ss, mol);
rxn->addAgentTemplate(ROMOL_SPTR(mol));
}
streamRead(ss, tag);
if (tag != ENDAGENTS) {
throw ReactionPicklerException(
"Bad pickle format: ENDAGENTS tag not found.");
}
}
if (version >= 3000) {
streamRead(ss, tag);
if (tag == BEGINSSSPARAMS) {
std::string json;
streamRead(ss, json, version);
updateSubstructMatchParamsFromJSON(rxn->getSubstructParams(), json);
streamRead(ss, tag);
if (tag != ENDSSSPARAMS) {
throw ReactionPicklerException(
"Bad pickle format: ENDSSSPARAMS tag not found.");
}
}
}
streamRead(ss, tag);
if (tag == BEGINPROPS) {
_unpickleProperties(ss, *rxn);
streamRead(ss, tag);
if (tag != ENDPROPS) {
throw ReactionPicklerException(
"Bad pickle format: ENDPROPS tag not found.");
}
}
// need to do this after we add reactants and products
rxn->df_needsInit = flag & 0x2;
} // end of _depickle
}; // namespace RDKit