/Object_Detection_Synthetic_Data

The project focuses on analyzing the training of object detection models using various types of datasets, including real data, augmented data, synthetic data, and hybrid data.

Primary LanguagePythonMIT LicenseMIT

Analysis of Training Object Detection Models using Different Types of Datasets

Requirements

Software:

Blender, Visual Studio Code with Python version >3.10 installed

Supported on the following operating systems:

Linux (recommended), macOS, Windows

Python Libraries:

See the "Installation Dependencies" section in the Readme file.

Project Description

The project focuses on analyzing the training of object detection models using various types of datasets, including real data, augmented data, synthetic data, and hybrid data. Consequently, we compare these approaches to train the model and determine the most effective method for object detection in real-world applications.

The experiment was conducted in a robotics lab called Industry 4.0 Cell (I4C), located at Brno University of Technology. The main hardware used for the experiment was a PhoXi 3D Scanner M. Since the project primarily focuses on object detection in 2D space, we utilized only the 2D camera of the scanner in our experiment. For future expansions of the project, I can also incorporate the third dimension to fully leverage the scanner's potential.

The experiment aims to detect two types of objects (see figure):

  • The T-joint object made of aluminum material. Figure left.
  • The Metal-Blank object made of aluminum material. Figure right.

To determine the most effective object detection method, we created six types of datasets and trained them using the YOLOv8 model (Ultralytics HUB).

Type of the Dataset Short Description
Type-0 A smaller dataset that contains real monochrome images from the Photoneo scanner.
Type-1 Augmented real monochrome images used to increase the size of the Type-0 dataset.
Type-2 A smaller dataset that contains synthetic monochrome images from the Blender camera.
Type-3 Augmented synthetic monochrome images used to increase the size of the Type-2 dataset.
Type-4 A larger dataset that contains synthetic monochrome images from the Blender camera.
Type-5 A hybrid dataset that contains synthetic monochrome images from the Blender camera, as well as real monochrome images from the Photoneo scanner.

The full dataset can be found here in a shared folder on Google Drive:

Data.zip

and the training results, models etc. can also be found here on Google Drive:

YOLO.zip

For information on object detection using the YOLOv8 model, please visit the Ultralytics YOLOv8 website.

More detailed information about the data acquisition process, generating synthetic data using Blender, data augmentation, training the object detection model on a custom dataset, and last but not least, the evaluation of the experiment can be found in the individual sections of the README file below.

The project was realized at the Institute of Automation and Computer Science, Brno University of Technology, Faculty of Mechanical Engineering (NETME Centre - Cybernetics and Robotics Division).

Project Hierarchy

../Object_Detection_Synthetic_Data/CAD/

3D models of individual objects.

../Object_Detection_Synthetic_Data/Blender/

Blender environment for access to synthetic data generation, robot workplace visualization, etc.

See the programmes below for more information.

$ ../src/Blender> ls
gen_object_bounding_box.py       gen_synthetic_data.py    save_image.py
gen_synthetic_background_data.py get_object_boundaries.py

../Object_Detection_Synthetic_Data/src/

The source code of the project with additional dependencies. The individual scripts for data collection, training, augmentation, validation of the results, etc. contain additional information.

The project also contains a Google Colab document that can be used to train the model in Collaboratory.

$ ../src/Training/Google_Colab> ls
YOLOv8_Train_Custom_Dataset.ipynb

../Object_Detection_Synthetic_Data/Template/

The Template section contains the required folders for the training and validation process. Both folders are explained below. To run the project smoothly, it is necessary to follow the structure of the individual folders.

Data

  • The Data folder contains the data from the camera, the individual datasets and the results for each model (PyTorch, ONNX, etc.).

YOLO

  • The YOLO folder contains the configuration file for training, the results of the training process and the models to be saved after training.

Data Acquisition from the PhoXi 3D Scanner M

The Harvester Python library was used for image acquisition. Harvester is a library for Python that aims to ease the process of image acquisition in computer vision applications.

