"""
Command-line interface for HBAT.
This module provides a command-line interface for running HBAT analysis
without the GUI, suitable for batch processing and scripting.
"""
import argparse
import json
import os
import sys
import time
from typing import Any, Dict, List, Optional
from .. import __version__
from ..constants.parameters import ParametersDefault
from ..core.analysis import AnalysisParameters, NPMolecularInteractionAnalyzer
from ..core.pdb_parser import PDBParser
[docs]
class ProgressBar:
"""Simple CLI progress bar for analysis operations."""
[docs]
def __init__(self, width: int = 50):
"""Initialize progress bar.
:param width: Width of the progress bar in characters
:type width: int
"""
self.width = width
self.current_step = ""
self.last_progress = -1
[docs]
def update(self, message: str, progress: Optional[int] = None) -> None:
"""Update progress bar with new message and optional percentage.
:param message: Current operation message
:type message: str
:param progress: Progress percentage (0-100), optional
:type progress: Optional[int]
"""
# Clear previous line and print new status
print(f"\r\033[K[INFO] {message}", end="", flush=True)
if progress is not None and progress != self.last_progress:
# Add progress bar for percentage updates with emoji
filled_width = int(self.width * progress / 100)
bar = "●" * filled_width + "○" * (self.width - filled_width)
print(f" [{bar}] {progress}%", end="", flush=True)
self.last_progress = progress
[docs]
def finish(self, message: str) -> None:
"""Finish progress bar with final message.
:param message: Final completion message
:type message: str
"""
print(f"\r\033[K[INFO] {message}")
self.last_progress = -1
[docs]
def create_parser() -> argparse.ArgumentParser:
"""Create command-line argument parser.
Creates and configures an ArgumentParser with all CLI options for HBAT,
including input/output options, analysis parameters, and preset management.
:returns: Configured argument parser
:rtype: argparse.ArgumentParser
"""
parser = argparse.ArgumentParser(
description="HBAT - Hydrogen Bond Analysis Tool",
formatter_class=argparse.RawDescriptionHelpFormatter,
epilog="""
Examples:
%(prog)s input.pdb # Basic analysis
%(prog)s input.pdb -o results.txt # Save results to text file
%(prog)s input.pdb -o results.csv # Save results to CSV file (single file)
%(prog)s input.pdb -o results.json # Save results to JSON file (single file)
%(prog)s input.pdb --csv results # Export to multiple CSV files (one per interaction type)
%(prog)s input.pdb --json results # Export to multiple JSON files (one per interaction type)
%(prog)s input.pdb --hb-distance 3.0 # Custom H-bond distance cutoff
%(prog)s input.pdb --mode local # Local interactions only
%(prog)s --list-presets # List available presets
%(prog)s input.pdb --preset high_resolution # Use preset with custom overrides
""",
)
# Version
parser.add_argument(
"--version", action="version", version=f"%(prog)s {__version__}"
)
# Input file (optional when listing presets)
parser.add_argument("input", nargs="?", help="Input PDB file")
# Output options
parser.add_argument("-o", "--output", help="Output file (format auto-detected from extension: .txt, .csv, .json)")
parser.add_argument("--json", help="Export to multiple JSON files (base name for files)")
parser.add_argument("--csv", help="Export to multiple CSV files (base name for files)")
# Preset options
preset_group = parser.add_argument_group("Preset Options")
preset_group.add_argument(
"--preset", type=str, help="Load parameters from preset file (.hbat or .json)"
)
preset_group.add_argument(
"--list-presets",
action="store_true",
help="List available example presets and exit",
)
# Analysis parameters
param_group = parser.add_argument_group("Analysis Parameters")
param_group.add_argument(
"--hb-distance",
type=float,
default=ParametersDefault.HB_DISTANCE_CUTOFF,
help=f"Hydrogen bond H...A distance cutoff in Å (default: {ParametersDefault.HB_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--hb-angle",
type=float,
default=ParametersDefault.HB_ANGLE_CUTOFF,
help=f"Hydrogen bond D-H...A angle cutoff in degrees (default: {ParametersDefault.HB_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--da-distance",
type=float,
default=ParametersDefault.HB_DA_DISTANCE,
help=f"Donor-acceptor distance cutoff in Å (default: {ParametersDefault.HB_DA_DISTANCE})",
)
param_group.add_argument(
"--whb-distance",
type=float,
default=ParametersDefault.WHB_DISTANCE_CUTOFF,
help=f"Weak hydrogen bond H...A distance cutoff in Å for carbon donors (default: {ParametersDefault.WHB_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--whb-angle",
type=float,
default=ParametersDefault.WHB_ANGLE_CUTOFF,
help=f"Weak hydrogen bond D-H...A angle cutoff in degrees for carbon donors (default: {ParametersDefault.WHB_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--whb-da-distance",
type=float,
default=ParametersDefault.WHB_DA_DISTANCE,
help=f"Weak hydrogen bond donor-acceptor distance cutoff in Å for carbon donors (default: {ParametersDefault.WHB_DA_DISTANCE})",
)
param_group.add_argument(
"--xb-distance",
type=float,
default=ParametersDefault.XB_DISTANCE_CUTOFF,
help=f"Halogen bond X...A distance cutoff in Å (default: {ParametersDefault.XB_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--xb-angle",
type=float,
default=ParametersDefault.XB_ANGLE_CUTOFF,
help=f"Halogen bond C-X...A angle cutoff in degrees (default: {ParametersDefault.XB_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--pi-distance",
type=float,
default=ParametersDefault.PI_DISTANCE_CUTOFF,
help=f"π interaction H...π distance cutoff in Å (default: {ParametersDefault.PI_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--pi-angle",
type=float,
default=ParametersDefault.PI_ANGLE_CUTOFF,
help=f"π interaction D-H...π angle cutoff in degrees (default: {ParametersDefault.PI_ANGLE_CUTOFF})",
)
# π interaction subtype parameters
param_group.add_argument(
"--pi-ccl-distance",
type=float,
default=ParametersDefault.PI_CCL_DISTANCE_CUTOFF,
help=f"C-Cl...π interaction distance cutoff in Å (default: {ParametersDefault.PI_CCL_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--pi-ccl-angle",
type=float,
default=ParametersDefault.PI_CCL_ANGLE_CUTOFF,
help=f"C-Cl...π interaction angle cutoff in degrees (default: {ParametersDefault.PI_CCL_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--pi-cbr-distance",
type=float,
default=ParametersDefault.PI_CBR_DISTANCE_CUTOFF,
help=f"C-Br...π interaction distance cutoff in Å (default: {ParametersDefault.PI_CBR_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--pi-cbr-angle",
type=float,
default=ParametersDefault.PI_CBR_ANGLE_CUTOFF,
help=f"C-Br...π interaction angle cutoff in degrees (default: {ParametersDefault.PI_CBR_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--pi-ci-distance",
type=float,
default=ParametersDefault.PI_CI_DISTANCE_CUTOFF,
help=f"C-I...π interaction distance cutoff in Å (default: {ParametersDefault.PI_CI_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--pi-ci-angle",
type=float,
default=ParametersDefault.PI_CI_ANGLE_CUTOFF,
help=f"C-I...π interaction angle cutoff in degrees (default: {ParametersDefault.PI_CI_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--pi-ch-distance",
type=float,
default=ParametersDefault.PI_CH_DISTANCE_CUTOFF,
help=f"C-H...π interaction distance cutoff in Å (default: {ParametersDefault.PI_CH_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--pi-ch-angle",
type=float,
default=ParametersDefault.PI_CH_ANGLE_CUTOFF,
help=f"C-H...π interaction angle cutoff in degrees (default: {ParametersDefault.PI_CH_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--pi-nh-distance",
type=float,
default=ParametersDefault.PI_NH_DISTANCE_CUTOFF,
help=f"N-H...π interaction distance cutoff in Å (default: {ParametersDefault.PI_NH_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--pi-nh-angle",
type=float,
default=ParametersDefault.PI_NH_ANGLE_CUTOFF,
help=f"N-H...π interaction angle cutoff in degrees (default: {ParametersDefault.PI_NH_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--pi-oh-distance",
type=float,
default=ParametersDefault.PI_OH_DISTANCE_CUTOFF,
help=f"O-H...π interaction distance cutoff in Å (default: {ParametersDefault.PI_OH_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--pi-oh-angle",
type=float,
default=ParametersDefault.PI_OH_ANGLE_CUTOFF,
help=f"O-H...π interaction angle cutoff in degrees (default: {ParametersDefault.PI_OH_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--pi-sh-distance",
type=float,
default=ParametersDefault.PI_SH_DISTANCE_CUTOFF,
help=f"S-H...π interaction distance cutoff in Å (default: {ParametersDefault.PI_SH_DISTANCE_CUTOFF})",
)
param_group.add_argument(
"--pi-sh-angle",
type=float,
default=ParametersDefault.PI_SH_ANGLE_CUTOFF,
help=f"S-H...π interaction angle cutoff in degrees (default: {ParametersDefault.PI_SH_ANGLE_CUTOFF})",
)
param_group.add_argument(
"--covalent-factor",
type=float,
default=ParametersDefault.COVALENT_CUTOFF_FACTOR,
help=f"Covalent bond detection factor (default: {ParametersDefault.COVALENT_CUTOFF_FACTOR})",
)
# Analysis mode
param_group.add_argument(
"--mode",
choices=["complete", "local"],
default=ParametersDefault.ANALYSIS_MODE,
help="Analysis mode: complete (all interactions) or local (intra-residue only)",
)
# PDB structure fixing options
fix_group = parser.add_argument_group("PDB Structure Fixing")
fix_group.add_argument(
"--fix-pdb",
action="store_true",
help="Enable PDB structure fixing",
)
fix_group.add_argument(
"--fix-method",
choices=["openbabel", "pdbfixer"],
default=ParametersDefault.FIX_PDB_METHOD,
help=f"PDB fixing method: openbabel or pdbfixer (default: {ParametersDefault.FIX_PDB_METHOD})",
)
fix_group.add_argument(
"--fix-add-hydrogens",
action="store_true",
default=ParametersDefault.FIX_PDB_ADD_HYDROGENS,
help="Add missing hydrogen atoms (both OpenBabel and PDBFixer)",
)
fix_group.add_argument(
"--fix-add-heavy-atoms",
action="store_true",
help="Add missing heavy atoms (PDBFixer only)",
)
fix_group.add_argument(
"--fix-replace-nonstandard",
action="store_true",
help="Replace nonstandard residues (PDBFixer only)",
)
fix_group.add_argument(
"--fix-remove-heterogens",
action="store_true",
help="Remove heterogens (PDBFixer only)",
)
fix_group.add_argument(
"--fix-keep-water",
action="store_true",
default=ParametersDefault.FIX_PDB_KEEP_WATER,
help="Keep water when removing heterogens (PDBFixer only)",
)
# Output control
output_group = parser.add_argument_group("Output Control")
output_group.add_argument(
"--verbose",
"-v",
action="store_true",
help="Verbose output with detailed progress",
)
output_group.add_argument(
"--quiet", "-q", action="store_true", help="Quiet mode with minimal output"
)
output_group.add_argument(
"--summary-only", action="store_true", help="Output summary statistics only"
)
# Analysis filters
filter_group = parser.add_argument_group("Analysis Filters")
filter_group.add_argument(
"--no-hydrogen-bonds", action="store_true", help="Skip hydrogen bond analysis"
)
filter_group.add_argument(
"--no-halogen-bonds", action="store_true", help="Skip halogen bond analysis"
)
filter_group.add_argument(
"--no-pi-interactions", action="store_true", help="Skip π interaction analysis"
)
return parser
[docs]
def get_example_presets_directory() -> str:
"""Get the example presets directory relative to the package.
Locates the example_presets directory that contains predefined
analysis parameter configurations.
:returns: Path to the example presets directory
:rtype: str
"""
# Get the directory of this file
current_dir = os.path.dirname(os.path.abspath(__file__))
# Go up to the hbat package root, then to example_presets
package_root = os.path.dirname(os.path.dirname(current_dir))
example_presets_dir = os.path.join(package_root, "example_presets")
return example_presets_dir
[docs]
def list_available_presets() -> None:
"""List all available example presets.
Displays a formatted list of all available preset files with descriptions
and icons to help users choose appropriate analysis parameters.
:returns: None
:rtype: None
"""
presets_dir = get_example_presets_directory()
if not os.path.exists(presets_dir):
print("No example presets directory found.")
return
print("🎯 Available HBAT Presets:")
print("=" * 50)
preset_files = [f for f in os.listdir(presets_dir) if f.endswith(".hbat")]
if not preset_files:
print("No preset files found in example_presets directory.")
return
# Define icons for different preset types
preset_icons = {
"high_resolution": "🔬",
"standard_resolution": "⚙️",
"low_resolution": "📐",
"nmr_structures": "🧬",
"strong_interactions_only": "💪",
"drug_design_strict": "💊",
"membrane_proteins": "🧱",
"weak_interactions_permissive": "🌐",
}
for preset_file in sorted(preset_files):
preset_name = preset_file.replace(".hbat", "")
icon = preset_icons.get(preset_name, "📄")
# Try to load and show description
preset_path = os.path.join(presets_dir, preset_file)
try:
with open(preset_path, "r") as f:
preset_data = json.load(f)
description = preset_data.get("description", "No description available")
print(f"{icon} {preset_name}")
print(f" {description}")
print()
except Exception:
print(f"{icon} {preset_name}")
print(" (Unable to load description)")
print()
[docs]
def load_preset_file(preset_path: str) -> AnalysisParameters:
"""Load parameters from a preset file.
Reads and parses a JSON preset file to create AnalysisParameters
with predefined values for various analysis scenarios.
:param preset_path: Path to the preset file to load
:type preset_path: str
:returns: Analysis parameters loaded from the preset
:rtype: AnalysisParameters
:raises ValueError: If preset file format is invalid
:raises FileNotFoundError: If preset file doesn't exist
"""
try:
with open(preset_path, "r", encoding="utf-8") as f:
preset_data = json.load(f)
# Validate basic structure
if "parameters" not in preset_data:
raise ValueError("Invalid preset file format: missing 'parameters' section")
params = preset_data["parameters"]
# Extract parameters with defaults
hb_params = params.get("hydrogen_bonds", {})
whb_params = params.get("weak_hydrogen_bonds", {})
xb_params = params.get("halogen_bonds", {})
pi_params = params.get("pi_interactions", {})
general_params = params.get("general", {})
fix_params = params.get("pdb_fixing", {})
return AnalysisParameters(
hb_distance_cutoff=hb_params.get(
"h_a_distance_cutoff", ParametersDefault.HB_DISTANCE_CUTOFF
),
hb_angle_cutoff=hb_params.get(
"dha_angle_cutoff", ParametersDefault.HB_ANGLE_CUTOFF
),
hb_donor_acceptor_cutoff=hb_params.get(
"d_a_distance_cutoff", ParametersDefault.HB_DA_DISTANCE
),
whb_distance_cutoff=whb_params.get(
"h_a_distance_cutoff", ParametersDefault.WHB_DISTANCE_CUTOFF
),
whb_angle_cutoff=whb_params.get(
"dha_angle_cutoff", ParametersDefault.WHB_ANGLE_CUTOFF
),
whb_donor_acceptor_cutoff=whb_params.get(
"d_a_distance_cutoff", ParametersDefault.WHB_DA_DISTANCE
),
xb_distance_cutoff=xb_params.get(
"x_a_distance_cutoff", ParametersDefault.XB_DISTANCE_CUTOFF
),
xb_angle_cutoff=xb_params.get(
"dxa_angle_cutoff", ParametersDefault.XB_ANGLE_CUTOFF
),
pi_distance_cutoff=pi_params.get(
"h_pi_distance_cutoff", ParametersDefault.PI_DISTANCE_CUTOFF
),
pi_angle_cutoff=pi_params.get(
"dh_pi_angle_cutoff", ParametersDefault.PI_ANGLE_CUTOFF
),
# π interaction subtype parameters
pi_ccl_distance_cutoff=pi_params.get(
"ccl_pi_distance_cutoff", ParametersDefault.PI_CCL_DISTANCE_CUTOFF
),
pi_ccl_angle_cutoff=pi_params.get(
"ccl_pi_angle_cutoff", ParametersDefault.PI_CCL_ANGLE_CUTOFF
),
pi_cbr_distance_cutoff=pi_params.get(
"cbr_pi_distance_cutoff", ParametersDefault.PI_CBR_DISTANCE_CUTOFF
),
pi_cbr_angle_cutoff=pi_params.get(
"cbr_pi_angle_cutoff", ParametersDefault.PI_CBR_ANGLE_CUTOFF
),
pi_ci_distance_cutoff=pi_params.get(
"ci_pi_distance_cutoff", ParametersDefault.PI_CI_DISTANCE_CUTOFF
),
pi_ci_angle_cutoff=pi_params.get(
"ci_pi_angle_cutoff", ParametersDefault.PI_CI_ANGLE_CUTOFF
),
pi_ch_distance_cutoff=pi_params.get(
"ch_pi_distance_cutoff", ParametersDefault.PI_CH_DISTANCE_CUTOFF
),
pi_ch_angle_cutoff=pi_params.get(
"ch_pi_angle_cutoff", ParametersDefault.PI_CH_ANGLE_CUTOFF
),
pi_nh_distance_cutoff=pi_params.get(
"nh_pi_distance_cutoff", ParametersDefault.PI_NH_DISTANCE_CUTOFF
),
pi_nh_angle_cutoff=pi_params.get(
"nh_pi_angle_cutoff", ParametersDefault.PI_NH_ANGLE_CUTOFF
),
pi_oh_distance_cutoff=pi_params.get(
"oh_pi_distance_cutoff", ParametersDefault.PI_OH_DISTANCE_CUTOFF
),
pi_oh_angle_cutoff=pi_params.get(
"oh_pi_angle_cutoff", ParametersDefault.PI_OH_ANGLE_CUTOFF
),
pi_sh_distance_cutoff=pi_params.get(
"sh_pi_distance_cutoff", ParametersDefault.PI_SH_DISTANCE_CUTOFF
),
pi_sh_angle_cutoff=pi_params.get(
"sh_pi_angle_cutoff", ParametersDefault.PI_SH_ANGLE_CUTOFF
),
covalent_cutoff_factor=general_params.get(
"covalent_cutoff_factor", ParametersDefault.COVALENT_CUTOFF_FACTOR
),
analysis_mode=general_params.get(
"analysis_mode", ParametersDefault.ANALYSIS_MODE
),
# PDB fixing parameters
fix_pdb_enabled=fix_params.get(
"enabled", ParametersDefault.FIX_PDB_ENABLED
),
fix_pdb_method=fix_params.get("method", ParametersDefault.FIX_PDB_METHOD),
fix_pdb_add_hydrogens=fix_params.get(
"add_hydrogens", ParametersDefault.FIX_PDB_ADD_HYDROGENS
),
fix_pdb_add_heavy_atoms=fix_params.get(
"add_heavy_atoms", ParametersDefault.FIX_PDB_ADD_HEAVY_ATOMS
),
fix_pdb_replace_nonstandard=fix_params.get(
"replace_nonstandard", ParametersDefault.FIX_PDB_REPLACE_NONSTANDARD
),
fix_pdb_remove_heterogens=fix_params.get(
"remove_heterogens", ParametersDefault.FIX_PDB_REMOVE_HETEROGENS
),
fix_pdb_keep_water=fix_params.get(
"keep_water", ParametersDefault.FIX_PDB_KEEP_WATER
),
)
except Exception as e:
print_error(f"Failed to load preset file '{preset_path}': {str(e)}")
sys.exit(1)
[docs]
def resolve_preset_path(preset_name: str) -> str:
"""Resolve preset name to full path.
Takes a preset name or partial path and resolves it to a full
path, searching in the example presets directory if needed.
:param preset_name: Name or path of the preset to resolve
:type preset_name: str
:returns: Full path to the preset file
:rtype: str
:raises SystemExit: If preset file cannot be found
"""
# If it's already a full path, use it
if os.path.isabs(preset_name) and os.path.exists(preset_name):
return preset_name
# If it's a relative path and exists, use it
if os.path.exists(preset_name):
return preset_name
# Try to find it in the example presets directory
presets_dir = get_example_presets_directory()
# If preset_name doesn't have extension, try adding .hbat
if not preset_name.endswith((".hbat", ".json")):
preset_name += ".hbat"
preset_path = os.path.join(presets_dir, preset_name)
if os.path.exists(preset_path):
return preset_path
# Try without directory (basename only)
basename = os.path.basename(preset_name)
preset_path = os.path.join(presets_dir, basename)
if os.path.exists(preset_path):
return preset_path
print_error(f"Preset file not found: {preset_name}")
print_error(f"Looked in: {presets_dir}")
print_error("Use --list-presets to see available presets")
sys.exit(1)
[docs]
def load_parameters_from_args(args: argparse.Namespace) -> AnalysisParameters:
"""Create AnalysisParameters from command-line arguments.
Processes command-line arguments to create analysis parameters,
with support for preset files and parameter overrides.
:param args: Parsed command-line arguments
:type args: argparse.Namespace
:returns: Analysis parameters configured from arguments
:rtype: AnalysisParameters
"""
# If preset is specified, load from preset file first
if hasattr(args, "preset") and args.preset:
preset_path = resolve_preset_path(args.preset)
print_progress(
f"Loading parameters from preset: {preset_path}",
not args.quiet if hasattr(args, "quiet") else True,
)
params = load_preset_file(preset_path)
# Override preset parameters with any explicitly provided CLI arguments
# Only override if the argument was explicitly set (not default)
parser = create_parser()
defaults = vars(parser.parse_args([])) # Get default values
if args.hb_distance != defaults.get("hb_distance"):
params.hb_distance_cutoff = args.hb_distance
if args.hb_angle != defaults.get("hb_angle"):
params.hb_angle_cutoff = args.hb_angle
if args.da_distance != defaults.get("da_distance"):
params.hb_donor_acceptor_cutoff = args.da_distance
if args.whb_distance != defaults.get("whb_distance"):
params.whb_distance_cutoff = args.whb_distance
if args.whb_angle != defaults.get("whb_angle"):
params.whb_angle_cutoff = args.whb_angle
if args.whb_da_distance != defaults.get("whb_da_distance"):
params.whb_donor_acceptor_cutoff = args.whb_da_distance
if args.xb_distance != defaults.get("xb_distance"):
params.xb_distance_cutoff = args.xb_distance
if args.xb_angle != defaults.get("xb_angle"):
params.xb_angle_cutoff = args.xb_angle
if args.pi_distance != defaults.get("pi_distance"):
params.pi_distance_cutoff = args.pi_distance
if args.pi_angle != defaults.get("pi_angle"):
params.pi_angle_cutoff = args.pi_angle
# π interaction subtype parameter overrides
if args.pi_ccl_distance != defaults.get("pi_ccl_distance"):
params.pi_ccl_distance_cutoff = args.pi_ccl_distance
if args.pi_ccl_angle != defaults.get("pi_ccl_angle"):
params.pi_ccl_angle_cutoff = args.pi_ccl_angle
if args.pi_cbr_distance != defaults.get("pi_cbr_distance"):
params.pi_cbr_distance_cutoff = args.pi_cbr_distance
if args.pi_cbr_angle != defaults.get("pi_cbr_angle"):
params.pi_cbr_angle_cutoff = args.pi_cbr_angle
if args.pi_ci_distance != defaults.get("pi_ci_distance"):
params.pi_ci_distance_cutoff = args.pi_ci_distance
if args.pi_ci_angle != defaults.get("pi_ci_angle"):
params.pi_ci_angle_cutoff = args.pi_ci_angle
if args.pi_ch_distance != defaults.get("pi_ch_distance"):
params.pi_ch_distance_cutoff = args.pi_ch_distance
if args.pi_ch_angle != defaults.get("pi_ch_angle"):
params.pi_ch_angle_cutoff = args.pi_ch_angle
if args.pi_nh_distance != defaults.get("pi_nh_distance"):
params.pi_nh_distance_cutoff = args.pi_nh_distance
if args.pi_nh_angle != defaults.get("pi_nh_angle"):
params.pi_nh_angle_cutoff = args.pi_nh_angle
if args.pi_oh_distance != defaults.get("pi_oh_distance"):
params.pi_oh_distance_cutoff = args.pi_oh_distance
if args.pi_oh_angle != defaults.get("pi_oh_angle"):
params.pi_oh_angle_cutoff = args.pi_oh_angle
if args.pi_sh_distance != defaults.get("pi_sh_distance"):
params.pi_sh_distance_cutoff = args.pi_sh_distance
if args.pi_sh_angle != defaults.get("pi_sh_angle"):
params.pi_sh_angle_cutoff = args.pi_sh_angle
if args.covalent_factor != defaults.get("covalent_factor"):
params.covalent_cutoff_factor = args.covalent_factor
if args.mode != defaults.get("mode"):
params.analysis_mode = args.mode
return params
else:
# Use CLI arguments only
return AnalysisParameters(
hb_distance_cutoff=args.hb_distance,
hb_angle_cutoff=args.hb_angle,
hb_donor_acceptor_cutoff=args.da_distance,
whb_distance_cutoff=args.whb_distance,
whb_angle_cutoff=args.whb_angle,
whb_donor_acceptor_cutoff=args.whb_da_distance,
xb_distance_cutoff=args.xb_distance,
xb_angle_cutoff=args.xb_angle,
pi_distance_cutoff=args.pi_distance,
pi_angle_cutoff=args.pi_angle,
# π interaction subtype parameters
pi_ccl_distance_cutoff=args.pi_ccl_distance,
pi_ccl_angle_cutoff=args.pi_ccl_angle,
pi_cbr_distance_cutoff=args.pi_cbr_distance,
pi_cbr_angle_cutoff=args.pi_cbr_angle,
pi_ci_distance_cutoff=args.pi_ci_distance,
pi_ci_angle_cutoff=args.pi_ci_angle,
pi_ch_distance_cutoff=args.pi_ch_distance,
pi_ch_angle_cutoff=args.pi_ch_angle,
pi_nh_distance_cutoff=args.pi_nh_distance,
pi_nh_angle_cutoff=args.pi_nh_angle,
pi_oh_distance_cutoff=args.pi_oh_distance,
pi_oh_angle_cutoff=args.pi_oh_angle,
pi_sh_distance_cutoff=args.pi_sh_distance,
pi_sh_angle_cutoff=args.pi_sh_angle,
covalent_cutoff_factor=args.covalent_factor,
analysis_mode=args.mode,
# PDB fixing parameters
fix_pdb_enabled=args.fix_pdb,
fix_pdb_method=args.fix_method,
fix_pdb_add_hydrogens=args.fix_add_hydrogens,
fix_pdb_add_heavy_atoms=args.fix_add_heavy_atoms,
fix_pdb_replace_nonstandard=args.fix_replace_nonstandard,
fix_pdb_remove_heterogens=args.fix_remove_heterogens,
fix_pdb_keep_water=args.fix_keep_water,
)
[docs]
def print_progress(message: str, verbose: bool = True) -> None:
"""Print progress message if verbose mode enabled.
:param message: Progress message to display
:type message: str
:param verbose: Whether to actually print the message
:type verbose: bool
:returns: None
:rtype: None
"""
if verbose:
print(f"[INFO] {message}")
[docs]
def print_error(message: str) -> None:
"""Print error message to stderr.
:param message: Error message to display
:type message: str
:returns: None
:rtype: None
"""
print(f"[ERROR] {message}", file=sys.stderr)
[docs]
def format_results_text(
analyzer: NPMolecularInteractionAnalyzer,
input_file: str,
summary_only: bool = False,
) -> str:
"""Format analysis results as text.
Creates a human-readable text report of analysis results,
with options for summary or detailed output.
:param analyzer: Analysis results to format
:type analyzer: MolecularInteractionAnalyzer
:param input_file: Path to the input file analyzed
:type input_file: str
:param summary_only: Whether to include only summary statistics
:type summary_only: bool
:returns: Formatted text report
:rtype: str
"""
lines = []
lines.append("HBAT Analysis Results")
lines.append("=" * 50)
lines.append(f"Input file: {input_file}")
lines.append(f"Analysis time: {time.strftime('%Y-%m-%d %H:%M:%S')}")
lines.append("")
# Statistics summary
summary = analyzer.get_summary()
lines.append("Summary:")
lines.append(f" Hydrogen bonds: {summary['hydrogen_bonds']['count']}")
lines.append(f" Halogen bonds: {summary['halogen_bonds']['count']}")
lines.append(f" π interactions: {summary['pi_interactions']['count']}")
lines.append(f" Cooperativity chains: {summary['cooperativity_chains']['count']}")
lines.append(f" Total interactions: {summary['total_interactions']}")
lines.append("")
# Bond detection statistics
if "bond_detection" in summary:
bond_stats = summary["bond_detection"]
lines.append("Bond Detection:")
lines.append(f" Total bonds detected: {bond_stats['total_bonds']}")
if bond_stats["breakdown"]:
for method, stats in bond_stats["breakdown"].items():
method_name = method.replace("_", " ").title()
lines.append(
f" {method_name}: {stats['count']} ({stats['percentage']}%)"
)
lines.append("")
if summary_only:
return "\n".join(lines)
# Detailed results
if analyzer.hydrogen_bonds:
lines.append("Hydrogen Bonds:")
lines.append("-" * 30)
for i, hb in enumerate(analyzer.hydrogen_bonds, 1):
lines.append(f"{i:3d}. {hb}")
lines.append("")
if analyzer.halogen_bonds:
lines.append("Halogen Bonds:")
lines.append("-" * 30)
for i, xb in enumerate(analyzer.halogen_bonds, 1):
lines.append(f"{i:3d}. {xb}")
lines.append("")
if analyzer.pi_interactions:
lines.append("π Interactions:")
lines.append("-" * 30)
for i, pi in enumerate(analyzer.pi_interactions, 1):
lines.append(f"{i:3d}. {pi}")
lines.append("")
if analyzer.cooperativity_chains:
lines.append("Cooperativity Chains:")
lines.append("-" * 30)
for i, chain in enumerate(analyzer.cooperativity_chains, 1):
lines.append(f"{i:3d}. {chain}")
lines.append("")
return "\n".join(lines)
[docs]
def export_to_json(
analyzer: NPMolecularInteractionAnalyzer, input_file: str, output_file: str
) -> None:
"""Export results to JSON format.
Exports complete analysis results to a structured JSON file
with metadata, statistics, and detailed interaction data.
:param analyzer: Analysis results to export
:type analyzer: MolecularInteractionAnalyzer
:param input_file: Path to the input file analyzed
:type input_file: str
:param output_file: Path to the JSON output file
:type output_file: str
:returns: None
:rtype: None
"""
import math
data: Dict[str, Any] = {
"metadata": {
"input_file": input_file,
"analysis_time": time.strftime("%Y-%m-%d %H:%M:%S"),
"hbat_version": __version__,
},
"summary": analyzer.get_summary(),
"hydrogen_bonds": [],
"halogen_bonds": [],
"pi_interactions": [],
"cooperativity_chains": [],
}
# Convert hydrogen bonds
for hb in analyzer.hydrogen_bonds:
data["hydrogen_bonds"].append(
{
"donor_residue": hb.donor_residue,
"donor_atom": hb.donor.name,
"donor_coords": hb.donor.coords.to_list(),
"hydrogen_atom": hb.hydrogen.name,
"hydrogen_coords": hb.hydrogen.coords.to_list(),
"acceptor_residue": hb.acceptor_residue,
"acceptor_atom": hb.acceptor.name,
"acceptor_coords": hb.acceptor.coords.to_list(),
"distance": round(hb.distance, 3),
"angle": round(math.degrees(hb.angle), 1),
"donor_acceptor_distance": round(hb.donor_acceptor_distance, 3),
"bond_type": hb.bond_type,
}
)
# Convert halogen bonds
for xb in analyzer.halogen_bonds:
data["halogen_bonds"].append(
{
"halogen_residue": xb.halogen_residue,
"halogen_atom": xb.halogen.name,
"halogen_coords": xb.halogen.coords.to_list(),
"acceptor_residue": xb.acceptor_residue,
"acceptor_atom": xb.acceptor.name,
"acceptor_coords": xb.acceptor.coords.to_list(),
"distance": round(xb.distance, 3),
"angle": round(math.degrees(xb.angle), 1),
"bond_type": xb.bond_type,
}
)
# Convert π interactions
for pi in analyzer.pi_interactions:
data["pi_interactions"].append(
{
"donor_residue": pi.donor_residue,
"donor_atom": pi.donor.name,
"donor_coords": pi.donor.coords.to_list(),
"hydrogen_atom": pi.hydrogen.name,
"hydrogen_coords": pi.hydrogen.coords.to_list(),
"pi_residue": pi.pi_residue,
"pi_center": pi.pi_center.to_list(),
"distance": round(pi.distance, 3),
"angle": round(math.degrees(pi.angle), 1),
}
)
# Convert cooperativity chains
for chain in analyzer.cooperativity_chains:
chain_data: Dict[str, Any] = {
"chain_length": chain.chain_length,
"chain_type": chain.chain_type,
"interactions": [],
}
for interaction in chain.interactions:
interaction_data: Dict[str, Any] = {
"interaction_type": interaction.interaction_type,
"donor_residue": interaction.get_donor_residue(),
"acceptor_residue": interaction.get_acceptor_residue(),
"distance": round(interaction.distance, 3),
"angle": round(math.degrees(interaction.angle), 1),
}
# Add type-specific fields
if hasattr(interaction, "bond_type"):
interaction_data["bond_type"] = interaction.bond_type
donor_atom = interaction.get_donor_atom()
if donor_atom:
interaction_data["donor_atom"] = donor_atom.name
acceptor_atom = interaction.get_acceptor_atom()
if acceptor_atom:
interaction_data["acceptor_atom"] = acceptor_atom.name
chain_data["interactions"].append(interaction_data)
data["cooperativity_chains"].append(chain_data)
with open(output_file, "w") as f:
json.dump(data, f, indent=2)
[docs]
def export_to_csv_files(analyzer: NPMolecularInteractionAnalyzer, base_filename: str) -> None:
"""Export results to multiple CSV files.
Creates separate CSV files for each interaction type.
:param analyzer: Analysis results to export
:type analyzer: NPMolecularInteractionAnalyzer
:param base_filename: Base filename (extension will be removed)
:type base_filename: str
:returns: None
:rtype: None
"""
import csv
import math
import os
from pathlib import Path
base_path = Path(base_filename)
base_name = base_path.stem
directory = base_path.parent
# Export hydrogen bonds
if analyzer.hydrogen_bonds:
hb_file = directory / f"{base_name}_h_bonds.csv"
with open(hb_file, 'w', newline='', encoding='utf-8') as csvfile:
writer = csv.writer(csvfile)
writer.writerow([
"Donor_Residue", "Donor_Atom", "Hydrogen_Atom",
"Acceptor_Residue", "Acceptor_Atom", "Distance_Angstrom",
"Angle_Degrees", "Donor_Acceptor_Distance_Angstrom",
"Bond_Type", "B/S_Interaction", "D-A_Properties"
])
for hb in analyzer.hydrogen_bonds:
writer.writerow([
hb.donor_residue, hb.donor.name, hb.hydrogen.name,
hb.acceptor_residue, hb.acceptor.name,
f"{hb.distance:.3f}", f"{math.degrees(hb.angle):.1f}",
f"{hb.donor_acceptor_distance:.3f}", hb.bond_type,
hb.get_backbone_sidechain_interaction(),
hb.donor_acceptor_properties
])
# Export halogen bonds
if analyzer.halogen_bonds:
xb_file = directory / f"{base_name}_x_bonds.csv"
with open(xb_file, 'w', newline='', encoding='utf-8') as csvfile:
writer = csv.writer(csvfile)
writer.writerow([
"Halogen_Residue", "Halogen_Atom", "Acceptor_Residue",
"Acceptor_Atom", "Distance_Angstrom", "Angle_Degrees",
"Bond_Type", "B/S_Interaction", "D-A_Properties"
])
for xb in analyzer.halogen_bonds:
writer.writerow([
xb.halogen_residue, xb.halogen.name,
xb.acceptor_residue, xb.acceptor.name,
f"{xb.distance:.3f}", f"{math.degrees(xb.angle):.1f}",
xb.bond_type,
xb.get_backbone_sidechain_interaction(),
xb.donor_acceptor_properties
])
# Export π interactions
if analyzer.pi_interactions:
pi_file = directory / f"{base_name}_pi_interactions.csv"
with open(pi_file, 'w', newline='', encoding='utf-8') as csvfile:
writer = csv.writer(csvfile)
writer.writerow([
"Donor_Residue", "Donor_Atom", "Hydrogen_Atom",
"Pi_Residue", "Distance_Angstrom", "Angle_Degrees",
"B/S_Interaction", "D-A_Properties"
])
for pi in analyzer.pi_interactions:
writer.writerow([
pi.donor_residue, pi.donor.name, pi.hydrogen.name,
pi.pi_residue, f"{pi.distance:.3f}",
f"{math.degrees(pi.angle):.1f}",
pi.get_backbone_sidechain_interaction(),
pi.donor_acceptor_properties
])
# Export cooperativity chains
if analyzer.cooperativity_chains:
chains_file = directory / f"{base_name}_cooperativity_chains.csv"
with open(chains_file, 'w', newline='', encoding='utf-8') as csvfile:
writer = csv.writer(csvfile)
writer.writerow(["Chain_ID", "Chain_Length", "Chain_Type", "Interactions"])
for i, chain in enumerate(analyzer.cooperativity_chains):
interactions_str = " -> ".join([
f"{interaction.get_donor_residue()}({interaction.get_donor_atom().name if interaction.get_donor_atom() else '?'})"
for interaction in chain.interactions
])
writer.writerow([
i + 1, chain.chain_length, chain.chain_type, interactions_str
])
[docs]
def export_to_json_files(analyzer: NPMolecularInteractionAnalyzer, base_filename: str, input_file: str) -> None:
"""Export results to multiple JSON files.
Creates separate JSON files for each interaction type.
:param analyzer: Analysis results to export
:type analyzer: NPMolecularInteractionAnalyzer
:param base_filename: Base filename (extension will be removed)
:type base_filename: str
:param input_file: Path to the input file analyzed
:type input_file: str
:returns: None
:rtype: None
"""
import json
import math
from pathlib import Path
base_path = Path(base_filename)
base_name = base_path.stem
directory = base_path.parent
# Export hydrogen bonds
if analyzer.hydrogen_bonds:
hb_file = directory / f"{base_name}_h_bonds.json"
data = {
"metadata": {
"input_file": input_file,
"analysis_engine": "HBAT",
"version": __version__,
"interaction_type": "Hydrogen Bonds"
},
"interactions": []
}
for hb in analyzer.hydrogen_bonds:
data["interactions"].append({
"donor_residue": hb.donor_residue,
"donor_atom": hb.donor.name,
"hydrogen_atom": hb.hydrogen.name,
"acceptor_residue": hb.acceptor_residue,
"acceptor_atom": hb.acceptor.name,
"distance_angstrom": round(hb.distance, 3),
"angle_degrees": round(math.degrees(hb.angle), 1),
"donor_acceptor_distance_angstrom": round(hb.donor_acceptor_distance, 3),
"bond_type": hb.bond_type,
"backbone_sidechain_interaction": hb.get_backbone_sidechain_interaction(),
"donor_acceptor_properties": hb.donor_acceptor_properties
})
with open(hb_file, 'w', encoding='utf-8') as jsonfile:
json.dump(data, jsonfile, indent=2, ensure_ascii=False)
# Export halogen bonds
if analyzer.halogen_bonds:
xb_file = directory / f"{base_name}_x_bonds.json"
data = {
"metadata": {
"input_file": input_file,
"analysis_engine": "HBAT",
"version": __version__,
"interaction_type": "Halogen Bonds"
},
"interactions": []
}
for xb in analyzer.halogen_bonds:
data["interactions"].append({
"halogen_residue": xb.halogen_residue,
"halogen_atom": xb.halogen.name,
"acceptor_residue": xb.acceptor_residue,
"acceptor_atom": xb.acceptor.name,
"distance_angstrom": round(xb.distance, 3),
"angle_degrees": round(math.degrees(xb.angle), 1),
"bond_type": xb.bond_type,
"backbone_sidechain_interaction": xb.get_backbone_sidechain_interaction(),
"donor_acceptor_properties": xb.donor_acceptor_properties
})
with open(xb_file, 'w', encoding='utf-8') as jsonfile:
json.dump(data, jsonfile, indent=2, ensure_ascii=False)
# Export π interactions
if analyzer.pi_interactions:
pi_file = directory / f"{base_name}_pi_interactions.json"
data = {
"metadata": {
"input_file": input_file,
"analysis_engine": "HBAT",
"version": __version__,
"interaction_type": "Pi Interactions"
},
"interactions": []
}
for pi in analyzer.pi_interactions:
data["interactions"].append({
"donor_residue": pi.donor_residue,
"donor_atom": pi.donor.name,
"hydrogen_atom": pi.hydrogen.name,
"pi_residue": pi.pi_residue,
"distance_angstrom": round(pi.distance, 3),
"angle_degrees": round(math.degrees(pi.angle), 1),
"backbone_sidechain_interaction": pi.get_backbone_sidechain_interaction(),
"donor_acceptor_properties": pi.donor_acceptor_properties
})
with open(pi_file, 'w', encoding='utf-8') as jsonfile:
json.dump(data, jsonfile, indent=2, ensure_ascii=False)
# Export cooperativity chains
if analyzer.cooperativity_chains:
chains_file = directory / f"{base_name}_cooperativity_chains.json"
data = {
"metadata": {
"input_file": input_file,
"analysis_engine": "HBAT",
"version": __version__,
"interaction_type": "Cooperativity Chains"
},
"chains": []
}
for i, chain in enumerate(analyzer.cooperativity_chains):
chain_data = {
"chain_id": i + 1,
"chain_length": chain.chain_length,
"chain_type": chain.chain_type,
"interactions": []
}
for interaction in chain.interactions:
interaction_data = {
"donor_residue": interaction.get_donor_residue(),
"acceptor_residue": interaction.get_acceptor_residue(),
"interaction_type": interaction.get_interaction_type()
}
donor_atom = interaction.get_donor_atom()
if donor_atom:
interaction_data["donor_atom"] = donor_atom.name
acceptor_atom = interaction.get_acceptor_atom()
if acceptor_atom:
interaction_data["acceptor_atom"] = acceptor_atom.name
chain_data["interactions"].append(interaction_data)
data["chains"].append(chain_data)
with open(chains_file, 'w', encoding='utf-8') as jsonfile:
json.dump(data, jsonfile, indent=2, ensure_ascii=False)
[docs]
def export_to_csv(analyzer: NPMolecularInteractionAnalyzer, output_file: str) -> None:
"""Export results to CSV format.
Exports analysis results to a CSV file with separate sections
for different interaction types.
:param analyzer: Analysis results to export
:type analyzer: MolecularInteractionAnalyzer
:param output_file: Path to the CSV output file
:type output_file: str
:returns: None
:rtype: None
"""
import csv
import math
with open(output_file, "w", newline="") as f:
writer = csv.writer(f)
# Hydrogen bonds section
if analyzer.hydrogen_bonds:
writer.writerow(["# Hydrogen Bonds"])
writer.writerow(
[
"Donor_Residue",
"Donor_Atom",
"Hydrogen_Atom",
"Acceptor_Residue",
"Acceptor_Atom",
"Distance_A",
"Angle_deg",
"DA_Distance_A",
"Bond_Type",
"D-A_Properties",
"B/S",
]
)
for hb in analyzer.hydrogen_bonds:
writer.writerow(
[
hb.donor_residue,
hb.donor.name,
hb.hydrogen.name,
hb.acceptor_residue,
hb.acceptor.name,
f"{hb.distance:.3f}",
f"{math.degrees(hb.angle):.1f}",
f"{hb.donor_acceptor_distance:.3f}",
hb.bond_type,
hb.donor_acceptor_properties,
hb.get_backbone_sidechain_interaction(),
]
)
writer.writerow([]) # Empty row
# Halogen bonds section
if analyzer.halogen_bonds:
writer.writerow(["# Halogen Bonds"])
writer.writerow(
[
"Halogen_Residue",
"Halogen_Atom",
"Acceptor_Residue",
"Acceptor_Atom",
"Distance_A",
"Angle_deg",
"Bond_Type",
"D-A_Properties",
"B/S",
]
)
for xb in analyzer.halogen_bonds:
writer.writerow(
[
xb.halogen_residue,
xb.halogen.name,
xb.acceptor_residue,
xb.acceptor.name,
f"{xb.distance:.3f}",
f"{math.degrees(xb.angle):.1f}",
xb.bond_type,
xb.donor_acceptor_properties,
xb.get_backbone_sidechain_interaction(),
]
)
writer.writerow([]) # Empty row
# π interactions section
if analyzer.pi_interactions:
writer.writerow(["# Pi Interactions"])
writer.writerow(
[
"Donor_Residue",
"Donor_Atom",
"Hydrogen_Atom",
"Pi_Residue",
"Distance_A",
"Angle_deg",
"Type",
"D-A_Properties",
"B/S",
]
)
for pi in analyzer.pi_interactions:
writer.writerow(
[
pi.donor_residue,
pi.donor.name,
pi.hydrogen.name,
pi.pi_residue,
f"{pi.distance:.3f}",
f"{math.degrees(pi.angle):.1f}",
pi.get_interaction_type_display(),
pi.donor_acceptor_properties,
pi.get_backbone_sidechain_interaction(),
]
)
writer.writerow([]) # Empty row
# Cooperativity chains section
if analyzer.cooperativity_chains:
writer.writerow(["# Cooperativity Chains"])
writer.writerow(["Chain_ID", "Chain_Length", "Chain_Type", "Interactions"])
for i, chain in enumerate(analyzer.cooperativity_chains):
interactions_str = " -> ".join(
[
f"{interaction.get_donor_residue()}({interaction.get_donor_atom().name if interaction.get_donor_atom() else '?'})"
for interaction in chain.interactions
]
)
writer.writerow(
[i + 1, chain.chain_length, chain.chain_type, interactions_str]
)
[docs]
def run_analysis(args: argparse.Namespace) -> int:
"""Run the analysis with given arguments.
Performs the complete analysis workflow including parameter loading,
analysis execution, and result output based on command-line arguments.
:param args: Parsed command-line arguments
:type args: argparse.Namespace
:returns: Exit code (0 for success, non-zero for failure)
:rtype: int
"""
# Validate input
if not validate_input_file(args.input):
return 1
verbose = args.verbose and not args.quiet
try:
# Load parameters
parameters = load_parameters_from_args(args)
print_progress(f"Starting analysis of {args.input}", verbose)
print_progress(f"Analysis mode: {parameters.analysis_mode}", verbose)
# Create analyzer
analyzer = NPMolecularInteractionAnalyzer(parameters)
# Set up progress tracking (show unless quiet mode)
progress_bar = None
if not args.quiet:
progress_bar = ProgressBar()
def cli_progress_callback(message: str) -> None:
"""Progress callback for CLI updates."""
# Use progress bar for clean display
if progress_bar:
# Extract percentage if present in message
if "%" in message:
try:
# Split on space to get the percentage part
parts = message.split()
percent_part = None
for part in parts:
if "%" in part:
percent_part = part.replace("%", "")
break
if percent_part and percent_part.isdigit():
progress = int(percent_part)
# Get the step name (everything before the percentage)
step_name = message.split(f" {percent_part}%")[0]
progress_bar.update(step_name, progress)
else:
progress_bar.update(message)
except (ValueError, IndexError):
progress_bar.update(message)
else:
progress_bar.update(message)
else:
# Fallback if no progress bar
print_progress(message, verbose)
analyzer.progress_callback = cli_progress_callback
# Run analysis
start_time = time.time()
success = analyzer.analyze_file(args.input)
analysis_time = time.time() - start_time
if not success:
if progress_bar:
progress_bar.finish("Analysis failed")
print_error("Analysis failed")
return 1
if progress_bar:
progress_bar.finish(f"Analysis completed in {analysis_time:.2f} seconds")
else:
print_progress(
f"Analysis completed in {analysis_time:.2f} seconds", verbose
)
# Get results
summary = analyzer.get_summary()
if not args.quiet:
print(
f"Found {summary['hydrogen_bonds']['count']} hydrogen bonds, "
f"{summary['halogen_bonds']['count']} halogen bonds, "
f"{summary['pi_interactions']['count']} π interactions, "
f"{summary['cooperativity_chains']['count']} cooperativity chains"
)
# Output results
if args.output:
try:
output_format = detect_output_format(args.output)
if output_format == 'text':
print_progress(f"Writing results to {args.output}", verbose)
with open(args.output, "w") as f:
f.write(format_results_text(analyzer, args.input, args.summary_only))
elif output_format == 'csv':
print_progress(f"Exporting to CSV: {args.output}", verbose)
export_to_csv(analyzer, args.output)
elif output_format == 'json':
print_progress(f"Exporting to JSON: {args.output}", verbose)
export_to_json(analyzer, args.input, args.output)
except ValueError as e:
print_error(str(e))
return 1
if args.json:
print_progress(f"Exporting to multiple JSON files: {args.json}", verbose)
export_to_json_files(analyzer, args.json, args.input)
if args.csv:
print_progress(f"Exporting to multiple CSV files: {args.csv}", verbose)
export_to_csv_files(analyzer, args.csv)
# Print to stdout if no output files specified
if not any([args.output, args.json, args.csv]) and not args.quiet:
print("\n" + format_results_text(analyzer, args.input, args.summary_only))
return 0
except KeyboardInterrupt:
print_error("Analysis interrupted by user")
return 130
except Exception as e:
print_error(f"Analysis failed: {e}")
if verbose:
import traceback
traceback.print_exc()
return 1
[docs]
def main() -> int:
"""Main CLI entry point for HBAT molecular interaction analysis.
Parses command-line arguments and dispatches to appropriate functionality.
Supports comprehensive analysis of molecular interactions including:
- Hydrogen bonds (classical N-H···O, O-H···O)
- Weak hydrogen bonds (C-H···O interactions)
- Halogen bonds (C-X···A with default 150° angle cutoff)
- π interactions with multiple subtypes:
• Hydrogen-π: C-H···π, N-H···π, O-H···π, S-H···π
• Halogen-π: C-Cl···π, C-Br···π, C-I···π
- Cooperativity chains and interaction networks
Includes built-in parameter presets and PDB structure fixing capabilities.
:returns: Exit code (0 for success, non-zero for failure)
:rtype: int
"""
# Initialize HBAT environment first
try:
from ..core.app_config import initialize_hbat_environment
initialize_hbat_environment(
verbose=False
) # We'll handle verbosity based on args
except ImportError:
pass # Continue without app config if import fails
parser = create_parser()
args = parser.parse_args()
# Show HBAT environment info if verbose
if hasattr(args, "verbose") and args.verbose:
try:
from ..core.app_config import get_hbat_config
config = get_hbat_config()
info = config.get_info()
print(f"📁 HBAT data directory: {info['hbat_directory']}")
if info["ccd_files_present"]:
print(f"✅ CCD data available")
else:
print(f"⚠️ CCD data will be downloaded as needed")
except ImportError:
pass
# Handle preset listing first
if hasattr(args, "list_presets") and args.list_presets:
list_available_presets()
return 0
# Validate that input file is provided for analysis
if not args.input:
print_error("Input PDB file is required for analysis")
print_error(
"Use --help for usage information or --list-presets to see available presets"
)
return 1
# Handle conflicting options
if args.verbose and args.quiet:
print_error("Cannot use both --verbose and --quiet options")
return 1
return run_analysis(args)
if __name__ == "__main__":
sys.exit(main())