blob: 1b951f2019ff77c7fd1cb4395725c2e8230b1c92 [file] [log] [blame]
#!/usr/bin/env python
""" UDP sender class """
import socket
import urlparse
from cp_protocol import Packet
from logger import define_logger
class SenderException(Exception):
""" Exceptions in Sender class """
pass
class Timeout(Exception):
""" Exceptions in Sender class """
pass
class Sender(object):
""" UDP sender class """
def __init__(self, packer, url=None, port=None, host="0.0.0.0", size=256):
""" Create connection object from input udp string or params"""
# test input
if url is None and port is None:
raise SenderException("Bad initialization")
if url is not None:
data = urlparse.urlparse(url)
# check schema
if data.scheme != "udp":
mes = "Bad protocol type: %s instead of UDP" % data.scheme
logger.error(mes)
raise SenderException("Bad protocol type")
# try to get port
try:
int_port = int(data.port)
except ValueError:
logger.error("Bad UDP port")
raise SenderException("Bad UDP port")
# save paths
self.sendto = (data.hostname, int_port)
self.bindto = (data.hostname, int_port)
# try to get size
try:
self.size = int(data.path.strip("/"))
except ValueError:
logger.error("Bad packet part size")
raise SenderException("Bad packet part size")
else:
# url is None - use size and port
self.sendto = (host, port)
self.bindto = ("0.0.0.0", port)
self.size = size
self.packer = packer
self.sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
self.binded = False
self.all_data = {}
self.send_packer = None
def bind(self):
""" Prepare for listening """
self.sock.bind(self.bindto)
self.sock.settimeout(0.5)
self.binded = True
def send(self, data):
""" Send data to udp socket"""
if self.sock.sendto(data, self.sendto) != len(data):
mes = "Cannot send data to %s:%s" % self.sendto
logger.error(mes)
raise SenderException("Cannot send data")
def send_by_protocol(self, data):
""" Send data by Packet protocol
data = dict"""
if self.send_packer is None:
self.send_packer = Packet(self.packer())
parts = self.send_packer.create_packet_v2(data, self.size)
for part in parts:
self.send(part)
def recv(self):
""" Receive data from udp socket"""
# check for binding
if not self.binded:
self.bind()
# try to recv
try:
data, (remote_ip, _) = self.sock.recvfrom(self.size)
return data, remote_ip
except socket.timeout:
raise Timeout()
def recv_by_protocol(self):
""" Receive data from udp socket by Packet protocol"""
data, remote_ip = self.recv()
if remote_ip not in self.all_data:
self.all_data[remote_ip] = Packet(self.packer())
return self.all_data[remote_ip].new_packet(data)
def recv_with_answer(self, stop_event=None):
""" Receive data from udp socket and send 'ok' back
Command port = local port + 1
Answer port = local port
Waiting for command is blocking """
# create command socket
command_sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
command_port = self.bindto[1]+1
command_sock.bind(("0.0.0.0", command_port))
command_sock.settimeout(1)
# try to recv
while True:
try:
data, (remote_ip, _) = command_sock.recvfrom(self.size)
self.send("ok")
return data, remote_ip
except socket.timeout:
if stop_event is not None and stop_event.is_set():
# return None if we are interrupted
return None
def verified_send(self, send_host, message, max_repeat=20):
""" Send and verify it by answer not more then max_repeat
Send port = local port + 1
Answer port = local port
Return True if send is verified """
# create send socket
send_sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
send_port = self.sendto[1]+1
for repeat in range(0, max_repeat):
send_sock.sendto(message, (send_host, send_port))
try:
data, remote_ip = self.recv()
if remote_ip == send_host and data == "ok":
return True
else:
logger.warning("No answer from %s, try %i", send_host, repeat)
except Timeout:
logger.warning("No answer from %s, try %i", send_host, repeat)
return False
logger = define_logger(__name__)