Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Per segment chunks #8272

Merged
merged 139 commits into from
Sep 24, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
139 commits
Select commit Hold shift + click to select a range
b32a9eb
Update frame provider and media cache
zhiltsov-max Jul 25, 2024
cb4ff93
t
zhiltsov-max Jul 25, 2024
d49233c
t
zhiltsov-max Jul 30, 2024
146a896
Support static chunk building, fix av memory leak, add caching media …
zhiltsov-max Aug 1, 2024
52d1bac
Refactor static chunk generation - extract function, revise threading
zhiltsov-max Aug 2, 2024
0c53436
Refactor and fix task chunk creation from segment chunks, any storage
zhiltsov-max Aug 2, 2024
c166123
Fix chunk number validation
zhiltsov-max Aug 5, 2024
630c97e
Enable formatting for updated components
zhiltsov-max Aug 5, 2024
8d710e7
Remove the checksum field
zhiltsov-max Aug 5, 2024
654a827
Be consistent about returned task chunk types (allow video chunks)
zhiltsov-max Aug 6, 2024
12e5f2a
Support iterator input in video chunk writing
zhiltsov-max Aug 6, 2024
a79a681
Fix type annotation
zhiltsov-max Aug 6, 2024
d5118a2
Refactor video reader memory leak fix, add to reader with manifest
zhiltsov-max Aug 6, 2024
1b429cf
Disable threading in video reading in frame provider
zhiltsov-max Aug 6, 2024
d512312
Fix keyframe search
zhiltsov-max Aug 6, 2024
167ee12
Return frames as generator in dynamic chunk creation
zhiltsov-max Aug 6, 2024
88a9cb2
Update chunk requests in UI
zhiltsov-max Aug 7, 2024
30bf8fd
Update cache indices in FrameDecoder, enable video play
zhiltsov-max Aug 7, 2024
ee3c905
Fix frame retrieval for video
zhiltsov-max Aug 7, 2024
dc03220
Fix frame reading in updated dynamic cache building
zhiltsov-max Aug 7, 2024
4bb8a74
Fix invalid frame quality
zhiltsov-max Aug 9, 2024
f7d2c4c
Fix video reading in media_extractors - exception handling, frame mis…
zhiltsov-max Aug 9, 2024
34d9ca0
Allow disabling static chunks, add seamless switching
zhiltsov-max Aug 9, 2024
8c97967
Extend code formatting
zhiltsov-max Aug 9, 2024
a0fd0ba
Rename function argument
zhiltsov-max Aug 9, 2024
c0480c9
Rename configuration parameter
zhiltsov-max Aug 9, 2024
5caf283
Add av version comment
zhiltsov-max Aug 12, 2024
efbe3a0
Refactor av video reading
zhiltsov-max Aug 12, 2024
fb1284d
Fix manifest access
zhiltsov-max Aug 12, 2024
8edcfc5
Add migration
zhiltsov-max Aug 12, 2024
51a7f83
Update downloading from cloud storage for packed data in task creation
zhiltsov-max Aug 12, 2024
5a2a746
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 12, 2024
65e4174
Update changelog
zhiltsov-max Aug 12, 2024
61f1735
Merge remote-tracking branch 'origin/zm/job-chunks' into zm/job-chunks
zhiltsov-max Aug 12, 2024
34f972f
Update migration name
zhiltsov-max Aug 12, 2024
2bb2b17
Polish some code
zhiltsov-max Aug 12, 2024
3788917
Fix frame retrieval by id
zhiltsov-max Aug 12, 2024
f695ae1
Remove extra import
zhiltsov-max Aug 12, 2024
14a9033
Fix frame access in gt jobs
zhiltsov-max Aug 12, 2024
e8bebe9
Fix frame access in export
zhiltsov-max Aug 12, 2024
bbef52f
Fix frame iteration for frame step and excluded frames, fix export in…
zhiltsov-max Aug 12, 2024
3d5bb52
Remove unused import
zhiltsov-max Aug 13, 2024
0e9c5c8
Fix error check in test
zhiltsov-max Aug 13, 2024
351bdc8
Fix cleanup in test
zhiltsov-max Aug 13, 2024
a71852c
Add handling for disabled static cache during task creation
zhiltsov-max Aug 13, 2024
d90ca0d
Refactor some code
zhiltsov-max Aug 13, 2024
03e749a
Fix downloading for cloud data in task creation
zhiltsov-max Aug 13, 2024
c0822a0
Fix preview reading for projects
zhiltsov-max Aug 13, 2024
56d413f
Fix failing sdk tests
zhiltsov-max Aug 13, 2024
48f4794
Fix other failing sdk tests
zhiltsov-max Aug 13, 2024
5c0cc1a
Improve logging for migration
zhiltsov-max Aug 14, 2024
5abd891
Fix invalid starting index
zhiltsov-max Aug 14, 2024
749b970
Fix frame reading in lambda functions
zhiltsov-max Aug 14, 2024
9105cd3
Fix unintended frame indexing changes
zhiltsov-max Aug 14, 2024
8dafcbe
Fix various indexing errors in media extractors
zhiltsov-max Aug 14, 2024
4cbf82f
Fix temp resource cleanup in server tests
zhiltsov-max Aug 14, 2024
88c34a3
Refactor some code
zhiltsov-max Aug 15, 2024
b0fd006
Remove duplicated tests
zhiltsov-max Aug 15, 2024
2eac04a
Remove extra change
zhiltsov-max Aug 15, 2024
640518c
Fix method name, remove extra method
zhiltsov-max Aug 15, 2024
3a246b3
Remove some shared code in tests, add temp data cleanup
zhiltsov-max Aug 15, 2024
a0704f4
Add checks for successful task creation in tests
zhiltsov-max Aug 15, 2024
cf026ef
Fix invalid variable access in test
zhiltsov-max Aug 15, 2024
f73cef3
Update default cache location in test checks
zhiltsov-max Aug 15, 2024
258c800
Update manifest validation logic, allow manifest input in any task da…
zhiltsov-max Aug 16, 2024
b3ae317
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 16, 2024
5e89ef4
Add task chunk caching, refactor chunk building
zhiltsov-max Aug 16, 2024
c5edcda
Refactor some code
zhiltsov-max Aug 16, 2024
7f5c722
Refactor some code
zhiltsov-max Aug 16, 2024
daf4035
Improve parameter name
zhiltsov-max Aug 16, 2024
8c1b82c
Fix function call
zhiltsov-max Aug 16, 2024
f172865
Add basic test set for meta, frames, and chunks reading in tasks
zhiltsov-max Aug 16, 2024
aacceee
Move class declaration for pylint compatibility
zhiltsov-max Aug 16, 2024
c8dbb7c
Add missing original chunk type field in job responses
zhiltsov-max Aug 16, 2024
6b9a3e9
Add tests for job data access
zhiltsov-max Aug 16, 2024
f5661e4
Update test assets
zhiltsov-max Aug 16, 2024
754757f
Clean imports
zhiltsov-max Aug 16, 2024
0c001a5
Python 3.8 compatibility
zhiltsov-max Aug 16, 2024
a9390eb
Python 3.8 compatibility
zhiltsov-max Aug 17, 2024
d2b1385
Python 3.8 compatibility
zhiltsov-max Aug 17, 2024
c9a5e31
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 19, 2024
621afa7
Add logging into shell command runs, fix invalid redis-cli invocation…
zhiltsov-max Aug 19, 2024
e40ffd1
Merge remote-tracking branch 'origin/zm/job-chunks' into zm/job-chunks
zhiltsov-max Aug 19, 2024
08c9f01
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 19, 2024
92a19f4
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 21, 2024
441d0e7
Allow calling flushall in redis in helm tests
zhiltsov-max Aug 21, 2024
0963f94
Update comment
zhiltsov-max Aug 21, 2024
0d78e63
Update redis cleanup command
zhiltsov-max Aug 21, 2024
f53948d
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 23, 2024
e69f2b7
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 23, 2024
1a9a813
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 23, 2024
e4db8ad
Reuse _get
zhiltsov-max Aug 28, 2024
b1c54f9
Make get_checksum private
zhiltsov-max Aug 28, 2024
5312b00
Add get_raw_data_dirname to the Data model
zhiltsov-max Aug 28, 2024
3c117fe
Make SegmentFrameProvider available in make_frame_provider
zhiltsov-max Aug 28, 2024
98eff81
Remove extra variable
zhiltsov-max Aug 28, 2024
316ec78
Include both cases of CVAT_ALLOW_STATIC_CACHE in CI checks
zhiltsov-max Aug 28, 2024
ebed825
Merge remote-tracking branch 'origin/zm/job-chunks' into zm/job-chunks
zhiltsov-max Aug 28, 2024
92f6083
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Aug 28, 2024
2b6e987
Remove extra import
zhiltsov-max Aug 28, 2024
f67a1a2
Update changelog
zhiltsov-max Sep 5, 2024
d72fe85
Refactor cache keys in media cache
zhiltsov-max Sep 5, 2024
d5bfb88
Refactor selective segment chunk creation
zhiltsov-max Sep 5, 2024
c5a1197
Remove the breaking change in the chunk retrieval API, add a new inde…
zhiltsov-max Sep 6, 2024
a5cf3b7
Update UI to use the new chunk index parameter
zhiltsov-max Sep 7, 2024
069f48c
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Sep 7, 2024
cfdde3f
Update test initialization
zhiltsov-max Sep 7, 2024
843b957
Update changelog
zhiltsov-max Sep 7, 2024
feb92cd
Add backward compatibility for chunk "number" in GT jobs, remove plac…
zhiltsov-max Sep 9, 2024
2424f2b
Update UI to support job chunks with non-sequential frame ids
zhiltsov-max Sep 9, 2024
fe60bdf
Fix job frame retrieval
zhiltsov-max Sep 9, 2024
6ddb6bf
Fix 3d task chunk writing
zhiltsov-max Sep 9, 2024
4fa7b97
Fix frame retrieval in UI
zhiltsov-max Sep 10, 2024
32f1be2
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Sep 10, 2024
0e95b40
Fix chunk availability check
zhiltsov-max Sep 11, 2024
21135b7
Merge remote-tracking branch 'origin/zm/job-chunks' into zm/job-chunks
zhiltsov-max Sep 11, 2024
b311f1e
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Sep 11, 2024
79bb1f7
Remove array comparisons
zhiltsov-max Sep 12, 2024
55a8424
Update validateFrameNumbers
zhiltsov-max Sep 12, 2024
add5ae6
Use builtins for range and binary search, convert frame step into a c…
zhiltsov-max Sep 12, 2024
643d998
Merge remote-tracking branch 'origin/zm/job-chunks' into zm/job-chunks
zhiltsov-max Sep 12, 2024
df90b33
Fix cached chunk indicators in frame player
zhiltsov-max Sep 12, 2024
6ccb7db
Fix chunk predecode logic
zhiltsov-max Sep 13, 2024
1fb68bc
Rename chunkNumber to chunkIndex where necessary
zhiltsov-max Sep 13, 2024
92d0c7a
Fix potential prefetch problem with reverse playback
zhiltsov-max Sep 13, 2024
67c1650
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Sep 13, 2024
3cdc4dc
Move env variable into docker-compose.yml
zhiltsov-max Sep 16, 2024
716042e
Merge remote-tracking branch 'origin/zm/job-chunks' into zm/job-chunks
zhiltsov-max Sep 16, 2024
19279c7
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Sep 16, 2024
bc5ed39
Fix invalid cached chunk display in GT jobs
zhiltsov-max Sep 17, 2024
08ddd28
Fix invalid task preview generation
zhiltsov-max Sep 17, 2024
1d969bd
Refactor CS previews, context image chunk generation, media cache cre…
zhiltsov-max Sep 17, 2024
e2cba8c
Merge remote-tracking branch 'origin/zm/job-chunks' into zm/job-chunks
zhiltsov-max Sep 17, 2024
d135475
Remove extra import
zhiltsov-max Sep 17, 2024
a1638c9
Fix CS preview in response
zhiltsov-max Sep 17, 2024
fc89c01
Add reverse migration
zhiltsov-max Sep 17, 2024
c6e65f6
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Sep 17, 2024
118828a
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Sep 23, 2024
3eea60d
Merge branch 'develop' into zm/job-chunks
zhiltsov-max Sep 24, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Refactor CS previews, context image chunk generation, media cache cre…
…ation and retrieval
  • Loading branch information
