Skip to content

Commit

Permalink
Merge pull request ReactionMechanismGenerator#2525 from rwest/pytest
Browse files Browse the repository at this point in the history
  • Loading branch information
rwest authored Aug 16, 2023
2 parents 32715ad + e946ec2 commit c6e2c19
Show file tree
Hide file tree
Showing 19 changed files with 574 additions and 442 deletions.
1 change: 1 addition & 0 deletions pytest.ini
Original file line number Diff line number Diff line change
Expand Up @@ -24,3 +24,4 @@ addopts =
--cov-report html
testpaths = test
python_files = *Test.py
python_classes = *Test Test*
386 changes: 386 additions & 0 deletions test/arkane/arkaneFunctionalInputTest.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,386 @@
#!/usr/bin/env python3

###############################################################################
# #
# RMG - Reaction Mechanism Generator #
# #
# Copyright (c) 2002-2023 Prof. William H. Green (whgreen@mit.edu), #
# Prof. Richard H. West (r.west@neu.edu) and the RMG Team (rmg_dev@mit.edu) #
# #
# Permission is hereby granted, free of charge, to any person obtaining a #
# copy of this software and associated documentation files (the 'Software'), #
# to deal in the Software without restriction, including without limitation #
# the rights to use, copy, modify, merge, publish, distribute, sublicense, #
# and/or sell copies of the Software, and to permit persons to whom the #
# Software is furnished to do so, subject to the following conditions: #
# #
# The above copyright notice and this permission notice shall be included in #
# all copies or substantial portions of the Software. #
# #
# THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR #
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, #
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE #
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER #
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING #
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER #
# DEALINGS IN THE SOFTWARE. #
# #
###############################################################################

"""
This module contains unit tests of the :mod:`arkane.input` module.
"""

import os

from rmgpy.pdep.collision import SingleExponentialDown
import pytest

import rmgpy
from rmgpy.exceptions import InputError
from rmgpy.kinetics.tunneling import Eckart
from rmgpy.statmech.rotation import NonlinearRotor
from rmgpy.statmech.translation import IdealGasTranslation
from rmgpy.statmech.vibration import HarmonicOscillator
from rmgpy.thermo.nasa import NASAPolynomial, NASA
from rmgpy.transport import TransportData

from arkane.input import (
species,
transitionState,
reaction,
SMILES,
load_input_file,
process_model_chemistry,
)
from arkane.modelchem import LevelOfTheory, CompositeLevelOfTheory


ADMONITION = (
"This unit test fails in the new pytest framework despite other similar tests passing. "
"This is likely due to a global state issue that was implicitly handled differently by the old nose testing "
"framework. \nThe best solution for this problem is to remove the abuse of the global "
"state in Arkane and RMG-Py rather than try and fix this single test."
)


class FunctionalInputTest:
"""
Contains unit tests for the Arkane input module
"""

def test_species(self):
"""
Test loading a species from input file-like kew word arguments
"""
label0 = "CH2O"
kwargs = {
"E0": (28.69, "kcal/mol"),
"structure": SMILES("C=O"),
"collisionModel": TransportData(sigma=(3.69e-10, "m"), epsilon=(4.0, "kJ/mol")),
"energyTransferModel": SingleExponentialDown(alpha0=(0.956, "kJ/mol"), T0=(300, "K"), n=0.95),
"spinMultiplicity": 1,
"opticalIsomers": 1,
"modes": [
HarmonicOscillator(frequencies=([1180, 1261, 1529, 1764, 2931, 2999], "cm^-1")),
NonlinearRotor(
rotationalConstant=(
[1.15498821005263, 1.3156969584727, 9.45570474524524],
"cm^-1",
),
symmetry=2,
quantum=False,
),
IdealGasTranslation(mass=(30.0106, "g/mol")),
],
}

spc0 = species(label0, **kwargs)
assert spc0.label == "CH2O"
assert spc0.smiles == "C=O"
assert round(abs(spc0.conformer.E0.value_si - 120038.96), 7) == 0
assert spc0.conformer.spin_multiplicity == 1
assert spc0.conformer.optical_isomers == 1
assert len(spc0.conformer.modes) == 3
assert isinstance(spc0.transport_data, TransportData)
assert isinstance(spc0.energy_transfer_model, SingleExponentialDown)

