Skip to content

patonlab/pyQRC

Repository files navigation

pyQRC

DOI PyPI version Python versions Downloads License CircleCI codecov

Introduction

QRC is an abbreviation of Quick Reaction Coordinate. This provides a quick alternative to IRC (intrinsic reaction coordinate) calculations. This was first described by Silva and Goodman.1 The original code was developed in Java for Jaguar output files. This Python version uses cclib to process a variety of computational chemistry outputs.

The program will read a Gaussian frequency calculation and will create a new input file which has been projected from the final coordinates along the Hessian eigenvector with a negative force constant. The magnitude of displacement can be adjusted on the command line. By default the projection will be in a positive sense (in relation to the imaginary normal mode) and the level of theory in the new input file will match that of the frequency calculation.

In addition to a pound-shop (dollar store) IRC calculation, a common application for pyQRC is in distorting ground state structures to remove annoying imaginary frequencies after reoptimization. This code has, in some form or other, been in use since around 2010.

pyQRC reads frequency calculations from Gaussian, ORCA, and Q-Chem. It also integrates with ASE and machine-learned interatomic potentials (MLIPs such as MACE, ANI, or AIMNet2): the bridge script in examples/ase_mlip writes ASE frequency results in a format pyQRC reads, so Hessians from an MLIP work exactly like QM output files. Runnable Jupyter notebooks demonstrating each route — Gaussian 16, ORCA 5 and 6, Q-Chem, and ASE/MLIP — ship in the examples directory.

Quick Start

# Install
pip install pyqrc

# Basic usage - displace along imaginary frequency
python -m pyqrc my_ts.log

# Specify processors and memory for the new input file
python -m pyqrc my_ts.log --nproc 4 --mem 8GB

Installation

Via PyPI (recommended):

pip install pyqrc

Via uv:

uv pip install pyqrc

Via pixi:

pixi add --pypi pyqrc

From source: Clone the repository https://github.com/patonlab/pyQRC.git and add to your PYTHONPATH variable.

ORCA 6 compatibility

Parsing ORCA 6 output files requires a newer cclib than the current PyPI release (1.8.1), which fails on ORCA 6's SCF block. Until cclib ships a release with the fix, install cclib from GitHub master alongside pyQRC:

pip install pyqrc
pip install --upgrade 'git+https://github.com/cclib/cclib.git'

Note: cclib master currently has a regression affecting some Q-Chem outputs. If you primarily use Q-Chem, stay on cclib 1.8.1.

Then run the script as a Python module with your computational chemistry output files (the program expects .log or .out extensions) and can accept wildcard arguments.

Usage

python -m pyqrc [options] <output_file(s)>

Command Line Options

Option Description Default
--amp AMPLITUDE Multiplier for the imaginary normal mode vector. Increase for larger displacements; use negative values for reverse direction. 0.2
--nproc NPROC Number of processors requested in the new input file. 1
--mem MEM Memory requested in the new input file. Format: XGB or X000MB. 4GB
--route ROUTE Route line for the new calculation, e.g. 'THEORY/BASIS opt'. Same as original
-q, --quiet Suppress verbose output (skips the .qrc summary file). Verbose by default
--auto Only process files with imaginary frequencies, skip others. Disabled
--name SUFFIX String appended to the filename for new input file(s). QRC
-f, --freq FREQ Displace along the normal mode nearest this frequency (cm⁻¹); errors if no mode is within 1 cm⁻¹. All imaginary
--freqnum FREQNUM Displace along frequency number N (from lowest); errors if N exceeds the number of modes. All imaginary
--qcoord Deprecated, removal in 3.0. Runs Gaussian single points along normal modes directly on the local machine (requires g16 on PATH). Generate inputs with the default mode and submit them through your scheduler instead. Disabled
--nummodes NUMMODES Deprecated, removal in 3.0. Number of modes for --qcoord calculations. all

Output Files

pyQRC generates the following files:

  • <filename>_QRC.com (Gaussian) or <filename>_QRC.inp (ORCA/Q-Chem): New input file with displaced geometry ready for optimization.
  • <filename>_QRC.qrc: Summary file containing:
    • Original geometry
    • Harmonic frequencies, reduced masses, and force constants
    • Normal mode displacement vectors
    • Mass-weighted Cartesian displacement magnitude

Dependencies

  • Python >= 3.9
  • cclib >= 1.8.1, < 2 (ORCA 6 outputs need a newer cclib than 1.8.1 — see "ORCA 6 compatibility" above)
  • NumPy >= 1.22
  • One of the following computational chemistry packages:

Examples