zhiltsov-max committed Sep 17, 2024
commit 1d969bd7f41c217d813c55d3cfdc9c8b20b737bb
155 changes: 92 additions & 63 deletions cvat/apps/engine/cache.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
Tuple,
Type,
Union,
overload,
)

import av
Expand Down Expand Up @@ -73,21 +74,20 @@ def _get_checksum(self, value: bytes) -> int:

def _get_or_set_cache_item(
self, key: str, create_callback: Callable[[], DataWithMime]
) -> DataWithMime:
) -> _CacheItem:
def create_item() -> _CacheItem:
slogger.glob.info(f"Starting to prepare chunk: key {key}")
item_data = create_callback()
slogger.glob.info(f"Ending to prepare chunk: key {key}")

if item_data[0]:
item = (item_data[0], item_data[1], self._get_checksum(item_data[0].getbuffer()))
item_data_bytes = item_data[0].getvalue()
item = (item_data[0], item_data[1], self._get_checksum(item_data_bytes))
if item_data_bytes:
self._cache.set(key, item)
else:
item = (item_data[0], item_data[1], None)

return item

item = self._get(key)
item = self._get_cache_item(key)
if not item:
item = create_item()
else:
Expand All @@ -98,9 +98,9 @@ def create_item() -> _CacheItem:
slogger.glob.info(f"Recreating cache item {key} due to checksum mismatch")
item = create_item()