https://github.com/genicam/harvesters

Python Support for Photoneo 3D Sensors using GenICam

https://www.photoneo.com/support/

The requirements to run the Python example with GenICam are:

- Python 3.*
- PhoXi Control 1.8 or higher

Examples located at (Windows):

C:\Program Files\Photoneo\PhoXiControl-x.x.x\API\examples\GenTL\python

Python Dependencies (packages)

NumPy
$ ../user_name> pip3 install numpy

Open3D
$ ../user_name> pip3 install open3d
Note:  Only if 3D data processing is used.

harvesters
$ ../user_name> pip3 install harvesters

The program for acquiring the raw 2D image from the Photoneo sensor can be found below. This program is compatible with various types of Photoneo scanners, including XS, S, M, L, and more.

$ ../src/Camera/Collection> python scan.py 

After scanning the environment, we obtain the raw image without adjusting the contrast and brightness. To adjust the contrast 'alpha' and brightness 'beta' of each image, we utilize the histogram clip function.

A program to adjust the contrast 'alpha' and brightness 'beta' of the raw image can be found below:

$ ../src/Camera> python image_processing.py 

The results of the adjustment can be saved to the images folder using the program bellow:

$ ../src/Evaluation/Camera_Data> python save_histogram.py 

Synthetic Data Generation

Generating synthetic data from Blender is a process that includes creating an environment similar to the real one and rendering it to make it more realistic. The data generation process only applies to the 2D field of view, as mentioned earlier. The process of generating synthetic images includes the function to randomly generate the parameters of the camera, as well as the object. In the case of the camera, the random function includes additional noise and illumination, while for the object, it involves positioning and rotation within certain limits.

Synthetic data consists of an image and a corresponding label, which is defined by a bounding box around the scanned object.

The main part to find the corresponging label of the object was to solve the equation of projection matrix 'P':

    P = K x [R | t],

where 'K' is the instrict matrix that contains the intrinsic parameters, and '[R | t]' is the extrinsic matrix that is combination of rotation matrix 'R' and a translation vector 't'.

The process of generating synthetic images takes approximately 17.22s per image. The time depends on the available GPU, along with the settings in Blender.

Information about the process of synthetic data generation

The files related to Blender can be found within the folder, see bellow.
$ ../Blender> ls
Gen_Synthetic_Data.blend Object_Bounding_Box.blend Robot_Environment_View.blend

Related Python programs used to work with Blender files can be found in the folder below.
$ ../src/Blender> ls
gen_object_bounding_box.py       gen_synthetic_data.py    save_image.py
gen_synthetic_background_data.py get_object_boundaries.py

See the individual programs with the *.py extension for more information.

Data Augmentation

Generating (augmenting) data from a small image dataset is a simple process that involves augmenting both the image and label data. The process of generating augmented images takes approximately 0.185 seconds per image.

We use the "Albumentations" library to generate augmented data. Information about the library (transformation functions, etc.) can be found here:

