Skip to content

A conceptual blockchain for gun ownership tracking written in Python

License

Notifications You must be signed in to change notification settings

niki864/ChainGun

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

5 Commits
 
 
 
 
 
 
 
 

Repository files navigation

ChainGun - A conceptual blockchain network for gun ownership

There's been a lot of talk about gun violence and calls for action against it in social media and through protests such as March For Our Lives. Hearing about it on a regular basis, I couldn't help but wonder if there was something that could be done to support this cause.

This project here is a manifestation of that desire and my personal interest in blockchain tech. Here, I conceptualize a theoretical blockchain, implemented in Python, that could be used to track gun ownership and transfer of possession. This follows an excellent tutorial listed on hackernoon by Daniel van Flymen.

Summary

Most licensed firearms have a serial id. It is this ID that police and ATF officials use, to try and trace the manufacturer and purchase history of the gun. This entire project is based on the concept that each transaction can be encoded into a blockchain and hence, a distribute ledger, and that the transaction can be recorded by all nodes in the system. Let me first define a few variables in this theoretical transaction system.

  • Owner - Person/Organization with a firearm for sale: e.g. Acme Firearms Inc.
  • Receiver - Person buying a firearm: e.g John Doe
  • Amount - Monetary value paid by John Doe for acquiring firearm from Owner
  • Gun_ID - Serial ID of firearm being sold.

Names can be substituted with some form of ID number for anonymity e.g. Firearms License Card ID Number

Its quite self-explanatory. When John walks up to to Acme Inc. to buy himself a new pistol, he initiates a transaction. The transaction records all these details as one dictionary variable. Now, when John decides to sell his firearm, he becomes the owner and Jane Doe who's buying his pistol becomes the receiver. Another, transaction is added to the chain.

In these two theoretical transactions, we observe that we can preserve information regarding the gun serial ID while still tracking the movement of the gun from one hand to another.

The ledger nature of a blockchain also allows for John, Jane and Acme to be simultaneously updated about the movement of the gun. Jane can trace the origins of the gun to Acme if necessary.

Packages Required

This project was done in Python3

  • Requests
  • Flask

I would also recommend using Postman over cURL for the HTTP client requests.

Prerequisites

A simple of understanding of blockchain basics and python programming is enough to make to implement this project. I would recommend the following guides for more knowledge on blockchain tech and the underlying priciples like Proof of Work, Hashing Algorithms and Network Consensus.

ChainGun

I will be explaining the crucial functions that make this program up

Chain Validity

This functions seeks to test the validity of the chain

class Chaingun:

...

def valid_chain(self, chain):
        last_block = chain[0]
        current_index = 1

        while current_index < len(chain):
            block = chain[current_index]
            print(f'{last_block}')
            print(f'{block}')
            print("\n-----------\n")
            # Check that the hash of the block is correct
            if block['previous_hash'] != self.hash(last_block):
                return False

            # Check that the Proof of Work is correct
            if not self.valid_proof(last_block['proof'], block['proof'], last_block['previous_hash']):
                return False

            last_block = block
            current_index += 1

        return True

...

Here chain is the current blockchain and the return variable is boolean true or false denoting if the chain passed the validity tests.

Network Consensus

def resolve_conflicts(self):
        neighbours = self.nodes
        new_chain = None

        # Look only for chains longer than current
        max_length = len(self.chain)

        # Find and verify the chains from all the nodes in network
        for node in neighbours:
            response = requests.get(f'http://{node}/chain')

            if response.status_code == 200:
                length = response.json()['length']
                chain = response.json()['chain']

                # Check to see if length is longer than current and the chain is valid
                if length > max_length and self.valid_chain(chain):
                    max_length = length
                    new_chain = chain

        # Replace our chain if we discovered a new, valid chain longer than ours
        if new_chain:
            self.chain = new_chain
            return True

        return False

This is the network consensus algorithm, it resolves conflicts by replacing our chain with the longest one in the network by checking all the other registered nodes. Our logic here is that the longest chain in the network is the most valid chain. We return True if our chain was replaced, and False if not.

Transaction and Hashing Function

The transaction function is next. This contains all the information for our transaction.

def new_transaction(self, owner, receiver, amount, gun_id):
        self.current_transactions.append({
            'owner': owner,
            'receiver': receiver,
            'amount': amount,
            'gun_id':gun_id
            
        })

        return self.last_block['index'] + 1

Creates a new transaction thats queued for the next mined Block

  • owner: ID of the current gun owner
  • receiver: ID of the buyer
  • amount: Amount paid in the transaction
  • gun_id: Unique gun id for the gun that's being transacted
  • return: The index of the Block that will hold this transaction

The hashing function takes all of this information, orders and encodes it performing a standard SHA-256 hash on it.

def hash(block):
        block_string = json.dumps(block, sort_keys=True).encode()
        return hashlib.sha256(block_string).hexdigest()

Proof of Work and Validation of PofW

Every blockchain network needs to have a proper proof of work system for the functioning of the system. Every time a new node is mined, this proof of work algo is what gives legitimacy to it. This PofW algo could be any mathematical equation or problem. Simple Proof of Work Algorithm:

  • Find a number p' such that hash(pp') contains leading 4 zeroes
  • Where p is the previous proof, and p' is the new proof
def proof_of_work(self, last_block):
        last_proof = last_block['proof']
        last_hash = self.hash(last_block)
        proof = 0
        while self.valid_proof(last_proof, proof, last_hash) is False:
            proof += 1
        return proof

   @staticmethod
   def valid_proof(last_proof, proof, last_hash):
        guess = f'{last_proof}{proof}{last_hash}'.encode()
        guess_hash = hashlib.sha256(guess).hexdigest()
        return guess_hash[:4] == "0000"

Demonstration

Let's try out how it works! Fire up Postman and do the following.

Initializing: Send a GET request to /mine

Drawing

Make a new transaction using POST and filling out the transaction details in JSON. Keep owner as the hash id of receiver in previous block /transactions/new. Mine again. You can add transactions and play around with this.

Drawing

Check out the addition in the chain using /chain

Drawing

Change port number from 5000 to 5001 and run another instance of chaingun.py. Then in a POST request on your old node, register your new node.

Drawing

Add Some transactions to your new node

Drawing

Try merging using nodes/resolve and see what you get!

Drawing

Conclusion

Obviously, this is not even close to the actual complexity of a real block-chain network like Ethereum. But it was fun learning in depth into blockchain systems and to come up with a simple network that demostrated it. Play around with it on your own systems.

Common Issues

I found that sometimes there were other processes that were still running and occupying ports in my system. Run this on your CLI ps -fA | grep python This will list the processes running on the port.

You can kill them by entering the process number kill <processnumber>. Run chaingun.py after that.

Acknowledgments

License

This project is licensed under the MIT License. See LICENSE.md for more information.

About

A conceptual blockchain for gun ownership tracking written in Python

Topics

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published