2017-01-03 13:07:48 +00:00
<!-- If you edit this, then please make the same changes to layout_for_doc_website.html, as that is used for the web
doc site generation which we put analytics tracking on to identify any potential problem pages -->
2016-11-11 11:52:29 +00:00
<!DOCTYPE html>
<!-- [if IE 8]><html class="no - js lt - ie9" lang="en" > <![endif] -->
<!-- [if gt IE 8]><! --> < html class = "no-js" lang = "en" > <!-- <![endif] -->
< head >
< meta charset = "utf-8" >
< meta name = "viewport" content = "width=device-width, initial-scale=1.0" >
2016-11-29 11:38:52 +00:00
< title > Transaction tear-offs — R3 Corda latest documentation< / title >
2016-11-11 11:52:29 +00:00
< link rel = "stylesheet" href = "_static/css/custom.css" type = "text/css" / >
< link rel = "top" title = "R3 Corda latest documentation" href = "index.html" / >
2017-01-26 16:57:23 +00:00
< link rel = "next" title = "Contract catalogue" href = "contract-catalogue.html" / >
< link rel = "prev" title = "Clauses" href = "clauses.html" / >
2016-11-11 11:52:29 +00:00
< script src = "_static/js/modernizr.min.js" > < / script >
< / head >
< body class = "wy-body-for-nav" role = "document" >
< div class = "wy-grid-for-nav" >
< nav data-toggle = "wy-nav-shift" class = "wy-nav-side" >
< div class = "wy-side-scroll" >
< div class = "wy-side-nav-search" >
< a href = "index.html" class = "icon icon-home" > R3 Corda
< / a >
< div class = "version" >
latest
< / div >
< div role = "search" >
< form id = "rtd-search-form" class = "wy-form" action = "search.html" method = "get" >
< input type = "text" name = "q" placeholder = "Search docs" / >
< input type = "hidden" name = "check_keywords" value = "yes" / >
< input type = "hidden" name = "area" value = "default" / >
< / form >
< / div >
< br >
2017-01-19 17:11:18 +00:00
API reference: < a href = "api/kotlin/corda/index.html" > Kotlin< / a > / < a href = "api/javadoc/index.html" > JavaDoc< / a >
2017-01-03 13:07:48 +00:00
< br >
< a href = "https://discourse.corda.net" > Discourse Forums< / a >
< br >
< a href = "http://slack.corda.net" > Slack< / a >
< br >
2016-11-11 11:52:29 +00:00
< / div >
< div class = "wy-menu wy-menu-vertical" data-spy = "affix" role = "navigation" aria-label = "main navigation" >
2016-11-23 12:50:02 +00:00
< p class = "caption" > < span class = "caption-text" > Getting started< / span > < / p >
< ul >
2016-11-11 11:52:29 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "inthebox.html" > What’ s included?< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "getting-set-up.html" > Getting set up< / a > < / li >
2017-01-06 17:38:23 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "getting-set-up-fault-finding.html" > Troubleshooting< / a > < / li >
2016-11-23 12:50:02 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "running-the-demos.html" > Running the demos< / a > < / li >
2016-11-29 18:44:59 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "CLI-vs-IDE.html" > CLI vs IDE< / a > < / li >
2016-11-23 12:50:02 +00:00
< / ul >
< p class = "caption" > < span class = "caption-text" > Key concepts< / span > < / p >
2017-01-26 16:57:23 +00:00
< ul >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts.html" > Overview< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts-ecosystem.html" > Corda ecosystem< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts-data-model.html" > Data model< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts-core-types.html" > Core types< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts-financial-model.html" > Financial model< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts-flow-framework.html" > Flow framework< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts-consensus-notaries.html" > Consensus and notaries< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts-vault.html" > Vault< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "key-concepts-security-model.html" > Security model< / a > < / li >
2016-11-29 18:44:59 +00:00
< / ul >
< p class = "caption" > < span class = "caption-text" > CorDapps< / span > < / p >
< ul >
2017-01-06 17:38:23 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "creating-a-cordapp.html" > CorDapp basics< / a > < / li >
2017-02-08 16:08:46 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "tutorial-cordapp.html" > The example CorDapp< / a > < / li >
2016-11-23 12:50:02 +00:00
< / ul >
< p class = "caption" > < span class = "caption-text" > The Corda node< / span > < / p >
< ul >
< li class = "toctree-l1" > < a class = "reference internal" href = "clientrpc.html" > Client RPC< / a > < / li >
2016-11-11 11:52:29 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "messaging.html" > Networking and messaging< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "persistence.html" > Persistence< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "node-administration.html" > Node administration< / a > < / li >
2016-11-29 11:38:52 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "corda-configuration-file.html" > Node configuration< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "corda-plugins.html" > The Corda plugin framework< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "node-services.html" > Brief introduction to the node services< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "node-explorer.html" > Node Explorer< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "permissioning.html" > Network permissioning< / a > < / li >
2016-11-23 12:50:02 +00:00
< / ul >
2016-11-11 11:52:29 +00:00
< p class = "caption" > < span class = "caption-text" > Tutorials< / span > < / p >
< ul >
< li class = "toctree-l1" > < a class = "reference internal" href = "tutorial-contract.html" > Writing a contract< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "tutorial-contract-clauses.html" > Writing a contract using clauses< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "tutorial-test-dsl.html" > Writing a contract test< / a > < / li >
2017-02-22 10:59:02 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "contract-upgrade.html" > Upgrading Contracts< / a > < / li >
2017-01-06 17:38:23 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "tutorial-integration-testing.html" > Integration testing< / a > < / li >
2016-11-29 11:38:52 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "tutorial-clientrpc-api.html" > Client RPC API tutorial< / a > < / li >
2017-01-06 17:38:23 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "tutorial-building-transactions.html" > Building transactions< / a > < / li >
2016-11-25 12:10:21 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "flow-state-machines.html" > Writing flows< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "flow-testing.html" > Writing flow tests< / a > < / li >
2016-11-29 11:38:52 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "running-a-notary.html" > Running a notary service< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "using-a-notary.html" > Using a notary service< / a > < / li >
2016-11-11 11:52:29 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "oracles.html" > Writing oracle services< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "tutorial-attachments.html" > Using attachments< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "event-scheduling.html" > Event scheduling< / a > < / li >
< / ul >
2016-11-23 12:50:02 +00:00
< p class = "caption" > < span class = "caption-text" > Other< / span > < / p >
2017-01-26 16:57:23 +00:00
< ul class = "current" >
2016-11-23 12:50:02 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "network-simulator.html" > Network Simulator< / a > < / li >
2017-01-26 16:57:23 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "clauses.html" > Clauses< / a > < / li >
< li class = "toctree-l1 current" > < a class = "current reference internal" href = "#" > Transaction tear-offs< / a > < ul >
< li class = "toctree-l2" > < a class = "reference internal" href = "#merkle-trees-in-corda" > Merkle trees in Corda< / a > < / li >
< li class = "toctree-l2" > < a class = "reference internal" href = "#hiding-data" > Hiding data< / a > < / li >
< li class = "toctree-l2" > < a class = "reference internal" href = "#example-of-usage" > Example of usage< / a > < / li >
< / ul >
< / li >
2016-11-11 11:52:29 +00:00
< / ul >
2016-11-23 12:50:02 +00:00
< p class = "caption" > < span class = "caption-text" > Component library< / span > < / p >
2016-11-11 11:52:29 +00:00
< ul >
2016-11-23 12:50:02 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "contract-catalogue.html" > Contract catalogue< / a > < / li >
2016-11-29 11:38:52 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "contract-irs.html" > Interest rate swaps< / a > < / li >
2016-11-11 11:52:29 +00:00
< / ul >
< p class = "caption" > < span class = "caption-text" > Appendix< / span > < / p >
< ul >
2016-11-18 12:57:39 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "loadtesting.html" > Load testing< / a > < / li >
2017-01-06 17:38:23 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "setting-up-a-corda-network.html" > What is a corda network?< / a > < / li >
2016-11-11 11:52:29 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "secure-coding-guidelines.html" > Secure coding guidelines< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "release-process.html" > Release process< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "release-notes.html" > Release notes< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "codestyle.html" > Code style guide< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "building-the-docs.html" > Building the documentation< / a > < / li >
2017-01-26 16:57:23 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "further-notes-on-kotlin.html" > Further notes on Kotlin< / a > < / li >
2017-01-03 13:07:48 +00:00
< li class = "toctree-l1" > < a class = "reference internal" href = "publishing-corda.html" > Publishing Corda< / a > < / li >
< li class = "toctree-l1" > < a class = "reference internal" href = "azure-vm.html" > Working with the Corda Demo on Azure Marketplace< / a > < / li >
2016-11-11 11:52:29 +00:00
< / ul >
2016-11-23 12:50:02 +00:00
< p class = "caption" > < span class = "caption-text" > Glossary< / span > < / p >
< ul >
< li class = "toctree-l1" > < a class = "reference internal" href = "glossary.html" > Glossary< / a > < / li >
< / ul >
2016-11-11 11:52:29 +00:00
< / div >
< / div >
< / nav >
< section data-toggle = "wy-nav-shift" class = "wy-nav-content-wrap" >
< nav class = "wy-nav-top" role = "navigation" aria-label = "top navigation" >
< i data-toggle = "wy-nav-top" class = "fa fa-bars" > < / i >
< a href = "index.html" > R3 Corda< / a >
< / nav >
< div class = "wy-nav-content" >
< div class = "rst-content" >
< div role = "navigation" aria-label = "breadcrumbs navigation" >
< ul class = "wy-breadcrumbs" >
< li > < a href = "index.html" > Docs< / a > » < / li >
2016-11-29 11:38:52 +00:00
< li > Transaction tear-offs< / li >
2016-11-11 11:52:29 +00:00
< li class = "wy-breadcrumbs-aside" >
< a href = "_sources/merkle-trees.txt" rel = "nofollow" > View page source< / a >
< / li >
< / ul >
< hr / >
< / div >
< div role = "main" class = "document" itemscope = "itemscope" itemtype = "http://schema.org/Article" >
< div itemprop = "articleBody" >
< div class = "section" id = "transaction-tear-offs" >
2016-11-29 11:38:52 +00:00
< h1 > Transaction tear-offs< a class = "headerlink" href = "#transaction-tear-offs" title = "Permalink to this headline" > ¶< / a > < / h1 >
2016-11-11 11:52:29 +00:00
< p > One of the basic data structures in our platform is a transaction. It can be passed around to be signed and verified,
also by third parties. The construction of transactions assumes that they form a whole entity with input and output states,
commands and attachments inside. However all sensitive data shouldn’ t be revealed to other nodes that take part in
the creation of transaction on validation level (a good example of this situation is the Oracle which validates only
2017-01-06 17:38:23 +00:00
embedded commands). How to achieve it in a way that convinces the other party the data they got for signing really did form
2016-11-11 11:52:29 +00:00
a part of the transaction?< / p >
< p > We decided to use well known and described cryptographic scheme to provide proofs of inclusion and data integrity.
Merkle trees are widely used in peer-to-peer networks, blockchain systems and git.
You can read more on the concept < a class = "reference external" href = "https://en.wikipedia.org/wiki/Merkle_tree" > here< / a > .< / p >
< div class = "section" id = "merkle-trees-in-corda" >
< h2 > Merkle trees in Corda< a class = "headerlink" href = "#merkle-trees-in-corda" title = "Permalink to this headline" > ¶< / a > < / h2 >
2017-02-08 16:08:46 +00:00
< p > Transactions are split into leaves, each of them contains either input, output, command or attachment. Additionally, in
transaction id calculation we use other fields of < code class = "docutils literal" > < span class = "pre" > WireTransaction< / span > < / code > like timestamp, notary, type and signers.
Next, the Merkle tree is built in the normal way by hashing the concatenation of nodes’ hashes below the current one together.
It’ s visible on the example image below, where < code class = "docutils literal" > < span class = "pre" > H< / span > < / code > denotes sha256 function, “ +” - concatenation.< / p >
2016-11-11 11:52:29 +00:00
< img alt = "_images/merkleTree.png" src = "_images/merkleTree.png" / >
2017-02-08 16:08:46 +00:00
< p > The transaction has two input states, one of output, attachment and command each and timestamp. For brevity we didn’ t
include all leaves on the diagram (type, notary and signers are presented as one leaf labelled Rest - in reality
they are separate leaves). Notice that if a tree is not a full binary tree, leaves are padded to the nearest power
of 2 with zero hash (since finding a pre-image of sha256(x) == 0 is hard computational task) - marked light green above.
Finally, the hash of the root is the identifier of the transaction, it’ s also used for signing and verification of data integrity.
2016-11-11 11:52:29 +00:00
Every change in transaction on a leaf level will change its identifier.< / p >
< / div >
< div class = "section" id = "hiding-data" >
< h2 > Hiding data< a class = "headerlink" href = "#hiding-data" title = "Permalink to this headline" > ¶< / a > < / h2 >
< p > Hiding data and providing the proof that it formed a part of a transaction is done by constructing Partial Merkle Trees
(or Merkle branches). A Merkle branch is a set of hashes, that given the leaves’ data, is used to calculate the root’ s hash.
Then that hash is compared with the hash of a whole transaction and if they match it means that data we obtained belongs
to that particular transaction.< / p >
< img alt = "_images/partialMerkle.png" src = "_images/partialMerkle.png" / >
2017-02-08 16:08:46 +00:00
< p > In the example above, the node < code class = "docutils literal" > < span class = "pre" > H(f)< / span > < / code > is the one holding command data for signing by Oracle service. Blue leaf < code class = "docutils literal" > < span class = "pre" > H(g)< / span > < / code > is also
included since it’ s holding timestamp information. Nodes labelled < code class = "docutils literal" > < span class = "pre" > Provided< / span > < / code > form the Partial Merkle Tree, black ones
are omitted. Having timestamp with the command that should be in a violet node place and branch we are able to calculate
root of this tree and compare it with original transaction identifier - we have a proof that this command and timestamp
belong to this transaction.< / p >
2016-11-11 11:52:29 +00:00
< / div >
< div class = "section" id = "example-of-usage" >
< h2 > Example of usage< a class = "headerlink" href = "#example-of-usage" title = "Permalink to this headline" > ¶< / a > < / h2 >
< p > Let’ s focus on a code example. We want to construct a transaction with commands containing interest rate fix data as in:
< a class = "reference internal" href = "oracles.html" > < span class = "doc" > Writing oracle services< / span > < / a > .
After construction of a partial transaction, with included < code class = "docutils literal" > < span class = "pre" > Fix< / span > < / code > commands in it, we want to send it to the Oracle for checking
and signing. To do so we need to specify which parts of the transaction are going to be revealed. That can be done by constructing
2017-02-08 16:08:46 +00:00
filtering function over fields of < code class = "docutils literal" > < span class = "pre" > WireTransaction< / span > < / code > of type < code class = "docutils literal" > < span class = "pre" > (Any)< / span > < span class = "pre" > -> < / span > < span class = "pre" > Boolean< / span > < / code > .< / p >
2016-11-11 11:52:29 +00:00
< div class = "codeset container" >
2017-02-08 16:08:46 +00:00
< div class = "highlight-kotlin" > < div class = "highlight" > < pre > < span > < / span > < span class = "k" > val< / span > < span class = "py" > partialTx< / span > < span class = "p" > =< / span > < span class = "p" > ...< / span >
2016-11-11 11:52:29 +00:00
< span class = "k" > val< / span > < span class = "py" > oracle< / span > < span class = "p" > :< / span > < span class = "n" > Party< / span > < span class = "p" > =< / span > < span class = "p" > ...< / span >
2017-02-08 16:08:46 +00:00
< span class = "k" > fun< / span > < span class = "nf" > filtering< / span > < span class = "p" > (< / span > < span class = "n" > elem< / span > < span class = "p" > :< / span > < span class = "n" > Any< / span > < span class = "p" > ):< / span > < span class = "n" > Boolean< / span > < span class = "p" > {< / span >
< span class = "k" > return< / span > < span class = "k" > when< / span > < span class = "p" > (< / span > < span class = "n" > elem< / span > < span class = "p" > )< / span > < span class = "p" > {< / span >
< span class = "k" > is< / span > < span class = "n" > Command< / span > < span class = "p" > -> < / span > < span class = "n" > oracleParty< / span > < span class = "p" > .< / span > < span class = "n" > owningKey< / span > < span class = "k" > in< / span > < span class = "n" > elem< / span > < span class = "p" > .< / span > < span class = "n" > signers< / span > < span class = "p" > & & < / span > < span class = "n" > elem< / span > < span class = "p" > .< / span > < span class = "n" > value< / span > < span class = "k" > is< / span > < span class = "n" > Fix< / span >
< span class = "k" > else< / span > < span class = "p" > -> < / span > < span class = "k" > false< / span >
< span class = "p" > }< / span >
< span class = "p" > }< / span >
2016-11-11 11:52:29 +00:00
< / pre > < / div >
< / div >
< / div >
< p > Assuming that we already assembled partialTx with some commands and know the identity of Oracle service,
2017-02-08 16:08:46 +00:00
we construct filtering function over commands - < code class = "docutils literal" > < span class = "pre" > filtering< / span > < / code > . It performs type checking and filters only < code class = "docutils literal" > < span class = "pre" > Fix< / span > < / code > commands
as in IRSDemo example. Then we can construct < code class = "docutils literal" > < span class = "pre" > FilteredTransaction< / span > < / code > :< / p >
2016-11-11 11:52:29 +00:00
< div class = "codeset container" >
< div class = "highlight-kotlin" > < div class = "highlight" > < pre > < span > < / span > < span class = "k" > val< / span > < span class = "py" > wtx< / span > < span class = "p" > :< / span > < span class = "n" > WireTransaction< / span > < span class = "p" > =< / span > < span class = "n" > partialTx< / span > < span class = "p" > .< / span > < span class = "n" > toWireTransaction< / span > < span class = "p" > ()< / span >
2017-02-08 16:08:46 +00:00
< span class = "k" > val< / span > < span class = "py" > ftx< / span > < span class = "p" > :< / span > < span class = "n" > FilteredTransaction< / span > < span class = "p" > =< / span > < span class = "n" > wtx< / span > < span class = "p" > .< / span > < span class = "n" > buildFilteredTransaction< / span > < span class = "p" > (< / span > < span class = "n" > filtering< / span > < span class = "p" > )< / span >
2016-11-11 11:52:29 +00:00
< / pre > < / div >
< / div >
< / div >
2017-02-08 16:08:46 +00:00
< p > In the Oracle example this step takes place in < code class = "docutils literal" > < span class = "pre" > RatesFixFlow< / span > < / code > by overriding < code class = "docutils literal" > < span class = "pre" > filtering< / span > < / code > function, see: < a class = "reference internal" href = "oracles.html#filtering-ref" > < span class = "std std-ref" > Using an oracle< / span > < / a > < / p >
2016-11-11 11:52:29 +00:00
< p > < code class = "docutils literal" > < span class = "pre" > FilteredTransaction< / span > < / code > holds < code class = "docutils literal" > < span class = "pre" > filteredLeaves< / span > < / code > (data that we wanted to reveal) and Merkle branch for them.< / p >
< div class = "codeset container" >
2017-02-08 16:08:46 +00:00
< div class = "highlight-kotlin" > < div class = "highlight" > < pre > < span > < / span > < span class = "c1" > // Direct accsess to included commands, inputs, outputs, attachments etc.< / span >
2016-11-11 11:52:29 +00:00
< span class = "k" > val< / span > < span class = "py" > cmds< / span > < span class = "p" > :< / span > < span class = "n" > List< / span > < span class = "p" > < < / span > < span class = "n" > Command< / span > < span class = "p" > > < / span > < span class = "p" > =< / span > < span class = "n" > ftx< / span > < span class = "p" > .< / span > < span class = "n" > filteredLeaves< / span > < span class = "p" > .< / span > < span class = "n" > commands< / span >
< span class = "k" > val< / span > < span class = "py" > ins< / span > < span class = "p" > :< / span > < span class = "n" > List< / span > < span class = "p" > < < / span > < span class = "n" > StateRef< / span > < span class = "p" > > < / span > < span class = "p" > =< / span > < span class = "n" > ftx< / span > < span class = "p" > .< / span > < span class = "n" > filteredLeaves< / span > < span class = "p" > .< / span > < span class = "n" > inputs< / span >
2017-02-08 16:08:46 +00:00
< span class = "k" > val< / span > < span class = "py" > timestamp< / span > < span class = "p" > :< / span > < span class = "n" > Timestamp< / span > < span class = "p" > ?< / span > < span class = "p" > =< / span > < span class = "n" > ftx< / span > < span class = "p" > .< / span > < span class = "n" > filteredLeaves< / span > < span class = "p" > .< / span > < span class = "n" > timestamp< / span >
< span class = "p" > ...< / span >
2016-11-11 11:52:29 +00:00
< / pre > < / div >
< / div >
< / div >
2017-02-22 10:59:02 +00:00
< div class = "highlight-kotlin" > < div class = "highlight" > < pre > < span > < / span > < span class = "k" > fun< / span > < span class = "nf" > sign< / span > < span class = "p" > (< / span > < span class = "n" > ftx< / span > < span class = "p" > :< / span > < span class = "n" > FilteredTransaction< / span > < span class = "p" > ):< / span > < span class = "n" > DigitalSignature< / span > < span class = "p" > .< / span > < span class = "n" > LegallyIdentifiable< / span > < span class = "p" > {< / span >
< span class = "k" > if< / span > < span class = "p" > (!< / span > < span class = "n" > ftx< / span > < span class = "p" > .< / span > < span class = "n" > verify< / span > < span class = "p" > ())< / span > < span class = "p" > {< / span >
2017-02-08 16:08:46 +00:00
< span class = "k" > throw< / span > < span class = "n" > MerkleTreeException< / span > < span class = "p" > (< / span > < span class = "s" > " Rate Fix Oracle: Couldn' t verify partial Merkle tree." < / span > < span class = "p" > )< / span >
< span class = "p" > }< / span >
< span class = "c1" > // Performing validation of obtained FilteredLeaves.< / span >
< span class = "k" > fun< / span > < span class = "nf" > commandValidator< / span > < span class = "p" > (< / span > < span class = "n" > elem< / span > < span class = "p" > :< / span > < span class = "n" > Command< / span > < span class = "p" > ):< / span > < span class = "n" > Boolean< / span > < span class = "p" > {< / span >
< span class = "k" > if< / span > < span class = "p" > (!(< / span > < span class = "n" > identity< / span > < span class = "p" > .< / span > < span class = "n" > owningKey< / span > < span class = "k" > in< / span > < span class = "n" > elem< / span > < span class = "p" > .< / span > < span class = "n" > signers< / span > < span class = "p" > & & < / span > < span class = "n" > elem< / span > < span class = "p" > .< / span > < span class = "n" > value< / span > < span class = "k" > is< / span > < span class = "n" > Fix< / span > < span class = "p" > ))< / span >
< span class = "k" > throw< / span > < span class = "n" > IllegalArgumentException< / span > < span class = "p" > (< / span > < span class = "s" > " Oracle received unknown command (not in signers or not Fix)." < / span > < span class = "p" > )< / span >
< span class = "k" > val< / span > < span class = "py" > fix< / span > < span class = "p" > =< / span > < span class = "n" > elem< / span > < span class = "p" > .< / span > < span class = "n" > value< / span > < span class = "k" > as< / span > < span class = "n" > Fix< / span >
< span class = "k" > val< / span > < span class = "py" > known< / span > < span class = "p" > =< / span > < span class = "n" > knownFixes< / span > < span class = "p" > [< / span > < span class = "n" > fix< / span > < span class = "p" > .< / span > < span class = "n" > of< / span > < span class = "p" > ]< / span >
< span class = "k" > if< / span > < span class = "p" > (< / span > < span class = "n" > known< / span > < span class = "p" > ==< / span > < span class = "k" > null< / span > < span class = "p" > ||< / span > < span class = "n" > known< / span > < span class = "p" > !=< / span > < span class = "n" > fix< / span > < span class = "p" > )< / span >
< span class = "k" > throw< / span > < span class = "n" > UnknownFix< / span > < span class = "p" > (< / span > < span class = "n" > fix< / span > < span class = "p" > .< / span > < span class = "n" > of< / span > < span class = "p" > )< / span >
< span class = "k" > return< / span > < span class = "k" > true< / span >
< span class = "p" > }< / span >
< span class = "k" > fun< / span > < span class = "nf" > check< / span > < span class = "p" > (< / span > < span class = "n" > elem< / span > < span class = "p" > :< / span > < span class = "n" > Any< / span > < span class = "p" > ):< / span > < span class = "n" > Boolean< / span > < span class = "p" > {< / span >
< span class = "k" > return< / span > < span class = "k" > when< / span > < span class = "p" > (< / span > < span class = "n" > elem< / span > < span class = "p" > )< / span > < span class = "p" > {< / span >
< span class = "k" > is< / span > < span class = "n" > Command< / span > < span class = "p" > -> < / span > < span class = "n" > commandValidator< / span > < span class = "p" > (< / span > < span class = "n" > elem< / span > < span class = "p" > )< / span >
< span class = "k" > else< / span > < span class = "p" > -> < / span > < span class = "k" > throw< / span > < span class = "n" > IllegalArgumentException< / span > < span class = "p" > (< / span > < span class = "s" > " Oracle received data of different type than expected." < / span > < span class = "p" > )< / span >
< span class = "p" > }< / span >
< span class = "p" > }< / span >
< span class = "k" > val< / span > < span class = "py" > leaves< / span > < span class = "p" > =< / span > < span class = "n" > ftx< / span > < span class = "p" > .< / span > < span class = "n" > filteredLeaves< / span >
< span class = "k" > if< / span > < span class = "p" > (!< / span > < span class = "n" > leaves< / span > < span class = "p" > .< / span > < span class = "n" > checkWithFun< / span > < span class = "p" > (< / span > < span class = "o" > ::< / span > < span class = "n" > check< / span > < span class = "p" > ))< / span >
< span class = "k" > throw< / span > < span class = "n" > IllegalArgumentException< / span > < span class = "p" > ()< / span >
< span class = "c1" > // It all checks out, so we can return a signature.< / span >
< span class = "c1" > //< / span >
< span class = "c1" > // Note that we will happily sign an invalid transaction, as we are only being presented with a filtered< / span >
< span class = "c1" > // version so we can' t resolve or check it ourselves. However, that doesn' t matter much, as if we sign< / span >
< span class = "c1" > // an invalid transaction the signature is worthless.< / span >
2017-02-22 10:59:02 +00:00
< span class = "k" > return< / span > < span class = "n" > signingKey< / span > < span class = "p" > .< / span > < span class = "n" > signWithECDSA< / span > < span class = "p" > (< / span > < span class = "n" > ftx< / span > < span class = "p" > .< / span > < span class = "n" > rootHash< / span > < span class = "p" > .< / span > < span class = "n" > bytes< / span > < span class = "p" > ,< / span > < span class = "n" > identity< / span > < span class = "p" > )< / span >
2017-02-08 16:08:46 +00:00
< span class = "p" > }< / span >
< / pre > < / div >
< / div >
< p > Above code snippet is taken from < code class = "docutils literal" > < span class = "pre" > NodeInterestRates.kt< / span > < / code > file and implements a signing part of an Oracle.
You can check only leaves using < code class = "docutils literal" > < span class = "pre" > leaves.checkWithFun< / span > < span class = "pre" > {< / span > < span class = "pre" > check(it)< / span > < span class = "pre" > }< / span > < / code > and then verify obtained < code class = "docutils literal" > < span class = "pre" > FilteredTransaction< / span > < / code >
to see if data from < code class = "docutils literal" > < span class = "pre" > PartialMerkleTree< / span > < / code > belongs to < code class = "docutils literal" > < span class = "pre" > WireTransaction< / span > < / code > with provided < code class = "docutils literal" > < span class = "pre" > id< / span > < / code > . All you need is the root hash
of the full transaction:< / p >
2016-11-11 11:52:29 +00:00
< div class = "codeset container" >
< div class = "highlight-kotlin" > < div class = "highlight" > < pre > < span > < / span > < span class = "k" > if< / span > < span class = "p" > (!< / span > < span class = "n" > ftx< / span > < span class = "p" > .< / span > < span class = "n" > verify< / span > < span class = "p" > (< / span > < span class = "n" > merkleRoot< / span > < span class = "p" > )){< / span >
< span class = "k" > throw< / span > < span class = "n" > MerkleTreeException< / span > < span class = "p" > (< / span > < span class = "s" > " Rate Fix Oracle: Couldn' t verify partial Merkle tree." < / span > < span class = "p" > )< / span >
< span class = "p" > }< / span >
< / pre > < / div >
< / div >
< / div >
2017-02-08 16:08:46 +00:00
< p > Or combine the two steps together:< / p >
< div class = "codeset container" >
< div class = "highlight-kotlin" > < div class = "highlight" > < pre > < span > < / span > < span class = "n" > ftx< / span > < span class = "p" > .< / span > < span class = "n" > verifyWithFunction< / span > < span class = "p" > (< / span > < span class = "n" > merkleRoot< / span > < span class = "p" > ,< / span > < span class = "o" > ::< / span > < span class = "n" > check< / span > < span class = "p" > )< / span >
< / pre > < / div >
< / div >
< / div >
2016-11-11 11:52:29 +00:00
< div class = "admonition note" >
< p class = "first admonition-title" > Note< / p >
< p class = "last" > The way the < code class = "docutils literal" > < span class = "pre" > FilteredTransaction< / span > < / code > is constructed ensures that after signing of the root hash it’ s impossible to add or remove
leaves. However, it can happen that having transaction with multiple commands one party reveals only subset of them to the Oracle.
2017-01-06 17:38:23 +00:00
As signing is done now over the Merkle root hash, the service signs all commands of given type, even though it didn’ t see
2016-11-11 11:52:29 +00:00
all of them. This issue will be handled after implementing partial signatures.< / p >
< / div >
< / div >
< / div >
< / div >
< / div >
< footer >
< div class = "rst-footer-buttons" role = "navigation" aria-label = "footer navigation" >
2017-01-26 16:57:23 +00:00
< a href = "contract-catalogue.html" class = "btn btn-neutral float-right" title = "Contract catalogue" accesskey = "n" > Next < span class = "fa fa-arrow-circle-right" > < / span > < / a >
2016-11-11 11:52:29 +00:00
2017-01-26 16:57:23 +00:00
< a href = "clauses.html" class = "btn btn-neutral" title = "Clauses" accesskey = "p" > < span class = "fa fa-arrow-circle-left" > < / span > Previous< / a >
2016-11-11 11:52:29 +00:00
< / div >
< hr / >
< div role = "contentinfo" >
< p >
2016-11-29 18:44:59 +00:00
© Copyright 2016, R3 Limited.
2016-11-11 11:52:29 +00:00
< / p >
< / div >
Built with < a href = "http://sphinx-doc.org/" > Sphinx< / a > using a < a href = "https://github.com/snide/sphinx_rtd_theme" > theme< / a > provided by < a href = "https://readthedocs.org" > Read the Docs< / a > .
< / footer >
< / div >
< / div >
< / section >
< / div >
< script type = "text/javascript" >
var DOCUMENTATION_OPTIONS = {
URL_ROOT:'./',
VERSION:'latest',
COLLAPSE_INDEX:false,
FILE_SUFFIX:'.html',
HAS_SOURCE: true
};
< / script >
< script type = "text/javascript" src = "_static/jquery.js" > < / script >
< script type = "text/javascript" src = "_static/underscore.js" > < / script >
< script type = "text/javascript" src = "_static/doctools.js" > < / script >
< script type = "text/javascript" src = "_static/js/theme.js" > < / script >
< script type = "text/javascript" >
jQuery(function () {
SphinxRtdTheme.StickyNav.enable();
});
< / script >
< / body >
< / html >