bpm-ocr 0.2.0

A library for attempting to extract a blood pressure monitor reading from an image using opencv.
Documentation
use std::rc::Rc;

use opencv::core::{Mat, Size, Vector};
use opencv::imgcodecs::ImreadModes;
use opencv::{imgcodecs, imgproc};

use crate::debug::BpmOcrDebugOutputter;
use crate::lcd_number_extractor::LcdNumberExtractor;
use crate::lcd_screen_extractor::LcdScreenExtractor;
use crate::models::{BloodPressureReading, ProcessingError};
pub mod debug;
mod digit_extractor;
mod lcd_number_extractor;
mod lcd_screen_extractor;
pub mod models;
mod rectangle;

pub struct BloodPressureReadingExtractor<T: BpmOcrDebugOutputter> {
    screen_extractor: LcdScreenExtractor<T>,
    screen_number_extractor: LcdNumberExtractor<T>,
    debugger: Rc<T>
}

impl<T: BpmOcrDebugOutputter> BloodPressureReadingExtractor<T> {
    pub fn new(debugger: T) -> Self {
        let shared_debugger = Rc::new(debugger);

        let screen_extractor = LcdScreenExtractor::new(&shared_debugger);
        let screen_number_extractor = LcdNumberExtractor::new(&shared_debugger);

        BloodPressureReadingExtractor {
            screen_extractor,
            screen_number_extractor,
            debugger: shared_debugger
        }
    }

    fn process_image(self: &Self, image: &Mat) -> Result<BloodPressureReading, ProcessingError> {
        self.debugger.debug_original_picture(&image)?;

        let mut resized_image = Mat::default();

        let interpolation: i32 = 0;
        imgproc::resize(
            &image,
            &mut resized_image,
            Size::new(800, 800),
            0.,
            0.,
            interpolation,
        )?;

        let birdseye_lcd_only = self.screen_extractor.extract_lcd(&resized_image)?;

        let reading = self
            .screen_number_extractor
            .extract_reading(&birdseye_lcd_only)?;

        Ok(reading)
    }

    /// Attempts to extract a blood pressure reading from a photo file of a blood pressure monitor screen
    /// * `filename` - the path to the photo file
    pub fn get_reading_from_file(
        self: &Self,
        filename: &str,
    ) -> Result<BloodPressureReading, ProcessingError> {
        let gray_scale_mode: i32 = ImreadModes::IMREAD_GRAYSCALE.into();
        let image = imgcodecs::imread(filename, gray_scale_mode)?;

        self.process_image(&image)
    }

    /// Attempts to extract a blood pressure reading from a byte buffer containing a photo file of a blood pressure monitor screen
    /// * `filename` - the byte buffer with the photo file
    pub fn get_reading_from_buffer(
        self: &Self,
        file_contents: Vec<u8>,
    ) -> Result<BloodPressureReading, ProcessingError> {
        let contents = Vector::from_slice(&file_contents);
        let image = imgcodecs::imdecode(&contents, ImreadModes::IMREAD_GRAYSCALE.into())?;

        self.process_image(&image)
    }
}


#[cfg(test)]
mod tests {
    use crate::debug::UnsafeTempFolderDebugger;

    use super::*;

    #[test]
    fn test_success_photo_at_angle() {
        let debugger: UnsafeTempFolderDebugger = UnsafeTempFolderDebugger::new("test_success", true);
        let testfile = Vec::from(include_bytes!("./test_resources/example_at_angle.jpg"));
        let extractor = BloodPressureReadingExtractor::new(debugger);

        let expected_result = BloodPressureReading {
            systolic: 133,
            diastolic: 93,
            pulse: 65
        };

        let result = extractor.get_reading_from_buffer(testfile).unwrap();

        assert_eq!(result, expected_result);
    }

    #[test]
    fn test_success_topdown_photo() {
        let debugger: UnsafeTempFolderDebugger = UnsafeTempFolderDebugger::new("test_topdown_photo", true);
        let testfile = Vec::from(include_bytes!("./test_resources/example_top_down.jpg"));
        let extractor = BloodPressureReadingExtractor::new(debugger);

        let expected_result = BloodPressureReading {
            systolic: 131,
            diastolic: 88,
            pulse: 77
        };

        let result = extractor.get_reading_from_buffer(testfile).unwrap();

        assert_eq!(result, expected_result);
    }

    #[test]
    fn test_with_2_digit() {
        let debugger: UnsafeTempFolderDebugger = UnsafeTempFolderDebugger::new("test_with_2_digit", true);
        let testfile = Vec::from(include_bytes!("./test_resources/contour_candidates.jpeg"));
        let extractor = BloodPressureReadingExtractor::new(debugger);

        let expected_result = BloodPressureReading {
            systolic: 123,
            diastolic: 85,
            pulse: 68
        };

        let result = extractor.get_reading_from_buffer(testfile).unwrap();

        assert_eq!(result, expected_result);
    }

}