gradio/README.md

463 lines
20 KiB
Markdown
Raw Normal View History

2021-10-14 13:27:34 +08:00
[![CircleCI](https://circleci.com/gh/gradio-app/gradio.svg?style=svg)](https://circleci.com/gh/gradio-app/gradio) [![PyPI version](https://badge.fury.io/py/gradio.svg)](https://badge.fury.io/py/gradio) [![codecov](https://codecov.io/gh/gradio-app/gradio/branch/master/graph/badge.svg?token=NNVPX9KEGS)](https://codecov.io/gh/gradio-app/gradio) [![PyPI - Downloads](https://img.shields.io/pypi/dm/gradio)](https://pypi.org/project/gradio/) [![Twitter Follow](https://img.shields.io/twitter/follow/gradio.svg?style=social&label=Follow)](https://twitter.com/gradio)
2021-10-14 13:25:27 +08:00
2020-06-09 17:01:12 +08:00
2020-10-27 06:27:28 +08:00
# Welcome to Gradio
2019-09-18 02:05:54 +08:00
2021-11-30 23:57:42 +08:00
Quickly create a GUI around your machine learning model, API, or function. Gradio (prounounced GRAY-dee-o) makes it easy for you to "play around" with your model in your browser by dragging-and-dropping in your own images, pasting your own text, recording your own voice, etc. and seeing what the model outputs.
2020-07-14 19:48:34 +08:00
2020-10-27 06:27:28 +08:00
![Interface montage](demo/screenshots/montage.gif)
2020-06-23 06:27:14 +08:00
2019-06-20 01:21:15 +08:00
Gradio is useful for:
2020-10-27 06:27:28 +08:00
2021-11-04 21:00:56 +08:00
* **Demoing** your machine learning models for clients / collaborators / users / students
* **Deploying** your models quickly with automatic shareable links and getting feedback on model performance
* **Debugging** your model interactively during development using built-in interpretation visualizations for any model
2019-06-20 01:18:07 +08:00
2019-01-25 15:34:09 +08:00
2020-10-27 06:27:28 +08:00
## Getting Started
You can find an interactive version of this README at [https://gradio.app/getting_started](https://gradio.app/getting_started).
### Quick Start
To get Gradio running with a simple example, follow these three steps:
2020-10-27 08:55:41 +08:00
<span>1.</span> Install Gradio from pip.
```bash
2019-02-20 12:39:21 +08:00
pip install gradio
2020-10-27 08:55:41 +08:00
```
2020-10-27 06:27:28 +08:00
2020-10-27 08:55:41 +08:00
<span>2.</span> Run the code below as a Python script or in a Python notebook (or in a [colab notebook](https://colab.research.google.com/drive/18ODkJvyxHutTN0P5APWyGFO_xwNcgHDZ?usp=sharing)).
2020-10-27 06:35:38 +08:00
2020-10-27 08:55:41 +08:00
```python
2020-10-27 06:27:28 +08:00
import gradio as gr
def greet(name):
2021-07-22 04:46:15 +08:00
return "Hello " + name + "!!"
2020-10-27 06:27:28 +08:00
iface = gr.Interface(fn=greet, inputs="text", outputs="text")
iface.launch()
2021-07-22 04:46:15 +08:00
2020-10-27 08:55:41 +08:00
```
2019-02-20 12:39:21 +08:00
2020-10-27 08:55:41 +08:00
<span>3.</span> The interface below will appear automatically within the Python notebook, or pop in a browser on [http://localhost:7860](http://localhost:7860/) if running from a script.
2020-10-27 06:35:38 +08:00
2020-10-27 06:27:28 +08:00
![hello_world interface](demo/screenshots/hello_world/1.gif)
### The Interface
Gradio can wrap almost any Python function with an easy to use interface. That function could be anything from a simple tax calculator to a pretrained model.
2019-02-20 12:39:21 +08:00
2020-10-27 06:27:28 +08:00
The core `Interface` class is initialized with three parameters:
2019-02-20 12:39:21 +08:00
2020-10-27 06:27:28 +08:00
- `fn`: the function to wrap
- `inputs`: the input component type(s)
- `outputs`: the output component type(s)
2019-02-20 12:39:21 +08:00
2020-10-27 06:27:28 +08:00
With these three arguments, we can quickly create interfaces and `launch()` them. But what if you want to change how the UI components look or behave?
2019-02-20 12:39:21 +08:00
2020-10-27 06:27:28 +08:00
### Customizable Components
What if we wanted to customize the input text field - for example, we wanted it to be larger and have a text hint? If we use the actual input class for `Textbox` instead of using the string shortcut, we have access to much more customizability. To see a list of all the components we support and how you can customize them, check out the [Docs](https://gradio.app/docs)
2020-10-27 08:55:41 +08:00
```python
2020-07-08 06:10:54 +08:00
import gradio as gr
2020-06-12 03:44:44 +08:00
2020-07-08 06:10:54 +08:00
def greet(name):
return "Hello " + name + "!"
2020-06-12 03:44:44 +08:00
2020-10-27 06:27:28 +08:00
iface = gr.Interface(
fn=greet,
inputs=gr.inputs.Textbox(lines=2, placeholder="Name Here..."),
outputs="text")
iface.launch()
2020-10-27 08:55:41 +08:00
```
2020-10-27 06:27:28 +08:00
![hello_world_2 interface](demo/screenshots/hello_world_2/1.gif)
2019-02-20 12:39:21 +08:00
2020-10-27 06:27:28 +08:00
### Multiple Inputs and Outputs
2020-06-12 03:48:38 +08:00
2020-10-27 06:27:28 +08:00
Let's say we had a much more complex function, with multiple inputs and outputs. In the example below, we have a function that takes a string, boolean, and number, and returns a string and number. Take a look how we pass a list of input and output components.
2020-06-12 03:48:38 +08:00
2020-10-27 08:55:41 +08:00
```python
2020-10-27 06:27:28 +08:00
import gradio as gr
def greet(name, is_morning, temperature):
salutation = "Good morning" if is_morning else "Good evening"
greeting = "%s %s. It is %s degrees today" % (
salutation, name, temperature)
celsius = (temperature - 32) * 5 / 9
return greeting, round(celsius, 2)
iface = gr.Interface(
fn=greet,
inputs=["text", "checkbox", gr.inputs.Slider(0, 100)],
outputs=["text", "number"])
iface.launch()
2020-10-27 08:55:41 +08:00
```
2020-10-27 06:27:28 +08:00
![hello_world_3 interface](demo/screenshots/hello_world_3/1.gif)
2019-02-20 12:39:21 +08:00
2020-10-27 06:27:28 +08:00
We simply wrap the components in a list. Furthermore, if we wanted to compare multiple functions that have the same input and return types, we can even pass a list of functions for quick comparison.
2019-02-20 12:39:21 +08:00
2020-10-27 06:27:28 +08:00
### Working with Images
2019-02-20 12:39:21 +08:00
2020-10-27 06:27:28 +08:00
Let's try an image to image function. When using the `Image` component, your function will receive a numpy array of your specified size, with the shape `(width, height, 3)`, where the last dimension represents the RGB values. We'll return an image as well in the form of a numpy array.
2019-02-20 12:39:21 +08:00
2020-10-27 08:55:41 +08:00
```python
2020-07-08 06:10:54 +08:00
import gradio as gr
import numpy as np
2019-02-20 12:46:44 +08:00
2021-09-22 03:09:41 +08:00
def sepia(input_img):
2020-10-27 06:27:28 +08:00
sepia_filter = np.array([[.393, .769, .189],
[.349, .686, .168],
[.272, .534, .131]])
2021-09-22 03:09:41 +08:00
sepia_img = input_img.dot(sepia_filter.T)
2020-10-27 06:27:28 +08:00
sepia_img /= sepia_img.max()
return sepia_img
2019-02-20 12:47:45 +08:00
2020-10-27 06:27:28 +08:00
iface = gr.Interface(sepia, gr.inputs.Image(shape=(200, 200)), "image")
2021-09-22 03:09:41 +08:00
2020-10-27 06:27:28 +08:00
iface.launch()
2020-10-27 08:55:41 +08:00
```
2020-10-27 06:27:28 +08:00
![sepia_filter interface](demo/screenshots/sepia_filter/1.gif)
2019-02-20 14:26:48 +08:00
2020-10-27 06:27:28 +08:00
Additionally, our `Image` input interface comes with an 'edit' button which opens tools for cropping, flipping, rotating, drawing over, and applying filters to images. We've found that manipulating images in this way will often reveal hidden flaws in a model.
2019-02-20 14:26:48 +08:00
2021-08-26 05:25:53 +08:00
In addition to images, Gradio supports other media input types, such as audio or video uploads. Read about these in the [Docs](https://gradio.app/docs).
### Working with Data
You can use Gradio to support inputs and outputs from your typical data libraries, such as numpy arrays, pandas dataframes, and Matplotlib plots. Take a look at the demo below (ignore the complicated data manipulation in the function!)
2021-08-26 05:25:53 +08:00
```python
import gradio as gr
import numpy as np
import pandas as pd
import matplotlib.pyplot as plt
def sales_projections(employee_data):
sales_data = employee_data.iloc[:, 1:4].astype("int").to_numpy()
regression_values = np.apply_along_axis(lambda row:
np.array(np.poly1d(np.polyfit([0,1,2], row, 2))), 0, sales_data)
projected_months = np.repeat(np.expand_dims(
np.arange(3,12), 0), len(sales_data), axis=0)
projected_values = np.array([
month * month * regression[0] + month * regression[1] + regression[2]
for month, regression in zip(projected_months, regression_values)])
plt.plot(projected_values.T)
plt.legend(employee_data["Name"])
return employee_data, plt.gcf(), regression_values
iface = gr.Interface(sales_projections,
gr.inputs.Dataframe(
headers=["Name", "Jan Sales", "Feb Sales", "Mar Sales"],
default=[["Jon", 12, 14, 18], ["Alice", 14, 17, 2], ["Sana", 8, 9.5, 12]]
),
[
"dataframe",
"plot",
"numpy"
],
description="Enter sales figures for employees to predict sales trajectory over year."
)
iface.launch()
```
![sales_projections interface](demo/screenshots/sales_projections/1.gif)
### Example Inputs
2019-02-20 14:24:29 +08:00
2021-02-27 02:51:51 +08:00
You can provide example data that a user can easily load into the model. This can be helpful to demonstrate the types of inputs the model expects, as well as to provide a way to explore your dataset in conjunction with your model. To load example data, you provide a **nested list** to the `examples=` keyword argument of the Interface constructor. Each sublist within the outer list represents a data sample, and each element within the sublist represents an input for each input component. The format of example data for each component is specified in the [Docs](https://gradio.app/docs).
2020-10-27 06:27:28 +08:00
2020-10-27 08:55:41 +08:00
```python
2020-07-08 06:10:54 +08:00
import gradio as gr
2020-10-27 06:27:28 +08:00
def calculator(num1, operation, num2):
if operation == "add":
return num1 + num2
elif operation == "subtract":
return num1 - num2
elif operation == "multiply":
return num1 * num2
elif operation == "divide":
return num1 / num2
iface = gr.Interface(calculator,
["number", gr.inputs.Radio(["add", "subtract", "multiply", "divide"]), "number"],
"number",
examples=[
[5, "add", 3],
2020-11-24 07:11:37 +08:00
[4, "divide", 2],
[-4, "multiply", 2.5],
[0, "subtract", 1.2],
2021-02-27 02:51:51 +08:00
],
2021-07-22 04:46:15 +08:00
title="test calculator",
description="heres a sample toy calculator. enjoy!",
2021-08-26 05:25:53 +08:00
flagging_options=["this", "or", "that"],
2020-10-27 06:27:28 +08:00
)
2020-11-24 07:11:37 +08:00
2020-10-27 06:27:28 +08:00
iface.launch()
2020-11-24 07:11:37 +08:00
2020-10-27 08:55:41 +08:00
```
2020-10-27 06:27:28 +08:00
![calculator interface](demo/screenshots/calculator/1.gif)
2021-02-27 02:51:51 +08:00
You can load a large dataset into the examples to browse and interact with the dataset through Gradio. The examples will be automatically paginated (you can configure this through the `examples_per_page` argument of Interface) and you can use CTRL + arrow keys to navigate through the examples quickly.
2020-11-24 07:11:37 +08:00
2021-08-26 05:25:53 +08:00
### Live Interfaces
You can make interfaces automatically responsive by setting `live=True` in the interface. Now the interface will recalculate as soon as the user input.
```python
import gradio as gr
def calculator(num1, operation, num2):
if operation == "add":
return num1 + num2
elif operation == "subtract":
return num1 - num2
elif operation == "multiply":
return num1 * num2
elif operation == "divide":
return num1 / num2
iface = gr.Interface(calculator,
["number", gr.inputs.Radio(["add", "subtract", "multiply", "divide"]), "number"],
"number",
live=True
)
iface.launch()
```
![calculator_live interface](demo/screenshots/calculator_live/1.gif)
2021-09-22 03:09:41 +08:00
Note there is no submit button, because the interface resubmits automatically on change.
### Using State
Your function may use data that persists beyond a single function call. If the data is something accessible to all function calls, you can create a global variable outside the function call and access it inside the function. For example, you may load a large model outside the function and use it inside the function so that every function call does not need to reload the model.
Another type of data persistence Gradio supports is session state, where data persists across multiple submits within a page load. To store data with this permanence, use `gr.get_state` and `gr.set_state` methods.
```python
import gradio as gr
import random
def chat(message):
history = gr.get_state() or []
if message.startswith("How many"):
response = random.randint(1,10)
elif message.startswith("How"):
response = random.choice(["Great", "Good", "Okay", "Bad"])
elif message.startswith("Where"):
response = random.choice(["Here", "There", "Somewhere"])
else:
response = "I don't know"
history.append((message, response))
gr.set_state(history)
html = "<div class='chatbot'>"
for user_msg, resp_msg in history:
html += f"<div class='user_msg'>{user_msg}</div>"
html += f"<div class='resp_msg'>{resp_msg}</div>"
html += "</div>"
return html
iface = gr.Interface(chat, "text", "html", css="""
.chatbox {display:flex;flex-direction:column}
.user_msg, .resp_msg {padding:4px;margin-bottom:4px;border-radius:4px;width:80%}
.user_msg {background-color:cornflowerblue;color:white;align-self:start}
.resp_msg {background-color:lightgray;align-self:self-end}
""", allow_screenshot=False, allow_flagging=False)
iface.launch()
```
![chatbot interface](demo/screenshots/chatbot/1.gif)
Notice how the state persists across submits within each page, but the state is not shared between the two pages.
2021-08-26 05:25:53 +08:00
2021-02-27 02:51:51 +08:00
### Flagging
2020-11-24 07:11:37 +08:00
2021-02-27 02:51:51 +08:00
Underneath the output interfaces, there is a button marked "Flag". When a user testing your model sees input with interesting output, such as erroneous or unexpected model behaviour, they can flag the input for the interface creator to review. Within the directory provided by the `flagging_dir=` argument to the Interface constructor, a CSV file will log the flagged inputs. If the interface involves file data, such as for Image and Audio components, folders will be created to store those flagged data as well.
2020-11-24 07:11:37 +08:00
2021-02-27 02:51:51 +08:00
For example, with the calculator interface shown above, we would have the flagged data stored in the flagged directory shown below:
2021-02-27 04:32:35 +08:00
```directory
2021-02-27 02:51:51 +08:00
+-- calculator.py
+-- flagged/
| +-- logs.csv
2021-02-27 02:56:19 +08:00
```
2021-02-27 02:51:51 +08:00
*flagged/logs.csv*
2021-02-27 04:32:35 +08:00
```csv
2021-02-27 02:51:51 +08:00
num1,operation,num2,Output
5,add,7,12
6,subtract,1.5,4.5
2021-02-27 02:57:52 +08:00
```
2021-02-27 02:51:51 +08:00
With the sepia interface shown above, we would have the flagged data stored in the flagged directory shown below:
2021-02-27 04:32:35 +08:00
```directory
2021-02-27 02:51:51 +08:00
+-- sepia.py
+-- flagged/
| +-- logs.csv
| +-- im/
| | +-- 0.png
| | +-- 1.png
| +-- Output/
| | +-- 0.png
| | +-- 1.png
2021-02-27 02:56:19 +08:00
```
2021-02-27 02:51:51 +08:00
2021-02-27 04:32:35 +08:00
*flagged/logs.csv*
```csv
2021-02-27 02:51:51 +08:00
im,Output
im/0.png,Output/0.png
im/1.png,Output/1.png
2021-02-27 02:57:52 +08:00
```
2021-02-27 02:51:51 +08:00
You can review these flagged inputs by manually exploring the flagging directory, or load them into the examples of the Gradio interface by pointing the `examples=` argument to the flagged directory. If you wish for the user to provide a reason for flagging, you can pass a list of strings to the `flagging_options` argument of Interface. Users will have to select one of the strings when flagging, which will be saved as an additional column to the CSV.
### Sharing Interfaces Publicly & Privacy
Interfaces can be easily shared publicly by setting `share=True` in the `launch()` method. Like this:
```python
gr.Interface(classify_image, "image", "label").launch(share=True)
```
This generates a public, shareable link that you can send to anybody! When you send this link, the user on the other side can try out the model in their browser. Because the processing happens on your device (as long as your device stays on!), you don't have to worry about any dependencies. If you're working out of colab notebook, a share link is always automatically created. It usually looks something like this: **XXXXX.gradio.app**. Although the link is served through a gradio link, we are only a proxy for your local server, and do not store any data sent through the interfaces.
2021-07-22 04:46:15 +08:00
Keep in mind, however, that these links are publicly accessible, meaning that anyone can use your model for prediction! Therefore, make sure not to expose any sensitive information through the functions you write, or allow any critical changes to occur on your device. If you set `share=False` (the default), only a local link is created, which can be shared by [port-forwarding](https://www.ssh.com/ssh/tunneling/example) with specific users.
2021-02-27 02:51:51 +08:00
2021-07-22 04:46:15 +08:00
Share links expire after 72 hours. For permanent hosting, see below.
2021-02-27 02:51:51 +08:00
![Sharing diagram](demo/images/sharing.svg)
2021-08-26 05:25:53 +08:00
### Authentication
You may wish to put an authentication page in front of your interface to limit access. With the `auth=` keyword argument in the `launch()` method, you can pass a list of acceptable username/password tuples; or, for custom authentication handling, pass a function that takes a username and password as arguments, and returns True to allow authentication, False otherwise.
2021-02-27 02:51:51 +08:00
### Permanent Hosting
You can share your interface publicly and permanently by hosting on Gradio's infrastructure. You will need to create a Gradio premium account. First, log into Gradio on [gradio.app](https://gradio.app) and click Sign In at the top. Once you've logged in with your Github account, you can specify which repositories from your Github profile you'd like to have hosted by Gradio. You must also specify the file within the repository that runs the Gradio `launch()` command. Once you've taken these steps, Gradio will launch your interface and provide a public link you can share.
2020-10-27 06:27:28 +08:00
2021-02-27 02:51:51 +08:00
## Advanced Features
2020-10-27 06:27:28 +08:00
### Interpretation
Most models are black boxes such that the internal logic of the function is hidden from the end user. To encourage transparency, we've added the ability for interpretation so that users can understand what parts of the input are responsible for the output. Take a look at the simple interface below:
2020-10-27 08:55:41 +08:00
```python
2020-10-27 06:27:28 +08:00
import gradio as gr
import re
2020-07-08 06:10:54 +08:00
2020-10-27 06:27:28 +08:00
male_words, female_words = ["he", "his", "him"], ["she", "her"]
def gender_of_sentence(sentence):
male_count = len([word for word in sentence.split() if word.lower() in male_words])
female_count = len([word for word in sentence.split() if word.lower() in female_words])
total = max(male_count + female_count, 1)
return {"male": male_count / total, "female": female_count / total}
2020-07-08 06:10:54 +08:00
2020-10-27 06:27:28 +08:00
iface = gr.Interface(
fn=gender_of_sentence, inputs=gr.inputs.Textbox(default="She went to his house to get her keys."),
outputs="label", interpretation="default")
iface.launch()
2020-07-08 06:10:54 +08:00
2020-10-27 08:55:41 +08:00
```
2020-10-27 06:27:28 +08:00
![gender_sentence_default_interpretation interface](demo/screenshots/gender_sentence_default_interpretation/1.gif)
Notice the `interpretation` keyword argument. We're going to use Gradio's default interpreter here. After you submit and click Interpret, you'll see the interface automatically highlights the parts of the text that contributed to the final output orange! The parts that conflict with the output are highlight blue.
You can also write your own interpretation function. The demo below adds custom interpretation to the previous demo. This function will take the same inputs as the main wrapped function. The output of this interpretation function will be used to highlight the input of each input interface - therefore the number of outputs here corresponds to the number of input interfaces. To see the format for interpretation for each input interface, check the [Docs](https://gradio.app/docs).
2020-10-27 08:55:41 +08:00
```python
2020-10-27 06:27:28 +08:00
import gradio as gr
import re
male_words, female_words = ["he", "his", "him"], ["she", "her"]
def gender_of_sentence(sentence):
male_count = len([word for word in sentence.split() if word.lower() in male_words])
female_count = len([word for word in sentence.split() if word.lower() in female_words])
total = max(male_count + female_count, 1)
return {"male": male_count / total, "female": female_count / total}
def interpret_gender(sentence):
result = gender_of_sentence(sentence)
is_male = result["male"] > result["female"]
interpretation = []
for word in re.split('( )', sentence):
score = 0
token = word.lower()
if (is_male and token in male_words) or (not is_male and token in female_words):
score = 1
2020-10-27 08:55:41 +08:00
elif (is_male and token in female_words) or (not is_male and token in male_words):
2020-10-27 06:27:28 +08:00
score = -1
interpretation.append((word, score))
return interpretation
iface = gr.Interface(
fn=gender_of_sentence, inputs=gr.inputs.Textbox(default="She went to his house to get her keys."),
2021-08-26 05:25:53 +08:00
outputs="label", interpretation=interpret_gender, enable_queue=True)
2020-10-27 06:27:28 +08:00
iface.launch()
2020-10-27 08:55:41 +08:00
```
2020-10-27 06:27:28 +08:00
![gender_sentence_custom_interpretation interface](demo/screenshots/gender_sentence_custom_interpretation/1.gif)
2020-07-08 06:10:54 +08:00
2021-02-27 02:51:51 +08:00
If you use Gradio's default interpretation, the output component must be a label or a number. All input components are supported for default interpretation. Below is an example with image input.
2020-10-27 08:55:41 +08:00
```python
2021-02-27 02:51:51 +08:00
import gradio as gr
import tensorflow as tf
import numpy as np
import json
from os.path import dirname, realpath, join
2020-10-27 08:55:41 +08:00
2021-02-27 02:51:51 +08:00
# Load human-readable labels for ImageNet.
current_dir = dirname(realpath(__file__))
with open(join(current_dir, "files/imagenet_labels.json")) as labels_file:
labels = json.load(labels_file)
2020-10-27 08:55:41 +08:00
2021-02-27 02:51:51 +08:00
mobile_net = tf.keras.applications.MobileNetV2()
def image_classifier(im):
arr = np.expand_dims(im, axis=0)
arr = tf.keras.applications.mobilenet.preprocess_input(arr)
prediction = mobile_net.predict(arr).flatten()
return {labels[i]: float(prediction[i]) for i in range(1000)}
2020-10-27 08:55:41 +08:00
2021-02-27 02:51:51 +08:00
iface = gr.Interface(
image_classifier,
gr.inputs.Image(shape=(224, 224)),
gr.outputs.Label(num_top_classes=3),
capture_session=True,
interpretation="default",
examples=[
["images/cheetah1.jpg"],
["images/lion.jpg"]
])
2020-10-27 08:55:41 +08:00
2021-02-27 02:51:51 +08:00
iface.launch()
2020-12-04 03:47:08 +08:00
2021-02-27 02:51:51 +08:00
```
![image_classifier interface](demo/screenshots/image_classifier/1.gif)
2020-12-04 03:47:08 +08:00
2020-10-27 06:27:28 +08:00
## Contributing:
2020-07-08 06:10:54 +08:00
2020-06-19 00:40:19 +08:00
If you would like to contribute and your contribution is small, you can directly open a pull request (PR). If you would like to contribute a larger feature, we recommend first creating an issue with a proposed design for discussion. Please see our contributing guidelines for more info.
2020-10-27 06:27:28 +08:00
## License:
2020-07-03 13:21:10 +08:00
Gradio is licensed under the Apache License 2.0
2020-10-27 06:27:28 +08:00
## See more:
2020-07-08 06:20:30 +08:00
You can find many more examples (like GPT-2, model comparison, multiple inputs, and numerical interfaces) as well as more info on usage on our website: www.gradio.app
2019-02-20 12:46:44 +08:00
2020-07-08 06:10:54 +08:00
See, also, the accompanying paper: ["Gradio: Hassle-Free Sharing and Testing of ML Models in the Wild"](https://arxiv.org/pdf/1906.02569.pdf), *ICML HILL 2019*, and please use the citation below.
```
@article{abid2019gradio,
title={Gradio: Hassle-Free Sharing and Testing of ML Models in the Wild},
author={Abid, Abubakar and Abdalla, Ali and Abid, Ali and Khan, Dawood and Alfozan, Abdulrahman and Zou, James},
journal={arXiv preprint arXiv:1906.02569},
year={2019}
}
2021-01-29 22:52:33 +08:00
```