|
@@ -1,21 +1,26 @@
|
|
|
# Verax
|
|
|
|
|
|
-## Theory
|
|
|
+## Data model
|
|
|
|
|
|
-Verax's data model is heavily inspired by Bitcoin's model. It is a data model where transactions are primarily by a set of `payments` that are going to be spend, and a set of new `payments` to be created.
|
|
|
+Verax's data model is heavily inspired by Bitcoin's unspent transaction output
|
|
|
+(UTXO) model. It is a data where transactions are a set of `payments` that are
|
|
|
+spent, and a set of new `payments` to be created.
|
|
|
|
|
|
Each `payment` is a data structure with this information
|
|
|
|
|
|
```
|
|
|
struct Payment {
|
|
|
- created_by_transaction: TransactionId,
|
|
|
- amount: Amount,
|
|
|
- destination: Address,
|
|
|
- spend_by: Option<TransactionId>
|
|
|
+ created_by_transaction: TransactionId,
|
|
|
+ amount: Amount,
|
|
|
+ destination: Address,
|
|
|
+ spend_by: Option<TransactionId>
|
|
|
}
|
|
|
```
|
|
|
|
|
|
-Any `payment` that has `spend_by: None` is part of the active balance of each address. Any other payment is not longer part of the active balance, but part of the historial record. These spent payments are read-only from that point forward.
|
|
|
+Any `payment` that has `spend_by: None` is part of the active balance of each
|
|
|
+address. Any other payment is not longer part of the active balance, but part of
|
|
|
+the historial record. These spent payments are read-only from that point
|
|
|
+forward.
|
|
|
|
|
|
Every `payment` is created from a transaction, no exception.
|
|
|
|
|
@@ -27,11 +32,15 @@ flowchart LR
|
|
|
B --> |1000 USD| UserB
|
|
|
```
|
|
|
|
|
|
-After this simple transction, `UserA` cannot longer spend their `1000 USD` and balance, and `UserB` can spend `1000 USD` more.
|
|
|
+After this simple transction, `UserA` cannot longer spend their `1000 USD` and
|
|
|
+balance, and `UserB` can spend `1000 USD` more.
|
|
|
|
|
|
-This data model does not care how many payments are being spend or created, as long as the amounts are the same on both ends.
|
|
|
+This data model does not care how many payments are being spend or created, as
|
|
|
+long as the amounts are the same on both ends.
|
|
|
|
|
|
-In the following example UserA transfer `1000 USD` to `UserB`, but `1 USD` is deducted from the transfer by the system and that is being transfer to the `FeeManager` account.
|
|
|
+In the following example UserA transfer `1000 USD` to `UserB`, but `1 USD` is
|
|
|
+deducted from the transfer by the system and that is being transfer to the
|
|
|
+`FeeManager` account.
|
|
|
|
|
|
```mermaid
|
|
|
flowchart LR
|
|
@@ -42,7 +51,10 @@ flowchart LR
|
|
|
|
|
|
### Multi-step transactions
|
|
|
|
|
|
-As mentioned before, the transaction can spend multiple payments and can create multiple as well. As long as the amounts are equal on both ends (in this case `1000 USD, 980 EUR` is equal to `999USD + 1 USD, 979EUR + 1EUR`), the transaction will be valid.
|
|
|
+As mentioned before, the transaction can spend multiple payments and can create
|
|
|
+multiple as well. As long as the amounts are equal on both ends (in this case
|
|
|
+`1000 USD, 980 EUR` is equal to `999USD + 1 USD, 979EUR + 1EUR`), the
|
|
|
+transaction will be valid.
|
|
|
|
|
|
```mermaid
|
|
|
flowchart LR
|
|
@@ -54,20 +66,28 @@ flowchart LR
|
|
|
B --> |1 EUR| FeeManager
|
|
|
```
|
|
|
|
|
|
-When the transaction will be attempted to be persisted, the storage layer will make sure to flag `UserA'` and `UserB'` payments. If that operation fails, the whole transaction creation fails.
|
|
|
+When the transaction will be attempted to be persisted, the storage layer will
|
|
|
+make sure to flag `UserA'` and `UserB'` payments. If that operation fails, the
|
|
|
+whole transaction creation fails.
|
|
|
|
|
|
## Concurrency model
|
|
|
|
|
|
-Because Verax is heavily inspired in Bitcoin's model, the concurrency model is quite simple. When a new transaction is commited into the database, each payment in `input` section is attempted to be spent (altering their `spend_by` field from `None` to `Some(new_transaction_id)`). If any `payment is already spent, or not valid, the whole transaction creation fails and a rollback is issued upper stream. The storage layer ensures that transaction creation and updates are atomic and updates.
|
|
|
+Because Verax is heavily inspired in Bitcoin's model, the concurrency model is
|
|
|
+quite simple. When a new transaction is commited into the database, each payment
|
|
|
+in `input` section is attempted to be spent (altering their `spend_by` field
|
|
|
+from `None` to `Some(new_transaction_id)`). If any `payment is already spent, or
|
|
|
+not valid, the whole transaction creation fails and a rollback is issued upper
|
|
|
+stream. The storage layer ensures that transaction creation and updates are
|
|
|
+atomic and updates.
|
|
|
|
|
|
```mermaid
|
|
|
sequenceDiagram
|
|
|
- Transaction->>+ DB:
|
|
|
+ Transaction ->>+ DB:s
|
|
|
critical Spend each input
|
|
|
loop Spend Inputs
|
|
|
Transaction ->>+ DB: Spend input
|
|
|
DB ->>+ Transaction: OK
|
|
|
- end
|
|
|
+ end
|
|
|
DB ->>+ Transaction: OK
|
|
|
loop Output
|
|
|
Transaction ->>+ DB: Creates new output
|
|
@@ -79,13 +99,23 @@ sequenceDiagram
|
|
|
DB ->>- Transaction: Error
|
|
|
Transaction ->>+ DB: Rollback
|
|
|
end
|
|
|
-
|
|
|
-
|
|
|
+
|
|
|
+
|
|
|
```
|
|
|
|
|
|
-Because of the `input` and `output` model, there is no need to check if the account has enough balance, and there is no need to enforce any locking mechanism, as long as each selected `payment` can be spendable at transaction storing time, the transaction will be created atomically. Each payment in the `inputs` must spendable, or else the whole operation fails, because every update is atomic, ensured by the storage layer.
|
|
|
+Because of the `input` and `output` model, there is no need to check if the
|
|
|
+account has enough balance, and there is no need to enforce any locking
|
|
|
+mechanism, as long as each selected `payment` can be spendable at transaction
|
|
|
+storing time, the transaction will be created atomically. Each payment in the
|
|
|
+`inputs` must spendable, or else the whole operation fails, because every update
|
|
|
+is atomic, ensured by the storage layer.
|
|
|
|
|
|
The conditions for a payment ot be spendable are:
|
|
|
|
|
|
-* It must be unspent: Payments are spendable once.
|
|
|
-* It must be finalized: This means that the transaction which created this new `payment` is settled. Any other state is not acceptable and will render this payment not usable.
|
|
|
+- It must be unspent: Payments are spendable once.
|
|
|
+- It must be finalized: This means that the transaction which created this new
|
|
|
+ `payment` is settled. Any other state is not acceptable and will render this
|
|
|
+
|
|
|
+No global state knowledge is required to be sure that no asset is being created
|
|
|
+or destroyed by mistake, as long as the inputs are spendable and the sum of
|
|
|
+amounts inside inputs and outputs matches.
|