mirror of
https://github.com/gradio-app/gradio.git
synced 2025-03-31 12:20:26 +08:00
[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:
parent
d1853625fd
commit
b66ecff671
@ -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)
|
||||
|
@ -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"
|
||||
|
||||
|
@ -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]:
|
||||
|
199
client/python/gradio_client/types.json
Normal file
199
client/python/gradio_client/types.json
Normal 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" }
|
||||
]
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
}
|
@ -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}")
|
||||
|
@ -1 +1 @@
|
||||
0.1.4
|
||||
0.2.0
|
||||
|
@ -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
|
||||
|
@ -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):
|
||||
|
@ -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
|
||||
|
@ -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
|
||||
|
@ -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,
|
||||
}
|
||||
)
|
||||
|
||||
|
@ -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):
|
||||
|
@ -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!"},
|
||||
},
|
||||
{
|
||||
|
@ -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
|
||||
|
@ -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
|
||||
|
@ -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
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user