rmm commited on
Commit
2157fef
·
1 Parent(s): 0124054

test: now using fixture that loads real data in the file_upload

Browse files

- added simple tests for author_email
- added a test that validates the file_uploader process (for multi-file
handling), by getting real image data, extracting metadata and
presenting it visually (see `test_mockupload_list_realdata`)
- added some explanations to the tests

Files changed (1) hide show
  1. tests/test_demo_multifile_upload.py +121 -6
tests/test_demo_multifile_upload.py CHANGED
@@ -8,13 +8,21 @@ import pytest
8
  from unittest.mock import MagicMock, patch
9
  from streamlit.testing.v1 import AppTest
10
 
11
- # - tests for apptest/demo_multifile_upload
 
 
 
 
 
 
 
12
 
13
- # zero test: no inputs -> empty session state
14
- # (or maybe even non-existent session state; for file_uploader we are not allowed to initialise the keyed variable, st borks)
15
 
16
- # many test: list of 2 inputs -> session state with 2 files
 
 
17
 
 
18
 
19
 
20
  # for expectations
@@ -118,15 +126,122 @@ def mock_uploadedFile_List_ImageData(mock_uploadedFile):
118
  return create_list_of_mocks_realdata
119
 
120
 
121
-
 
 
122
  def test_no_input_no_interaction():
 
 
 
 
 
123
  at = AppTest.from_file("src/apptest/demo_multifile_upload.py").run()
124
-
125
  assert at.session_state.observations == {}
126
  assert at.session_state.input_author_email == spoof_metadata.get("author_email")
127
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
128
 
129
 
 
 
130
  @patch("streamlit.file_uploader")
131
  def test_mockupload_list(mock_file_uploader_rtn: MagicMock, mock_uploadedFile_List):
132
  # Create a list of 2 mock files
 
8
  from unittest.mock import MagicMock, patch
9
  from streamlit.testing.v1 import AppTest
10
 
11
+ # tests for apptest/demo_multifile_upload
12
+ # - the functionality in the test harness is a file_uploader that is configured
13
+ # for multi-file input; and uses a callback to buffer the files into session state.
14
+ # - the handling of individual files includes extracting metadata from the files
15
+ # - a text_area is created for each file, to display the metadata extracted;
16
+ # this deviates from the presentation in the real app, but the extracted info
17
+ # is the same (here we put it all in text which is far easier to validate using AppTest)
18
+ # - the demo also has the author email input
19
 
 
 
20
 
21
+ # zero test: no inputs -> empty session state
22
+ # (or maybe even non-existent session state; for file_uploader we are not
23
+ # allowed to initialise the keyed variable, st borks)
24
 
25
+ # many test: list of >=2 inputs -> session state with 2 files
26
 
27
 
28
  # for expectations
 
126
  return create_list_of_mocks_realdata
127
 
128
 
129
+ # simple tests on the author email input via AppTest
130
+ # - empty input should propagate to session state
131
+ # - invalid email should trigger an error
132
  def test_no_input_no_interaction():
133
+ with patch.dict(spoof_metadata, {"author_email": None}):
134
+ at = AppTest.from_file("src/apptest/demo_multifile_upload.py").run()
135
+ assert at.session_state.observations == {}
136
+ assert at.session_state.input_author_email == None
137
+
138
  at = AppTest.from_file("src/apptest/demo_multifile_upload.py").run()
 
139
  assert at.session_state.observations == {}
140
  assert at.session_state.input_author_email == spoof_metadata.get("author_email")
141
 
