4 changed files with 200 additions and 220 deletions
-
234app.py
-
114conection.py
-
1shared_manager.py
-
71utils.py
@ -0,0 +1,114 @@ |
|||
# Helper class to track client connection status |
|||
import threading |
|||
import time |
|||
import socket |
|||
|
|||
from message_queue.Manager import Manager |
|||
import shared_manager |
|||
from utils import manager_callback |
|||
|
|||
|
|||
def start_discovery_service(port): |
|||
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) |
|||
sock.bind(('0.0.0.0', port)) |
|||
print(f"Discovery service listening on port {port}...") |
|||
|
|||
while True: |
|||
data, addr = sock.recvfrom(1024) |
|||
if data.decode() == "DISCOVER_SERVER": |
|||
print(f"Received discovery request from {addr}") |
|||
sock.sendto(b"SERVER_RESPONSE", addr) |
|||
break |
|||
sock.close() |
|||
return addr |
|||
|
|||
class ConnectionTracker: |
|||
def __init__(self): |
|||
self.last_message_time = time.time() |
|||
self.timeout = 15 # 15-second timeout |
|||
self.last_client_ip = None # Track last connected client IP |
|||
|
|||
def update_last_message_time(self, client_ip=None): |
|||
self.last_message_time = time.time() |
|||
if client_ip: |
|||
self.last_client_ip = client_ip # Update last known client |
|||
|
|||
def is_client_active(self): |
|||
"""Check if the last client is still active within the last 15 seconds.""" |
|||
return (time.time() - self.last_message_time) < self.timeout |
|||
# Create an instance of ConnectionTracker |
|||
connection_tracker = ConnectionTracker() |
|||
|
|||
|
|||
class ConnectionThread(threading.Thread): |
|||
def __init__(self,debug = False,core = None): |
|||
super().__init__() |
|||
self.running = True |
|||
self.__debug = debug |
|||
self.__core = core |
|||
|
|||
def run(self): |
|||
while self.running: |
|||
time.sleep(0.5) |
|||
|
|||
if connection_tracker.last_client_ip: |
|||
|
|||
if self.__debug: |
|||
print(f"Checking if last client {connection_tracker.last_client_ip} is still available...") |
|||
|
|||
if self.query_last_client(connection_tracker.last_client_ip): |
|||
connection_tracker.update_last_message_time() |
|||
|
|||
if self.__debug: |
|||
print(f"Last client {connection_tracker.last_client_ip} responded. Continuing...") |
|||
continue # Skip discovering a new client |
|||
if not connection_tracker.is_client_active(): |
|||
|
|||
|
|||
print("Client inactive for 15 seconds. Searching for another client...") |
|||
cl_ip, _ = start_discovery_service(12345) |
|||
print(f"New client found: {cl_ip}") |
|||
|
|||
# Reinitialize the Manager with the new client address |
|||
shared_manager.manager = Manager(f"tcp://{cl_ip}:5558", f"tcp://{cl_ip}:5557") |
|||
shared_manager.manager.start(manager_callback) |
|||
|
|||
connection_tracker.update_last_message_time(cl_ip) # Update with new client |
|||
|
|||
def query_last_client(self, client_ip): |
|||
"""Send a heartbeat packet to the last known client IP on port 29170 and wait for a response.""" |
|||
|
|||
if self.__debug: |
|||
print(f"Sending heartbeat to {client_ip}:29170") # Debugging |
|||
|
|||
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) |
|||
sock.settimeout(5) # 5-second timeout |
|||
heartbeat_port = 29170 # New port for heartbeat |
|||
|
|||
try: |
|||
sock.sendto(b"HEARTBEAT", (client_ip, heartbeat_port)) # Send heartbeat message |
|||
data, addr = sock.recvfrom(1024) # Wait for response |
|||
|
|||
if self.__debug: |
|||
print(f"Received response from {addr}: {data.decode()}") # Debugging |
|||
|
|||
if data.decode() == "HEARTBEAT_ACK": |
|||
connection_tracker.update_last_message_time() # Update the last message time |
|||
|
|||
if self.__debug: |
|||
print(f"Client {client_ip} is still responding.") |
|||
return True |
|||
|
|||
except socket.timeout: |
|||
|
|||
print(f"Client {client_ip} did not respond. Marking as inactive.") |
|||
|
|||
finally: |
|||
sock.close() |
|||
|
|||
return False # Client did not respond |
|||
|
|||
|
|||
def stop(self): |
|||
self.running = False |
|||
|
@ -0,0 +1 @@ |
|||
manager = "nothing" |
@ -0,0 +1,71 @@ |
|||
import os |
|||
import sys |
|||
proto_dir = os.path.join(os.path.dirname(__file__), 'message_queue', 'proto') |
|||
if proto_dir not in sys.path: |
|||
sys.path.append(proto_dir) |
|||
import shared_manager |
|||
from cvStreamer import cvStreamer |
|||
from message_queue.proto.Message_pb2 import Message |
|||
from message_queue.proto.TrackCommandMessage_pb2 import TrackCommand |
|||
from message_queue.proto.enums_pb2 import MessageType |
|||
from video_streamer.gst_video_streamer import GstVideoStreamer |
|||
|
|||
def findUsbCams(): |
|||
cv_cameras = [] |
|||
for i in range(50): |
|||
cap = cvStreamer(i) |
|||
if cap.isOpened(): |
|||
cv_cameras.append(cap) |
|||
return cv_cameras |
|||
|
|||
core = "a" |
|||
|
|||
def manager_callback(msg_str): |
|||
msg = Message() |
|||
msg.ParseFromString(msg_str) |
|||
|
|||
if msg.msgType == MessageType.MESSAGE_TYPE_TOGGLE_TRACK: |
|||
if msg.track_cmd.command == TrackCommand.TRACK_COMMAND_STOP: |
|||
core.stop_track() |
|||
elif msg.track_cmd.command == TrackCommand.TRACK_COMMAND_START: |
|||
core.start_track(msg.track_cmd.x, msg.track_cmd.y, msg.track_cmd.w, msg.track_cmd.h) |
|||
elif msg.track_cmd.command == TrackCommand.TRACK_COMMAND_START_DETECT: |
|||
core.start_detect(msg.track_cmd.x, msg.track_cmd.y, msg.track_cmd.w, msg.track_cmd.h) |
|||
elif msg.msgType == MessageType.MESSAGE_TYPE_TRACK_SETTINGS: |
|||
core.set_thickness(msg.track_settings.thickness) |
|||
elif msg.msgType == MessageType.MESSAGE_TYPE_SWITCH_CAMERA: |
|||
print(f"switch camera detected ,primary value is : {msg.cam_switch.primaryCamType}") |
|||
print(f"switch camera detected ,secondary value is : {msg.cam_switch.secondaryCamType}") |
|||
core.set_source(msg.cam_switch.primaryCamType) |
|||
elif msg.msgType == MessageType.MESSAGE_TYPE_SET_CAMERA: |
|||
if msg.cam_set.cameraSource == 1: |
|||
print(f"set camera source to network") |
|||
ip = msg.cam_set.ip |
|||
videoStreamers = [] |
|||
for src in range(2): |
|||
rtsp_link = f'rtsp://admin:Abc.12345@{ip}:554/ch{src}/stream0' |
|||
# ic(rtsp_link) |
|||
videoStreamer = GstVideoStreamer(rtsp_link, [1920, 1080, 3], str(idx), fps=15) |
|||
videoStreamer.cameraStatus.connect(handle_camera_status) |
|||
print(f'{videoStreamer.id} connected') |
|||
videoStreamers.append(videoStreamer) |
|||
core.set_video_sources(videoStreamers) |
|||
# videoStreamers = [] |
|||
# for idx, rtsp_link in enumerate(rtsp_links): |
|||
# videoStreamer = GstVideoStreamer(rtsp_link, [1920, 1080, 3], str(idx), fps=15) |
|||
# videoStreamer.cameraStatus.connect(handle_camera_status) |
|||
# print(f'{videoStreamer.id} connected') |
|||
# videoStreamers.append(videoStreamer) |
|||
# core.set_video_sources(videoStreamers) |
|||
else: |
|||
sources = findUsbCams() |
|||
if len(sources) >= 2: |
|||
print("set camera source to captutre card") |
|||
core.set_video_sources(findUsbCams()) |
|||
|
|||
|
|||
def handle_camera_status(status: int): |
|||
m = Message() |
|||
m.msgType = MessageType.MESSAGE_TYPE_CAMERA_CONNECTION_STATUS |
|||
m.cam_status.status = status |
|||
shared_manager.manager.send_message(m.SerializeToString()) |
Write
Preview
Loading…
Cancel
Save
Reference in new issue