import gradio as gr
from gradio_huggingfacehub_search import HuggingfaceHubSearch
import nbformat as nbf
from huggingface_hub import HfApi
from httpx import Client
import logging
import pandas as pd
from utils.notebook_utils import (
replace_wildcards,
load_json_files_from_folder,
)
from dotenv import load_dotenv
import os
from nbconvert import HTMLExporter
import uuid
load_dotenv()
HF_TOKEN = os.getenv("HF_TOKEN")
assert HF_TOKEN is not None, "You need to set HF_TOKEN in your environment variables"
NOTEBOOKS_REPOSITORY = os.getenv("NOTEBOOKS_REPOSITORY")
assert (
NOTEBOOKS_REPOSITORY is not None
), "You need to set NOTEBOOKS_REPOSITORY in your environment variables"
BASE_DATASETS_SERVER_URL = "https://datasets-server.huggingface.co"
HEADERS = {"Accept": "application/json", "Content-Type": "application/json"}
client = Client(headers=HEADERS)
logging.basicConfig(level=logging.INFO)
# TODO: Validate notebook templates format
folder_path = "notebooks"
notebook_templates = load_json_files_from_folder(folder_path)
logging.info(f"Available notebooks {notebook_templates.keys()}")
def get_compatible_libraries(dataset: str):
try:
response = client.get(
f"{BASE_DATASETS_SERVER_URL}/compatible-libraries?dataset={dataset}"
)
response.raise_for_status()
return response.json()
except Exception as e:
logging.error(f"Error fetching compatible libraries: {e}")
raise
def create_notebook_file(cells, notebook_name):
nb = nbf.v4.new_notebook()
nb["cells"] = [
nbf.v4.new_code_cell(
cmd["source"]
if isinstance(cmd["source"], str)
else "\n".join(cmd["source"])
)
if cmd["cell_type"] == "code"
else nbf.v4.new_markdown_cell(cmd["source"])
for cmd in cells
]
with open(notebook_name, "w") as f:
nbf.write(nb, f)
logging.info(f"Notebook {notebook_name} created successfully")
html_exporter = HTMLExporter()
html_data, _ = html_exporter.from_notebook_node(nb)
return html_data
def get_first_rows_as_df(dataset: str, config: str, split: str, limit: int):
try:
resp = client.get(
f"{BASE_DATASETS_SERVER_URL}/first-rows?dataset={dataset}&config={config}&split={split}"
)
resp.raise_for_status()
content = resp.json()
rows = content["rows"]
rows = [row["row"] for row in rows]
first_rows_df = pd.DataFrame.from_dict(rows).sample(frac=1).head(limit)
return first_rows_df
except Exception as e:
logging.error(f"Error fetching first rows: {e}")
raise
def longest_string_column(df):
longest_col = None
max_length = 0
for col in df.select_dtypes(include=["object", "string"]):
max_col_length = df[col].str.len().max()
if max_col_length > max_length:
max_length = max_col_length
longest_col = col
return longest_col
def _push_to_hub(
dataset_id,
notebook_file,
):
logging.info(f"Pushing notebook to hub: {dataset_id} on file {notebook_file}")
notebook_name = notebook_file.split("/")[-1]
api = HfApi(token=HF_TOKEN)
try:
logging.info(f"About to push {notebook_file} - {dataset_id}")
api.upload_file(
path_or_fileobj=notebook_file,
path_in_repo=notebook_name,
repo_id=NOTEBOOKS_REPOSITORY,
repo_type="dataset",
)
except Exception as e:
logging.info("Failed to push notebook", e)
raise
def generate_cells(dataset_id, notebook_title):
logging.info(f"Generating {notebook_title} notebook for dataset {dataset_id}")
cells = notebook_templates[notebook_title]["notebook_template"]
notebook_type = notebook_templates[notebook_title]["notebook_type"]
try:
libraries = get_compatible_libraries(dataset_id)
except Exception as err:
gr.Error("Unable to retrieve dataset info from HF Hub.")
logging.error(f"Failed to fetch compatible libraries: {err}")
return "", "## ❌ This dataset is not accessible from the Hub ❌"
if not libraries:
logging.error(f"Dataset not compatible with pandas library - not libraries")
return "", "## ❌ This dataset is not compatible with pandas library ❌"
pandas_library = next(
(lib for lib in libraries.get("libraries", []) if lib["library"] == "pandas"),
None,
)
if not pandas_library:
logging.error("Dataset not compatible with pandas library - not pandas library")
return "", "## ❌ This dataset is not compatible with pandas library ❌"
first_config_loading_code = pandas_library["loading_codes"][0]
first_code = first_config_loading_code["code"]
first_config = first_config_loading_code["config_name"]
first_split = list(first_config_loading_code["arguments"]["splits"].keys())[0]
df = get_first_rows_as_df(dataset_id, first_config, first_split, 3)
longest_col = longest_string_column(df)
html_code = f""
wildcards = ["{dataset_name}", "{first_code}", "{html_code}", "{longest_col}"]
replacements = [dataset_id, first_code, html_code, longest_col]
has_numeric_columns = len(df.select_dtypes(include=["number"]).columns) > 0
has_categoric_columns = len(df.select_dtypes(include=["object"]).columns) > 0
# TODO: Validate by notebook type
if notebook_type in ("rag", "embeddings") and not has_categoric_columns:
logging.error(
"Dataset does not have categorical columns, which are required for RAG generation."
)
return (
"",
"## ❌ This dataset does not have categorical columns, which are required for Embeddings/RAG generation ❌",
)
if notebook_type == "eda" and not (has_categoric_columns or has_numeric_columns):
logging.error(
"Dataset does not have categorical or numeric columns, which are required for EDA generation."
)
return (
"",
"## ❌ This dataset does not have categorical or numeric columns, which are required for EDA generation ❌",
)
cells = replace_wildcards(
cells, wildcards, replacements, has_numeric_columns, has_categoric_columns
)
notebook_name = (
f"{dataset_id.replace('/', '-')}-{notebook_type}-{uuid.uuid4()}.ipynb"
)
html_content = create_notebook_file(cells, notebook_name=notebook_name)
_push_to_hub(dataset_id, notebook_name)
notebook_link = f"https://colab.research.google.com/#fileId=https%3A//huggingface.co/datasets/asoria/dataset-notebook-creator-content/blob/main/{notebook_name}"
return (
html_content,
f"## 🎉 Ready to explore? Play and run the generated notebook 👉 [here]({notebook_link})!",
)
css = """
#box {
height: 650px;
overflow-y: scroll !important;
}
"""
with gr.Blocks(
fill_height=True,
fill_width=True,
css=css,
) as demo:
gr.Markdown("# 🤖 Dataset notebook creator 🕵️")
text_input = gr.Textbox(label="Suggested notebook type", visible=False)
gr.Markdown("## 1. Select and preview a dataset from Huggingface Hub")
dataset_name = HuggingfaceHubSearch(
label="Hub Dataset ID",
placeholder="Search for dataset id on Huggingface",
search_type="dataset",
value="",
)
dataset_samples = gr.Examples(
examples=[
[
"scikit-learn/iris",
"Try this dataset for Exploratory Data Analysis",
],
[
"infinite-dataset-hub/GlobaleCuisineRecipes",
"Try this dataset for Embeddings generation",
],
[
"infinite-dataset-hub/GlobalBestSellersSummaries",
"Try this dataset for RAG generation",
],
],
inputs=[dataset_name, text_input],
cache_examples=False,
)
@gr.render(inputs=dataset_name)
def embed(name):
if not name:
return gr.Markdown("### No dataset provided")
html_code = f"""
"""
return gr.HTML(value=html_code, elem_classes="viewer")
gr.Markdown("## 2. Select the type of notebook you want to generate")
with gr.Row():
notebook_type = gr.Dropdown(
choices=notebook_templates.keys(),
label="Notebook type",
value="Text Embeddings",
)
generate_button = gr.Button("Generate Notebook", variant="primary")
contribute_btn = gr.Button(
"Or Contribute",
visible=True,
variant="secondary",
size="sm",
link="https://huggingface.co/spaces/asoria/auto-notebook-creator/blob/main/CONTRIBUTING.md",
)
gr.Markdown("## 3. Notebook code result")
code_component = gr.HTML(elem_id="box")
go_to_notebook = gr.Markdown("", visible=True)
generate_button.click(
generate_cells,
inputs=[dataset_name, notebook_type],
outputs=[code_component, go_to_notebook],
)
gr.Markdown(
"🚧 Note: Some code may not be compatible with datasets that contain binary data or complex structures. 🚧"
)
demo.launch()