cocotools 0.0.7

Package providing functionalities to work with COCO format datasets.
Documentation
use pyo3::class::basic::CompareOp;
use pyo3::prelude::*;

use crate::coco::object_detection::*;

#[pymethods]
impl Annotation {
    #[new]
    fn new(
        id: u64,
        image_id: u64,
        category_id: u32,
        segmentation: Segmentation,
        area: f64,
        bbox: Bbox,
        iscrowd: u32,
    ) -> Self {
        Self {
            id,
            image_id,
            category_id,
            segmentation,
            area,
            bbox,
            iscrowd,
        }
    }

    fn __repr__(&self) -> String {
        format!(
            "Annotation(id={}, image_id={}, category_id={}, segmentation={}, area={}, bbox={}, iscrowd={})",
            self.id, self.image_id, self.category_id, &self.segmentation.__repr__(), self.area, &self.bbox.__repr__(), self.iscrowd
        )
    }

    fn __richcmp__(&self, other: &Self, op: CompareOp, py: Python<'_>) -> PyObject {
        match op {
            CompareOp::Eq => (self.id == other.id
                && self.image_id == other.image_id
                && self.category_id == other.category_id
                && self.segmentation == other.segmentation
                && self.area == other.area
                && self.bbox == other.bbox
                && self.iscrowd == other.iscrowd)
                .into_py(py),
            CompareOp::Ne => (self.id != other.id
                || self.image_id != other.image_id
                || self.category_id != other.category_id
                || self.segmentation != other.segmentation
                || self.area != other.area
                || self.bbox != other.bbox
                || self.iscrowd != other.iscrowd)
                .into_py(py),
            _ => py.NotImplemented(),
        }
    }
}

#[pymethods]
impl Category {
    #[new]
    fn new(id: u32, name: String, supercategory: String) -> Self {
        Self {
            id,
            name,
            supercategory,
        }
    }

    fn __repr__(&self) -> String {
        format!(
            "Category(id={}, name='{}', supercategory='{}')",
            self.id, self.name, self.supercategory
        )
    }

    fn __richcmp__(&self, other: &Self, op: CompareOp, py: Python<'_>) -> PyObject {
        match op {
            CompareOp::Eq => (self.id == other.id
                && self.name == other.name
                && self.supercategory == other.supercategory)
                .into_py(py),
            CompareOp::Ne => (self.id != other.id
                || self.name != other.name
                || self.supercategory != other.supercategory)
                .into_py(py),
            _ => py.NotImplemented(),
        }
    }
}

#[pymethods]
impl Image {
    #[new]
    fn new(id: u64, width: u32, height: u32, file_name: String) -> Self {
        Self {
            id,
            width,
            height,
            file_name,
            ..Default::default()
        }
    }

    fn __repr__(&self) -> String {
        format!(
            "Image(id={}, width='{}', height='{}', file_name='{}')",
            self.id, self.width, self.height, self.file_name
        )
    }

    fn __richcmp__(&self, other: &Self, op: CompareOp, py: Python<'_>) -> PyObject {
        match op {
            CompareOp::Eq => (self.id == other.id
                && self.width == other.width
                && self.height == other.height
                && self.file_name == other.file_name)
                .into_py(py),
            CompareOp::Ne => (self.id != other.id
                || self.width != other.width
                || self.height != other.height
                || self.file_name != other.file_name)
                .into_py(py),
            _ => py.NotImplemented(),
        }
    }
}

#[pyclass]
struct BboxIter {
    inner: std::vec::IntoIter<f64>,
}

#[pymethods]
impl BboxIter {
    fn __iter__(slf: PyRef<'_, Self>) -> PyRef<'_, Self> {
        slf
    }

    fn __next__(mut slf: PyRefMut<'_, Self>) -> Option<f64> {
        slf.inner.next()
    }
}

#[pymethods]
impl Bbox {
    #[new]
    fn new(left: f64, top: f64, width: f64, height: f64) -> Self {
        Self {
            left,
            top,
            width,
            height,
        }
    }

    fn __repr__(&self) -> String {
        format!(
            "BBox(left={}, top={}, width={}, height={})",
            self.left, self.top, self.width, self.height
        )
    }

    fn __len__(&self) -> usize {
        4
    }