return item[0], item[1]
return item

def _get(self, key: str) -> Optional[DataWithMime]:
def _get_cache_item(self, key: str) -> Optional[_CacheItem]:
slogger.glob.info(f"Starting to get chunk from cache: key {key}")
try:
item = self._cache.get(key)
Expand Down Expand Up @@ -152,20 +152,36 @@ def _make_segment_task_chunk_key(
def _make_context_image_preview_key(self, db_data: models.Data, frame_number: int) -> str:
return f"context_image_{db_data.id}_{frame_number}_preview"

def get_segment_chunk(
@overload
def _to_data_with_mime(self, cache_item: _CacheItem) -> DataWithMime: ...

@overload
def _to_data_with_mime(self, cache_item: Optional[_CacheItem]) -> Optional[DataWithMime]: ...

def _to_data_with_mime(self, cache_item: Optional[_CacheItem]) -> Optional[DataWithMime]:
if not cache_item:
return None

return cache_item[:2]

def get_or_set_segment_chunk(
self, db_segment: models.Segment, chunk_number: int, *, quality: FrameQuality
) -> DataWithMime:
return self._get_or_set_cache_item(
key=self._make_chunk_key(db_segment, chunk_number, quality=quality),
create_callback=lambda: self.prepare_segment_chunk(
db_segment, chunk_number, quality=quality
),
return self._to_data_with_mime(
self._get_or_set_cache_item(
key=self._make_chunk_key(db_segment, chunk_number, quality=quality),
create_callback=lambda: self.prepare_segment_chunk(
db_segment, chunk_number, quality=quality
),
)
)

def get_task_chunk(
self, db_task: models.Task, chunk_number: int, *, quality: FrameQuality
) -> Optional[DataWithMime]:
return self._get(key=self._make_chunk_key(db_task, chunk_number, quality=quality))
return self._to_data_with_mime(
self._get_cache_item(key=self._make_chunk_key(db_task, chunk_number, quality=quality))
)

def get_or_set_task_chunk(
self,
Expand All @@ -175,16 +191,20 @@ def get_or_set_task_chunk(
quality: FrameQuality,
set_callback: Callable[[], DataWithMime],
) -> DataWithMime:
return self._get_or_set_cache_item(
key=self._make_chunk_key(db_task, chunk_number, quality=quality),
create_callback=set_callback,
return self._to_data_with_mime(
self._get_or_set_cache_item(
key=self._make_chunk_key(db_task, chunk_number, quality=quality),
create_callback=set_callback,
)
)

def get_segment_task_chunk(
self, db_segment: models.Segment, chunk_number: int, *, quality: FrameQuality
) -> Optional[DataWithMime]:
return self._get(
key=self._make_segment_task_chunk_key(db_segment, chunk_number, quality=quality)
return self._to_data_with_mime(
self._get_cache_item(
key=self._make_segment_task_chunk_key(db_segment, chunk_number, quality=quality)
)
)

def get_or_set_segment_task_chunk(
Expand All @@ -195,40 +215,52 @@ def get_or_set_segment_task_chunk(
quality: FrameQuality,
set_callback: Callable[[], DataWithMime],
) -> DataWithMime:
return self._get_or_set_cache_item(
key=self._make_segment_task_chunk_key(db_segment, chunk_number, quality=quality),
create_callback=set_callback,
return self._to_data_with_mime(
self._get_or_set_cache_item(
key=self._make_segment_task_chunk_key(db_segment, chunk_number, quality=quality),
create_callback=set_callback,
)
)

def get_selective_job_chunk(
def get_or_set_selective_job_chunk(
self, db_job: models.Job, chunk_number: int, *, quality: FrameQuality
) -> DataWithMime:
return self._get_or_set_cache_item(
key=self._make_chunk_key(db_job, chunk_number, quality=quality),
create_callback=lambda: self.prepare_masked_range_segment_chunk(
db_job.segment, chunk_number, quality=quality
),
return self._to_data_with_mime(
self._get_or_set_cache_item(
key=self._make_chunk_key(db_job, chunk_number, quality=quality),
create_callback=lambda: self.prepare_masked_range_segment_chunk(
db_job.segment, chunk_number, quality=quality
),
)
)

def get_or_set_segment_preview(self, db_segment: models.Segment) -> DataWithMime:
return self._get_or_set_cache_item(
self._make_preview_key(db_segment),
create_callback=lambda: self._prepare_segment_preview(db_segment),
return self._to_data_with_mime(
self._get_or_set_cache_item(
self._make_preview_key(db_segment),
create_callback=lambda: self._prepare_segment_preview(db_segment),
)
)

def get_cloud_preview(self, db_storage: models.CloudStorage) -> Optional[DataWithMime]:
return self._get(self._make_preview_key(db_storage))
return self._to_data_with_mime(self._get_cache_item(self._make_preview_key(db_storage)))

def get_or_set_cloud_preview(self, db_storage: models.CloudStorage) -> DataWithMime:
return self._get_or_set_cache_item(
self._make_preview_key(db_storage),
create_callback=lambda: self._prepare_cloud_preview(db_storage),
return self._to_data_with_mime(
self._get_or_set_cache_item(
self._make_preview_key(db_storage),
create_callback=lambda: self._prepare_cloud_preview(db_storage),
)
)

def get_frame_context_images(self, db_data: models.Data, frame_number: int) -> DataWithMime:
return self._get_or_set_cache_item(
key=self._make_context_image_preview_key(db_data, frame_number),
create_callback=lambda: self.prepare_context_images(db_data, frame_number),
def get_or_set_frame_context_images_chunk(
self, db_data: models.Data, frame_number: int
) -> DataWithMime:
return self._to_data_with_mime(
self._get_or_set_cache_item(
key=self._make_context_image_preview_key(db_data, frame_number),
create_callback=lambda: self.prepare_context_images_chunk(db_data, frame_number),
)
)

def _read_raw_images(
Expand Down Expand Up @@ -536,59 +568,56 @@ def _prepare_segment_preview(self, db_segment: models.Segment) -> DataWithMime:

return prepare_preview_image(preview)

def _prepare_cloud_preview(self, db_storage):
def _prepare_cloud_preview(self, db_storage: models.CloudStorage) -> DataWithMime:
storage = db_storage_to_storage_instance(db_storage)
if not db_storage.manifests.count():
raise ValidationError("Cannot get the cloud storage preview. There is no manifest file")

preview_path = None
for manifest_model in db_storage.manifests.all():
manifest_prefix = os.path.dirname(manifest_model.filename)
for db_manifest in db_storage.manifests.all():
manifest_prefix = os.path.dirname(db_manifest.filename)

full_manifest_path = os.path.join(
db_storage.get_storage_dirname(), manifest_model.filename
db_storage.get_storage_dirname(), db_manifest.filename
)
if not os.path.exists(full_manifest_path) or datetime.fromtimestamp(
os.path.getmtime(full_manifest_path), tz=timezone.utc
) < storage.get_file_last_modified(manifest_model.filename):
storage.download_file(manifest_model.filename, full_manifest_path)
) < storage.get_file_last_modified(db_manifest.filename):
storage.download_file(db_manifest.filename, full_manifest_path)

manifest = ImageManifestManager(
os.path.join(db_storage.get_storage_dirname(), manifest_model.filename),
os.path.join(db_storage.get_storage_dirname(), db_manifest.filename),
db_storage.get_storage_dirname(),
)
# need to update index
manifest.set_index()
if not len(manifest):
continue

preview_info = manifest[0]
preview_filename = "".join([preview_info["name"], preview_info["extension"]])
preview_path = os.path.join(manifest_prefix, preview_filename)
break

if not preview_path:
msg = "Cloud storage {} does not contain any images".format(db_storage.pk)
slogger.cloud_storage[db_storage.pk].info(msg)
raise NotFound(msg)

buff = storage.download_fileobj(preview_path)
mime_type = mimetypes.guess_type(preview_path)[0]

return buff, mime_type
image = PIL.Image.open(buff)
return prepare_preview_image(image)

def prepare_context_images(
self, db_data: models.Data, frame_number: int
) -> Optional[DataWithMime]:
def prepare_context_images_chunk(self, db_data: models.Data, frame_number: int) -> DataWithMime:
zip_buffer = io.BytesIO()
try:
image = models.Image.objects.get(data_id=db_data.id, frame=frame_number)
except models.Image.DoesNotExist:
return None

if not image.related_files.count():
return None, None
related_images = db_data.related_files.filter(primary_image__frame=frame_number).all()
if not related_images:
return zip_buffer, ""

with zipfile.ZipFile(zip_buffer, "a", zipfile.ZIP_DEFLATED, False) as zip_file:
common_path = os.path.commonpath(
list(map(lambda x: str(x.path), image.related_files.all()))
)
for i in image.related_files.all():
common_path = os.path.commonpath(list(map(lambda x: str(x.path), related_images)))
for i in related_images:
path = os.path.realpath(str(i.path))
name = os.path.relpath(str(i.path), common_path)
image = cv2.imread(path)
Expand Down
28 changes: 16 additions & 12 deletions cvat/apps/engine/frame_provider.py
Original file line number Diff line number Diff line change
Expand Up @@ -200,7 +200,7 @@ def get_frame(
) -> DataWithMeta[AnyFrame]: ...

@abstractmethod
def get_frame_context_images(
def get_frame_context_images_chunk(
self,
frame_number: int,
) -> Optional[DataWithMeta[BytesIO]]: ...
Expand Down Expand Up @@ -350,11 +350,13 @@ def get_frame(
frame_number, quality=quality, out_type=out_type
)

def get_frame_context_images(
def get_frame_context_images_chunk(
self,
frame_number: int,
) -> Optional[DataWithMeta[BytesIO]]:
return self._get_segment_frame_provider(frame_number).get_frame_context_images(frame_number)
return self._get_segment_frame_provider(frame_number).get_frame_context_images_chunk(
frame_number
)

def iterate_frames(
self,
Expand Down Expand Up @@ -432,15 +434,15 @@ def __init__(self, db_segment: models.Segment) -> None:
self._loaders[FrameQuality.COMPRESSED] = _BufferChunkLoader(
reader_class=reader_class[db_data.compressed_chunk_type][0],
reader_params=reader_class[db_data.compressed_chunk_type][1],
get_chunk_callback=lambda chunk_idx: cache.get_segment_chunk(
get_chunk_callback=lambda chunk_idx: cache.get_or_set_segment_chunk(
db_segment, chunk_idx, quality=FrameQuality.COMPRESSED
),
)

self._loaders[FrameQuality.ORIGINAL] = _BufferChunkLoader(
reader_class=reader_class[db_data.original_chunk_type][0],
reader_params=reader_class[db_data.original_chunk_type][1],
get_chunk_callback=lambda chunk_idx: cache.get_segment_chunk(
get_chunk_callback=lambda chunk_idx: cache.get_or_set_segment_chunk(
db_segment, chunk_idx, quality=FrameQuality.ORIGINAL
),
)
Expand Down Expand Up @@ -549,19 +551,21 @@ def get_frame(

return return_type(frame, mime=mimetypes.guess_type(frame_name)[0])

def get_frame_context_images(
def get_frame_context_images_chunk(
self,
frame_number: int,
) -> Optional[DataWithMeta[BytesIO]]:
# TODO: refactor, optimize
cache = MediaCache()
self.validate_frame_number(frame_number)

if self._db_segment.task.data.storage_method == models.StorageMethodChoice.CACHE:
data, mime = cache.get_frame_context_images(self._db_segment.task.data, frame_number)
db_data = self._db_segment.task.data

cache = MediaCache()
if db_data.storage_method == models.StorageMethodChoice.CACHE:
data, mime = cache.get_or_set_frame_context_images_chunk(db_data, frame_number)
else:
data, mime = cache.prepare_context_images(self._db_segment.task.data, frame_number)
data, mime = cache.prepare_context_images_chunk(db_data, frame_number)

if not data:
if not data.getvalue():
return None

return DataWithMeta[BytesIO](data, mime=mime)
Expand Down
2 changes: 1 addition & 1 deletion cvat/apps/engine/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -703,7 +703,7 @@ def __call__(self):
return HttpResponse(data.data.getvalue(), content_type=data.mime)

elif self.type == 'context_image':
data = frame_provider.get_frame_context_images(self.number)
data = frame_provider.get_frame_context_images_chunk(self.number)
if not data:
return HttpResponseNotFound()

Expand Down