{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Exploring RNAcentral APIs \n",
    "In this exercise we will explore the RNAcentral APIs to find some information about the most differentially expressed RNAs in an experiment.\n",
    "\n",
    "The data for this exercise comes from this experiment: https://www.ebi.ac.uk/ena/data/view/PRJNA601750 a single-end RNA-seq dataset based on Drosophila melanogaster S2 cells sequenced under normal conditions and amino-acid starvation.\n",
    "\n",
    "The data has been analysed using support protocol 3 detailed in our 2020 Current Protocols in Bioinformatics paper, which you can read here: https://currentprotocols.onlinelibrary.wiley.com/doi/full/10.1002/cpbi.104\n",
    "\n",
    "The output is a table with URS_taxid (a unique RNA identifier) and some data about the differential expression of the RNA in this experiment. For the purposes of this webinar, we have prepared just the output, which we will analyse here. NOte - it has been slightly reduced in size to allow us to get things done in a reasonable amount of time.\n",
    "\n",
    "The first step is to download the analysis output, which you can get from our FTP site\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "vscode": {
     "languageId": "shellscript"
    }
   },
   "outputs": [],
   "source": [
    "## Get all data we will need from RNAcentral FTP\n",
    "!wget -O deseq2_results.csv https://ftp.ebi.ac.uk/pub/databases/RNAcentral/outreach/20231120-webinar/deseq2_results.csv"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Next, we will import some useful python libraries\n",
    "\n",
    "- `pandas`: The go-to dataframe library in python.\n",
    "- `requests`: To be able to make requests to the RNAcentral API\n",
    "- `ratelimit`: The RNAcentral API has a limit of 20 requests per second, we will try to respect that\n",
    "- `tqdm`: Gives us some nice progressbars and other feedback on progress"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "if 'google.colab' in str(get_ipython()):\n",
    "  import pip\n",
    "  pip.main(['install', 'ratelimit'])\n",
    "\n",
    "import pandas as pd\n",
    "pd.set_option('display.width', None)\n",
    "import requests\n",
    "from ratelimit import limits\n",
    "from tqdm.autonotebook import tqdm \n",
    "tqdm.pandas()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "de_data = pd.read_csv(\"deseq2_results.csv\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now we have the data, we can filter out only those RNAs with significant differential expression, by looking at `pvalue < 0.01`. The standard threshold for significance is `0.05`, but to cut down what we need to evaluate, we will set the bound tighter. We will visualise the dataframe again to get a feel for what's going on."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "de_data = de_data[de_data[\"pvalue\"] < 0.01]\n",
    "with pd.option_context('display.max_colwidth', 20):\n",
    "    display(de_data)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "You may recognise this as the output of the DESeq2 program.\n",
    "\n",
    "So we have 36 RNAs to look at. Lets see what type of RNAs they are by querying the RNAcentral API. We will write a small python function to process the output of the API and append columns to our DESeq2 output.\n",
    "\n",
    "Note that we try to respect the API rate limits, and we get a brand new dataframe with this data in it."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "@limits(calls=20, period=1)\n",
    "def get_rna_data(rnacentral_id):\n",
    "    url = f\"https://rnacentral.org/api/v1/rna/{rnacentral_id}\"\n",
    "    response = requests.get(url)\n",
    "    return response.json()\n",
    "\n",
    "## Query the API to get what RNAcentral knows about these RNAs\n",
    "r = de_data[\"urs_taxid\"].progress_map(get_rna_data)\n",
    "\n",
    "## Convert the results to a DataFrame\n",
    "rnacentral_data = pd.DataFrame(list(r))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "That should take less than a minute hopefully.\n",
    "\n",
    "Let's visualise the data here in the notebook. Note - you should be able to scroll sideways to look at more columns"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "with pd.option_context('display.max_colwidth', 20):\n",
    "    display(rnacentral_data)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We get a lot more data than just the type of the RNA back, including:\n",
    "- The sequence for the URS_taxid stored in RNAcentral, \n",
    "- its length\n",
    "- what species the RNA comes from \n",
    "- a brief description\n",
    "- A list of gene names for the RNA\n",
    "- The databases from which RNAcentral knows the RNA\n",
    "\n",
    "From this, we can see that many of the most differentially expressed ncRNAs are lncRNAs. Lets quantify that with a little aggregation:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "rnacentral_data.groupby(\"rna_type\")['rna_type'].count()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "So from this we can see most of the differentially expressed RNAs are indeed lncRNA, but there are a couple of interesting things - for example one mature miRNA is defferentially expressed. Let's have a look at that one:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "miRNA = rnacentral_data.loc[rnacentral_data[\"rna_type\"] == \"miRNA\"]\n",
    "with pd.option_context('display.max_colwidth', None):\n",
    "    display(miRNA)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This gives us the genes we would need to search for if we wanted to find more information, e.g. `dme-mir-4968`. Let's see what else we can get from RNAcentral's APIs. \n",
    "\n",
    "We can use the LitScan API to search for relevant publications about this RNA:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "@limits(calls=20, period=1)\n",
    "def get_publications(genes):\n",
    "    format_gene_query = ['job_id:\"{}\"'.format(gene) for gene in genes]\n",
    "    url = f\"https://www.ebi.ac.uk/ebisearch/ws/rest/rnacentral-litscan?query=entry_type:Publication AND ({' OR '.join(format_gene_query)})&fields=title,pmcid&format=json\"\n",
    "    response = requests.get(url)\n",
    "    r = response.json()\n",
    "    titles = [hit['fields']['title'][0] for hit in r['entries']]\n",
    "    pmcids = [hit['fields']['pmcid'][0] for hit in r['entries']]\n",
    "    links = [f\"https://www.ncbi.nlm.nih.gov/pmc/articles/{pmcid}/\" for pmcid in pmcids]\n",
    "\n",
    "    publication_data = pd.DataFrame({\"titles\": titles, \"pmcids\": pmcids, \"links\": links})\n",
    "    return publication_data\n",
    "\n",
    "## We can get away with this because we only have one RNA in our selection - \n",
    "# think how it would need to change if we wanted to run across a whole dataset\n",
    "publication_data = get_publications(set(list(miRNA['genes'])[0]))\n",
    "with pd.option_context('display.max_colwidth', None):\n",
    "    display(publication_data)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "So this RNA has been found in 4 publications, the titles of which you can see here, and you should be able to go and get the full text at these links.\n",
    "\n",
    "## What next?\n",
    "Hopefully this gives you an idea how to get started using the RNAcentral APIs for your own analyses. If you want to play a little more with this data, you could try getting all the papers for other RNA types identified in this analysis, or filtering to only look at those RNAs with no papers about them, or try some of the other techniques we've talked about in this webinar!"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "rnac",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.3"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}