    fn __getitem__(&self, idx: usize) -> f64 {
        // https://pyo3.rs/main/doc/pyo3/types/struct.pysequence
        // https://docs.rs/pyo3/0.14.3/pyo3/class/sequence/trait.PySequenceProtocol.html
        [self.left, self.top, self.width, self.height][idx]
    }

    fn __iter__(slf: PyRef<'_, Self>) -> PyResult<Py<BboxIter>> {
        let iter = BboxIter {
            inner: vec![slf.left, slf.top, slf.width, slf.height].into_iter(),
        };
        Py::new(slf.py(), iter)
    }

    fn __richcmp__(&self, other: &Self, op: CompareOp, py: Python<'_>) -> PyObject {
        match op {
            // For some reason, the COCO dataset uses floats instead of ints, hence the < 0.01.
            CompareOp::Eq => ((self.left - other.left).abs() < 0.01
                && (self.top - other.top).abs() < 0.01
                && (self.width - other.width).abs() < 0.01
                && (self.height - other.height).abs() < 0.01)
                .into_py(py),
            CompareOp::Ne => ((self.left - other.left).abs() > 0.01
                || (self.top - other.top).abs() > 0.01
                || (self.width - other.width).abs() > 0.01
                || (self.height - other.height).abs() > 0.01)
                .into_py(py),
            _ => py.NotImplemented(),
        }
    }
}

#[pymethods]
impl Rle {
    #[new]
    fn new(size: Vec<u32>, counts: Vec<u32>) -> Self {
        Self { size, counts }
    }

    fn __repr__(&self) -> String {
        format!("RLE(size={:?}, counts={:?})", self.size, self.counts)
    }

    fn __richcmp__(&self, other: &Self, op: CompareOp, py: Python<'_>) -> PyObject {
        match op {
            CompareOp::Eq => (self.size == other.size && self.counts == other.counts).into_py(py),
            CompareOp::Ne => (self.size != other.size || self.counts != other.counts).into_py(py),
            _ => py.NotImplemented(),
        }
    }
}

#[pymethods]
impl CocoRle {
    #[new]
    fn new(size: Vec<u32>, counts: String) -> Self {
        Self { size, counts }
    }

    fn __repr__(&self) -> String {
        format!("COCO_RLE(size={:?}, counts={:?})", self.size, self.counts)
    }

    fn __richcmp__(&self, other: &Self, op: CompareOp, py: Python<'_>) -> PyObject {
        match op {
            CompareOp::Eq => (self.size == other.size && self.counts == other.counts).into_py(py),
            CompareOp::Ne => (self.size != other.size || self.counts != other.counts).into_py(py),
            _ => py.NotImplemented(),
        }
    }
}

#[pymethods]
impl PolygonsRS {
    #[new]
    fn new(size: Vec<u32>, counts: Vec<Vec<f64>>) -> Self {
        Self { size, counts }
    }

    fn __repr__(&self) -> String {
        format!("PolygonsRS(size={:?}, counts={:?})", self.size, self.counts)
    }

    fn __richcmp__(&self, other: &Self, op: CompareOp, py: Python<'_>) -> PyObject {
        match op {
            // FIXME: `counts` are floats, don't compare them with "==".
            CompareOp::Eq => (self.size == other.size && self.counts == other.counts).into_py(py),
            CompareOp::Ne => (self.size != other.size || self.counts != other.counts).into_py(py),
            _ => py.NotImplemented(),
        }
    }
}

impl Segmentation {
    fn __repr__(&self) -> String {
        match self {
            Segmentation::Rle(rle) => rle.__repr__(),
            Segmentation::CocoRle(coco_rle) => coco_rle.__repr__(),
            Segmentation::Polygons(poly) => format!("Polygons(counts={:?})", poly),
            Segmentation::PolygonsRS(poly) => poly.__repr__(),
        }
    }
}

impl IntoPy<PyObject> for Segmentation {
    fn into_py(self, py: Python<'_>) -> PyObject {
        match self {
            Segmentation::Rle(rle) => rle.into_py(py),
            Segmentation::CocoRle(coco_rle) => coco_rle.into_py(py),
            Segmentation::Polygons(poly) => poly.into_py(py),
            Segmentation::PolygonsRS(poly) => poly.into_py(py),
        }
    }
}