rdworks 0.38.1__py3-none-any.whl → 0.39.1__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
rdworks/__init__.py CHANGED
@@ -1,4 +1,4 @@
1
- __version__ = '0.38.1'
1
+ __version__ = '0.39.1'
2
2
 
3
3
  from rdworks.conf import Conf
4
4
  from rdworks.mol import Mol
rdworks/conf.py CHANGED
@@ -39,11 +39,20 @@ class Conf:
39
39
  """
40
40
  assert isinstance(molecule, str | Chem.Mol) or molecule is None
41
41
 
42
- self.rdmol = None # must contain one and only one rdkit conformer
43
42
  self.name = name
43
+ self.rdmol = None # must contain one and only one rdkit conformer
44
44
  self.natoms = 0
45
- self.charge = 0 # molecular charge
46
- self.spin = 1 # molecular spin multiplicity for ASE
45
+ self.charge = 0 # molecular formal charge
46
+ self.spin = 1 # molecular spin multiplicity for ASE
47
+ # Molecular spin multiplicity describes the number of possible orientations of
48
+ # spin angular momentum for a given molecule, essentially indicating the total
49
+ # number of unpaired electrons.
50
+ # Spin Angular Momentum (S): up (+1/2) or down (-1/2)
51
+ # Spin Multiplicity (2S + 1)
52
+ # 0 unpaired electron has S = 0, 2S + 1 = 0, called a singlet.
53
+ # 1 unpaired electron has S = 1/2, 2S + 1 = 2, called a doublet (radical).
54
+ # 2 unpaired electrons has S = 1, 2S + 1 = 3, called a triplet.
55
+
47
56
  self.props = {}
48
57
 
49
58
  if molecule is None:
@@ -63,10 +72,12 @@ class Conf:
63
72
 
64
73
  num_atoms = self.rdmol.GetNumAtoms()
65
74
  tot_atoms = self.rdmol.GetNumAtoms(onlyExplicit=False)
75
+
66
76
  assert num_atoms == tot_atoms, "Conf() Error: missing hydrogens"
77
+
67
78
  self.natoms = num_atoms
68
79
  self.charge = rdmolops.GetFormalCharge(self.rdmol)
69
- self.props.update({'atoms': num_atoms})
80
+ self.props.update({'atoms': self.natoms, 'charge': self.charge})
70
81
 
71
82
  assert self.rdmol.GetConformer().Is3D(), "Conf() Error: not 3D"
72
83
 
@@ -148,7 +159,6 @@ class Conf:
148
159
 
149
160
  return self
150
161
 
151
-
152
162
 
153
163
  def optimize(self,
154
164
  calculator: str | Callable = 'MMFF94',
@@ -402,6 +412,95 @@ class Conf:
402
412
  return {i: d[:4] for i, d in enumerate(get_torsion_atoms(self.rdmol, strict))}
403
413
 
404
414
 
415
+ def torsion_energies_one(self,
416
+ calculator: str | Callable,
417
+ indices: tuple,
418
+ simplify: bool = True,
419
+ fmax: float = 0.05,
420
+ interval: float = 20.0,
421
+ use_converged_only: bool = True,
422
+ **kwargs) -> Self:
423
+ """Calculate potential energy profile for a torsion angle.
424
+
425
+ Args:
426
+ calculator (str | Callable): 'MMFF', 'UFF', or ASE calculator.
427
+ indices (tuple): atom indices (i,j,k,l) for a torsion angle
428
+ simplify (bool, optional): whether to use fragementation. Defaults to True.
429
+ fmax (float, optional): convergence limit for optimize. Defaults to 0.05.
430
+ interval (float, optional): angle intervals. Defaults to 20.0.
431
+ use_converged_only (bool, optional): whether to use only converged data. Defaults to True.
432
+
433
+ Returns:
434
+ Self: modified self.
435
+ """
436
+ ref_conf = self.copy()
437
+
438
+ data = {'indices': indices,
439
+ 'angle': [],
440
+ 'init': [],
441
+ 'last': [],
442
+ 'Converged': [],
443
+ }
444
+
445
+ if simplify:
446
+ frag, frag_ijkl = create_torsion_fragment(ref_conf.rdmol, indices)
447
+ frag_conf = Conf(frag)
448
+ for angle in np.arange(-180.0, 180.0, interval):
449
+ # Iterated numpy.ndarray does not contain the last 180: -180., ..., (180).
450
+ conf = frag_conf.copy()
451
+ conf.set_torsion(*frag_ijkl, angle) # atoms bonded to `l` move.
452
+ conf = conf.optimize(calculator, fmax, **kwargs)
453
+ # conf.optimize() updates coordinates and conf.props:
454
+ # `E_tot_init(kcal/mol)`, `E_tot(kcal/mol)`, `Converged`.
455
+ data['angle'].append(angle)
456
+ data['init'].append(conf.props['E_tot_init(kcal/mol)'])
457
+ data['last'].append(conf.props['E_tot(kcal/mol)'])
458
+ data['Converged'].append(conf.props['Converged'])
459
+ frag_cleaned, _ = clean_2d(frag, reset_isotope=True, remove_H=True)
460
+ # to serialize the molecule
461
+ data['frag'] = Chem.MolToMolBlock(frag_cleaned)
462
+ data['frag_indices'] = frag_ijkl
463
+ else:
464
+ for angle in np.arange(-180.0, 180.0, interval):
465
+ # Iterated numpy.ndarray does not contain the last 180: -180., ..., (180).
466
+ conf = ref_conf.copy()
467
+ conf.set_torsion(*indices, angle) # atoms bonded to `l` move.
468
+ conf = conf.optimize(calculator, fmax, **kwargs)
469
+ # conf.optimize() updates coordinates and conf.props:
470
+ # `E_tot_init(kcal/mol)`, `E_tot(kcal/mol)`, `Converged`.
471
+ data['angle'].append(conf.props['angle'])
472
+ data['init'].append(conf.props['E_tot_init(kcal/mol)'])
473
+ data['last'].append(conf.props['E_tot(kcal/mol)'])
474
+ data['Converged'].append(conf.props['Converged'])
475
+
476
+ # Post-processing
477
+ if use_converged_only:
478
+ data['angle'] = list(itertools.compress(data['angle'], data['Converged']))
479
+ data['init' ] = list(itertools.compress(data['init' ], data['Converged']))
480
+ data['last' ] = list(itertools.compress(data['last' ], data['Converged']))
481
+
482
+ relax = np.array(data['init']) - np.median(data['last'])
483
+ E_rel = relax - np.min(relax)
484
+
485
+ torsion_energy_profile = {
486
+ 'indices' : data['indices'],
487
+ 'angle' : np.round(np.array(data['angle']), 1).tolist(), # np.ndarray -> list for serialization
488
+ 'E_rel(kcal/mol)': np.round(E_rel, 2).tolist(), # np.ndarray -> list for serialization
489
+ }
490
+
491
+ if simplify:
492
+ torsion_energy_profile.update({
493
+ 'frag' : data.get('frag', None),
494
+ 'frag_indices' : data.get('frag_indices', None),
495
+ })
496
+
497
+ self.props['torsion'] = torsion_energy_profile
498
+ self.props['torsion_calculator'] = str(calculator)
499
+
500
+ return self
501
+
502
+
503
+
405
504
  def torsion_energies(self,
406
505
  calculator: str | Callable,
407
506
  torsion_key: int | None = None,
@@ -427,80 +526,23 @@ class Conf:
427
526
  Self: modified self.
428
527
  """
