-
Notifications
You must be signed in to change notification settings - Fork 901
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Lina Tang
committed
Oct 23, 2023
1 parent
42be774
commit a399589
Showing
3 changed files
with
132 additions
and
7 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
127 changes: 127 additions & 0 deletions
127
src/promptflow/tests/executor/unittests/_utils/test_multimedia_utils.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,127 @@ | ||
import pytest | ||
import re | ||
from pathlib import Path | ||
from unittest.mock import mock_open | ||
|
||
from promptflow._utils.multimedia_utils import ( | ||
_create_image_from_file, | ||
convert_multimedia_data_to_base64, | ||
create_image, | ||
load_multimedia_data, | ||
persist_multimedia_data, | ||
) | ||
from promptflow.contracts._errors import InvalidImageInput | ||
from promptflow.contracts.flow import FlowInputDefinition | ||
from promptflow.contracts.tool import ValueType | ||
|
||
from ...utils import DATA_ROOT | ||
|
||
TEST_IMAGE_PATH = DATA_ROOT / "test_image.jpg" | ||
|
||
|
||
@pytest.mark.unittest | ||
class TestMultimediaUtils: | ||
def test_create_image(self, mocker): | ||
# Test create image from dict | ||
## From path | ||
image_dict = {"data:image/jpg;path": TEST_IMAGE_PATH} | ||
image_from_path = create_image(image_dict) | ||
assert image_from_path._mime_type == "image/jpg" | ||
|
||
## From base64 | ||
image_dict = {"data:image/jpg;base64": image_from_path.to_base64()} | ||
image_from_base64 = create_image(image_dict) | ||
assert str(image_from_path) == str(image_from_base64) | ||
assert image_from_base64._mime_type == "image/jpg" | ||
|
||
## From url | ||
mocker.patch("requests.get", return_value=mocker.Mock(content=image_from_path, status_code=200)) | ||
image_dict = {"data:image/jpg;url": ""} | ||
image_from_url = create_image(image_dict) | ||
assert str(image_from_path) == str(image_from_url) | ||
assert image_from_url._mime_type == "image/jpg" | ||
|
||
mocker.patch("requests.get", return_value=mocker.Mock(content=None, status_code=404)) | ||
with pytest.raises(InvalidImageInput) as ex: | ||
create_image(image_dict) | ||
assert "Error while fetching image from URL" in ex.value.message_format | ||
|
||
# Test create image from string | ||
## From path | ||
image_from_path = create_image(str(TEST_IMAGE_PATH)) | ||
assert image_from_path._mime_type == "image/jpg" | ||
|
||
# From base64 | ||
image_from_base64 = create_image(image_from_path.to_base64()) | ||
assert str(image_from_path) == str(image_from_base64) | ||
assert image_from_base64._mime_type in ["image/jpg", "image/jpeg"] | ||
|
||
## From url | ||
mocker.patch("promptflow._utils.multimedia_utils._is_url", return_value=True) | ||
mocker.patch("promptflow._utils.multimedia_utils._is_base64", return_value=False) | ||
mocker.patch("requests.get", return_value=mocker.Mock(content=image_from_path, status_code=200)) | ||
image_from_url = create_image("") | ||
assert str(image_from_path) == str(image_from_url) | ||
assert image_from_url._mime_type in ["image/jpg", "image/jpeg"] | ||
|
||
## From image | ||
image_from_image = create_image(image_from_path) | ||
assert str(image_from_path) == str(image_from_image) | ||
|
||
# Test invalid input type | ||
with pytest.raises(InvalidImageInput) as ex: | ||
create_image(0) | ||
assert "Unsupported image input type" in ex.value.message_format | ||
|
||
# Test invalid image dict | ||
with pytest.raises(InvalidImageInput) as ex: | ||
invalid_image_dict = {"invalid_image": "invalid_image"} | ||
create_image(invalid_image_dict) | ||
assert "Invalid image input format" in ex.value.message_format | ||
|
||
def test_persist_multimedia_date(self, mocker): | ||
image = _create_image_from_file(TEST_IMAGE_PATH) | ||
mocker.patch('builtins.open', mock_open()) | ||
data = {"image": image, "images": [image, image, "other_data"], "other_data": "other_data"} | ||
persisted_data = persist_multimedia_data(data, base_dir=Path(__file__).parent) | ||
file_name = re.compile(r"^[0-9a-z]{8}-[0-9a-z]{4}-[0-9a-z]{4}-[0-9a-z]{4}-[0-9a-z]{12}.jpg$") | ||
assert re.match(file_name, persisted_data["image"]["data:image/jpg;path"]) | ||
assert re.match(file_name, persisted_data["images"][0]["data:image/jpg;path"]) | ||
assert re.match(file_name, persisted_data["images"][1]["data:image/jpg;path"]) | ||
|
||
def test_convert_multimedia_date_to_base64(self): | ||
image = _create_image_from_file(TEST_IMAGE_PATH) | ||
data = {"image": image, "images": [image, image, "other_data"], "other_data": "other_data"} | ||
base64_data = convert_multimedia_data_to_base64(data) | ||
assert base64_data == { | ||
"image": image.to_base64(), | ||
"images": [image.to_base64(), image.to_base64(), "other_data"], | ||
"other_data": "other_data", | ||
} | ||
|
||
base64_data = convert_multimedia_data_to_base64(data, with_type=True) | ||
prefix = f"data:{image._mime_type};base64," | ||
assert base64_data == { | ||
"image": prefix + image.to_base64(), | ||
"images": [prefix + image.to_base64(), prefix + image.to_base64(), "other_data"], | ||
"other_data": "other_data", | ||
} | ||
|
||
def test_load_multimedia_data(self): | ||
inputs = { | ||
"image": FlowInputDefinition(type=ValueType.IMAGE), | ||
"images": FlowInputDefinition(type=ValueType.LIST), | ||
"object": FlowInputDefinition(type=ValueType.OBJECT), | ||
} | ||
line_inputs = { | ||
"image": {"data:image/jpg;path": str(TEST_IMAGE_PATH)}, | ||
"images": [{"data:image/jpg;path": str(TEST_IMAGE_PATH)}, {"data:image/jpg;path": str(TEST_IMAGE_PATH)}], | ||
"object": {"image": {"data:image/jpg;path": str(TEST_IMAGE_PATH)}, "other_data": "other_data"} | ||
} | ||
updated_inputs = load_multimedia_data(inputs, line_inputs) | ||
image = _create_image_from_file(TEST_IMAGE_PATH) | ||
assert updated_inputs == { | ||
"image": image, | ||
"images": [image, image], | ||
"object": {"image": image, "other_data": "other_data"} | ||
} |