From ec610a963c10d354098a19e43366edd9c7571c9a Mon Sep 17 00:00:00 2001 From: Eugene Liu Date: Fri, 22 Nov 2024 11:46:04 +0000 Subject: [PATCH] Allow empty tile annotation (#4124) * Add warnings for empty annotations in OTXTileDetTestDataset and OTXTileInstSegTestDataset * Fix empty annotation handling in tiling --- CHANGELOG.md | 2 ++ src/otx/core/data/dataset/tile.py | 21 ++++++++++++++------- 2 files changed, 16 insertions(+), 7 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 19a7fac40b..1ddf602a0f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -124,6 +124,8 @@ All notable changes to this project will be documented in this file. () - Disable tiling classifier toggle in configurable parameters () +- Fix empty annotation in tiling + () ## \[v2.1.0\] diff --git a/src/otx/core/data/dataset/tile.py b/src/otx/core/data/dataset/tile.py index a729ddc186..4187935aa4 100644 --- a/src/otx/core/data/dataset/tile.py +++ b/src/otx/core/data/dataset/tile.py @@ -7,6 +7,7 @@ import logging as log import operator +import warnings from copy import deepcopy from itertools import product from typing import TYPE_CHECKING, Callable @@ -372,14 +373,17 @@ def _get_item_impl(self, index: int) -> TileDetDataEntity: # type: ignore[overr img = item.media_as(Image) img_data, img_shape, _ = self._get_img_data_and_shape(img) - bbox_anns = [ann for ann in item.annotations if isinstance(ann, Bbox)] + gt_bboxes = [ann for ann in item.annotations if isinstance(ann, Bbox)] + + if empty_anno := len(gt_bboxes) == 0: + warnings.warn(f"Empty annotation for image {item.id}!", stacklevel=2) bboxes = ( - np.stack([ann.points for ann in bbox_anns], axis=0).astype(np.float32) - if len(bbox_anns) > 0 - else np.zeros((0, 4), dtype=np.float32) + np.empty((0, 4), dtype=np.float32) + if empty_anno + else np.stack([ann.points for ann in gt_bboxes], axis=0).astype(np.float32) ) - labels = torch.as_tensor([ann.label for ann in bbox_anns]) + labels = torch.as_tensor([ann.label for ann in gt_bboxes]) tile_entities, tile_attrs = self.get_tiles(img_data, item, index) @@ -476,11 +480,14 @@ def _get_item_impl(self, index: int) -> TileInstSegDataEntity: # type: ignore[o else: gt_masks.append(polygon_to_bitmap([annotation], *img_shape)[0]) + if empty_anno := len(gt_bboxes) == 0: + warnings.warn(f"Empty annotation for image {item.id}", stacklevel=2) + # convert xywh to xyxy format - bboxes = np.array(gt_bboxes, dtype=np.float32) + bboxes = np.empty((0, 4), dtype=np.float32) if empty_anno else np.stack(gt_bboxes, dtype=np.float32) bboxes[:, 2:] += bboxes[:, :2] - masks = np.stack(gt_masks, axis=0) if gt_masks else np.zeros((0, *img_shape), dtype=bool) + masks = np.stack(gt_masks, axis=0) if gt_masks else np.empty((0, *img_shape), dtype=bool) labels = np.array(gt_labels, dtype=np.int64) tile_entities, tile_attrs = self.get_tiles(img_data, item, index)