2022-02-05 05:42:49 +08:00
|
|
|
import html
|
2022-01-21 21:44:12 +08:00
|
|
|
import inspect
|
2021-12-14 14:02:19 +08:00
|
|
|
import json
|
2022-01-21 21:44:12 +08:00
|
|
|
import os
|
2021-12-14 14:02:19 +08:00
|
|
|
import re
|
2022-01-21 21:44:12 +08:00
|
|
|
|
|
|
|
import markdown2
|
|
|
|
import requests
|
|
|
|
from jinja2 import Template
|
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
from gradio.inputs import InputComponent
|
|
|
|
from gradio.interface import Interface
|
2022-01-21 21:44:12 +08:00
|
|
|
from gradio.outputs import OutputComponent
|
2021-12-14 14:02:19 +08:00
|
|
|
|
2022-02-05 05:42:49 +08:00
|
|
|
import cairo
|
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
GRADIO_DIR = "../../"
|
|
|
|
GRADIO_GUIDES_DIR = os.path.join(GRADIO_DIR, "guides")
|
|
|
|
GRADIO_DEMO_DIR = os.path.join(GRADIO_DIR, "demo")
|
|
|
|
|
2022-02-05 05:42:49 +08:00
|
|
|
|
|
|
|
guides = []
|
2021-12-14 14:02:19 +08:00
|
|
|
for guide in sorted(os.listdir(GRADIO_GUIDES_DIR)):
|
2022-01-19 05:39:55 +08:00
|
|
|
if "template" in guide or "getting_started" in guide:
|
2021-12-14 14:02:19 +08:00
|
|
|
continue
|
|
|
|
guide_name = guide[:-3]
|
2022-01-21 21:44:12 +08:00
|
|
|
pretty_guide_name = " ".join(
|
|
|
|
[word.capitalize().replace("Ml", "ML") for word in guide_name.split("_")]
|
|
|
|
)
|
2022-02-05 05:42:49 +08:00
|
|
|
with open(os.path.join(GRADIO_GUIDES_DIR, guide), "r") as f:
|
|
|
|
guide_content = f.read()
|
|
|
|
|
|
|
|
tags = []
|
|
|
|
if "tags: " in guide_content:
|
|
|
|
tags = guide_content.split("tags: ")[1].split("\n")[0].split(", ")
|
|
|
|
tags = tags[:3]
|
|
|
|
tags = " • ".join(tags)
|
|
|
|
|
|
|
|
guide_dict = {
|
|
|
|
"guide_name": guide_name,
|
|
|
|
"pretty_guide_name": pretty_guide_name,
|
|
|
|
"guide_content": guide_content,
|
|
|
|
"tags": tags
|
|
|
|
}
|
|
|
|
guides.append(guide_dict)
|
2021-12-14 14:02:19 +08:00
|
|
|
|
2022-02-05 05:42:49 +08:00
|
|
|
with open("src/navbar.html", encoding="utf-8") as navbar_file:
|
|
|
|
navbar_html = navbar_file.read()
|
2022-01-21 21:44:12 +08:00
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
def render_index():
|
|
|
|
os.makedirs("generated", exist_ok=True)
|
2022-01-21 21:44:12 +08:00
|
|
|
with open("src/tweets.json", encoding="utf-8") as tweets_file:
|
2021-12-14 14:02:19 +08:00
|
|
|
tweets = json.load(tweets_file)
|
2022-01-21 21:44:12 +08:00
|
|
|
star_count = "{:,}".format(
|
|
|
|
requests.get("https://api.github.com/repos/gradio-app/gradio").json()[
|
|
|
|
"stargazers_count"
|
|
|
|
]
|
2022-02-05 05:42:49 +08:00
|
|
|
)
|
2022-01-21 21:44:12 +08:00
|
|
|
with open("src/index_template.html", encoding="utf-8") as template_file:
|
2021-12-14 14:02:19 +08:00
|
|
|
template = Template(template_file.read())
|
2022-02-05 05:42:49 +08:00
|
|
|
output_html = template.render(tweets=tweets, star_count=star_count,
|
|
|
|
navbar_html=navbar_html)
|
|
|
|
with open(os.path.join("generated", "index.html"), "w", encoding='utf-8') as generated_template:
|
2021-12-14 14:02:19 +08:00
|
|
|
generated_template.write(output_html)
|
|
|
|
|
2022-01-21 21:44:12 +08:00
|
|
|
|
2022-02-05 05:42:49 +08:00
|
|
|
def render_guides_main():
|
|
|
|
with open("src/guides_main_template.html", encoding='utf-8') as template_file:
|
|
|
|
template = Template(template_file.read())
|
|
|
|
output_html = template.render(guides=guides, navbar_html=navbar_html)
|
|
|
|
with open(os.path.join("generated", "guides.html"), "w", encoding='utf-8') as generated_template:
|
|
|
|
generated_template.write(output_html)
|
|
|
|
|
|
|
|
|
|
|
|
def add_line_breaks(text, num_char):
|
|
|
|
if len(text) > num_char:
|
|
|
|
text_list = text.split()
|
|
|
|
text = ""
|
|
|
|
total_count = 0
|
|
|
|
count = 0
|
|
|
|
for word in text_list:
|
|
|
|
if total_count > num_char*5:
|
|
|
|
text = text[:-1]
|
|
|
|
text += "..."
|
|
|
|
break
|
|
|
|
count += len(word)
|
|
|
|
if count > num_char:
|
|
|
|
text += word + "\n"
|
|
|
|
total_count += count
|
|
|
|
count = 0
|
|
|
|
else:
|
|
|
|
text += word + " "
|
|
|
|
total_count += len(word+" ")
|
|
|
|
return text
|
|
|
|
return text
|
|
|
|
|
|
|
|
|
|
|
|
def generate_guide_meta_tags(title, tags, url, guide_path_name):
|
|
|
|
surface = cairo.ImageSurface.create_from_png("src/assets/img/guides/base-image.png")
|
|
|
|
ctx = cairo.Context(surface)
|
|
|
|
ctx.scale(500, 500)
|
|
|
|
ctx.set_source_rgba(0.611764706,0.639215686,0.6862745098,1)
|
|
|
|
ctx.select_font_face("Arial", cairo.FONT_SLANT_NORMAL,
|
|
|
|
cairo.FONT_WEIGHT_NORMAL)
|
|
|
|
ctx.set_font_size(0.15)
|
|
|
|
ctx.move_to(0.3, 0.55)
|
|
|
|
ctx.show_text("gradio.app/guides")
|
|
|
|
if len(tags) > 5:
|
|
|
|
tags = tags[:5]
|
|
|
|
tags = " | ".join(tags)
|
|
|
|
ctx.move_to(0.3, 2.2)
|
|
|
|
ctx.show_text(tags)
|
|
|
|
ctx.set_source_rgba(0.352941176,0.352941176,0.352941176,1)
|
|
|
|
ctx.set_font_size(0.28)
|
|
|
|
title_breaked = add_line_breaks(title, 10)
|
|
|
|
|
|
|
|
if "\n" in title_breaked:
|
|
|
|
for i, t in enumerate(title_breaked.split("\n")):
|
|
|
|
ctx.move_to(0.3, 0.9+i*0.4)
|
|
|
|
ctx.show_text(t)
|
|
|
|
else:
|
|
|
|
ctx.move_to(0.3, 1.0)
|
|
|
|
ctx.show_text(title_breaked)
|
|
|
|
image_path = f"src/assets/img/guides/{guide_path_name}.png"
|
|
|
|
surface.write_to_png(image_path)
|
|
|
|
load_path = f"/assets/img/guides/{guide_path_name}.png"
|
|
|
|
meta_tags = f"""
|
|
|
|
<title>{title}</title>
|
|
|
|
<meta property="og:title" content="{title}" />
|
|
|
|
<meta property="og:type" content="article" />
|
|
|
|
<meta property="og:url" content="{url}" />
|
|
|
|
<meta property="og:image" content="{load_path}" />
|
|
|
|
<meta name="twitter:title" content="{title}">
|
|
|
|
<meta name="twitter:image" content="{load_path}">
|
|
|
|
<meta name="twitter:card" content="summary_large_image">
|
|
|
|
"""
|
|
|
|
return meta_tags
|
|
|
|
|
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
def render_guides():
|
|
|
|
for guide in os.listdir(GRADIO_GUIDES_DIR):
|
|
|
|
if "template" in guide:
|
|
|
|
continue
|
2022-01-21 21:44:12 +08:00
|
|
|
with open(
|
|
|
|
os.path.join(GRADIO_GUIDES_DIR, guide), encoding="utf-8"
|
|
|
|
) as guide_file:
|
2021-12-14 14:02:19 +08:00
|
|
|
guide_text = guide_file.read()
|
2022-02-05 05:42:49 +08:00
|
|
|
|
|
|
|
if "related_spaces: " in guide_text:
|
|
|
|
spaces = guide_text.split("related_spaces: ")[1].split("\n")[0].split(", ")
|
|
|
|
spaces_html = "<div id='spaces-holder'><a href='https://hf.co/spaces' target='_blank'><img src='/assets/img/spaces-logo.svg'></a><p style='margin: 0;display: inline;font-size: large;font-weight: 400;'>Related Spaces: </p>"
|
|
|
|
for space in spaces:
|
|
|
|
spaces_html += f"<div class='space-link'><a href='{space}' target='_blank'>{space[30:]}</a></div>"
|
|
|
|
spaces_html += "</div>"
|
|
|
|
guide_text = guide_text.split("related_spaces: ")[0] + spaces_html + "\n".join(guide_text.split("related_spaces: ")[1].split("\n")[1:])
|
|
|
|
|
|
|
|
tags = ""
|
|
|
|
if "tags: " in guide_text:
|
|
|
|
tags = guide_text.split("tags: ")[1].split("\n")[0].split(", ")
|
|
|
|
guide_text = guide_text.split("tags: ")[0] + "\n" + "\n".join(guide_text.split("tags: ")[1].split("\n")[1:])
|
|
|
|
|
|
|
|
title = " ".join(
|
|
|
|
[word.capitalize().replace("Ml", "ML").replace("Gan", "GAN") for word in guide[:-3].split("_")]
|
|
|
|
)
|
|
|
|
url = f"https://gradio.app/{guide[:-3]}/"
|
|
|
|
meta_tags = generate_guide_meta_tags(title, tags, url, guide[:-3])
|
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
code_tags = re.findall(r'\{\{ code\["([^\s]*)"\] \}\}', guide_text)
|
|
|
|
demo_names = re.findall(r'\{\{ demos\["([^\s]*)"\] \}\}', guide_text)
|
|
|
|
code, demos = {}, {}
|
2022-01-21 21:44:12 +08:00
|
|
|
guide_text = (
|
|
|
|
guide_text.replace("website/src/assets", "/assets")
|
|
|
|
.replace("```python\n", "<pre><code class='lang-python'>")
|
|
|
|
.replace("```bash\n", "<pre><code class='lang-bash'>")
|
|
|
|
.replace("```directory\n", "<pre><code class='lang-bash'>")
|
|
|
|
.replace("```csv\n", "<pre><code class='lang-bash'>")
|
|
|
|
.replace("```", "</code></pre>")
|
|
|
|
)
|
2022-02-05 05:42:49 +08:00
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
for code_src in code_tags:
|
|
|
|
with open(os.path.join(GRADIO_DEMO_DIR, code_src, "run.py")) as code_file:
|
|
|
|
python_code = code_file.read().replace(
|
2022-01-21 21:44:12 +08:00
|
|
|
'if __name__ == "__main__":\n iface.launch()', "iface.launch()"
|
|
|
|
)
|
|
|
|
code[code_src] = (
|
|
|
|
"<pre><code class='lang-python'>" + python_code + "</code></pre>"
|
|
|
|
)
|
2022-02-05 05:42:49 +08:00
|
|
|
|
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
for demo_name in demo_names:
|
|
|
|
demos[demo_name] = "<div id='interface_" + demo_name + "'></div>"
|
|
|
|
guide_template = Template(guide_text)
|
|
|
|
guide_output = guide_template.render(code=code, demos=demos)
|
2022-02-05 05:42:49 +08:00
|
|
|
|
|
|
|
# Escape HTML tags inside python code blocks so they show up properly
|
|
|
|
pattern = "<code class='lang-python'>\n?((.|\n)*?)\n?</code>"
|
|
|
|
guide_output = re.sub(pattern, lambda x: "<code class='lang-python'>" + html.escape(x.group(1)) + "</code>", guide_output)
|
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
output_html = markdown2.markdown(guide_output)
|
|
|
|
output_html = output_html.replace("<a ", "<a target='blank' ")
|
2022-02-05 05:42:49 +08:00
|
|
|
|
2022-01-21 21:44:12 +08:00
|
|
|
for match in re.findall(r"<h3>([A-Za-z0-9 ]*)<\/h3>", output_html):
|
2021-12-14 14:02:19 +08:00
|
|
|
output_html = output_html.replace(
|
2022-01-21 21:44:12 +08:00
|
|
|
f"<h3>{match}</h3>",
|
|
|
|
f"<h3 id={match.lower().replace(' ', '_')}>{match}</h3>",
|
|
|
|
)
|
2022-02-05 05:42:49 +08:00
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
os.makedirs("generated", exist_ok=True)
|
|
|
|
guide = guide[:-3]
|
2022-01-21 21:44:12 +08:00
|
|
|
os.makedirs(os.path.join("generated", guide), exist_ok=True)
|
|
|
|
with open(
|
|
|
|
"src/guides_template.html", encoding="utf-8"
|
|
|
|
) as general_template_file:
|
2021-12-14 14:02:19 +08:00
|
|
|
general_template = Template(general_template_file.read())
|
2022-02-05 05:42:49 +08:00
|
|
|
with open(os.path.join("generated", guide, "index.html"), "w", encoding='utf-8') as generated_template:
|
2022-01-21 21:44:12 +08:00
|
|
|
output_html = general_template.render(
|
2022-02-05 05:42:49 +08:00
|
|
|
template_html=output_html, demo_names=demo_names,
|
|
|
|
meta_tags=meta_tags, navbar_html=navbar_html)
|
2021-12-14 14:02:19 +08:00
|
|
|
generated_template.write(output_html)
|
|
|
|
|
2022-01-21 21:44:12 +08:00
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
def render_docs():
|
|
|
|
if os.path.exists("generated/colab_links.json"):
|
|
|
|
with open("generated/colab_links.json") as demo_links_file:
|
2021-12-22 16:45:23 +08:00
|
|
|
try:
|
|
|
|
demo_links = json.load(demo_links_file)
|
|
|
|
except ValueError:
|
|
|
|
demo_links = {}
|
2021-12-14 14:02:19 +08:00
|
|
|
else: # docs will be missing demo links
|
|
|
|
demo_links = {}
|
|
|
|
SCREENSHOT_FOLDER = "dist/assets/demo_screenshots"
|
|
|
|
os.makedirs(SCREENSHOT_FOLDER, exist_ok=True)
|
|
|
|
|
|
|
|
def get_function_documentation(func):
|
|
|
|
doc_str = inspect.getdoc(func)
|
|
|
|
func_doc, params_doc, return_doc = [], [], []
|
|
|
|
documented_params = {"self"}
|
|
|
|
mode = "pre"
|
|
|
|
for line in doc_str.split("\n"):
|
|
|
|
if line.startswith("Parameters:"):
|
|
|
|
mode = "in"
|
|
|
|
continue
|
|
|
|
if line.startswith("Returns:"):
|
|
|
|
mode = "out"
|
|
|
|
continue
|
|
|
|
if "DEPRECATED" in line:
|
|
|
|
continue
|
|
|
|
if mode == "pre":
|
|
|
|
func_doc.append(line)
|
|
|
|
elif mode == "in":
|
|
|
|
space_index = line.index(" ")
|
|
|
|
colon_index = line.index(":")
|
|
|
|
name = line[:space_index]
|
|
|
|
documented_params.add(name)
|
|
|
|
params_doc.append(
|
2022-01-21 21:44:12 +08:00
|
|
|
(
|
|
|
|
name,
|
|
|
|
line[space_index + 2 : colon_index - 1],
|
|
|
|
line[colon_index + 2 :],
|
|
|
|
)
|
|
|
|
)
|
2021-12-14 14:02:19 +08:00
|
|
|
elif mode == "out":
|
|
|
|
colon_index = line.index(":")
|
2022-01-21 21:44:12 +08:00
|
|
|
return_doc.append((line[1 : colon_index - 1], line[colon_index + 2 :]))
|
2021-12-14 14:02:19 +08:00
|
|
|
params = inspect.getfullargspec(func)
|
|
|
|
param_set = []
|
|
|
|
for i in range(len(params.args)):
|
|
|
|
neg_index = -1 - i
|
|
|
|
if params.args[neg_index] not in documented_params:
|
|
|
|
continue
|
|
|
|
if params.defaults and i < len(params.defaults):
|
|
|
|
default = params.defaults[neg_index]
|
|
|
|
if type(default) == str:
|
|
|
|
default = '"' + default + '"'
|
|
|
|
else:
|
|
|
|
default = str(default)
|
|
|
|
param_set.insert(0, (params.args[neg_index], default))
|
|
|
|
else:
|
|
|
|
param_set.insert(0, (params.args[neg_index],))
|
|
|
|
return "\n".join(func_doc), param_set, params_doc, return_doc
|
|
|
|
|
|
|
|
def get_class_documentation(cls):
|
|
|
|
inp = {}
|
|
|
|
inp["name"] = cls.__name__
|
|
|
|
doc = inspect.getdoc(cls)
|
|
|
|
doc_lines = doc.split("\n")
|
|
|
|
inp["doc"] = "\n".join(doc_lines[:-2])
|
|
|
|
inp["type"] = doc_lines[-2].split("type: ")[-1]
|
|
|
|
inp["demos"] = doc_lines[-1][7:].split(", ")
|
2022-01-21 21:44:12 +08:00
|
|
|
_, inp["params"], inp["params_doc"], _ = get_function_documentation(
|
|
|
|
cls.__init__
|
|
|
|
)
|
2021-12-14 14:02:19 +08:00
|
|
|
inp["shortcuts"] = list(cls.get_shortcut_implementations().items())
|
|
|
|
if "interpret" in cls.__dict__:
|
2022-01-21 21:44:12 +08:00
|
|
|
(
|
|
|
|
inp["interpret"],
|
|
|
|
inp["interpret_params"],
|
|
|
|
inp["interpret_params_doc"],
|
|
|
|
_,
|
|
|
|
) = get_function_documentation(cls.interpret)
|
2021-12-14 14:02:19 +08:00
|
|
|
_, _, _, inp["interpret_returns_doc"] = get_function_documentation(
|
2022-01-21 21:44:12 +08:00
|
|
|
cls.get_interpretation_scores
|
|
|
|
)
|
2021-12-14 14:02:19 +08:00
|
|
|
|
|
|
|
return inp
|
|
|
|
|
|
|
|
inputs = [get_class_documentation(cls) for cls in InputComponent.__subclasses__()]
|
|
|
|
outputs = [get_class_documentation(cls) for cls in OutputComponent.__subclasses__()]
|
|
|
|
interface_params = get_function_documentation(Interface.__init__)
|
|
|
|
interface = {
|
|
|
|
"doc": inspect.getdoc(Interface),
|
|
|
|
"params": interface_params[1],
|
|
|
|
"params_doc": interface_params[2],
|
|
|
|
}
|
|
|
|
launch_params = get_function_documentation(Interface.launch)
|
|
|
|
launch = {
|
|
|
|
"params": launch_params[1],
|
|
|
|
"params_doc": launch_params[2],
|
|
|
|
}
|
|
|
|
load_params = get_function_documentation(Interface.load)
|
|
|
|
load = {
|
|
|
|
"params": load_params[1],
|
|
|
|
"params_doc": load_params[2],
|
|
|
|
"return_doc": load_params[3],
|
|
|
|
}
|
|
|
|
docs = {
|
|
|
|
"input": inputs,
|
|
|
|
"output": outputs,
|
|
|
|
"interface": interface,
|
|
|
|
"launch": launch,
|
|
|
|
"load": load,
|
|
|
|
}
|
|
|
|
os.makedirs("generated", exist_ok=True)
|
|
|
|
with open("src/docs_template.html") as template_file:
|
|
|
|
template = Template(template_file.read())
|
2022-02-05 05:42:49 +08:00
|
|
|
output_html = template.render(docs=docs, demo_links=demo_links,
|
|
|
|
navbar_html=navbar_html)
|
2021-12-14 14:02:19 +08:00
|
|
|
os.makedirs(os.path.join("generated", "docs"), exist_ok=True)
|
2022-01-21 21:44:12 +08:00
|
|
|
with open(
|
|
|
|
os.path.join("generated", "docs", "index.html"), "w"
|
|
|
|
) as generated_template:
|
2021-12-14 14:02:19 +08:00
|
|
|
generated_template.write(output_html)
|
|
|
|
|
2022-01-21 21:44:12 +08:00
|
|
|
|
2021-12-17 01:15:23 +08:00
|
|
|
def render_other():
|
|
|
|
os.makedirs("generated", exist_ok=True)
|
|
|
|
for template_filename in os.listdir("src/other_templates"):
|
2022-01-21 21:44:12 +08:00
|
|
|
with open(
|
|
|
|
os.path.join("src/other_templates", template_filename)
|
|
|
|
) as template_file:
|
2021-12-17 01:15:23 +08:00
|
|
|
template = Template(template_file.read())
|
2022-02-05 05:42:49 +08:00
|
|
|
output_html = template.render()
|
2021-12-17 01:15:23 +08:00
|
|
|
folder_name = template_filename[:-14]
|
|
|
|
os.makedirs(os.path.join("generated", folder_name), exist_ok=True)
|
2022-01-21 21:44:12 +08:00
|
|
|
with open(
|
|
|
|
os.path.join("generated", folder_name, "index.html"), "w", encoding="utf-8"
|
|
|
|
) as generated_template:
|
2021-12-17 01:15:23 +08:00
|
|
|
generated_template.write(output_html)
|
|
|
|
|
|
|
|
|
2021-12-14 14:02:19 +08:00
|
|
|
if __name__ == "__main__":
|
|
|
|
render_index()
|
2022-02-05 05:42:49 +08:00
|
|
|
render_guides_main()
|
2021-12-14 14:02:19 +08:00
|
|
|
render_guides()
|
|
|
|
render_docs()
|
2021-12-17 01:15:23 +08:00
|
|
|
render_other()
|