* Refine atachment detect.

* Fix duplicated sending msg.

Signed-off-by: peter-tank <30540412+peter-tank@users.noreply.github.com>
This commit is contained in:
peter-tank 2018-07-19 00:03:23 +08:00
parent 425eded1a2
commit fd7bb7c917
No known key found for this signature in database
GPG Key ID: 5C422EFEA3A30F52

View File

@ -110,7 +110,6 @@ inputShorts = dict({
'yes': ['y', 'yes'], 'yes': ['y', 'yes'],
'no': ['n', 'no'], 'no': ['n', 'no'],
'exit': ['e', 'ex', 'exit'], 'exit': ['e', 'ex', 'exit'],
'save': ['save', 's', 'sv'],
'deterministic': ['d', 'dt'], 'deterministic': ['d', 'dt'],
'random': ['r', 'rd', 'random'], 'random': ['r', 'rd', 'random'],
'message': ['m', 'msg', 'message'], 'message': ['m', 'msg', 'message'],
@ -1087,7 +1086,9 @@ def genAdd(lbl, deterministic, passphrase, numOfAdd, addVNum, streamNum, ripe):
def getBase64Len(x=''): def getBase64Len(x=''):
return int(len(x)*(3/4)) - 2 if x[-2:] == '==' else 1 if x[-1] == '=' else 0
strip = len(x) if len(x) < 4 else 2 if x[-2:] == '==' else 1 if x[-1] == '=' else 0
return int(len(x) * 3 /4) - strip
def dump2File(fileName, fileData, deCoded): def dump2File(fileName, fileData, deCoded):
@ -1122,7 +1123,7 @@ def dump2File(fileName, fileData, deCoded):
print '\n'.join([ print '\n'.join([
' -----------------------------------', ' -----------------------------------',
' Contents seems not "BASE64" encoded. (base on length check)', ' Contents seems not "BASE64" encoded. (base on length check)',
' Start[%d] ~ Ends[%d].' % (x[:3], x[-3:]), ' Start[%s] ~ Ends[%s].' % (x[:3], x[-3:]),
' About: %d(bytes).' % getBase64Len(x), ' About: %d(bytes).' % getBase64Len(x),
' FileName: "%s"' % fileName, ' FileName: "%s"' % fileName,
]) ])
@ -1331,6 +1332,8 @@ def sendMsg(toAddress, fromAddress, subject=None, message=None):
uInput = userInput('Would you like to try again, (n)o or (Y)es?').lower() uInput = userInput('Would you like to try again, (n)o or (Y)es?').lower()
if uInput in inputShorts['no']: if uInput in inputShorts['no']:
break break
else:
break
if ackData['error'] == 0: if ackData['error'] == 0:
print ' Fetching send status...' print ' Fetching send status...'
@ -1418,6 +1421,8 @@ def sendBrd(fromAddress=None, subject=None, message=None):
uInput = userInput('Would you like to try again, no or (Y)es?').lower() uInput = userInput('Would you like to try again, no or (Y)es?').lower()
if uInput in inputShorts['no']: if uInput in inputShorts['no']:
break break
else:
break
if ackData['error'] == 0: if ackData['error'] == 0:
print ' Fetching send status...' print ' Fetching send status...'
@ -1466,7 +1471,7 @@ def inbox(unreadOnly=False, pageNum=20):
print ' From:', getLabelForAddress(message['fromAddress']) # Get the from address print ' From:', getLabelForAddress(message['fromAddress']) # Get the from address
print ' Subject:', message['subject'].decode('base64') # Get the subject print ' Subject:', message['subject'].decode('base64') # Get the subject
print ' Received:', datetime.datetime.fromtimestamp(float(message['receivedTime'])).strftime('%Y-%m-%d %H:%M:%S') print ' Received:', datetime.datetime.fromtimestamp(float(message['receivedTime'])).strftime('%Y-%m-%d %H:%M:%S')
print ' Base64Len:', str(len(message['message'])) print ' Size: %d(bytes)' % getBase64Len(message['message'])
messagesPrinted += 1 messagesPrinted += 1
if not message['read']: if not message['read']:
messagesUnread += 1 messagesUnread += 1
@ -1507,7 +1512,7 @@ def outbox(pageNum=20):
print ' Status:', message['status'] # Get the status print ' Status:', message['status'] # Get the status
print ' Ack:', message['ackData'] # Get the ackData print ' Ack:', message['ackData'] # Get the ackData
print ' Last Action Time:', datetime.datetime.fromtimestamp(float(message['lastActionTime'])).strftime('%Y-%m-%d %H:%M:%S') print ' Last Action Time:', datetime.datetime.fromtimestamp(float(message['lastActionTime'])).strftime('%Y-%m-%d %H:%M:%S')
print ' Base64Len:', str(len(message['message'])) print ' Size: %d(bytes)' % getBase64Len(message['message'])
if msgNum % pageNum == 0 and msgNum != 0: if msgNum % pageNum == 0 and msgNum != 0:
uInput = userInput('Paused on %d/%d, next [%d],' % (msgNum, numMessages - 1, msgNum - pageNum if msgNum > pageNum else 0)) uInput = userInput('Paused on %d/%d, next [%d],' % (msgNum, numMessages - 1, msgNum - pageNum if msgNum > pageNum else 0))
@ -1524,56 +1529,47 @@ def attDetect(content=None, textmsg=None, attPrefix=None, askSave=True):
global inputShorts global inputShorts
attPos = msgPos = 0 attPos = msgPos = 0
counter = 0
prefixRe = re.compile(r'<(\w*)\s*[^<>]*src="[^"]*;base64,$')
suffixRe = re.compile(r'("\s*/?>)')
fnRe1 = re.compile(r'alt="([^"]*)"*\s*[^<>]*;base64,$') # alt= preferred for attachment filename
fnRe2 = re.compile(r'src="([^"]*)"*\s*[^<>]*;base64,$')
# Hard way search attachments # Hard way search attachments
for counter in range(1, 3): # Allows maximum 3 of attachments to be downloaded/saved while counter < 4: # Allows maximum 3 of attachments to be downloaded/saved
try: try:
attPos = content.index(';base64,', attPos) + 9 # Finds the attachment position attEndPos = attPos = content.index(';base64,', attPos) + len(';base64,') # Finds the attachment position
attEndPos = content.index('/>', attPos) - 1 # Finds the end of the attachment hasEnd = suffixRe.search(content[attPos:])
if hasEnd:
attEndPos = attPos + hasEnd.start() + len(hasEnd.group(1))
else:
raise ValueError
attPre = attPos - 9 # back for ;base64, | <img src=";base64, | <attachment src=";base64, attPre = attPos - len(';base64,') - 1 # back forward searching based on ;base64
attOffset = attPos - msgPos
# try remove prefix <xxxx | <xxxxxxxxxxx # try remove prefix <xxxx | <xxxxxxxxxxx
pBack = 0 prefix_max = attOffset if attOffset <= 180 else 180 # define max forward searching for base64 embedded codes '<xxxx
if attPre >= (msgPos + 12): prefixStr = content[attPos - prefix_max:attPos]
if '<' == content[attPre - 12]: iscomplete = prefixRe.search(prefixStr)
pBack = 12 if iscomplete and hasEnd.start() >= 4: # too small not attachments, leave as is
elif attPre - msgPos + 5 >= 0: havefn = fnRe1.search(prefixStr)
if '<' == content[attPre - 5]: if not havefn:
pBack = 5 havefn = fnRe2.search(prefixStr)
attPre = attPre - pBack fn = '%s_%d_%s_%s' % (attPrefix, attPos, iscomplete.group(1), havefn.group(1) if havefn else 'notdetected')
attPre = attPos - prefix_max + iscomplete.start()
try: this_attachment = content[attPos:attPos + hasEnd.start()]
fnPos = content.index('alt="', msgPos, attPos) + 5 # Finds position of the filename
fnEndPos = content.index('" src=', msgPos, attPos) # Finds the end position
# fnLen = fnEndPos - fnPos #Finds the length of the filename
fn = content[fnPos:fnEndPos]
attPre = fnPos - 5 # back for alt=" | <img alt=" | <attachment alt="
# try remove prefix <xxxx | <xxxxxxxxxxx
pBack = 0
if attPre >= (msgPos + 12):
if '<' == content[attPre - 12]:
pBack = 12
elif attPre - msgPos + 5 >= 0:
if '<' == content[attPre - 5]:
pBack = 5
attPre = attPre - pBack
except ValueError:
fn = 'notdetected'
fn = '%d_attachment_%d_%s' % (attPrefix, attPos, fn)
this_attachment = content[attPos:attEndPos]
x = filter(lambda z: not re.match(r'^\s*$', z), this_attachment) x = filter(lambda z: not re.match(r'^\s*$', z), this_attachment)
# x = x.replace('\n', '').strip() # x = x.replace('\n', '').strip()
trydecode = False trydecode = False
counter += 1
if len(x) % 4 == 0: # check by length before decode. if len(x) % 4 == 0: # check by length before decode.
trydecode = True trydecode = True
else: else:
print '\n'.join([ print '\n'.join([
' -----------------------------------', ' -----------------------------------',
' Embeded mesaage seems not "BASE64" encoded. (base on length check)', ' Embeded mesaage seems not "BASE64" encoded. (base on length check)',
' Offset: %d, about: %d(bytes).' % (attPos, (int(len(x)*3/4)) - 2 if x[-2:] == '==' else 1 if x[-1] == '=' else 0), ' Offset: %d, about: %d(bytes).' % (attPre, getBase64Len(x)),
' Start[%d] ~ Ends[%d].' % (x[:3], x[-3:]), ' Start[%s] ~ Ends[%s].' % (x[:3], x[-3:]),
' FileName: "%s"' % fn, ' FileName: "%s"' % fn,
]) ])
uInput = userInput('Try to decode anyway, (n)o or (Y)es?') uInput = userInput('Try to decode anyway, (n)o or (Y)es?')
@ -1583,31 +1579,32 @@ def attDetect(content=None, textmsg=None, attPrefix=None, askSave=True):
try: try:
y = x.decode('base64', 'strict') y = x.decode('base64', 'strict')
if x == y.encode('base64').replace('\n', ''): # double check decoded string. if x == y.encode('base64').replace('\n', ''): # double check decoded string.
print ' This embeded message decoded successfully:', fn
if askSave is True: if askSave is True:
uInput = userInput('Download the "decoded" attachment, (y)es or (No)?\nName(%d): %s,' % (counter, fn)).lower() uInput = userInput('Download the "decoded" attachment, (y)es or (No)?\nNames[%d]: %s,' % (counter, fn)).lower()
if uInput in inputShorts['yes']: if uInput in inputShorts['yes']:
src = dump2File(fn, y, True) src = dump2File(fn, y, True)
else:
src = ' Attachment skiped[%d] "%s".' % (counter, fn)
else: else:
src = dump2File(fn, y, True) src = dump2File(fn, y, True)
print src print src
attmsg = '\n'.join([ attmsg = '\n'.join([
' -----------------------------------', ' -----------------------------------',
' Attachment: "%s"' % fn, ' Attachment[%d]: "%s"' % (counter, fn),
' Size: %d(bytes)' % getBase64Len(x), ' Size: %d(bytes)' % getBase64Len(x),
' -----------------------------------', ' -----------------------------------',
]) ])
# remove base64 and '<att' prefix and suffix '/>' stuff # remove base64 and '<att' prefix and suffix '/>' stuff
textmsg = textmsg + content[msgPos:attPre] + attmsg textmsg = textmsg + content[msgPos:attPre] + attmsg
attEndPos += 3
msgPos = attEndPos msgPos = attEndPos
else: else:
print '\n Failed on decode this embeded "BASE64" like message on re-encode check.\n' print '\n Failed on decode this embeded "BASE64" like message on re-encode check.\n'
except ValueError: except ValueError:
pass
print '\n Failed on decode this emdeded "BASE64" encoded like message.\n' print '\n Failed on decode this emdeded "BASE64" encoded like message.\n'
except InputException:
raise
except Exception: except Exception:
print '\n Unexpected error: %s.\n' % sys.exc_info()[0] print '\n Unexpected error: %s.\n' % sys.exc_info()[0]
@ -1621,7 +1618,10 @@ def attDetect(content=None, textmsg=None, attPrefix=None, askSave=True):
except ValueError: except ValueError:
textmsg = textmsg + content[msgPos:] textmsg = textmsg + content[msgPos:]
break break
except InputException:
raise
except Exception: except Exception:
traceback.print_exc()
print '\n Unexpected error: %s.\n' % sys.exc_info()[0] print '\n Unexpected error: %s.\n' % sys.exc_info()[0]
return textmsg return textmsg
@ -1630,8 +1630,8 @@ def attDetect(content=None, textmsg=None, attPrefix=None, askSave=True):
def readSentMsg(cmd='read', msgNum=-1, messageID=None, trunck=380, withAtta=False): def readSentMsg(cmd='read', msgNum=-1, messageID=None, trunck=380, withAtta=False):
"""Opens a sent message for reading""" """Opens a sent message for reading"""
print ' All outbox messages downloading...', str(msgNum) print ' All outbox messages downloading...', str(msgNum), messageID
if messageID is None: if not messageID:
response = api.getAllSentMessages() response = api.getAllSentMessages()
if response['error'] != 0: if response['error'] != 0:
return response['errormsg'] return response['errormsg']
@ -1667,7 +1667,7 @@ def readSentMsg(cmd='read', msgNum=-1, messageID=None, trunck=380, withAtta=Fals
print ' ', 74 * '-' print ' ', 74 * '-'
if cmd == 'save': if cmd == 'save':
ret = dump2File('outbox_' + subject, textmsg, withAtta) ret = dump2File('outbox_' + subject, textmsg, True)
print ret print ret
return '' return ''
@ -1703,7 +1703,7 @@ def readMsg(cmd='read', msgNum=-1, messageID=None, trunck=380, withAtta=False):
print ' ', 74 * '-' print ' ', 74 * '-'
if cmd == 'save': if cmd == 'save':
ret = dump2File('inbox_' + subject + str(full), textmsg, withAtta) ret = dump2File('inbox_' + subject + str(full), textmsg, True)
print ret print ret
return '' return ''
@ -1800,7 +1800,7 @@ def delMsg(msgNum=-1, messageID=None):
def delSentMsg(msgNum=-1, messageID=None): def delSentMsg(msgNum=-1, messageID=None):
"""Deletes a specified message from the outbox""" """Deletes a specified message from the outbox"""
if messageID is None: if not messageID:
print ' All outbox messages downloading...', str(msgNum) print ' All outbox messages downloading...', str(msgNum)
response = api.getAllSentMessages() response = api.getAllSentMessages()
if response['error'] != 0: if response['error'] != 0:
@ -2374,7 +2374,7 @@ def UI(cmdInput=None):
elif cmdInput in cmdShorts['send']: # Sends a message or broadcast elif cmdInput in cmdShorts['send']: # Sends a message or broadcast
uInput = userInput('Would you like to send a (b)roadcast or (M)essage?').lower() uInput = userInput('Would you like to send a (b)roadcast or (M)essage?').lower()
null = '' null = None
if uInput in inputShorts['broadcast']: if uInput in inputShorts['broadcast']:
src = sendBrd(null, null, null) src = sendBrd(null, null, null)
else: else: