I will not lie: I often struggle to find a snippet of code that did something in PyRosetta or I spend hours facing a problem caused by something not working as I expect it to. I recently did a tricky project involving RFdiffusion and I kept slipping on the PyRosetta side. So to make future me, others, and ChatGTP5 happy, here are some common operations to make working with PyRosetta for RFdiffusion easier.
Import
For easy copypasting, let’s start PyRosetta and import things. As said in previous post I am not keen on star imports, because they are not meant to be used indiscriminately and can cause issues —a common pythonic example is when someone mixes collections.Counter
and typing.Counter
.
import pyrosetta import pyrosetta_help as ph from types import ModuleType # Better than star imports: prc: ModuleType = pyrosetta.rosetta.core prp: ModuleType = pyrosetta.rosetta.protocols pru: ModuleType = pyrosetta.rosetta.utility prn: ModuleType = pyrosetta.rosetta.numeric pr_conf: ModuleType = pyrosetta.rosetta.core.conformation pr_scoring: ModuleType = pyrosetta.rosetta.core.scoring pr_res: ModuleType = pyrosetta.rosetta.core.select.residue_selector pr_options: ModuleType = pyrosetta.rosetta.basic.options logger = ph.configure_logger() pyrosetta.distributed.maybe_init(extra_options=ph.make_option_string(no_optH=False, ex1=None, ex2=None, ignore_unrecognized_res=True, load_PDB_components=False, ignore_waters=True, ) )
Load from a PDB file:
pyrosetta.pose_from_file('👾👾👾.pdb')
or from string
pose = pyrosetta.Pose() prc.import_pose.pose_from_pdbstring(pose, 'ATOM 👾👾👾...')
or from a mmCIF file. Note this will not work straight from PyMol because an extra entry at end is needed as discussed previously. I should say working with mmCIF is uncommon, but does have some nice metadata handling advantages.
pose: pyrosetta.Pose = prc.import_pose.pose_from_file('👾👾👾.cif', read_fold_tree=True, type=prc.import_pose.FileType.CIF_file)
There are several other pose importers. Another useful one, for threading especially, is pyrosetta.pose_from_sequence
is rather helpful —just watch out for the eye of Sauron when threading.
Extra complication happens when dealing with residue types that are not in the database. These need to be added to the pose before loading the file.
As always with PyRosetta, if you pass an illegal value, say prc.import_pose.pose_from_file(Exception)
you’ll get a TypeError
that will tell you the accepted arguments of the overloaded function, which sometimes differs from the help(fun
) info.
A new residue type can be added from a params file or from memory. The official way to create a params file is from the Python 2.7 script from the Rosetta download, but that was not good for my purposes so I wrote a new converter, the rdkit_to_params module, which I also made into a web app due to popular demand.
A params file can be passed via the initialisation options (-extra_res_fa
) or via the following:
pru: ModuleType = pyrosetta.rosetta.utility params_filenames: List[str] = ... pose: pyrosetta.Pose = ... params_vector = pru.vector1_string() params_vector.extend([f for f in params_filenames if f]) pyrosetta.generate_nonstandard_residue_set(pose, params_vector)
Having added a residue type does not mean a residue was added to the pose. For that you’d need to do:
prc: ModuleType = pyrosetta.rosetta.core resiset: prc.chemical.ResidueTypeSet = pyrosetta.generate_nonstandard_residue_set(pose, params_vector) new_res = prc.conformation.ResidueFactory.create_residue( resiset.name_map( name ) ) pose.append_residue_by_jump(new_res, pose.num_jump() + 1)
In the rdkit-to-params module, Params.add_residuetype
adds a params but from a string basically.
def add_residuetype(self, pose: pyrosetta.Pose, params_block: str, name3: str, reset:bool=True) \ -> pyrosetta.rosetta.core.chemical.ResidueTypeSet: """ Adds the params to a copy of a residue type set of a pose. If reset is True it will also save it as the default RTS —it keeps other custom residue types. """ rts = pose.conformation().modifiable_residue_type_set_for_conf(prc.chemical.FULL_ATOM_t) buffer = pyrosetta.rosetta.std.stringbuf(params_block) stream = pyrosetta.rosetta.std.istream(buffer) new = prc.chemical.read_topology_file(stream, name3, rts) rts.add_base_residue_type(new) if reset: pose.conformation().reset_residue_type_set_for_conf(rts) return rts
Connections, either polymeric (LOWER
and UPPER
) or otherwise (CONN1
etc.) are a topic that would require its own blog post, so I won’t cover it here.
PDBInfo
PDB information (residue numbering, chain, occupancy and more) is stored in a pyrosetta.rosetta.core.pose.PDBInfo
instance attached to the pose (accessible via .pdb_info()
method, which returns the instance attached and not a copy).
This very handy. Two of its methods are pose2pdb(res=👾)
and pdb2pose(chain=👾, res=👾)
which allows one to jump between the two standards. A nice thing is that it does not segfault when a pose residue index that does not exist is called (RuntimeError
). If the a PDB residue is request that does not exist it will return 0.
A parenthetical warning — one of its methods is .remarks()
, which is a no go. In the PDB format, REMARK is basically a comment that has really strict tools-specific grammar and is a total mess. The .remarks()
does not work in PyRosetta and segfaults even when the file loaded did indeed have REMARK lines. And passing a pyrosetta.rosetta.core.io.Remarks
instance will not work nor will appending to a Remarks
or even making one:
remarks = pyrosetta.rosetta.core.io.Remarks() remark = pyrosetta.rosetta.core.io.RemarkInfo() remark.value = 'Hello world' remarks.append(remark) # ValueError: vector
So were metadata needed to be added to a file, editing the PDB block might be a better bet —editing the mmCIF dictionary data is even betterer. In a pinch appending comment lines to a PDB helps too (but don’t tell anyone).
Certain operations (RemodelMover, grafting etc.) will make the PDB information obsolete or make it get lost.
The Pose class has a method split_by_chain
which returns the chains (as defined in the fold tree) preserving the PDBInfo, but the function (external to pyrosetta.Pose
) pyrosetta.rosetta.core.pose.append_pose_to_pose
does not, so it needs correcting:
def add_chain(built: pyrosetta.Pose, new: pyrosetta.Pose, reset: bool = False) -> None: """ Add a chain ``new`` to a pose ``built`` preserving the residue numbering. :param built: this is the pyrosetta.Pose that will be built into... :param new: the addendum :param reset: resets the PDBInfo for the chain present to A :return: """ built_pi = built.pdb_info() if built_pi is None or reset: built_pi = prc.pose.PDBInfo(built) built.pdb_info(built_pi) for r in range(1, built.total_residue() + 1): built_pi.set_resinfo(res=r, chain_id='A', pdb_res=r) for chain in new.split_by_chain(): offset: int = built.total_residue() pyrosetta.rosetta.core.pose.append_pose_to_pose(built, chain, new_chain=True) # the new `built` residues will not have PDBinfo chain_pi = chain.pdb_info() for r in range(1, chain.total_residue() + 1): built_pi.set_resinfo(res=r + offset, chain_id=chain_pi.chain(r), pdb_res=chain_pi.number(r)) built_pi.obsolete(False)
The functions pyrosetta.rosetta.protocols.grafting.delete_region
or pyrosetta.rosetta.protocols.grafting.return_region
do not impact the PDBinfo.
Given immutation start sequence one can fix the chain infomation thusly:
def fix_starts(pose: pyrosetta.Pose, chain_letters: str, start_seqs: List[str]): """ Fix the chains in place .. code-block:: python strep_seq = 'MEAGIT' start_seqs = ['MKIYY', strep_seq, strep_seq, 'GEFAR', strep_seq, strep_seq, 'FKDET'] fix_starts(pose, chain_letters='ACDEFGB', start_seq=start_seq) :param pose: :param chain_letters: :param start_seq: Confusingly, the first is ignored: the start of the pose is the start of the first chain. :return: """ pi = pose.pdb_info() seq = pose.sequence() seq_iter = iter(start_seqs[1:]+[None]) chain_iter = iter(chain_letters) start_idx = 1 while True: this_chain = next(chain_iter) next_seq = next(seq_iter) if next_seq is None: for i in range(start_idx, len(seq)+1): pi.chain(i, this_chain) break else: next_start = seq.find(next_seq, start_idx) + 1 for i in range(start_idx, next_start): pi.chain(i, this_chain) start_idx = next_start pose.update_pose_chains_from_pdb_chains() assert pose.num_chains() == len(chain_letters), f'{pose.num_chains()} != {len(chain_letters)}'
Threading set-up
The next steps address the fact, that the backbone skeleton lacks sidechains and needs to be threaded with a sequence generated with proteinMPNN. When one imports a pose with missing sidechains into PyRosetta, these get added. They are good, but not perfect, so copying the coordinates of the conserved sidechains is a must. A key move is superposition.
Superposition
First, a silly pedantic lexical side note. When one has a sequence of elements and one spaces out the elements so they match, that is called to align
. When one places one object over another but letting one still be visible, say with a stagger like the traces in a joy plot or translating without rotating (e.g. ▽+△=✡︎) that is called to superimpose
, when one rototranslates fully (e.g. ▽+△↺=▽), that is called to superpose
. Just to go through both words, below I will the align sequences, to superposed structures. Although, in this specific case, the metadata has the information so best use that.
There are actually several functions that allow superpositions, including presets for CA atoms or all atoms and so forth. The above is customisable as you have to tell it what atom goes to which.
In PyRosetta one can superpose two poses with the following:
mobile: pyrosetta.Pose = ... ref: pyrosetta.Pose = ... atom_map: prs.map_core_id_AtomID_core_id_AtomID = ... # this is a hash-mapping of mobile `pyrosetta.AtomID` to reference `pyrosetta.AtomID` rmsd: float = pr_scoring.superimpose_pose(mod_pose=mobile, ref_pose=ref, atom_map=atom_map)
If pyrosetta.AtomID
seems familiar, that is because for constraints one has to play around with them a lot. So where we to align two poses by the CA atoms of a specified PDB chain that ought to be common, we can do:
prc: ModuleType = pyrosetta.rosetta.core prp: ModuleType = pyrosetta.rosetta.protocols pru: ModuleType = pyrosetta.rosetta.utility prn: ModuleType = pyrosetta.rosetta.numeric prs: ModuleType = pyrosetta.rosetta.std pr_conf: ModuleType = pyrosetta.rosetta.core.conformation pr_scoring: ModuleType = pyrosetta.rosetta.core.scoring pr_res: ModuleType = pyrosetta.rosetta.core.select.residue_selector pr_options: ModuleType = pyrosetta.rosetta.basic.options def superpose_pose_by_chain(pose, ref, chain: str, strict: bool=True) -> float: """ superpose by PDB chain letter :param pose: :param ref: :param chain: :return: """ atom_map = prs.map_core_id_AtomID_core_id_AtomID() chain_sele: pr_res.ResidueSelector = pr_res.ChainSelector(chain) r: int # reference pose residue number (Fortran) m: int # mobile pose residue number (Fortran) for r, m in zip(pr_res.selection_positions(chain_sele.apply(ref)), pr_res.selection_positions(chain_sele.apply(pose)) ): if strict: assert pose.residue(m).name3() == ref.residue(r).name3(), 'Mismatching residue positions!' ref_atom = pyrosetta.AtomID(ref.residue(r).atom_index("CA"), r) mobile_atom = pyrosetta.AtomID(pose.residue(m).atom_index("CA"), m) atom_map[mobile_atom] = ref_atom return pr_scoring.superimpose_pose(mod_pose=pose, ref_pose=ref, atom_map=atom_map)
So what operations move the pose? A lot. The main two to keep an eye out for are RFdiffusion itself and Relax and its pesky drift.
TRB file
Now that we know how to superpose two poses, we need to figure out which are the conserved ones. To determine what is conserved in a RFdiffusion run, the pickle .trb
file is a good option, as it will have some good information. If this file was deleted or similar, then see alignment below. The designed protein, is termed the hallucination (technically the field is moving away from that term, but it is a nice one), while the parent template, the reference. The dictionary in the .trb
file will have several keys some in the pattern [complex_/]con_[ref/hal]_[pdb_idx/idx0]. In RFdiffusion, one designs only chain A and all other chains become chain B: complex_
includes chain B. So to get the mapping of the conserved residues is as easy as dict(zip(trb['complex_con_ref_idx0'], trb['complex_con_hal_idx0']))
. inpaint_seq
is a sequence of booleans spanning the length of the complex.
Digression: Alignment
Sequence alignment comes into play when superposing protein by the conserved regions —outside of RFdiffusion with the trb file. For that Bio.Align.PairwiseAligner
can be used (since pairwise2
was deprecated).
In the default settings, the character ‘-‘ in the input sequence is treated like a regular character and not a free gap. Were one hellbent to do so a custom substitution matrix would be needed.
However, if the poses to superpose differed by a span that any identity is coincidental, such as a redesigned part of a domain, then we need to strongly penalise mismatches and not penalise gap extensions.
from Bio.Align import PairwiseAligner, Alignment, substitution_matrices def superpose_by_common_alignment(mobile: pyrosetta.Pose, ref: pyrosetta.Pose) -> float: """ Pairwise alignment of the sequences of the poses. This time only spans that are common. return (ref_index, mobile_index) :param mobile: :param ref: :return: """ # ## align aligner = PairwiseAligner() # We don't want to get mismatches aligned, so no to: # aligner.substitution_matrix = substitution_matrices.load("BLOSUM62") aligner.internal_gap_score = -10 aligner.extend_gap_score = -0.01 aligner.end_gap_score = -0.01 ref_seq: str = ref.sequence() pose_seq: str = mobile.sequence() aln: Alignment = aligner.align(ref_seq, pose_seq)[0] # like before but with an extra condition `ref_seq[t] == pose_seq[q]` aln_map: Dict[int, int] = {t: q for t, q in zip(aln.indices[0], aln.indices[1]) if q != -1 and t != -1 and ref_seq[t] == pose_seq[q]} # ## make pyrosetta atom map ... # return RMSD return pr_scoring.superimpose_pose(mod_pose=mobile, ref_pose=ref, atom_map=atom_map)
In the case that one end does not matter in the sequence, there are attributes that control the left (N-terminal) an right (C-terminal) side gaps. For example, the attributes target_right_gap_score
and target_right_extend_gap_score
could be set to zero, which make the C-terminal difference an inkshed of gaps.
Were one to want align to homologues, this would be fine:
from Bio.Align import PairwiseAligner, Alignment, substitution_matrices def superpose_by_alignment(mobile: pyrosetta.Pose, ref: pyrosetta.Pose) -> float: """ Pairwise alignment of the sequences of the poses. return (ref_index, mobile_index) :param mobile: :param ref: :return: """ # ## align aligner = PairwiseAligner() aligner.substitution_matrix = substitution_matrices.load("BLOSUM62") ref_seq: str = ref.sequence() pose_seq: str = mobile.sequence() aln: Alignment = aligner.align(ref_seq, pose_seq)[0] # `aln.indices` has the mapping # an index of -1 is a map to a gap aln_map: Dict[int, int] = {t: q for t, q in zip(aln.indices[0], aln.indices[1]) if q != -1 and t != -1} # ## make pyrosetta atom map # for the purpose of explanation the following block is not in its own function, but will be repeated in a minute. # where these all part of the same code, the repeated part would need to be its own function as it is very modular anyway! atom_map = prs.map_core_id_AtomID_core_id_AtomID() for r, m in aln_map.items(): ref_atom = pyrosetta.AtomID(ref.residue(r + offset).atom_index("CA"), r + offset) mobile_atom = pyrosetta.AtomID(mobile.residue(m + offset).atom_index("CA"), m + offset) atom_map[mobile_atom] = ref_atom # ## superpose and return RMSD return pr_scoring.superimpose_pose(mod_pose=mobile, ref_pose=ref, atom_map=atom_map)
Stealing sidechains
One critical detail to keep in mind is that the RFdiffusion/hallucination is a backbone skeleton without sidechains. When this is imported, PyRosetta will add the missing atoms, sequentially by residue. Intriguingly, the sidechains match somewhat where they should be, but the match is not perfect and the potential energy is worse (higher). As a result we need to steal the sidechains from the template. To do this we need to be careful with variable names as the superposition runs on the words ‘ref’ and ‘mobile’, while the trb metadata runs off ‘hal’ and ‘ref’.
def steal_frozen(acceptor: pyrosetta.Pose, donor: pyrosetta.Pose, trb: Dict[str, Any], move_acceptor: bool = False ): """ Copy all the conserved coordinates from the donor to the acceptor. These are determined by ``trb`` dict from RFdiffusion. This is similar to ``prp.comparative_modeling.StealSideChainsMover``, but does not require the threading infrastructure. The hallucination is the acceptor, the parent is the donor. The RFdiffused pose is skeleton, but when imported the sidechains are added. The theft is done in 3 steps. 1. A mapping of residue idx, atom idx to residue idx, atom idx is made. 2. The hallucination is superimposed on the parent if move_acceptor is True, else vice versa. 3. The coordinates are copied from the parent to the hallucination. The term 'ref' gets confusing. hallucination is fixed / mutanda, parent is mobile fixed is called ref, but ref means parent for RFdiffusion so it is flipped) :param acceptor: :param donor: :param trb: :return: """ # ## Make mapping of all atoms of conserved residues donor2acceptor_idx1s: Dict[Tuple[FTypeIdx, FTypeIdx], Tuple[FTypeIdx, FTypeIdx, str]] = {} # these run off 0-based indices for donor_res_idx0, acceptor_res_idx0 in zip(trb['complex_con_ref_idx0'], trb['complex_con_hal_idx0']): donor_res_idx1 = donor_res_idx0 + 1 acceptor_res_idx1 = acceptor_res_idx0 + 1 acceptor_res = acceptor.residue(acceptor_res_idx1) donor_res = donor.residue(donor_res_idx1) assert donor_res.name3() == acceptor_res.name3(), f'donor {donor_res.name3()} != acceptor {acceptor_res.name3()}' mob_atomnames = [donor_res.atom_name(ai1) for ai1 in range(1, donor_res.natoms() + 1)] for fixed_atm_idx1 in range(1, acceptor_res.natoms() + 1): aname = acceptor_res.atom_name(fixed_atm_idx1) # key to map one to other: overkill bar for HIE/HID if aname not in mob_atomnames: print(f'Template residue {donor_res.annotated_name()}{donor_res_idx1} lacks atom {aname}') continue donor_atm_idx1 = donor_res.atom_index(aname) donor2acceptor_idx1s[(donor_res_idx1, donor_atm_idx1)] = (acceptor_res_idx1, fixed_atm_idx1, aname) # ## Superpose atom_map = prs.map_core_id_AtomID_core_id_AtomID() if move_acceptor: mobile: pyrosetta.Pose = acceptor fixed: pyrosetta.Pose = donor else: mobile: pyrosetta.Pose = donor fixed: pyrosetta.Pose = acceptor for (donor_res_idx1, donor_atm_idx1), (acceptor_res_idx1, acceptor_atm_idx1, aname) in donor2acceptor_idx1s.items(): if move_acceptor: mob_res_idx1, mob_atm_idx1 = acceptor_res_idx1, acceptor_atm_idx1 fixed_res_idx1, fixed_atm_idx1 = donor_res_idx1, donor_atm_idx1 else: mob_res_idx1, mob_atm_idx1 = donor_res_idx1, donor_atm_idx1 fixed_res_idx1, fixed_atm_idx1 = acceptor_res_idx1, acceptor_atm_idx1 if aname.strip() not in ('N', 'CA', 'C', 'O'): # BB superpositon continue fixed_atom = pyrosetta.AtomID(fixed_atm_idx1, fixed_res_idx1) mobile_atom = pyrosetta.AtomID(mob_atm_idx1, mob_res_idx1) atom_map[mobile_atom] = fixed_atom rmsd = prc.scoring.superimpose_pose(mod_pose=mobile, ref_pose=fixed, atom_map=atom_map) # I am unsure why this is not near zero but around 0.1–0.3 assert rmsd < 1, f'RMSD {rmsd} is too high' # ## Copy coordinates to_move_atomIDs = pru.vector1_core_id_AtomID() to_move_to_xyz = pru.vector1_numeric_xyzVector_double_t() for (donor_res_idx1, donor_atm_idx1), (acceptor_res_idx1, acceptor_atm_idx1, aname) in donor2acceptor_idx1s.items(): # if aname in ('N', 'CA', 'C', 'O'): # BB if common # continue # this does not stick: fixed_res.set_xyz( fixed_ai1, mob_res.xyz(mob_ai1) ) to_move_atomIDs.append(pyrosetta.AtomID(acceptor_atm_idx1, acceptor_res_idx1)) to_move_to_xyz.append(donor.residue(donor_res_idx1).xyz(donor_atm_idx1)) acceptor.batch_set_xyz(to_move_atomIDs, to_move_to_xyz) # ## Fix HIE/HID the brutal way v = prc.select.residue_selector.ResidueNameSelector('HIS').apply(acceptor) relax = prp.relax.FastRelax(pyrosetta.get_score_function(), 1) movemap = pyrosetta.MoveMap() movemap.set_bb(False) movemap.set_chi(v) movemap.set_jump(False) relax.apply(acceptor) return rmsd, donor2acceptor_idx1s
Once in a while, I get an email asking me about HIE/HID, because I mention it somewhere, so I am reticent to address a wee bit of code there: the HIS tautomerisation. In Rosetta, the HID tautomer is a patch on HIS, (residue.annotated_name() == 'H[HIS_D]'
. For a specific change one can mutate the residue to alanine, then the patched/unpatched residue (cf. §Thread). There are tasks that can added to packers to sample the his tautomers / HNQ flips, but the laziest way is to let Relax deal with it.
Thread
In PyRosetta one can mutate a residue via
prp: ModuleType = pyrosetta.rosetta.protocols prp.simple_moves.MutateResidue(1, 'ALA').apply(pose)
This allows mutations to not only base residues, but also patched residues, such as NtermProteinFull
(with extra N-terminal proton) or acetylated
etc. To see if in the database folder there is what you are after it is honestly easier simply looking in the folder:
import pyrosetta, itertools from pathlib import Path from itertools import chain fa = 'database/chemical/residue_type_sets/fa_standard' # residue types print([p.name for p in (Path(pyrosetta.__file__).parent / fa / 'residue_types' ).glob('*/*.params')]) # params types patches_folder = (Path(pyrosetta.__file__).parent / fa / 'patches' ) print([p.name for p in itertools.chain(patches_folder.glob('*.txt'), patches_folder.glob('*/*.txt'))])
Just be aware, that MutateResidue
segfaults on error, so try the patch first and it is fiddly at first.
To remove the terminus patches one can do the following:
import pyrosetta prc: ModuleType = pyrosetta.rosetta.core pru: ModuleType = pyrosetta.rosetta.utility def remove_terminus_patches(pose): clean_template_pose = template_pose.clone() prc.pose.remove_nonprotein_residues(pose) ### find lowers = pru.vector1_std_pair_unsigned_long_protocols_sic_dock_Vec3_t() uppers = pru.vector1_std_pair_unsigned_long_protocols_sic_dock_Vec3_t() prc.sic_dock.get_termini_from_pose(pose, lowers, uppers) ### remove for upper, _ in uppers: prc.conformation.remove_upper_terminus_type_from_conformation_residue(clean_template_pose.conformation(), upper) for lower, _ in lowers: prc.conformation.remove_lower_terminus_type_from_conformation_residue(clean_template_pose.conformation(), lower)
However, mutating a whole pose based on a sequence is a pillar of homology modelling, namely threading. I have discussed in an other post threading and hybridisation in PyRosetta and most examples here are simply ported from a helper module of mine, so I’ll be brief. Threading in Rosetta requires a “Grishin file”, which is a specific form of pairwise sequence alignment file. The mover is ThreadingMover
with the aid of StealSideChainsMover
, namely:
prc: ModuleType = pyrosetta.rosetta.core prp: ModuleType = pyrosetta.rosetta.protocols clean_template_pose = remove_terminus_patches(pose) # no termini from earlier! target_pose = pyrosetta.Pose() prc.pose.make_pose_from_sequence(target_pose, target_sequence, 'fa_standard') ## Thread align = prc.sequence.read_aln(format='grishin', filename=aln_file)[1] threader = prp.comparative_modeling.ThreadingMover(align=align, template_pose=clean_template_pose) threader.apply(target_pose) ## Steal sidechains qt = threader.get_qt_mapping(target_pose) steal = prp.comparative_modeling.StealSideChainsMover(clean_template_pose, qt) steal.apply(target_pose)
The main point of the aforementioned post about threading was loop modelling by cannibalising AF2, which can be done with the following settings before using the ThreadingMover
mover:
# optional set for loop modelling from reference poses poses: List[pyrosetta.Pose] = ... # poses to cannibilise for loops lengths: List[int] = [3,] # 3,6,9 are traditional choices fragment_sets = pru.vector1_std_shared_ptr_core_fragment_FragSet_t(len(lengths)) for i, l in enumerate(lengths): fragment_sets[i+1] = prc.fragment.ConstantLengthFragSet(l) for pose in poses: prc.fragment.steal_constant_length_frag_set_from_pose(pose, fragment_sets[i+1]) threader = ... threader.build_loops(True) threader.randomize_loop_coords(True) # default threader.frag_libs(fragment_sets) threader.apply(...)
As may be apparent, the ThreadingMover
does not like non-base residues. So for ligand residues, one can just use append_subpose_to_pose
function.
prc: ModuleType = pyrosetta.rosetta.core pr_res: ModuleType = pyrosetta.rosetta.core.select.residue_selector def steal_ligands(donor_pose, acceptor_pose) -> None: """ Steals non-Protein residues from donor_pose and adds them to acceptor_pose Do not use with nucleic acid polymers. :param donor_pose: :param acceptor_pose: :return: """ PROTEIN = prc.chemical.ResidueProperty.PROTEIN prot_sele = pr_res.ResiduePropertySelector(PROTEIN) not_sele = pr_res.NotResidueSelector(prot_sele) rv = pr_res.ResidueVector(not_sele.apply(donor_pose)) # if it were DNA... # for from_res, to_res in ph.rangify(rv): # prc.pose.append_subpose_to_pose(acceptor_pose, donor_pose, from_res, to_res, True) for res in rv: prc.pose.append_subpose_to_pose(acceptor_pose, donor_pose, res, res, True)
Now back to threading. As mentioned, I just use my helper function to deal with the above:
import pyrosetta_help def thread(template_block, target_seq, target_name, template_name, temp_folder='/tmp'): # load template template = pyrosetta.Pose() prc.import_pose.pose_from_pdbstring(template, template_block) # thread aln_filename = f'{temp_folder}/{template_name}-{target_name}.grishin' ph.write_grishin(target_name=target_name, target_sequence=target_seq, template_name=template_name, template_sequence=template.sequence(), outfile=aln_filename ) aln: prc.sequence.SequenceAlignment = prc.sequence.read_aln(format='grishin', filename=aln_filename)[1] threaded: pyrosetta.Pose threader: prp.comparative_modeling.ThreadingMover threadites: pru.vector1_bool threaded, threader, threadites = ph.thread(target_sequence=target_seq, template_pose=template, target_name=target_name, template_name=template_name, align=aln ) # no need to superpose. It is already aligned # superpose(template, threaded) # fix pdb info n = threaded.total_residue() pi = prc.pose.PDBInfo(n) for i in range(1, n + 1): pi.number(i, i) pi.chain(i, 'A') threaded.pdb_info(pi) return threaded
A pythonic parenthesis. Names of callable objects are, by good practice, verbs, however stuff gets confusing when a word can be both a noun and a verb: I have tripped up so often with thread and fragment. Hence the use of the past particle in the above. For fragment, which has initial-stress derivation I naughtily use fràgment
and fragm
. è
nt
Relax
The workhorse of Rosetta is without question the FastRelax mover. It is very customisable, so most often there is no near to play with packers, Monte Carlo samplers etc. This is true for design mode. I won’t go into detail on the basics as this is covered in every tutorial.
scorefxn: pr_scoring.ScoreFunction = pyrosetta.get_fa_scorefxn() # don't forget to set weights! scorefxn.set_weight(pr_scoring.ScoreType.atom_pair_constraint, 3) scorefxn.set_weight(pr_scoring.ScoreType.coordinate_constraint, 5) cycles = 3 # more than 5 is probably not going to do much relax = prp.relax.FastRelax(scorefxn, cycles) # set movemap to freeze residues # say repack the sidechains of chain A and its neighbours # allow backbone movements for chain A only movemap = pyrosetta.MoveMap() rs: ModuleType = prc.select.residue_selector chainA_sele: rs.ResidueSelector = rs.ChainSelector('A') chainA: pru.vector1_bool = chainA_sele.apply(pose) neigh_sele: rs.ResidueSelector = rs.NeighborhoodResidueSelector(chainA_sele, True, distance) neighs: pru.vector1_bool = neigh_sele.apply(pose) movemap.set_chi(neighs) movemap.set_bb(chainA) movemap.set_jump(False) relax.set_movemap(movemap) # go! replax.apply(pose)
In the case of RFdiffusion the template ought to have been minimised otherwise either the scores will be for which case was minimised better if the other chains are also minimised (don’t) or the scores will be for cases where the designed binder interacts or not with a spuriously strained part of the binding partner.
In the above example the neighbourhood was moved as an example. If you do repack these sidechains, which is reasonable, do make sure to keep the complex and the original binding partner will no longer work.
A cool thing once can do with PyRosetta is use electron design as a constraint (tutorial) —theoretically one could used RFdiffusion via blender API, gemmi and numpy manupulations to make custom shaped protein, but that would be for fun and not actually relevant here.
If there are covalent bond lengths that need fixing, the we have do it cartesian:
constraint_weight = 5 scorefxn: pr_scoring.ScoreFunction = pyrosetta.create_score_function('ref2015_cart') scorefxn.set_weight(pr_scoring.ScoreType.coordinate_constraint, constraint_weight) scorefxn.set_weight(pr_scoring.ScoreType.angle_constraint, constraint_weight) scorefxn.set_weight(pr_scoring.ScoreType.atom_pair_constraint, constraint_weight) relax = pyrosetta.rosetta.protocols.relax.FastRelax(scorefxn, 3) relax.cartesian(True) relax.minimize_bond_angles(True) relax.minimize_bond_lengths(True) relax.apply(pose)
In both example I allude to constraints. These are well covered and are set up in various way for example, given the pose index and atom name of the atoms we can do:
def constrain_distance(pose, fore_idx, fore_name, aft_idx, aft_name, x0_in=1.334, sd_in=0.2, tol_in=0.02, weight=1): AtomPairConstraint = pr_scoring.constraints.AtomPairConstraint # noqa fore = pyrosetta.AtomID(atomno_in=pose.residue(fore_idx).atom_index(fore_name), rsd_in=fore_idx) aft = pyrosetta.AtomID(atomno_in=pose.residue(aft_idx).atom_index(aft_name), rsd_in=aft_idx) fun = pr_scoring.func.FlatHarmonicFunc(x0_in=x0_in, sd_in=sd_in, tol_in=tol_in) if weight != 1: fun = pr_scoring.func.ScalarWeightedFunc(weight, fun) con = AtomPairConstraint(fore, aft, fun) pose.add_constraint(con) return con def constrain_angle(pose, fore_idx, fore_name, mid_name, mid_idx, aft_idx, aft_name, x0_in=109/180*3.14, sd_in=10/180*3.14, weight=1): AngleConstraint = pr_scoring.constraints.AngleConstraint fore = pyrosetta.AtomID(atomno_in=pose.residue(fore_idx).atom_index(fore_name), rsd_in=fore_idx) mid = pyrosetta.AtomID(atomno_in=pose.residue(mid_idx).atom_index(mid_name), rsd_in=mid_idx) aft = pyrosetta.AtomID(atomno_in=pose.residue(aft_idx).atom_index(aft_name), rsd_in=aft_idx) fun = pr_scoring.func.CircularHarmonicFunc(x0_radians=x0_in, sd_radians=sd_in) if weight != 1: fun = pr_scoring.func.ScalarWeightedFunc(weight, fun) con = AngleConstraint(fore, mid, aft, fun) pose.add_constraint(con) return con def constrain_position(pose: pyrosetta.Pose, target_index: int, ref_index: int, x0_in=0., sd_in=0.01): ref_ca = pyrosetta.AtomID(atomno_in=pose.residue(ref_index).atom_index('CA'), rsd_in=ref_index) target_ca = pyrosetta.AtomID(atomno_in=pose.residue(target_index).atom_index('CA'), rsd_in=frozen_index) target_xyz = pose.residue(frozen_index).xyz(target_ca.atomno()) fun = pr_scoring.func.HarmonicFunc(x0_in=x0_in, sd_in=sd_in) if weight != 1: fun = pr_scoring.func.ScalarWeightedFunc(weight, fun) con = pr_scoring.constraints.CoordinateConstraint(a1=target_ca, fixed_atom_in=ref_ca, xyz_target_in=target_xyz, func=fun, scotype=pr_scoring.ScoreType.coordinate_constraint) pose.add_constraint(con) def constrain_peptide_gap(pose, chain_break, x0_in=1.334, sd_in=0.2, tol_in=0.02): """ Close a gap between """ AtomPairConstraint = pr_scoring.constraints.AtomPairConstraint # noqa fore_c = pyrosetta.AtomID(atomno_in=pose.residue(chain_break).atom_index('C'), rsd_in=chain_break) aft_n = pyrosetta.AtomID(atomno_in=pose.residue(chain_break + 1).atom_index('N'), rsd_in=chain_break + 1) fun = pr_scoring.func.FlatHarmonicFunc(x0_in=x0_in, sd_in=sd_in, tol_in=tol_in) con = AtomPairConstraint(fore_c, aft_n, fun) pose.add_constraint(con)
One thing to keep an eye out for is how healthy are the constraints after minimisation:
def show_cons(pose): """ print the score for the various constraints """ get_atomname = lambda atomid: pose.residue(atomid.rsd()).atom_name(atomid.atomno()).strip() get_description = lambda atomid: f'{pose.residue(atomid.rsd()).name3()}{pi.pose2pdb(atomid.rsd()).strip().replace(" ", ":")}.{get_atomname(atomid)}' for con in pose.constraint_set().get_all_constraints(): if con.type() == 'AtomPair': print(con.type(), get_description(con.atom1()), get_description(con.atom2()), con.score(pose)) elif con.type() == 'Angle': print(con.type(), get_description(con.atom1()), get_description(con.atom2()), get_description(con.atom3()), con.score(pose)) else: print(con.type(), con.score(pose))
Design
One can do design with FastRelax. Due to the way the mover samples, it fixes sub-optimal residues one by one as opposed to large scale remodelling —for that Remodel might be better suitable (tutorial). Design is really useful for RFdiffusion as the latter can in essence be thought as coarse-grain so there may be some residue combinations that are subpar. Obviously, when FastDesign choses a residue it is based on the scorefunction, which is imperfect: a quirky example of this is its proclivity to change certain residues in Streptag when bound to streptavidin even though the former is the product of rounds and rounds of phage display.
To run design one needs to create a task factory for it onto which restrictions (operations) are pushed. Confusingly, a default task factory designs everything, while the default task factory in relax does not. The ProhibitSpecifiedBaseResidueTypes
is really handy for preventing cysteine from making life hard and also for correcting surface charges —protein are not very soluble if the isoelectric point is close to pH 7.4 or aggregate if there are exposed patches. On the latter note, proteinMPNN when using the vanilla weights potentially might make inadvertently a transmembrane helix, so that is something to keep an eye out for.
def create_design_tf(pose:pyrosetta.Pose, design_sele: pr_res.ResidueSelector, distance:int) -> prc.pack.task.TaskFactory: """ Create design task factory for relax. Designs the ``design_sele`` and repacks around ``distance`` of it. Remember to do ... code-block:: python relax.set_enable_design(True) relax.set_task_factory(task_factory) """ # design is default, so this is not done: # residues_to_design = design_sele.apply(pose) # design_ops = prc.pack.task.operation.OperateOnResidueSubset(????, residues_to_design) no_cys = pru.vector1_std_string(1) no_cys[1] = 'CYS' no_cys_ops = prc.pack.task.operation.ProhibitSpecifiedBaseResidueTypes(no_cys) # No design, but repack repack_sele = pr_res.NeighborhoodResidueSelector(design_sele, distance, False) residues_to_repack = repack_sele.apply(pose) repack_rtl = prc.pack.task.operation.RestrictToRepackingRLT() repack_ops = prc.pack.task.operation.OperateOnResidueSubset(repack_rtl, residues_to_repack) # No repack, no design frozen_sele = pr_res.NotResidueSelector(pr_res.OrResidueSelector(design_sele, repack_sele)) residues_to_freeze = frozen_sele.apply(pose) prevent_rtl = prc.pack.task.operation.PreventRepackingRLT() frozen_ops = prc.pack.task.operation.OperateOnResidueSubset(prevent_rtl, residues_to_freeze) # pyrosetta.rosetta.core.pack.task.operation.RestrictAbsentCanonicalAASRLT # pyrosetta.rosetta.core.pack.task.operation.PreserveCBetaRLT task_factory = prc.pack.task.TaskFactory() task_factory.push_back(no_cys_ops) task_factory.push_back(repack_ops) task_factory.push_back(frozen_ops) return task_factory
Favour native residues
A residue is changed because the variant has a better score than the wild type —technically within the metropolis criterion of the Monte Carlo sampling. Therefore it is possible for the sequence to drift near neutrally, which is not what we want here. To this effect the constraint ResidueTypeConstraint
addresses this by favouring the native residue. This is a constraint, not a packer operation, and is controlled by the weight pyrosetta.rosetta.core.scoring.ScoreType.res_type_constraint
. There are a least three ways to declare the constraint:
# native residues relative to pose object — the pose is used to get name3 and is not kept (so can change) # very much like the 'constrain to starting positions' in relax which creates a myriad coordinate constraints prc.scoring.constraints.ResidueTypeConstraint(pose, seqpos=1, native_residue_bonus=1) # or relative to user specified residue prc.scoring.constraints.ResidueTypeConstraint(seqpos=1, aa_in='A', name3_in='ALA', native_residue_bonus=1) # or using a wrapper for every residue in the pose prp.protein_interface_design.FavorNativeResidue(pose, 1) # all options require scorefxn.set_weight(pyrosetta.rosetta.core.scoring.ScoreType.res_type_constraint, 1)
The bonus and weight to add is 1 and 1 in this example, but on preliminary tests, I feel like 1 and 2 is a better call, but that is a discussion for another time.
The ProhibitSpecifiedBaseResidueTypes
operation restricts against given amino acids, while to allow non-canonical amino acids a packer palette needs to be added (task_factory.set_packer_palette
), but neither gives weights to given residues. One simple option to do that would be to add multiple FavorNativeResidue
constraints —they need not be wrapped in AmbiguousConstraint
because the non-scoring ones get zero.
Variable length
When I used RFdiffusion I gave the algorithm a range in the contigmap, thus allowing the best length of an insertion to win. This does however make the pipeline a bit complicated. First I had to adapt the helper scripts of ProteinMPNN to be used as Python functions (see footnote). There are then two options for the design step: refer to the metadata or do a sequence alignment. The former requires being organised, while the latter could use the alignment code discussed above.
def design_different_by_trb(pose: pyrosetta.Pose, trb: Dict[str, Any], cycles = 5, scorefxn=None): """The RFdiffusion case""" idx_sele = prc.select.residue_selector.ResidueIndexSelector() for idx0, b in enumerate(trb['inpaint_seq']): if b: continue idx_sele.append_index(idx0 + 1) task_factory: prc.pack.task.TaskFactory = create_design_tf(pose, design_sele=idx_sele, distance=0) if scorefxn is None: scorefxn = pyrosetta.get_fa_scorefxn() relax = pyrosetta.rosetta.protocols.relax.FastRelax(scorefxn, cycles) relax.set_enable_design(True) relax.set_task_factory(task_factory) relax.apply(pose) def design_different_by_alignment(pose: pyrosetta.Pose, ref: pyrosetta.Pose, cycles = 5, scorefxn=None): """Not the RFdiffusion case""" ref = ref.split_by_chain(1) ref2pose: dict = align_for_atom_map(pose.split_by_chain(1), ref) conserved = list(ref2pose.values()) idx_sele = pr_res.ResidueIndexSelector() for i in range(1, len(pose.chain_sequence(1))): if i not in conserved: idx_sele.append_index(i) task_factory: prc.pack.task.TaskFactory = create_design_tf(pose, design_sele=idx_sele, distance=0) if scorefxn is None: scorefxn = pyrosetta.get_fa_scorefxn() relax = pyrosetta.rosetta.protocols.relax.FastRelax(scorefxn, cycles) relax.set_enable_design(True) relax.set_task_factory(task_factory) relax.apply(pose)
The degree of change during a FastDesign run depends on how strained the residues are. So it important that the design be FastRelax properly first and not all at once.
Actually I often use FastRelax in a first pass with the backbone fixed to prevent the model from blowing up.
This degree of change _I think_ makes a good metric for scoring. If the pose.sequence() changed by more than 20% (irrespective of native favouring), then that is hard no.
Scoring
RFdiffusion will be made to generate lots of designs. The reason is because they are not all amazing. So they need to be scored.
Coarse-grain clashes
First, we need to determine if the polyglycine skeleton will clash with the extended neighbourhood when the protein is part of a greater machinery that could not be feasibly fed into RFdiffusion. An example is a membrane, say one were designing a binder to a transmembrane protein, the design should not cross the membrane, so one could get the transmembrane protein from OPM database, superpose the skeleton and use the layers of O / N atoms. For a project of mine, I had a multidimensional polymeric lattice. Scoring via a scorefunction and using only the Lenard–Jones repulsion is a tad overkill and prone to errors. So simply doing distance based calculations works:
skeleton: pyrosetta.Pose = ... clasher: pyrosetta.Pose = ... # where the skeleton combined with this pose, it should not have clashes xyz_skeleton = extract_coords(skeleton) xyz_clasher = extract_coords(clasher) all_distances = np.sqrt(np.sum((xyz_clasher[:, np.newaxis, :] - xyz_skeleton[np.newaxis, :, :]) ** 2, axis=-1)) n_clashing = np.count_nonzero(all_distances < 1.)
If designing interactions, the above but with < 3. will given backbone hydrogen bonding to any atom of the reference.
Say the skeleton passed, we then generate the sequence with proteinMPNN, thread it, relax it and tune the sequence. Now we can do some proper scoring.
∆∆G
The first is the predicted Gibbs free energy from a scorefunction. The catch is that this absolute energy, which should not be taken too seriously. The reason being is that they are optimised for ∆∆G calculations, so have empirical terms (i.e. fudge factors) to make them work, including the REF value (a per residue type weight, ref2015 values shown below), which means that were one to make a linear string of peptide it will not have a predicted absolute energy of folding of zero. However, in terms of magnitude is there is a screaming high value then the model is for the bin.
{'ALA': 1.32468, 'ARG': -0.09474, 'ASN': -1.34026, 'ASP': -2.14574, 'CYS': 3.25479, 'GLN': -1.45095, 'GLU': -2.72453, 'GLY': 0.79816, 'HIS': -0.30065, 'ILE': 2.30374, 'LEU': 1.66147, 'LYS': -0.71458, 'MET': 1.65735, 'PHE': 1.21829, 'PRO': -1.64321, 'SER': -0.28969, 'THR': 1.15175, 'TRP': 2.26099, 'TYR': 0.58223, 'VAL': 2.64269}
Per residue score
The scorefunction is a sum of the scores of each particle/residue, and the per-residue scores are obtainable. These are a common way to determine how accurate the model likely is. The per-residue score should not >+10 kcal/mol, and most residues in minimised PDB protein are under 5 kcal/mol, with one or two at the +8 kcal/mol mark likely due to removed ligands and crystalline waters (implicit solvent is an approximation of bulk solvent, not frozen waters). Two things are worth remembering, the ScoreFunction
does not store any data, but Pose.energies
does. But both can access this data. Personally I prefer the latter, but the former has weights which is handy.
prc: ModuleType = pyrosetta.rosetta.core pr_scoring: ModuleType = pyrosetta.rosetta.core.scoring scorefxn = pyrosetta.ScoreFunction = pyrosetta.get_fa_scorefxn() # showcase weights: weights: Dict[str, float] = {w.name: scorefxn.get_weight(w) for w in scorefxn.get_nonzero_weighted_scoretypes()} print(weights) # get per residue scores: res_scores = [] for i in range(1, monomer.total_residue() + 1): v = pru.vector1_bool(pose.total_residue()) v[i] = 1 res_scores.append(scorefxn.get_sub_score(pose, v)) # some magic can be done with the annoying EMapVector v: pr_scoring.EMapVector = scorefxn.weights() print( v[pr_scoring.total_score], v[pr_scoring.fa_sol] )
Where using pose.energies
is a lot easier
scorefxn(pose) # fills pose.energies energies: pr_scoring.Energies = pose.energies() typed_energies: Dict[str, float] = energies.active_total_energies() a: npt.NDArray = energies.residue_total_energies_array() energy = pd.DataFrame(a) # amazingly civilised # alternative a specific residue can be queried via EMapVector as above v: pr_scoring.EMapVector = energies.residue_total_energies(1)
Do note that due to the fact that it takes two particles for a two body score (e.g. hydrogen bond), there’s a function to toggle whether these are halved or not (cf. this post).
scorefxn.weights() # Create the EnergyMap emopts = pr_scoring.methods.EnergyMethodOptions(scorefxn.energy_method_options()) emopts.hbond_options().decompose_bb_hb_into_pair_energies(True) scorefxn.set_energy_method_options(emopts)
Parenthetically, if the pd.DataFrame(pose.energies().residue_total_energies_array())
was not saved, but is needed again, here’s a function to read the junk on a scored dumped PDB.
def rosetta_pdb_to_df(pdbblock: str) -> pd.DataFrame: parsable = False _data = [] for line in pdbblock.split('\n'): if '#BEGIN_POSE_ENERGIES_TABLE' in line: parsable = True continue elif '#END_POSE_ENERGIES_TABLE' in line: break elif not parsable: continue parts = line.strip().split() if parts[0] == 'label': _data.append(parts) elif parts[0] == 'weights': _data.append([parts[0]] + list(map(float, parts[1:-1])) + [float('nan')]) else: _data.append([parts[0]] + list(map(float, parts[1:]))) data = pd.DataFrame(_data) data.columns = data.iloc[0] data = data.iloc[1:].copy() return data
Interface energy
The thing we want to have is a nice interface. There are two ways to this. One is civilised, but prone to segfaults: the InterfaceAnalyzerMover
mover.
def score_interface(complex: Union[pyrosetta.Pose, Sequence[pyrosetta.Pose]], interface: str): if isinstance(complex, Sequence): _complex = complex[0].clone() for c in complex[1:]: add_chain(_complex, c) complex = _complex ia = pyrosetta.rosetta.protocols.analysis.InterfaceAnalyzerMover(interface) ia.apply(complex) return {'complex_energy': ia.get_complex_energy(), 'separated_interface_energy': ia.get_separated_interface_energy(), 'complexed_sasa': ia.get_complexed_sasa(), 'crossterm_interface_energy': ia.get_crossterm_interface_energy(), 'interface_dG': ia.get_interface_dG(), 'interface_delta_sasa': ia.get_interface_delta_sasa()}
Like many other movers, no ligands are allowed.
Interface is a chain letter form, say A_BC
. This crashes out above 3 chains I believe.
The other way is scoring the monomers in isolation (pose.split_chain
) or translating a silly distance after having reset the pose.energies
but preferably repacked the surface layer.
Changed sequence
As said above, how many residues changed when tweaked can be a proxy for something being not quite right.
Number of neighbours
pose2pdb = oligomer.pdb_info().pose2pdb chainA_sele = pr_res.ChainSelector('A') # boolean vector: v = pr_res.NeighborhoodResidueSelector(chainA_sele, distance, False).apply(oligomer) # vector of pose idxs: close_residues = prc.select.get_residues_from_subset(v) # array of PDB numbers print( [pose2pdb(r) for r in close_residues] )
In another blog post I go through the two different residue neighbourhood selectors, but briefly, NeighborhoodResidueSelector
is centroid (roughly the beta carbon), while CloseContactResidueSelector
is closest atom.
cc_sele = pr_res.CloseContactResidueSelector() cc_sele.central_residue_group_selector(chainA_sele) cc_sele.threshold(3) # Å
Hydrophobicity & co.
In a tool developed in OPIG, Therapeutic Antibody Profiler, an set of metrics where characterised, such as patches of surface hydrophobicity etc. which most likely would be well worth using in protein binder design. (I’ve not used these yet so don’t have a snippet to share).
Movement
One may be after rigidity. An MD run is best for that, but pyrosetta can do (discussed in detail in another blog post) via the BackrubMover
mover.
def movement(original: pyrosetta.Pose, trials: int = 100, temperature: int = 1.0, replicate_number: int = 20) -> List[float]: scorefxn = pyrosetta.get_fa_scorefxn() backrub = pyrosetta.rosetta.protocols.backrub.BackrubMover() monégasque = pyrosetta.rosetta.protocols.monte_carlo.GenericMonteCarloMover(maxtrials=trials, max_accepted_trials=trials, # gen.max_accepted_trials() = 0 task_scaling=5, # gen.task_scaling() mover=backrub, temperature=temperature, sample_type='low', drift=True) monégasque.set_scorefxn(scorefxn) # find most deviant rs = [] for i in range(replicate_number): variant = original.clone() monégasque.apply(variant) if monégasque.accept_counter() > 0: rs.append(pr_scoring.bb_rmsd(variant, original)) else: rs.append(float('nan')) return rs
Conclusion
Hopefully this exhaustive, and exhausting, overview of the movers and functions in PyRosetta, has adequately showcases what operations can be done in PyRosetta to better tune and analyse variants from RFdiffusion.
Footnote: ProteinMPNN helper as API functions
As mentioned, my designs had different lengths and I wanted to create the definitions for ProteinMPNN via my script as it was reading from tar.gz archives and other details. But briefly, I parsed the RFdiffusion skeletons thusly:
no_fix = [] definitions = [] global_fixed_chains: Dict[str, List[List[str]]] = {} global_fixed_positions = {} for path in paths: # ## Load data name = path.stem pdbblock = get_ATOM_only(path.read_text()) # ## Parse definition (i.e. coordinates) definition = parse_PDBblock(pdbblock, name) definitions.append(definition) # ## Fixed chain fixed_chains = define_fixed_chains(definition, designed_chain_list='A') # chain A is designed global_fixed_chains[name] = fixed_chains # ## Fixed pos sequence = get_chainA_sequence(pdbblock) # assuming the sequence to change is chain A # The skeleton will have stretches of glycines where the design was made: masked = re.sub(r'(G{3,})', lambda match: '_' * len(match.group(1)), sequence) fixed_list = [[i for i, r in enumerate(masked) if r == '_']] fixed_positions = define_unfixed_positions(definition, ['A'], fixed_list) global_fixed_positions[name] = fixed_positions # write out definitions, global_fixed_chains, global_fixed_positions with open(chains_definitions_path, 'w') as fh: # only chains_definitions.jsonl is a JSONL for definition in definitions: fh.write(json.dumps(definition) + '\n') with open(fixed_chains_path, 'w') as fh: json.dump(global_fixed_chains, fh) with open(fixed_positions_path, 'w') as fh: json.dump(global_fixed_positions, fh)
Where the functions are:
""" This is a minor refactoring of the original code, with the following changes: `parse_PDBblock(pdbblock: str, name: str, chain_alphabet, ca_only=False)` accepts a PDB block ``pdbblock`` and a name ``name`` and returns a dictionary that forms a line for the JSONL. The code is mostly kept as was, with minor chances. """ import json from typing import List, Sequence, Dict, Any import numpy as np alpha_1 = list("ARNDCQEGHILKMFPSTWYV-") states = len(alpha_1) alpha_3 = ['ALA', 'ARG', 'ASN', 'ASP', 'CYS', 'GLN', 'GLU', 'GLY', 'HIS', 'ILE', 'LEU', 'LYS', 'MET', 'PHE', 'PRO', 'SER', 'THR', 'TRP', 'TYR', 'VAL', 'GAP'] aa_1_N = {a: n for n, a in enumerate(alpha_1)} aa_3_N = {a: n for n, a in enumerate(alpha_3)} aa_N_1 = {n: a for n, a in enumerate(alpha_1)} aa_1_3 = {a: b for a, b in zip(alpha_1, alpha_3)} aa_3_1 = {b: a for a, b in zip(alpha_1, alpha_3)} def get_ATOM_only(pdbblock: str) -> str: """ This gets all ATOM, regardless of name and chain """ return '\n'.join([line for line in pdbblock.splitlines() if line.startswith('ATOM')]) def get_chainA_sequence(pdbblock: str) -> str: sequence = '' residues_seen = set() for line in pdbblock.splitlines(): if line.startswith("ATOM") and " CA " in line and " A " in line: res_info = line[17:26] # Residue name and number for uniqueness if res_info not in residues_seen: residues_seen.add(res_info) res_name = line[17:20].strip() sequence += three_to_one.get(res_name, '?') return sequence three_to_one = { 'ALA': 'A', 'CYS': 'C', 'ASP': 'D', 'GLU': 'E', 'PHE': 'F', 'GLY': 'G', 'HIS': 'H', 'ILE': 'I', 'LYS': 'K', 'LEU': 'L', 'MET': 'M', 'ASN': 'N', 'PRO': 'P', 'GLN': 'Q', 'ARG': 'R', 'SER': 'S', 'THR': 'T', 'VAL': 'V', 'TRP': 'W', 'TYR': 'Y' } def AA_to_N(x): # ["ARND"] -> [[0,1,2,3]] x = np.array(x); if x.ndim == 0: x = x[None] return [[aa_1_N.get(a, states - 1) for a in y] for y in x] def N_to_AA(x): # [[0,1,2,3]] -> ["ARND"] x = np.array(x); if x.ndim == 1: x = x[None] return ["".join([aa_N_1.get(a, "-") for a in y]) for y in x] def inner_parse_PDBblock(pdbblock, atoms=['N', 'CA', 'C'], chain=None) -> tuple: ''' input: pdb_filename = PDB filename atoms = atoms to extract (optional) output: (length, atoms, coords=(x,y,z)), sequence ''' xyz, seq, min_resn, max_resn = {}, {}, 1e6, -1e6 for line in pdbblock.split("\n"): if line[:6] == "HETATM" and line[17:17 + 3] == "MSE": line = line.replace("HETATM", "ATOM ") line = line.replace("MSE", "MET") if line[:4] == "ATOM": ch = line[21:22] if ch == chain or chain is None: atom = line[12:12 + 4].strip() resi = line[17:17 + 3] resn = line[22:22 + 5].strip() x, y, z = [float(line[i:(i + 8)]) for i in [30, 38, 46]] if resn[-1].isalpha(): resa, resn = resn[-1], int(resn[:-1]) - 1 else: resa, resn = "", int(resn) - 1 # resn = int(resn) if resn < min_resn: min_resn = resn if resn > max_resn: max_resn = resn if resn not in xyz: xyz[resn] = {} if resa not in xyz[resn]: xyz[resn][resa] = {} if resn not in seq: seq[resn] = {} if resa not in seq[resn]: seq[resn][resa] = resi if atom not in xyz[resn][resa]: xyz[resn][resa][atom] = np.array([x, y, z]) # ^^ end of xyz loop # convert to numpy arrays, fill in missing values seq_, xyz_ = [], [] try: for resn in range(min_resn, max_resn + 1): if resn in seq: for k in sorted(seq[resn]): seq_.append(aa_3_N.get(seq[resn][k], 20)) else: seq_.append(20) if resn in xyz: for k in sorted(xyz[resn]): for atom in atoms: if atom in xyz[resn][k]: xyz_.append(xyz[resn][k][atom]) else: xyz_.append(np.full(3, np.nan)) else: for atom in atoms: xyz_.append(np.full(3, np.nan)) return np.array(xyz_).reshape(-1, len(atoms), 3), N_to_AA(np.array(seq_)) except TypeError: return 'no_chain', 'no_chain' def get_chain_alphabet(): init_alphabet = ['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z'] extra_alphabet = [str(item) for item in list(np.arange(300))] return init_alphabet + extra_alphabet def parse_PDBblock(pdbblock: str, name: str, ca_only=False): chain_alphabet = get_chain_alphabet() my_dict = {} s = 0 concat_seq = '' for letter in chain_alphabet: if ca_only: sidechain_atoms = ['CA'] else: sidechain_atoms = ['N', 'CA', 'C', 'O'] xyz, seq = inner_parse_PDBblock(pdbblock=pdbblock, atoms=sidechain_atoms, chain=letter) if type(xyz) != str: concat_seq += seq[0] my_dict['seq_chain_' + letter] = seq[0] coords_dict_chain = {} if ca_only: coords_dict_chain['CA_chain_' + letter] = xyz.tolist() else: coords_dict_chain['N_chain_' + letter] = xyz[:, 0, :].tolist() coords_dict_chain['CA_chain_' + letter] = xyz[:, 1, :].tolist() coords_dict_chain['C_chain_' + letter] = xyz[:, 2, :].tolist() coords_dict_chain['O_chain_' + letter] = xyz[:, 3, :].tolist() my_dict['coords_chain_' + letter] = coords_dict_chain s += 1 my_dict['name'] = name my_dict['num_of_chains'] = s my_dict['seq'] = concat_seq if s < len(chain_alphabet): return my_dict else: raise Exception('Too many chains') def define_fixed_chains(chains_definition: Dict[str, Any], designed_chain_list=('A',)): """ The fixed chain definition file is a JSON dictionary of name to tuple/list of two: designed_chain_list and fixed_chain_list """ all_chain_list = [item[-1:] for item in list(chains_definition) if item[:9] == 'seq_chain'] # ['A','B', 'C',...] fixed_chain_list = [letter for letter in all_chain_list if letter not in designed_chain_list] # fix/do not redesign these chains return list(designed_chain_list), fixed_chain_list def define_global_fixed_chains(chains_definitions, global_designed_chain_list=('A',)): return {chains_definition['name']: define_fixed_chains(chains_definition, global_designed_chain_list) for chains_definition in chains_definitions} def define_fixed_positions(chains_definition: Dict[str, Any], designed_chain_list: Sequence[str], fixed_list: Sequence[Sequence[int]]): all_chain_list = [item[-1:] for item in list(chains_definition) if item[:9] == 'seq_chain'] fixed_position_dict = {} for i, chain in enumerate(designed_chain_list): fixed_position_dict[chain] = fixed_list[i] for chain in all_chain_list: if chain not in designed_chain_list: fixed_position_dict[chain] = [] return fixed_position_dict def define_unfixed_positions(chains_definition, designed_chain_list: Sequence[str], unfixed_list: Sequence[Sequence[int]]): """ This will be an entry in a dictionary passed to ``chain_id_jsonl`` (Misnomer: it's not a JSONL, but a JSON, only ``jsonl_path`` is) :param chains_definition: :param designed_chain_list: :param unfixed_list: :return: """ all_chain_list = [item[-1:] for item in list(chains_definition) if item[:9] == 'seq_chain'] fixed_position_dict = {} for chain in all_chain_list: seq_length = len(chains_definition[f'seq_chain_{chain}']) all_residue_list = (np.arange(seq_length) + 1).tolist() if chain not in designed_chain_list: fixed_position_dict[chain] = all_residue_list else: idx = np.argwhere(np.array(designed_chain_list) == chain)[0][0] fixed_position_dict[chain] = list(set(all_residue_list) - set(unfixed_list[idx])) return fixed_position_dict