diff --git a/demo/uploader.py b/demo/uploader.py index a3257b5eb..5a3492dbd 100644 --- a/demo/uploader.py +++ b/demo/uploader.py @@ -5,10 +5,7 @@ @me.stateclass class State: - name: str - size: int - mime_type: str - contents: str + file: me.UploadedFile @me.page( @@ -29,19 +26,22 @@ def app(): style=me.Style(font_weight="bold"), ) - if state.contents: + if state.file.size: with me.box(style=me.Style(margin=me.Margin.all(10))): - me.text(f"File name: {state.name}") - me.text(f"File size: {state.size}") - me.text(f"File type: {state.mime_type}") + me.text(f"File name: {state.file.name}") + me.text(f"File size: {state.file.size}") + me.text(f"File type: {state.file.mime_type}") with me.box(style=me.Style(margin=me.Margin.all(10))): - me.image(src=state.contents) + me.image(src=_convert_contents_data_url(state.file)) def handle_upload(event: me.UploadEvent): state = me.state(State) - state.name = event.file.name - state.size = event.file.size - state.mime_type = event.file.mime_type - state.contents = f"data:{event.file.mime_type};base64,{base64.b64encode(event.file.getvalue()).decode()}" + state.file = event.file + + +def _convert_contents_data_url(file: me.UploadedFile) -> str: + return ( + f"data:{file.mime_type};base64,{base64.b64encode(file.getvalue()).decode()}" + ) diff --git a/mesop/components/uploader/BUILD b/mesop/components/uploader/BUILD index da63c5888..dbccee4c8 100644 --- a/mesop/components/uploader/BUILD +++ b/mesop/components/uploader/BUILD @@ -8,6 +8,7 @@ package( mesop_component( name = "uploader", assets = [":uploader.css"], + py_deps = [":uploaded_file"], ) sass_binary( @@ -15,3 +16,8 @@ sass_binary( src = "uploader.scss", sourcemap = False, ) + +py_library( + name = "uploaded_file", + srcs = ["uploaded_file.py"], +) diff --git a/mesop/components/uploader/e2e/uploader_app.py b/mesop/components/uploader/e2e/uploader_app.py index 4591bacac..b83abb623 100644 --- a/mesop/components/uploader/e2e/uploader_app.py +++ b/mesop/components/uploader/e2e/uploader_app.py @@ -5,34 +5,38 @@ @me.stateclass class State: - name: str - size: int - mime_type: str - contents: str + file: me.UploadedFile @me.page(path="/components/uploader/e2e/uploader_app") def app(): state = me.state(State) - me.uploader( - label="Upload Image", - accepted_file_types=["image/jpeg", "image/png"], - on_upload=handle_upload, - ) - - if state.contents: - with me.box(style=me.Style(margin=me.Margin.all(10))): - me.text(f"File name: {state.name}") - me.text(f"File size: {state.size}") - me.text(f"File type: {state.mime_type}") - - with me.box(style=me.Style(margin=me.Margin.all(10))): - me.image(src=state.contents) + with me.box(style=me.Style(padding=me.Padding.all(15))): + me.uploader( + label="Upload Image", + accepted_file_types=["image/jpeg", "image/png"], + on_upload=handle_upload, + type="flat", + color="primary", + style=me.Style(font_weight="bold"), + ) + + if state.file.size: + with me.box(style=me.Style(margin=me.Margin.all(10))): + me.text(f"File name: {state.file.name}") + me.text(f"File size: {state.file.size}") + me.text(f"File type: {state.file.mime_type}") + + with me.box(style=me.Style(margin=me.Margin.all(10))): + me.image(src=_convert_contents_data_url(state.file)) def handle_upload(event: me.UploadEvent): state = me.state(State) - state.name = event.file.name - state.size = event.file.size - state.mime_type = event.file.mime_type - state.contents = f"data:{event.file.mime_type};base64,{base64.b64encode(event.file.getvalue()).decode()}" + state.file = event.file + + +def _convert_contents_data_url(file: me.UploadedFile) -> str: + return ( + f"data:{file.mime_type};base64,{base64.b64encode(file.getvalue()).decode()}" + ) diff --git a/mesop/components/uploader/e2e/uploader_test.ts b/mesop/components/uploader/e2e/uploader_test.ts index 0a898bbbf..263e01f4f 100644 --- a/mesop/components/uploader/e2e/uploader_test.ts +++ b/mesop/components/uploader/e2e/uploader_test.ts @@ -13,6 +13,8 @@ test('test upload file', async ({page}) => { await expect(page.getByText('File size: 30793')).toHaveCount(1); await expect(page.getByText('File type: image/jpeg')).toHaveCount(1); await expect( - page.locator(`//img[contains(@src, "data:image/jpeg;base64")]`), + page.locator( + `//img[@src=""]`, + ), ).toHaveCount(1); }); diff --git a/mesop/components/uploader/uploaded_file.py b/mesop/components/uploader/uploaded_file.py new file mode 100644 index 000000000..371c4a989 --- /dev/null +++ b/mesop/components/uploader/uploaded_file.py @@ -0,0 +1,42 @@ +import io + + +# Store this class in a separate file so we can more easily reference +# in dataclass utils. +class UploadedFile(io.BytesIO): + """Uploaded file contents and metadata.""" + + def __init__( + self, + contents: bytes = b"", + *, + name: str = "", + size: int = 0, + mime_type: str = "", + ): + super().__init__(contents) + self._name = name + self._size = size + self._mime_type = mime_type + + @property + def name(self): + return self._name + + @property + def size(self): + return self._size + + @property + def mime_type(self): + return self._mime_type + + def __eq__(self, other): + if isinstance(other, UploadedFile): + return (self.getvalue(), self.name, self.size, self.mime_type) == ( + other.getvalue(), + other.name, + other.size, + other.mime_type, + ) + return False diff --git a/mesop/components/uploader/uploader.py b/mesop/components/uploader/uploader.py index 9b32ad696..bcc67e239 100644 --- a/mesop/components/uploader/uploader.py +++ b/mesop/components/uploader/uploader.py @@ -1,4 +1,3 @@ -import io from dataclasses import dataclass from typing import Any, Callable, Literal, Sequence @@ -10,32 +9,11 @@ register_event_mapper, register_native_component, ) +from mesop.components.uploader.uploaded_file import UploadedFile from mesop.events import MesopEvent from mesop.exceptions import MesopDeveloperException -class UploadedFile(io.BytesIO): - """Uploaded file contents and metadata.""" - - def __init__(self, contents: bytes, *, name: str, size: int, mime_type: str): - super().__init__(contents) - self._name = name - self._size = size - self._mime_type = mime_type - - @property - def name(self): - return self._name - - @property - def size(self): - return self._size - - @property - def mime_type(self): - return self._mime_type - - @dataclass(kw_only=True) class UploadEvent(MesopEvent): """Event for file uploads. diff --git a/mesop/dataclass_utils/BUILD b/mesop/dataclass_utils/BUILD index f6909e01d..f86151989 100644 --- a/mesop/dataclass_utils/BUILD +++ b/mesop/dataclass_utils/BUILD @@ -10,17 +10,26 @@ py_library( ["*.py"], exclude = ["*_test.py"], ), - deps = ["//mesop/exceptions"] + THIRD_PARTY_PY_DEEPDIFF, + deps = [ + "//mesop/components/uploader:uploaded_file", + "//mesop/exceptions", + ] + THIRD_PARTY_PY_DEEPDIFF, ) py_test( name = "dataclass_utils_test", srcs = ["dataclass_utils_test.py"], - deps = [":dataclass_utils"] + THIRD_PARTY_PY_PYTEST + THIRD_PARTY_PY_PANDAS, + deps = [ + ":dataclass_utils", + "//mesop/components/uploader:uploaded_file", + ] + THIRD_PARTY_PY_PYTEST + THIRD_PARTY_PY_PANDAS, ) py_test( name = "diff_state_test", srcs = ["diff_state_test.py"], - deps = [":dataclass_utils"] + THIRD_PARTY_PY_PYTEST + THIRD_PARTY_PY_PANDAS, + deps = [ + ":dataclass_utils", + "//mesop/components/uploader:uploaded_file", + ] + THIRD_PARTY_PY_PYTEST + THIRD_PARTY_PY_PANDAS, ) diff --git a/mesop/dataclass_utils/dataclass_utils.py b/mesop/dataclass_utils/dataclass_utils.py index 2fc142df1..acac68e10 100644 --- a/mesop/dataclass_utils/dataclass_utils.py +++ b/mesop/dataclass_utils/dataclass_utils.py @@ -10,12 +10,15 @@ from deepdiff.operator import BaseOperator from deepdiff.path import parse_path +from mesop.components.uploader.uploaded_file import UploadedFile from mesop.exceptions import MesopDeveloperException, MesopException _PANDAS_OBJECT_KEY = "__pandas.DataFrame__" _DATETIME_OBJECT_KEY = "__datetime.datetime__" _BYTES_OBJECT_KEY = "__python.bytes__" +_UPLOADED_FILE_OBJECT_KEY = "__mesop.UploadedFile__" _DIFF_ACTION_DATA_FRAME_CHANGED = "data_frame_changed" +_DIFF_ACTION_UPLOADED_FILE_CHANGED = "mesop_uploaded_file_changed" C = TypeVar("C") @@ -174,8 +177,19 @@ def default(self, obj): except ImportError: pass + if isinstance(obj, UploadedFile): + return { + _UPLOADED_FILE_OBJECT_KEY: { + "contents": base64.b64encode(obj.getvalue()).decode("utf-8"), + "name": obj.name, + "size": obj.size, + "mime_type": obj.mime_type, + } + } + if isinstance(obj, datetime): return {_DATETIME_OBJECT_KEY: obj.isoformat()} + if isinstance(obj, bytes): return {_BYTES_OBJECT_KEY: base64.b64encode(obj).decode("utf-8")} @@ -210,6 +224,14 @@ def decode_mesop_json_state_hook(dct): if _BYTES_OBJECT_KEY in dct: return base64.b64decode(dct[_BYTES_OBJECT_KEY]) + if _UPLOADED_FILE_OBJECT_KEY in dct: + return UploadedFile( + base64.b64decode(dct[_UPLOADED_FILE_OBJECT_KEY]["contents"]), + name=dct[_UPLOADED_FILE_OBJECT_KEY]["name"], + size=dct[_UPLOADED_FILE_OBJECT_KEY]["size"], + mime_type=dct[_UPLOADED_FILE_OBJECT_KEY]["mime_type"], + ) + return dct @@ -241,6 +263,29 @@ def give_up_diffing(self, level, diff_instance) -> bool: return True +class UploadedFileOperator(BaseOperator): + """Custom operator to detect changes in UploadedFile class. + + DeepDiff does not diff the UploadedFile class correctly, so we will just use a normal + equality check, rather than diffing further into the io.BytesIO parent class. + + This class could probably be made more generic to handle other classes where we want + to diff using equality checks. + """ + + def match(self, level) -> bool: + return isinstance(level.t1, UploadedFile) and isinstance( + level.t2, UploadedFile + ) + + def give_up_diffing(self, level, diff_instance) -> bool: + if level.t1 != level.t2: + diff_instance.custom_report_result( + _DIFF_ACTION_UPLOADED_FILE_CHANGED, level, {"value": level.t2} + ) + return True + + def diff_state(state1: Any, state2: Any) -> str: """ Diffs two state objects and returns the difference using DeepDiff's Delta format as a @@ -255,11 +300,13 @@ def diff_state(state1: Any, state2: Any) -> str: raise MesopException("Tried to diff state which was not a dataclass") custom_actions = [] - + custom_operators = [UploadedFileOperator()] # Only use the `DataFrameOperator` if pandas exists. if _has_pandas: differences = DeepDiff( - state1, state2, custom_operators=[DataFrameOperator()] + state1, + state2, + custom_operators=[*custom_operators, DataFrameOperator()], ) # Manually format dataframe diffs to flat dict format. @@ -273,7 +320,18 @@ def diff_state(state1: Any, state2: Any) -> str: for path, diff in differences[_DIFF_ACTION_DATA_FRAME_CHANGED].items() ] else: - differences = DeepDiff(state1, state2) + differences = DeepDiff(state1, state2, custom_operators=custom_operators) + + # Manually format UploadedFile diffs to flat dict format. + if _DIFF_ACTION_UPLOADED_FILE_CHANGED in differences: + custom_actions = [ + { + "path": parse_path(path), + "action": _DIFF_ACTION_UPLOADED_FILE_CHANGED, + **diff, + } + for path, diff in differences[_DIFF_ACTION_UPLOADED_FILE_CHANGED].items() + ] return json.dumps( Delta(differences, always_include_values=True).to_flat_dicts() diff --git a/mesop/dataclass_utils/dataclass_utils_test.py b/mesop/dataclass_utils/dataclass_utils_test.py index 3b48cbd48..7ebf5257a 100644 --- a/mesop/dataclass_utils/dataclass_utils_test.py +++ b/mesop/dataclass_utils/dataclass_utils_test.py @@ -6,6 +6,7 @@ import pytest import mesop.protos.ui_pb2 as pb +from mesop.components.uploader.uploaded_file import UploadedFile from mesop.dataclass_utils.dataclass_utils import ( dataclass_with_defaults, has_parent, @@ -43,6 +44,11 @@ class WithBytes: data: bytes = b"" +@dataclass +class WithUploadedFile: + data: UploadedFile = field(default_factory=UploadedFile) + + JSON_STR = """{"b": {"c": {"val": ""}}, "list_b": [ {"c": {"val": "1"}}, @@ -159,6 +165,21 @@ def test_serialize_pandas_dataframe(): ) +def test_serialize_uploaded_file(): + serialized_dataclass = serialize_dataclass( + WithUploadedFile( + data=UploadedFile( + b"data", name="file.png", size=10, mime_type="image/png" + ) + ) + ) + + assert ( + serialized_dataclass + == '{"data": {"__mesop.UploadedFile__": {"contents": "ZGF0YQ==", "name": "file.png", "size": 10, "mime_type": "image/png"}}}' + ) + + @pytest.mark.parametrize( "input_bytes, expected_json", [ @@ -238,6 +259,18 @@ def test_update_dataclass_with_pandas_dataframe(): ) +def test_update_dataclass_with_uploaded_file(): + uploaded_file = UploadedFile( + b"data", name="file.png", size=10, mime_type="image/png" + ) + serialized_dataclass = serialize_dataclass( + WithUploadedFile(data=uploaded_file) + ) + uploaded_file_state = WithUploadedFile() + update_dataclass_from_json(uploaded_file_state, serialized_dataclass) + assert uploaded_file_state.data == uploaded_file + + def test_update_dataclass_with_bytes(): bytes_data = b"hello world" serialized_dataclass = serialize_dataclass(WithBytes(data=bytes_data)) diff --git a/mesop/dataclass_utils/diff_state_test.py b/mesop/dataclass_utils/diff_state_test.py index 39e918cc8..fd284e5f8 100644 --- a/mesop/dataclass_utils/diff_state_test.py +++ b/mesop/dataclass_utils/diff_state_test.py @@ -6,6 +6,7 @@ import pandas as pd import pytest +from mesop.components.uploader.uploaded_file import UploadedFile from mesop.dataclass_utils.dataclass_utils import diff_state from mesop.exceptions import MesopException @@ -395,6 +396,47 @@ class C: ] +def test_diff_uploaded_file(): + @dataclass + class C: + data: UploadedFile + + s1 = C(data=UploadedFile()) + s2 = C( + data=UploadedFile(b"data", name="file.png", size=10, mime_type="image/png") + ) + + assert json.loads(diff_state(s1, s2)) == [ + { + "path": ["data"], + "action": "mesop_uploaded_file_changed", + "value": { + "__mesop.UploadedFile__": { + "contents": "ZGF0YQ==", + "name": "file.png", + "size": 10, + "mime_type": "image/png", + }, + }, + } + ] + + +def test_diff_uploaded_file_same_no_diff(): + @dataclass + class C: + data: UploadedFile + + s1 = C( + data=UploadedFile(b"data", name="file.png", size=10, mime_type="image/png") + ) + s2 = C( + data=UploadedFile(b"data", name="file.png", size=10, mime_type="image/png") + ) + + assert json.loads(diff_state(s1, s2)) == [] + + # The diff will pass, but the Mesop JSON serializer currently fails on sets in state class. # See https://github.com/google/mesop/issues/387 def test_diff_set(): diff --git a/mesop/web/src/utils/diff.ts b/mesop/web/src/utils/diff.ts index 33f366a5f..bb763261b 100644 --- a/mesop/web/src/utils/diff.ts +++ b/mesop/web/src/utils/diff.ts @@ -78,6 +78,7 @@ export function applyComponentDiff(component: Component, diff: ComponentDiff) { const STATE_DIFF_VALUES_CHANGED = 'values_changed'; const STATE_DIFF_TYPE_CHANGES = 'type_changes'; const STATE_DIFF_DATA_FRAME_CHANGED = 'data_frame_changed'; +const STATE_DIFF_UPLOADED_FILE_CHANGED = 'mesop_uploaded_file_changed'; const STATE_DIFF_ITERABLE_ITEM_REMOVED = 'iterable_item_removed'; const STATE_DIFF_ITERABLE_ITEM_ADDED = 'iterable_item_added'; const STATE_DIFF_DICT_ITEM_REMOVED = 'dictionary_item_removed'; @@ -114,7 +115,8 @@ export function applyStateDiff(stateJson: string, diffJson: string): string { if ( row.action === STATE_DIFF_VALUES_CHANGED || row.action === STATE_DIFF_TYPE_CHANGES || - row.action === STATE_DIFF_DATA_FRAME_CHANGED + row.action === STATE_DIFF_DATA_FRAME_CHANGED || + row.action === STATE_DIFF_UPLOADED_FILE_CHANGED ) { updateValue(root, row.path, row.value); } else if (row.action === STATE_DIFF_DICT_ITEM_ADDED) { diff --git a/mesop/web/src/utils/diff_state_spec.ts b/mesop/web/src/utils/diff_state_spec.ts index ad960190d..52c0d8ced 100644 --- a/mesop/web/src/utils/diff_state_spec.ts +++ b/mesop/web/src/utils/diff_state_spec.ts @@ -373,4 +373,44 @@ describe('applyStateDiff functionality', () => { }), ); }); + + it('applies UploadedFile updates', () => { + const state1 = JSON.stringify({ + data: { + '__mesop.UploadedFile__': { + 'contents': '', + 'name': '', + 'size': 0, + 'mime_type': '', + }, + }, + }); + const diff = JSON.stringify([ + { + path: ['data'], + action: 'mesop_uploaded_file_changed', + value: { + '__mesop.UploadedFile__': { + 'contents': 'data', + 'name': 'file.png', + 'size': 10, + 'mime_type': 'image/png', + }, + }, + }, + ]); + + expect(applyStateDiff(state1, diff)).toBe( + JSON.stringify({ + data: { + '__mesop.UploadedFile__': { + 'contents': 'data', + 'name': 'file.png', + 'size': 10, + 'mime_type': 'image/png', + }, + }, + }), + ); + }); });