diff --git a/frontend/client/tasks/server-docking-task.tsx b/frontend/client/tasks/server-docking-task.tsx index 10e610d..7086a77 100644 --- a/frontend/client/tasks/server-docking-task.tsx +++ b/frontend/client/tasks/server-docking-task.tsx @@ -1,4 +1,4 @@ -import { PredictionInfo } from "../prankweb-api"; +import { PredictionInfo, getApiEndpoint } from "../prankweb-api"; import { PocketData, Point3D, ServerTaskInfo, ServerTaskLocalStorageData } from "../custom-types"; import { getPocketAtomCoordinates } from "../viewer/molstar-visualise"; @@ -107,13 +107,10 @@ export async function dockingHash(pocket: string, smiles: string, exhaustiveness /** * Downloads the result of the task. - * @param smiles SMILES identifier * @param fileURL URL to download the result from - * @param pocket Pocket identifier - * @param exhaustiveness exhaustiveness value (for Autodock Vina) * @returns void */ -export async function downloadDockingResult(smiles: string, fileURL: string, pocket: string, exhaustiveness: string) { +export async function downloadDockingResult(fileURL: string) { // https://stackoverflow.com/questions/50694881/how-to-download-file-in-react-js fetch(fileURL) .then((response) => response.blob()) @@ -182,3 +179,74 @@ export async function pollForDockingTask(predictionInfo: PredictionInfo) { } return localStorage.getItem(`${predictionInfo.id}_serverTasks`); } + +/** + * Generates a bash script that can be used to run the docking task locally. + * @param smiles SMILES identifier of the ligand + * @param pocket Pocket data + * @param plugin Mol* plugin + * @param prediction Prediction info + * @returns A bash script that can be used to run the docking task locally. + */ +export function generateBashScriptForDockingTask(smiles: string, pocket: PocketData, plugin: PluginUIContext, prediction: PredictionInfo) { + const box = computeBoundingBox(plugin, pocket); + const url = getApiEndpoint(prediction.database, prediction.id).replace(".", window.location.host) + `/public/${prediction.metadata.structureName}`; + + const script = + `#!/bin/bash +# This script runs the docking task locally. +# It requires Docker to be installed on the system. + +json_content='{ + "receptor": "${prediction.metadata.structureName}", + "ligand": "ligand.smi", + "output": "output.pdbqt", + "center": { + "x": ${box.center.x}, + "y": ${box.center.y}, + "z": ${box.center.z} + }, + "size": { + "x": ${box.size.x}, + "y": ${box.size.y}, + "z": ${box.size.z} + } +}' + +echo "$json_content" > docking_parameters.json +echo "${smiles}" > ligand.smi + +wget "${url}" -O ${prediction.metadata.structureName}.gz + +gunzip ${prediction.metadata.structureName}.gz + +docker run -it -v \$\{PWD\}/:/data ghcr.io/kiarka7/dodo:latest`; + + return script; +} + +/** + * Downloads the generated bash script with the current timestamp. + * @param script Bash script + * @returns void + */ +export async function downloadDockingBashScript(script: string) { + const today = new Date(); + const filename = `docking-task-${today.toISOString()}.sh`; + + const url = window.URL.createObjectURL( + new Blob([script]), + ); + const link = document.createElement('a'); + link.href = url; + link.setAttribute( + 'download', + filename, + ); + + document.body.appendChild(link); + link.click(); + link.parentNode!.removeChild(link); + + return; +} \ No newline at end of file diff --git a/frontend/client/viewer/components/tasks-tab.tsx b/frontend/client/viewer/components/tasks-tab.tsx index 5d1f5f8..a4798e1 100644 --- a/frontend/client/viewer/components/tasks-tab.tsx +++ b/frontend/client/viewer/components/tasks-tab.tsx @@ -9,7 +9,7 @@ import { Button, Paper, Typography } from "@mui/material"; import "./tasks-tab.css"; import { PredictionInfo } from "../../prankweb-api"; -import { computeDockingTaskOnBackend } from "../../tasks/server-docking-task"; +import { computeDockingTaskOnBackend, generateBashScriptForDockingTask, downloadDockingBashScript } from "../../tasks/server-docking-task"; import { PluginUIContext } from "molstar/lib/mol-plugin-ui/context"; import { computePocketVolume } from "../../tasks/client-atoms-volume"; import { TasksTable } from "./tasks-table"; @@ -73,28 +73,39 @@ export default function TasksTab(props: { pockets: PocketData[], predictionInfo: type: TaskType.Server, name: "Docking", compute: (params, customName, pocketIndex) => { + const smiles = params[0].replaceAll(" ", ""); + + const handleInvalidDockingInput = (baseMessage: string) => { + if (baseMessage === "") { + setInvalidInputMessage(""); + return; + } + + setInvalidInputMessage(`${baseMessage} Try running the script below to run docking locally.`); + setDockingScript(generateBashScriptForDockingTask(smiles, props.pockets[pocketIndex], props.plugin, props.predictionInfo)); + }; + // check if exhaustiveness is a number const exhaustiveness = params[1].replaceAll(",", ".").replaceAll(" ", ""); - if (isNaN(parseFloat(exhaustiveness))) { - setInvalidInput(true); + if (isNaN(parseInt(exhaustiveness))) { + handleInvalidDockingInput("Exhaustiveness must be an integer."); return; } // 1-64 is the allowed range if (Number(exhaustiveness) < 1 || Number(exhaustiveness) > 64) { - setInvalidInput(true); + handleInvalidDockingInput("Exhaustiveness must be in the range 1-64."); return; } - setInvalidInput(false); - const smiles = params[0].replaceAll(" ", ""); // check if SMILES is < 300 characters, otherwise - // TODO: add an option to prepare a script that will run DODO with the ligand if (smiles.length > 300) { - setInvalidInput(true); + handleInvalidDockingInput("SMILES must be shorter than 300 characters."); return; } + handleInvalidDockingInput(""); + let savedTasks = localStorage.getItem(`${props.predictionInfo.id}_serverTasks`); if (!savedTasks) savedTasks = "[]"; const tasks: ServerTaskLocalStorageData[] = JSON.parse(savedTasks); @@ -127,7 +138,8 @@ export default function TasksTab(props: { pockets: PocketData[], predictionInfo: const [name, setName] = React.useState(""); const [parameters, setParameters] = React.useState([]); const [forceUpdate, setForceUpdate] = React.useState(0); - const [invalidInput, setInvalidInput] = React.useState(false); + const [invalidInputMessage, setInvalidInputMessage] = React.useState(""); + const [dockingScript, setDockingScript] = React.useState(""); const handleTaskTypeChange = (event: SelectChangeEvent) => { const newTask = tasks.find(task => task.id == Number(event.target.value))!; @@ -228,16 +240,17 @@ export default function TasksTab(props: { pockets: PocketData[], predictionInfo: ) } { - invalidInput && - - - Error: The task could not be created. Check the formatting. - - + invalidInputMessage === "" ? null : + + + {invalidInputMessage} + + } + {dockingScript !== "" && <> } diff --git a/frontend/client/viewer/components/tasks-table.tsx b/frontend/client/viewer/components/tasks-table.tsx index 95aa3bb..4fe5bbf 100644 --- a/frontend/client/viewer/components/tasks-table.tsx +++ b/frontend/client/viewer/components/tasks-table.tsx @@ -116,7 +116,7 @@ export function TasksTable(props: { pocket: PocketData | null, predictionInfo: P const handleResultClick = (serverTask: ServerTaskLocalStorageData) => { switch (serverTask.type) { case ServerTaskType.Docking: - downloadDockingResult(serverTask.params[0], serverTask.responseData[0].url, serverTask.pocket.toString(), serverTask.params[1]); + downloadDockingResult(serverTask.responseData[0].url); break; default: break; @@ -237,4 +237,4 @@ export function TasksTable(props: { pocket: PocketData | null, predictionInfo: P ); -} \ No newline at end of file +}; \ No newline at end of file