@pytest.mark.skip(reason=ADMONITION)
def test_species_atomic_nasa_polynomial(self):
"""
Test loading a atom with NASA polynomials
"""
label0 = "H(1)"
kwargs = {
"structure": SMILES("[H]"),
"thermo": NASA(
polynomials=[
NASAPolynomial(
coeffs=[2.5, 0, 0, 0, 0, 25473.7, -0.446683],
Tmin=(200, "K"),
Tmax=(1000, "K"),
),
NASAPolynomial(
coeffs=[2.5, 0, 0, 0, 0, 25473.7, -0.446683],
Tmin=(1000, "K"),
Tmax=(6000, "K"),
),
],
Tmin=(200, "K"),
Tmax=(6000, "K"),
comment="""Thermo library: FFCM1(-)""",
),
"energyTransferModel": SingleExponentialDown(alpha0=(3.5886, "kJ/mol"), T0=(300, "K"), n=0.85),
}
spc0 = species(label0, **kwargs)
assert spc0.label == label0
assert spc0.smiles == "[H]"
assert spc0.has_statmech()
assert spc0.thermo == kwargs["thermo"]

@pytest.mark.skip(reason=ADMONITION)
def test_species_polyatomic_nasa_polynomial(self):
"""
Test loading a species with NASA polynomials
"""
label0 = "benzyl"
kwargs = {
"structure": SMILES("[c]1ccccc1"),
"thermo": NASA(
polynomials=[
NASAPolynomial(
coeffs=[
2.78632,
0.00784632,
7.97887e-05,
-1.11617e-07,
4.39429e-11,
39695,
11.5114,
],
Tmin=(100, "K"),
Tmax=(943.73, "K"),
),
NASAPolynomial(
coeffs=[
13.2455,
0.0115667,
-2.49996e-06,
4.66496e-10,
-4.12376e-14,
35581.1,
-49.6793,
],
Tmin=(943.73, "K"),
Tmax=(5000, "K"),
),
],
Tmin=(100, "K"),
Tmax=(5000, "K"),
comment="""Thermo library: Fulvene_H + radical(CbJ)""",
),
"energyTransferModel": SingleExponentialDown(alpha0=(3.5886, "kJ/mol"), T0=(300, "K"), n=0.85),
}
spc0 = species(label0, **kwargs)
assert spc0.label == label0
assert spc0.has_statmech()
assert spc0.thermo == kwargs["thermo"]

def test_transition_state(self):
"""
Test loading a transition state from input file-like kew word arguments
"""
label0 = "TS1"
kwargs = {
"E0": (39.95, "kcal/mol"),
"spinMultiplicity": 2,
"opticalIsomers": 1,
"frequency": (-1934, "cm^-1"),
"modes": [
HarmonicOscillator(
frequencies=(
[792, 987, 1136, 1142, 1482, 2441, 3096, 3183],
"cm^-1",
)
),
NonlinearRotor(
rotationalConstant=([0.928, 0.962, 5.807], "cm^-1"),
symmetry=1,
quantum=False,
),
IdealGasTranslation(mass=(31.01843, "g/mol")),
],
}

ts0 = transitionState(label0, **kwargs)
assert ts0.label == "TS1"
assert round(abs(ts0.conformer.E0.value_si - 167150.8), 7) == 0
assert ts0.conformer.spin_multiplicity == 2
assert ts0.conformer.optical_isomers == 1
assert ts0.frequency.value_si == -1934.0
assert len(ts0.conformer.modes) == 3

def test_reaction(self):
"""
Test loading a reaction from input file-like kew word arguments
"""

species(
label="methoxy",
structure=SMILES("C[O]"),
E0=(9.44, "kcal/mol"),
modes=[
HarmonicOscillator(
frequencies=(
[758, 960, 1106, 1393, 1403, 1518, 2940, 3019, 3065],
"cm^-1",
)
),
NonlinearRotor(
rotationalConstant=([0.916, 0.921, 5.251], "cm^-1"),
symmetry=3,
quantum=False,
),
IdealGasTranslation(mass=(31.01843, "g/mol")),
],
spinMultiplicity=2,
opticalIsomers=1,
molecularWeight=(31.01843, "amu"),
collisionModel=TransportData(sigma=(3.69e-10, "m"), epsilon=(4.0, "kJ/mol")),
energyTransferModel=SingleExponentialDown(alpha0=(0.956, "kJ/mol"), T0=(300, "K"), n=0.95),
)

species(
label="formaldehyde",
E0=(28.69, "kcal/mol"),
molecularWeight=(30.0106, "g/mol"),
collisionModel=TransportData(sigma=(3.69e-10, "m"), epsilon=(4.0, "kJ/mol")),
energyTransferModel=SingleExponentialDown(alpha0=(0.956, "kJ/mol"), T0=(300, "K"), n=0.95),
spinMultiplicity=1,
opticalIsomers=1,
modes=[
HarmonicOscillator(frequencies=([1180, 1261, 1529, 1764, 2931, 2999], "cm^-1")),
NonlinearRotor(
rotationalConstant=(
[1.15498821005263, 1.3156969584727, 9.45570474524524],
"cm^-1",
),
symmetry=2,
quantum=False,
),
IdealGasTranslation(mass=(30.0106, "g/mol")),
],
)