429
528
 
430
- if torsion_key is None:
431
- torsion_atoms_indices = self.torsion_atoms()
432
- # {0: (5, 4, 3, 1)}
433
- else:
434
- torsion_atoms_indices = {torsion_key: self.torsion_atoms()[torsion_key]}
529
+ torsion_atoms_dict = self.torsion_atoms() # {0: (5, 4, 3, 1)}
530
+ if (torsion_key is not None) and torsion_atoms_dict.get(torsion_key):
531
+ torsion_atoms_dict = {torsion_key: torsion_atoms_dict.get(torsion_key)}
532
+ # single torsion angle atom indices
435
533
 
436
- ref_conf = self.copy()
437
-
438
- data = {}
439
-
440
- if simplify:
441
- for tk, indices in torsion_atoms_indices.items():
442
- frag, frag_ijkl = create_torsion_fragment(ref_conf.rdmol, indices)
443
- frag_conf = Conf(frag)
444
- data[tk] = {'indices': indices, 'angle':[], 'init':[], 'last':[], 'Converged':[]}
445
- for angle in np.arange(-180.0, 180.0, interval):
446
- # Iterated numpy.ndarray does not contain the last 180: -180., ..., (180).
447
- conf = frag_conf.copy()
448
- conf.props.update({'torsion_key': tk, 'angle': float(angle)})
449
- conf.set_torsion(*frag_ijkl, angle) # atoms bonded to `l` move.
450
- conf = conf.optimize(calculator, fmax, **kwargs)
451
- # conf.optimize() updates coordinates and conf.props:
452
- # `angle`, `E_tot_init(kcal/mol)`, `E_tot(kcal/mol)`, `Converged`.
453
- tk = conf.props['torsion_key']
454
- data[tk]['angle'].append(conf.props['angle'])
455
- data[tk]['init'].append(conf.props['E_tot_init(kcal/mol)'])
456
- data[tk]['last'].append(conf.props['E_tot(kcal/mol)'])
457
- data[tk]['Converged'].append(conf.props['Converged'])
458
- frag_cleaned, _ = clean_2d(frag, reset_isotope=True, remove_H=True)
459
- # to serialize the molecule
460
- data[tk]['frag'] = Chem.MolToMolBlock(frag_cleaned)
461
- data[tk]['frag_indices'] = frag_ijkl
534
+ conf = self.copy()
462
535
 
