[WIP] Language Agnostic Typing in /info route (#4039)

* First commit

* All serializers

* Remove output type

* Add route

* Format json

* Modify dropdown and slider choices

* Fix impl

* Lint

* Add tests

* Fix lint

* remove breakpoint

* Tests passing locally

* Format code

* Address comments

* Use union + fix tests

* handle multiple file case

* Add serializer to info payload

* lint

* Add to CHANGELOG

* grc version

* requirements

* fix test

---------

Co-authored-by: Abubakar Abid <abubakar@huggingface.co>
This commit is contained in:
Freddy Boulton 2023-05-05 20:10:34 -04:00 committed by GitHub
parent d1853625fd
commit b66ecff671
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
16 changed files with 662 additions and 282 deletions

View File

@ -2,11 +2,10 @@
## New Features:
No changes to highlight.
- Returning language agnostic types in the `/info` route by [@freddyaboulton](https://github.com/freddyaboulton) in [PR 4039](https://github.com/gradio-app/gradio/pull/4039)
## Bug Fixes:
- Fixed bug where type hints in functions caused the event handler to crash by [@freddyaboulton](https://github.com/freddyaboulton) in [PR 4068](https://github.com/gradio-app/gradio/pull/4068)
- Fix dropdown default value not appearing by [@aliabid94](https://github.com/aliabid94) in [PR 4072](https://github.com/gradio-app/gradio/pull/4072).
- Soft theme label color fix by [@aliabid94](https://github.com/aliabid94) in [PR 4070](https://github.com/gradio-app/gradio/pull/4070)

View File

@ -30,7 +30,12 @@ from typing_extensions import Literal
from gradio_client import serializing, utils
from gradio_client.documentation import document, set_documentation_group
from gradio_client.serializing import Serializable
from gradio_client.utils import Communicator, JobStatus, Status, StatusUpdate
from gradio_client.utils import (
Communicator,
JobStatus,
Status,
StatusUpdate,
)
set_documentation_group("py-client")
@ -399,17 +404,17 @@ class Client:
api_info_url = urllib.parse.urljoin(self.src, utils.RAW_API_INFO_URL)
r = requests.get(api_info_url, headers=self.headers)
# Versions of Gradio older than 3.26 returned format of the API info
# Versions of Gradio older than 3.28.3 returned format of the API info
# from the /info endpoint
if (
version.parse(self.config.get("version", "2.0")) >= version.Version("3.26")
version.parse(self.config.get("version", "2.0")) > version.Version("3.28.3")
and r.ok
):
info = r.json()
else:
fetch = requests.post(
utils.SPACE_FETCHER_URL,
json={"serialize": self.serialize, "config": json.dumps(self.config)},
json={"config": json.dumps(self.config), "serialize": self.serialize},
)
if fetch.ok:
info = fetch.json()["api"]
@ -449,7 +454,7 @@ class Client:
def _render_endpoints_info(
self,
name_or_index: str | int,
endpoints_info: dict[str, list[dict[str, str]]],
endpoints_info: dict[str, list[dict[str, Any]]],
) -> str:
parameter_names = [p["label"] for p in endpoints_info["parameters"]]
parameter_names = [utils.sanitize_parameter_names(p) for p in parameter_names]
@ -473,13 +478,25 @@ class Client:
human_info += " Parameters:\n"
if endpoints_info["parameters"]:
for info in endpoints_info["parameters"]:
human_info += f" - [{info['component']}] {utils.sanitize_parameter_names(info['label'])}: {info['type_python']} ({info['type_description']})\n"
desc = (
f" ({info['python_type']['description']})"
if info["python_type"].get("description")
else ""
)
type_ = info["python_type"]["type"]
human_info += f" - [{info['component']}] {utils.sanitize_parameter_names(info['label'])}: {type_}{desc} \n"
else:
human_info += " - None\n"
human_info += " Returns:\n"
if endpoints_info["returns"]:
for info in endpoints_info["returns"]:
human_info += f" - [{info['component']}] {utils.sanitize_parameter_names(info['label'])}: {info['type_python']} ({info['type_description']})\n"
desc = (
f" ({info['python_type']['description']})"
if info["python_type"].get("description")
else ""
)
type_ = info["python_type"]["type"]
human_info += f" - [{info['component']}] {utils.sanitize_parameter_names(info['label'])}: {type_}{desc} \n"
else:
human_info += " - None\n"

View File

@ -9,8 +9,17 @@ from typing import Any
from gradio_client import media_data, utils
from gradio_client.data_classes import FileData
serializer_types = json.load(open(Path(__file__).parent / "types.json"))
class Serializable:
def serialized_info(self):
"""
The typing information for this component as a dictionary whose values are a list of 2 strings: [Python type, language-agnostic description].
Keys of the dictionary are: raw_input, raw_output, serialized_input, serialized_output
"""
return self.api_info()
def api_info(self) -> dict[str, list[str]]:
"""
The typing information for this component as a dictionary whose values are a list of 2 strings: [Python type, language-agnostic description].
@ -57,12 +66,10 @@ class Serializable:
class SimpleSerializable(Serializable):
"""General class that does not perform any serialization or deserialization."""
def api_info(self) -> dict[str, str | list[str]]:
def api_info(self) -> dict[str, bool | dict]:
return {
"raw_input": ["Any", ""],
"raw_output": ["Any", ""],
"serialized_input": ["Any", ""],
"serialized_output": ["Any", ""],
"info": serializer_types["SimpleSerializable"],
"serialized_info": False,
}
def example_inputs(self) -> dict[str, Any]:
@ -75,12 +82,10 @@ class SimpleSerializable(Serializable):
class StringSerializable(Serializable):
"""Expects a string as input/output but performs no serialization."""
def api_info(self) -> dict[str, list[str]]:
def api_info(self) -> dict[str, bool | dict]:
return {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
"info": serializer_types["StringSerializable"],
"serialized_info": False,
}
def example_inputs(self) -> dict[str, Any]:
@ -93,12 +98,10 @@ class StringSerializable(Serializable):
class ListStringSerializable(Serializable):
"""Expects a list of strings as input/output but performs no serialization."""
def api_info(self) -> dict[str, list[str]]:
def api_info(self) -> dict[str, bool | dict]:
return {
"raw_input": ["List[str]", "list of string values"],
"raw_output": ["List[str]", "list of string values"],
"serialized_input": ["List[str]", "list of string values"],
"serialized_output": ["List[str]", "list of string values"],
"info": serializer_types["ListStringSerializable"],
"serialized_info": False,
}
def example_inputs(self) -> dict[str, Any]:
@ -111,12 +114,10 @@ class ListStringSerializable(Serializable):
class BooleanSerializable(Serializable):
"""Expects a boolean as input/output but performs no serialization."""
def api_info(self) -> dict[str, list[str]]:
def api_info(self) -> dict[str, bool | dict]:
return {
"raw_input": ["bool", "boolean value"],
"raw_output": ["bool", "boolean value"],
"serialized_input": ["bool", "boolean value"],
"serialized_output": ["bool", "boolean value"],
"info": serializer_types["BooleanSerializable"],
"serialized_info": False,
}
def example_inputs(self) -> dict[str, Any]:
@ -129,12 +130,10 @@ class BooleanSerializable(Serializable):
class NumberSerializable(Serializable):
"""Expects a number (int/float) as input/output but performs no serialization."""
def api_info(self) -> dict[str, list[str]]:
def api_info(self) -> dict[str, bool | dict]:
return {
"raw_input": ["int | float", "numeric value"],
"raw_output": ["int | float", "numeric value"],
"serialized_input": ["int | float", "numeric value"],
"serialized_output": ["int | float", "numeric value"],
"info": serializer_types["NumberSerializable"],
"serialized_info": False,
}
def example_inputs(self) -> dict[str, Any]:
@ -147,13 +146,11 @@ class NumberSerializable(Serializable):
class ImgSerializable(Serializable):
"""Expects a base64 string as input/output which is serialized to a filepath."""
def api_info(self) -> dict[str, list[str]]:
return {
"raw_input": ["str", "base64 representation of image"],
"raw_output": ["str", "base64 representation of image"],
"serialized_input": ["str", "filepath or URL to image"],
"serialized_output": ["str", "filepath or URL to image"],
}
def serialized_info(self):
return {"type": "string", "description": "filepath or URL to image"}
def api_info(self) -> dict[str, bool | dict]:
return {"info": serializer_types["ImgSerializable"], "serialized_info": True}
def example_inputs(self) -> dict[str, Any]:
return {
@ -204,20 +201,34 @@ class ImgSerializable(Serializable):
class FileSerializable(Serializable):
"""Expects a dict with base64 representation of object as input/output which is serialized to a filepath."""
def api_info(self) -> dict[str, list[str]]:
def serialized_info(self):
return self._single_file_serialized_info()
def _single_file_api_info(self):
return {
"raw_input": [
"str | Dict",
"base64 string representation of file; or a dictionary-like object, the keys should be either: is_file (False), data (base64 representation of file) or is_file (True), name (str filename)",
],
"raw_output": [
"Dict",
"dictionary-like object with keys: name (str filename), data (base64 representation of file), is_file (bool, set to False)",
],
"serialized_input": ["str", "filepath or URL to file"],
"serialized_output": ["str", "filepath or URL to file"],
"info": serializer_types["SingleFileSerializable"],
"serialized_info": True,
}
def _single_file_serialized_info(self):
return {"type": "string", "description": "filepath or URL to file"}
def _multiple_file_serialized_info(self):
return {
"type": "array",
"description": "List of filepath(s) or URL(s) to files",
"items": {"type": "string", "description": "filepath or URL to file"},
}
def _multiple_file_api_info(self):
return {
"info": serializer_types["MultipleFileSerializable"],
"serialized_info": True,
}
def api_info(self) -> dict[str, dict | bool]:
return self._single_file_api_info()
def example_inputs(self) -> dict[str, Any]:
return {
"raw": {"is_file": False, "data": media_data.BASE64_FILE},
@ -331,19 +342,11 @@ class FileSerializable(Serializable):
class VideoSerializable(FileSerializable):
def api_info(self) -> dict[str, list[str]]:
return {
"raw_input": [
"str | Dict",
"base64 string representation of file; or a dictionary-like object, the keys should be either: is_file (False), data (base64 representation of file) or is_file (True), name (str filename)",
],
"raw_output": [
"Tuple[Dict, Dict]",
"a tuple of 2 dictionary-like object with keys: name (str filename), data (base64 representation of file), is_file (bool, set to False). First dictionary is for the video, second dictionary is for the subtitles.",
],
"serialized_input": ["str", "filepath or URL to file"],
"serialized_output": ["str", "filepath or URL to file"],
}
def serialized_info(self):
return {"type": "string", "description": "filepath or URL to video file"}
def api_info(self) -> dict[str, dict | bool]:
return {"info": serializer_types["FileSerializable"], "serialized_info": True}
def example_inputs(self) -> dict[str, Any]:
return {
@ -378,13 +381,11 @@ class VideoSerializable(FileSerializable):
class JSONSerializable(Serializable):
def api_info(self) -> dict[str, list[str]]:
return {
"raw_input": ["str | Dict | List", "JSON-serializable object or a string"],
"raw_output": ["Dict | List", "dictionary- or list-like object"],
"serialized_input": ["str", "filepath to JSON file"],
"serialized_output": ["str", "filepath to JSON file"],
}
def serialized_info(self):
return {"type": "string", "description": "filepath to JSON file"}
def api_info(self) -> dict[str, dict | bool]:
return {"info": serializer_types["JSONSerializable"], "serialized_info": True}
def example_inputs(self) -> dict[str, Any]:
return {
@ -430,24 +431,16 @@ class JSONSerializable(Serializable):
class GallerySerializable(Serializable):
def api_info(self) -> dict[str, list[str]]:
def serialized_info(self):
return {
"raw_input": [
"List[List[str | None]]",
"List of lists. The inner lists should contain two elements: a base64 file representation and an optional caption, the outer list should contain one such list for each image in the gallery.",
],
"raw_output": [
"List[List[str | None]]",
"List of lists. The inner lists should contain two elements: a base64 file representation and an optional caption, the outer list should contain one such list for each image in the gallery.",
],
"serialized_input": [
"str",
"path to directory with images and a file associating images with captions called captions.json",
],
"serialized_output": [
"str",
"path to directory with images and a file associating images with captions called captions.json",
],
"type": "string",
"description": "path to directory with images and a file associating images with captions called captions.json",
}
def api_info(self) -> dict[str, dict | bool]:
return {
"info": serializer_types["GallerySerializable"],
"serialized_info": True,
}
def example_inputs(self) -> dict[str, Any]:

View File

@ -0,0 +1,199 @@
{
"SimpleSerializable": {
"type": {},
"description": "any valid value"
},
"StringSerializable": {
"type": "string"
},
"ListStringSerializable": {
"type": "array",
"items": {
"type": "string"
}
},
"BooleanSerializable": {
"type": "boolean"
},
"NumberSerializable": {
"type": "number"
},
"ImgSerializable": {
"type": "string",
"description": "base64 representation of an image"
},
"FileSerializable": {
"oneOf": [
{
"type": "string",
"description": "filepath or URL to file"
},
{
"type": "object",
"properties": {
"name": { "type": "string", "description": "name of file" },
"data": {
"type": "string",
"description": "base64 representation of file"
},
"size": {
"type": "integer",
"description": "size of image in bytes"
},
"is_file": {
"type": "boolean",
"description": "true if the file has been uploaded to the server"
},
"orig_name": {
"type": "string",
"description": "original name of the file"
}
},
"required": ["name", "data"]
},
{
"type": "array",
"items": {
"anyOf": [
{
"type": "string",
"description": "filepath or URL to file"
},
{
"type": "object",
"properties": {
"name": { "type": "string", "description": "name of file" },
"data": {
"type": "string",
"description": "base64 representation of file"
},
"size": {
"type": "integer",
"description": "size of image in bytes"
},
"is_file": {
"type": "boolean",
"description": "true if the file has been uploaded to the server"
},
"orig_name": {
"type": "string",
"description": "original name of the file"
}
},
"required": ["name", "data"]
}
]
}
}
]
},
"SingleFileSerializable": {
"oneOf": [
{
"type": "string",
"description": "filepath or URL to file"
},
{
"type": "object",
"properties": {
"name": { "type": "string", "description": "name of file" },
"data": {
"type": "string",
"description": "base64 representation of file"
},
"size": {
"type": "integer",
"description": "size of image in bytes"
},
"is_file": {
"type": "boolean",
"description": "true if the file has been uploaded to the server"
},
"orig_name": {
"type": "string",
"description": "original name of the file"
}
},
"required": ["name", "data"]
}
]
},
"MultipleFileSerializable": {
"type": "array",
"items": {
"anyOf": [
{
"type": "string",
"description": "filepath or URL to file"
},
{
"type": "object",
"properties": {
"name": { "type": "string", "description": "name of file" },
"data": {
"type": "string",
"description": "base64 representation of file"
},
"size": {
"type": "integer",
"description": "size of image in bytes"
},
"is_file": {
"type": "boolean",
"description": "true if the file has been uploaded to the server"
},
"orig_name": {
"type": "string",
"description": "original name of the file"
}
},
"required": ["name", "data"]
}
]
}
},
"JSONSerializable": {
"type": {},
"description": "any valid json"
},
"GallerySerializable": {
"type": "array",
"items": {
"type": "array",
"items": false,
"maxSize": 2,
"minSize": 2,
"prefixItems": [
{
"type": "object",
"properties": {
"name": { "type": "string", "description": "name of file" },
"data": {
"type": "string",
"description": "base64 representation of file"
},
"size": {
"type": "integer",
"description": "size of image in bytes"
},
"is_file": {
"type": "boolean",
"description": "true if the file has been uploaded to the server"
},
"orig_name": {
"type": "string",
"description": "original name of the file"
}
},
"required": ["name", "data"]
},
{
"oneOf": [
{ "type": "string", "description": "caption of image" },
{ "type": "null" }
]
}
]
}
}
}

View File

@ -29,7 +29,7 @@ UPLOAD_URL = "/upload"
CONFIG_URL = "/config"
API_INFO_URL = "/info"
RAW_API_INFO_URL = "/info?serialize=False"
SPACE_FETCHER_URL = "https://gradio-space-api-fetcher.hf.space/api"
SPACE_FETCHER_URL = "https://gradio-space-api-fetcher-v2.hf.space/api"
RESET_URL = "/reset"
SPACE_URL = "https://hf.space/{}"
@ -487,3 +487,61 @@ def synchronize_async(func: Callable, *args, **kwargs) -> Any:
**kwargs:
"""
return fsspec.asyn.sync(fsspec.asyn.get_loop(), func, *args, **kwargs) # type: ignore
class APIInfoParseError(ValueError):
pass
def get_type(schema: dict):
if "type" in schema:
return schema["type"]
elif schema.get("oneOf"):
return "oneOf"
elif schema.get("anyOf"):
return "anyOf"
else:
raise APIInfoParseError(f"Cannot parse type for {schema}")
def json_schema_to_python_type(schema: Any) -> str:
"""Convert the json schema into a python type hint"""
type_ = get_type(schema)
if type_ == {}:
if "json" in schema["description"]:
return "Dict[Any, Any]"
else:
return "Any"
elif type_ == "null":
return "None"
elif type_ == "integer":
return "int"
elif type_ == "string":
return "str"
elif type_ == "boolean":
return "bool"
elif type_ == "number":
return "int | float"
elif type_ == "array":
items = schema.get("items")
if "prefixItems" in items:
elements = ", ".join(
[json_schema_to_python_type(i) for i in items["prefixItems"]]
)
return f"Tuple[{elements}]"
else:
elements = json_schema_to_python_type(items)
return f"List[{elements}]"
elif type_ == "object":
des = ", ".join(
[
f"{n}: {json_schema_to_python_type(v)} ({v.get('description')})"
for n, v in schema["properties"].items()
]
)
return f"Dict({des})"
elif type_ in ["oneOf", "anyOf"]:
desc = " | ".join([json_schema_to_python_type(i) for i in schema[type_]])
return desc
else:
raise APIInfoParseError(f"Cannot parse schema {schema}")

View File

@ -1 +1 @@
0.1.4
0.2.0

View File

@ -157,3 +157,14 @@ def count_generator_demo():
list_btn.click(show, num, out)
return demo.queue()
@pytest.fixture
def file_io_demo():
demo = gr.Interface(
lambda x: print("foox"),
[gr.File(file_count="multiple"), "file"],
[gr.File(file_count="multiple"), "file"],
)
return demo

View File

@ -14,7 +14,7 @@ import pytest
from huggingface_hub.utils import RepositoryNotFoundError
from gradio_client import Client
from gradio_client.serializing import SimpleSerializable
from gradio_client.serializing import Serializable
from gradio_client.utils import Communicator, ProgressUnit, Status, StatusUpdate
os.environ["HF_HUB_DISABLE_TELEMETRY"] = "1"
@ -490,32 +490,45 @@ class TestAPIInfo:
"parameters": [
{
"label": "Sex",
"type_python": "str",
"type_description": "string value",
"type": {"type": "string"},
"python_type": {"type": "str", "description": ""},
"component": "Radio",
"example_input": "Howdy!",
"serializer": "StringSerializable",
},
{
"label": "Age",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Slider",
"example_input": 5,
"serializer": "NumberSerializable",
},
{
"label": "Fare (british pounds)",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Slider",
"example_input": 5,
"serializer": "NumberSerializable",
},
],
"returns": [
{
"label": "output",
"type_python": "str",
"type_description": "filepath to JSON file",
"type": {"type": {}, "description": "any valid json"},
"python_type": {
"type": "str",
"description": "filepath to JSON file",
},
"component": "Label",
"serializer": "JSONSerializable",
}
],
},
@ -523,32 +536,45 @@ class TestAPIInfo:
"parameters": [
{
"label": "Sex",
"type_python": "str",
"type_description": "string value",
"type": {"type": "string"},
"python_type": {"type": "str", "description": ""},
"component": "Radio",
"example_input": "Howdy!",
"serializer": "StringSerializable",
},
{
"label": "Age",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Slider",
"example_input": 5,
"serializer": "NumberSerializable",
},
{
"label": "Fare (british pounds)",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Slider",
"example_input": 5,
"serializer": "NumberSerializable",
},
],
"returns": [
{
"label": "output",
"type_python": "str",
"type_description": "filepath to JSON file",
"type": {"type": {}, "description": "any valid json"},
"python_type": {
"type": "str",
"description": "filepath to JSON file",
},
"component": "Label",
"serializer": "JSONSerializable",
}
],
},
@ -556,32 +582,45 @@ class TestAPIInfo:
"parameters": [
{
"label": "Sex",
"type_python": "str",
"type_description": "string value",
"type": {"type": "string"},
"python_type": {"type": "str", "description": ""},
"component": "Radio",
"example_input": "Howdy!",
"serializer": "StringSerializable",
},
{
"label": "Age",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Slider",
"example_input": 5,
"serializer": "NumberSerializable",
},
{
"label": "Fare (british pounds)",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Slider",
"example_input": 5,
"serializer": "NumberSerializable",
},
],
"returns": [
{
"label": "output",
"type_python": "str",
"type_description": "filepath to JSON file",
"type": {"type": {}, "description": "any valid json"},
"python_type": {
"type": "str",
"description": "filepath to JSON file",
},
"component": "Label",
"serializer": "JSONSerializable",
}
],
},
@ -593,8 +632,7 @@ class TestAPIInfo:
def test_serializable_in_mapping(self, calculator_demo):
with connect(calculator_demo) as client:
assert all(
isinstance(c, SimpleSerializable)
for c in client.endpoints[0].serializers
isinstance(c, Serializable) for c in client.endpoints[0].serializers
)
@pytest.mark.flaky
@ -609,18 +647,20 @@ class TestAPIInfo:
"parameters": [
{
"label": "x",
"type_python": "str",
"type_description": "string value",
"type": {"type": "string"},
"python_type": {"type": "str", "description": ""},
"component": "Textbox",
"example_input": "Howdy!",
"serializer": "StringSerializable",
}
],
"returns": [
{
"label": "output",
"type_python": "str",
"type_description": "string value",
"type": {"type": "string"},
"python_type": {"type": "str", "description": ""},
"component": "Textbox",
"serializer": "StringSerializable",
}
],
}
@ -636,32 +676,45 @@ class TestAPIInfo:
"parameters": [
{
"label": "num1",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Number",
"example_input": 5,
"serializer": "NumberSerializable",
},
{
"label": "operation",
"type_python": "str",
"type_description": "string value",
"type": {"type": "string"},
"python_type": {"type": "str", "description": ""},
"component": "Radio",
"example_input": "Howdy!",
"serializer": "StringSerializable",
},
{
"label": "num2",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Number",
"example_input": 5,
"serializer": "NumberSerializable",
},
],
"returns": [
{
"label": "output",
"type_python": "int | float",
"type_description": "numeric value",
"type": {"type": "number"},
"python_type": {
"type": "int | float",
"description": "",
},
"component": "Number",
"serializer": "NumberSerializable",
}
],
}
@ -676,6 +729,28 @@ class TestAPIInfo:
assert "fn_index=0" in info
assert "api_name" not in info
def test_file_io(self, file_io_demo):
with connect(file_io_demo) as client:
info = client.view_api(return_format="dict")
inputs = info["named_endpoints"]["/predict"]["parameters"]
outputs = info["named_endpoints"]["/predict"]["returns"]
assert inputs[0]["python_type"] == {
"type": "List[str]",
"description": "List of filepath(s) or URL(s) to files",
}
assert inputs[1]["python_type"] == {
"type": "str",
"description": "filepath or URL to file",
}
assert outputs[0]["python_type"] == {
"type": "List[str]",
"description": "List of filepath(s) or URL(s) to files",
}
assert outputs[1]["python_type"] == {
"type": "str",
"description": "filepath or URL to file",
}
class TestEndpoints:
def test_upload(self):

