Back to Publications

ExpirioBot

Table of contents


The ExpirioBot

Dofbot Pi

Overview

ExpirioBot is an automated robotic system designed to identify and sort products based on their expiry dates. The system utilizes computer vision to capture images of products, extract expiry dates using Optical Character Recognition (OCR), and control a robotic arm to move expired or valid products to designated locations.


Features

  • OCR-Based Expiry Date Detection: Extract expiry dates from product labels using Tesseract OCR.
  • Robotic Arm Sorting: A robotic arm sorts products into expired and valid categories.
  • Real-Time Video Feed: Displays a live feed of the camera input
  • Counters for Products: Tracks the number of expired and valid products in real-time.
  • Threaded Architecture: Ensures smooth operation with concurrent frame capturing and processing.

Requirements

Hardware

  • DOFBOT-Pi robotic arm
  • Processing Computer (Rasperry Pi 4 or equivalent recommended)
  • Camera (USB or Pi Camera)

Software

  • Python 3
  • OpenCV (cv2)
  • Pillow (PIL)
  • Pytesseract
  • tkinter
  • threading
  • queue
  • re
  • time
  • datetime
  • Arm_Lib (custom library for controlling the robotic arm)
  • Tesseract OCR installed on your system

Installation

  1. Clone the repository:

    bash code

    git clone https://github.com/EdwinAbdonShayo/ExpirioBot.git
    
    cd ExpirioBot
    
  2. Install the required Python packages:

    bash code

    pip install opencv-python pillow pytesseract tkinter
    
  3. Install Tesseract OCR:

    • For Linux, use your package manager:

      bash code

      sudo apt-get install tesseract-ocr
      
    • For Windows, download the installer from Tesseract at UB Mannheim.

    • For macOS, use Homebrew:

      bash code

      brew install tesseract
      
  4. Installing Arm Library: Ensure the py_install is in your project folder:

    bash code

    cd py_install
    sudo python3 setup.py install
    

Repository Structure

  ExpirioBot/
  │
  ├── ExpirioBot.py          # Main script for image capture, processing, and robotic arm movement.
  │
  ├── py_install             # Custom library for robotic arm control.
  │   ├── Arm_Lib            # Folder with robotic arm library dependencies.
  │   └── setup.py           # Arm_Lib library setup file.
  │
  ├── ocr.py                 # Script with the optical character recognition (extracting text out of the images and gets date using patterns).
  │
  ├── objectMover.py         # Script for the robot arm movement.
  │
  ├── image.jpg              # Sample image captured from camera and preprocessed.
  │
  ├── Learning Curve/        # Experimental scripts for trials and testing.
  │   ├── Main
  │   │   ├── ExpirioBot1.py    # Experimental script 1.
  │   │   ├── ExpirioBot2.py    # Experimental script 2.
  │   │   ├── ...               # Additional experimental scripts.
  │   │
  │   ├── ...                # Additional experimental scripts.
  │
  └── README.md              # This README file.

Usage

  1. Connect Hardware:
    • Attach the camera (in this case, the Raspberry Pi) and ensure it is accessible.
    • Connect the DOFBOT robotic arm to the System.
  2. Run the Program:

    bash code

    python3 ExpirioBot.py
    
  3. Control Through the GUI:
  • Start: Begin capturing and processing frames.
  • Stop: Pause the system.
  • View live video feed and counters for expired and valid products.

Key Code Snippets

  1. Arm Control Functions

def arm_clamp_block(enable):
    """
    Clamp or release the robotic arm.
    enable: 1 to clamp, 0 to release.
    """
    position = 100 if enable else 10
    Arm.Arm_serial_servo_write(6, position, 400)
    time.sleep(0.5)

def arm_move(positions, s_time=500):
    """
    Moves the arm to specified positions.
    positions: List of servo positions.
    s_time: Time duration for movement.
    """
    for i, pos in enumerate(positions):
        servo_id = i + 1
        Arm.Arm_serial_servo_write(servo_id, pos, s_time)
        time.sleep(0.01)
    time.sleep(s_time / 1000)

  1. Image Preprocessing

def preprocess_image(frame):
    """
    Convert frame to grayscale and apply binary thresholding.
    """
    gray = cv2.cvtColor(frame, cv2.COLOR_BGR2GRAY)
    _, binary = cv2.threshold(gray, 128, 255, cv2.THRESH_BINARY)
    return binary

  1. Extracting Expiry Dates with OCR

def extract_expiry_date(image_path):
    """
    Extract expiry date from an image using OCR.
    image_path: Path to the image file.
    Returns: Extracted expiry date in DD/MM/YYYY format.
    """
    image = Image.open(image_path)
    text = pytesseract.image_to_string(image)
    pattern = r'\b\d{2}[./]\d{2}[./]\d{4}\b'
    match = re.search(pattern, text)
    return match.group(0) if match else None

  1. Threading and Queues for Real-Time Processing

def capture_frames(cap, frame_queue_container, producer_allowed_event):
    """
    Continuously captures frames from the camera and adds them to a queue.
    Uses threading to ensure real-time video feed and reduced stress on the Raspberry Pi.
    """
    while True:
        producer_allowed_event.wait()
        ret, frame = cap.read()
        if not ret:
            print("Failed to grab frame")
            break

        with queue_lock:
            current_queue = frame_queue_container[0]
            if current_queue.full():
                try:
                    current_queue.get_nowait()
                except queue.Empty:
                    pass
            current_queue.put(frame)
        time.sleep(0.1)

def process_frames(frame_queue_container, processing_event, producer_allowed_event):
    """
    Processes frames from the queue and handles OCR and robotic arm control.
    Uses threading to offload work from the main capture loop.
    """
    while True:
        processing_event.wait()
        with queue_lock:
            current_queue = frame_queue_container[0]

        if not current_queue.empty():
            frame = current_queue.get()
            image_path = "image.jpg"
            processed_frame = preprocess_image(frame)
            cv2.imwrite(image_path, processed_frame)
            expiry_date = extract_expiry_date(image_path)
            if expiry_date:
                # Process expiry date and control the arm...
                pass


Customization

  1. Modify Predefined Arm Positions:
    • Update positions in the script (p_front, p_left, etc.) to match your setup.
  2. Change Thresholds for Image Preprocessing:
    • Edit the preprocess_image function to adjust grayscale or binary thresholds.
  3. Extend OCR Patterns:
    • Modify the extract_expiry_date function to handle additional date formats.

Troubleshooting

  • Camera Not Detected:
    • Ensure the camera is connected and accessible through OpenCV.
  • Robotic Arm Not Responding:
    • Verify the arm is powered, correctly configured & arm library is installed.
  • OCR Not Extracting Dates:
    • Check the Tesseract installation and ensure the image has clear, legible text.

Acknowledgments

  • OpenCV for image processing.
  • Tesseract OCR for text extraction.
  • DOFBOT for providing a reliable robotic arm solution.
  • tkinter for GUI development.

Models

Datasets

There are no datasets linked

Files