2013-09-28 12:09:15 +00:00
#!/usr/bin/env python
2012-11-19 19:45:05 +00:00
# Copyright (c) 2012 Jonathan Warren
# Copyright (c) 2012 The Bitmessage developers
# Distributed under the MIT/X11 software license. See the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
2013-06-13 18:00:56 +00:00
# Right now, PyBitmessage only support connecting to stream 1. It doesn't
# yet contain logic to expand into further streams.
2012-11-19 19:45:05 +00:00
2013-06-13 18:00:56 +00:00
# The software version variable is now held in shared.py
2013-02-18 20:22:48 +00:00
2013-06-13 18:00:56 +00:00
import signal # Used to capture a Ctrl-C keypress so that Bitmessage can shutdown gracefully.
# The next 3 are used for the API
2013-05-13 09:29:14 +00:00
import singleton
2013-06-25 20:26:12 +00:00
import os
2014-02-16 16:21:20 +00:00
import socket
import ctypes
from struct import pack
2013-05-01 20:06:55 +00:00
2013-12-30 00:53:44 +00:00
from SimpleXMLRPCServer import SimpleXMLRPCServer
2014-01-20 20:25:02 +00:00
from api import MySimpleXMLRPCRequestHandler
from helper_startup import isOurOperatingSystemLimitedToHavingVeryFewHalfOpenConnections
2013-12-30 00:53:44 +00:00
import shared
from helper_sql import sqlQuery
import threading
2013-06-20 21:23:03 +00:00
# Classes
2013-12-30 00:53:44 +00:00
#from helper_sql import *
#from class_sqlThread import *
from class_sqlThread import sqlThread
from class_singleCleaner import singleCleaner
#from class_singleWorker import *
from class_objectProcessor import objectProcessor
from class_outgoingSynSender import outgoingSynSender
from class_singleListener import singleListener
from class_singleWorker import singleWorker
#from class_addressGenerator import *
from class_addressGenerator import addressGenerator
2013-08-25 20:23:28 +00:00
from debug import logger
2013-06-20 21:23:03 +00:00
2013-06-20 22:55:04 +00:00
# Helper Functions
import helper_bootstrap
2013-12-30 00:53:44 +00:00
import helper_generic
2013-06-20 22:55:04 +00:00
2013-12-30 00:53:44 +00:00
from subprocess import call
import time
2013-10-25 23:35:59 +00:00
2014-01-13 00:30:01 +00:00
# OSX python version check
import sys
if sys . platform == ' darwin ' :
if float ( " {1} . {2} " . format ( * sys . version_info ) ) < 7.5 :
msg = " You should use python 2.7.5 or greater. Your version: %s " , " {0} . {1} . {2} " . format ( * sys . version_info )
logger . critical ( msg )
print msg
sys . exit ( 0 )
2013-05-01 20:06:55 +00:00
def connectToStream ( streamNumber ) :
2013-11-20 06:29:37 +00:00
shared . streamsInWhichIAmParticipating [ streamNumber ] = ' no data '
2013-05-01 20:06:55 +00:00
selfInitiatedConnections [ streamNumber ] = { }
2013-09-06 22:55:12 +00:00
shared . inventorySets [ streamNumber ] = set ( )
queryData = sqlQuery ( ''' SELECT hash FROM inventory WHERE streamnumber=? ''' , streamNumber )
for row in queryData :
shared . inventorySets [ streamNumber ] . add ( row [ 0 ] )
2014-01-20 20:25:02 +00:00
if isOurOperatingSystemLimitedToHavingVeryFewHalfOpenConnections ( ) :
# Some XP and Vista systems can only have 10 outgoing connections at a time.
2013-05-08 21:11:16 +00:00
maximumNumberOfHalfOpenConnections = 9
else :
2014-01-20 20:25:02 +00:00
maximumNumberOfHalfOpenConnections = 64
2013-05-08 21:11:16 +00:00
for i in range ( maximumNumberOfHalfOpenConnections ) :
2013-05-01 20:06:55 +00:00
a = outgoingSynSender ( )
2013-06-21 23:49:50 +00:00
a . setup ( streamNumber , selfInitiatedConnections )
2013-05-01 20:06:55 +00:00
a . start ( )
2014-02-16 16:21:20 +00:00
def _fixWinsock ( ) :
if not ( ' win32 ' in sys . platform ) and not ( ' win64 ' in sys . platform ) :
return
# Python 2 on Windows doesn't define a wrapper for
# socket.inet_ntop but we can make one ourselves using ctypes
if not hasattr ( socket , ' inet_ntop ' ) :
addressToString = ctypes . windll . ws2_32 . WSAAddressToStringA
def inet_ntop ( family , host ) :
if family == socket . AF_INET :
if len ( host ) != 4 :
raise ValueError ( " invalid IPv4 host " )
host = pack ( " hH4s8s " , socket . AF_INET , 0 , host , " \0 " * 8 )
elif family == socket . AF_INET6 :
if len ( host ) != 16 :
raise ValueError ( " invalid IPv6 host " )
host = pack ( " hHL16sL " , socket . AF_INET6 , 0 , 0 , host , 0 )
else :
raise ValueError ( " invalid address family " )
buf = " \0 " * 64
lengthBuf = pack ( " I " , len ( buf ) )
addressToString ( host , len ( host ) , None , buf , lengthBuf )
return buf [ 0 : buf . index ( " \0 " ) ]
socket . inet_ntop = inet_ntop
# Same for inet_pton
if not hasattr ( socket , ' inet_pton ' ) :
stringToAddress = ctypes . windll . ws2_32 . WSAStringToAddressA
def inet_pton ( family , host ) :
buf = " \0 " * 28
lengthBuf = pack ( " I " , len ( buf ) )
if stringToAddress ( str ( host ) ,
int ( family ) ,
None ,
buf ,
lengthBuf ) != 0 :
raise socket . error ( " illegal IP address passed to inet_pton " )
if family == socket . AF_INET :
return buf [ 4 : 8 ]
elif family == socket . AF_INET6 :
return buf [ 8 : 24 ]
else :
raise ValueError ( " invalid address family " )
socket . inet_pton = inet_pton
# These sockopts are needed on for IPv6 support
if not hasattr ( socket , ' IPPROTO_IPV6 ' ) :
socket . IPPROTO_IPV6 = 41
if not hasattr ( socket , ' IPV6_V6ONLY ' ) :
socket . IPV6_V6ONLY = 27
2013-04-26 17:20:30 +00:00
2013-06-13 18:00:56 +00:00
# This thread, of which there is only one, runs the API.
2013-05-01 20:06:55 +00:00
class singleAPI ( threading . Thread ) :
2013-06-13 18:00:56 +00:00
2013-05-01 20:06:55 +00:00
def __init__ ( self ) :
threading . Thread . __init__ ( self )
2013-03-19 17:32:37 +00:00
def run ( self ) :
2013-06-13 18:00:56 +00:00
se = SimpleXMLRPCServer ( ( shared . config . get ( ' bitmessagesettings ' , ' apiinterface ' ) , shared . config . getint (
' bitmessagesettings ' , ' apiport ' ) ) , MySimpleXMLRPCRequestHandler , True , True )
2013-03-19 17:32:37 +00:00
se . register_introspection_functions ( )
se . serve_forever ( )
2013-08-06 17:19:26 +00:00
# This is a list of current connections (the thread pointers at least)
2013-06-24 19:51:01 +00:00
selfInitiatedConnections = { }
if shared . useVeryEasyProofOfWorkForTesting :
2013-06-13 18:00:56 +00:00
shared . networkDefaultProofOfWorkNonceTrialsPerByte = int (
shared . networkDefaultProofOfWorkNonceTrialsPerByte / 16 )
shared . networkDefaultPayloadLengthExtraBytes = int (
shared . networkDefaultPayloadLengthExtraBytes / 7000 )
2013-01-18 22:38:09 +00:00
2013-08-06 11:23:56 +00:00
class Main :
2013-08-07 21:02:53 +00:00
def start ( self , daemon = False ) :
2014-02-16 16:21:20 +00:00
_fixWinsock ( )
2013-09-05 00:14:25 +00:00
shared . daemon = daemon
2013-08-06 11:23:56 +00:00
# is the application already running? If yes then exit.
2014-01-13 00:50:44 +00:00
thisapp = singleton . singleinstance ( )
2013-08-06 11:23:56 +00:00
signal . signal ( signal . SIGINT , helper_generic . signal_handler )
# signal.signal(signal.SIGINT, signal.SIG_DFL)
helper_bootstrap . knownNodes ( )
# Start the address generation thread
addressGeneratorThread = addressGenerator ( )
addressGeneratorThread . daemon = True # close the main program even if there are threads left
addressGeneratorThread . start ( )
# Start the thread that calculates POWs
singleWorkerThread = singleWorker ( )
singleWorkerThread . daemon = True # close the main program even if there are threads left
singleWorkerThread . start ( )
# Start the SQL thread
sqlLookup = sqlThread ( )
sqlLookup . daemon = False # DON'T close the main program even if there are threads left. The closeEvent should command this thread to exit gracefully.
sqlLookup . start ( )
2013-12-02 06:35:34 +00:00
# Start the thread that calculates POWs
objectProcessorThread = objectProcessor ( )
objectProcessorThread . daemon = False # DON'T close the main program even the thread remains. This thread checks the shutdown variable after processing each object.
objectProcessorThread . start ( )
2013-08-06 11:23:56 +00:00
# Start the cleanerThread
singleCleanerThread = singleCleaner ( )
singleCleanerThread . daemon = True # close the main program even if there are threads left
singleCleanerThread . start ( )
shared . reloadMyAddressHashes ( )
shared . reloadBroadcastSendersForWhichImWatching ( )
if shared . safeConfigGetBoolean ( ' bitmessagesettings ' , ' apienabled ' ) :
try :
apiNotifyPath = shared . config . get (
' bitmessagesettings ' , ' apinotifypath ' )
except :
apiNotifyPath = ' '
if apiNotifyPath != ' ' :
with shared . printLock :
print ' Trying to call ' , apiNotifyPath
2013-05-02 15:53:54 +00:00
2013-08-06 11:23:56 +00:00
call ( [ apiNotifyPath , " startingUp " ] )
singleAPIThread = singleAPI ( )
singleAPIThread . daemon = True # close the main program even if there are threads left
singleAPIThread . start ( )
2013-06-29 17:29:35 +00:00
2013-08-06 11:23:56 +00:00
connectToStream ( 1 )
2013-05-01 20:06:55 +00:00
2013-08-06 11:23:56 +00:00
singleListenerThread = singleListener ( )
singleListenerThread . setup ( selfInitiatedConnections )
singleListenerThread . daemon = True # close the main program even if there are threads left
singleListenerThread . start ( )
2013-05-01 20:06:55 +00:00
2013-08-07 21:02:53 +00:00
if daemon == False and shared . safeConfigGetBoolean ( ' bitmessagesettings ' , ' daemon ' ) == False :
2013-08-06 11:23:56 +00:00
try :
from PyQt4 import QtCore , QtGui
except Exception as err :
print ' PyBitmessage requires PyQt unless you want to run it as a daemon and interact with it using the API. You can download PyQt from http://www.riverbankcomputing.com/software/pyqt/download or by searching Google for \' PyQt Download \' . If you want to run in daemon mode, see https://bitmessage.org/wiki/Daemon '
print ' Error message: ' , err
os . _exit ( 0 )
import bitmessageqt
bitmessageqt . run ( )
else :
shared . config . remove_option ( ' bitmessagesettings ' , ' dontconnect ' )
2013-05-02 20:55:13 +00:00
2013-08-07 21:02:53 +00:00
if daemon :
2013-08-06 11:23:56 +00:00
with shared . printLock :
print ' Running as a daemon. The main program should exit this thread. '
else :
with shared . printLock :
print ' Running as a daemon. You can use Ctrl+C to exit. '
while True :
time . sleep ( 20 )
2013-09-28 12:09:15 +00:00
2013-08-06 11:23:56 +00:00
def stop ( self ) :
2013-06-29 17:29:35 +00:00
with shared . printLock :
2013-08-06 11:23:56 +00:00
print ' Stopping Bitmessage Deamon. '
shared . doCleanShutdown ( )
2013-09-28 12:09:15 +00:00
2013-12-30 00:53:44 +00:00
#TODO: nice function but no one is using this
2013-08-06 11:23:56 +00:00
def getApiAddress ( self ) :
if not shared . safeConfigGetBoolean ( ' bitmessagesettings ' , ' apienabled ' ) :
return None
address = shared . config . get ( ' bitmessagesettings ' , ' apiinterface ' )
port = shared . config . getint ( ' bitmessagesettings ' , ' apiport ' )
return { ' address ' : address , ' port ' : port }
2013-09-28 12:09:15 +00:00
2013-08-05 20:29:06 +00:00
if __name__ == " __main__ " :
2013-08-06 11:23:56 +00:00
mainprogram = Main ( )
mainprogram . start ( )
2013-02-18 20:22:48 +00:00
2013-09-28 12:09:15 +00:00
2013-06-17 20:42:30 +00:00
# So far, the creation of and management of the Bitmessage protocol and this
# client is a one-man operation. Bitcoin tips are quite appreciated.
2012-11-29 10:39:39 +00:00
# 1H5XaDA6fYENLbknwZyjiYXYPQaFjjLX2u