[http://albumentations.ai](http://albumentations.ai)

The transformation declaration used to augment the image/bounding box

transformation = A.Compose([A.Affine(translate_px={'x': (-10, 10), 'y': (-10, 10)}, p = 0.75),
                            A.ColorJitter(brightness=(0.25, 1.5), contrast=(0.25, 1.5), saturation=(0.1, 1.0), 
                                            always_apply=True),
                            A.GaussianBlur(blur_limit=(5, 5), sigma_limit=(0.01, 1.0), p = 0.5),
                            A.RandomResizedCrop(height= 1544, width = 2064, scale = (0.95, 1.0), p = 0.5)], 
                            bbox_params=A.BboxParams(format='yolo', label_fields=['class_labels']))

Information about the process of augmented data generation

Augmented data can be easily generated using a Python program, see below. More information about the generation parameters can be found in the program. 
$ ../src/Augmentation> python generate.py

Train YOLOv8 Object Detection on a Custom Dataset

The YOLOv8n (nano) model was used to train the object detection experiment. To quickly retrain the model on the new data without retraining the entire network, we use a function to freeze the backbone layers of the model. The backbone of the model consists of layers 0-9 (10 layers). The training process was performed using an internal NVIDIA GeForce GTX 1080 Ti 16 GB GPU, , but it is also possible to use Google Colab, as I mentioned earlier.

Information about the training process

The 'config.yaml' file can be found here:
../YOLO/Configuration/Type_{dataset_type}/config.yaml
Note:
  dataset_type: The identification number of the dataset type.

Training the YOLOv8 model on a custom dataset. Just change the constant 'CONST_DATASET_TYPE' in the program parameters.
$ ../src/Training/Internal> python train.py

Programs useful for evaluation of the training process can be found here:
$ ../src/Evaluation/Model/Results> python save_train_results.py
$ ../src/Evaluation/Model/Results> python save_train_comparison.py

Configuration file (config.yaml) for the Type-0 dataset

# Description:
#   The contents of the configuration data, such as the specified partition path, the total 
#   number of classes in the dataset, and the name of each class.

# The path to the main directory of the dataset.
#   Name of the dataset:
#     Dataset_Type_{dataset_type}
#   Note:
#     dataset_type: 
#       The identification number of the dataset type.
#   Warning:
#     path(Google Colab): '/content/drive/MyDrive/YOLOv8_Custom_Dataset/Data/Dataset_Type_{dataset_type}'
#     path(Internal): '../../../../Data/Dataset_Type_{dataset_type}'
path: '../../../../Data/Dataset_Type_0'

# The absolute path to the specified partition folder (train, validation, test) that contains 
# the dataset images as well as the labels.
train: images/train
val: images/valid
test: images/test

# Parameters:
#   The total number of classes in the dataset.
nc: 2 
#   ID name of each class.
names: ['T_Joint', 'Metal_Blank']

Training parameters of the YOLOv8 object detection model

Model: YOLOv8n (Nano)
Image Size: 640x480 (rectangular training with each batch collated for minimum padding).
Maximum Number of Epochs: 300
Patience: 0
Batch Size: Automatically estimate the best YOLO batch size to use a fraction of the available CUDA memory.

The partitioning of image data within datasets

Partitioning of the Dataset:
    - Train: 80% + Background Images.
    - Valid: 20%
    - Test: 24 images from real hardware (camera) and 1 synthetic image.

Type of images:
    R: Real
    A: Augmented
    S: Synthetic

Type of objects:
    ID_0: T-Joint (polypropylene)
    ID_1: Metal-Blank (aluminium)
    ID_B: Background

Type_0:
    Size: 0.134 GB
    Number of images:
        - Train: 
            - ID_0: 12 = 12(R) + 0(A) + 0(S)
            - ID_1: 12 = 12(R) + 0(A) + 0(S)
            - ID_B:  2 =  2(R) + 0(A) + 0(S)
            - N: 24(ID_{0,1}) + 2(B)
        - Valid: 
            - ID_0: 3 = 3(R) + 0(A) + 0(S)
            - ID_1: 3 = 3(R) + 0(A) + 0(S)
            - N: 6(ID_{0,1})
        - Test: 
            - N: 24(R) + 0(A) + 1(S)
    Freeze Backbone: True
    Epochs: 300 
    Training Time: 0.038 hours

Type_1:
    Size: 1.800 GB
    Number of images:
        - Train: 
            - ID_0: 240 = 12(R) + 228(A) + 0(S)
            - ID_1: 240 = 12(R) + 228(A) + 0(S)
            - ID_B:  20 =  2(R) +  18(A) + 0(S)
            - N: 480(ID_{0,1}) + 20(B)
        - Valid: 
            - ID_0: 60 = 3(R) + 57(A) + 0(S)
            - ID_1: 60 = 3(R) + 57(A) + 0(S)
            - N: 120(ID_{0,1})
        - Test: 
            - N: 24(R) + 0(A) + 1(S)
    Freeze Backbone: True
    Epochs: 300 
    Training Time: 0.357 hours

Type_2:
    Size: 0.128 GB
    Number of images:
        - Train: 
            - ID_0: 12 =  0(R) + 0(A) + 12(S)
            - ID_1: 12 =  0(R) + 0(A) + 12(S)
            - ID_B:  2 =  0(R) + 0(A) +  2(S)
            - N: 24(ID_{0,1}) + 2(B)
        - Valid: 
            - ID_0: 3 = 0(R) + 0(A) + 3(S)
            - ID_1: 3 = 0(R) + 0(A) + 3(S)
            - N: 6(ID_{0,1})
        - Test: 
            - N: 24(R) + 0(A) + 1(S)
    Freeze Backbone: True
    Epochs: 300 
    Training Time: 0.037 hours

Type_3:
    Size: 2.500 GB
    Number of images:
        - Train: 
            - ID_0: 240 =  0(R) + 228(A) + 12(S)
            - ID_1: 240 =  0(R) + 228(A) + 12(S)
            - ID_B:  20 =  0(R) +  18(A) +  2(S)
            - N: 480(ID_{0,1}) + 20(B)
        - Valid: 
            - ID_0: 60 = 0(R) + 57(A) + 3(S)
            - ID_1: 60 = 0(R) + 57(A) + 3(S)
            - N: 120(ID_{0,1})
        - Test: 
            - N: 24(R) + 0(A) + 1(S)
    Freeze Backbone: True
    Epochs: 300 
    Training Time: 0.380 hours

Type_4:
    Size: 1.400 GB
    Number of images:
        - Train: 
            - ID_0: 240 =  0(R) + 0(A) + 240(S)
            - ID_1: 240 =  0(R) + 0(A) + 240(S)
            - ID_B:  20 =  0(R) + 0(A) +  20(S)
            - N: 480(ID_{0,1}) + 20(B)
        - Valid: 
            - ID_0: 60 = 0(R) + 0(A) + 60(S)
            - ID_1: 60 = 0(R) + 0(A) + 60(S)
            - N: 120(ID_{0,1})
        - Test: 
            - N: 24(R) + 0(A) + 1(S)
    Freeze Backbone: True
    Epochs: 300 
    Training Time: 0.301 hours

Type_5:
    Size: 1.400 GB
    Number of images:
        - Train: 
            - ID_0: 240 = 12(R) + 0(A) + 228(S)
            - ID_1: 240 = 12(R) + 0(A) + 228(S)
            - ID_B:  20 =  2(R) + 0(A) + 18(S)
            - N: 480(ID_{0,1}) + 20(B)
        - Valid: 
            - ID_0: 60 = 3(R) + 0(A) + 57(S)
            - ID_1: 60 = 3(R) + 0(A) + 57(S)
            - N: 120(ID_{0,1})
        - Test: 
            - N: 24(R) + 0(A) + 1(S)
    Freeze Backbone: True
    Epochs: 300 
    Training Time: 0.300 hours

The results of the object detection model training process

Comparison of training results

Type of the Dataset GIoU: Train GIoU: Valid Objectness: Train Objectness: Valid Classification: Train Classification: Valid Precision Recall mAP@0.5 mAP@0.5:0.95
Type-0 0.2558 0.3898 0.3012 0.3412 0.7736 0.7618 0.9852 1.0000 0.9950 0.9665
Type-1 0.1823 0.3379 0.1525 0.2144 0.7558 0.7706 0.9952 1.0000 0.9950 0.9596
Type-2 0.3291 0.4206 0.3748 0.3191 0.7643 0.8098 0.9796 1.0000 0.9950 0.9658
Type-3 0.2254 0.3305 0.1790 0.2019 0.7871 0.7923 0.9991 1.0000 0.9950 0.9695
Type-4 0.2039 0.1957 0.1595 0.1675 0.7758 0.7436 0.9991 1.0000 0.9950 0.9933
Type-5 0.1974 0.2079 0.1619 0.1712 0.7662 0.7404 0.9991 1.0000 0.9950 0.9930

Evaluation of the Experiment

Information about the validation and prediction process

Validation of the custom-trained YOLOv8 model on a test partition of an individual dataset. Just change the constant 'CONST_DATASET_TYPE' in the program parameters.
$ ../src/Evaluation/Model/Results> python save_valid_results.py
$ ../src/Evaluation/Model/Results> python save_valid_comparison.py

Prediction of the custom-trained YOLOv8 model on a test partition of an individual dataset. Just change the constant 'CONST_DATASET_TYPE' in the program parameters.
$ ../src/Evaluation/Model> python predict.py

Comparison of prediction results on the test dataset

Type of the Dataset Precision Recall mAP@0.5 mAP@0.5:0.95
Type-0 0.9738 0.9908 0.9949 0.8806
Type-1 0.9957 1.0000 0.9950 0.9042
Type-2 0.9499 0.9757 0.9921 0.8326
Type-3 0.9762 0.9899 0.9950 0.8578
Type-4 0.9829 0.9959 0.9950 0.8696
Type-5 0.9980 1.0000 0.9950 0.9163

As we can see in the table above, the best predictions were found from the object detection model that was trained with the Type-5 dataset. We can, therefore, conclude that the combination of real data and synthetic data is the most effective method for object detection in real-world applications. The prediction results for the entire test partition can be found below.

Validation: : Conf(0.001), IoU(0.6)

Prediction: Conf(0.5), IoU(0.7)

Installation Dependencies

It will be useful for the project to create a virtual environment using Conda. Conda is an open source package management system and environment management system that runs on Windows, macOS, and Linux. Conda quickly installs, runs and updates packages and their dependencies.

Set up a new virtual environment called {name} with python {version}

$ ../user_name> conda create -n {name} python={version}
$ ../user_name> conda activate {name}

Installation of packages needed for the project

OpenCV
$ ../user_name> conda install -c conda-forge opencv

Matplotlib
$ ../user_name> conda install -c conda-forge matplotlib

SciencePlots
$ ../user_name> conda install -c conda-forge scienceplots

Pandas
$ ../user_name> conda install -c conda-forge pandas

Albumentations
$ ../user_name> conda install -c conda-forge albumentations

PyTorch, Torchvision, etc.
$ ../user_name> conda install pytorch::pytorch torchvision torchaudio -c pytorch
or 
$ ../user_name> conda install pytorch-nightly::pytorch torchvision torchaudio -c pytorch-nightly

Installation of packages that are not available using conda-forge

After conda env. will be activated, install pip.
$ ../user_name> conda activate {name}
$ ../user_name> conda install -c conda-forge pip

Locate the bin folder in the directory of the activated conda environment.
$ ../user_name> cd {directory_path}/bin

Installation of new packages such as Ultralytics.
$ ../user_name> pip install ultralytics

Other useful commands for working with the Conda environment

Deactivate environment.
$ ../user_name> conda deactivate

Remove environment.
$ ../user_name> conda remove --name {name} --all

To verify that the environment was removed, in your terminal window or an Anaconda Prompt, run.
$ ../user_name> conda info --envs

Rename the environment from the old name to the new one.
$ ../user_name> conda rename -n {old_name} {name_name}

Video

Contact Info:

Roman.Parak@outlook.com

Citation (BibTex)

@misc{RomanParak_BlenderSynth,
  author = {Roman Parak},
  title = {Analysis of training object detection models using different types of datasets},
  year = {2023-2023},
  publisher = {GitHub},
  journal = {GitHub repository},
  howpublished = {\url{https://github.com/rparak/Object_Detection_Synthetic_Data}}
}

License

MIT