View File

@ -1,12 +1,24 @@
import os
import tempfile
import pytest
from gradio import components
from gradio_client.serializing import COMPONENT_MAPPING, FileSerializable
from gradio_client.serializing import COMPONENT_MAPPING, FileSerializable, Serializable
from gradio_client.utils import encode_url_or_file_to_base64
@pytest.mark.parametrize("serializer_class", Serializable.__subclasses__())
def test_duplicate(serializer_class):
if "gradio_client" not in serializer_class.__module__:
pytest.skip(f"{serializer_class} not defined in gradio_client")
serializer = serializer_class()
info = serializer.api_info()
assert "info" in info and "serialized_info" in info
if "serialized_info" in info:
assert serializer.serialized_info()
def test_check_component_fallback_serializers():
for component_name, class_type in COMPONENT_MAPPING.items():
if component_name == "dataset": # cannot be instantiated without parameters

View File

@ -1,3 +1,4 @@
import importlib.resources
import json
import tempfile
from copy import deepcopy
@ -9,6 +10,13 @@ from requests.exceptions import HTTPError
from gradio_client import media_data, utils
types = json.loads(importlib.resources.read_text("gradio_client", "types.json"))
types["MultipleFile"] = {
"type": "array",
"items": {"type": "string", "description": "filepath or URL to file"},
}
types["SingleFile"] = {"type": "string", "description": "filepath or URL to file"}
def test_encode_url_or_file_to_base64():
output_base64 = utils.encode_url_or_file_to_base64(
@ -120,3 +128,36 @@ def test_sleep_successful(mock_post):
def test_sleep_unsuccessful(mock_post):
with pytest.raises(utils.SpaceDuplicationError):
utils.set_space_timeout("gradio/calculator")
@pytest.mark.parametrize("schema", types)
def test_json_schema_to_python_type(schema):
if schema == "SimpleSerializable":
answer = "Any"
elif schema == "StringSerializable":
answer = "str"
elif schema == "ListStringSerializable":
answer = "List[str]"
elif schema == "BooleanSerializable":
answer = "bool"
elif schema == "NumberSerializable":
answer = "int | float"
elif schema == "ImgSerializable":
answer = "str"
elif schema == "FileSerializable":
answer = "str | Dict(name: str (name of file), data: str (base64 representation of file), size: int (size of image in bytes), is_file: bool (true if the file has been uploaded to the server), orig_name: str (original name of the file)) | List[str | Dict(name: str (name of file), data: str (base64 representation of file), size: int (size of image in bytes), is_file: bool (true if the file has been uploaded to the server), orig_name: str (original name of the file))]"
elif schema == "JSONSerializable":
answer = "Dict[Any, Any]"
elif schema == "GallerySerializable":
answer = "Tuple[Dict(name: str (name of file), data: str (base64 representation of file), size: int (size of image in bytes), is_file: bool (true if the file has been uploaded to the server), orig_name: str (original name of the file)), str | None]"
elif schema == "SingleFileSerializable":
answer = "str | Dict(name: str (name of file), data: str (base64 representation of file), size: int (size of image in bytes), is_file: bool (true if the file has been uploaded to the server), orig_name: str (original name of the file))"
elif schema == "MultipleFileSerializable":
answer = "List[str | Dict(name: str (name of file), data: str (base64 representation of file), size: int (size of image in bytes), is_file: bool (true if the file has been uploaded to the server), orig_name: str (original name of the file))]"
elif schema == "SingleFile":
answer = "str"
elif schema == "MultipleFile":
answer = "List[str]"
else:
raise ValueError(f"This test has not been modified to check {schema}")
assert utils.json_schema_to_python_type(types[schema]) == answer

View File

@ -20,6 +20,7 @@ from anyio import CapacityLimiter
from gradio_client import serializing
from gradio_client import utils as client_utils
from gradio_client.documentation import document, set_documentation_group
from packaging import version
from typing_extensions import Literal
from gradio import (
@ -468,6 +469,9 @@ def get_api_info(config: dict, serialize: bool = True):
"""
api_info = {"named_endpoints": {}, "unnamed_endpoints": {}}
mode = config.get("mode", None)
after_new_format = version.parse(config.get("version", "2.0")) > version.Version(
"3.28.3"
)
for d, dependency in enumerate(config["dependencies"]):
dependency_info = {"parameters": [], "returns": []}
@ -494,29 +498,36 @@ def get_api_info(config: dict, serialize: bool = True):
# The config has the most specific API info (taking into account the parameters
# of the component), so we use that if it exists. Otherwise, we fallback to the
# Serializer's API info.
if component.get("api_info"):
if serialize:
info = component["api_info"]["serialized_input"]
example = component["example_inputs"]["serialized"]
else:
info = component["api_info"]["raw_input"]
example = component["example_inputs"]["raw"]
serializer = serializing.COMPONENT_MAPPING[type]()
if component.get("api_info") and after_new_format:
info = component["api_info"]
example = component["example_inputs"]["serialized"]
else:
serializer = serializing.COMPONENT_MAPPING[type]()
assert isinstance(serializer, serializing.Serializable)
if serialize:
info = serializer.api_info()["serialized_input"]
example = serializer.example_inputs()["serialized"]
else:
info = serializer.api_info()["raw_input"]
example = serializer.example_inputs()["raw"]
info = serializer.api_info()
example = serializer.example_inputs()["raw"]
python_info = info["info"]
if serialize and info["serialized_info"]:
python_info = serializer.serialized_info()
if (
isinstance(serializer, serializing.FileSerializable)
and component["props"].get("file_count", "single") != "single"
):
python_info = serializer._multiple_file_serialized_info()
python_type = client_utils.json_schema_to_python_type(python_info)
serializer_name = serializing.COMPONENT_MAPPING[type].__name__
dependency_info["parameters"].append(
{
"label": label,
"type_python": info[0],
"type_description": info[1],
"type": info["info"],
"python_type": {
"type": python_type,
"description": python_info.get("description", ""),
},
"component": type.capitalize(),
"example_input": example,
"serializer": serializer_name,
}
)
@ -540,16 +551,27 @@ def get_api_info(config: dict, serialize: bool = True):
label = component["props"].get("label", f"value_{o}")
serializer = serializing.COMPONENT_MAPPING[type]()
assert isinstance(serializer, serializing.Serializable)
if serialize:
info = serializer.api_info()["serialized_output"]
else:
info = serializer.api_info()["raw_output"]
info = serializer.api_info()
python_info = info["info"]
if serialize and info["serialized_info"]:
python_info = serializer.serialized_info()
if (
isinstance(serializer, serializing.FileSerializable)
and component["props"].get("file_count", "single") != "single"
):
python_info = serializer._multiple_file_serialized_info()
python_type = client_utils.json_schema_to_python_type(python_info)
serializer_name = serializing.COMPONENT_MAPPING[type].__name__
dependency_info["returns"].append(
{
"label": label,
"type_python": info[0],
"type_description": info[1],
"type": info["info"],
"python_type": {
"type": python_type,
"description": python_info.get("description", ""),
},
"component": type.capitalize(),
"serializer": serializer_name,
}
)

View File

@ -845,13 +845,13 @@ class Slider(
NeighborInterpretable.__init__(self)
self.cleared_value = self.value
def api_info(self) -> dict[str, tuple[str, str]]:
description = f"numeric value between {self.minimum} and {self.maximum}"
def api_info(self) -> dict[str, dict | bool]:
return {
"raw_input": ("int | float", description),
"raw_output": ("int | float", description),
"serialized_input": ("int | float", description),
"serialized_output": ("int | float", description),
"info": {
"type": "number",
"description": f"numeric value between {self.minimum} and {self.maximum}",
},
"serialized_info": False,
}
def example_inputs(self) -> dict[str, Any]:
@ -1483,19 +1483,16 @@ class Dropdown(
self.cleared_value = self.value or ([] if multiselect else "")
def api_info(self) -> dict[str, tuple[str, str]]:
def api_info(self) -> dict[str, dict | bool]:
if self.multiselect:
type = "List[str]"
description = f"List of options from: {self.choices}"
type = {
"type": "array",
"items": {"type": "string"},
"description": f"List of options from: {self.choices}",
}
else:
type = "str"
description = f"Option from: {self.choices}"
return {
"raw_input": (type, description),
"raw_output": (type, description),
"serialized_input": (type, description),
"serialized_output": (type, description),
}
type = {"type": "string", "description": f"Option from: {self.choices}"}
return {"info": type, "serialized_info": False}
def example_inputs(self) -> dict[str, Any]:
if self.multiselect:
@ -2788,6 +2785,18 @@ class File(
else:
return Path(input_data).name
def api_info(self) -> dict[str, dict | bool]:
if self.file_count == "single":
return self._single_file_api_info()
else:
return self._multiple_file_api_info()
def serialized_info(self):
if self.file_count == "single":
return self._single_file_serialized_info()
else:
return self._multiple_file_serialized_info()
@document("style")
class Dataframe(Changeable, Selectable, IOComponent, JSONSerializable):

View File

@ -13,13 +13,8 @@ XRAY_CONFIG = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"api_info": {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
},
"serializer": "StringSerializable",
"api_info": {"info": {"type": "string"}, "serialized_info": False},
"example_inputs": {"raw": "Howdy!", "serialized": "Howdy!"},
},
{
@ -34,12 +29,10 @@ XRAY_CONFIG = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"serializer": "ListStringSerializable",
"api_info": {
"raw_input": ["List[str]", "list of string values"],
"raw_output": ["List[str]", "list of string values"],
"serialized_input": ["List[str]", "list of string values"],
"serialized_output": ["List[str]", "list of string values"],
"info": {"type": "array", "items": {"type": "string"}},
"serialized_info": False,
},
"example_inputs": {"raw": "Covid", "serialized": "Covid"},
},
@ -76,10 +69,11 @@ XRAY_CONFIG = {
},
"serializer": "ImgSerializable",
"api_info": {
"raw_input": ["str", "base64 representation of image"],
"raw_output": ["str", "base64 representation of image"],
"serialized_input": ["str", "filepath or URL to image"],
"serialized_output": ["str", "filepath or URL to image"],
"info": {
"type": "string",
"description": "base64 representation of an image",
},
"serialized_info": True,
},
"example_inputs": {
"raw": "",
@ -92,13 +86,8 @@ XRAY_CONFIG = {
"props": {"show_label": True, "name": "json", "visible": True, "style": {}},
"serializer": "JSONSerializable",
"api_info": {
"raw_input": [
"str | Dict | List",
"JSON-serializable object or a string",
],
"raw_output": ["Dict | List", "dictionary- or list-like object"],
"serialized_input": ["str", "filepath to JSON file"],
"serialized_output": ["str", "filepath to JSON file"],
"info": {"type": {}, "description": "any valid json"},
"serialized_info": True,
},
"example_inputs": {"raw": {"a": 1, "b": 2}, "serialized": None},
},
@ -113,13 +102,8 @@ XRAY_CONFIG = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"api_info": {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
},
"serializer": "StringSerializable",
"api_info": {"info": {"type": "string"}, "serialized_info": False},
"example_inputs": {"raw": "Howdy!", "serialized": "Howdy!"},
},
{
@ -154,10 +138,11 @@ XRAY_CONFIG = {
},
"serializer": "ImgSerializable",
"api_info": {
"raw_input": ["str", "base64 representation of image"],
"raw_output": ["str", "base64 representation of image"],
"serialized_input": ["str", "filepath or URL to image"],
"serialized_output": ["str", "filepath or URL to image"],
"info": {
"type": "string",
"description": "base64 representation of an image",
},
"serialized_info": True,
},
"example_inputs": {
"raw": "",
@ -170,13 +155,8 @@ XRAY_CONFIG = {
"props": {"show_label": True, "name": "json", "visible": True, "style": {}},
"serializer": "JSONSerializable",
"api_info": {
"raw_input": [
"str | Dict | List",
"JSON-serializable object or a string",
],
"raw_output": ["Dict | List", "dictionary- or list-like object"],
"serialized_input": ["str", "filepath to JSON file"],
"serialized_output": ["str", "filepath to JSON file"],
"info": {"type": {}, "description": "any valid json"},
"serialized_info": True,
},
"example_inputs": {"raw": {"a": 1, "b": 2}, "serialized": None},
},
@ -191,13 +171,8 @@ XRAY_CONFIG = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"api_info": {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
},
"serializer": "StringSerializable",
"api_info": {"info": {"type": "string"}, "serialized_info": False},
"example_inputs": {"raw": "Howdy!", "serialized": "Howdy!"},
},
{
@ -213,13 +188,8 @@ XRAY_CONFIG = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"api_info": {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
},
"serializer": "StringSerializable",
"api_info": {"info": {"type": "string"}, "serialized_info": False},
"example_inputs": {"raw": "Howdy!", "serialized": "Howdy!"},
},
{
@ -353,13 +323,8 @@ XRAY_CONFIG_DIFF_IDS = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"api_info": {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
},
"serializer": "StringSerializable",
"api_info": {"info": {"type": "string"}, "serialized_info": False},
"example_inputs": {"raw": "Howdy!", "serialized": "Howdy!"},
},
{
@ -374,12 +339,10 @@ XRAY_CONFIG_DIFF_IDS = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"serializer": "ListStringSerializable",
"api_info": {
"raw_input": ["List[str]", "list of string values"],
"raw_output": ["List[str]", "list of string values"],
"serialized_input": ["List[str]", "list of string values"],
"serialized_output": ["List[str]", "list of string values"],
"info": {"type": "array", "items": {"type": "string"}},
"serialized_info": False,
},
"example_inputs": {"raw": "Covid", "serialized": "Covid"},
},
@ -416,10 +379,11 @@ XRAY_CONFIG_DIFF_IDS = {
},
"serializer": "ImgSerializable",
"api_info": {
"raw_input": ["str", "base64 representation of image"],
"raw_output": ["str", "base64 representation of image"],
"serialized_input": ["str", "filepath or URL to image"],
"serialized_output": ["str", "filepath or URL to image"],
"info": {
"type": "string",
"description": "base64 representation of an image",
},
"serialized_info": True,
},
"example_inputs": {
"raw": "",
@ -432,13 +396,8 @@ XRAY_CONFIG_DIFF_IDS = {
"props": {"show_label": True, "name": "json", "visible": True, "style": {}},
"serializer": "JSONSerializable",
"api_info": {
"raw_input": [
"str | Dict | List",
"JSON-serializable object or a string",
],
"raw_output": ["Dict | List", "dictionary- or list-like object"],
"serialized_input": ["str", "filepath to JSON file"],
"serialized_output": ["str", "filepath to JSON file"],
"info": {"type": {}, "description": "any valid json"},
"serialized_info": True,
},
"example_inputs": {"raw": {"a": 1, "b": 2}, "serialized": None},
},
@ -453,13 +412,8 @@ XRAY_CONFIG_DIFF_IDS = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"api_info": {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
},
"serializer": "StringSerializable",
"api_info": {"info": {"type": "string"}, "serialized_info": False},
"example_inputs": {"raw": "Howdy!", "serialized": "Howdy!"},
},
{
@ -494,10 +448,11 @@ XRAY_CONFIG_DIFF_IDS = {
},
"serializer": "ImgSerializable",
"api_info": {
"raw_input": ["str", "base64 representation of image"],
"raw_output": ["str", "base64 representation of image"],
"serialized_input": ["str", "filepath or URL to image"],
"serialized_output": ["str", "filepath or URL to image"],
"info": {
"type": "string",
"description": "base64 representation of an image",
},
"serialized_info": True,
},
"example_inputs": {
"raw": "",
@ -510,13 +465,8 @@ XRAY_CONFIG_DIFF_IDS = {
"props": {"show_label": True, "name": "json", "visible": True, "style": {}},
"serializer": "JSONSerializable",
"api_info": {
"raw_input": [
"str | Dict | List",
"JSON-serializable object or a string",
],
"raw_output": ["Dict | List", "dictionary- or list-like object"],
"serialized_input": ["str", "filepath to JSON file"],
"serialized_output": ["str", "filepath to JSON file"],
"info": {"type": {}, "description": "any valid json"},
"serialized_info": True,
},
"example_inputs": {"raw": {"a": 1, "b": 2}, "serialized": None},
},
@ -531,13 +481,8 @@ XRAY_CONFIG_DIFF_IDS = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"api_info": {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
},
"serializer": "StringSerializable",
"api_info": {"info": {"type": "string"}, "serialized_info": False},
"example_inputs": {"raw": "Howdy!", "serialized": "Howdy!"},
},
{
@ -553,13 +498,8 @@ XRAY_CONFIG_DIFF_IDS = {
"visible": True,
"style": {},
},
"serializer": "Serializable",
"api_info": {
"raw_input": ["str", "string value"],
"raw_output": ["str", "string value"],
"serialized_input": ["str", "string value"],
"serialized_output": ["str", "string value"],
},
"serializer": "StringSerializable",
"api_info": {"info": {"type": "string"}, "serialized_info": False},
"example_inputs": {"raw": "Howdy!", "serialized": "Howdy!"},
},
{

View File

@ -36,6 +36,7 @@ import aiohttp
import httpx
import matplotlib
import requests
from gradio_client.serializing import Serializable
from markdown_it import MarkdownIt
from mdit_py_plugins.dollarmath.index import dollarmath_plugin
from mdit_py_plugins.footnote.index import footnote_plugin
@ -990,6 +991,9 @@ def get_serializer_name(block: Block) -> str | None:
and getattr(meth.__self__, "__class__", None)
):
for cls in inspect.getmro(meth.__self__.__class__):
# Find the first serializer defined in gradio_client that
if issubclass(cls, Serializable) and "gradio_client" in cls.__module__:
return cls
if meth.__name__ in cls.__dict__:
return cls
meth = getattr(meth, "__func__", meth) # fallback to __qualname__ parsing

View File

@ -3,7 +3,7 @@ aiohttp
altair>=4.2.0
fastapi
ffmpy
gradio_client>=0.1.3
gradio_client>=0.2.0
httpx
huggingface_hub>=0.13.0
Jinja2

View File

@ -955,7 +955,7 @@ class TestFile:
x_file["is_example"] = True
assert file_input.preprocess(x_file) is not None
zero_size_file = {"name": "document.txt", "size": 0, "data": "data:"}
zero_size_file = {"name": "document.txt", "size": 0, "data": ""}
temp_file = file_input.preprocess(zero_size_file)
assert os.stat(temp_file.name).st_size == 0