-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathimage_processor.py
132 lines (103 loc) · 5.07 KB
/
image_processor.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
import os
import base64
from datetime import datetime
from dotenv import load_dotenv
from typing import Union, List, Tuple
load_dotenv()
DEBUG = os.environ.get('DEBUG', False) == 'True'
static_url = os.environ.get('static_url')
upload_folder = os.environ.get("upload_folder")
def get_py_stamp(js_timestamp: str):
""" Converts js timestamp to python timestamp
Args:
js_timestamp (str): JS timestamp in format: "dd/mm/yyyy, hh:mm:ss AM/PM"
Returns:
str: Python timestamp in format: "yyyy-mm-dd_hh-mm-ss"
"""
try:
dt_timestamp = datetime.strptime(js_timestamp, "%d/%m/%Y, %I:%M:%S %p")
py_timestamp = dt_timestamp.strftime("%Y-%m-%d_%Hh%Mm%Ss")
return py_timestamp
except ValueError as e:
raise ValueError(f"Invalid JS timestamp format: {js_timestamp}") from e
def process_image(
timestamps: Union[List[str], str],
base64s: Union[List[str], str]
) -> Tuple[List[str], List[str], List[str]]:
"""
Takes js timestamps and base64s
Converts into py stamps, and also, saves the images
Returns modified js stamps with _1, _2 etc. as well
Args:
timestamps (Union[list, str]): List of timestamps in format: "dd/mm/yyyy, hh:mm:ss AM/PM"
base64s (Union[list, str]): List of base64 strings
Returns:
Tuple of:
- List: List of file names
- List: List of python timestamps
- List: List of modified js timestamps
"""
if isinstance(timestamps, str):
timestamps = [timestamps]
if isinstance(base64s, str):
base64s = [base64s]
if DEBUG:
print(f"[Img-processor Info]: Got {len(base64s)} images and {len(timestamps)} timestamps...")
if len(timestamps) != len(base64s):
raise ValueError(
"Mismatched lengths: timestamps and base64 strings must have the same number of elements.")
curr_stamp = datetime.now()
py_time_stamps = []
js_modified_time_stamps = []
file_names = []
# Issues is that, when multiple frames under same second are passed, our naming scheme does not support that. The same name is returned by function and only one image is over-written again n again with that PARTICULAR name.
# So keeping the last saved name in memory to add some _1, _2 after the repeated names.
last_saved = ""
same_name_count = 0
# with open("formData.json", 'w') as f:
# f.write(json.dumps({'timestamps': timestamps, 'bases': base64s}))
for idx, (timestamp, base64_str) in enumerate(zip(timestamps, base64s)):
try:
# get the extension from: "_base64_string"
base64_split = base64_str.split(',')
extension = base64_split[0].split('/')[1].split(';')[0]
# remove that part: "data:image/jpeg;base64"
image_data = base64.b64decode(base64_split[1])
# Create subfolder for current session:
# "uploads/curr_timestamp_folder/all_images_in_that_session"
subfolder = f'{curr_stamp.strftime("%Y-%m-%d_%Hh%Mm%Ss")}'
folder = os.path.join(static_url, upload_folder, subfolder)
os.makedirs(folder, exist_ok=True)
# Generate unique file name
file_base_name = get_py_stamp(timestamp)
if file_base_name == last_saved:
same_name_count += 1
file_base_name += f'_{same_name_count}'
else:
last_saved = file_base_name
same_name_count = 0
file_name = f'{file_base_name}.{extension}'
file_path = os.path.join(folder, file_name)
# Save the image to the specified path
with open(file_path, 'wb') as f:
f.write(image_data)
file_names.append(file_path)
py_time_stamps.append(file_base_name)
js_modified_time_stamps.append(f"{timestamp}, {same_name_count}")
except base64.binascii.Error as e:
print(f"[Img-processor Error] Invalid Base64 string at index {idx}: {e}")
except ValueError as e:
print(f"[Img-processor Error] Timestamp processing failed at index {idx}: {e}")
except Exception as e:
print(f"[Img-processor Error] Failed to process image at index {idx}: {e}")
if DEBUG:
print(f'[Img-processor Info]: Saved image `{file_name}` successfully...')
return file_names, py_time_stamps, js_modified_time_stamps
# Example usage:
# DEBUG = True
# timestamps = ["01/12/2024, 12:30:45 PM", "01/12/2024, 12:30:46 PM"]
# base64s = [
# "",
# "",
# ]
# process_image(timestamps, base64s)