463
- else:
464
- # mol.confs will be populated with torsion conformers.
465
- # It is designed for a batch optimization in the future.
466
- torsion_angle_confs = []
467
- for tk, indices in torsion_atoms_indices.items():
468
- data[tk] = {'indices': indices, 'angle':[], 'init':[], 'last':[], 'Converged':[]}
469
- for angle in np.arange(-180.0, 180.0, interval):
470
- # Iterated numpy.ndarray does not contain the last 180: -180., ..., (180).
471
- x = ref_conf.copy()
472
- x.props.update({'torsion_key': tk, 'angle': float(angle)})
473
- x.set_torsion(*indices, angle) # atoms bonded to `l` move.
474
- torsion_angle_confs.append(x)
475
-
476
- # Calculate relaxation energies
477
- for conf in torsion_angle_confs:
478
- conf = conf.optimize(calculator, fmax, **kwargs)
479
- # conf.optimize() updates coordinates and conf.props:
480
- # `angle`, `E_tot_init(kcal/mol)`, `E_tot(kcal/mol)`, `Converged`.
481
- tk = conf.props['torsion_key']
482
- data[tk]['angle'].append(conf.props['angle'])
483
- data[tk]['init'].append(conf.props['E_tot_init(kcal/mol)'])
484
- data[tk]['last'].append(conf.props['E_tot(kcal/mol)'])
485
- data[tk]['Converged'].append(conf.props['Converged'])
486
-
487
- # Post-processing
488
536
  torsion_energy_profiles = {}