The input and output files for the examples below ship in the examples directory. Each format subdirectory (g16, orca5, orca6, qchem, ase_mlip) also contains a runnable Jupyter notebook (generate_qrc_inputs.ipynb) that walks through generating QRC inputs from those files.

Example 1: Remove an Unwanted Imaginary Frequency

python -m pyqrc acetaldehyde.log --nproc 4 --mem 8GB

This initial optimization inadvertently produced a transition structure. The code displaces along the normal mode and creates a new input file. A subsequent optimization then fixes the problem since the imaginary frequency disappears. Note that by default this displacement occurs along all imaginary modes - if there is more than one imaginary frequency, and displacement is only desired along one of these (e.g. the lowest) then the use of --freqnum 1 is necessary.

Example 2: Map a Reaction Coordinate (QRC)

python -m pyqrc claisen_ts.log --nproc 4 --mem 8GB --amp 0.3 --name QRCF
python -m pyqrc claisen_ts.log --nproc 4 --mem 8GB --amp -0.3 --name QRCR

The initial optimization located a transition structure. The quick reaction coordinate (QRC) is obtained from two optimizations, started from two points displaced along the reaction coordinate in either direction.

Example 3: Conformational Sampling via Normal Mode Displacement

python -m pyqrc planar_chex.log --nproc 4 --freqnum 1 --name mode1
python -m pyqrc planar_chex.log --nproc 4 --freqnum 3 --name mode3

In this example, the initial optimization located a (3rd order) saddle point - planar cyclohexane - with three imaginary frequencies. Two new inputs are created by displacing along (i) only the first (i.e., lowest) normal mode and (ii) only the third normal mode. This contrasts from the --auto function of pyQRC which displaces along all imaginary modes. Subsequent optimizations of these new inputs results in different minima, producing (i) chair-shaped cyclohexane and (ii) twist-boat cyclohexane. This example illustrates that displacement along particular normal modes could be used for e.g. conformational sampling.

Example 4: QRC from an ASE / MLIP Frequency Calculation

When the Hessian comes from a machine-learned interatomic potential driven through ASE rather than a QM package, there is no output file for pyQRC to read. The helper script examples/ase_mlip/ase2gaussian.py bridges the gap: it writes an ASE Vibrations result as a Gaussian-format log file that cclib parses, after which pyQRC works exactly as in the examples above. The accompanying notebook walks through the full loop with MACE-OFF: locating the planar NH₃ inversion transition state and relaxing the QRC-displaced geometry to the pyramidal minimum, then mapping the Claisen reaction coordinate of Example 2 entirely on the MLIP — the forward and reverse QRC displacements from the DFT transition state relax to 4-pentenal and allyl vinyl ether without any further QM calculations.

Comparison with IRC

To benchmark QRC against intrinsic reaction coordinate (IRC) calculations, 544 transition states from the Grambow dataset were used. Transition states were reoptimized at the wB97XD/6-31G(d) level, and QRC calculations were performed with three displacement amplitudes (0.1, 0.3, and 0.5). The resulting reactant and product identities (canonical SMILES) and energies were compared against IRC results for the same structures.

Amplitude Barrier MAE (kcal/mol) Rxn Energy MAE (kcal/mol) Reactant Match Product Match
0.1 4.55 5.73 97.8% 98.2%
0.3 0.65 1.39 98.9% 98.9%
0.5 0.58 3.44 98.7% 98.2%

An amplitude of 0.3 gives the best overall performance, with the highest SMILES match rates for both reactants and products (98.9%) and low MAE for barriers (0.65 kcal/mol) and reaction energies (1.39 kcal/mol). While an amplitude of 0.5 gives a marginally lower barrier MAE (0.58 kcal/mol), it produces larger errors in reaction energies and lower product match rates. An amplitude of 0.1 gives insufficient displacement, leading to higher energy errors and more mismatched products. Full details and data are available in the irc_comparison directory.

Development

git clone https://github.com/patonlab/pyQRC.git
cd pyQRC
pip install -e ".[dev]"
pytest            # run the test suite
pylint pyqrc      # lint (CI requires a score >= 9.0)

Planned work is tracked in ROADMAP.md.

Citation

If you use pyQRC in your research, please cite:

Paton, R. S.; Sowndarya S. V., S.; Landis, J.; Goodfellow, A. S. pyQRC. DOI: 10.5281/zenodo.3365476

References

  1. (a) Goodman, J. M.; Silva, M. A. Tetrahedron Lett. 2003, 44, 8233-8236 DOI: 10.1016/j.tetlet.2003.09.074; (b) Goodman, J. M.; Silva, M. A. Tetrahedron Lett. 2005, 46, 2067-2069 DOI: 10.1016/j.tetlet.2005.01.142

Contributors


License: MIT

Packages

 
 
 

Contributors