2016-06-30 10:11:33 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
2017-06-04 12:11:19 +02:00
|
|
|
import argparse
|
2017-07-01 15:05:06 +02:00
|
|
|
import base64
|
2016-06-30 10:11:33 +02:00
|
|
|
import csv
|
|
|
|
import logging
|
2017-07-01 15:05:06 +02:00
|
|
|
import multiprocessing
|
2016-06-30 10:29:52 +02:00
|
|
|
import os
|
2016-06-30 10:11:33 +02:00
|
|
|
import pickle
|
2016-10-15 16:12:09 +02:00
|
|
|
import signal
|
2016-06-30 10:11:33 +02:00
|
|
|
import socket
|
|
|
|
|
|
|
|
from advertiser import Advertiser
|
|
|
|
from manager import Manager
|
|
|
|
from listener import Listener
|
2017-06-09 20:41:33 +02:00
|
|
|
import i2p.controller
|
|
|
|
import i2p.listener
|
2016-06-30 10:11:33 +02:00
|
|
|
import shared
|
|
|
|
|
|
|
|
|
2016-10-15 17:20:14 +02:00
|
|
|
def handler(s, f):
|
2016-10-15 16:12:09 +02:00
|
|
|
logging.info('Gracefully shutting down MiNode')
|
|
|
|
shared.shutting_down = True
|
|
|
|
|
|
|
|
|
2017-06-04 12:11:19 +02:00
|
|
|
def parse_arguments():
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument('-p', '--port', help='Port to listen on', type=int)
|
2017-07-01 15:05:06 +02:00
|
|
|
parser.add_argument('--host', help='Listening host')
|
2017-06-04 12:11:19 +02:00
|
|
|
parser.add_argument('--debug', help='Enable debug logging', action='store_true')
|
|
|
|
parser.add_argument('--data-dir', help='Path to data directory')
|
|
|
|
parser.add_argument('--no-incoming', help='Do not listen for incoming connections', action='store_true')
|
|
|
|
parser.add_argument('--no-outgoing', help='Do not send outgoing connections', action='store_true')
|
2017-07-01 15:05:06 +02:00
|
|
|
parser.add_argument('--no-ip', help='Do not use IP network', action='store_true')
|
2017-06-04 12:11:19 +02:00
|
|
|
parser.add_argument('--trusted-peer', help='Specify a trusted peer we should connect to')
|
|
|
|
parser.add_argument('--connection-limit', help='Maximum number of connections', type=int)
|
2017-06-09 20:41:33 +02:00
|
|
|
parser.add_argument('--i2p', help='Enable I2P support (uses SAMv3)', action='store_true')
|
2017-06-11 07:55:53 +02:00
|
|
|
parser.add_argument('--i2p-tunnel-length', help='Length of I2P tunnels', type=int)
|
|
|
|
parser.add_argument('--i2p-sam-host', help='Host of I2P SAMv3 bridge')
|
|
|
|
parser.add_argument('--i2p-sam-port', help='Port of I2P SAMv3 bridge', type=int)
|
2017-07-20 16:12:32 +02:00
|
|
|
parser.add_argument('--i2p-transient', help='Generate new I2P destination on start', action='store_true')
|
2017-06-04 12:11:19 +02:00
|
|
|
|
|
|
|
args = parser.parse_args()
|
|
|
|
if args.port:
|
|
|
|
shared.listening_port = args.port
|
2017-07-01 15:05:06 +02:00
|
|
|
if args.host:
|
|
|
|
shared.listening_host = args.host
|
2017-06-04 12:11:19 +02:00
|
|
|
if args.debug:
|
|
|
|
shared.log_level = logging.DEBUG
|
|
|
|
if args.data_dir:
|
|
|
|
dir_path = args.data_dir
|
|
|
|
if not dir_path.endswith('/'):
|
|
|
|
dir_path += '/'
|
|
|
|
shared.data_directory = dir_path
|
|
|
|
if args.no_incoming:
|
|
|
|
shared.listen_for_connections = False
|
|
|
|
if args.no_outgoing:
|
|
|
|
shared.send_outgoing_connections = False
|
2017-07-01 15:05:06 +02:00
|
|
|
if args.no_ip:
|
|
|
|
shared.ip_enabled = False
|
2017-06-04 12:11:19 +02:00
|
|
|
if args.trusted_peer:
|
2017-06-09 20:41:33 +02:00
|
|
|
if len(args.trusted_peer) > 50:
|
|
|
|
# I2P
|
|
|
|
shared.trusted_peer = (args.trusted_peer.encode(), 'i2p')
|
|
|
|
else:
|
|
|
|
colon_count = args.trusted_peer.count(':')
|
|
|
|
if colon_count == 0:
|
|
|
|
shared.trusted_peer = (args.trusted_peer, 8444)
|
|
|
|
if colon_count == 1:
|
|
|
|
addr = args.trusted_peer.split(':')
|
|
|
|
shared.trusted_peer = (addr[0], int(addr[1]))
|
|
|
|
if colon_count >= 2:
|
|
|
|
# IPv6 <3
|
|
|
|
addr = args.trusted_peer.split(']:')
|
|
|
|
addr[0] = addr[0][1:]
|
|
|
|
shared.trusted_peer = (addr[0], int(addr[1]))
|
2017-06-04 12:11:19 +02:00
|
|
|
if args.connection_limit:
|
|
|
|
shared.connection_limit = args.connection_limit
|
2017-06-09 20:41:33 +02:00
|
|
|
if args.i2p:
|
|
|
|
shared.i2p_enabled = True
|
2017-06-11 07:55:53 +02:00
|
|
|
if args.i2p_tunnel_length:
|
|
|
|
shared.i2p_tunnel_length = args.i2p_tunnel_length
|
|
|
|
if args.i2p_sam_host:
|
|
|
|
shared.i2p_sam_host = args.i2p_sam_host
|
|
|
|
if args.i2p_sam_port:
|
|
|
|
shared.i2p_sam_port = args.i2p_sam_port
|
2017-07-20 16:12:32 +02:00
|
|
|
if args.i2p_transient:
|
|
|
|
shared.i2p_transient = True
|
2017-06-04 12:11:19 +02:00
|
|
|
|
|
|
|
|
2017-07-20 18:22:59 +02:00
|
|
|
def load_data():
|
2016-06-30 10:11:33 +02:00
|
|
|
try:
|
|
|
|
with open(shared.data_directory + 'objects.pickle', mode='br') as file:
|
|
|
|
shared.objects = pickle.load(file)
|
|
|
|
except Exception as e:
|
|
|
|
logging.warning('Error while loading objects from disk.')
|
|
|
|
logging.warning(e)
|
|
|
|
|
|
|
|
try:
|
|
|
|
with open(shared.data_directory + 'nodes.pickle', mode='br') as file:
|
2016-07-09 22:09:20 +02:00
|
|
|
shared.node_pool = pickle.load(file)
|
2016-06-30 10:11:33 +02:00
|
|
|
except Exception as e:
|
|
|
|
logging.warning('Error while loading nodes from disk.')
|
|
|
|
logging.warning(e)
|
|
|
|
|
2017-07-01 15:05:06 +02:00
|
|
|
try:
|
|
|
|
with open(shared.data_directory + 'i2p_nodes.pickle', mode='br') as file:
|
|
|
|
shared.i2p_node_pool = pickle.load(file)
|
|
|
|
except Exception as e:
|
|
|
|
logging.warning('Error while loading nodes from disk.')
|
|
|
|
logging.warning(e)
|
|
|
|
|
2016-07-19 11:53:24 +02:00
|
|
|
with open(os.path.join(shared.source_directory, 'core_nodes.csv'), mode='r', newline='') as f:
|
2016-06-30 10:11:33 +02:00
|
|
|
reader = csv.reader(f)
|
|
|
|
shared.core_nodes = {tuple(row) for row in reader}
|
|
|
|
shared.node_pool.update(shared.core_nodes)
|
|
|
|
|
2017-07-01 15:05:06 +02:00
|
|
|
with open(os.path.join(shared.source_directory, 'i2p_core_nodes.csv'), mode='r', newline='') as f:
|
|
|
|
reader = csv.reader(f)
|
|
|
|
shared.i2p_core_nodes = {(row[0].encode(), 'i2p') for row in reader}
|
|
|
|
shared.i2p_node_pool.update(shared.i2p_core_nodes)
|
|
|
|
|
2017-07-20 18:22:59 +02:00
|
|
|
|
|
|
|
def bootstrap_from_dns():
|
|
|
|
try:
|
|
|
|
for item in socket.getaddrinfo('bootstrap8080.bitmessage.org', 80):
|
|
|
|
shared.unchecked_node_pool.add((item[4][0], 8080))
|
|
|
|
logging.debug('Adding ' + item[4][0] + ' to unchecked_node_pool based on DNS bootstrap method')
|
|
|
|
for item in socket.getaddrinfo('bootstrap8444.bitmessage.org', 80):
|
|
|
|
shared.unchecked_node_pool.add((item[4][0], 8444))
|
|
|
|
logging.debug('Adding ' + item[4][0] + ' to unchecked_node_pool based on DNS bootstrap method')
|
|
|
|
except Exception as e:
|
|
|
|
logging.error('Error during DNS bootstrap')
|
|
|
|
logging.error(e)
|
|
|
|
|
|
|
|
|
|
|
|
def start_ip_listener():
|
|
|
|
listener_ipv4 = None
|
|
|
|
listener_ipv6 = None
|
|
|
|
|
|
|
|
if socket.has_ipv6:
|
2017-01-14 13:42:15 +01:00
|
|
|
try:
|
2017-07-20 18:22:59 +02:00
|
|
|
listener_ipv6 = Listener(shared.listening_host, shared.listening_port, family=socket.AF_INET6)
|
|
|
|
listener_ipv6.start()
|
2017-01-14 13:42:15 +01:00
|
|
|
except Exception as e:
|
2017-07-20 18:22:59 +02:00
|
|
|
logging.warning('Error while starting IPv6 listener on port {}'.format(shared.listening_port))
|
|
|
|
logging.warning(e)
|
|
|
|
|
|
|
|
try:
|
|
|
|
listener_ipv4 = Listener(shared.listening_host, shared.listening_port)
|
|
|
|
listener_ipv4.start()
|
|
|
|
except Exception as e:
|
|
|
|
if listener_ipv6:
|
|
|
|
logging.warning('Error while starting IPv4 listener on port {}. '.format(shared.listening_port) +
|
|
|
|
'However the IPv6 one seems to be working and will probably accept IPv4 connections.')
|
|
|
|
else:
|
|
|
|
logging.error('Error while starting IPv4 listener on port {}. '.format(shared.listening_port) +
|
|
|
|
'You will not receive incoming connections. Please check your port configuration')
|
2017-01-14 13:42:15 +01:00
|
|
|
logging.error(e)
|
2016-06-30 10:11:33 +02:00
|
|
|
|
2017-06-09 20:41:33 +02:00
|
|
|
|
2017-07-20 18:22:59 +02:00
|
|
|
def start_i2p_listener():
|
|
|
|
# Grab I2P destinations from old object file
|
|
|
|
for obj in shared.objects.values():
|
|
|
|
if obj.object_type == shared.i2p_dest_obj_type:
|
|
|
|
shared.i2p_unchecked_node_pool.add((base64.b64encode(obj.object_payload, altchars=b'-~'), 'i2p'))
|
|
|
|
|
|
|
|
dest_priv = b''
|
|
|
|
|
|
|
|
if not shared.i2p_transient:
|
|
|
|
try:
|
|
|
|
with open(shared.data_directory + 'i2p_dest_priv.key', mode='br') as file:
|
|
|
|
dest_priv = file.read()
|
|
|
|
logging.debug('Loaded I2P destination private key.')
|
|
|
|
except Exception as e:
|
|
|
|
logging.warning('Error while loading I2P destination private key.')
|
|
|
|
logging.warning(e)
|
|
|
|
|
|
|
|
logging.info('Starting I2P Controller and creating tunnels. This may take a while.')
|
|
|
|
i2p_controller = i2p.controller.I2PController(shared.i2p_sam_host, shared.i2p_sam_port, dest_priv)
|
|
|
|
i2p_controller.start()
|
|
|
|
|
|
|
|
shared.i2p_dest_pub = i2p_controller.dest_pub
|
|
|
|
shared.i2p_session_nick = i2p_controller.nick
|
|
|
|
|
|
|
|
logging.info('Local I2P destination: {}'.format(shared.i2p_dest_pub.decode()))
|
|
|
|
logging.info('I2P session nick: {}'.format(shared.i2p_session_nick.decode()))
|
|
|
|
|
|
|
|
logging.info('Starting I2P Listener')
|
|
|
|
i2p_listener = i2p.listener.I2PListener(i2p_controller.nick)
|
|
|
|
i2p_listener.start()
|
|
|
|
|
|
|
|
if not shared.i2p_transient:
|
|
|
|
try:
|
|
|
|
with open(shared.data_directory + 'i2p_dest_priv.key', mode='bw') as file:
|
|
|
|
file.write(i2p_controller.dest_priv)
|
|
|
|
logging.debug('Saved I2P destination private key.')
|
|
|
|
except Exception as e:
|
|
|
|
logging.warning('Error while saving I2P destination private key.')
|
|
|
|
logging.warning(e)
|
|
|
|
|
|
|
|
try:
|
|
|
|
with open(shared.data_directory + 'i2p_dest.pub', mode='bw') as file:
|
|
|
|
file.write(shared.i2p_dest_pub)
|
|
|
|
logging.debug('Saved I2P destination public key.')
|
|
|
|
except Exception as e:
|
|
|
|
logging.warning('Error while saving I2P destination public key.')
|
|
|
|
logging.warning(e)
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
signal.signal(signal.SIGINT, handler)
|
|
|
|
signal.signal(signal.SIGTERM, handler)
|
|
|
|
|
|
|
|
parse_arguments()
|
|
|
|
|
|
|
|
logging.basicConfig(level=shared.log_level, format='[%(asctime)s] [%(levelname)s] %(message)s')
|
|
|
|
logging.info('Starting MiNode')
|
|
|
|
|
|
|
|
logging.info('Data directory: {}'.format(shared.data_directory))
|
|
|
|
if not os.path.exists(shared.data_directory):
|
2017-06-09 20:41:33 +02:00
|
|
|
try:
|
2017-07-20 18:22:59 +02:00
|
|
|
os.makedirs(shared.data_directory)
|
2017-06-09 20:41:33 +02:00
|
|
|
except Exception as e:
|
2017-07-20 18:22:59 +02:00
|
|
|
logging.warning('Error while creating data directory in: {}'.format(shared.data_directory))
|
2017-06-09 20:41:33 +02:00
|
|
|
logging.warning(e)
|
|
|
|
|
2017-07-20 18:22:59 +02:00
|
|
|
load_data()
|
|
|
|
|
|
|
|
if shared.ip_enabled and not shared.trusted_peer:
|
|
|
|
bootstrap_from_dns()
|
|
|
|
|
|
|
|
if shared.i2p_enabled:
|
|
|
|
# We are starting it before cleaning expired objects so we can collect I2P destination objects
|
|
|
|
start_i2p_listener()
|
|
|
|
|
2017-08-23 18:43:52 +02:00
|
|
|
for vector in set(shared.objects):
|
|
|
|
if not shared.objects[vector].is_valid():
|
2017-08-23 18:58:22 +02:00
|
|
|
if shared.objects[vector].is_expired():
|
|
|
|
logging.debug('Deleted expired object: {}'.format(base64.b16encode(vector).decode()))
|
|
|
|
else:
|
|
|
|
logging.warning('Deleted invalid object: {}'.format(base64.b16encode(vector).decode()))
|
2017-08-23 18:43:52 +02:00
|
|
|
del shared.objects[vector]
|
|
|
|
|
2017-07-01 15:05:06 +02:00
|
|
|
manager = Manager()
|
|
|
|
manager.start()
|
|
|
|
|
|
|
|
advertiser = Advertiser()
|
|
|
|
advertiser.start()
|
|
|
|
|
2017-01-14 13:45:39 +01:00
|
|
|
if shared.listen_for_connections:
|
2017-07-20 18:22:59 +02:00
|
|
|
start_ip_listener()
|
2017-01-14 13:42:15 +01:00
|
|
|
|
2016-06-30 10:11:33 +02:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2017-07-01 15:05:06 +02:00
|
|
|
multiprocessing.set_start_method('spawn')
|
2016-06-30 10:11:33 +02:00
|
|
|
main()
|