489
- for tk, dictdata in data.items():
490
- if use_converged_only:
491
- dictdata['angle'] = list(itertools.compress(dictdata['angle'], dictdata['Converged']))
492
- dictdata['init'] = list(itertools.compress(dictdata['init'], dictdata['Converged']))
493
- dictdata['last'] = list(itertools.compress(dictdata['last'], dictdata['Converged']))
494
- relax = np.array(dictdata['init']) - np.median(dictdata['last'])
495
- E_rel = relax - np.min(relax)
496
- torsion_energy_profiles[tk] = {
497
- 'indices' : dictdata['indices'],
498
- 'angle' : np.round(np.array(dictdata['angle']), 1).tolist(), # np.ndarray -> list for serialization
499
- 'E_rel(kcal/mol)': np.round(E_rel, 2).tolist(), # np.ndarray -> list for serialization
500
- 'frag' : dictdata.get('frag', None),
501
- 'frag_indices' : dictdata.get('frag_indices', None),
502
- }
503
-
537
+ for tk, indices in torsion_atoms_dict.items():
538
+ conf = conf.torsion_energies_one(calculator,
539
+ indices,
540
+ simplify,
541
+ fmax,
542
+ interval,
543
+ use_converged_only)
544
+ torsion_energy_profiles[tk] = conf.props['torsion']
545
+
504
546
  self.props['torsion'] = torsion_energy_profiles
505
547
  self.props['torsion_calculator'] = str(calculator)
506
548
 
@@ -548,8 +590,10 @@ class Conf:
548
590
  str: serialized string for json.loads()
549
591
  """
550
592
  serialized = json.dumps({
551
- 'name' : self.name,
552
- 'natoms': self.natoms,
593
+ 'name' : self.name,
594
+ 'natoms': self.natoms,
595
+ 'charge': self.charge,
596
+ 'spin': self.spin,
553
597
  'props' : recursive_round(self.props, decimals),
554
598
  'molblock' : self.to_molblock(),
555
599
  })
@@ -573,7 +617,9 @@ class Conf:
573
617
  data = json.loads(serialized)
574
618
 
575
619
  self.name = data['name']
576
- self.natoms = data['natoms']
620
+ self.natoms = int(data['natoms'])
621
+ self.charge = int(data['charge'])
622
+ self.spin = int(data['spin'])
577
623
  self.props = data['props']
578
624
  self.rdmol = Chem.MolFromMolBlock(data['molblock'], sanitize=False, removeHs=False)
579
625
 
rdworks/mol.py CHANGED
@@ -83,17 +83,18 @@ class Mol:
83
83
  """
84
84
  assert isinstance(molecule, str | Chem.Mol | Conf) or molecule is None
85
85
 
86
+ self.name = ''
86
87
  self.rdmol = None # 2D, one and only one Conformer
87
88
  self.smiles = '' # isomeric SMILES
88
89
  self.confs = [] # container for 3D conformers
89
- self.name = ''
90
90
  self.InChIKey = '' # 27 characters (SHA-256 hash of InChI)
91
- self.InChI = ''
92
91
  self.props = {}
93
- self.fp = None
92
+
94
93
  self.max_workers = max_workers
95
94
  self.chunksize = chunksize
96
95
  self.progress = progress
96
+
97
+ self.fp = None
97
98
 
98
99
  if molecule is None:
99
100
  return
@@ -1569,6 +1570,7 @@ class Mol:
1569
1570
  serialized = json.dumps({
1570
1571
  'name' : self.name,
1571
1572
  'smiles': self.smiles,
1573
+ 'InChIKey': self.InChIKey,
1572
1574
  'props' : recursive_round(self.props, decimals),
1573
1575
  'confs' : [conf.serialize() for conf in self.confs],
1574
1576
  })
@@ -1595,7 +1597,7 @@ class Mol:
1595
1597
  self.smiles = data['smiles'] # isomeric SMILES, no H
1596
1598
  self.rdmol = Chem.MolFromSmiles(data['smiles']) # for 2D depiction
1597
1599
  self.rdmol.SetProp('_Name', self.name)
1598
- self.InChIKey = generate_inchi_key(self.rdmol)
1600
+ self.InChIKey = data['InChIKey']
1599
1601
  self.props = data['props']
1600
1602
  self.confs = [Conf().deserialize(_) for _ in data['confs']] # for 3D conformers (iterable)
1601
1603
 
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: rdworks
3
- Version: 0.38.1
3
+ Version: 0.39.1
4
4
  Summary: Frequently used tasks built on RDKit and other tools
5
5
  Author-email: Sung-Hun Bae <sunghun.bae@gmail.com>
6
6
  Maintainer-email: Sung-Hun Bae <sunghun.bae@gmail.com>
