vision-agent 0.2.165__py3-none-any.whl → 0.2.166__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- vision_agent/utils/video.py +23 -3
- {vision_agent-0.2.165.dist-info → vision_agent-0.2.166.dist-info}/METADATA +1 -1
- {vision_agent-0.2.165.dist-info → vision_agent-0.2.166.dist-info}/RECORD +5 -5
- {vision_agent-0.2.165.dist-info → vision_agent-0.2.166.dist-info}/LICENSE +0 -0
- {vision_agent-0.2.165.dist-info → vision_agent-0.2.166.dist-info}/WHEEL +0 -0
vision_agent/utils/video.py
CHANGED
@@ -11,6 +11,9 @@ import numpy as np
|
|
11
11
|
_LOGGER = logging.getLogger(__name__)
|
12
12
|
# The maximum length of the clip to extract frames from, in seconds
|
13
13
|
|
14
|
+
_DEFAULT_VIDEO_FPS = 24
|
15
|
+
_DEFAULT_INPUT_FPS = 1.0
|
16
|
+
|
14
17
|
|
15
18
|
def play_video(video_base64: str) -> None:
|
16
19
|
"""Play a video file"""
|
@@ -51,7 +54,9 @@ def _resize_frame(frame: np.ndarray) -> np.ndarray:
|
|
51
54
|
|
52
55
|
|
53
56
|
def video_writer(
|
54
|
-
frames: List[np.ndarray],
|
57
|
+
frames: List[np.ndarray],
|
58
|
+
fps: float = _DEFAULT_INPUT_FPS,
|
59
|
+
filename: Optional[str] = None,
|
55
60
|
) -> str:
|
56
61
|
if filename is None:
|
57
62
|
filename = tempfile.NamedTemporaryFile(delete=False, suffix=".mp4").name
|
@@ -78,7 +83,7 @@ def video_writer(
|
|
78
83
|
|
79
84
|
|
80
85
|
def frames_to_bytes(
|
81
|
-
frames: List[np.ndarray], fps: float =
|
86
|
+
frames: List[np.ndarray], fps: float = _DEFAULT_INPUT_FPS, file_ext: str = ".mp4"
|
82
87
|
) -> bytes:
|
83
88
|
r"""Convert a list of frames to a video file encoded into a byte string.
|
84
89
|
|
@@ -101,7 +106,7 @@ def frames_to_bytes(
|
|
101
106
|
# same file name and the time savings are very large.
|
102
107
|
@lru_cache(maxsize=8)
|
103
108
|
def extract_frames_from_video(
|
104
|
-
video_uri: str, fps: float =
|
109
|
+
video_uri: str, fps: float = _DEFAULT_INPUT_FPS
|
105
110
|
) -> List[Tuple[np.ndarray, float]]:
|
106
111
|
"""Extract frames from a video along with the timestamp in seconds.
|
107
112
|
|
@@ -118,6 +123,16 @@ def extract_frames_from_video(
|
|
118
123
|
|
119
124
|
cap = cv2.VideoCapture(video_uri)
|
120
125
|
orig_fps = cap.get(cv2.CAP_PROP_FPS)
|
126
|
+
if not orig_fps or orig_fps <= 0:
|
127
|
+
_LOGGER.warning(
|
128
|
+
f"Input video, {video_uri}, has no fps, using the default value {_DEFAULT_VIDEO_FPS}"
|
129
|
+
)
|
130
|
+
orig_fps = _DEFAULT_VIDEO_FPS
|
131
|
+
if not fps or fps <= 0:
|
132
|
+
_LOGGER.warning(
|
133
|
+
f"Input fps, {fps}, is illegal, using the default value: {_DEFAULT_INPUT_FPS}"
|
134
|
+
)
|
135
|
+
fps = _DEFAULT_INPUT_FPS
|
121
136
|
orig_frame_time = 1 / orig_fps
|
122
137
|
targ_frame_time = 1 / fps
|
123
138
|
frames: List[Tuple[np.ndarray, float]] = []
|
@@ -129,10 +144,15 @@ def extract_frames_from_video(
|
|
129
144
|
break
|
130
145
|
|
131
146
|
elapsed_time += orig_frame_time
|
147
|
+
# This is to prevent float point precision loss issue, which can cause
|
148
|
+
# the elapsed time to be slightly less than the target frame time, which
|
149
|
+
# causes the last frame to be skipped
|
150
|
+
elapsed_time = round(elapsed_time, 8)
|
132
151
|
if elapsed_time >= targ_frame_time:
|
133
152
|
frames.append((cv2.cvtColor(frame, cv2.COLOR_BGR2RGB), i / orig_fps))
|
134
153
|
elapsed_time -= targ_frame_time
|
135
154
|
|
136
155
|
i += 1
|
137
156
|
cap.release()
|
157
|
+
_LOGGER.info(f"Extracted {len(frames)} frames from {video_uri}")
|
138
158
|
return frames
|
@@ -28,8 +28,8 @@ vision_agent/utils/execute.py,sha256=FqSOr5gtBeKB1g2hbV6-bhox6qItDQNn2o9efq1w6f4
|
|
28
28
|
vision_agent/utils/image_utils.py,sha256=rm9GfXvD4JrjnqKrP_f2gfq4SzmqYC0IdC1kKwdn6sk,11303
|
29
29
|
vision_agent/utils/sim.py,sha256=ZuSS07TUXFGjipmiQoY8TKRmSes7XXCdtU9PI8PC1sw,5609
|
30
30
|
vision_agent/utils/type_defs.py,sha256=BE12s3JNQy36QvauXHjwyeffVh5enfcvd4vTzSwvEZI,1384
|
31
|
-
vision_agent/utils/video.py,sha256=
|
32
|
-
vision_agent-0.2.
|
33
|
-
vision_agent-0.2.
|
34
|
-
vision_agent-0.2.
|
35
|
-
vision_agent-0.2.
|
31
|
+
vision_agent/utils/video.py,sha256=fOPR48-SuwMbE5eB5rc2F7lVo6k1mVHn26eEJ0QCslc,5602
|
32
|
+
vision_agent-0.2.166.dist-info/LICENSE,sha256=xx0jnfkXJvxRnG63LTGOxlggYnIysveWIZ6H3PNdCrQ,11357
|
33
|
+
vision_agent-0.2.166.dist-info/METADATA,sha256=e15d4yNaAJvLCViaBUFo_RNHII88W-y9WgJauEFfbyU,18034
|
34
|
+
vision_agent-0.2.166.dist-info/WHEEL,sha256=7Z8_27uaHI_UZAc4Uox4PpBhQ9Y5_modZXWMxtUi4NU,88
|
35
|
+
vision_agent-0.2.166.dist-info/RECORD,,
|
File without changes
|
File without changes
|