2015-01-21 17:38:25 +00:00
from __future__ import division
2013-06-21 21:32:22 +00:00
import threading
import shared
import time
2013-06-24 20:25:31 +00:00
from time import strftime , localtime , gmtime
2013-06-21 21:32:22 +00:00
import random
2013-09-30 03:01:56 +00:00
from subprocess import call # used when the API must execute an outside program
2013-06-21 21:32:22 +00:00
from addresses import *
import highlevelcrypto
import proofofwork
2013-06-23 19:52:39 +00:00
import sys
2013-06-24 19:51:01 +00:00
import tr
2017-02-22 08:34:54 +00:00
from bmconfigparser import BMConfigParser
2013-08-08 19:37:48 +00:00
from debug import logger
2017-02-08 19:37:42 +00:00
import defaults
2013-08-29 11:27:30 +00:00
from helper_sql import *
2013-09-30 03:01:56 +00:00
import helper_inbox
2014-05-02 14:46:36 +00:00
from helper_generic import addDataPadding
2016-11-14 19:23:58 +00:00
import helper_msgcoding
2015-11-24 00:55:17 +00:00
from helper_threading import *
2017-01-19 18:48:12 +00:00
from inventory import Inventory , PendingUpload
2014-08-06 02:01:01 +00:00
import l10n
2017-01-11 13:27:19 +00:00
import protocol
2017-02-08 12:41:56 +00:00
import queues
2017-01-14 22:20:15 +00:00
import state
2016-03-23 22:26:57 +00:00
from binascii import hexlify , unhexlify
2013-06-21 21:32:22 +00:00
# This thread, of which there is only one, does the heavy lifting:
# calculating POWs.
2015-11-26 01:38:55 +00:00
def sizeof_fmt ( num , suffix = ' h/s ' ) :
for unit in [ ' ' , ' k ' , ' M ' , ' G ' , ' T ' , ' P ' , ' E ' , ' Z ' ] :
if abs ( num ) < 1000.0 :
return " %3.1f %s %s " % ( num , unit , suffix )
num / = 1024.0
return " %.1f %s %s " % ( num , ' Yi ' , suffix )
2013-06-21 21:32:22 +00:00
2015-11-24 00:55:17 +00:00
class singleWorker ( threading . Thread , StoppableThread ) :
2013-06-21 21:32:22 +00:00
def __init__ ( self ) :
# QThread.__init__(self, parent)
2015-11-18 15:22:17 +00:00
threading . Thread . __init__ ( self , name = " singleWorker " )
2015-11-24 00:55:17 +00:00
self . initStop ( )
2017-08-15 10:24:43 +00:00
proofofwork . init ( )
2015-11-24 00:55:17 +00:00
def stopThread ( self ) :
try :
2017-02-08 12:41:56 +00:00
queues . workerQueue . put ( ( " stopThread " , " data " ) )
2015-11-24 00:55:17 +00:00
except :
pass
super ( singleWorker , self ) . stopThread ( )
2013-06-21 21:32:22 +00:00
def run ( self ) :
2017-02-26 19:44:56 +00:00
while not state . sqlReady and state . shutdown == 0 :
self . stop . wait ( 2 )
if state . shutdown > 0 :
return
2015-03-09 06:35:32 +00:00
# Initialize the neededPubkeys dictionary.
2013-08-29 11:27:30 +00:00
queryreturn = sqlQuery (
2014-08-27 07:14:32 +00:00
''' SELECT DISTINCT toaddress FROM sent WHERE (status= ' awaitingpubkey ' AND folder= ' sent ' ) ''' )
2013-06-21 21:32:22 +00:00
for row in queryreturn :
2014-08-27 07:14:32 +00:00
toAddress , = row
toStatus , toAddressVersionNumber , toStreamNumber , toRipe = decodeAddress ( toAddress )
2013-09-13 04:27:34 +00:00
if toAddressVersionNumber < = 3 :
2017-01-14 22:20:15 +00:00
state . neededPubkeys [ toAddress ] = 0
2013-09-13 04:27:34 +00:00
elif toAddressVersionNumber > = 4 :
2013-09-15 01:06:26 +00:00
doubleHashOfAddressData = hashlib . sha512 ( hashlib . sha512 ( encodeVarint (
toAddressVersionNumber ) + encodeVarint ( toStreamNumber ) + toRipe ) . digest ( ) ) . digest ( )
privEncryptionKey = doubleHashOfAddressData [ : 32 ] # Note that this is the first half of the sha512 hash.
tag = doubleHashOfAddressData [ 32 : ]
2017-01-14 22:20:15 +00:00
state . neededPubkeys [ tag ] = ( toAddress , highlevelcrypto . makeCryptor ( hexlify ( privEncryptionKey ) ) ) # We'll need this for when we receive a pubkey reply: it will be encrypted and we'll need to decrypt it.
2013-06-21 21:32:22 +00:00
2014-08-27 07:14:32 +00:00
# Initialize the shared.ackdataForWhichImWatching data structure
2013-08-29 11:27:30 +00:00
queryreturn = sqlQuery (
2016-10-05 18:06:47 +00:00
''' SELECT ackdata FROM sent WHERE status = ' msgsent ' ''' )
2013-06-21 21:32:22 +00:00
for row in queryreturn :
ackdata , = row
2016-03-23 22:26:57 +00:00
logger . info ( ' Watching for ackdata ' + hexlify ( ackdata ) )
2013-06-24 19:51:01 +00:00
shared . ackdataForWhichImWatching [ ackdata ] = 0
2013-06-21 21:32:22 +00:00
2015-11-24 00:55:17 +00:00
self . stop . wait (
2015-03-09 06:35:32 +00:00
10 ) # give some time for the GUI to start before we start on existing POW tasks.
2017-01-14 22:20:15 +00:00
if state . shutdown == 0 :
2016-01-22 13:47:26 +00:00
# just in case there are any pending tasks for msg
# messages that have yet to be sent.
2017-02-08 12:41:56 +00:00
queues . workerQueue . put ( ( ' sendmessage ' , ' ' ) )
2016-01-22 13:47:26 +00:00
# just in case there are any tasks for Broadcasts
# that have yet to be sent.
2017-02-08 12:41:56 +00:00
queues . workerQueue . put ( ( ' sendbroadcast ' , ' ' ) )
2013-06-21 21:32:22 +00:00
2017-01-14 22:20:15 +00:00
while state . shutdown == 0 :
2016-04-20 13:33:01 +00:00
self . busy = 0
2017-02-08 12:41:56 +00:00
command , data = queues . workerQueue . get ( )
2016-04-20 13:33:01 +00:00
self . busy = 1
2013-06-21 21:32:22 +00:00
if command == ' sendmessage ' :
2016-04-17 18:31:25 +00:00
try :
self . sendMsg ( )
except :
pass
2013-06-21 21:32:22 +00:00
elif command == ' sendbroadcast ' :
2016-04-17 18:31:25 +00:00
try :
self . sendBroadcast ( )
except :
pass
2013-06-21 21:32:22 +00:00
elif command == ' doPOWForMyV2Pubkey ' :
2016-04-17 18:31:25 +00:00
try :
self . doPOWForMyV2Pubkey ( data )
except :
pass
2013-07-22 05:10:22 +00:00
elif command == ' sendOutOrStoreMyV3Pubkey ' :
2016-04-17 18:31:25 +00:00
try :
self . sendOutOrStoreMyV3Pubkey ( data )
except :
pass
2013-09-13 04:27:34 +00:00
elif command == ' sendOutOrStoreMyV4Pubkey ' :
2016-04-17 18:31:25 +00:00
try :
self . sendOutOrStoreMyV4Pubkey ( data )
except :
pass
2017-02-28 21:59:44 +00:00
elif command == ' resetPoW ' :
try :
proofofwork . resetPoW ( )
except :
pass
2015-11-24 00:55:17 +00:00
elif command == ' stopThread ' :
2016-04-20 13:33:01 +00:00
self . busy = 0
2015-11-24 00:55:17 +00:00
return
2013-06-21 21:32:22 +00:00
else :
2015-11-18 15:22:17 +00:00
logger . error ( ' Probable programming error: The command sent to the workerThread is weird. It is: %s \n ' % command )
2013-06-29 17:29:35 +00:00
2017-02-08 12:41:56 +00:00
queues . workerQueue . task_done ( )
2016-10-05 18:06:47 +00:00
logger . info ( " Quitting... " )
2013-06-21 21:32:22 +00:00
def doPOWForMyV2Pubkey ( self , hash ) : # This function also broadcasts out the pubkey message once it is done with the POW
# Look up my stream number based on my address hash
2017-05-15 10:18:07 +00:00
""" configSections = shared.config.addresses()
2013-06-21 21:32:22 +00:00
for addressInKeysFile in configSections :
if addressInKeysFile < > ' bitmessagesettings ' :
status , addressVersionNumber , streamNumber , hashFromThisParticularAddress = decodeAddress ( addressInKeysFile )
if hash == hashFromThisParticularAddress :
myAddress = addressInKeysFile
break """
myAddress = shared . myAddressesByHash [ hash ]
status , addressVersionNumber , streamNumber , hash = decodeAddress (
myAddress )
2014-08-27 07:14:32 +00:00
2014-11-13 21:32:31 +00:00
TTL = int ( 28 * 24 * 60 * 60 + random . randrange ( - 300 , 300 ) ) # 28 days from now plus or minus five minutes
embeddedTime = int ( time . time ( ) + TTL )
2014-08-27 07:14:32 +00:00
payload = pack ( ' >Q ' , ( embeddedTime ) )
payload + = ' \x00 \x00 \x00 \x01 ' # object type: pubkey
2013-06-21 21:32:22 +00:00
payload + = encodeVarint ( addressVersionNumber ) # Address version number
payload + = encodeVarint ( streamNumber )
2017-01-11 13:27:19 +00:00
payload + = protocol . getBitfield ( myAddress ) # bitfield of features supported by me (see the wiki).
2013-06-21 21:32:22 +00:00
try :
2017-01-11 13:27:19 +00:00
privSigningKeyBase58 = BMConfigParser ( ) . get (
2013-06-21 21:32:22 +00:00
myAddress , ' privsigningkey ' )
2017-01-11 13:27:19 +00:00
privEncryptionKeyBase58 = BMConfigParser ( ) . get (
2013-06-21 21:32:22 +00:00
myAddress , ' privencryptionkey ' )
except Exception as err :
2015-11-18 15:22:17 +00:00
logger . error ( ' Error within doPOWForMyV2Pubkey. Could not read the keys from the keys.dat file for a requested address. %s \n ' % err )
2013-06-21 21:32:22 +00:00
return
2016-03-23 22:26:57 +00:00
privSigningKeyHex = hexlify ( shared . decodeWalletImportFormat (
privSigningKeyBase58 ) )
privEncryptionKeyHex = hexlify ( shared . decodeWalletImportFormat (
privEncryptionKeyBase58 ) )
pubSigningKey = unhexlify ( highlevelcrypto . privToPub (
privSigningKeyHex ) )
pubEncryptionKey = unhexlify ( highlevelcrypto . privToPub (
privEncryptionKeyHex ) )
2013-06-21 21:32:22 +00:00
payload + = pubSigningKey [ 1 : ]
payload + = pubEncryptionKey [ 1 : ]
# Do the POW for this pubkey message
2017-02-08 19:37:42 +00:00
target = 2 * * 64 / ( defaults . networkDefaultProofOfWorkNonceTrialsPerByte * ( len ( payload ) + 8 + defaults . networkDefaultPayloadLengthExtraBytes + ( ( TTL * ( len ( payload ) + 8 + defaults . networkDefaultPayloadLengthExtraBytes ) ) / ( 2 * * 16 ) ) ) )
2015-11-18 15:22:17 +00:00
logger . info ( ' (For pubkey message) Doing proof of work... ' )
2013-06-21 21:32:22 +00:00
initialHash = hashlib . sha512 ( payload ) . digest ( )
trialValue , nonce = proofofwork . run ( target , initialHash )
2015-11-18 15:22:17 +00:00
logger . info ( ' (For pubkey message) Found proof of work ' + str ( trialValue ) , ' Nonce: ' , str ( nonce ) )
2013-06-21 21:32:22 +00:00
payload = pack ( ' >Q ' , nonce ) + payload
inventoryHash = calculateInventoryHash ( payload )
2014-08-27 07:14:32 +00:00
objectType = 1
2017-01-10 20:15:35 +00:00
Inventory ( ) [ inventoryHash ] = (
2013-09-13 04:27:34 +00:00
objectType , streamNumber , payload , embeddedTime , ' ' )
2017-01-19 18:48:12 +00:00
PendingUpload ( ) . add ( inventoryHash )
2013-06-21 21:32:22 +00:00
2016-03-23 22:26:57 +00:00
logger . info ( ' broadcasting inv with hash: ' + hexlify ( inventoryHash ) )
2013-06-29 17:29:35 +00:00
2017-08-09 15:36:52 +00:00
queues . invQueue . put ( ( streamNumber , inventoryHash ) )
2017-02-08 12:41:56 +00:00
queues . UISignalQueue . put ( ( ' updateStatusBar ' , ' ' ) )
2013-11-07 04:38:19 +00:00
try :
2017-01-11 13:27:19 +00:00
BMConfigParser ( ) . set (
2013-11-07 04:38:19 +00:00
myAddress , ' lastpubkeysendtime ' , str ( int ( time . time ( ) ) ) )
2017-01-15 09:50:02 +00:00
BMConfigParser ( ) . save ( )
2013-11-07 04:38:19 +00:00
except :
# The user deleted the address out of the keys.dat file before this
# finished.
pass
2013-06-21 21:32:22 +00:00
2013-07-22 05:10:22 +00:00
# If this isn't a chan address, this function assembles the pubkey data,
# does the necessary POW and sends it out. If it *is* a chan then it
# assembles the pubkey and stores is in the pubkey table so that we can
# send messages to "ourselves".
def sendOutOrStoreMyV3Pubkey ( self , hash ) :
2013-11-07 04:38:19 +00:00
try :
myAddress = shared . myAddressesByHash [ hash ]
except :
#The address has been deleted.
return
2017-01-11 13:27:19 +00:00
if BMConfigParser ( ) . safeGetBoolean ( myAddress , ' chan ' ) :
2015-11-18 15:22:17 +00:00
logger . info ( ' This is a chan address. Not sending pubkey. ' )
2013-09-29 23:24:27 +00:00
return
2013-06-21 21:32:22 +00:00
status , addressVersionNumber , streamNumber , hash = decodeAddress (
myAddress )
2014-08-27 07:14:32 +00:00
2014-11-13 21:32:31 +00:00
TTL = int ( 28 * 24 * 60 * 60 + random . randrange ( - 300 , 300 ) ) # 28 days from now plus or minus five minutes
embeddedTime = int ( time . time ( ) + TTL )
2014-08-27 07:14:32 +00:00
signedTimeForProtocolV2 = embeddedTime - TTL
"""
According to the protocol specification , the expiresTime along with the pubkey information is
signed . But to be backwards compatible during the upgrade period , we shall sign not the
expiresTime but rather the current time . There must be precisely a 28 day difference
between the two . After the upgrade period we ' ll switch to signing the whole payload with the
expiresTime time .
"""
payload = pack ( ' >Q ' , ( embeddedTime ) )
payload + = ' \x00 \x00 \x00 \x01 ' # object type: pubkey
2013-06-21 21:32:22 +00:00
payload + = encodeVarint ( addressVersionNumber ) # Address version number
payload + = encodeVarint ( streamNumber )
2017-01-11 13:27:19 +00:00
payload + = protocol . getBitfield ( myAddress ) # bitfield of features supported by me (see the wiki).
2013-06-21 21:32:22 +00:00
try :
2017-01-11 13:27:19 +00:00
privSigningKeyBase58 = BMConfigParser ( ) . get (
2013-06-21 21:32:22 +00:00
myAddress , ' privsigningkey ' )
2017-01-11 13:27:19 +00:00
privEncryptionKeyBase58 = BMConfigParser ( ) . get (
2013-06-21 21:32:22 +00:00
myAddress , ' privencryptionkey ' )
except Exception as err :
2015-11-18 15:22:17 +00:00
logger . error ( ' Error within sendOutOrStoreMyV3Pubkey. Could not read the keys from the keys.dat file for a requested address. %s \n ' % err )
2013-06-29 17:29:35 +00:00
2013-06-21 21:32:22 +00:00
return
2016-03-23 22:26:57 +00:00
privSigningKeyHex = hexlify ( shared . decodeWalletImportFormat (
privSigningKeyBase58 ) )
privEncryptionKeyHex = hexlify ( shared . decodeWalletImportFormat (
privEncryptionKeyBase58 ) )
pubSigningKey = unhexlify ( highlevelcrypto . privToPub (
privSigningKeyHex ) )
pubEncryptionKey = unhexlify ( highlevelcrypto . privToPub (
privEncryptionKeyHex ) )
2013-06-21 21:32:22 +00:00
payload + = pubSigningKey [ 1 : ]
payload + = pubEncryptionKey [ 1 : ]
2017-01-11 13:27:19 +00:00
payload + = encodeVarint ( BMConfigParser ( ) . getint (
2013-06-21 21:32:22 +00:00
myAddress , ' noncetrialsperbyte ' ) )
2017-01-11 13:27:19 +00:00
payload + = encodeVarint ( BMConfigParser ( ) . getint (
2013-06-21 21:32:22 +00:00
myAddress , ' payloadlengthextrabytes ' ) )
2014-08-27 07:14:32 +00:00
2014-12-25 08:57:34 +00:00
signature = highlevelcrypto . sign ( payload , privSigningKeyHex )
2013-06-21 21:32:22 +00:00
payload + = encodeVarint ( len ( signature ) )
payload + = signature
2013-09-29 23:24:27 +00:00
# Do the POW for this pubkey message
2017-02-08 19:37:42 +00:00
target = 2 * * 64 / ( defaults . networkDefaultProofOfWorkNonceTrialsPerByte * ( len ( payload ) + 8 + defaults . networkDefaultPayloadLengthExtraBytes + ( ( TTL * ( len ( payload ) + 8 + defaults . networkDefaultPayloadLengthExtraBytes ) ) / ( 2 * * 16 ) ) ) )
2015-11-18 15:22:17 +00:00
logger . info ( ' (For pubkey message) Doing proof of work... ' )
2013-09-29 23:24:27 +00:00
initialHash = hashlib . sha512 ( payload ) . digest ( )
trialValue , nonce = proofofwork . run ( target , initialHash )
2015-11-18 15:22:17 +00:00
logger . info ( ' (For pubkey message) Found proof of work. Nonce: ' + str ( nonce ) )
2013-06-21 21:32:22 +00:00
2013-09-29 23:24:27 +00:00
payload = pack ( ' >Q ' , nonce ) + payload
inventoryHash = calculateInventoryHash ( payload )
2014-08-27 07:14:32 +00:00
objectType = 1
2017-01-10 20:15:35 +00:00
Inventory ( ) [ inventoryHash ] = (
2013-09-29 23:24:27 +00:00
objectType , streamNumber , payload , embeddedTime , ' ' )
2017-01-19 18:48:12 +00:00
PendingUpload ( ) . add ( inventoryHash )
2013-06-21 21:32:22 +00:00
2016-03-23 22:26:57 +00:00
logger . info ( ' broadcasting inv with hash: ' + hexlify ( inventoryHash ) )
2013-06-29 17:29:35 +00:00
2017-08-09 15:36:52 +00:00
queues . invQueue . put ( ( streamNumber , inventoryHash ) )
2017-02-08 12:41:56 +00:00
queues . UISignalQueue . put ( ( ' updateStatusBar ' , ' ' ) )
2013-11-07 04:38:19 +00:00
try :
2017-01-11 13:27:19 +00:00
BMConfigParser ( ) . set (
2013-11-07 04:38:19 +00:00
myAddress , ' lastpubkeysendtime ' , str ( int ( time . time ( ) ) ) )
2017-01-15 09:50:02 +00:00
BMConfigParser ( ) . save ( )
2013-11-07 04:38:19 +00:00
except :
# The user deleted the address out of the keys.dat file before this
# finished.
pass
2013-06-21 21:32:22 +00:00
2013-09-13 04:27:34 +00:00
# If this isn't a chan address, this function assembles the pubkey data,
2013-09-29 23:24:27 +00:00
# does the necessary POW and sends it out.
2013-09-13 04:27:34 +00:00
def sendOutOrStoreMyV4Pubkey ( self , myAddress ) :
2017-01-11 13:27:19 +00:00
if not BMConfigParser ( ) . has_section ( myAddress ) :
2013-11-07 04:38:19 +00:00
#The address has been deleted.
return
2017-01-11 13:27:19 +00:00
if shared . BMConfigParser ( ) . safeGetBoolean ( myAddress , ' chan ' ) :
2015-11-18 15:22:17 +00:00
logger . info ( ' This is a chan address. Not sending pubkey. ' )
2013-09-29 23:24:27 +00:00
return
2013-09-13 04:27:34 +00:00
status , addressVersionNumber , streamNumber , hash = decodeAddress (
myAddress )
2014-08-27 07:14:32 +00:00
2014-11-13 21:32:31 +00:00
TTL = int ( 28 * 24 * 60 * 60 + random . randrange ( - 300 , 300 ) ) # 28 days from now plus or minus five minutes
embeddedTime = int ( time . time ( ) + TTL )
2013-09-13 04:27:34 +00:00
payload = pack ( ' >Q ' , ( embeddedTime ) )
2014-08-27 07:14:32 +00:00
payload + = ' \x00 \x00 \x00 \x01 ' # object type: pubkey
2013-09-13 04:27:34 +00:00
payload + = encodeVarint ( addressVersionNumber ) # Address version number
payload + = encodeVarint ( streamNumber )
2017-01-11 13:27:19 +00:00
dataToEncrypt = protocol . getBitfield ( myAddress )
2013-09-13 04:27:34 +00:00
try :
2017-01-11 13:27:19 +00:00
privSigningKeyBase58 = BMConfigParser ( ) . get (
2013-09-13 04:27:34 +00:00
myAddress , ' privsigningkey ' )
2017-01-11 13:27:19 +00:00
privEncryptionKeyBase58 = BMConfigParser ( ) . get (
2013-09-13 04:27:34 +00:00
myAddress , ' privencryptionkey ' )
except Exception as err :
2015-11-18 15:22:17 +00:00
logger . error ( ' Error within sendOutOrStoreMyV4Pubkey. Could not read the keys from the keys.dat file for a requested address. %s \n ' % err )
2013-09-13 04:27:34 +00:00
return
2016-03-23 22:26:57 +00:00
privSigningKeyHex = hexlify ( shared . decodeWalletImportFormat (
privSigningKeyBase58 ) )
privEncryptionKeyHex = hexlify ( shared . decodeWalletImportFormat (
privEncryptionKeyBase58 ) )
pubSigningKey = unhexlify ( highlevelcrypto . privToPub (
privSigningKeyHex ) )
pubEncryptionKey = unhexlify ( highlevelcrypto . privToPub (
privEncryptionKeyHex ) )
2013-09-13 04:27:34 +00:00
dataToEncrypt + = pubSigningKey [ 1 : ]
dataToEncrypt + = pubEncryptionKey [ 1 : ]
2017-01-11 13:27:19 +00:00
dataToEncrypt + = encodeVarint ( BMConfigParser ( ) . getint (
2013-09-13 04:27:34 +00:00
myAddress , ' noncetrialsperbyte ' ) )
2017-01-11 13:27:19 +00:00
dataToEncrypt + = encodeVarint ( BMConfigParser ( ) . getint (
2013-09-13 04:27:34 +00:00
myAddress , ' payloadlengthextrabytes ' ) )
2014-08-27 07:14:32 +00:00
# When we encrypt, we'll use a hash of the data
2013-09-29 23:24:27 +00:00
# contained in an address as a decryption key. This way in order to
# read the public keys in a pubkey message, a node must know the address
# first. We'll also tag, unencrypted, the pubkey with part of the hash
# so that nodes know which pubkey object to try to decrypt when they
# want to send a message.
doubleHashOfAddressData = hashlib . sha512 ( hashlib . sha512 ( encodeVarint (
addressVersionNumber ) + encodeVarint ( streamNumber ) + hash ) . digest ( ) ) . digest ( )
payload + = doubleHashOfAddressData [ 32 : ] # the tag
2014-12-25 08:57:34 +00:00
signature = highlevelcrypto . sign ( payload + dataToEncrypt , privSigningKeyHex )
2014-08-27 07:14:32 +00:00
dataToEncrypt + = encodeVarint ( len ( signature ) )
dataToEncrypt + = signature
2013-09-29 23:24:27 +00:00
privEncryptionKey = doubleHashOfAddressData [ : 32 ]
2014-05-21 10:15:07 +00:00
pubEncryptionKey = highlevelcrypto . pointMult ( privEncryptionKey )
2013-09-29 23:24:27 +00:00
payload + = highlevelcrypto . encrypt (
2016-03-23 22:26:57 +00:00
dataToEncrypt , hexlify ( pubEncryptionKey ) )
2013-09-18 04:04:01 +00:00
2013-09-29 23:24:27 +00:00
# Do the POW for this pubkey message
2017-02-08 19:37:42 +00:00
target = 2 * * 64 / ( defaults . networkDefaultProofOfWorkNonceTrialsPerByte * ( len ( payload ) + 8 + defaults . networkDefaultPayloadLengthExtraBytes + ( ( TTL * ( len ( payload ) + 8 + defaults . networkDefaultPayloadLengthExtraBytes ) ) / ( 2 * * 16 ) ) ) )
2015-11-18 15:22:17 +00:00
logger . info ( ' (For pubkey message) Doing proof of work... ' )
2013-09-29 23:24:27 +00:00
initialHash = hashlib . sha512 ( payload ) . digest ( )
trialValue , nonce = proofofwork . run ( target , initialHash )
2015-11-18 15:22:17 +00:00
logger . info ( ' (For pubkey message) Found proof of work ' + str ( trialValue ) + ' Nonce: ' + str ( nonce ) )
2013-09-13 04:27:34 +00:00
2013-09-29 23:24:27 +00:00
payload = pack ( ' >Q ' , nonce ) + payload
inventoryHash = calculateInventoryHash ( payload )
2014-08-27 07:14:32 +00:00
objectType = 1
2017-01-10 20:15:35 +00:00
Inventory ( ) [ inventoryHash ] = (
2013-09-29 23:24:27 +00:00
objectType , streamNumber , payload , embeddedTime , doubleHashOfAddressData [ 32 : ] )
2017-01-19 18:48:12 +00:00
PendingUpload ( ) . add ( inventoryHash )
2013-09-13 04:27:34 +00:00
2016-03-23 22:26:57 +00:00
logger . info ( ' broadcasting inv with hash: ' + hexlify ( inventoryHash ) )
2013-09-18 04:04:01 +00:00
2017-08-09 15:36:52 +00:00
queues . invQueue . put ( ( streamNumber , inventoryHash ) )
2017-02-08 12:41:56 +00:00
queues . UISignalQueue . put ( ( ' updateStatusBar ' , ' ' ) )
2013-10-25 23:35:59 +00:00
try :
2017-01-11 13:27:19 +00:00
BMConfigParser ( ) . set (
2013-10-25 23:35:59 +00:00
myAddress , ' lastpubkeysendtime ' , str ( int ( time . time ( ) ) ) )
2017-01-15 09:50:02 +00:00
BMConfigParser ( ) . save ( )
2013-12-06 06:52:19 +00:00
except Exception as err :
logger . error ( ' Error: Couldn \' t add the lastpubkeysendtime to the keys.dat file. Error message: %s ' % err )
2013-09-13 04:27:34 +00:00
2013-06-21 21:32:22 +00:00
def sendBroadcast ( self ) :
2016-10-05 18:06:47 +00:00
# Reset just in case
sqlExecute (
''' UPDATE sent SET status= ' broadcastqueued ' WHERE status = ' doingbroadcastpow ' ''' )
2013-08-29 11:27:30 +00:00
queryreturn = sqlQuery (
2016-11-14 19:23:58 +00:00
''' SELECT fromaddress, subject, message, ackdata, ttl, encodingtype FROM sent WHERE status=? and folder= ' sent ' ''' , ' broadcastqueued ' )
2016-10-05 18:06:47 +00:00
2013-06-21 21:32:22 +00:00
for row in queryreturn :
2016-11-14 19:23:58 +00:00
fromaddress , subject , body , ackdata , TTL , encoding = row
2013-06-21 21:32:22 +00:00
status , addressVersionNumber , streamNumber , ripe = decodeAddress (
fromaddress )
2013-07-31 16:36:51 +00:00
if addressVersionNumber < = 1 :
2015-11-18 15:22:17 +00:00
logger . error ( ' Error: In the singleWorker thread, the sendBroadcast function doesn \' t understand the address version. \n ' )
2013-07-31 16:36:51 +00:00
return
# We need to convert our private keys to public keys in order
# to include them.
try :
2017-01-11 13:27:19 +00:00
privSigningKeyBase58 = BMConfigParser ( ) . get (
2013-07-31 16:36:51 +00:00
fromaddress , ' privsigningkey ' )
2017-01-11 13:27:19 +00:00
privEncryptionKeyBase58 = BMConfigParser ( ) . get (
2013-07-31 16:36:51 +00:00
fromaddress , ' privencryptionkey ' )
except :
2017-02-08 12:41:56 +00:00
queues . UISignalQueue . put ( ( ' updateSentItemStatusByAckdata ' , (
2016-04-30 09:07:27 +00:00
ackdata , tr . _translate ( " MainWindow " , " Error! Could not find sender address (your address) in the keys.dat file. " ) ) ) )
2013-07-31 16:36:51 +00:00
continue
2016-10-05 18:06:47 +00:00
sqlExecute (
''' UPDATE sent SET status= ' doingbroadcastpow ' WHERE ackdata=? AND status= ' broadcastqueued ' ''' ,
ackdata )
2016-03-23 22:26:57 +00:00
privSigningKeyHex = hexlify ( shared . decodeWalletImportFormat (
privSigningKeyBase58 ) )
privEncryptionKeyHex = hexlify ( shared . decodeWalletImportFormat (
privEncryptionKeyBase58 ) )
2013-07-31 16:36:51 +00:00
pubSigningKey = highlevelcrypto . privToPub ( privSigningKeyHex ) . decode (
' hex ' ) # At this time these pubkeys are 65 bytes long because they include the encoding byte which we won't be sending in the broadcast message.
2016-03-23 22:26:57 +00:00
pubEncryptionKey = unhexlify ( highlevelcrypto . privToPub (
privEncryptionKeyHex ) )
2013-07-31 16:36:51 +00:00
2015-03-09 06:35:32 +00:00
if TTL > 28 * 24 * 60 * 60 :
TTL = 28 * 24 * 60 * 60
if TTL < 60 * 60 :
TTL = 60 * 60
TTL = int ( TTL + random . randrange ( - 300 , 300 ) ) # add some randomness to the TTL
2014-11-13 21:32:31 +00:00
embeddedTime = int ( time . time ( ) + TTL )
2014-08-27 07:14:32 +00:00
payload = pack ( ' >Q ' , embeddedTime )
payload + = ' \x00 \x00 \x00 \x03 ' # object type: broadcast
2014-12-25 08:57:34 +00:00
if addressVersionNumber < = 3 :
payload + = encodeVarint ( 4 ) # broadcast version
else :
payload + = encodeVarint ( 5 ) # broadcast version
2014-08-27 07:14:32 +00:00
2013-07-31 16:36:51 +00:00
payload + = encodeVarint ( streamNumber )
2013-09-15 01:06:26 +00:00
if addressVersionNumber > = 4 :
doubleHashOfAddressData = hashlib . sha512 ( hashlib . sha512 ( encodeVarint (
addressVersionNumber ) + encodeVarint ( streamNumber ) + ripe ) . digest ( ) ) . digest ( )
2013-12-01 05:45:37 +00:00
tag = doubleHashOfAddressData [ 32 : ]
payload + = tag
else :
tag = ' '
2013-07-31 16:36:51 +00:00
2014-12-25 08:57:34 +00:00
dataToEncrypt = encodeVarint ( addressVersionNumber )
2013-07-31 16:36:51 +00:00
dataToEncrypt + = encodeVarint ( streamNumber )
2017-01-11 13:27:19 +00:00
dataToEncrypt + = protocol . getBitfield ( fromaddress ) # behavior bitfield
2013-07-31 16:36:51 +00:00
dataToEncrypt + = pubSigningKey [ 1 : ]
dataToEncrypt + = pubEncryptionKey [ 1 : ]
if addressVersionNumber > = 3 :
2017-01-11 13:27:19 +00:00
dataToEncrypt + = encodeVarint ( BMConfigParser ( ) . getint ( fromaddress , ' noncetrialsperbyte ' ) )
dataToEncrypt + = encodeVarint ( BMConfigParser ( ) . getint ( fromaddress , ' payloadlengthextrabytes ' ) )
2016-11-14 19:23:58 +00:00
dataToEncrypt + = encodeVarint ( encoding ) # message encoding type
encodedMessage = helper_msgcoding . MsgEncode ( { " subject " : subject , " body " : body } , encoding )
dataToEncrypt + = encodeVarint ( encodedMessage . length )
dataToEncrypt + = encodedMessage . data
2014-12-25 08:57:34 +00:00
dataToSign = payload + dataToEncrypt
2014-08-27 07:14:32 +00:00
2013-07-31 16:36:51 +00:00
signature = highlevelcrypto . sign (
2014-08-27 07:14:32 +00:00
dataToSign , privSigningKeyHex )
2013-07-31 16:36:51 +00:00
dataToEncrypt + = encodeVarint ( len ( signature ) )
dataToEncrypt + = signature
2014-11-13 21:32:31 +00:00
# Encrypt the broadcast with the information contained in the broadcaster's address.
# Anyone who knows the address can generate the private encryption key to decrypt
# the broadcast. This provides virtually no privacy; its purpose is to keep
# questionable and illegal content from flowing through the Internet connections
# and being stored on the disk of 3rd parties.
2013-09-15 01:06:26 +00:00
if addressVersionNumber < = 3 :
privEncryptionKey = hashlib . sha512 ( encodeVarint (
addressVersionNumber ) + encodeVarint ( streamNumber ) + ripe ) . digest ( ) [ : 32 ]
else :
privEncryptionKey = doubleHashOfAddressData [ : 32 ]
2013-12-01 05:45:37 +00:00
2014-05-21 10:15:07 +00:00
pubEncryptionKey = highlevelcrypto . pointMult ( privEncryptionKey )
2013-07-31 16:36:51 +00:00
payload + = highlevelcrypto . encrypt (
2016-03-23 22:26:57 +00:00
dataToEncrypt , hexlify ( pubEncryptionKey ) )
2013-07-31 16:36:51 +00:00
2017-02-08 19:37:42 +00:00
target = 2 * * 64 / ( defaults . networkDefaultProofOfWorkNonceTrialsPerByte * ( len ( payload ) + 8 + defaults . networkDefaultPayloadLengthExtraBytes + ( ( TTL * ( len ( payload ) + 8 + defaults . networkDefaultPayloadLengthExtraBytes ) ) / ( 2 * * 16 ) ) ) )
2015-11-18 15:22:17 +00:00
logger . info ( ' (For broadcast message) Doing proof of work... ' )
2017-02-08 12:41:56 +00:00
queues . UISignalQueue . put ( ( ' updateSentItemStatusByAckdata ' , (
2016-04-30 09:07:27 +00:00
ackdata , tr . _translate ( " MainWindow " , " Doing work necessary to send broadcast... " ) ) ) )
2013-07-31 16:36:51 +00:00
initialHash = hashlib . sha512 ( payload ) . digest ( )
trialValue , nonce = proofofwork . run ( target , initialHash )
2015-11-18 15:22:17 +00:00
logger . info ( ' (For broadcast message) Found proof of work ' + str ( trialValue ) + ' Nonce: ' + str ( nonce ) )
2013-07-31 16:36:51 +00:00
payload = pack ( ' >Q ' , nonce ) + payload
2014-08-27 07:14:32 +00:00
# Sanity check. The payload size should never be larger than 256 KiB. There should
# be checks elsewhere in the code to not let the user try to send a message this large
# until we implement message continuation.
if len ( payload ) > 2 * * 18 : # 256 KiB
logger . critical ( ' This broadcast object is too large to send. This should never happen. Object size: %s ' % len ( payload ) )
continue
2013-06-29 17:29:35 +00:00
2013-07-31 16:36:51 +00:00
inventoryHash = calculateInventoryHash ( payload )
2014-08-27 07:14:32 +00:00
objectType = 3
2017-01-10 20:15:35 +00:00
Inventory ( ) [ inventoryHash ] = (
2014-08-27 07:14:32 +00:00
objectType , streamNumber , payload , embeddedTime , tag )
2017-01-19 18:48:12 +00:00
PendingUpload ( ) . add ( inventoryHash )
2016-03-23 22:26:57 +00:00
logger . info ( ' sending inv (within sendBroadcast function) for object: ' + hexlify ( inventoryHash ) )
2017-08-09 15:36:52 +00:00
queues . invQueue . put ( ( streamNumber , inventoryHash ) )
2013-07-31 16:36:51 +00:00
2017-02-08 12:41:56 +00:00
queues . UISignalQueue . put ( ( ' updateSentItemStatusByAckdata ' , ( ackdata , tr . _translate ( " MainWindow " , " Broadcast sent on % 1 " ) . arg ( l10n . formatTimestamp ( ) ) ) ) )
2013-07-31 16:36:51 +00:00
# Update the status of the message in the 'sent' table to have
# a 'broadcastsent' status
2013-08-29 11:27:30 +00:00
sqlExecute (
' UPDATE sent SET msgid=?, status=?, lastactiontime=? WHERE ackdata=? ' ,
inventoryHash ,
' broadcastsent ' ,
int ( time . time ( ) ) ,
ackdata )
2013-07-31 16:36:51 +00:00
2013-06-21 21:32:22 +00:00
def sendMsg ( self ) :
2016-10-05 18:06:47 +00:00
# Reset just in case
sqlExecute (
''' UPDATE sent SET status= ' msgqueued ' WHERE status IN ( ' doingpubkeypow ' , ' doingmsgpow ' ) ''' )
queryreturn = sqlQuery (
2016-11-14 19:23:58 +00:00
''' SELECT toaddress, fromaddress, subject, message, ackdata, status, ttl, retrynumber, encodingtype FROM sent WHERE (status= ' msgqueued ' or status= ' forcepow ' ) and folder= ' sent ' ''' )
2016-10-05 18:06:47 +00:00
for row in queryreturn : # while we have a msg that needs some work
2016-11-14 19:23:58 +00:00
toaddress , fromaddress , subject , message , ackdata , status , TTL , retryNumber , encoding = row
2014-08-27 07:14:32 +00:00
toStatus , toAddressVersionNumber , toStreamNumber , toRipe = decodeAddress (
toaddress )
fromStatus , fromAddressVersionNumber , fromStreamNumber , fromRipe = decodeAddress (
fromaddress )
# We may or may not already have the pubkey for this toAddress. Let's check.
if status == ' forcepow ' :
# if the status of this msg is 'forcepow' then clearly we have the pubkey already
# because the user could not have overridden the message about the POW being
# too difficult without knowing the required difficulty.
pass
2015-03-09 06:35:32 +00:00
elif status == ' doingmsgpow ' :
# We wouldn't have set the status to doingmsgpow if we didn't already have the pubkey
# so let's assume that we have it.
pass
2014-08-27 07:14:32 +00:00
# If we are sending a message to ourselves or a chan then we won't need an entry in the pubkeys table; we can calculate the needed pubkey using the private keys in our keys.dat file.
2017-01-11 13:27:19 +00:00
elif BMConfigParser ( ) . has_section ( toaddress ) :
2013-08-29 11:27:30 +00:00
sqlExecute (
''' UPDATE sent SET status= ' doingmsgpow ' WHERE toaddress=? AND status= ' msgqueued ' ''' ,
toaddress )
2014-08-27 07:14:32 +00:00
status = ' doingmsgpow '
2015-03-09 06:35:32 +00:00
elif status == ' msgqueued ' :
2014-08-27 07:14:32 +00:00
# Let's see if we already have the pubkey in our pubkeys table
queryreturn = sqlQuery (
2015-03-09 06:35:32 +00:00
''' SELECT address FROM pubkeys WHERE address=? ''' , toaddress )
2014-08-27 07:14:32 +00:00
if queryreturn != [ ] : # If we have the needed pubkey in the pubkey table already,
# set the status of this msg to doingmsgpow
2013-08-29 11:27:30 +00:00
sqlExecute (
2014-08-27 07:14:32 +00:00
''' UPDATE sent SET status= ' doingmsgpow ' WHERE toaddress=? AND status= ' msgqueued ' ''' ,
toaddress )
status = ' doingmsgpow '
2015-03-09 06:35:32 +00:00
# mark the pubkey as 'usedpersonally' so that we don't delete it later. If the pubkey version
# is >= 4 then usedpersonally will already be set to yes because we'll only ever have
# usedpersonally v4 pubkeys in the pubkeys table.
2014-08-27 07:14:32 +00:00
sqlExecute (