2022-10-20 13:05:23 +00:00
|
|
|
"""
|
|
|
|
Mqtt gateway for oak-lite device
|
|
|
|
"""
|
|
|
|
import argparse
|
2022-01-15 17:42:14 +00:00
|
|
|
import logging
|
|
|
|
import os
|
2022-10-20 15:06:55 +00:00
|
|
|
import signal
|
2022-10-27 08:34:04 +00:00
|
|
|
import typing, types
|
2022-10-20 13:05:23 +00:00
|
|
|
|
2022-10-27 08:34:04 +00:00
|
|
|
import depthai as dai
|
2022-01-15 17:42:14 +00:00
|
|
|
import paho.mqtt.client as mqtt
|
2022-10-20 13:05:23 +00:00
|
|
|
|
2023-10-01 15:24:05 +00:00
|
|
|
from camera import oak_pipeline as cam
|
2022-01-15 17:42:14 +00:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2022-10-20 13:05:23 +00:00
|
|
|
_DEFAULT_CLIENT_ID = "robocar-depthai"
|
2022-01-15 17:42:14 +00:00
|
|
|
|
|
|
|
|
2022-10-20 15:06:55 +00:00
|
|
|
def _parse_args_cli() -> argparse.Namespace:
|
2022-10-20 08:48:35 +00:00
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument("-u", "--mqtt-username",
|
|
|
|
help="MQTT user",
|
|
|
|
default=_get_env_value("MQTT_USERNAME", ""))
|
|
|
|
parser.add_argument("-p", "--mqtt-password",
|
|
|
|
help="MQTT password",
|
|
|
|
default=_get_env_value("MQTT_PASSWORD", ""))
|
|
|
|
parser.add_argument("-b", "--mqtt-broker-host",
|
|
|
|
help="MQTT broker host",
|
|
|
|
default=_get_env_value("MQTT_BROKER_HOST", "localhost"))
|
|
|
|
parser.add_argument("-P", "--mqtt-broker-port",
|
|
|
|
help="MQTT broker port",
|
|
|
|
type=int,
|
|
|
|
default=_get_env_int_value("MQTT_BROKER_PORT", 1883))
|
|
|
|
parser.add_argument("-C", "--mqtt-client-id",
|
|
|
|
help="MQTT client id",
|
2022-10-20 13:05:23 +00:00
|
|
|
default=_get_env_value("MQTT_CLIENT_ID", _DEFAULT_CLIENT_ID))
|
2022-10-20 08:48:35 +00:00
|
|
|
parser.add_argument("-c", "--mqtt-topic-robocar-oak-camera",
|
|
|
|
help="MQTT topic where to publish robocar-oak-camera frames",
|
2022-10-20 13:05:23 +00:00
|
|
|
default=_get_env_value("MQTT_TOPIC_CAMERA", "/oak/camera_rgb"))
|
2022-10-20 08:48:35 +00:00
|
|
|
parser.add_argument("-o", "---mqtt-topic-robocar-objects",
|
|
|
|
help="MQTT topic where to publish objects detection results",
|
|
|
|
default=_get_env_value("MQTT_TOPIC_OBJECTS", "/objects"))
|
|
|
|
parser.add_argument("-t", "--objects-threshold",
|
|
|
|
help="threshold to filter detected objects",
|
|
|
|
type=float,
|
|
|
|
default=_get_env_float_value("OBJECTS_THRESHOLD", 0.2))
|
2023-10-01 15:24:05 +00:00
|
|
|
parser.add_argument("-f", "--camera-fps",
|
|
|
|
help="set rate at which camera should produce frames",
|
|
|
|
type=int,
|
|
|
|
default=30)
|
2022-10-20 08:48:35 +00:00
|
|
|
parser.add_argument("-H", "--image-height", help="image height",
|
|
|
|
type=int,
|
|
|
|
default=_get_env_int_value("IMAGE_HEIGHT", 120))
|
|
|
|
parser.add_argument("-W", "--image-width", help="image width",
|
|
|
|
type=int,
|
|
|
|
default=_get_env_int_value("IMAGE_WIDTH", 126))
|
2022-11-09 19:21:03 +00:00
|
|
|
parser.add_argument("--log", help="Log level",
|
|
|
|
type=str,
|
|
|
|
default="info",
|
|
|
|
choices=["info", "debug"])
|
2022-10-20 08:48:35 +00:00
|
|
|
args = parser.parse_args()
|
2022-10-20 15:06:55 +00:00
|
|
|
return args
|
|
|
|
|
|
|
|
|
2022-10-27 08:34:04 +00:00
|
|
|
def _init_mqtt_client(broker_host: str, broker_port: int, user: str, password: str, client_id: str) -> mqtt.Client:
|
2022-10-20 15:06:55 +00:00
|
|
|
logger.info("Start part.py-robocar-oak-camera")
|
|
|
|
client = mqtt.Client(client_id=client_id, clean_session=True, userdata=None, protocol=mqtt.MQTTv311)
|
|
|
|
|
|
|
|
client.username_pw_set(user, password)
|
|
|
|
logger.info("Connect to mqtt broker %s", broker_host)
|
|
|
|
client.connect(host=broker_host, port=broker_port, keepalive=60)
|
|
|
|
logger.info("Connected to mqtt broker")
|
|
|
|
return client
|
|
|
|
|
|
|
|
|
|
|
|
def execute_from_command_line() -> None:
|
|
|
|
"""
|
|
|
|
Cli entrypoint
|
|
|
|
:return:
|
|
|
|
"""
|
|
|
|
|
|
|
|
args = _parse_args_cli()
|
2022-11-09 19:21:03 +00:00
|
|
|
if args.log == "info":
|
|
|
|
logging.basicConfig(level=logging.INFO)
|
|
|
|
elif args.log == "debug":
|
|
|
|
logging.basicConfig(level=logging.DEBUG)
|
2022-01-15 17:42:14 +00:00
|
|
|
|
2022-10-20 13:05:23 +00:00
|
|
|
client = _init_mqtt_client(broker_host=args.mqtt_broker_host,
|
|
|
|
broker_port=args.mqtt_broker_port,
|
|
|
|
user=args.mqtt_username,
|
|
|
|
password=args.mqtt_password,
|
|
|
|
client_id=args.mqtt_client_id,
|
|
|
|
)
|
2022-10-20 14:57:33 +00:00
|
|
|
frame_processor = cam.FrameProcessor(mqtt_client=client, frame_topic=args.mqtt_topic_robocar_oak_camera)
|
|
|
|
object_processor = cam.ObjectProcessor(mqtt_client=client,
|
|
|
|
objects_topic=args.mqtt_topic_robocar_objects,
|
|
|
|
objects_threshold=args.objects_threshold)
|
|
|
|
|
2022-10-27 08:34:04 +00:00
|
|
|
pipeline = dai.Pipeline()
|
2022-11-02 16:33:36 +00:00
|
|
|
pipeline_controller = cam.PipelineController(pipeline=pipeline,
|
|
|
|
frame_processor=frame_processor,
|
2022-10-27 08:34:04 +00:00
|
|
|
object_processor=object_processor,
|
|
|
|
object_node=cam.ObjectDetectionNN(pipeline=pipeline),
|
|
|
|
camera=cam.CameraSource(pipeline=pipeline,
|
|
|
|
img_width=args.image_width,
|
2022-12-25 10:22:06 +00:00
|
|
|
img_height=args.image_height,
|
2023-10-01 15:24:05 +00:00
|
|
|
fps=args.camera_fps,
|
2022-10-27 08:34:04 +00:00
|
|
|
))
|
|
|
|
|
|
|
|
def sigterm_handler(signum: int, frame: typing.Optional[
|
|
|
|
types.FrameType]) -> None: # pylint: disable=unused-argument # need to implement handler signature
|
2022-10-20 15:06:55 +00:00
|
|
|
logger.info("exit on SIGTERM")
|
|
|
|
pipeline_controller.stop()
|
|
|
|
|
|
|
|
signal.signal(signal.SIGTERM, sigterm_handler)
|
2022-10-20 14:57:33 +00:00
|
|
|
pipeline_controller.run()
|
2022-01-15 17:42:14 +00:00
|
|
|
|
|
|
|
|
2022-10-20 08:48:35 +00:00
|
|
|
def _get_env_value(env_var: str, default_value: str) -> str:
|
2022-01-15 17:42:14 +00:00
|
|
|
if env_var in os.environ:
|
|
|
|
return os.environ[env_var]
|
|
|
|
return default_value
|
2022-10-20 08:48:35 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _get_env_int_value(env_var: str, default_value: int) -> int:
|
|
|
|
value = _get_env_value(env_var, str(default_value))
|
|
|
|
return int(value)
|
|
|
|
|
|
|
|
|
|
|
|
def _get_env_float_value(env_var: str, default_value: float) -> float:
|
|
|
|
value = _get_env_value(env_var, str(default_value))
|
|
|
|
return float(value)
|