miner.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. #!/usr/bin/env python3
  2. """ The executable that participates in the P2P network and optionally mines new blocks. """
  3. __all__ = []
  4. import argparse
  5. import json
  6. from urllib.parse import urlparse
  7. import flask
  8. app = flask.Flask(__name__)
  9. from src.crypto import Signing
  10. from src.protocol import Protocol
  11. from src.block import GENESIS_BLOCK
  12. from src.chainbuilder import ChainBuilder
  13. from src.mining import Miner
  14. from src.transaction import TransactionInput
  15. from src.persistence import Persistence
  16. def parse_addr_port(val):
  17. url = urlparse("//" + val)
  18. assert url.scheme == ''
  19. assert url.path == ''
  20. assert url.params == ''
  21. assert url.query == ''
  22. assert url.fragment == ''
  23. assert url.port is not None
  24. assert url.hostname is not None
  25. return (url.hostname, url.port)
  26. def rpc_server(port, chainbuilder, persist):
  27. @app.route("/network-info", methods=['GET'])
  28. def get_network_info():
  29. return json.dumps([list(peer.peer_addr)[:2] for peer in chainbuilder.protocol.peers if peer.is_connected])
  30. @app.route("/new-transaction", methods=['PUT'])
  31. def send_transaction():
  32. chainbuilder.protocol.received("transaction", flask.request.json, None, 0)
  33. # TODO: from the right thread, we want to call persist.store() here
  34. return b""
  35. @app.route("/show-balance", methods=['POST'])
  36. def show_balance():
  37. pubkeys = {Signing.from_json_compatible(pk): i for (i, pk) in enumerate(flask.request.json)}
  38. amounts = [0 for _ in pubkeys.values()]
  39. for output in chainbuilder.primary_block_chain.unspent_coins.values():
  40. if output.recipient_pk in pubkeys:
  41. amounts[pubkeys[output.recipient_pk]] += output.amount
  42. return json.dumps(amounts)
  43. @app.route("/build-transaction", methods=['POST'])
  44. def build_transaction():
  45. sender_pks = {Signing.from_json_compatible(o): i for i, o in enumerate(flask.request.json['sender-pubkeys'])}
  46. amount = flask.request.json['amount']
  47. inputs = []
  48. used_keys = []
  49. for (inp, output) in chainbuilder.primary_block_chain.unspent_coins.items():
  50. if output.recipient_pk in sender_pks:
  51. amount -= output.amount
  52. inputs.append(inp.to_json_compatible())
  53. used_keys.append(sender_pks[output.recipient_pk])
  54. if amount <= 0:
  55. break
  56. if amount > 0:
  57. inputs = []
  58. used_keys = []
  59. return json.dumps({
  60. "inputs": inputs,
  61. "remaining_amount": -amount,
  62. "key_indices": used_keys,
  63. })
  64. @app.route("/transactions", methods=['POST'])
  65. def get_transactions_for_key():
  66. key = Signing(flask.request.data)
  67. transactions = set()
  68. outputs = set()
  69. for b in chainbuilder.primary_block_chain.blocks:
  70. for t in b.transactions:
  71. for i, target in enumerate(t.targets):
  72. if target.recipient_pk == key:
  73. transactions.add(t)
  74. outputs.add(TransactionInput(t.get_hash(), i))
  75. for b in chainbuilder.primary_block_chain.blocks:
  76. for t in b.transactions:
  77. for inp in t.inputs:
  78. if inp in outputs:
  79. transactions.add(t)
  80. return json.dumps([t.to_json_compatible() for t in transactions])
  81. app.run(port=port)
  82. def main():
  83. parser = argparse.ArgumentParser(description="Blockchain Miner.")
  84. parser.add_argument("--listen-address", default="",
  85. help="The IP address where the P2P server should bind to.")
  86. parser.add_argument("--listen-port", default=0, type=int,
  87. help="The port where the P2P server should listen. Defaults a dynamically assigned port.")
  88. parser.add_argument("--mining-pubkey", type=argparse.FileType('rb'),
  89. help="The public key where mining rewards should be sent to. No mining is performed if this is left unspecified.")
  90. parser.add_argument("--bootstrap-peer", action='append', type=parse_addr_port, default=[],
  91. help="Addresses of other P2P peers in the network.")
  92. parser.add_argument("--rpc-port", type=int, default=40203,
  93. help="The port number where the wallet can find an RPC server.")
  94. parser.add_argument("--persist-path",
  95. help="The file where data is persisted.")
  96. args = parser.parse_args()
  97. proto = Protocol(args.bootstrap_peer, GENESIS_BLOCK, args.listen_port, args.listen_address)
  98. if args.mining_pubkey is not None:
  99. pubkey = Signing(args.mining_pubkey.read())
  100. args.mining_pubkey.close()
  101. miner = Miner(proto, pubkey)
  102. miner.start_mining()
  103. chainbuilder = miner.chainbuilder
  104. else:
  105. chainbuilder = ChainBuilder(proto)
  106. if args.persist_path:
  107. persist = Persistence(args.persist_path, chainbuilder)
  108. try:
  109. persist.load()
  110. except FileNotFoundError:
  111. pass
  112. else:
  113. persist = None
  114. rpc_server(args.rpc_port, chainbuilder, persist)
  115. if __name__ == '__main__':
  116. main()