miner.py 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  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. def parse_addr_port(val):
  16. url = urlparse("//" + val)
  17. assert url.scheme == ''
  18. assert url.path == ''
  19. assert url.params == ''
  20. assert url.query == ''
  21. assert url.fragment == ''
  22. assert url.port is not None
  23. assert url.hostname is not None
  24. return (url.hostname, url.port)
  25. def rpc_server(port, chainbuilder):
  26. @app.route("/network-info", methods=['GET'])
  27. def get_network_info():
  28. return json.dumps([list(peer.peer_addr)[:2] for peer in chainbuilder.protocol.peers if peer.is_connected])
  29. @app.route("/new-transaction", methods=['PUT'])
  30. def send_transaction():
  31. chainbuilder.protocol.received("transaction", flask.request.json, None, 0)
  32. return b""
  33. @app.route("/show-balance", methods=['POST'])
  34. def show_balance():
  35. pubkeys = {Signing.from_json_compatible(pk): i for (i, pk) in enumerate(flask.request.json)}
  36. amounts = [0 for _ in pubkeys.values()]
  37. for output in chainbuilder.primary_block_chain.unspent_coins.values():
  38. if output.recipient_pk in pubkeys:
  39. amounts[pubkeys[output.recipient_pk]] += output.amount
  40. return json.dumps(amounts)
  41. @app.route("/build-transaction", methods=['POST'])
  42. def build_transaction():
  43. sender_pks = {Signing.from_json_compatible(o): i for i, o in enumerate(flask.request.json['sender-pubkeys'])}
  44. amount = flask.request.json['amount']
  45. inputs = []
  46. used_keys = []
  47. for (inp, output) in chainbuilder.primary_block_chain.unspent_coins.items():
  48. if output.recipient_pk in sender_pks:
  49. amount -= output.amount
  50. inputs.append(inp.to_json_compatible())
  51. used_keys.append(sender_pks[output.recipient_pk])
  52. if amount <= 0:
  53. break
  54. if amount > 0:
  55. inputs = []
  56. used_keys = []
  57. return json.dumps({
  58. "inputs": inputs,
  59. "remaining_amount": -amount,
  60. "key_indices": used_keys,
  61. })
  62. @app.route("/transactions", methods=['POST'])
  63. def get_transactions_for_key():
  64. key = Signing(flask.request.data)
  65. transactions = set()
  66. outputs = set()
  67. for b in chainbuilder.primary_block_chain.blocks:
  68. for t in b.transactions:
  69. for i, target in enumerate(t.targets):
  70. if target.recipient_pk == key:
  71. transactions.add(t)
  72. outputs.add(TransactionInput(t.get_hash(), i))
  73. for b in chainbuilder.primary_block_chain.blocks:
  74. for t in b.transactions:
  75. for inp in t.inputs:
  76. if inp in outputs:
  77. transactions.add(t)
  78. return json.dumps([t.to_json_compatible() for t in transactions])
  79. app.run(port=port)
  80. def main():
  81. parser = argparse.ArgumentParser(description="Blockchain Miner.")
  82. parser.add_argument("--listen-address", default="",
  83. help="The IP address where the P2P server should bind to.")
  84. parser.add_argument("--listen-port", default=0, type=int,
  85. help="The port where the P2P server should listen. Defaults a dynamically assigned port.")
  86. parser.add_argument("--mining-pubkey", type=argparse.FileType('rb'),
  87. help="The public key where mining rewards should be sent to. No mining is performed if this is left unspecified.")
  88. parser.add_argument("--bootstrap-peer", action='append', type=parse_addr_port, default=[],
  89. help="Addresses of other P2P peers in the network.")
  90. parser.add_argument("--rpc-port", type=int, default=40203,
  91. help="The port number where the wallet can find an RPC server.")
  92. args = parser.parse_args()
  93. proto = Protocol(args.bootstrap_peer, GENESIS_BLOCK, args.listen_port, args.listen_address)
  94. if args.mining_pubkey is not None:
  95. pubkey = Signing(args.mining_pubkey.read())
  96. args.mining_pubkey.close()
  97. miner = Miner(proto, pubkey)
  98. miner.start_mining()
  99. chainbuilder = miner.chainbuilder
  100. else:
  101. chainbuilder = ChainBuilder(proto)
  102. rpc_server(args.rpc_port, chainbuilder)
  103. if __name__ == '__main__':
  104. main()