@@ -1,10 +1,10 @@
1
- rdworks/__init__.py,sha256=I7VqtqGi1BTUMH2KV5AU3UWjjteuPPjdB3td8SDB4sE,1368
2
- rdworks/conf.py,sha256=5j6KxncNn9eb3F-O3_S631azx-SK_KQ6oDAnf3Cke0A,28782
1
+ rdworks/__init__.py,sha256=nclDr-b9IUvI96Eo_QJwkhl31eATK109moqGVi8sVOA,1368
2
+ rdworks/conf.py,sha256=65c39R90Mv4Iau5T959fMgJsa6FEoGgjzcPambUFg9Q,30432
3
3
  rdworks/descriptor.py,sha256=34T_dQ6g8v3u-ym8TLKbQtxIIV5TEo-d3pdedq3o-cg,2106
4
4
  rdworks/display.py,sha256=JR0gR26UpH-JCxVOaqXZCUj2MiGZSrx9Me87FncspVI,13469
5
5
  rdworks/ionized.py,sha256=5oIjMRpkX792RIpEEE2Ir96icfFaN_h21mSihhfQPAw,6713
6
6
  rdworks/matchedseries.py,sha256=A3ON4CUpQV159mu9VqgNiJ8uoQ9ePOry9d3ra4NCAgc,10377
7
- rdworks/mol.py,sha256=R6ifd-KIquR9ayl1_HCOE9-tXm89vk9vkHPkQtHsRNU,67787
7
+ rdworks/mol.py,sha256=3wNeM_zVr1-2JKVJzh437X-7EpoEY2srM5WEFmcAV1Y,67808
8
8
  rdworks/mollibr.py,sha256=X4UBO6Ga-QmNS7RwUiaDYAx0Q5hnWs71yTkEpH02Qb4,37696
9
9
  rdworks/pka.py,sha256=NVJVfpcNEMlX5QRyLBgUM7GIT7VMjO-llAR4LWc8J2c,1656
10
10
  rdworks/readin.py,sha256=0bnVcZcAmSLqc6zu1mYcv0LdBv2agQfOpKGwpSRL9VE,11742
@@ -65,8 +65,8 @@ rdworks/predefined/misc/reactive-part-3.xml,sha256=LgWHSEbRTVmgBoIO45xbTo1xQJs0X
65
65
  rdworks/predefined/misc/reactive.xml,sha256=syedoQ6VYUfRLnxy99ObuDniJ_a_WhrWAJbTKFfJ6VY,11248
66
66
  rdworks/xtb/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
67
67
  rdworks/xtb/wrapper.py,sha256=I0nW89vlJZ5Za5pCjIpjsEOFbTm7HpeNGiPgssheWn8,11350
68
- rdworks-0.38.1.dist-info/licenses/LICENSE,sha256=UOkJSBqYyQUvtCp7a-vdCANeEcLE2dnTie_eB1By5SY,1074
69
- rdworks-0.38.1.dist-info/METADATA,sha256=_uZpCzy6HddNOJ1adyBo5oM0VX2-bhCBKz8P_7XHKbg,1183
70
- rdworks-0.38.1.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
71
- rdworks-0.38.1.dist-info/top_level.txt,sha256=05C98HbvBK2axUBogC_hAT_CdpOeQYGnQ6vRAgawr8s,8
72
- rdworks-0.38.1.dist-info/RECORD,,
68
+ rdworks-0.39.1.dist-info/licenses/LICENSE,sha256=UOkJSBqYyQUvtCp7a-vdCANeEcLE2dnTie_eB1By5SY,1074
69
+ rdworks-0.39.1.dist-info/METADATA,sha256=NUoA9z42Law1A2K3bnHMRQkhFYmIGw1kAtgm51G4gmE,1183
70
+ rdworks-0.39.1.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
71
+ rdworks-0.39.1.dist-info/top_level.txt,sha256=05C98HbvBK2axUBogC_hAT_CdpOeQYGnQ6vRAgawr8s,8
72
+ rdworks-0.39.1.dist-info/RECORD,,