Add video and live stream processing and tests for Image Classifier C API
PiperOrigin-RevId: 578242391
This commit is contained in:
		
							parent
							
								
									7da2810b83
								
							
						
					
					
						commit
						a4048eee11
					
				| 
						 | 
					@ -30,13 +30,12 @@ cc_library(
 | 
				
			||||||
        "//mediapipe/tasks/c/components/processors:classifier_options_converter",
 | 
					        "//mediapipe/tasks/c/components/processors:classifier_options_converter",
 | 
				
			||||||
        "//mediapipe/tasks/c/core:base_options",
 | 
					        "//mediapipe/tasks/c/core:base_options",
 | 
				
			||||||
        "//mediapipe/tasks/c/core:base_options_converter",
 | 
					        "//mediapipe/tasks/c/core:base_options_converter",
 | 
				
			||||||
 | 
					        "//mediapipe/tasks/cc/vision/core:running_mode",
 | 
				
			||||||
        "//mediapipe/tasks/cc/vision/image_classifier",
 | 
					        "//mediapipe/tasks/cc/vision/image_classifier",
 | 
				
			||||||
        "//mediapipe/tasks/cc/vision/utils:image_utils",
 | 
					        "//mediapipe/tasks/cc/vision/utils:image_utils",
 | 
				
			||||||
        "@com_google_absl//absl/log:absl_log",
 | 
					        "@com_google_absl//absl/log:absl_log",
 | 
				
			||||||
        "@com_google_absl//absl/status",
 | 
					        "@com_google_absl//absl/status",
 | 
				
			||||||
        "@com_google_absl//absl/status:statusor",
 | 
					        "@com_google_absl//absl/status:statusor",
 | 
				
			||||||
        "@com_google_absl//absl/strings:str_format",
 | 
					 | 
				
			||||||
        "@com_google_absl//absl/time",
 | 
					 | 
				
			||||||
    ],
 | 
					    ],
 | 
				
			||||||
    alwayslink = 1,
 | 
					    alwayslink = 1,
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -15,6 +15,8 @@ limitations under the License.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "mediapipe/tasks/c/vision/image_classifier/image_classifier.h"
 | 
					#include "mediapipe/tasks/c/vision/image_classifier/image_classifier.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <cstdint>
 | 
				
			||||||
 | 
					#include <cstdlib>
 | 
				
			||||||
#include <memory>
 | 
					#include <memory>
 | 
				
			||||||
#include <utility>
 | 
					#include <utility>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -26,6 +28,7 @@ limitations under the License.
 | 
				
			||||||
#include "mediapipe/tasks/c/components/containers/classification_result_converter.h"
 | 
					#include "mediapipe/tasks/c/components/containers/classification_result_converter.h"
 | 
				
			||||||
#include "mediapipe/tasks/c/components/processors/classifier_options_converter.h"
 | 
					#include "mediapipe/tasks/c/components/processors/classifier_options_converter.h"
 | 
				
			||||||
#include "mediapipe/tasks/c/core/base_options_converter.h"
 | 
					#include "mediapipe/tasks/c/core/base_options_converter.h"
 | 
				
			||||||
 | 
					#include "mediapipe/tasks/cc/vision/core/running_mode.h"
 | 
				
			||||||
#include "mediapipe/tasks/cc/vision/image_classifier/image_classifier.h"
 | 
					#include "mediapipe/tasks/cc/vision/image_classifier/image_classifier.h"
 | 
				
			||||||
#include "mediapipe/tasks/cc/vision/utils/image_utils.h"
 | 
					#include "mediapipe/tasks/cc/vision/utils/image_utils.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -41,7 +44,10 @@ using ::mediapipe::tasks::c::components::processors::
 | 
				
			||||||
    CppConvertToClassifierOptions;
 | 
					    CppConvertToClassifierOptions;
 | 
				
			||||||
using ::mediapipe::tasks::c::core::CppConvertToBaseOptions;
 | 
					using ::mediapipe::tasks::c::core::CppConvertToBaseOptions;
 | 
				
			||||||
