From a73c93d5a44f27df854c7eb76d3098a86b6b5255 Mon Sep 17 00:00:00 2001 From: Joseph White Date: Fri, 1 Dec 2023 10:54:36 -0500 Subject: [PATCH 1/2] Add reports --- ACCESS/PIs and Allocations.ipynb | 312 +++++++++++++++++++++++++ ACCESS/README.md | 11 + ACCESS/User-Resource-Overlaps.ipynb | 348 ++++++++++++++++++++++++++++ 3 files changed, 671 insertions(+) create mode 100644 ACCESS/PIs and Allocations.ipynb create mode 100644 ACCESS/README.md create mode 100644 ACCESS/User-Resource-Overlaps.ipynb diff --git a/ACCESS/PIs and Allocations.ipynb b/ACCESS/PIs and Allocations.ipynb new file mode 100644 index 0000000..a891c09 --- /dev/null +++ b/ACCESS/PIs and Allocations.ipynb @@ -0,0 +1,312 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "db4061c7-2b36-4284-af31-4ae940a07768", + "metadata": { + "tags": [] + }, + "source": [ + "# XDMoD Data Analytics Framework — PIs with allocations\n", + "v1.0.0\n", + "\n", + "© 2023 University at Buffalo Center for Computational Research\n", + "\n", + "See the [xdmod-notebooks](https://github.com/ubccr/xdmod-notebooks) repository for licensing information." + ] + }, + { + "cell_type": "markdown", + "id": "77f9e969-5120-4a7a-a844-fa6fda893755", + "metadata": {}, + "source": [ + "## Introduction\n", + "The XDMoD Data Analytics Framework provides API access to the data in XDMoD via the [`xdmod_data` Python module](https://pypi.org/project/xdmod-data). This notebook provides an introductory example showing how to use the module. You will use the XDMoD API to request data, load them into a [Pandas](https://pandas.pydata.org/) [DataFrame](https://pandas.pydata.org/docs/reference/api/pandas.DataFrame.html), and generate plots. The dataset in this example contains the number of active users of [ACCESS](https://access-ci.org/)-allocated resources per day of the week over a 4-month period." + ] + }, + { + "cell_type": "markdown", + "id": "84713a67-80c4-4803-8a45-0decdb65b500", + "metadata": {}, + "source": [ + "## Install/upgrade the required modules\n", + "Run the code below to install/upgrade the modules needed to run this notebook." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "43521c0d-ca9e-47cd-ba5a-c1f9d0b83290", + "metadata": {}, + "outputs": [], + "source": [ + "import sys\n", + "! {sys.executable} -m pip install --upgrade xdmod-data python-dotenv tabulate" + ] + }, + { + "cell_type": "markdown", + "id": "56c7d25d-241b-4c65-a77b-7eca37386714", + "metadata": {}, + "source": [ + "If running that code caused a new version of Plotly to be installed/upgraded, you may need to refresh your browser window for plots to appear correctly." + ] + }, + { + "cell_type": "markdown", + "id": "cdc4dfd8-5e8c-407c-ace6-f484608e4141", + "metadata": { + "tags": [] + }, + "source": [ + "## Configure notebook formatting" + ] + }, + { + "cell_type": "markdown", + "id": "7242d35a-7d1b-4b14-9aa5-b4a9bcb10a91", + "metadata": {}, + "source": [ + "### Tables\n", + "Run the code below to set up for displaying Pandas DataFrames as Markdown tables in this notebook." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "2678201f-feba-4b2a-897f-284f05119b9c", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "from IPython.display import display, Markdown\n", + "def display_df_md_table(df):\n", + " return display(Markdown(df.replace('\\n', '
', regex=True).to_markdown()))" + ] + }, + { + "cell_type": "markdown", + "id": "67311e00-9fd8-4b1c-80cc-44b0d4b18a7f", + "metadata": {}, + "source": [ + "## Obtain an API token\n", + "Follow [these instructions](https://github.com/ubccr/xdmod-data#api-token-access) to obtain an API token." + ] + }, + { + "cell_type": "markdown", + "id": "035a27ba-a47b-4fe6-a7ea-b73e5f5a308a", + "metadata": { + "tags": [] + }, + "source": [ + "## Create an environment file\n", + "The `xdmod-data.env` file will store your XDMoD API token.\n", + "\n", + "Run the code below to create the file in your home directory (if it does not already exist) and allow only you to read and write to it." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7d216a32-11c6-4ae4-98e9-1b2c19d442d2", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "from pathlib import Path\n", + "from os.path import expanduser\n", + "xdmod_data_env_path = Path(expanduser('~/xdmod-data.env'))\n", + "try:\n", + " with open(xdmod_data_env_path):\n", + " pass\n", + "except FileNotFoundError:\n", + " with open(xdmod_data_env_path, 'w') as xdmod_data_env_file:\n", + " xdmod_data_env_file.write('XDMOD_API_TOKEN=')\n", + " xdmod_data_env_path.chmod(0o600)" + ] + }, + { + "cell_type": "markdown", + "id": "ae55bfed-eedc-4a72-88d9-510fe271ed36", + "metadata": {}, + "source": [ + "## Store your API token in the environment file\n", + "Open the `xdmod-data.env` file and paste your token after `XDMOD_API_TOKEN=`.\n", + "\n", + "Save the file." + ] + }, + { + "cell_type": "markdown", + "id": "02193bb2-1b80-4b9d-8733-ad292a877ccf", + "metadata": { + "tags": [] + }, + "source": [ + "## Load your XDMoD API token into the environment\n", + "Run the code below to load the contents of the `xdmod-data.env` file into the environment. It will print `True` if it successfully loaded the file." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "d5eefbd5-8eab-4ac2-87fe-2c6271c7b9cd", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "from dotenv import load_dotenv\n", + "load_dotenv(xdmod_data_env_path, override=True)" + ] + }, + { + "cell_type": "markdown", + "id": "bc6c166c-5e03-403b-88e1-4b4b2735c690", + "metadata": {}, + "source": [ + "## Initialize the XDMoD Data Warehouse\n", + "Run the code below to prepare for getting data from the XDMoD data warehouse at the given URL." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "57583c14-0196-4e1d-bc63-3a66c493fa44", + "metadata": {}, + "outputs": [], + "source": [ + "from xdmod_data.warehouse import DataWarehouse\n", + "dw = DataWarehouse('https://xdmod.access-ci.org')" + ] + }, + { + "cell_type": "markdown", + "id": "3738a176-5b11-4a27-bf5c-2e785f07210d", + "metadata": {}, + "source": [ + "## Get the data from XDMoD\n", + "\n", + "The following code queries XDMoD to get the list of PIs from UC San Diego for projects that have had any usage during the report period. It then queries the allocation information for each PI." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "00e141b1-3554-44e6-a08b-85b3f6b19fd3", + "metadata": {}, + "outputs": [], + "source": [ + "report_start = '2022-12-01'\n", + "report_end = '2023-11-30'\n", + "\n", + "output = {'PI': [], 'Allocation': [], 'ACCESS Credit Equivalents': []}\n", + "with dw:\n", + " pis = dw.get_data(\n", + " duration=(report_start, report_end),\n", + " realm='Jobs',\n", + " metric='Number of Jobs Ended',\n", + " dataset_type='aggregate',\n", + " dimension='PI',\n", + " filters={\n", + " 'PI Institution': ['UC San Diego - University of California, San Diego']\n", + " }\n", + " )\n", + " for pi in pis.index:\n", + " allocs = dw.get_data(\n", + " duration=(report_start, report_end),\n", + " realm='Jobs',\n", + " metric='ACCESS Credit Equivalents Charged: Total (SU)',\n", + " dataset_type='aggregate',\n", + " dimension='Allocation',\n", + " filters={\n", + " 'PI': [pi]\n", + " }\n", + " )\n", + " for alloc in allocs.index:\n", + " output['PI'].append(pi)\n", + " output['Allocation'].append(alloc)\n", + " output['ACCESS Credit Equivalents'].append(allocs[alloc])\n", + " " + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "85992b84-6c73-4110-80e2-288f64e84db4", + "metadata": {}, + "outputs": [], + "source": [ + "import pandas as pd\n", + "data= pd.DataFrame(output)\n", + "display_df_md_table(data)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7f77dfa4-69d1-4ad8-bb87-170470940846", + "metadata": {}, + "outputs": [], + "source": [ + "from IPython.display import Javascript\n", + "js_download = \"\"\"\n", + "var csv = `%s`;\n", + "\n", + "var filename = 'UC_PI_Allocations.csv';\n", + "var blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });\n", + "if (navigator.msSaveBlob) { // IE 10+\n", + " navigator.msSaveBlob(blob, filename);\n", + "} else {\n", + " var link = document.createElement(\"a\");\n", + " link.innerHTML = 'Download as CSV';\n", + " if (link.download !== undefined) { // feature detection\n", + " // Browsers that support HTML5 download attribute\n", + " var url = URL.createObjectURL(blob);\n", + " link.setAttribute(\"href\", url);\n", + " link.setAttribute(\"download\", filename);\n", + " document.body.appendChild(link);\n", + " }\n", + "}\n", + "\"\"\" % data.to_csv(index=False)\n", + "\n", + "# Note this javascript link is intended for use when you export the notebook to html\n", + "# via jupyter nbconvert --to html [NOTEBOOK]\n", + "Javascript(js_download)" + ] + }, + { + "cell_type": "markdown", + "id": "b9cb8874-138a-45f4-a06b-244c67f510f2", + "metadata": {}, + "source": [ + "---\n", + "![XDMoD]() XDMoD Data Analytics Framework v1.0.0" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "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.10.9" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/ACCESS/README.md b/ACCESS/README.md new file mode 100644 index 0000000..6fedf24 --- /dev/null +++ b/ACCESS/README.md @@ -0,0 +1,11 @@ +# xdmod-notebooks/ACCESS + +This directory contains example Jupyter notebooks demonstrating how to use the XDMoD Data Analytics +framework for various analysis of usage and performance of ACCESS-allocated CI resources. + +If you are new to the XDMoD Data Analytics Framework then please see the `XDMoD-Data-First-Example.ipynb` +notebook in the parent directory. + +## Setup + +Please see the [setup instructions](../README.md#Setup) diff --git a/ACCESS/User-Resource-Overlaps.ipynb b/ACCESS/User-Resource-Overlaps.ipynb new file mode 100644 index 0000000..5f159dc --- /dev/null +++ b/ACCESS/User-Resource-Overlaps.ipynb @@ -0,0 +1,348 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "db4061c7-2b36-4284-af31-4ae940a07768", + "metadata": { + "tags": [] + }, + "source": [ + "# XDMoD Data Analytics Framework — User Resource Overlaps\n", + "v1.0.0\n", + "\n", + "© 2023 University at Buffalo Center for Computational Research\n", + "\n", + "See the [xdmod-notebooks](https://github.com/ubccr/xdmod-notebooks) repository for licensing information." + ] + }, + { + "cell_type": "markdown", + "id": "77f9e969-5120-4a7a-a844-fa6fda893755", + "metadata": {}, + "source": [ + "## Introduction\n", + "The XDMoD Data Analytics Framework provides API access to the data in XDMoD via the [`xdmod_data` Python module](https://pypi.org/project/xdmod-data). This notebook provides an introductory example showing how to use the module. You will use the XDMoD API to request data, load them into a [Pandas](https://pandas.pydata.org/) [DataFrame](https://pandas.pydata.org/docs/reference/api/pandas.DataFrame.html), and generate plots. The dataset in this example contains the number of active users of [ACCESS](https://access-ci.org/)-allocated resources per day of the week over a 4-month period." + ] + }, + { + "cell_type": "markdown", + "id": "84713a67-80c4-4803-8a45-0decdb65b500", + "metadata": {}, + "source": [ + "## Install/upgrade the required modules\n", + "Run the code below to install/upgrade the modules needed to run this notebook." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "43521c0d-ca9e-47cd-ba5a-c1f9d0b83290", + "metadata": {}, + "outputs": [], + "source": [ + "import sys\n", + "! {sys.executable} -m pip install --upgrade xdmod-data python-dotenv tabulate\n", + "\n", + "import pandas as pd" + ] + }, + { + "cell_type": "markdown", + "id": "56c7d25d-241b-4c65-a77b-7eca37386714", + "metadata": {}, + "source": [ + "If running that code caused a new version of Plotly to be installed/upgraded, you may need to refresh your browser window for plots to appear correctly." + ] + }, + { + "cell_type": "markdown", + "id": "cdc4dfd8-5e8c-407c-ace6-f484608e4141", + "metadata": { + "tags": [] + }, + "source": [ + "## Configure notebook formatting" + ] + }, + { + "cell_type": "markdown", + "id": "7242d35a-7d1b-4b14-9aa5-b4a9bcb10a91", + "metadata": {}, + "source": [ + "### Tables\n", + "Run the code below to set up for displaying Pandas DataFrames as Markdown tables in this notebook." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "2678201f-feba-4b2a-897f-284f05119b9c", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "from IPython.display import display, Markdown\n", + "def display_df_md_table(df):\n", + " return display(Markdown(df.replace('\\n', '
', regex=True).to_markdown()))" + ] + }, + { + "cell_type": "markdown", + "id": "035a27ba-a47b-4fe6-a7ea-b73e5f5a308a", + "metadata": {}, + "source": [ + "## Create an environment file\n", + "The `xdmod-data.env` file will store your XDMoD API token.\n", + "\n", + "Run the code below to create the file in your home directory (if it does not already exist) and allow only you to read and write to it." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "7d216a32-11c6-4ae4-98e9-1b2c19d442d2", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "from pathlib import Path\n", + "from os.path import expanduser\n", + "xdmod_data_env_path = Path(expanduser('~/xdmod-data.env'))\n", + "try:\n", + " with open(xdmod_data_env_path):\n", + " pass\n", + "except FileNotFoundError:\n", + " with open(xdmod_data_env_path, 'w') as xdmod_data_env_file:\n", + " xdmod_data_env_file.write('XDMOD_API_TOKEN=')\n", + " xdmod_data_env_path.chmod(0o600)" + ] + }, + { + "cell_type": "markdown", + "id": "67311e00-9fd8-4b1c-80cc-44b0d4b18a7f", + "metadata": {}, + "source": [ + "## Obtain an API token\n", + "Follow [these instructions](https://github.com/ubccr/xdmod-data#api-token-access) to obtain an API token." + ] + }, + { + "cell_type": "markdown", + "id": "ae55bfed-eedc-4a72-88d9-510fe271ed36", + "metadata": {}, + "source": [ + "## Store your API token in the environment file\n", + "Open the `xdmod-data.env` file and paste your token after `XDMOD_API_TOKEN=`.\n", + "\n", + "Save the file." + ] + }, + { + "cell_type": "markdown", + "id": "02193bb2-1b80-4b9d-8733-ad292a877ccf", + "metadata": { + "tags": [] + }, + "source": [ + "## Load your XDMoD API token into the environment\n", + "Run the code below to load the contents of the `xdmod-data.env` file into the environment. It will print `True` if it successfully loaded the file." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "d5eefbd5-8eab-4ac2-87fe-2c6271c7b9cd", + "metadata": { + "tags": [] + }, + "outputs": [], + "source": [ + "from dotenv import load_dotenv\n", + "load_dotenv(xdmod_data_env_path, override=True)" + ] + }, + { + "cell_type": "markdown", + "id": "bc6c166c-5e03-403b-88e1-4b4b2735c690", + "metadata": {}, + "source": [ + "## Initialize the XDMoD Data Warehouse\n", + "Run the code below to prepare for getting data from the XDMoD data warehouse at the given URL." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "57583c14-0196-4e1d-bc63-3a66c493fa44", + "metadata": {}, + "outputs": [], + "source": [ + "from xdmod_data.warehouse import DataWarehouse\n", + "dw = DataWarehouse('https://xdmod.access-ci.org')" + ] + }, + { + "cell_type": "markdown", + "id": "3738a176-5b11-4a27-bf5c-2e785f07210d", + "metadata": {}, + "source": [ + "## Get the list of resource that have run compute jobs\n", + "\n", + "Get the list of compute resources that have run compute jobs during the specific times and how many distinct ACCESS users have used it." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "00e141b1-3554-44e6-a08b-85b3f6b19fd3", + "metadata": {}, + "outputs": [], + "source": [ + "report_start = '2022-10-01'\n", + "report_end = '2023-09-30'\n", + "\n", + "with dw:\n", + " resources = dw.get_data(\n", + " duration=(report_start, report_end),\n", + " realm='Jobs',\n", + " metric='Number of Users: Active',\n", + " dataset_type='aggregate',\n", + " dimension='Resource'\n", + " )" + ] + }, + { + "cell_type": "markdown", + "id": "258e366e-04fb-4a99-b1cf-31dc07bfc676", + "metadata": {}, + "source": [ + "Then for each resource we can obtain the list of users and the number of service units consumed during the report period. In this case we just stored the list of users for each resource." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "79a95d2c-eba2-48bd-8770-1b957679da54", + "metadata": {}, + "outputs": [], + "source": [ + "resource_people = {}\n", + "for resource in resources.index:\n", + " with dw: \n", + " people = dw.get_data(\n", + " duration=(report_start, report_end),\n", + " realm='Jobs',\n", + " metric='ACCESS Credit Equivalents Charged: Total (SU)',\n", + " dataset_type='aggregate',\n", + " dimension='User',\n", + " filters={\n", + " 'Resource': [resource]\n", + " }\n", + " )\n", + " resource_people[resource] = set(people.index)\n", + "\n" + ] + }, + { + "cell_type": "markdown", + "id": "5edfdc87-1134-4ed6-be8b-f720e6e4dc8d", + "metadata": {}, + "source": [ + "Then compute the overlap of users between different resources and display in a table." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "70e10722-6ddf-4667-9930-059b1b3721b4", + "metadata": {}, + "outputs": [], + "source": [ + "out = {'Resource': []}\n", + "for r1, r1p in resource_people.items():\n", + " out['Resource'].append(r1)\n", + " for r2, r2p in resource_people.items():\n", + " if r2 not in out:\n", + " out[r2] = []\n", + " out[r2].append(len(r1p.intersection(r2p)))\n", + " \n", + "data = pd.DataFrame(out)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "a3f5cba7-9f1e-4545-8aa0-7c1ddc81506d", + "metadata": {}, + "outputs": [], + "source": [ + "data" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "62b2bc6b-3ce2-4d70-ae50-73f125d184c5", + "metadata": {}, + "outputs": [], + "source": [ + "from IPython.display import Javascript\n", + "js_download = \"\"\"\n", + "var csv = `%s`;\n", + "\n", + "var filename = 'User-Resource-Overlap.csv';\n", + "var blob = new Blob([csv], { type: 'text/csv;charset=utf-8;' });\n", + "if (navigator.msSaveBlob) { // IE 10+\n", + " navigator.msSaveBlob(blob, filename);\n", + "} else {\n", + " var link = document.createElement(\"a\");\n", + " link.innerHTML = 'Download as CSV';\n", + " if (link.download !== undefined) { // feature detection\n", + " // Browsers that support HTML5 download attribute\n", + " var url = URL.createObjectURL(blob);\n", + " link.setAttribute(\"href\", url);\n", + " link.setAttribute(\"download\", filename);\n", + " document.body.appendChild(link);\n", + " }\n", + "}\n", + "\"\"\" % data.to_csv(index=False)\n", + "\n", + "# Note this javascript link is intended for use when you export the notebook to html\n", + "# via jupyter nbconvert --to html [NOTEBOOK]\n", + "Javascript(js_download)" + ] + }, + { + "cell_type": "markdown", + "id": "b9cb8874-138a-45f4-a06b-244c67f510f2", + "metadata": {}, + "source": [ + "---\n", + "![XDMoD]() XDMoD Data Analytics Framework v1.0.0" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "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.10.9" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} From ed2752884e2b49e0a2bcfff3c4e238526d119b90 Mon Sep 17 00:00:00 2001 From: Joseph White Date: Fri, 1 Dec 2023 11:41:28 -0500 Subject: [PATCH 2/2] Add histogram plot too. --- ACCESS/User-Resource-Overlaps.ipynb | 56 +++++++++++++++++++++++++++++ 1 file changed, 56 insertions(+) diff --git a/ACCESS/User-Resource-Overlaps.ipynb b/ACCESS/User-Resource-Overlaps.ipynb index 5f159dc..8aa0796 100644 --- a/ACCESS/User-Resource-Overlaps.ipynb +++ b/ACCESS/User-Resource-Overlaps.ipynb @@ -87,6 +87,28 @@ " return display(Markdown(df.replace('\\n', '
', regex=True).to_markdown()))" ] }, + { + "cell_type": "markdown", + "id": "d9e37669-ca38-42a6-b97c-a361cc3ab21e", + "metadata": {}, + "source": [ + "## Plotting\n", + "Run the code below to import plotly libraries and set the XDMoD theme" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "0cccc2e6-48e8-474f-bbb3-83e2e36869a1", + "metadata": {}, + "outputs": [], + "source": [ + "import plotly.express as px\n", + "import plotly.io as pio\n", + "import xdmod_data.themes\n", + "pio.templates.default = \"timeseries\"" + ] + }, { "cell_type": "markdown", "id": "035a27ba-a47b-4fe6-a7ea-b73e5f5a308a", @@ -314,6 +336,40 @@ "Javascript(js_download)" ] }, + { + "cell_type": "code", + "execution_count": null, + "id": "2eb42a63-bf40-493e-85c3-37e20ece8c97", + "metadata": {}, + "outputs": [], + "source": [ + "allpeople = set()\n", + "for r, p in resource_people.items():\n", + " allpeople |= p\n", + " \n", + "out = {'Person': list(allpeople), 'Resource Count': []}\n", + "for p in out['Person']:\n", + " count = 0\n", + " for pet in resource_people.values():\n", + " if p in pet:\n", + " count += 1\n", + " out['Resource Count'].append(count)\n", + "\n", + "data = pd.DataFrame(out)" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "c8a20af7-90e8-4c6f-a276-5413a60be922", + "metadata": {}, + "outputs": [], + "source": [ + "fig = px.histogram(data, x='Resource Count', log_y=True, text_auto=True)\n", + "fig.update_layout(yaxis_title=\"Number of Users (log scale)\")\n", + "fig.show()" + ] + }, { "cell_type": "markdown", "id": "b9cb8874-138a-45f4-a06b-244c67f510f2",