Why GraphQL Isn't Sufficient for Enterprise Blockchains and What We Did About It
We looked at the fundamentals of GraphQL in our earlier blog post Why We Combined GraphQL and a Serverless Distributed Ledger. In this post, we're going to dig in a bit deeper on some of the features Vendia has added to improve the GraphQL developer experience and address known limitations to what the core GraphQL specification allows.
Vendia Transactions
GraphQL natively supports running multiple mutations as a part of the same transaction but their order of execution is not guaranteed by the GraphQL specification.
In cases where ordering isn't important, or when you don't need to guarantee both operations succeed or fail as a single unit, you can use a single mutation like the one below. This is completely inline with the GraphQL spec:
Mutation with Multiple Operations
mutation m {
warehouse1: update_Inventory(id: "017d92a7-0ab5-5513-fac1-c50be330f057", input: {quantityWarehouse1: 90, lastUpdated: "2021-12-06T18:30:00Z"}) {
transaction {
transactionId
}
}
warehouse2: update_Inventory(id: "017d92a7-0ab5-5513-fac1-c50be330f057", input: {quantityWarehouse2: 10, lastUpdated: "2021-12-06T18:30:00Z"}) {
transaction {
transactionId
}
}
}
As noted above, the two update_Inventory
operations can occur in any order. The warehouse2
operation could run before the warehouse1
operation. Each operation also runs independently - there is no grouping. The warehouse1
operation could succeed and the warehouse2
operation could fail. Even though they are submitted together, they should be thought of as two completely separate transactions.
These GraphQL limitations led us to create Vendia Transactions. The idea originates with database transactions but we've applied them to a GraphQL interface for enterprise blockchain. Vendia Transactions are a custom GraphQL directive (@vendia_transaction) that can be used to decorate mutations and queries submitted to Vendia Share.
NOTE: Our co-founder, Dr. Tim Wager, has an interesting post that raises the question Are Blockchains Databases?.
When used with a mutation to create, change, or delete data, Vendia Transactions ensure multiple operations are run in a single transaction. They also guarantee the order of the operations. Most importantly for use cases requiring operational certainty, they ensure multiple operations either succeed or fail together as a unit, analogous to relational database transactions.
Mutation with Vendia Transaction
mutation m @vendia_transaction {
warehouse1: update_Inventory(id: "017d92a7-0ab5-5513-fac1-c50be330f057", input: {quantityWarehouse1: 90, lastUpdated: "2021-12-06T18:30:00Z"}) {
transaction {
transactionId
}
}
warehouse2: update_Inventory(id: "017d92a7-0ab5-5513-fac1-c50be330f057", input: {quantityWarehouse2: 10, lastUpdated: "2021-12-06T18:30:00Z"}) {
transaction {
transactionId
}
}
}
When used to query data, Vendia Transactions ensure that all nodes will see the same, consistent value until all nodes have any updates applied. Due to the asynchronous nature of Vendia Share, it may take a brief amount of time for updated values to be reflected on all nodes in a Uni.
Query with Vendia Transaction
query q @vendia_transaction {
get_Inventory(id: "017d92a7-0ab5-5513-fac1-c50be330f057") {
_id
itemName
itemNumber
quantityWarehouse1
quantityWarehouse2
}
}
Take a look at our documentation for more information on using Vendia Transactions.
Conditional Updates
Customers want flexibility when updating or deleting records and may only need to perform such updates when a condition is met. The GraphQL specification does not allow for such logic. For REST developers, this may seem like a feature gap. Conditional requests can be used to update resources when defined validation conditions are met. Vendia Share introduced conditional updates to our GraphQL API to address just this scenario.
Let's work through an example where we want to update the shipment date and item quantity of a purchase order but only if the warehouse has a minimum reserve of at least 100 units available.
Mutation with a Conditional Update
mutation m {
update_PurchaseOrder(id: "017d92a7-0ab5-5513-fac1-c50be330f092", input: {shipmentDate: "2021-02-16T09:00:00Z", quantity: 10}, condition: {warehouseQuantity: {gte: 100}}) {
transaction {
transactionId
}
}
}
You can include other operators like _and
, _not
, and _or
in your condition should you need more complexity.
Examining Changes to Data
Vendia Share captures all changes to data in a Uni in a ledger. This is something that comes out of the box - customers automatically have the ability to determine who introduced change, when it was introduced, and what changes were made. Vendia stores this data in blocks. Uni participants have the ability to issue GraphQL queries to get at data stored in blocks.
Examine blocks in the ledger
query blocksQuery {
listVendia_BlockItems {
Vendia_BlockItems {
blockId
blockHash
previousBlockId
commitTime
transactions {
_id
_owner
mutations
}
}
}
}
This has enormous implications. First, participants have full transparency into when changes are introduced. They are not dependent upon the auditing and reporting capabilities of a central data source. These are two characteristics of what our co-founder, Dr. Tim Wager, has described as a Lean App. Second, because the data can be returned via a GraphQL query, participants can take this data and store it in a downstream search or analytics system.
Conclusion
Vendia has provided a number of capabilities to improve the GraphQL development experience. From transactional semantics to GraphQL subscriptions to exposing data in the built-in ledger, Vendia Share allows developers to easily leverage their existing knowledge of GraphQL while also creating enterprise-grade solutions using the additional GraphQL features provided by Vendia Share.
We've expanded our code examples to highlight these three capabilities using several programming languages.