PPIformer / app.py
Anton Bushuiev
Add link to cpu version of the web
688dca6
# print("""
# __ __ _ ___ _ _ _____ _____ _ _ _ _ _ ____ _____
# | \/ | / \ |_ _| \ | |_ _| ____| \ | | / \ | \ | |/ ___| ____|
# | |\/| | / _ \ | || \| | | | | _| | \| | / _ \ | \| | | | _|
# | | | |/ ___ \ | || |\ | | | | |___| |\ |/ ___ \| |\ | |___| |___
# |_| |_/_/ \_\___|_| \_| |_| |_____|_| \_/_/ \_\_| \_|\____|_____|
# ____ ____ _____ _ _ __
# | __ )| _ \| ____| / \ | |/ /
# | _ \| |_) | _| / _ \ | ' /
# | |_) | _ <| |___ / ___ \| . \
# |____/|_| \_\_____/_/ \_\_|\_\
# """)
import os
# os.system("pip uninstall -y gradio")
# os.system("pip install gradio==3.50.2")
# os.system("pip uninstall -y spaces")
# os.system("pip install spaces==0.8")
os.system("pip uninstall -y torch")
os.system("pip install torch==2.0.1")
import sys
import copy
import random
import tempfile
import shutil
import logging
from pathlib import Path
from functools import partial
import spaces
import gradio as gr
import torch
import numpy as np
import pandas as pd
from Bio.PDB.Polypeptide import protein_letters_3to1
from biopandas.pdb import PandasPdb
from colour import Color
from colour import RGB_TO_COLOR_NAMES
from mutils.proteins import AMINO_ACID_CODES_1
from mutils.pdb import download_pdb
from mutils.mutations import Mutation
from ppiref.extraction import PPIExtractor
from ppiref.utils.ppi import PPIPath
from ppiref.utils.residue import Residue
from ppiformer.tasks.node import DDGPPIformer
from ppiformer.utils.api import download_from_zenodo
from ppiformer.utils.api import predict_ddg as predict_ddg_
from ppiformer.utils.torch import fill_diagonal
from ppiformer.definitions import PPIFORMER_WEIGHTS_DIR
import pkg_resources
import sys
def print_package_versions():
installed_packages = sorted([f"{pkg.key}=={pkg.version}" for pkg in pkg_resources.working_set])
print("Installed packages and their versions:")
for package in installed_packages:
print(package)
print("\nPython version:")
print(sys.version)
print_package_versions()
logging.basicConfig(
level=logging.INFO,
format='%(asctime)s - %(levelname)s - %(message)s',
handlers=[logging.StreamHandler(sys.stdout)]
)
random.seed(0)
@spaces.GPU
def predict_ddg(models, ppi, muts, return_attn):
device = 'cuda' if torch.cuda.is_available() else 'cpu'
print(f"[INFO] Device on prediction: {device}")
models = [model.to(device) for model in models]
if return_attn:
ddg_pred, attns = predict_ddg_(models, ppi, muts, return_attn=return_attn)
return ddg_pred.detach().cpu(), attns.detach().cpu()
else:
ddg_pred = predict_ddg_(models, ppi, muts, return_attn=return_attn)
return ddg_pred.detach().cpu()
def process_inputs(inputs, temp_dir):
pdb_code, pdb_path, partners, muts, muts_path = inputs
# Check inputs
if not pdb_code and not pdb_path:
raise gr.Error("PPI structure not specified.")
if pdb_code and pdb_path:
gr.Warning("Both PDB code and PDB file specified. Using PDB file.")
if not partners:
raise gr.Error("Partners not specified.")
if not muts and not muts_path:
raise gr.Error("Mutations not specified.")
if muts and muts_path:
gr.Warning("Both mutations and mutations file specified. Using mutations file.")
# Prepare PDB input
if pdb_path:
# convert file name to PPIRef format
new_pdb_path = temp_dir / f"pdb/{pdb_path.name.replace('_', '-')}"
new_pdb_path.parent.mkdir(parents=True, exist_ok=True)
shutil.copy(str(pdb_path), str(new_pdb_path))
pdb_path = new_pdb_path
pdb_path = Path(pdb_path)
else:
try:
pdb_code = pdb_code.strip().lower()
pdb_path = temp_dir / f'pdb/{pdb_code}.pdb'
download_pdb(pdb_code, path=pdb_path)
except:
raise gr.Error("PDB download failed.")
# Parse partners
partners = list(map(lambda x: x.strip(), partners.split(',')))
# Add partners to file name
pdb_path = pdb_path.rename(pdb_path.with_stem(f"{pdb_path.stem}-{'-'.join(partners)}"))
# Extract PPI into temp dir
try:
ppi_dir = temp_dir / 'ppi'
extractor = PPIExtractor(out_dir=ppi_dir, nest_out_dir=True, join=True, radius=10.0)
extractor.extract(pdb_path, partners=partners)
ppi_path = PPIPath.construct(ppi_dir, pdb_path.stem, partners)
except:
raise gr.Error("PPI extraction failed.")
# Prepare mutations input
if muts_path:
muts_path = Path(muts_path)
muts = muts_path.read_text()
# Check mutations
# Basic format
try:
muts = [Mutation.from_str(m) for m in muts.strip().split(';') if m.strip()]
except Exception as e:
raise gr.Error(f'Mutations parsing failed: {e}')
# Partners
for mut in muts:
for pmut in mut.muts:
if pmut.chain not in partners:
raise gr.Error(f'Chain of point mutation {pmut} is not in the list of partners {partners}.')
# Consistency with provided .pdb
muts_on_interface = []
for mut in muts:
if mut.wt_in_pdb(ppi_path):
val = True
elif mut.wt_in_pdb(pdb_path):
val = False
else:
raise gr.Error(f'Wild-type of mutation {mut} is not in the provided .pdb file.')
muts_on_interface.append(val)
muts = [str(m) for m in muts]
return pdb_path, ppi_path, muts, muts_on_interface
def plot_3dmol(pdb_path, ppi_path, mut, attn, attn_mut_id=0):
# NOTE 3DMol.js adapted from https://huggingface.co/spaces/huhlim/cg2all/blob/main/app.py
# Read PDB for 3Dmol.js
with open(pdb_path, "r") as fp:
lines = fp.readlines()
mol = ""
for l in lines:
mol += l
mol = mol.replace("OT1", "O ")
mol = mol.replace("OT2", "OXT")
# Read PPI to customize 3Dmol.js visualization
ppi_df = PandasPdb().read_pdb(ppi_path).df['ATOM']
ppi_df = ppi_df.groupby(list(Residue._fields)).apply(lambda df: df[df['atom_name'] == 'CA'].iloc[0]).reset_index(drop=True)
ppi_df['id'] = ppi_df.apply(lambda row: ':'.join([row['residue_name'], row['chain_id'], str(row['residue_number']), row['insertion']]), axis=1)
ppi_df['id'] = ppi_df['id'].apply(lambda x: x[:-1] if x[-1] == ':' else x)
muts_id = Mutation.from_str(mut).wt_to_graphein() # flatten ids of all sp muts
ppi_df['mutated'] = ppi_df.apply(lambda row: row['id'] in muts_id, axis=1)
# Prepare attention coeffictients per residue (normalized sum of direct attention from mutated residues)
attn = torch.nan_to_num(attn, nan=1e-10)
attn_sub = attn[:, attn_mut_id, 0, :, 0, :, :, :] # models, layers, heads, tokens, tokens
idx_mutated = torch.from_numpy(ppi_df.index[ppi_df['mutated']].to_numpy())
attn_sub = fill_diagonal(attn_sub, 1e-10)
attn_mutated = attn_sub[..., idx_mutated, :]
attn_mutated.shape
attns_per_token = torch.sum(attn_mutated, dim=(0, 1, 2, 3))
attns_per_token = (attns_per_token - attns_per_token.min()) / (attns_per_token.max() - attns_per_token.min())
attns_per_token += 1e-10
ppi_df['attn'] = attns_per_token.numpy()
chains = ppi_df.sort_values('attn', ascending=False)['chain_id'].unique()
# Customize 3Dmol.js visualization https://3dmol.csb.pitt.edu/doc/
styles = []
zoom_atoms = []
# Cartoon chains
preferred_colors = ['LimeGreen', 'HotPink', 'RoyalBlue']
all_colors = [c[0] for c in RGB_TO_COLOR_NAMES.values()]
all_colors = [c for c in all_colors if c not in preferred_colors + ['Black', 'White']]
random.shuffle(all_colors)
all_colors = preferred_colors + all_colors
all_colors = [Color(c) for c in all_colors]
chain_to_color = dict(zip(chains, all_colors))
for chain in chains:
styles.append([{"chain": chain}, {"cartoon": {"color": chain_to_color[chain].hex_l, "opacity": 0.6}}])
# Stick PPI and atoms for zoom
# TODO Insertions
for _, row in ppi_df.iterrows():
color = copy.deepcopy(chain_to_color[row['chain_id']])
color.saturation = row['attn']
color = color.hex_l
if row['mutated']:
styles.append([
{'chain': row['chain_id'], 'resi': str(row['residue_number'])},
{'stick': {'color': 'red', 'radius': 0.2, 'opacity': 1.0}}
])
zoom_atoms.append(row['atom_number'])
else:
styles.append([
{'chain': row['chain_id'], 'resi': str(row['residue_number'])},
{'stick': {'color': color, 'radius': row['attn'] / 5, 'opacity': row['attn']}}
])
# Convert style dicts to JS lines
styles = ''.join(['viewer.addStyle(' + ', '.join([str(s).replace("'", '"') for s in dcts]) + ');\n' for dcts in styles])
# Convert zoom atoms to 3DMol.js selection and add labels for mutated residues
zoom_animation_duration = 500
sel = '{\"or\": [' + ', '.join(["{\"serial\": " + str(a) + "}" for a in zoom_atoms]) + ']}'
zoom = 'viewer.zoomTo(' + sel + ',' + f'{zoom_animation_duration});'
for atom in zoom_atoms:
sel = '{\"serial\": ' + str(atom) + '}'
row = ppi_df[ppi_df['atom_number'] == atom].iloc[0]
label = protein_letters_3to1[row['residue_name']] + row['chain_id'] + str(row['residue_number']) + row['insertion']
styles += 'viewer.addLabel(' + f"\"{label}\"," + "{fontSize:16, fontColor:\"red\", backgroundOpacity: 0.0}," + sel + ');\n'
# Construct 3Dmol.js visualization script embedded in HTML
html = (
"""<!DOCTYPE html>
<html>
<head>
<meta http-equiv="content-type" content="text/html; charset=UTF-8" />
<style>
body{
font-family:sans-serif
}
.mol-container {
width: 100%;
height: 600px;
position: relative;
}
.mol-container select{
background-image:None;
}
</style>
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.3/jquery.min.js" integrity="sha512-STof4xm1wgkfm7heWqFJVn58Hm3EtS31XFaagaa8VMReCXAkQnJZ+jEy8PCC/iT18dFy95WcExNHFTqLyp72eQ==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>
<script src="https://3Dmol.csb.pitt.edu/build/3Dmol-min.js"></script>
</head>
<body>
<div id="container" class="mol-container"></div>
<script>
let pdb = `"""
+ mol
+ """`
$(document).ready(function () {
let element = $("#container");
let config = { backgroundColor: "white" };
let viewer = $3Dmol.createViewer(element, config);
viewer.addModel(pdb, "pdb");
viewer.setStyle({"model": 0}, {"ray_opaque_background": "off"}, {"stick": {"color": "lightgrey", "opacity": 0.5}});
"""
+ styles
+ zoom
+ """
viewer.render();
})
</script>
</body></html>"""
)
return f"""<iframe style="width: 100%; height: 600px" name="result" allow="midi; geolocation; microphone; camera;
display-capture; encrypted-media;" sandbox="allow-modals allow-forms
allow-scripts allow-same-origin allow-popups
allow-top-navigation-by-user-activation allow-downloads" allowfullscreen=""
allowpaymentrequest="" frameborder="0" srcdoc='{html}'></iframe>"""
def predict(models, temp_dir, *inputs):
logging.info('Starting prediction')
# Process input
pdb_path, ppi_path, muts, muts_on_interface = process_inputs(inputs, temp_dir)
# Create dataframe
df = pd.DataFrame({
'Mutation': muts,
'ddG [kcal/mol]': len(muts) * [np.nan],
'10A Interface': muts_on_interface,
'Attn Id': len(muts) * [np.nan],
})
# Show warning if some mutations are not on the interface
muts_not_on_interface = df[~df['10A Interface']]['Mutation'].tolist()
n_muts_not_on_interface = len(muts_not_on_interface)
if n_muts_not_on_interface:
n_muts_warn = 5
muts_not_on_interface = ';'.join(muts_not_on_interface[:n_muts_warn])
if n_muts_not_on_interface > n_muts_warn:
muts_not_on_interface += f'... (and {n_muts_not_on_interface - n_muts_warn} more)'
gr.Warning((
f"{muts_not_on_interface} {'is' if n_muts_not_on_interface == 1 else 'are'} not on the interface. "
f"The model will predict the effect{'s' if n_muts_not_on_interface > 1 else ''} of "
f"mutation{'s' if n_muts_not_on_interface > 1 else ''} on the whole complex. "
f"This may lead to less accurate predictions."
))
logging.info('Inputs processed')
# Predict using interface for mutations on the interface and using the whole complex otherwise
attn_ppi, attn_pdb = None, None
for df_sub, path in [
[df[df['10A Interface']], ppi_path],
[df[~df['10A Interface']], pdb_path]
]:
if not len(df_sub):
continue
# Predict
try:
ddg, attn = predict_ddg(models, path, df_sub['Mutation'].tolist(), return_attn=True)
except Exception as e:
print(f"Prediction failed. {str(e)}")
raise gr.Error(f"Prediction failed. {str(e)}")
ddg = ddg.detach().numpy().tolist()
logging.info(f'Predictions made for {path}')
# Update dataframe and attention tensor
idx = df_sub.index
df.loc[idx, 'ddG [kcal/mol]'] = ddg
df.loc[idx, 'Attn Id'] = np.arange(len(idx))
if path == ppi_path:
attn_ppi = attn
else:
attn_pdb = attn
df['Attn Id'] = df['Attn Id'].astype(int)
# Round ddG values
df['ddG [kcal/mol]'] = df['ddG [kcal/mol]'].round(3)
# Create PPI-specific dropdown
dropdown = gr.Dropdown(
df['Mutation'].tolist(), value=df['Mutation'].iloc[0],
interactive=True, visible=True, label="Mutation to visualize",
)
# Predefine plot arguments for all dropdown choices
dropdown_choices_to_plot_args = {
mut: (
pdb_path,
ppi_path if df[df['Mutation'] == mut]['10A Interface'].iloc[0] else pdb_path,
mut,
attn_ppi if df[df['Mutation'] == mut]['10A Interface'].iloc[0] else attn_pdb,
df[df['Mutation'] == mut]['Attn Id'].iloc[0]
)
for mut in df['Mutation']
}
# Create dataframe file
path = 'ppiformer_ddg_predictions.csv'
if n_muts_not_on_interface:
df = df[['Mutation', 'ddG [kcal/mol]', '10A Interface']]
df.to_csv(path, index=False)
df = gr.Dataframe(
value=df,
headers=['Mutation', 'ddG [kcal/mol]', '10A Interface'],
datatype=['str', 'number', 'bool'],
col_count=(3, 'fixed'),
)
else:
df = df[['Mutation', 'ddG [kcal/mol]']]
df.to_csv(path, index=False)
df = gr.Dataframe(
value=df,
headers=['Mutation', 'ddG [kcal/mol]'],
datatype=['str', 'number'],
col_count=(2, 'fixed'),
)
logging.info('Prediction results prepared')
return df, path, dropdown, dropdown_choices_to_plot_args
def update_plot(dropdown, dropdown_choices_to_plot_args):
return plot_3dmol(*dropdown_choices_to_plot_args[dropdown])
app = gr.Blocks(theme=gr.themes.Default(primary_hue="green", secondary_hue="pink"))
with app:
# Input GUI
gr.Markdown(value="""
# PPIformer Web
### Computational Design of Protein-Protein Interactions
""")
gr.Image("assets/readme-dimer-close-up.png")
gr.Markdown(value="""
📣 **New!** The model now runs on [Zero GPU](https://huggingface.co/zero-gpu-explorers), which is currently in beta. Please see the [CPU-only version of PPIformer Web](https://huggingface.co/spaces/anton-bushuiev/PPIformer-CPU) if you experience any issues.
[PPIformer](https://github.com/anton-bushuiev/PPIformer/tree/main) is a state-of-the-art predictor of the effects of mutations
on protein-protein interactions (PPIs), as quantified by the binding free energy changes (ddG). PPIformer was shown to successfully
identify known favourable mutations of the [staphylokinase thrombolytics](https://pubmed.ncbi.nlm.nih.gov/10942387/)
and a [human antibody](https://www.pnas.org/doi/10.1073/pnas.2122954119) against the SARS-CoV-2 spike protein. The model was pre-trained
on the [PPIRef](https://github.com/anton-bushuiev/PPIRef)
dataset via a coarse-grained structural masked modeling and fine-tuned on the [SKEMPI v2.0](https://life.bsc.es/pid/skempi2) dataset via log odds.
Please see more details in [our ICLR 2024 paper](https://arxiv.org/abs/2310.18515).
**Inputs.** To use PPIformer on your data, please specify the PPI structure (PDB code or .pdb file), interacting proteins of interest
(chain codes in the file) and mutations (semicolon-separated list or file with mutations in the
[standard format](https://foldxsuite.crg.eu/parameter/mutant-file): wild-type residue, chain, residue number, mutant residue).
For inspiration, you can use one of the examples below: click on one of the rows to pre-fill the inputs. After specifying the inputs,
press the button to predict the effects of mutations on the PPI. Currently the model runs on CPU, so the predictions may take a few minutes.
**Outputs.** After making a prediction with the model, you will see binding free energy changes for each mutation (ddG values in kcal/mol).
A more negative value indicates an improvement in affinity, whereas a more positive value means a reduction in affinity.
Below you will also see a 3D visualization of the PPI with wild types of mutated residues highlighted in red. The visualization additionally shows
the attention coefficients of the model for the nearest neighboring residues, which quantifies the contribution of the residues
to the predicted ddG value. The brighter and thicker a residue is, the more attention the model paid to it.
""")
with gr.Row(equal_height=True):
with gr.Column():
gr.Markdown("## PPI structure")
with gr.Row(equal_height=True):
pdb_code = gr.Textbox(placeholder="1BUI", label="PDB code", info="Protein Data Bank identifier for the structure (https://www.rcsb.org/)")
partners = gr.Textbox(placeholder="A,B,C", label="Partners", info="Protein chain identifiers in the PDB file forming the PPI interface (two or more)")
pdb_path = gr.File(file_count="single", label="Or .pdb file instead of PDB code (your structure will only be used for this prediction and not stored anywhere)")
with gr.Column():
gr.Markdown("## Mutations")
muts = gr.Textbox(placeholder="SC16A;FC47A;SC16A,FC47A", label="List of (multi-point) mutations", info="SC16A;FC47A;SC16A,FC47A for three mutations: serine to alanine at position 16 in chain C, phenylalanine to alanine at position 47 in chain C, and their double-point combination")
muts_path = gr.File(file_count="single", label="Or file with mutations")
examples = gr.Examples(
examples=[
["1BUI", "A,B,C", "SC16A,FC47A;SC16A;FC47A"],
["3QIB", "A,B,P,C,D", "YP7F,TP12S;YP7F;TP12S"],
["1KNE", "A,P", ';'.join([f"TP6{a}" for a in AMINO_ACID_CODES_1])]
],
inputs=[pdb_code, partners, muts],
label="Examples (click on a line to pre-fill the inputs)",
cache_examples=False
)
# Predict GUI
predict_button = gr.Button(value="Predict effects of mutations on PPI", variant="primary")
# Output GUI
gr.Markdown("## Predictions")
df_file = gr.File(label="Download predictions as .csv", interactive=False, visible=True)
df = gr.Dataframe(
headers=["Mutation", "ddG [kcal/mol]"],
datatype=["str", "number"],
col_count=(2, "fixed"),
)
dropdown = gr.Dropdown(interactive=True, visible=False)
dropdown_choices_to_plot_args = gr.State([])
plot = gr.HTML()
# Bottom info box
gr.Markdown(value="""
<br/>
## About this web
**Use cases**. The predictor can be used in: (i) Drug Discovery for the development of novel drugs and vaccines for various diseases such as cancer,
neurodegenerative disorders, and infectious diseases, (ii) Biotechnological Applications to develop new biocatalysts for biofuels,
industrial chemicals, and pharmaceuticals (iii) Therapeutic Protein Design to develop therapeutic proteins with enhanced stability,
specificity, and efficacy, and (iv) Mechanistic Studies to gain insights into fundamental biological processes, such as signal transduction,
gene regulation, and immune response.
**Acknowledgement**. Please, use the following citation to acknowledge the use of our service. The web server is provided free of charge for non-commercial use.
> Bushuiev, Anton, Roman Bushuiev, Petr Kouba, Anatolii Filkin, Marketa Gabrielova, Michal Gabriel, Jiri Sedlar, Tomas Pluskal, Jiri Damborsky, Stanislav Mazurenko, Josef Sivic.
> "Learning to design protein-protein interactions with enhanced generalization". The Twelfth International Conference on Learning Representations (ICLR 2024).
> [https://arxiv.org/abs/2310.18515](https://arxiv.org/abs/2310.18515).
**Contact**. Please share your feedback or report any bugs through [GitHub Issues](https://github.com/anton-bushuiev/PPIformer/issues/new), or feel free to contact us directly at [anton.bushuiev@cvut.cz](mailto:anton.bushuiev@cvut.cz).
""")
gr.Image("assets/logos.png")
# Download weights from Zenodo
download_from_zenodo('weights.zip')
# Set device
device = 'cuda' if torch.cuda.is_available() else 'cpu'
print(f"[INFO] Device on start: {device}")
# Load models
models = [
DDGPPIformer.load_from_checkpoint(
PPIFORMER_WEIGHTS_DIR / f'ddg_regression/{i}.ckpt',
map_location=torch.device('cpu')
).eval()
for i in range(3)
]
models = [model.to(device) for model in models]
# Create temporary directory for storing downloaded PDBs and extracted PPIs
temp_dir_obj = tempfile.TemporaryDirectory()
temp_dir = Path(temp_dir_obj.name)
# Main logic
inputs = [pdb_code, pdb_path, partners, muts, muts_path]
outputs = [df, df_file, dropdown, dropdown_choices_to_plot_args]
predict = partial(predict, models, temp_dir)
predict_button.click(predict, inputs=inputs, outputs=outputs)
# Update plot on dropdown change
dropdown.change(update_plot, inputs=[dropdown, dropdown_choices_to_plot_args], outputs=[plot])
app.launch(allowed_paths=['./assets'])