142
+ def test_bad_email():
143
+ with patch.dict(spoof_metadata, {"author_email": "notanemail"}):
144
+ at = AppTest.from_file("src/apptest/demo_multifile_upload.py").run()
145
+ assert at.session_state.input_author_email == "notanemail"
146
+ assert at.error[0].value == "Please enter a valid email address."
147
+
148
+
149
+ # test when we load real data files, with all properties as per real app
150
+ # - if files loaded correctly and metadata is extracted correctly, we should see the
151
+ # the data in both the session state and in the visual elements.
152
+ @patch("streamlit.file_uploader")
153
+ def test_mockupload_list_realdata(mock_file_rv: MagicMock, mock_uploadedFile_List_ImageData):
154
+ #def test_mockupload_list(mock_file_uploader_rtn: MagicMock, mock_uploadedFile_List):
155
+ num_files = 3
156
+ PRINT_PROPS = False
157
+ # Create a list of n mock files
158
+ mock_files = mock_uploadedFile_List_ImageData(num_files=num_files)
159
+
160
+ # Set the return value of the mocked file_uploader to the list of mock files
161
+ mock_file_rv.return_value = mock_files
162
+
163
+ # Run the Streamlit app
164
+ at = AppTest.from_file("src/apptest/demo_multifile_upload.py").run()
165
+
166
+ # put the mocked file_upload into session state, as if it were the result of a file upload, with the key 'file_uploader_data'
167
+ at.session_state["file_uploader_data"] = mock_files
168
+
169
+ #print(f"[I] session state: {at.session_state}")
170
+ #print(f"[I] uploaded files: {at.session_state.file_uploader_data}")
171
+
172
+ if PRINT_PROPS:
173
+ print(f"[I] uploaded files: ({len(at.session_state.file_uploader_data)}) {at.session_state.file_uploader_data}")
174
+ for _f in at.session_state.file_uploader_data:
175
+ #print(f"\t[I] props: {dir(_f)}")
176
+ print(f" [I] name: {_f.name}")
177
+ print(f"\t[I] size: {_f.size}")
178
+ print(f"\t[I] type: {_f.type}")
179
+ # lets make an image from the data
180
+ im = Image.open(_f)
181
+
182
+ # lets see what metadata we can get to.
183
+ dt = get_image_datetime(_f)
184
+ print(f"\t[I] datetime: {dt}")
185
+ lat, lon = get_image_latlon(_f)
186
+ print(f"\t[I] lat, lon: {lat}, {lon}")
187
+
188
+
189
+ # we expect to get the following info from the files
190
+ # file1:
191
+ # datetime: 2024:10:24 15:59:45
192
+ # lat, lon: 46.51860277777778, 6.562075
193
+ # file2:
194
+ # datetime: None
195
+ # lat, lon: 46.51860277777778, 6.562075
196
+
197
+ # let's run assertions on the backend data (session_state)
198
+ # and then on the front end too (visual elements)
199
+ f1 = at.session_state.file_uploader_data[0]
200
+ f2 = at.session_state.file_uploader_data[1]
201
+
202
+ assert get_image_datetime(f1) == "2024:10:24 15:59:45"
203
+ assert get_image_datetime(f2) == None
204
+ # use a tolerance of 1e-6, assert that the lat, lon is close to 46.5186
205
+ assert abs(get_image_latlon(f1)[0] - 46.51860277777778) < 1e-6
206
+ assert abs(get_image_latlon(f1)[1] - 6.562075) < 1e-6
207
+ assert abs(get_image_latlon(f2)[0] - 46.51860277777778) < 1e-6
208
+ assert abs(get_image_latlon(f2)[1] - 6.562075) < 1e-6
209
+
210
+ # need to run the script top-to-bottom to get the text_area elements
211
+ # since they are dynamically created.
212
+ at.run()
213
+
214
+ # since we uplaoded num_files files, hopefully we get num_files text areas
215
+ assert len(at.text_area) == num_files
216
+ # expecting
217
+ exp0 = "index: 0, name: cakes.jpg, datetime: 2024:10:24 15:59:45, lat: 46.51860277777778, lon:6.562075"
218
+ exp1 = "index: 1, name: cakes_no_exif_datetime.jpg, datetime: None, lat: 46.51860277777778, lon:6.562075"
219
+ exp2 = "index: 2, name: cakes_no_exif_gps.jpg, datetime: 2024:10:24 15:59:45, lat: None, lon:None"
220
+
221
+ assert at.text_area[0].value == exp0
222
+ assert at.text_area[1].value == exp1
223
+ if num_files >= 1:
224
+ assert at.text_area(key='metadata_0').value == exp0
225
+ if num_files >= 2:
226
+ assert at.text_area(key='metadata_1').value == exp1
227
+ if num_files >= 3:
228
+ assert at.text_area(key='metadata_2').value == exp2
229
+
230
+ # {"fname": "cakes.jpg", "size": 1234, "type": "image/jpeg"},
231
+ # {"fname": "cakes_no_exif_datetime.jpg", "size": 12345, "type": "image/jpeg"},
232
+ # {"fname": "cakes_no_exif_gps.jpg", "size": 123456, "type": "image/jpeg"},
233
+ #]
234
+
235
+
236
+ # Verify the behavior in your app
237
+ assert len(at.session_state.file_uploader_data) == num_files
238
+
239
+ assert at.session_state.file_uploader_data[0].size == 1234 # Check properties of the first file
240
+ assert at.session_state.file_uploader_data[1].name == "cakes_no_exif_datetime.jpg"
241
 
242
 
243
+ # this test was a stepping stone; when I was mocking files that didn't have any real data
244
+ # - it helped to explore how properties should be set in the mock object and generator funcs.
245
  @patch("streamlit.file_uploader")
246
  def test_mockupload_list(mock_file_uploader_rtn: MagicMock, mock_uploadedFile_List):
247
  # Create a list of 2 mock files