From e369f5e81b966e6c28b6d290b2c9d672fa3e5cc7 Mon Sep 17 00:00:00 2001 From: s3goat <41472927+dimyme@users.noreply.github.com> Date: Wed, 19 Sep 2018 18:02:05 +0200 Subject: [PATCH] new menu 9 --- src/bitmessagecurses/modding/__init__.py | 1119 ++++++++++++++++++++++ 1 file changed, 1119 insertions(+) diff --git a/src/bitmessagecurses/modding/__init__.py b/src/bitmessagecurses/modding/__init__.py index f5cac494..8611a5ee 100644 --- a/src/bitmessagecurses/modding/__init__.py +++ b/src/bitmessagecurses/modding/__init__.py @@ -1 +1,1120 @@ # drop-in replacement of the regular file for extended menu asf. +''' + Copyright (c) 2014 Luke Montalvo and the BM developers + This file adds a alternative commandline interface. useful e.g. if a distro (such as KaOS) does not ship Qt4 any longer + feel free to critique and fork + + This has been tested on Arch Linux and Linux Mint only + + Dependencies: + * from python2-pip + * python2-pythondialog + * dialog +''' + +import os +import sys +#import StringIO +from textwrap import * + +import time +#from time import strftime, localtime +from threading import Timer + +import curses +#import dialog +from dialog import Dialog +from helper_sql import * +from helper_ackPayload import genAckPayload + +from addresses import * +import ConfigParser +from bmconfigparser import BMConfigParser +from inventory import Inventory +import l10n +#from pyelliptic.openssl import OpenSSL +import queues +import shared +import shutdown + +qquit = False +menutab = 1 +menu = ["Inbox", "Send", "Sent", "Your Identities", "Subscriptions", "Address Book", "Blacklist", "Network Status", "AddOns"] +naptime = 100 +log = "" +logpad = None +inventorydata = 0 +startuptime = time.time() + +inbox = [] +inboxcur = 0 +sentbox = [] +sentcur = 0 +addresses = [] +addrcur = 0 +addrcopy = 0 +subscriptions = [] +subcur = 0 +addrbook = [] +abookcur = 0 +blacklist = [] +blackcur = 0 +bwtype = "black" + +BROADCAST_STR = "[Broadcast subscribers]" + + +class printLog: + def write(self, output): + global log + log += output + def flush(self): + pass + + +class errLog: + def write(self, output): + global log + log += "!"+output + def flush(self): + pass +printlog = printLog() +errlog = errLog() + + +def cpair(a): + r = curses.color_pair(a) + if r not in list(range(1, curses.COLOR_PAIRS-1)): + r = curses.color_pair(0) + return r + + +def ascii(s): + r = "" + for c in s: + if ord(c) in range(128): + r += c + return r + + +def drawmenu(stdscr): + menustr = " " + for i in range(0, len(menu)): + if menutab == i+1: + menustr = menustr[:-1] + menustr += "[" + menustr += str(i+1)+menu[i] + if menutab == i+1: + menustr += "] " + elif i != len(menu)-1: + menustr += " " + stdscr.addstr(2, 5, menustr, curses.A_UNDERLINE) + + +def set_background_title(d, title): + try: + d.set_background_title(title) + except: + d.add_persistent_args(("--backtitle", title)) + + +def scrollbox(d, text, height=None, width=None): + try: d.scrollbox(text, height, width, exit_label = "Continue") + except: d.msgbox( text, height or 0, width or 0, ok_label = "Continue") + + +def resetlookups(): + global inventorydata + inventorydata = Inventory().numberOfInventoryLookupsPerformed + Inventory().numberOfInventoryLookupsPerformed = 0 + Timer(1, resetlookups, ()).start() + + +def drawtab(stdscr): + if menutab in range(1, len(menu)+1): + if menutab == 1: # Inbox + stdscr.addstr(3, 5, "To", curses.A_BOLD) + stdscr.addstr(3, 40, "From", curses.A_BOLD) + stdscr.addstr(3, 80, "Subject", curses.A_BOLD) + stdscr.addstr(3, 120, "Time Received", curses.A_BOLD) + stdscr.hline(4, 5, '-', 121) + for i, item in enumerate(inbox[max(min(len(inbox)-curses.LINES+6, inboxcur-5), 0):]): + if 6+i < curses.LINES: + a = 0 + if i == inboxcur - max(min(len(inbox)-curses.LINES+6, inboxcur-5), 0): # Highlight current address + a = a | curses.A_REVERSE + if item[7] is False: # If not read, highlight + a = a | curses.A_BOLD + stdscr.addstr(5+i, 5, item[1][:34], a) + stdscr.addstr(5+i, 40, item[3][:39], a) + stdscr.addstr(5+i, 80, item[5][:39], a) + stdscr.addstr(5+i, 120, item[6][:39], a) + elif menutab == 3: # Sent + stdscr.addstr(3, 5, "To", curses.A_BOLD) + stdscr.addstr(3, 40, "From", curses.A_BOLD) + stdscr.addstr(3, 80, "Subject", curses.A_BOLD) + stdscr.addstr(3, 120, "Status", curses.A_BOLD) + stdscr.hline(4, 5, '-', 121) + for i, item in enumerate(sentbox[max(min(len(sentbox)-curses.LINES+6, sentcur-5), 0):]): + if 6+i < curses.LINES: + a = 0 + if i == sentcur - max(min(len(sentbox)-curses.LINES+6, sentcur-5), 0): # Highlight current address + a = a | curses.A_REVERSE + stdscr.addstr(5+i, 5, item[0][:34], a) + stdscr.addstr(5+i, 40, item[2][:39], a) + stdscr.addstr(5+i, 80, item[4][:39], a) + stdscr.addstr(5+i, 120, item[5][:39], a) + elif menutab == 2 or menutab == 4: # Send or Identities + stdscr.addstr(3, 5, "Label", curses.A_BOLD) + stdscr.addstr(3, 40, "Address", curses.A_BOLD) + stdscr.addstr(3, 80, "Stream", curses.A_BOLD) + stdscr.hline(4, 5, '-', 81) + for i, item in enumerate(addresses[max(min(len(addresses)-curses.LINES+6, addrcur-5), 0):]): + if 6+i < curses.LINES: + a = 0 + if i == addrcur - max(min(len(addresses)-curses.LINES+6, addrcur-5), 0): # Highlight current address + a = a | curses.A_REVERSE + if item[1] is True and item[3] not in [8,9]: # Embolden enabled, non-special addresses + a = a | curses.A_BOLD + stdscr.addstr(5+i, 5, item[0][:34], a) + stdscr.addstr(5+i, 40, item[2][:39], cpair(item[3]) | a) + stdscr.addstr(5+i, 80, str(1)[:39], a) + elif menutab == 5: # Subscriptions + stdscr.addstr(3, 5, "Label", curses.A_BOLD) + stdscr.addstr(3, 80, "Address", curses.A_BOLD) + stdscr.addstr(3, 120, "Enabled", curses.A_BOLD) + stdscr.hline(4, 5, '-', 121) + for i, item in enumerate(subscriptions[max(min(len(subscriptions)-curses.LINES+6, subcur-5), 0):]): + if 6+i < curses.LINES: + a = 0 + if i == subcur - max(min(len(subscriptions)-curses.LINES+6, subcur-5), 0): # Highlight current address + a = a | curses.A_REVERSE + if item[2] == True: # Embolden enabled subscriptions + a = a | curses.A_BOLD + stdscr.addstr(5+i, 5, item[0][:74], a) + stdscr.addstr(5+i, 80, item[1][:39], a) + stdscr.addstr(5+i, 120, str(item[2]), a) + + elif menutab == 6: # Address book + stdscr.addstr(3, 5, "Label", curses.A_BOLD) + stdscr.addstr(3, 40, "Address", curses.A_BOLD) + stdscr.hline(4, 5, '-', 41) + for i, item in enumerate(addrbook[max(min(len(addrbook)-curses.LINES+6, abookcur-5), 0):]): + if 6+i < curses.LINES: + a = 0 + if i == abookcur - max(min(len(addrbook)-curses.LINES+6, abookcur-5), 0): # Highlight current address + a = a | curses.A_REVERSE + stdscr.addstr(5+i, 5, item[0][:34], a) + stdscr.addstr(5+i, 40, item[1][:39], a) + + elif menutab == 7: # Blacklist + stdscr.addstr(3, 5, "Type: "+bwtype) + stdscr.addstr(4, 5, "Label", curses.A_BOLD) + stdscr.addstr(4, 80, "Address", curses.A_BOLD) + stdscr.addstr(4, 120, "Enabled", curses.A_BOLD) + stdscr.hline(5, 5, '-', 121) + for i, item in enumerate(blacklist[max(min(len(blacklist)-curses.LINES+6, blackcur-5), 0):]): + if 7+i < curses.LINES: + a = 0 + if i == blackcur - max(min(len(blacklist)-curses.LINES+6, blackcur-5), 0): # Highlight current address + a = a | curses.A_REVERSE + if item[2] == True: # Embolden enabled subscriptions + a = a | curses.A_BOLD + stdscr.addstr(6+i, 5, item[0][:74], a) + stdscr.addstr(6+i, 80, item[1][:39], a) + stdscr.addstr(6+i, 120, str(item[2]), a) + + + + elif menutab == 8: # Network status + # Connection data + stdscr.addstr(4, 5, "Total Connections: "+str(len(shared.connectedHostsList)).ljust(2)) + stdscr.addstr(6, 6, "Stream #", curses.A_BOLD) + stdscr.addstr(6, 18, "Connections", curses.A_BOLD) + stdscr.hline(7, 6, '-', 23) + streamcount = [] + for host, stream in list(shared.connectedHostsList.items()): + if stream >= len(streamcount): + streamcount.append(1) + else: + streamcount[stream] += 1 + for i, item in enumerate(streamcount): + if i < 4: + if i == 0: + stdscr.addstr(8+i, 6, "?") + else: + stdscr.addstr(8+i, 6, str(i)) + stdscr.addstr(8+i, 18, str(item).ljust(2)) + + # Uptime and processing data + stdscr.addstr(6, 35, "Since startup on "+l10n.formatTimestamp(startuptime, False)) + stdscr.addstr(7, 40, "Processed "+str(shared.numberOfMessagesProcessed).ljust(4)+" person-to-person messages.") + stdscr.addstr(8, 40, "Processed "+str(shared.numberOfBroadcastsProcessed).ljust(4)+" broadcast messages.") + stdscr.addstr(9, 40, "Processed "+str(shared.numberOfPubkeysProcessed).ljust(4)+" public keys.") + + # Inventory data + stdscr.addstr(11, 35, "Inventory lookups per second: "+str(inventorydata).ljust(3)) + + # Log + stdscr.addstr(13, 6, "Log", curses.A_BOLD) + n = log.count('\n') + if n > 0: + l = log.split('\n') + if n > 512: + del l[:(n-256)] + logpad.erase() + n = len(l) + for i, item in enumerate(l): + a = 0 + if len(item) > 0 and item[0] == '!': + a = curses.color_pair(1) + item = item[1:] + logpad.addstr(i, 0, item, a) + logpad.refresh(n-curses.LINES+2, 0, 14, 6, curses.LINES-2, curses.COLS-7) + + + elif menutab == 9: # AddOns special user extensions and modifications ##### #TODO + ###USER_FEATURE # see https://github.com/Bitmessage/PyBitmessage/wiki/6-Tips-&-Tricks + stdscr.addstr(5, 5, "here you can put your own menu options to extend pyBitmessage ") + stdscr.addstr(7, 5, "hit q to quit ") + stdscr.hline( 11, 5, '-', 121) + + + stdscr.refresh() + + + + + +def redraw(stdscr): + stdscr.erase() + stdscr.border() + drawmenu(stdscr) + stdscr.refresh() + + +def dialogreset(stdscr): + stdscr.clear() + stdscr.keypad(1) + curses.curs_set(0) + + +def handlech(c, stdscr): + try: + addr="" + if c != curses.ERR: + global inboxcur, addrcur, sentcur, subcur, abookcur, blackcur + if c in range(256): + if chr(c) in '123456789': + global menutab + menutab = int(chr(c)) + elif chr(c) == 'q': + global qquit + qquit = True + elif chr(c) == '\n': + curses.curs_set(1) + d = Dialog(dialog="dialog") + if menutab == 1: + set_background_title(d, "Inbox Message Dialog Box") + r, t = d.menu("Do what with \""+inbox[inboxcur][5]+"\" from \""+inbox[inboxcur][3]+"\"?", + choices=[("1", "View message"), + ("2", "Mark message as unread"), + ("3", "Reply"), + ("4", "Add sender to Address Book"), + ("5", "Save message as text file"), + ("6", "Move to trash")]) + if r == d.DIALOG_OK: + if t == "1": # View + set_background_title(d, "\""+inbox[inboxcur][5]+"\" from \""+inbox[inboxcur][3]+"\" to \""+inbox[inboxcur][1]+"\"") + data = "" + ret = sqlQuery("SELECT message FROM inbox WHERE msgid=?", inbox[inboxcur][0]) + if ret != []: + for row in ret: + data, = row + data = shared.fixPotentiallyInvalidUTF8Data(data) + msg = "" + for i, item in enumerate(data.split("\n")): + msg += fill(item, replace_whitespace=False)+"\n" + scrollbox(d, str(ascii(msg)), 30, 80) + sqlExecute("UPDATE inbox SET read=1 WHERE msgid=?", inbox[inboxcur][0]) + inbox[inboxcur][7] = 1 + scrollbox(d, str("Could not fetch message.")) + #scrollbox(d, unicode("Could not fetch message.")) + elif t == "2": # Mark unread + sqlExecute("UPDATE inbox SET read=0 WHERE msgid=?", inbox[inboxcur][0]) + inbox[inboxcur][7] = 0 + elif t == "3": # Reply + curses.curs_set(1) + m = inbox[inboxcur] + fromaddr = m[4] + ischan = False + for i, item in enumerate(addresses): + if fromaddr == item[2] and item[3] != 0: + ischan = True + break + if not addresses[i][1]: + scrollbox(d, str("Sending address disabled, please either enable it or choose a different address.")) + return + toaddr = m[2] + if ischan: + toaddr = fromaddr + + subject = m[5] + if not m[5][:4] == "Re: ": + subject = "Re: "+m[5] + body = "" + ret = sqlQuery("SELECT message FROM inbox WHERE msgid=?", m[0]) + if ret != []: + body = "\n\n------------------------------------------------------\n" + for row in ret: + body, = row + + sendMessage(fromaddr, toaddr, ischan, subject, body, True) + dialogreset(stdscr) + elif t == "4": # Add to Address Book + addr = inbox[inboxcur][4] + if addr not in [item[1] for i,item in enumerate(addrbook)]: + r, t = d.inputbox("Label for address \""+addr+"\"") + if r == d.DIALOG_OK: + label = t + sqlExecute("INSERT INTO addressbook VALUES (?,?)", label, addr) + # Prepend entry + addrbook.reverse() + addrbook.append([label, addr]) + addrbook.reverse() + else: + scrollbox(d, str("The selected address is already in the Address Book.")) + elif t == "5": # Save message + set_background_title(d, "Save \""+inbox[inboxcur][5]+"\" as text file") + r, t = d.inputbox("Filename", init=inbox[inboxcur][5]+".txt") + if r == d.DIALOG_OK: + msg = "" + ret = sqlQuery("SELECT message FROM inbox WHERE msgid=?", inbox[inboxcur][0]) + if ret != []: + for row in ret: + msg, = row + fh = open(t, "a") # Open in append mode just in case + fh.write(msg) + fh.close() + else: + scrollbox(d, str("Could not fetch message.")) + elif t == "6": # Move to trash + sqlExecute("UPDATE inbox SET folder='trash' WHERE msgid=?", inbox[inboxcur][0]) + scrollbox(d, str( "Message moved to trash. There is no interface to view your trash, \nbut the message is still on disk if you are desperate to recover it.")) + #scrollbox(d, unicode("Message moved to trash. There is no interface to view your trash, \nbut the message is still on disk if you are desperate to recover it.")) + elif menutab == 2: + a = "" + if addresses[addrcur][3] != 0: # if current address is a chan + a = addresses[addrcur][2] + sendMessage(addresses[addrcur][2], a) + elif menutab == 3: + set_background_title(d, "Sent Messages Dialog Box") + r, t = d.menu("Do what with \""+sentbox[sentcur][4]+"\" to \""+sentbox[sentcur][0]+"\"?", + choices=[("1", "View message"), + ("2", "Move to trash")]) + if r == d.DIALOG_OK: + if t == "1": # View + set_background_title(d, "\""+sentbox[sentcur][4]+"\" from \""+sentbox[sentcur][3]+"\" to \""+sentbox[sentcur][1]+"\"") + data = "" + ret = sqlQuery("SELECT message FROM sent WHERE subject=? AND ackdata=?", sentbox[sentcur][4], sentbox[sentcur][6]) + if ret != []: + for row in ret: + data, = row + data = shared.fixPotentiallyInvalidUTF8Data(data) + scrollbox(d, str(ascii(msg)), 30, 80) + msg += fill(item, replace_whitespace=False)+"\n" + scrollbox(d, unicode(ascii(msg)), 30, 80) + #scrollbox(d, unicode("Could not fetch message.")) + scrollbox(d, str("Could not fetch message.")) + elif t == "2": # Move to trash + sqlExecute("UPDATE sent SET folder='trash' WHERE subject=? AND ackdata=?", sentbox[sentcur][4], sentbox[sentcur][6]) + del sentbox[sentcur] + #scrollbox(d, unicode("Message moved to trash. There is no interface to view your trash, \nbut the message is still on disk if you are desperate to recover it.")) + scrollbox(d, str( "Message moved to trash. There is no interface to view your trash, \nbut the message is still on disk if you are desperate to recover it.")) + elif menutab == 4: + set_background_title(d, "Your Identities Dialog Box") + if len(addresses) <= addrcur: + r, t = d.menu("Do what with addresses?", + choices=[("1", "Create new address")]) + else: + r, t = d.menu("Do what with \""+addresses[addrcur][0]+"\" : \""+addresses[addrcur][2]+"\"?", + choices=[("1", "Create new address"), + ("2", "Send a message from this address"), + ("3", "Rename"), + ("4", "Enable"), + ("5", "Disable"), + ("6", "Delete"), + ("7", "Special address behavior")]) + if r == d.DIALOG_OK: + if t == "1": # Create new address + set_background_title(d, "Create new address") + #scrollbox(d, str( "Here you may generate as many addresses as you like.\n" + scrollbox(d, unicode("Here you may generate as many addresses as you like.\n" + "Indeed, creating and abandoning addresses is encouraged.\n" + "Deterministic addresses have several pros and cons:\n" + "\nPros:\n" + " * You can recreate your addresses on any computer from memory\n" + " * You need not worry about backing up your keys.dat file as long as you \n can remember your passphrase\n" + "Cons:\n" + " * You must remember (or write down) your passphrase in order to recreate \n your keys if they are lost\n" + " * You must also remember the address version and stream numbers\n" + " * If you choose a weak passphrase someone may be able to brute-force it \n and then send and receive messages as you")) + r, t = d.menu("Choose an address generation technique", + choices=[("1", "Use a random number generator"), + ("2", "Use a passphrase")]) + if r == d.DIALOG_OK: + if t == "1": + set_background_title(d, "Randomly generate address") + r, t = d.inputbox("Label (not shown to anyone except you)") + label = "" + if r == d.DIALOG_OK and len(t) > 0: + label = t + r, t = d.menu("Choose a stream", + choices=[("1", "Use the most available stream"),("", "(Best if this is the first of many addresses you will create)"), + ("2", "Use the same stream as an existing address"),("", "(Saves you some bandwidth and processing power)")]) + if r == d.DIALOG_OK: + if t == "1": + stream = 1 + elif t == "2": + addrs = [] + for i, item in enumerate(addresses): + addrs.append([str(i), item[2]]) + r, t = d.menu("Choose an existing address's stream", choices=addrs) + if r == d.DIALOG_OK: + stream = decodeAddress(addrs[int(t)][1])[2] + shorten = False + r, t = d.checklist("Miscellaneous options", + choices=[("1", "Spend time shortening the address", 1 if shorten else 0)]) + if r == d.DIALOG_OK and "1" in t: + shorten = True + queues.addressGeneratorQueue.put(("createRandomAddress", 4, stream, label, 1, "", shorten)) + elif t == "2": + set_background_title(d, "Make deterministic addresses") + r, t = d.passwordform("Enter passphrase", + [("Passphrase", 1, 1, "", 2, 1, 64, 128), + ("Confirm passphrase", 3, 1, "", 4, 1, 64, 128)], + form_height=4, insecure=True) + if r == d.DIALOG_OK: + if t[0] == t[1]: + passphrase = t[0] + r, t = d.rangebox("Number of addresses to generate", width=48, min=1, max=99, init=8) + if r == d.DIALOG_OK: + number = t + shorten = False + r, t = d.checklist("Miscellaneous options", + choices=[("1", "Spend time shortening the address", 1 if shorten else 0)]) + if r == d.DIALOG_OK and "1" in t: + shorten = True + #scrollbox(d, unicode("In addition to your passphrase, be sure to remember the following numbers:\n" + scrollbox(d, str("In addition to your passphrase, be sure to remember the following numbers:\n" + "\n * Address version number: "+str(4)+"\n" + " * Stream number: "+str(stream))) + queues.addressGeneratorQueue.put(('createDeterministicAddresses', 4, stream, "unused deterministic address", number, str(passphrase), shorten)) + else: + scrollbox(d, str("Passphrases do not match")) + #scrollbox(d, unicode("Passphrases do not match")) + elif t == "2": # Send a message + a = "" + if addresses[addrcur][3] != 0: # if current address is a chan + a = addresses[addrcur][2] + sendMessage(addresses[addrcur][2], a) + elif t == "3": # Rename address label + a = addresses[addrcur][2] + label = addresses[addrcur][0] + r, t = d.inputbox("New address label", init=label) + if r == d.DIALOG_OK: + label = t + BMConfigParser().set(a, "label", label) + # Write config + BMConfigParser().save() + addresses[addrcur][0] = label + elif t == "4": # Enable address + a = addresses[addrcur][2] + BMConfigParser().set(a, "enabled", "true") # Set config + # Write config + BMConfigParser().save() + # Change color + if BMConfigParser().safeGetBoolean(a, 'chan'): + addresses[addrcur][3] = 9 # orange + elif BMConfigParser().safeGetBoolean(a, 'mailinglist'): + addresses[addrcur][3] = 5 # magenta + else: + addresses[addrcur][3] = 0 # black + addresses[addrcur][1] = True + shared.reloadMyAddressHashes() # Reload address hashes + elif t == "5": # Disable address + a = addresses[addrcur][2] + BMConfigParser().set(a, "enabled", "false") # Set config + addresses[addrcur][3] = 8 # Set color to gray + # Write config + BMConfigParser().save() + shared.reloadMyAddressHashes() # Reload address hashes + elif t == "6": # Delete address + r, t = d.inputbox("Type in \"I want to delete this address\"", width=50) + if r == d.DIALOG_OK and t == "I want to delete this address": + BMConfigParser().remove_section(addresses[addrcur][2]) + BMConfigParser().save() + del addresses[addrcur] + elif t == "7": # Special address behavior + a = addresses[addrcur][2] + set_background_title(d, "Special address behavior") + if BMConfigParser().safeGetBoolean(a, "chan"): + #scrollbox(d, unicode("This is a chan address. You cannot use it as a pseudo-mailing list.")) + scrollbox(d, str("This is a chan address. You cannot use it as a pseudo-mailing list.")) + else: + m = BMConfigParser().safeGetBoolean(a, "mailinglist") + r, t = d.radiolist("Select address behavior", choices=[("1", "Behave as a normal address", not m), ("2", "Behave as a pseudo-mailing-list address", m)]) + if r == d.DIALOG_OK: + if t == "1" and m == True: + BMConfigParser().set(a, "mailinglist", "false") + if addresses[addrcur][1]: + addresses[addrcur][3] = 0 # Set color to black + else: + addresses[addrcur][3] = 8 # Set color to gray + elif t == "2" and m == False: + try: + mn = BMConfigParser().get(a, "mailinglistname") + #except ConfigParser.NoOptionError: + except ConfigParser.NoOptionError: + mn = "" + r, t = d.inputbox("Mailing list name", init=mn) + if r == d.DIALOG_OK: + mn = t + BMConfigParser().set(a, "mailinglist", "true") + BMConfigParser().set(a, "mailinglistname", mn) + addresses[addrcur][3] = 6 # Set color to magenta + # Write config + BMConfigParser().save() + elif menutab == 5: + set_background_title(d, "Subscriptions Dialog Box") + if len(subscriptions) <= subcur: + r, t = d.menu("Do what with subscription to \""+subscriptions[subcur][0]+"\"?", + choices=[("1", "Add new subscription")]) + else: + r, t = d.menu("Do what with subscription to \""+subscriptions[subcur][0]+"\"?", + choices=[("1", "Add new subscription"), + ("2", "Delete this subscription"), + ("3", "Enable"), + ("4", "Disable")]) + if r == d.DIALOG_OK: + if t == "1": + r, t = d.inputbox("New subscription address") + if r == d.DIALOG_OK: + addr = addBMIfNotPresent(t) + if not shared.isAddressInMySubscriptionsList(addr): + r, t = d.inputbox("New subscription label") + if r == d.DIALOG_OK: + label = t + # Prepend entry + subscriptions.reverse() + subscriptions.append([label, addr, True]) + subscriptions.reverse() + + sqlExecute("INSERT INTO subscriptions VALUES (?,?,?)", label, addr, True) + shared.reloadBroadcastSendersForWhichImWatching() + elif t == "2": + r, t = d.inpuxbox("Type in \"I want to delete this subscription\"") + if r == d.DIALOG_OK and t == "I want to delete this subscription": + sqlExecute("DELETE FROM subscriptions WHERE label=? AND address=?", subscriptions[subcur][0], subscriptions[subcur][1]) + shared.reloadBroadcastSendersForWhichImWatching() + del subscriptions[subcur] + elif t == "3": + sqlExecute("UPDATE subscriptions SET enabled=1 WHERE label=? AND address=?", subscriptions[subcur][0], subscriptions[subcur][1]) + shared.reloadBroadcastSendersForWhichImWatching() + subscriptions[subcur][2] = True + elif t == "4": + sqlExecute("UPDATE subscriptions SET enabled=0 WHERE label=? AND address=?", subscriptions[subcur][0], subscriptions[subcur][1]) + shared.reloadBroadcastSendersForWhichImWatching() + subscriptions[subcur][2] = False + elif menutab == 6: + set_background_title(d, "Address Book Dialog Box") + if len(addrbook) <= abookcur: + r, t = d.menu("Do what with addressbook?", + choices=[("3", "Add new address to Address Book")]) + else: + r, t = d.menu("Do what with \""+addrbook[abookcur][0]+"\" : \""+addrbook[abookcur][1]+"\"", + choices=[("1", "Send a message to this address"), + ("2", "Subscribe to this address"), + ("3", "Add new address to Address Book"), + ("4", "Delete this address")]) + if r == d.DIALOG_OK: + if t == "1": + sendMessage(recv=addrbook[abookcur][1]) + elif t == "2": + r, t = d.inputbox("New subscription label") + if r == d.DIALOG_OK: + label = t + # Prepend entry + subscriptions.reverse() + subscriptions.append([label, addr, True]) + subscriptions.reverse() + + shared.reloadBroadcastSendersForWhichImWatching() + elif t == "3": + r, t = d.inputbox("Input new address") + if r == d.DIALOG_OK: + addr = t + if addr not in [item[1] for i,item in enumerate(addrbook)]: + r, t = d.inputbox("Label for address \""+addr+"\"") + if r == d.DIALOG_OK: + sqlExecute("INSERT INTO addressbook VALUES (?,?)", t, addr) + # Prepend entry + addrbook.reverse() + addrbook.append([t, addr]) + addrbook.reverse() + else: + #scrollbox(d, unicode("The selected address is already in the Address Book.")) + scrollbox(d, str("The selected address is already in the Address Book.")) + elif t == "4": + r, t = d.inputbox("Type in \"I want to delete this Address Book entry\"") + if r == d.DIALOG_OK and t == "I want to delete this Address Book entry": + sqlExecute("DELETE FROM addressbook WHERE label=? AND address=?", addrbook[abookcur][0], addrbook[abookcur][1]) + del addrbook[abookcur] + elif menutab == 7: + set_background_title(d, "Blacklist Dialog Box") + r, t = d.menu("Do what with \""+blacklist[blackcur][0]+"\" : \""+blacklist[blackcur][1]+"\"?", + choices=[("1", "Delete"), + ("2", "Enable"), + ("3", "Disable")]) + if r == d.DIALOG_OK: + if t == "1": + r, t = d.inputbox("Type in \"I want to delete this Blacklist entry\"") + if r == d.DIALOG_OK and t == "I want to delete this Blacklist entry": + sqlExecute("DELETE FROM blacklist WHERE label=? AND address=?", blacklist[blackcur][0], blacklist[blackcur][1]) + del blacklist[blackcur] + elif t == "2": + sqlExecute("UPDATE blacklist SET enabled=1 WHERE label=? AND address=?", blacklist[blackcur][0], blacklist[blackcur][1]) + blacklist[blackcur][2] = True + elif t== "3": + sqlExecute("UPDATE blacklist SET enabled=0 WHERE label=? AND address=?", blacklist[blackcur][0], blacklist[blackcur][1]) + blacklist[blackcur][2] = False + dialogreset(stdscr) + else: + if c == curses.KEY_UP: + if menutab == 1 and inboxcur > 0: + inboxcur -= 1 + if (menutab == 2 or menutab == 4) and addrcur > 0: + addrcur -= 1 + if menutab == 3 and sentcur > 0: + sentcur -= 1 + if menutab == 5 and subcur > 0: + subcur -= 1 + if menutab == 6 and abookcur > 0: + abookcur -= 1 + if menutab == 7 and blackcur > 0: + blackcur -= 1 + elif c == curses.KEY_DOWN: + if menutab == 1 and inboxcur < len(inbox)-1: + inboxcur += 1 + if (menutab == 2 or menutab == 4) and addrcur < len(addresses)-1: + addrcur += 1 + if menutab == 3 and sentcur < len(sentbox)-1: + sentcur += 1 + if menutab == 5 and subcur < len(subscriptions)-1: + subcur += 1 + if menutab == 6 and abookcur < len(addrbook)-1: + abookcur += 1 + if menutab == 7 and blackcur < len(blacklist)-1: + blackcur += 1 + elif c == curses.KEY_HOME: + if menutab == 1: + inboxcur = 0 + if menutab == 2 or menutab == 4: + addrcur = 0 + if menutab == 3: + sentcur = 0 + if menutab == 5: + subcur = 0 + if menutab == 6: + abookcur = 0 + if menutab == 7: + blackcur = 0 + elif c == curses.KEY_END: + if menutab == 1: + inboxcur = len(inbox)-1 + if menutab == 2 or menutab == 4: + addrcur = len(addresses)-1 + if menutab == 3: + sentcur = len(sentbox)-1 + if menutab == 5: + subcur = len(subscriptions)-1 + if menutab == 6: + abookcur = len(addrbook)-1 + if menutab == 7: + blackcur = len(blackcur)-1 + + redraw(stdscr) + + + except: + pass # else the screen gets corrupted + print("some exception just occurred ") + #redraw(stdscr) + + +def sendMessage(sender="", recv="", broadcast=None, subject="", body="", reply=False): + try: + global streamNumber ### TODO necess ?? + streamNumber = 0 + if sender == "": + return + d = Dialog(dialog="dialog") + set_background_title(d, "Send a message") + if recv == "": + r, t = d.inputbox("Recipient address (Cancel to load from the Address Book or leave blank to broadcast)", 10, 60) + if r != d.DIALOG_OK: + global menutab + menutab = 6 + return + recv = t + if broadcast == None and sender != recv: + r, t = d.radiolist("How to send the message?", + choices=[("1", "Send to one or more specific people", 1), + ("2", "Broadcast to everyone who is subscribed to your address", 0)]) + if r != d.DIALOG_OK: + return + broadcast = False + if t == "2": # Broadcast + broadcast = True + if subject == "" or reply: + r, t = d.inputbox("Message subject", width=80, init=subject) + if r != d.DIALOG_OK: + return + subject = t + if body == "" or reply: + r, t = d.inputbox("Message body", 23, 120, init=body) + if r != d.DIALOG_OK: + return + body = t + body = body.replace("\\n", "\n").replace("\\t", "\t") + + + if not broadcast: + recvlist = [] + for i, item in enumerate(recv.replace(",", ";").split(";")): + recvlist.append(item.strip()) + list(set(recvlist)) # Remove exact duplicates + for addr in recvlist: + if addr != "": + status, version, stream, ripe = decodeAddress(addr) + if status != "success": + set_background_title(d, "Recipient address error") + scrollbox(d, str(err)) + if status == "missingbm": + err += "Bitmessage addresses should start with \"BM-\"." + elif status == "checksumfailed": + err += "The address was not typed or copied correctly." + elif status == "invalidcharacters": + err += "The address contains invalid characters." + elif status == "ripetooshort": + err += "Some data encoded in the address is too short. There might be something wrong with the software of your acquaintance." + elif status == "ripetoolong": + err += "Some data encoded in the address is too long. There might be something wrong with the software of your acquaintance." + elif status == "varintmalformed": + err += "Some data encoded in the address is malformed. There might be something wrong with the software of your acquaintance." + else: + err += "It is unknown what is wrong with the address." + scrollbox(d, unicode(err)) + else: + addr = addBMIfNotPresent(addr) + if version > 4 or version <= 1: + set_background_title(d, "Recipient address error") + scrollbox(d, str("Could not understand version number " + version + "of address" + addr + ".")) + continue + if stream > 1 or stream == 0: + set_background_title(d, "Recipient address error") + scrollbox(d, str("Bitmessage currently only supports stream numbers of 1, unlike as requested for address " + addr + ".")) + continue + if len(shared.connectedHostsList) == 0: + set_background_title(d, "Not connected warning") + scrollbox(d, str("Because you are not currently connected to the network, no sending will actually happen now."), 8 , 120) + # scrollbox(d, str("You unicode("Because + + + + #stealthLevel = BMConfigParser().safeGetInt('bitmessagesettings', 'ackstealthlevel') + sqlExecute( + "INSERT INTO sent VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)", + "", + addr, + ripe, + sender, + subject, + body, + ackdata, + int(time.time()), # sentTime (this will never change) + int(time.time()), # lastActionTime + 0, # sleepTill time. This will get set when the POW gets done. + "msgqueued", + 0, # retryNumber + "sent", + 2, # encodingType + BMConfigParser().getint('bitmessagesettings', 'ttl')) + queues.workerQueue.put(("sendmessage", addr)) + else: # Broadcast + if recv == "": + set_background_title(d, "Empty sender error") + scrollbox(d, unicode("You must specify an address to send the message from.")) + else: + # dummy ackdata, no need for stealth + ackdata = genAckPayload(streamNumber, 0) + recv = BROADCAST_STR + ripe = "" + sqlExecute( + "INSERT INTO sent VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)", + "", + recv, + ripe, + sender, + subject, + body, + ackdata, + int(time.time()), # sentTime (this will never change) + int(time.time()), # lastActionTime + 0, # sleepTill time. This will get set when the POW gets done. + "broadcastqueued", + 0, # retryNumber + "sent", # folder + 2, # encodingType + BMConfigParser().getint('bitmessagesettings', 'ttl')) + queues.workerQueue.put(('sendbroadcast', '')) + + except: + d.msgbox(" sendmessage exception error of some kind " , 10 , 70 , ok_label = "Continue") + + + + + + +def loadInbox(): + sys.stdout = sys.__stdout__ + print("Loading inbox messages...") + sys.stdout = printlog + + where = "toaddress || fromaddress || subject || message" + what = "%%" + ret = sqlQuery("""SELECT msgid, toaddress, fromaddress, subject, received, read + FROM inbox WHERE folder='inbox' AND %s LIKE ? + ORDER BY received + """ % (where,), what) + for row in ret: + msgid, toaddr, fromaddr, subject, received, read = row + subject = ascii(shared.fixPotentiallyInvalidUTF8Data(subject)) + + # Set label for to address + try: + if toaddr == BROADCAST_STR: + tolabel = BROADCAST_STR + else: + tolabel = BMConfigParser().get(toaddr, "label") + except: + tolabel = "" + if tolabel == "": + tolabel = toaddr + tolabel = shared.fixPotentiallyInvalidUTF8Data(tolabel) + + # Set label for from address + fromlabel = "" + if BMConfigParser().has_section(fromaddr): + fromlabel = BMConfigParser().get(fromaddr, "label") + if fromlabel == "": # Check Address Book + qr = sqlQuery("SELECT label FROM addressbook WHERE address=?", fromaddr) + if qr != []: + for r in qr: + fromlabel, = r + if fromlabel == "": # Check Subscriptions + qr = sqlQuery("SELECT label FROM subscriptions WHERE address=?", fromaddr) + if qr != []: + for r in qr: + fromlabel, = r + if fromlabel == "": + fromlabel = fromaddr + fromlabel = shared.fixPotentiallyInvalidUTF8Data(fromlabel) + + # Load into array + inbox.append([msgid, tolabel, toaddr, fromlabel, fromaddr, subject, + l10n.formatTimestamp(received, False), read]) + inbox.reverse() +def loadSent(): + sys.stdout = sys.__stdout__ + print("Loading sent messages...") + sys.stdout = printlog + + where = "toaddress || fromaddress || subject || message" + what = "%%" + ret = sqlQuery("""SELECT toaddress, fromaddress, subject, status, ackdata, lastactiontime + FROM sent WHERE folder='sent' AND %s LIKE ? + ORDER BY lastactiontime + """ % (where,), what) + for row in ret: + toaddr, fromaddr, subject, status, ackdata, lastactiontime = row + subject = ascii(shared.fixPotentiallyInvalidUTF8Data(subject)) + + # Set label for to address + tolabel = "" + qr = sqlQuery("SELECT label FROM addressbook WHERE address=?", toaddr) + if qr != []: + for r in qr: + tolabel, = r + if tolabel == "": + qr = sqlQuery("SELECT label FROM subscriptions WHERE address=?", toaddr) + if qr != []: + for r in qr: + tolabel, = r + if tolabel == "": + if BMConfigParser().has_section(toaddr): + tolabel = BMConfigParser().get(toaddr, "label") + if tolabel == "": + tolabel = toaddr + + # Set label for from address + fromlabel = "" + if BMConfigParser().has_section(fromaddr): + fromlabel = BMConfigParser().get(fromaddr, "label") + if fromlabel == "": + fromlabel = fromaddr + + # Set status string + if status == "awaitingpubkey": + statstr = "Waiting for their public key. Will request it again soon" + elif status == "doingpowforpubkey": + statstr = "Encryption key request queued" + elif status == "msgqueued": + statstr = "Message queued" + elif status == "msgsent": + t = l10n.formatTimestamp(lastactiontime, False) + statstr = "Message sent at "+t+".Waiting for acknowledgement." + elif status == "msgsentnoackexpected": + t = l10n.formatTimestamp(lastactiontime, False) + statstr = "Message sent at "+t+"." + elif status == "doingmsgpow": + statstr = "The proof of work required to send the message has been queued." + elif status == "ackreceived": + t = l10n.formatTimestamp(lastactiontime, False) + statstr = "Acknowledgment of the message received at "+t+"." + elif status == "broadcastqueued": + statstr = "Broadcast queued." + elif status == "broadcastsent": + t = l10n.formatTimestamp(lastactiontime, False) + statstr = "Broadcast sent at "+t+"." + elif status == "forcepow": + statstr = "Forced difficulty override. Message will start sending soon." + elif status == "badkey": + statstr = "Warning: Could not encrypt message because the recipient's encryption key is no good." + elif status == "toodifficult": + statstr = "Error: The work demanded by the recipient is more difficult than you are willing to do." + else: + t = l10n.formatTimestamp(lastactiontime, False) + statstr = "Unknown status "+status+" at "+t+"." + + # Load into array + sentbox.append([tolabel, toaddr, fromlabel, fromaddr, subject, statstr, ackdata, + l10n.formatTimestamp(lastactiontime, False)]) + sentbox.reverse() +def loadAddrBook(): + sys.stdout = sys.__stdout__ + print("Loading address book...") + sys.stdout = printlog + + ret = sqlQuery("SELECT label, address FROM addressbook") + for row in ret: + label, addr = row + label = shared.fixPotentiallyInvalidUTF8Data(label) + addrbook.append([label, addr]) + addrbook.reverse() +def loadSubscriptions(): + ret = sqlQuery("SELECT label, address, enabled FROM subscriptions") + for row in ret: + label, address, enabled = row + subscriptions.append([label, address, enabled]) + subscriptions.reverse() +def loadBlackWhiteList(): + global bwtype + bwtype = BMConfigParser().get("bitmessagesettings", "blackwhitelist") + if bwtype == "black": + ret = sqlQuery("SELECT label, address, enabled FROM blacklist") + else: + ret = sqlQuery("SELECT label, address, enabled FROM whitelist") + for row in ret: + label, address, enabled = row + blacklist.append([label, address, enabled]) + blacklist.reverse() + +def runwrapper(): + sys.stdout = printlog + #sys.stderr = errlog + + # Load messages from database + loadInbox() + loadSent() + loadAddrBook() + loadSubscriptions() + loadBlackWhiteList() + + stdscr = curses.initscr() + + global logpad + logpad = curses.newpad(1024, curses.COLS) + + stdscr.nodelay(0) + curses.curs_set(0) + stdscr.timeout(1000) + + curses.wrapper(run) + doShutdown() + +def run(stdscr): + # Schedule inventory lookup data + resetlookups() + + # Init color pairs + if curses.has_colors(): + curses.init_pair(1, curses.COLOR_RED, curses.COLOR_BLACK) # red + curses.init_pair(2, curses.COLOR_GREEN, curses.COLOR_BLACK) # green + curses.init_pair(3, curses.COLOR_YELLOW, curses.COLOR_BLACK) # yellow + curses.init_pair(4, curses.COLOR_BLUE, curses.COLOR_BLACK) # blue + curses.init_pair(5, curses.COLOR_MAGENTA, curses.COLOR_BLACK) # magenta + curses.init_pair(6, curses.COLOR_CYAN, curses.COLOR_BLACK) # cyan + curses.init_pair(7, curses.COLOR_WHITE, curses.COLOR_BLACK) # white + if curses.can_change_color(): + curses.init_color(8, 500, 500, 500) # gray + curses.init_pair(8, 8, 0) + curses.init_color(9, 844, 465, 0) # orange + curses.init_pair(9, 9, 0) + else: + curses.init_pair(8, curses.COLOR_WHITE, curses.COLOR_BLACK) # grayish + curses.init_pair(9, curses.COLOR_YELLOW, curses.COLOR_BLACK) # orangish + + # Init list of address in 'Your Identities' tab + configSections = BMConfigParser().addresses() + for addressInKeysFile in configSections: + isEnabled = BMConfigParser().getboolean(addressInKeysFile, "enabled") + addresses.append([BMConfigParser().get(addressInKeysFile, "label"), isEnabled, addressInKeysFile]) + # Set address color + if not isEnabled: + addresses[len(addresses)-1].append(8) # gray + elif BMConfigParser().safeGetBoolean(addressInKeysFile, 'chan'): + addresses[len(addresses)-1].append(9) # orange + elif BMConfigParser().safeGetBoolean(addressInKeysFile, 'mailinglist'): + addresses[len(addresses)-1].append(5) # magenta + else: + addresses[len(addresses)-1].append(0) # black + addresses.reverse() + + stdscr.clear() + redraw(stdscr) + while qquit == False: + drawtab(stdscr) + handlech(stdscr.getch(), stdscr) + + +def doShutdown(): + sys.stdout = sys.__stdout__ + print("Shutting down with err code 0 ") + sys.stdout = printlog + shutdown.doCleanShutdown() + sys.stdout = sys.__stdout__ + sys.stderr = sys.__stderr__ + + os._exit(0)