species(
label="H",
E0=(0.000, "kcal/mol"),
molecularWeight=(1.00783, "g/mol"),
collisionModel=TransportData(sigma=(3.69e-10, "m"), epsilon=(4.0, "kJ/mol")),
energyTransferModel=SingleExponentialDown(alpha0=(0.956, "kJ/mol"), T0=(300, "K"), n=0.95),
modes=[IdealGasTranslation(mass=(1.00783, "g/mol"))],
spinMultiplicity=2,
opticalIsomers=1,
)

transitionState(
label="TS3",
E0=(34.1, "kcal/mol"),
spinMultiplicity=2,
opticalIsomers=1,
frequency=(-967, "cm^-1"),
modes=[
HarmonicOscillator(
frequencies=(
[466, 581, 1169, 1242, 1499, 1659, 2933, 3000],
"cm^-1",
)
),
NonlinearRotor(
rotationalConstant=([0.970, 1.029, 3.717], "cm^-1"),
symmetry=1,
quantum=False,
),
IdealGasTranslation(mass=(31.01843, "g/mol")),
],
)

reactants = ["formaldehyde", "H"]
products = ["methoxy"]
tunneling = "Eckart"

rxn = reaction("CH2O+H=Methoxy", reactants, products, "TS3", tunneling=tunneling)
assert rxn.label == "CH2O+H=Methoxy"
assert len(rxn.reactants) == 2
assert len(rxn.products) == 1
assert round(abs(rxn.reactants[0].conformer.E0.value_si - 0), 7) == 0
assert round(abs(rxn.reactants[1].conformer.E0.value_si - 120038.96), 7) == 0
assert round(abs(rxn.products[0].conformer.E0.value_si - 39496.96), 7) == 0
assert round(abs(rxn.transition_state.conformer.E0.value_si - 142674.4), 7) == 0
assert round(abs(rxn.transition_state.frequency.value_si - -967.0), 7) == 0
assert isinstance(rxn.transition_state.tunneling, Eckart)

def test_load_input_file(self):
"""Test loading an Arkane input file"""
path = os.path.join(
os.path.dirname(os.path.dirname(rmgpy.__file__)),
"examples",
"arkane",
"networks",
"acetyl+O2",
"input.py",
)
(
job_list,
reaction_dict,
species_dict,
transition_state_dict,
network_dict,
model_chemistry,
) = load_input_file(path)

assert len(job_list) == 1

assert len(reaction_dict) == 5
assert "entrance1" in reaction_dict
assert "exit2" in reaction_dict

assert len(species_dict) == 9
assert "acetyl" in species_dict
assert "hydroperoxyl" in species_dict

assert len(transition_state_dict) == 5
assert "entrance1" in transition_state_dict
assert "isom1" in transition_state_dict

assert len(network_dict) == 1
assert "acetyl + O2" in network_dict

assert model_chemistry is None

def test_process_model_chemistry(self):
"""
Test processing the model chemistry to derive the sp and freq levels
"""
mc = "ccsd(t)-f12a/aug-cc-pvtz//b3lyp/6-311++g(3df,3pd)"
lot = process_model_chemistry(mc)
assert isinstance(lot, CompositeLevelOfTheory)
assert lot.energy == LevelOfTheory("ccsd(t)-f12a", "aug-cc-pvtz")
assert lot.freq == LevelOfTheory("b3lyp", "6-311++g(3df,3pd)")

mc = "b3lyp-d3/def2-tzvp"
lot = process_model_chemistry(mc)
assert isinstance(lot, LevelOfTheory)
assert lot == LevelOfTheory("b3lyp-d3", "def2-tzvp")

mc = "cbs-qb3"
lot = process_model_chemistry(mc)
assert isinstance(lot, LevelOfTheory)
assert lot == LevelOfTheory("cbs-qb3")

mc = LevelOfTheory("test")
lot = process_model_chemistry(mc)
assert mc is lot

with pytest.raises(InputError):
process_model_chemistry("CCSD(T)-F12a/aug-cc-pVTZ//CCSD(T)-F12a/aug-cc-pVTZ//B3LYP/6-311++G(3df,3pd)")
Loading

0 comments on commit c6e2c19

Please sign in to comment.