Handcrafting Transactions¶
For those who wish to assemble transaction payloads “by hand”, with examples in Python.
Overview¶
Submitting a transaction to a BigchainDB node consists of three main steps:
- Preparing the transaction payload;
- Fulfilling the prepared transaction payload; and
- Sending the transaction payload via HTTPS.
Step 1 and 2 can be performed offline on the client. That is, they do not require any connection to any BigchainDB node.
For convenience’s sake, some utilites are provided to prepare and fulfill a
transaction via the BigchainDB class, and via the
offchain module. For an introduction on using these
utilities, see the Basic Usage Examples or Advanced Usage Examples sections.
The rest of this document will guide you through completing steps 1 and 2 manually by revisiting some of the examples provided in the usage sections. We will:
- provide all values, including the default ones;
- generate the transaction id;
- learn to use crypto-conditions to generate a condition that locks the transaction, hence protecting it from being consumed by an unauthorized user;
- learn to use crypto-conditions to generate a fulfillment that unlocks the transaction asset, and consequently enact an ownership transfer.
In order to perform all of the above, we’ll use the following Python libraries:
- json: to serialize the transaction dictionary into a JSON formatted string;
- sha3: to hash the serialized transaction; and
- cryptoconditions: to create conditions and fulfillments
High-level view of a transaction in Python¶
For detailled documentation on the transaction schema, please consult The Transaction Model and The Transaction Schema.
From the point of view of Python, a transaction is simply a dictionary:
{
    'id': 'c10671ba9b1d3be3cb68959416bf88a711a4eeff2dbe29f5aca13e8dc39579ff',
    'operation': 'CREATE',
    'asset': {
        'data': {
            'bicycle': {
                'manufacturer': 'bkfab',
                'serial_number': 'abcd1234',
            },
        },
    },
    'metadata': {'planet': 'earth'},
    'outputs': [{
        'amount': 1,
        'condition': {
            'details': {
                'bitmask': 32,
                'public_key': '6FCKbDMmCiM37pN9qzNmgJxKqebWzGxZUcAqB8CNg84J',
                'signature': None,
                'type': 'fulfillment',
                'type_id': 4,
            },
            'uri': 'cc:4:20:Te1duSG0oFpsm-5Sn9uQT1QIngxmhZSOEry8xeCna8M:96',
        },
        'public_keys': ['6FCKbDMmCiM37pN9qzNmgJxKqebWzGxZUcAqB8CNg84J'],
    }],
    'inputs': [{
        'fulfillment': 'cf:4:Te1duSG0oFpsm-5Sn9uQT1QIngxmhZSOEry8xeCna8OJSbCmmVoQddD14yzvLRC0XxC5CsK7KnOORFOe5gOiCkEUh-KqCBgia_38jx4B-KDUkhcMaT-oP2TcjIRZhhkJ',
        'fulfills': None,
        'owners_before': ['6FCKbDMmCiM37pN9qzNmgJxKqebWzGxZUcAqB8CNg84J'],
    }],
    'version': '0.9',
}
Because a transaction must be signed before being sent, the id and
fulfillment must be provided by the client.
Important
Implications of Signed Payloads
Because BigchainDB relies on cryptographic signatures, the payloads need to be fully prepared and signed on the client side. This prevents the server(s) from tempering with the provided data.
This enhanced security puts more work on the clients, as various values that could traditionally be generated on the server side need to be generated on the client side.
Bicycle Asset Creation Revisited¶
The Prepared Transaction¶
Recall that in order to prepare a transaction, we had to do something similar to:
In [1]: from bigchaindb_driver.crypto import generate_keypair
In [2]: from bigchaindb_driver.offchain import prepare_transaction
In [3]: alice = generate_keypair()
In [4]: bicycle = {
   ...:     'data': {
   ...:         'bicycle': {
   ...:             'serial_number': 'abcd1234',
   ...:             'manufacturer': 'bkfab',
   ...:         },
   ...:     },
   ...: }
   ...: 
In [5]: metadata = {'planet': 'earth'}
In [6]: prepared_creation_tx = prepare_transaction(
   ...:     operation='CREATE',
   ...:     signers=alice.public_key,
   ...:     asset=bicycle,
   ...:     metadata=metadata,
   ...: )
   ...: 
and the payload of the prepared transaction looked similar to:
In [7]: prepared_creation_tx
Out[7]: 
{'asset': {'data': {'bicycle': {'manufacturer': 'bkfab',
    'serial_number': 'abcd1234'}}},
 'id': '11aadb005123fb1d7f76536f51b330574744b87f44bd1de7a5172589318c9d72',
 'inputs': [{'fulfillment': {'bitmask': 32,
    'public_key': 'ATEi7Kos3xgpgogneUABuujia22Pi5RWoqkRKPhG3HZc',
    'signature': None,
    'type': 'fulfillment',
    'type_id': 4},
   'fulfills': None,
   'owners_before': ['ATEi7Kos3xgpgogneUABuujia22Pi5RWoqkRKPhG3HZc']}],
 'metadata': {'planet': 'earth'},
 'operation': 'CREATE',
 'outputs': [{'amount': 1,
   'condition': {'details': {'bitmask': 32,
     'public_key': 'ATEi7Kos3xgpgogneUABuujia22Pi5RWoqkRKPhG3HZc',
     'signature': None,
     'type': 'fulfillment',
     'type_id': 4},
    'uri': 'cc:4:20:jHHIsAp5OWQ1RHmYfwAPJ09HM6FlRp46_YFwNi4Db2M:96'},
   'public_keys': ['ATEi7Kos3xgpgogneUABuujia22Pi5RWoqkRKPhG3HZc']}],
 'version': '0.9'}
Note alice‘s public key is listed in the public keys of outputs:
In [8]: alice.public_key
Out[8]: 'ATEi7Kos3xgpgogneUABuujia22Pi5RWoqkRKPhG3HZc'
In [9]: prepared_creation_tx['outputs'][0]['public_keys'][0] == alice.public_key