-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
22 changed files
with
1,235 additions
and
377 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,222 @@ | ||
// Released under MIT License. | ||
// Copyright (c) 2023 Ladislav Bartos | ||
|
||
//! Implementation of a command line argument parser. | ||
use std::path::Path; | ||
|
||
use clap::Parser; | ||
use groan_rs::files::FileType; | ||
|
||
use crate::errors::RunError; | ||
|
||
// Center Gromacs trajectory or structure file. | ||
#[derive(Parser, Debug)] | ||
#[command( | ||
author, | ||
version, | ||
about, | ||
long_about = "Center your chosen group within a Gromacs trajectory or structure file effortlessly using the Bai & Breen algorithm.\n | ||
With `gcenter`, you can accurately center atom groups, even when they span multiple molecules that may extend beyond the box boundaries. | ||
`gcenter` does not employ connectivity information, so it doesn't require a tpr file as input. | ||
Be aware that `gcenter` exclusively supports orthogonal simulation boxes." | ||
)] | ||
pub struct Args { | ||
#[arg( | ||
short = 'c', | ||
long = "structure", | ||
help = "Input structure file", | ||
long_help = "Path to a gro or pdb file containing the system structure. If a trajectory is also provided, the coordinates from the structure file will be ignored.", | ||
value_parser = validate_structure_type, | ||
)] | ||
pub structure: String, | ||
|
||
#[arg( | ||
short = 'f', | ||
long = "trajectory", | ||
help = "Input trajectory file(s)", | ||
long_help = "Path to xtc or trr file(s) containing the trajectory or trajectories to be manipulated. | ||
If not provided, the centering operation will use the structure file itself. | ||
Multiple files separated by whitespace can be provided. These will be concatenated into one output file. | ||
When joining trajectories, the last frame of each trajectory and the first frame of the following trajectory are checked for matching simulation steps. | ||
If the simulation steps coincide, only the first of these frames is centered and written to output.", | ||
num_args = 0.., | ||
value_parser = validate_trajectory_type, | ||
)] | ||
pub trajectories: Vec<String>, | ||
|
||
#[arg( | ||
short = 'n', | ||
long = "index", | ||
help = "Input index file [default: index.ndx]", | ||
long_help = "Path to an ndx file containing groups associated with the system.\n\n[default: index.ndx]" | ||
)] | ||
pub index: Option<String>, | ||
|
||
#[arg( | ||
short = 'o', | ||
long = "output", | ||
help = "Output file name", | ||
long_help = "Name of the output file, which can be in gro, pdb (if no trajectory is provided), xtc, or trr format." | ||
)] | ||
pub output: String, | ||
|
||
#[arg( | ||
short = 'r', | ||
long = "reference", | ||
help = "Group to center", | ||
default_value = "Protein", | ||
long_help = "Specify the group to be centered. Use VMD-like 'groan selection language' to define the group. This language also supports ndx group names." | ||
)] | ||
pub reference: String, | ||
|
||
#[arg( | ||
short = 'b', | ||
long = "begin", | ||
help = "Time of the first frame to read (in ps) [default: 0.0]", | ||
requires = "trajectories", | ||
long_help = "Time of the first frame to read from the trajectory (in ps). All previous frames will be skipped. This option is only applicable when trajectory file(s) is/are provided.\n\n[default: 0.0]" | ||
)] | ||
pub start_time: Option<f32>, | ||
|
||
#[arg( | ||
short = 'e', | ||
long = "end", | ||
help = "Time of the last frame to read (in ps) [default: NaN]", | ||
requires = "trajectories", | ||
long_help = "Time of the last frame to read from the trajectory (in ps). All following frames will be skipped. This option is only applicable when trajectory file(s) is/are provided.\n\n[default: NaN]" | ||
)] | ||
pub end_time: Option<f32>, | ||
|
||
#[arg( | ||
short = 's', | ||
long = "step", | ||
help = "Write every <STEP>th frame", | ||
default_value_t = 1, | ||
requires = "trajectories", | ||
long_help = "Center and write only every <STEP>th frame of the trajectory to the output file. This option is only applicable when a SINGLE trajectory file is provided." | ||
)] | ||
pub step: usize, | ||
|
||
#[arg( | ||
short = 'x', | ||
action, | ||
help = "Center in the x dimension", | ||
default_value_t = false, | ||
long_help = "Perform centering operation in the x-dimension. This can be combined with other dimensions. If no dimensions are selected, it defaults to '-xyz'." | ||
)] | ||
pub xdimension: bool, | ||
|
||
#[arg( | ||
short = 'y', | ||
action, | ||
help = "Center in the y dimension", | ||
default_value_t = false, | ||
long_help = "Perform centering operation in the y-dimension. This can be combined with other dimensions. If no dimensions are selected, it defaults to '-xyz'." | ||
)] | ||
pub ydimension: bool, | ||
|
||
#[arg( | ||
short = 'z', | ||
action, | ||
help = "Center in the z dimension", | ||
default_value_t = false, | ||
long_help = "Perform centering operation in the z-dimension. This can be combined with other dimensions. If no dimensions are selected, it defaults to '-xyz'." | ||
)] | ||
pub zdimension: bool, | ||
|
||
#[arg( | ||
long = "silent", | ||
action, | ||
help = "Suppress standard output", | ||
default_value_t = false, | ||
long_help = "Suppress all standard output generated by the 'gcenter' tool, except for error messages written to stderr." | ||
)] | ||
pub silent: bool, | ||
|
||
#[arg( | ||
long = "overwrite", | ||
action, | ||
help = "Overwrite existing files with the same name", | ||
default_value_t = false, | ||
long_help = "Enable this option to overwrite existing files with the same name as the output file. No backup copies will be created." | ||
)] | ||
pub overwrite: bool, | ||
} | ||
|
||
/// Validate that the structure is gro or pdb file. | ||
fn validate_structure_type(s: &str) -> Result<String, String> { | ||
match FileType::from_name(s) { | ||
FileType::GRO | FileType::PDB => Ok(s.to_owned()), | ||
_ => Err(String::from("unsupported file extension")), | ||
} | ||
} | ||
|
||
/// Validate that the trajectories are xtc or trr files. | ||
/// Validate that no trajectory is provided multiple times. | ||
fn validate_trajectory_type(s: &str) -> Result<String, String> { | ||
match FileType::from_name(s) { | ||
FileType::XTC | FileType::TRR => Ok(s.to_owned()), | ||
_ => Err(String::from("unsupported file extension")), | ||
} | ||
} | ||
|
||
/// Perform various sanity checks: | ||
/// a) Check that the input and output files are not identical. | ||
/// This protects the user from accidentaly overwriting their data. | ||
/// b) Check that the output file has the correct file extension. | ||
fn sanity_check_inputs(args: &Args) -> Result<(), RunError> { | ||
// check that the input structure exists | ||
if !Path::new(&args.structure).exists() { | ||
return Err(RunError::InputStructureNotFound(args.structure.to_string())); | ||
} | ||
|
||
// check for input-output matches | ||
if args.trajectories.is_empty() { | ||
if args.structure == args.output { | ||
return Err(RunError::IOMatch(args.structure.to_string())); | ||
} | ||
} else { | ||
for (t, traj) in args.trajectories.iter().enumerate() { | ||
// check that the trajectory exists | ||
if !Path::new(traj).exists() { | ||
return Err(RunError::InputTrajectoryNotFound(traj.to_string())); | ||
} | ||
|
||
// check that the trajectory does not match the output | ||
if traj.as_str() == args.output { | ||
return Err(RunError::IOMatch(traj.to_string())); | ||
} | ||
|
||
// check that no other trajectory file matches this one | ||
for traj2 in args.trajectories.iter().skip(t + 1) { | ||
if traj == traj2 { | ||
return Err(RunError::IdenticalInputFiles( | ||
traj.to_owned(), | ||
traj2.to_owned(), | ||
)); | ||
} | ||
} | ||
} | ||
} | ||
|
||
if args.step != 1 && args.trajectories.len() > 1 { | ||
return Err(RunError::StepJoinUnsupported(args.step)); | ||
} | ||
|
||
// check the extension of the output file | ||
let output_type = FileType::from_name(&args.output); | ||
match (args.trajectories.is_empty(), output_type) { | ||
(true, FileType::GRO | FileType::PDB) => Ok(()), | ||
(true, _) => Err(RunError::OutputUnsupported(args.output.clone())), | ||
(false, FileType::XTC | FileType::TRR) => Ok(()), | ||
(false, _) => Err(RunError::OutputUnsupported(args.output.clone())), | ||
} | ||
} | ||
|
||
pub fn parse() -> Result<Args, Box<dyn std::error::Error>> { | ||
let args = Args::parse(); | ||
sanity_check_inputs(&args)?; | ||
|
||
Ok(args) | ||
} |
Oops, something went wrong.