Coverage for /private/tmp/im/impacket/impacket/mqtt.py : 26%
 
         
         
    Hot-keys on this page
r m x p toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
| #!/usr/bin/env python # SECUREAUTH LABS. Copyright 2018 SecureAuth Corporation. All rights reserved. # # This software is provided under under a slightly modified version # of the Apache Software License. See the accompanying LICENSE file # for more information. # # Author: Alberto Solino (@agsolino) # # Description: # Minimalistic MQTT implementation, just focused on connecting, subscribing and publishing basic # messages on topics. # # References: # https://docs.oasis-open.org/mqtt/mqtt/v3.1.1/mqtt-v3.1.1.html # # ToDo: # [ ] Implement all the MQTT Control Packets and operations # [ ] Implement QoS = QOS_ASSURED_DELIVERY when publishing messages # except ImportError: logging.critical("pyOpenSSL is not installed, can't continue") raise 
 # Packet Types 
 # CONNECT Flags 
 # CONNECT_ACK Return Errors 0x00: 'Connection Accepted', 0x01: 'Connection Refused, unacceptable protocol version', 0x02: 'Connection Refused, identifier rejected', 0x03: 'Connection Refused, Server unavailable', 0x04: 'Connection Refused, bad user name or password', 0x05: 'Connection Refused, not authorized' } 
 # QoS Levels 
 ('PacketType','B=0'), ('MessageLength','<L=0'), ) ('_VariablePart', '_-VariablePart', 'self["MessageLength"]'), ('VariablePart', ':'), ) self['PacketType'] |= (QoS << 1) 
 if data is not None and len(data) > 2: # Get the Length index = 1 multiplier = 1 value = 0 encodedByte = 128 packetType = data[0] while (encodedByte & 128) != 0: encodedByte = ord(data[index]) value += (encodedByte & 127) * multiplier multiplier *= 128 index += 1 if multiplier > 128 * 128 * 128: raise Exception('Malformed Remaining Length') data = packetType + struct.pack('<L', value) + data[index:value+index] return Structure.fromString(self, data) raise Exception('Dont know') 
 packetType = self['PacketType'] self.commonHdr = () packetLen = len(Structure.getData(self)) output = '' while packetLen > 0: encodedByte = packetLen % 128 packetLen /= 128 if packetLen > 0: encodedByte |= 128 output += chr(encodedByte) self.commonHdr = ( ('PacketType','B=0'), ('MessageLength',':'), ) self['PacketType'] = packetType self['MessageLength'] = output if output == '': self['MessageLength'] = chr(00) 
 return Structure.getData(self) 
 
 ('Length','>H-Name'), ('Name',':'), ) 
 ('ProtocolName',':', MQTT_String), ('Version','B=3'), ('Flags','B=2'), ('KeepAlive','>H=60'), ('ClientID',':', MQTT_String), ('Payload',':=""'), ) MQTT_Packet.__init__(self, data, alignment) if data is None: self['PacketType'] = PACKET_CONNECT 
 ('ReturnCode', '>H=0'), ) 
 ('Topic',':', MQTT_String), ('Message',':'), ) MQTT_Packet.__init__(self, data, alignment) if data is None: self['PacketType'] = PACKET_PUBLISH 
 if self['PacketType'] & 6 > 0: # We have QoS enabled, we need to have a MessageID field self.structure = ( ('Topic', ':', MQTT_String), ('MessageID', '>H=0'), ('Message', ':'), ) return MQTT_Packet.getData(self) 
 ) MQTT_Packet.__init__(self, data, alignment) if data is None: self['PacketType'] = PACKET_DISCONNECT 
 ('MessageID','>H=1'), ('Topic',':', MQTT_String), ('Flags','B=0'), ) MQTT_Packet.__init__(self, data, alignment) if data is None: self['PacketType'] = PACKET_SUBSCRIBE 
 ('MessageID','>H=0'), ('ReturnCode','B=0'), ) MQTT_Packet.__init__(self, data, alignment) if data is None: self['PacketType'] = PACKET_SUBSCRIBEACK 
 ('MessageID','>H=1'), ('Topics',':'), ) MQTT_Packet.__init__(self, data, alignment) if data is None: self['PacketType'] = PACKET_UNSUBSCRIBE 
 """ This is the exception every client should catch """ 
 Exception.__init__(self) self.error = error self.packet = packet self.errorString = errorString 
 return self.error 
 return self.packet 
 return self.errorString 
 return self.errorString 
 self._targetHost = host self._targetPort = port self._isSSL = isSSL self._socket = None self._messageId = 1 self.connectSocket() 
 return self._socket 
 s = socket.socket() s.connect((self._targetHost, int(self._targetPort))) 
 if self._isSSL is True: ctx = SSL.Context(SSL.TLSv1_METHOD) self._socket = SSL.Connection(ctx, s) self._socket.set_connect_state() self._socket.do_handshake() else: self._socket = s 
 return self._socket.sendall(str(request)) 
 self.send(request) return self.recv() 
 REQUEST_SIZE = 8192 data = '' done = False while not done: recvData = self._socket.recv(REQUEST_SIZE) if len(recvData) < REQUEST_SIZE: done = True data += recvData 
 response = [] while len(data) > 0: try: message = MQTT_Packet(data) remaining = data[len(message):] except Exception: # We need more data remaining = data + self._socket.recv(REQUEST_SIZE) else: response.append(message) data = remaining 
 self._messageId += 1 return response 
 """ 
 :param clientId: Whatever cliend Id that represents you :param username: if None, anonymous connection will be attempted :param password: if None, anonymous connection will be attempted :param protocolName: specification states default should be 'MQTT' but some brokers might expect 'MQIsdp' :param version: Allowed versions are 3 or 4 (some brokers might like 4) :param flags: :param keepAlive: default 60 :return: True or MQTTSessionError if something went wrong """ 
 # Let's build the packet connectPacket = MQTT_Connect() connectPacket['Version'] = version connectPacket['Flags'] = flags connectPacket['KeepAlive'] = keepAlive connectPacket['ProtocolName'] = MQTT_String() connectPacket['ProtocolName']['Name'] = protocolName 
 connectPacket['ClientID'] = MQTT_String() connectPacket['ClientID']['Name'] = clientId 
 if username is not None: connectPacket['Flags'] |= CONNECT_USERNAME | CONNECT_PASSWORD if username is None: user = '' else: user = username if password is None: pwd = '' else: pwd = password 
 username = MQTT_String() username['Name'] = user password = MQTT_String() password['Name'] = pwd connectPacket['Payload'] = str(username) + str(password) 
 data= self.sendReceive(connectPacket)[0] 
 response = MQTT_ConnectAck(str(data)) if response['ReturnCode'] != 0: raise MQTTSessionError(error = response['ReturnCode'], errorString = CONNECT_ACK_ERROR_MSGS[response['ReturnCode']] ) 
 return True 
 """ 
 :param topic: Topic name you want to subscribe to :param messageID: optional messageId :param flags: Message flags :param QoS: define the QoS requested :return: True or MQTTSessionError if something went wrong """ subscribePacket = MQTT_Subscribe() subscribePacket['MessageID'] = messageID subscribePacket['Topic'] = MQTT_String() subscribePacket['Topic']['Name'] = topic subscribePacket['Flags'] = flags subscribePacket.setQoS(QoS) 
 try: data = self.sendReceive(subscribePacket)[0] except Exception as e: raise MQTTSessionError(errorString=str(e)) 
 subAck = MQTT_SubscribeACK(str(data)) 
 if subAck['ReturnCode'] > 2: raise MQTTSessionError(errorString = 'Failure to subscribe') 
 return True 
 """ Unsubscribes from a topic 
 :param topic: :param messageID: :param QoS: define the QoS requested :return: """ # ToDo: Support more than one topic packet = MQTT_UnSubscribe() packet['MessageID'] = messageID packet['Topics'] = MQTT_String() packet['Topics']['Name'] = topic packet.setQoS( QoS ) 
 return self.sendReceive(packet) 
 
 packet = MQTT_Publish() packet['Topic'] = MQTT_String() packet['Topic']['Name'] = topic packet['Message'] = message packet['MessageID'] = messageID packet.setQoS( QoS ) 
 return self.sendReceive(packet) 
 return self.send(str(MQTT_Disconnect())) 
 HOST = '192.168.45.162' USER = 'test' PASS = 'test' 
 mqtt = MQTTConnection(HOST, 1883, False) mqtt.connect('secure-', username=USER, password=PASS, version = 3) #mqtt.connect(protocolName='MQTT', version=4) #mqtt.connect() 
 #mqtt.subscribe('/test/beto') #mqtt.unSubscribe('/test/beto') #mqtt.publish('/test/beto', 'Hey There, I"d like to talk to you', QoS=1) mqtt.subscribe('$SYS/#') 
 
 while True: 
 packets = mqtt.recv() for packet in packets: publish = MQTT_Publish(str(packet)) print('%s -> %s' % (publish['Topic']['Name'], publish['Message'])) 
 mqtt.disconnect() |