1 #!/usr/bin/perl -w 2 # 3 # File: ModifySDFilesDataFields.pl 4 # Author: Manish Sud <msud@san.rr.com> 5 # 6 # Copyright (C) 2025 Manish Sud. All rights reserved. 7 # 8 # This file is part of MayaChemTools. 9 # 10 # MayaChemTools is free software; you can redistribute it and/or modify it under 11 # the terms of the GNU Lesser General Public License as published by the Free 12 # Software Foundation; either version 3 of the License, or (at your option) any 13 # later version. 14 # 15 # MayaChemTools is distributed in the hope that it will be useful, but without 16 # any warranty; without even the implied warranty of merchantability of fitness 17 # for a particular purpose. See the GNU Lesser General Public License for more 18 # details. 19 # 20 # You should have received a copy of the GNU Lesser General Public License 21 # along with MayaChemTools; if not, see <http://www.gnu.org/licenses/> or 22 # write to the Free Software Foundation Inc., 59 Temple Place, Suite 330, 23 # Boston, MA, 02111-1307, USA. 24 # 25 26 use strict; 27 use FindBin; use lib "$FindBin::Bin/../lib"; 28 use Getopt::Long; 29 use File::Basename; 30 use Text::ParseWords; 31 use Benchmark; 32 use FileUtil; 33 use SDFileUtil; 34 use TextUtil; 35 36 my($ScriptName, %Options, $StartTime, $EndTime, $TotalTime); 37 38 # Autoflush STDOUT 39 $| = 1; 40 41 # Starting message... 42 $ScriptName = basename($0); 43 print "\n$ScriptName: Starting...\n\n"; 44 $StartTime = new Benchmark; 45 46 # Get the options and setup script... 47 SetupScriptUsage(); 48 if ($Options{help} || @ARGV < 1) { 49 die GetUsageFromPod("$FindBin::Bin/$ScriptName"); 50 } 51 52 my(@SDFilesList); 53 @SDFilesList = ExpandFileNames(\@ARGV, "sdf sd"); 54 55 # Process options... 56 print "Processing options...\n"; 57 my(%OptionsInfo); 58 ProcessOptions(); 59 60 print "Checking input SD file(s)...\n"; 61 my(%SDFilesInfo); 62 RetrieveSDFilesInfo(); 63 64 # Generate output files... 65 my($FileIndex); 66 if (@SDFilesList > 1) { 67 print "\nProcessing SD files...\n"; 68 } 69 for $FileIndex (0 .. $#SDFilesList) { 70 if ($SDFilesInfo{FileOkay}[$FileIndex]) { 71 print "\nProcessing file $SDFilesList[$FileIndex]...\n"; 72 ModifySDFile($FileIndex); 73 } 74 } 75 print "\n$ScriptName:Done...\n\n"; 76 77 $EndTime = new Benchmark; 78 $TotalTime = timediff ($EndTime, $StartTime); 79 print "Total time: ", timestr($TotalTime), "\n"; 80 81 ############################################################################### 82 83 # Modify SD file data fields.... 84 sub ModifySDFile { 85 my($Index) = @_; 86 my($SDFile, $NewSDFile); 87 88 $SDFile = $SDFilesList[$Index]; 89 $NewSDFile = $SDFilesInfo{OutFile}[$Index]; 90 91 print "Generating new SD file $NewSDFile...\n"; 92 open NEWSDFILE, ">$NewSDFile" or die "Error: Couldn't open $NewSDFile: $! \n"; 93 open SDFILE, "$SDFile" or die "Error: Can't open $SDFile: $! \n"; 94 95 my($CmpdCount, $CmpdString, $CmpdData, $MolName, $OldSDField, $NewSDField, $CommonSDField, $Label, $Value, $FieldValues, $MolNameDataField, $URLCmpdIdFieldName, @CmpdLines, %DataFieldAndValues, @DataFieldLabels); 96 $CmpdCount = 0; 97 98 COMPOUND: while ($CmpdString = ReadCmpdString(\*SDFILE)) { 99 $CmpdCount++; 100 @CmpdLines = split "\n", $CmpdString; 101 if ($OptionsInfo{UseDataFieldForMolName} || $OptionsInfo{ModifyDataFields}) { 102 %DataFieldAndValues = GetCmpdDataHeaderLabelsAndValues(\@CmpdLines); 103 } 104 if ($OptionsInfo{ModifyMolName}) { 105 if ($OptionsInfo{AlwaysReplaceMolName} || !IsNotEmpty($CmpdLines[0])) { 106 $MolNameDataField = $OptionsInfo{MolNameDataField}; 107 if ($OptionsInfo{UseDataFieldForMolName} && exists($DataFieldAndValues{$MolNameDataField})) { 108 $MolName = $DataFieldAndValues{$MolNameDataField}; 109 if (length($MolName) > 80) { 110 $MolName = substr($MolName, 0, 80); 111 } 112 } 113 else { 114 $MolName = "$OptionsInfo{MolNamePrefix}${CmpdCount}"; 115 } 116 $CmpdLines[0] = $MolName; 117 $CmpdString = join "\n", @CmpdLines; 118 } 119 } 120 if (!$OptionsInfo{ModifyDataFields}) { 121 # Just write the data and get the next compound... 122 print NEWSDFILE "$CmpdString\n"; 123 next COMPOUND; 124 } 125 # Write out the structure data now and handle the old data fields later... 126 ($CmpdData) = split /\n>/, $CmpdString; 127 print NEWSDFILE "$CmpdData\n"; 128 129 # Modify specified data fields... 130 for $NewSDField (sort keys %{$OptionsInfo{SpecifiedNewToOldSDFieldMap}}) { 131 $FieldValues = ""; 132 for $OldSDField (@{$OptionsInfo{SpecifiedNewToOldSDFieldMap}{$NewSDField}}) { 133 if (exists($DataFieldAndValues{$OldSDField}) && length($DataFieldAndValues{$OldSDField})) { 134 $Value = $DataFieldAndValues{$OldSDField}; 135 $FieldValues .= ($FieldValues) ? "\n$Value" : $Value; 136 } 137 } 138 print NEWSDFILE "> <$NewSDField>\n$FieldValues\n\n"; 139 } 140 # Add specified common fields... 141 for $CommonSDField (sort keys %{$OptionsInfo{SpecifiedCommonFieldMap}}) { 142 $Value = $OptionsInfo{SpecifiedCommonFieldMap}{$CommonSDField}; 143 print NEWSDFILE "> <$CommonSDField>\n$Value\n\n"; 144 } 145 if ($OptionsInfo{CreateDataFieldURL}) { 146 $Value = ""; 147 $URLCmpdIdFieldName = $OptionsInfo{URLCmpdIdFieldName}; 148 if (exists($DataFieldAndValues{$URLCmpdIdFieldName}) && length($DataFieldAndValues{$URLCmpdIdFieldName})) { 149 $Value = $DataFieldAndValues{$URLCmpdIdFieldName}; 150 $Value = "$OptionsInfo{URLCGIScriptName}?$OptionsInfo{URLParamName}=${Value}"; 151 } 152 print NEWSDFILE "> <$OptionsInfo{URLDataFieldLabel}>\n$Value\n\n"; 153 } 154 155 # Handle old data fields and write 'em in the same order as they appear in the input 156 # files... 157 if ($OptionsInfo{KeepAllOldDataFields} || $OptionsInfo{KeepUnMappedOldDataFields}) { 158 my($KeepLabel); 159 @DataFieldLabels = GetCmpdDataHeaderLabels(\@CmpdLines); 160 LABEL: for $Label (@DataFieldLabels) { 161 $KeepLabel = $OptionsInfo{KeepAllOldDataFields} ? 1 : ( exists($OptionsInfo{SpecifiedOldToNewSDFieldMap}{$Label}) ? 0 : 1 ); 162 if (!$KeepLabel) { 163 next LABEL; 164 } 165 $Value = $DataFieldAndValues{$Label}; 166 print NEWSDFILE "> <$Label>\n$Value\n\n"; 167 } 168 } 169 170 print NEWSDFILE "\$\$\$\$\n"; 171 } 172 close NEWSDFILE; 173 close SDFILE; 174 } 175 176 # Process option values... 177 sub ProcessOptions { 178 %OptionsInfo = (); 179 180 $OptionsInfo{Mode} = $Options{mode}; 181 182 $OptionsInfo{ModifyMolName} = 1; $OptionsInfo{ModifyDataFields} = 0; 183 if ($Options{mode} =~ /^both$/i) { 184 $OptionsInfo{ModifyMolName} = 1; $OptionsInfo{ModifyDataFields} = 1; 185 } 186 elsif ($Options{mode} =~ /^datafields$/i) { 187 $OptionsInfo{ModifyMolName} = 0; $OptionsInfo{ModifyDataFields} = 1; 188 } 189 190 $OptionsInfo{KeepOldDataFields} = $Options{keepolddatafields}; 191 $OptionsInfo{KeepAllOldDataFields} = ($Options{keepolddatafields} =~ /^all$/i) ? 1 : 0; 192 $OptionsInfo{KeepUnMappedOldDataFields} = ($Options{keepolddatafields} =~ /^unmappedonly$/i) ? 1 : 0; 193 194 $OptionsInfo{MolNameMode} = $Options{molnamemode}; 195 $OptionsInfo{UseDataFieldForMolName} = ($Options{molnamemode} =~ /^datafield$/i) ? 1 : 0; 196 197 $OptionsInfo{MolName} = $Options{molname}; 198 $OptionsInfo{MolNameDataField} = ""; $OptionsInfo{MolNamePrefix} = "Cmpd"; 199 if ($Options{molname}) { 200 if ($OptionsInfo{UseDataFieldForMolName}) { 201 $OptionsInfo{MolNameDataField} = $Options{molname}; 202 } 203 else { 204 $OptionsInfo{MolNamePrefix} = $Options{molname}; 205 } 206 } 207 208 $OptionsInfo{MolNameReplace} = $Options{molnamereplace}; 209 $OptionsInfo{AlwaysReplaceMolName} = ($Options{molnamereplace} =~ /^always$/i) ? 1 : 0; 210 211 if ($Options{datafieldsmap} && $Options{datafieldsmapfile}) { 212 die "Error: Both \"--datafieldsmap\" and \"--datafieldsmapfile\" options specified: only one is allowed at a time\n"; 213 } 214 215 $OptionsInfo{DataFieldsMap} = $Options{datafieldsmap} ? $Options{datafieldsmap} : ''; 216 $OptionsInfo{DataFieldsMapFile} = $Options{datafieldsmapfile} ? $Options{datafieldsmapfile} : ''; 217 218 my($SpecifiedDataFieldMap); 219 220 %{$OptionsInfo{SpecifiedNewToOldSDFieldMap}} = (); 221 %{$OptionsInfo{SpecifiedOldToNewSDFieldMap}} = (); 222 223 $SpecifiedDataFieldMap = ""; 224 if ($Options{datafieldsmap}) { 225 $SpecifiedDataFieldMap = $Options{datafieldsmap}; 226 } 227 elsif ($Options{datafieldsmapfile}) { 228 my($Line, @LineWords); 229 open DATAFIELDSFILE, "$Options{datafieldsmapfile}" or die "Couldn't open $Options{datafieldsmapfile}: $! \n"; 230 while ($Line = GetTextLine(\*DATAFIELDSFILE)) { 231 @LineWords = quotewords(";", 0, $Line); 232 $SpecifiedDataFieldMap .= JoinWords(\@LineWords, ";", 0); 233 } 234 close DATAFIELDSFILE; 235 } 236 237 if ($SpecifiedDataFieldMap) { 238 my($DataFieldMap, $DataField, $NewSDField, @OldSDFields, @DataFieldMapSplit, @DataFieldsSplit, $FirstField); 239 @DataFieldMapSplit = split ";", $SpecifiedDataFieldMap; 240 for $DataFieldMap (@DataFieldMapSplit) { 241 @DataFieldsSplit = split ",", $DataFieldMap; 242 if (@DataFieldsSplit == 1) { 243 die "Error: Invalid number of comma delimited values, ", scalar(@DataFieldsSplit), ", specified, @DataFieldsSplit, using \"--datafieldsmap or --datafieldsmapfile\" option: it must contain more than one value.\n"; 244 } 245 $FirstField = 1; 246 @OldSDFields = (); 247 for $DataField (@DataFieldsSplit) { 248 if (!(defined($DataField) && length($DataField))) { 249 die "Error: One of the comma delimited values, \"", join(",", @DataFieldsSplit), "\", specified using \"--datafieldsmap or --datafieldsmapfile\" option is empty.\n"; 250 } 251 if ($FirstField) { 252 $FirstField = 0; 253 $NewSDField = $DataField; 254 } 255 else { 256 push @OldSDFields, $DataField; 257 } 258 } 259 # Make sure a datafield is only specified once... 260 if (exists $OptionsInfo{SpecifiedNewToOldSDFieldMap}{$NewSDField}) { 261 die "Error: New data field, $NewSDField, specified more than once using \"--datafieldsmap or --datafieldsmapfile\" option.\n"; 262 } 263 @{$OptionsInfo{SpecifiedNewToOldSDFieldMap}{$NewSDField}} = (); 264 push @{$OptionsInfo{SpecifiedNewToOldSDFieldMap}{$NewSDField}}, @OldSDFields; 265 for $DataField (@OldSDFields) { 266 if (exists $OptionsInfo{SpecifiedOldToNewSDFieldMap}{$DataField} ) { 267 die "Error: SD field, $DataField, specified more than once using \"--datafieldsmap or --datafieldsmapfile\" option.\n"; 268 } 269 else { 270 $OptionsInfo{SpecifiedOldToNewSDFieldMap}{$DataField} = $NewSDField; 271 } 272 } 273 274 } 275 } 276 277 $OptionsInfo{DataFieldsCommon} = $Options{datafieldscommon} ? $Options{datafieldscommon} : ''; 278 %{$OptionsInfo{SpecifiedCommonFieldMap}} = (); 279 280 if ($Options{datafieldscommon}) { 281 my($DataFieldName, $DataFieldValue, $Index, @CommonDataFieldsSplit); 282 @CommonDataFieldsSplit = split ",", $Options{datafieldscommon}; 283 if (@CommonDataFieldsSplit % 2) { 284 die "Error: Invalid number of comma delimited values, ", scalar(@CommonDataFieldsSplit), ", specified \"", join(",", @CommonDataFieldsSplit), "\" using \"--datafieldscommon\" option: it must contain even number of values.\n"; 285 } 286 for ($Index = 0; $Index < @CommonDataFieldsSplit; $Index += 2) { 287 $DataFieldName = $CommonDataFieldsSplit[$Index]; 288 $DataFieldValue = $CommonDataFieldsSplit[$Index + 1]; 289 if (exists $OptionsInfo{SpecifiedCommonFieldMap}{$DataFieldName}) { 290 die "Error: Common data field, $DataFieldName, specified more than once using \"--datafieldscommon\" option.\n"; 291 } 292 if (exists($OptionsInfo{SpecifiedNewToOldSDFieldMap}{$DataFieldName}) || exists($OptionsInfo{SpecifiedOldToNewSDFieldMap}{$DataFieldName})) { 293 die "Error: Common data field, $DataFieldName, specified using \"--datafieldscommon\" option cannot be specified in \"--datafieldsmap or --datafieldsmapfile\" option.\n"; 294 } 295 $OptionsInfo{SpecifiedCommonFieldMap}{$DataFieldName} = $DataFieldValue; 296 } 297 } 298 299 $OptionsInfo{DataFieldURL} = $Options{datafieldurl} ? $Options{datafieldurl} : ''; 300 $OptionsInfo{CreateDataFieldURL} = (exists($Options{datafieldurl}) && length($Options{datafieldurl}) ) ? 1 : 0; 301 302 $OptionsInfo{URLDataFieldLabel} = ""; $OptionsInfo{URLCGIScriptName} = ""; 303 $OptionsInfo{URLParamName} = ""; $OptionsInfo{URLCmpdIdFieldName} = ""; 304 305 if ($OptionsInfo{CreateDataFieldURL}) { 306 my(@DataFieldURLSplit, $Value); 307 @DataFieldURLSplit = split ",", $Options{datafieldurl}; 308 if (@DataFieldURLSplit != 4) { 309 die "Error: Invalid number of values, ", scalar(@DataFieldURLSplit), ", specified using \"--datafieldURL\" option: it must contain 4 values.\n"; 310 } 311 for $Value (@DataFieldURLSplit) { 312 if (!IsNotEmpty($Value)) { 313 die "Error: One of the values, $Options{datafieldurl}, specified using \"--datafieldURL\" option is empty.\n"; 314 } 315 } 316 $OptionsInfo{URLDataFieldLabel} = $DataFieldURLSplit[0]; 317 $OptionsInfo{URLCGIScriptName} = $DataFieldURLSplit[1]; 318 $OptionsInfo{URLParamName} = $DataFieldURLSplit[2]; 319 $OptionsInfo{URLCmpdIdFieldName} = $DataFieldURLSplit[3]; 320 } 321 322 } 323 324 # Retrieve information about input SD files... 325 sub RetrieveSDFilesInfo { 326 my($Index, $SDFile, $FileDir, $FileName, $FileExt, $OutFileRoot, $OutFile, $DataFieldName); 327 328 %SDFilesInfo = (); 329 @{$SDFilesInfo{FileOkay}} = (); 330 @{$SDFilesInfo{OutFile}} = (); 331 332 FILELIST: for $Index (0 .. $#SDFilesList) { 333 $SDFile = $SDFilesList[$Index]; 334 335 $SDFilesInfo{FileOkay}[$Index] = 0; 336 $SDFilesInfo{OutFile}[$Index] = ''; 337 338 if (!(-e $SDFile)) { 339 warn "Warning: Ignoring file $SDFile: It doesn't exist\n"; 340 next FILELIST; 341 } 342 if (!CheckFileType($SDFile, "sd sdf")) { 343 warn "Warning: Ignoring file $SDFile: It's not a SD file\n"; 344 next FILELIST; 345 } 346 $FileDir = ""; $FileName = ""; $FileExt = ""; 347 ($FileDir, $FileName, $FileExt) = ParseFileName($SDFile); 348 if ($Options{root} && (@SDFilesList == 1)) { 349 my ($RootFileDir, $RootFileName, $RootFileExt) = ParseFileName($Options{root}); 350 if ($RootFileName && $RootFileExt) { 351 $FileName = $RootFileName; 352 } 353 else { 354 $FileName = $Options{root}; 355 } 356 $OutFileRoot = $FileName; 357 } 358 else { 359 $OutFileRoot = $FileName . "ModifiedDataFields"; 360 } 361 362 $OutFile = $OutFileRoot . ".$FileExt"; 363 if (lc($OutFile) eq lc($SDFile)) { 364 warn "Warning: Ignoring file $SDFile:Output file name, $OutFile, is same as input SD file name, $SDFile\n"; 365 next FILELIST; 366 } 367 if (!$Options{overwrite}) { 368 if (-e $OutFile) { 369 warn "Warning: Ignoring file $SDFile: The file $OutFile already exists\n"; 370 next FILELIST; 371 } 372 } 373 374 $SDFilesInfo{FileOkay}[$Index] = 1; 375 $SDFilesInfo{OutFile}[$Index] = $OutFile; 376 } 377 } 378 379 # Setup script usage and retrieve command line arguments specified using various options... 380 sub SetupScriptUsage { 381 382 # Retrieve all the options... 383 %Options = (); 384 $Options{detail} = 1; 385 $Options{keepolddatafields} = "none"; 386 $Options{mode} = "molname"; 387 $Options{molnamemode} = "labelprefix"; 388 $Options{molnamereplace} = "empty"; 389 390 if (!GetOptions(\%Options, "detail|d=i", "datafieldscommon=s", "datafieldsmap=s", "datafieldsmapfile=s", "datafieldurl=s", "help|h", "keepolddatafields|k=s", "mode|m=s", "molname=s", "molnamemode=s", "molnamereplace=s", "overwrite|o", "root|r=s", "workingdir|w=s")) { 391 die "\nTo get a list of valid options and their values, use \"$ScriptName -h\" or\n\"perl -S $ScriptName -h\" command and try again...\n"; 392 } 393 if ($Options{workingdir}) { 394 if (! -d $Options{workingdir}) { 395 die "Error: The value specified, $Options{workingdir}, for option \"-w --workingdir\" is not a directory name.\n"; 396 } 397 chdir $Options{workingdir} or die "Error: Couldn't chdir $Options{workingdir}: $! \n"; 398 } 399 if ($Options{keepolddatafields} !~ /^(all|unmappedonly|none)$/i) { 400 die "Error: The value specified, $Options{keepolddatafields}, for option \"-k --keepolddatafields\" is not valid. Allowed values: all, unmappedonly, or none\n"; 401 } 402 if ($Options{mode} !~ /^(molname|datafields|both)$/i) { 403 die "Error: The value specified, $Options{mode}, for option \"-m --mode\" is not valid. Allowed values: molname, datafields, or both\n"; 404 } 405 if ($Options{molnamemode} !~ /^(datafield|labelprefix)$/i) { 406 die "Error: The value specified, $Options{molnamemode}, for option \"--molnamemode\" is not valid. Allowed values: datafield or labelprefix\n"; 407 } 408 if ($Options{molnamereplace} !~ /^(always|empty)$/i) { 409 die "Error: The value specified, $Options{molnamereplace}, for option \"--molnamereplace\" is not valid. Allowed values: always or empty\n"; 410 } 411 if (!IsPositiveInteger($Options{detail})) { 412 die "Error: The value specified, $Options{detail}, for option \"-d --detail\" is not valid. Allowed values: > 0\n"; 413 } 414 } 415