using ::mediapipe::tasks::vision::CreateImageFromBuffer;
 | 
					using ::mediapipe::tasks::vision::CreateImageFromBuffer;
 | 
				
			||||||
 | 
					using ::mediapipe::tasks::vision::core::RunningMode;
 | 
				
			||||||
using ::mediapipe::tasks::vision::image_classifier::ImageClassifier;
 | 
					using ::mediapipe::tasks::vision::image_classifier::ImageClassifier;
 | 
				
			||||||
 | 
					typedef ::mediapipe::tasks::vision::image_classifier::ImageClassifierResult
 | 
				
			||||||
 | 
					    CppImageClassifierResult;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
int CppProcessError(absl::Status status, char** error_msg) {
 | 
					int CppProcessError(absl::Status status, char** error_msg) {
 | 
				
			||||||
  if (error_msg) {
 | 
					  if (error_msg) {
 | 
				
			||||||
| 
						 | 
					@ -60,6 +66,53 @@ ImageClassifier* CppImageClassifierCreate(const ImageClassifierOptions& options,
 | 
				
			||||||
  CppConvertToBaseOptions(options.base_options, &cpp_options->base_options);
 | 
					  CppConvertToBaseOptions(options.base_options, &cpp_options->base_options);
 | 
				
			||||||
  CppConvertToClassifierOptions(options.classifier_options,
 | 
					  CppConvertToClassifierOptions(options.classifier_options,
 | 
				
			||||||
                                &cpp_options->classifier_options);
 | 
					                                &cpp_options->classifier_options);
 | 
				
			||||||
 | 
					  cpp_options->running_mode = static_cast<RunningMode>(options.running_mode);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // Enable callback for processing live stream data when the running mode is
 | 
				
			||||||
 | 
					  // set to RunningMode::LIVE_STREAM.
 | 
				
			||||||
 | 
					  if (cpp_options->running_mode == RunningMode::LIVE_STREAM) {
 | 
				
			||||||
 | 
					    if (options.result_callback == nullptr) {
 | 
				
			||||||
 | 
					      const absl::Status status = absl::InvalidArgumentError(
 | 
				
			||||||
 | 
					          "Provided null pointer to callback function.");
 | 
				
			||||||
 | 
					      ABSL_LOG(ERROR) << "Failed to create ImageClassifier: " << status;
 | 
				
			||||||
 | 
					      CppProcessError(status, error_msg);
 | 
				
			||||||
 | 
					      return nullptr;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ImageClassifierOptions::result_callback_fn result_callback =
 | 
				
			||||||
 | 
					        options.result_callback;
 | 
				
			||||||
 | 
					    cpp_options->result_callback =
 | 
				
			||||||
 | 
					        [result_callback](absl::StatusOr<CppImageClassifierResult> cpp_result,
 | 
				
			||||||
 | 
					                          const Image& image, int64_t timestamp) {
 | 
				
			||||||
 | 
					          char* error_msg = nullptr;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          if (!cpp_result.ok()) {
 | 
				
			||||||
 | 
					            ABSL_LOG(ERROR) << "Classification failed: " << cpp_result.status();
 | 
				
			||||||
 | 
					            CppProcessError(cpp_result.status(), &error_msg);
 | 
				
			||||||
 | 
					            result_callback(nullptr, MpImage(), timestamp, error_msg);
 | 
				
			||||||
 | 
					            free(error_msg);
 | 
				
			||||||
 | 
					            return;
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          // Result is valid for the lifetime of the callback function.
 | 
				
			||||||
 | 
					          ImageClassifierResult result;
 | 
				
			||||||
 | 
					          CppConvertToClassificationResult(*cpp_result, &result);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          const auto& image_frame = image.GetImageFrameSharedPtr();
 | 
				
			||||||
 | 
					          const MpImage mp_image = {
 | 
				
			||||||
 | 
					              .type = MpImage::IMAGE_FRAME,
 | 
				
			||||||
 | 
					              .image_frame = {
 | 
				
			||||||
 | 
					                  .format = static_cast<::ImageFormat>(image_frame->Format()),
 | 
				
			||||||
 | 
					                  .image_buffer = image_frame->PixelData(),
 | 
				
			||||||
 | 
					                  .width = image_frame->Width(),
 | 
				
			||||||
 | 
					                  .height = image_frame->Height()}};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          result_callback(&result, mp_image, timestamp,
 | 
				
			||||||
 | 
					                          /* error_msg= */ nullptr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          CppCloseClassificationResult(&result);
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  auto classifier = ImageClassifier::Create(std::move(cpp_options));
 | 
					  auto classifier = ImageClassifier::Create(std::move(cpp_options));
 | 
				
			||||||
  if (!classifier.ok()) {
 | 
					  if (!classifier.ok()) {
 | 
				
			||||||
| 
						 | 
					@ -75,8 +128,8 @@ int CppImageClassifierClassify(void* classifier, const MpImage* image,
 | 
				
			||||||
                               ImageClassifierResult* result,
 | 
					                               ImageClassifierResult* result,
 | 
				
			||||||
                               char** error_msg) {
 | 
					                               char** error_msg) {
 | 
				
			||||||
  if (image->type == MpImage::GPU_BUFFER) {
 | 
					  if (image->type == MpImage::GPU_BUFFER) {
 | 
				
			||||||
    absl::Status status =
 | 
					    const absl::Status status =
 | 
				
			||||||
        absl::InvalidArgumentError("gpu buffer not supported yet");
 | 
					        absl::InvalidArgumentError("GPU Buffer not supported yet.");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    ABSL_LOG(ERROR) << "Classification failed: " << status.message();
 | 
					    ABSL_LOG(ERROR) << "Classification failed: " << status.message();
 | 
				
			||||||
    return CppProcessError(status, error_msg);
 | 
					    return CppProcessError(status, error_msg);
 | 
				
			||||||
| 
						 | 
					@ -102,6 +155,68 @@ int CppImageClassifierClassify(void* classifier, const MpImage* image,
 | 
				
			||||||
  return 0;
 | 
					  return 0;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int CppImageClassifierClassifyForVideo(void* classifier, const MpImage* image,
 | 
				
			||||||
 | 
					                                       int64_t timestamp_ms,
 | 
				
			||||||
 | 
					                                       ImageClassifierResult* result,
 | 
				
			||||||
 | 
					                                       char** error_msg) {
 | 
				
			||||||
 | 
					  if (image->type == MpImage::GPU_BUFFER) {
 | 
				
			||||||
 | 
					    absl::Status status =
 | 
				
			||||||
 | 
					        absl::InvalidArgumentError("GPU Buffer not supported yet");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ABSL_LOG(ERROR) << "Classification failed: " << status.message();
 | 
				
			||||||
 | 
					    return CppProcessError(status, error_msg);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const auto img = CreateImageFromBuffer(
 | 
				
			||||||
 | 
					      static_cast<ImageFormat::Format>(image->image_frame.format),
 | 
				
			||||||
 | 
					      image->image_frame.image_buffer, image->image_frame.width,
 | 
				
			||||||
 | 
					      image->image_frame.height);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  if (!img.ok()) {
 | 
				
			||||||
 | 
					    ABSL_LOG(ERROR) << "Failed to create Image: " << img.status();
 | 
				
			||||||
 | 
					    return CppProcessError(img.status(), error_msg);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  auto cpp_classifier = static_cast<ImageClassifier*>(classifier);
 | 
				
			||||||
 | 
					  auto cpp_result = cpp_classifier->ClassifyForVideo(*img, timestamp_ms);
 | 
				
			||||||
 | 
					  if (!cpp_result.ok()) {
 | 
				
			||||||
 | 
					    ABSL_LOG(ERROR) << "Classification failed: " << cpp_result.status();
 | 
				
			||||||
 | 
					    return CppProcessError(cpp_result.status(), error_msg);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  CppConvertToClassificationResult(*cpp_result, result);
 | 
				
			||||||
 | 
					  return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int CppImageClassifierClassifyAsync(void* classifier, const MpImage* image,
 | 
				
			||||||
 | 
					                                    int64_t timestamp_ms, char** error_msg) {
 | 
				
			||||||
 | 
					  if (image->type == MpImage::GPU_BUFFER) {
 | 
				
			||||||
 | 
					    absl::Status status =
 | 
				
			||||||
 | 
					        absl::InvalidArgumentError("GPU Buffer not supported yet");
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    ABSL_LOG(ERROR) << "Classification failed: " << status.message();
 | 
				
			||||||
 | 
					    return CppProcessError(status, error_msg);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const auto img = CreateImageFromBuffer(
 | 
				
			||||||
 | 
					      static_cast<ImageFormat::Format>(image->image_frame.format),
 | 
				
			||||||
 | 
					      image->image_frame.image_buffer, image->image_frame.width,
 | 
				
			||||||
 | 
					      image->image_frame.height);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  if (!img.ok()) {
 | 
				
			||||||
 | 
					    ABSL_LOG(ERROR) << "Failed to create Image: " << img.status();
 | 
				
			||||||
 | 
					    return CppProcessError(img.status(), error_msg);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  auto cpp_classifier = static_cast<ImageClassifier*>(classifier);
 | 
				
			||||||
 | 
					  auto cpp_result = cpp_classifier->ClassifyAsync(*img, timestamp_ms);
 | 
				
			||||||
 | 
					  if (!cpp_result.ok()) {
 | 
				
			||||||
 | 
					    ABSL_LOG(ERROR) << "Data preparation for the image classification failed: "
 | 
				
			||||||
 | 
					                    << cpp_result;
 | 
				
			||||||
 | 
					    return CppProcessError(cpp_result, error_msg);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void CppImageClassifierCloseResult(ImageClassifierResult* result) {
 | 
					void CppImageClassifierCloseResult(ImageClassifierResult* result) {
 | 
				
			||||||
  CppCloseClassificationResult(result);
 | 
					  CppCloseClassificationResult(result);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -134,6 +249,22 @@ int image_classifier_classify_image(void* classifier, const MpImage* image,
 | 
				
			||||||
      CppImageClassifierClassify(classifier, image, result, error_msg);
 | 
					      CppImageClassifierClassify(classifier, image, result, error_msg);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int image_classifier_classify_for_video(void* classifier, const MpImage* image,
 | 
				
			||||||
 | 
					                                        int64_t timestamp_ms,
 | 
				
			||||||
 | 
					                                        ImageClassifierResult* result,
 | 
				
			||||||
 | 
					                                        char** error_msg) {
 | 
				
			||||||
 | 
					  return mediapipe::tasks::c::vision::image_classifier::
 | 
				
			||||||
 | 
					      CppImageClassifierClassifyForVideo(classifier, image, timestamp_ms,
 | 
				
			||||||
 | 
					                                         result, error_msg);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					int image_classifier_classify_async(void* classifier, const MpImage* image,
 | 
				
			||||||
 | 
					                                    int64_t timestamp_ms, char** error_msg) {
 | 
				
			||||||
 | 
					  return mediapipe::tasks::c::vision::image_classifier::
 | 
				
			||||||
 | 
					      CppImageClassifierClassifyAsync(classifier, image, timestamp_ms,
 | 
				
			||||||
 | 
					                                      error_msg);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void image_classifier_close_result(ImageClassifierResult* result) {
 | 
					void image_classifier_close_result(ImageClassifierResult* result) {
 | 
				
			||||||
  mediapipe::tasks::c::vision::image_classifier::CppImageClassifierCloseResult(
 | 
					  mediapipe::tasks::c::vision::image_classifier::CppImageClassifierCloseResult(
 | 
				
			||||||
      result);
 | 
					      result);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -92,9 +92,16 @@ struct ImageClassifierOptions {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // The user-defined result callback for processing live stream data.
 | 
					  // The user-defined result callback for processing live stream data.
 | 
				
			||||||
  // The result callback should only be specified when the running mode is set
 | 
					  // The result callback should only be specified when the running mode is set
 | 
				
			||||||
  // to RunningMode::LIVE_STREAM.
 | 
					  // to RunningMode::LIVE_STREAM. Arguments of the callback function include:
 | 
				
			||||||
  typedef void (*result_callback_fn)(ImageClassifierResult*, const MpImage*,
 | 
					  // the pointer to classification result, the image that result was obtained
 | 
				
			||||||
                                     int64_t);
 | 
					  // on, the timestamp relevant to classification results and pointer to error
 | 
				
			||||||
 | 
					  // message in case of any failure. The validity of the passed arguments is
 | 
				
			||||||
 | 
					  // true for the lifetime of the callback function.
 | 
				
			||||||
 | 
					  //
 | 
				
			||||||
 | 
					  // A caller is responsible for closing image classifier result.
 | 
				
			||||||
 | 
					  typedef void (*result_callback_fn)(ImageClassifierResult* result,
 | 
				
			||||||
 | 
					                                     const MpImage image, int64_t timestamp_ms,
 | 
				
			||||||
 | 
					                                     char* error_msg);
 | 
				
			||||||
  result_callback_fn result_callback;
 | 
					  result_callback_fn result_callback;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -110,13 +117,22 @@ MP_EXPORT void* image_classifier_create(struct ImageClassifierOptions* options,
 | 
				
			||||||
// If an error occurs, returns an error code and sets the error parameter to an
 | 
					// If an error occurs, returns an error code and sets the error parameter to an
 | 
				
			||||||
// an error message (if `error_msg` is not nullptr). You must free the memory
 | 
					// an error message (if `error_msg` is not nullptr). You must free the memory
 | 
				
			||||||
// allocated for the error message.
 | 
					// allocated for the error message.
 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// TODO: Add API for video and live stream processing.
 | 
					 | 
				
			||||||
MP_EXPORT int image_classifier_classify_image(void* classifier,
 | 
					MP_EXPORT int image_classifier_classify_image(void* classifier,
 | 
				
			||||||
                                              const MpImage* image,
 | 
					                                              const MpImage* image,
 | 
				
			||||||
                                              ImageClassifierResult* result,
 | 
					                                              ImageClassifierResult* result,
 | 
				
			||||||
                                              char** error_msg = nullptr);
 | 
					                                              char** error_msg = nullptr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					MP_EXPORT int image_classifier_classify_for_video(void* classifier,
 | 
				
			||||||
 | 
					                                                  const MpImage* image,
 | 
				
			||||||
 | 
					                                                  int64_t timestamp_ms,
 | 
				
			||||||
 | 
					                                                  ImageClassifierResult* result,
 | 
				
			||||||
 | 
					                                                  char** error_msg = nullptr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					MP_EXPORT int image_classifier_classify_async(void* classifier,
 | 
				
			||||||
 | 
					                                              const MpImage* image,
 | 
				
			||||||
 | 
					                                              int64_t timestamp_ms,
 | 
				
			||||||
 | 
					                                              char** error_msg = nullptr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Frees the memory allocated inside a ImageClassifierResult result.
 | 
					// Frees the memory allocated inside a ImageClassifierResult result.
 | 
				
			||||||
// Does not free the result pointer itself.
 | 
					// Does not free the result pointer itself.
 | 
				
			||||||
MP_EXPORT void image_classifier_close_result(ImageClassifierResult* result);
 | 
					MP_EXPORT void image_classifier_close_result(ImageClassifierResult* result);
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -15,6 +15,7 @@ limitations under the License.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "mediapipe/tasks/c/vision/image_classifier/image_classifier.h"
 | 
					#include "mediapipe/tasks/c/vision/image_classifier/image_classifier.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include <cstdint>
 | 
				
			||||||
#include <cstdlib>
 | 
					#include <cstdlib>
 | 
				
			||||||
#include <string>
 | 
					#include <string>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -36,12 +37,13 @@ using testing::HasSubstr;
 | 
				
			||||||
constexpr char kTestDataDirectory[] = "/mediapipe/tasks/testdata/vision/";
 | 
					constexpr char kTestDataDirectory[] = "/mediapipe/tasks/testdata/vision/";
 | 
				
			||||||
constexpr char kModelName[] = "mobilenet_v2_1.0_224.tflite";
 | 
					constexpr char kModelName[] = "mobilenet_v2_1.0_224.tflite";
 | 
				
			||||||
constexpr float kPrecision = 1e-4;
 | 
					constexpr float kPrecision = 1e-4;
 | 
				
			||||||
 | 
					constexpr int kIterations = 100;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
std::string GetFullPath(absl::string_view file_name) {
 | 
					std::string GetFullPath(absl::string_view file_name) {
 | 
				
			||||||
  return JoinPath("./", kTestDataDirectory, file_name);
 | 
					  return JoinPath("./", kTestDataDirectory, file_name);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
TEST(ImageClassifierTest, SmokeTest) {
 | 
					TEST(ImageClassifierTest, ImageModeTest) {
 | 
				
			||||||
  const auto image = DecodeImageFromFile(GetFullPath("burger.jpg"));
 | 
					  const auto image = DecodeImageFromFile(GetFullPath("burger.jpg"));
 | 
				
			||||||
  ASSERT_TRUE(image.ok());
 | 
					  ASSERT_TRUE(image.ok());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					@ -63,14 +65,13 @@ TEST(ImageClassifierTest, SmokeTest) {
 | 
				
			||||||
  void* classifier = image_classifier_create(&options);
 | 
					  void* classifier = image_classifier_create(&options);
 | 
				
			||||||
  EXPECT_NE(classifier, nullptr);
 | 
					  EXPECT_NE(classifier, nullptr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const auto& image_frame = image->GetImageFrameSharedPtr();
 | 
				
			||||||
  const MpImage mp_image = {
 | 
					  const MpImage mp_image = {
 | 
				
			||||||
      .type = MpImage::IMAGE_FRAME,
 | 
					      .type = MpImage::IMAGE_FRAME,
 | 
				
			||||||
      .image_frame = {
 | 
					      .image_frame = {.format = static_cast<ImageFormat>(image_frame->Format()),
 | 
				
			||||||
          .format = static_cast<ImageFormat>(
 | 
					                      .image_buffer = image_frame->PixelData(),
 | 
				
			||||||
              image->GetImageFrameSharedPtr()->Format()),
 | 
					                      .width = image_frame->Width(),
 | 
				
			||||||
          .image_buffer = image->GetImageFrameSharedPtr()->PixelData(),
 | 
					                      .height = image_frame->Height()}};
 | 
				
			||||||
          .width = image->GetImageFrameSharedPtr()->Width(),
 | 
					 | 
				
			||||||
          .height = image->GetImageFrameSharedPtr()->Height()}};
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  ImageClassifierResult result;
 | 
					  ImageClassifierResult result;
 | 
				
			||||||
  image_classifier_classify_image(classifier, &mp_image, &result);
 | 
					  image_classifier_classify_image(classifier, &mp_image, &result);
 | 
				
			||||||
| 
						 | 
					@ -84,6 +85,120 @@ TEST(ImageClassifierTest, SmokeTest) {
 | 
				
			||||||
  image_classifier_close(classifier);
 | 
					  image_classifier_close(classifier);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					TEST(ImageClassifierTest, VideoModeTest) {
 | 
				
			||||||
 | 
					  const auto image = DecodeImageFromFile(GetFullPath("burger.jpg"));
 | 
				
			||||||
 | 
					  ASSERT_TRUE(image.ok());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const std::string model_path = GetFullPath(kModelName);
 | 
				
			||||||
 | 
					  ImageClassifierOptions options = {
 | 
				
			||||||
 | 
					      /* base_options= */ {/* model_asset_buffer= */ nullptr,
 | 
				
			||||||
 | 
					                           /* model_asset_path= */ model_path.c_str()},
 | 
				
			||||||
 | 
					      /* running_mode= */ RunningMode::VIDEO,
 | 
				
			||||||
 | 
					      /* classifier_options= */
 | 
				
			||||||
 | 
					      {/* display_names_locale= */ nullptr,
 | 
				
			||||||
 | 
					       /* max_results= */ 3,
 | 
				
			||||||
 | 
					       /* score_threshold= */ 0.0,
 | 
				
			||||||
 | 
					       /* category_allowlist= */ nullptr,
 | 
				
			||||||
 | 
					       /* category_allowlist_count= */ 0,
 | 
				
			||||||
 | 
					       /* category_denylist= */ nullptr,
 | 
				
			||||||
 | 
					       /* category_denylist_count= */ 0},
 | 
				
			||||||
 | 
					      /* result_callback= */ nullptr,
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  void* classifier = image_classifier_create(&options);
 | 
				
			||||||
 | 
					  EXPECT_NE(classifier, nullptr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const auto& image_frame = image->GetImageFrameSharedPtr();
 | 
				
			||||||
 | 
					  const MpImage mp_image = {
 | 
				
			||||||
 | 
					      .type = MpImage::IMAGE_FRAME,
 | 
				
			||||||
 | 
					      .image_frame = {.format = static_cast<ImageFormat>(image_frame->Format()),
 | 
				
			||||||
 | 
					                      .image_buffer = image_frame->PixelData(),
 | 
				
			||||||
 | 
					                      .width = image_frame->Width(),
 | 
				
			||||||
 | 
					                      .height = image_frame->Height()}};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  for (int i = 0; i < kIterations; ++i) {
 | 
				
			||||||
 | 
					    ImageClassifierResult result;
 | 
				
			||||||
 | 
					    image_classifier_classify_for_video(classifier, &mp_image, i, &result);
 | 
				
			||||||
 | 
					    EXPECT_EQ(result.classifications_count, 1);
 | 
				
			||||||
 | 
					    EXPECT_EQ(result.classifications[0].categories_count, 3);
 | 
				
			||||||
 | 
					    EXPECT_EQ(
 | 
				
			||||||
 | 
					        std::string{result.classifications[0].categories[0].category_name},
 | 
				
			||||||
 | 
					        "cheeseburger");
 | 
				
			||||||
 | 
					    EXPECT_NEAR(result.classifications[0].categories[0].score, 0.7939f,
 | 
				
			||||||
 | 
					                kPrecision);
 | 
				
			||||||
 | 
					    image_classifier_close_result(&result);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  image_classifier_close(classifier);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// A structure to support LiveStreamModeTest below. This structure holds a
 | 
				
			||||||
 | 
					// static method `Fn` for a callback function of C API. A `static` qualifier
 | 
				
			||||||
 | 
					// allows to take an address of the method to follow API style. Another static
 | 
				
			||||||
 | 
					// struct member is `last_timestamp` that is used to verify that current
 | 
				
			||||||
 | 
					// timestamp is greater than the previous one.
 | 
				
			||||||
 | 
					struct LiveStreamModeCallback {
 | 
				
			||||||
 | 
					  static int64_t last_timestamp;
 | 
				
			||||||
 | 
					  static void Fn(ImageClassifierResult* classifier_result, const MpImage image,
 | 
				
			||||||
 | 
					                 int64_t timestamp, char* error_msg) {
 | 
				
			||||||
 | 
					    ASSERT_NE(classifier_result, nullptr);
 | 
				
			||||||
 | 
					    ASSERT_EQ(error_msg, nullptr);
 | 
				
			||||||
 | 
					    EXPECT_EQ(
 | 
				
			||||||
 | 
					        std::string{
 | 
				
			||||||
 | 
					            classifier_result->classifications[0].categories[0].category_name},
 | 
				
			||||||
 | 
					        "cheeseburger");
 | 
				
			||||||
 | 
					    EXPECT_NEAR(classifier_result->classifications[0].categories[0].score,
 | 
				
			||||||
 | 
					                0.7939f, kPrecision);
 | 
				
			||||||
 | 
					    EXPECT_GT(image.image_frame.width, 0);
 | 
				
			||||||
 | 
					    EXPECT_GT(image.image_frame.height, 0);
 | 
				
			||||||
 | 
					    EXPECT_GT(timestamp, last_timestamp);
 | 
				
			||||||
 | 
					    last_timestamp++;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					int64_t LiveStreamModeCallback::last_timestamp = -1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					TEST(ImageClassifierTest, LiveStreamModeTest) {
 | 
				
			||||||
 | 
					  const auto image = DecodeImageFromFile(GetFullPath("burger.jpg"));
 | 
				
			||||||
 | 
					  ASSERT_TRUE(image.ok());
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const std::string model_path = GetFullPath(kModelName);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  ImageClassifierOptions options = {
 | 
				
			||||||
 | 
					      /* base_options= */ {/* model_asset_buffer= */ nullptr,
 | 
				
			||||||
 | 
					                           /* model_asset_path= */ model_path.c_str()},
 | 
				
			||||||
 | 
					      /* running_mode= */ RunningMode::LIVE_STREAM,
 | 
				
			||||||
 | 
					      /* classifier_options= */
 | 
				
			||||||
 | 
					      {/* display_names_locale= */ nullptr,
 | 
				
			||||||
 | 
					       /* max_results= */ 3,
 | 
				
			||||||
 | 
					       /* score_threshold= */ 0.0,
 | 
				
			||||||
 | 
					       /* category_allowlist= */ nullptr,
 | 
				
			||||||
 | 
					       /* category_allowlist_count= */ 0,
 | 
				
			||||||
 | 
					       /* category_denylist= */ nullptr,
 | 
				
			||||||
 | 
					       /* category_denylist_count= */ 0},
 | 
				
			||||||
 | 
					      /* result_callback= */ LiveStreamModeCallback::Fn,
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  void* classifier = image_classifier_create(&options);
 | 
				
			||||||
 | 
					  EXPECT_NE(classifier, nullptr);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  const auto& image_frame = image->GetImageFrameSharedPtr();
 | 
				
			||||||
 | 
					  const MpImage mp_image = {
 | 
				
			||||||
 | 
					      .type = MpImage::IMAGE_FRAME,
 | 
				
			||||||
 | 
					      .image_frame = {.format = static_cast<ImageFormat>(image_frame->Format()),
 | 
				
			||||||
 | 
					                      .image_buffer = image_frame->PixelData(),
 | 
				
			||||||
 | 
					                      .width = image_frame->Width(),
 | 
				
			||||||
 | 
					                      .height = image_frame->Height()}};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  for (int i = 0; i < kIterations; ++i) {
 | 
				
			||||||
 | 
					    EXPECT_GE(image_classifier_classify_async(classifier, &mp_image, i), 0);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  image_classifier_close(classifier);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // Due to the flow limiter, the total of outputs might be smaller than the
 | 
				
			||||||
 | 
					  // number of iterations.
 | 
				
			||||||
 | 
					  EXPECT_LE(LiveStreamModeCallback::last_timestamp, kIterations);
 | 
				
			||||||
 | 
					  EXPECT_GT(LiveStreamModeCallback::last_timestamp, 0);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
TEST(ImageClassifierTest, InvalidArgumentHandling) {
 | 
					TEST(ImageClassifierTest, InvalidArgumentHandling) {
 | 
				
			||||||
  // It is an error to set neither the asset buffer nor the path.
 | 
					  // It is an error to set neither the asset buffer nor the path.
 | 
				
			||||||
  ImageClassifierOptions options = {
 | 
					  ImageClassifierOptions options = {
 | 
				
			||||||
| 
						 | 
					@ -124,7 +239,7 @@ TEST(ImageClassifierTest, FailedClassificationHandling) {
 | 
				
			||||||
  ImageClassifierResult result;
 | 
					  ImageClassifierResult result;
 | 
				
			||||||
  char* error_msg;
 | 
					  char* error_msg;
 | 
				
			||||||
  image_classifier_classify_image(classifier, &mp_image, &result, &error_msg);
 | 
					  image_classifier_classify_image(classifier, &mp_image, &result, &error_msg);
 | 
				
			||||||
  EXPECT_THAT(error_msg, HasSubstr("gpu buffer not supported yet"));
 | 
					  EXPECT_THAT(error_msg, HasSubstr("GPU Buffer not supported yet"));
 | 
				
			||||||
  free(error_msg);
 | 
					  free(error_msg);
 | 
				
			||||||
  image_classifier_close(classifier);
 | 
					  image_classifier_close(classifier);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue
	
	Block a user