This document introduces the background and guidance regarding Gate's audit process of Proof-of-Reserves, in order to transparently prove to customers that Gate held full reserves of their funds.
Report Release Date | Snapshot Time | Audit Company | Currency | Report | Website | Status |
---|---|---|---|---|---|---|
May 11, 2020 | May 04, 2020, 00:00 UTC | Armanino LLP | BTC | Gate.io Proof-of-Reserves Assessment Report [BTC] [May-25-2020] | Trust Explorer - Proof of Reservers (May-25-2020) | Released |
Oct 28, 2022 | Oct 19, 2022, 00:00 UTC | Armanino LLP | BTCÐ | Gate.io Proof-of-Reserves Assessment Report [BTCÐ] [Oct-28-2022] | Trust Explorer - Proof of Reservers (Oct-28-2022) | Released |
One of the core problems with cryptocurrency exchanges is transparency, which primarily involves in the proof of reserves. Because customers need to know and confirm that the service they are using does in fact hold 100% of their funds. Hence, Gate came up with this solution utilizing the Merkle tree approach to give customers the ability to verify their fund is fully held by Gate; besides, an independent and cryptographically-verified audit was employed to help with the audit process.
-
Gate provides the auditor with all the details of user balances on a token basis. The auditor will then import the user balances into generator.html to generate the Merkle tree, as shown below:
-
After the Merkle tree successfully generated in generator.html, its root hash together with user count and total amount of user balances will be calculated and displayed to the auditor for verification.
The leaves of the Merkle tree will be saved in a plain text file, which will be publicly shared to customers on Github to verify individual's account balance.
-
User needs to first get the published Merkle tree from Github, import into verifier.html, and then input his/her own hashed User ID and token balance to trigger the verification process. The hashed user id can be retrieved at https://www.gate.io/myaccount/myavailableproof. If the hashed UID and balance provided by user matches the record in the Merkle tree, a successful result will be displayed together with the node location of user information within the Merkle tree. The Merkle tree's root hash will be re-calculated using the imported file so that user can verify the root hash to ensure the correctness and completeness of the Merkle tree.
-
In cryptography and computer science, a hash tree or Merkle tree is a tree in which every leaf node is labelled with the cryptographic hash of a data block, and every non-leaf node is labelled with the hash of the labels of its child nodes. Hash trees allow efficient and secure verification of the contents of large data structures.
-
Hashed user id (UID) and user balances are first exported from Gate's database. Each pair of hashed UID and user balance will be hashed respectively, and then concatenated to form the underlying data block. For each data block, the same hash function will be applied to generate the leaf nodes of the Merkle tree. The resulting hashed data are subsequently hashed together in pairs to create the parent nodes of the leaf nodes. This process continues until it results in a single hash known as the merkle root. Please refer to the diagram below for illustration. After the merkle tree is successfully built, the leaf nodes will be exported into a plain text file, which will be published together with the merkle root hash by the auditor.
-
In order to verify the input hashed user id (UID) and user balance, we need to construct a merkle proof to verify the inclusion of such data within the Merkle tree.
Merkle proofs are established by hashing the concatenation of hashed UID and hashed user balance, and climbing up the tree until obtaining the root hash, which has been published by the auditor in step #2.
The merkle proofs are explained with following example.
In order to verify the inclusion of data (UID, Balance) from user input, in the merkle tree root, we use a one way function to hash the hashed value of UID and Balance to obtain data K, and then we apply the same hash function on data K to obtain H(K), denoted by K'. In order to validate the inclusivity of K, K doesn't have to be revealed, similarly, the hash of data A can be revealed without any implicit security repercussions and so on.
Taking the calculations steps below:
- K' when hashed with the hash of the unknown dataset A, yields A'K', which is H(A' + K')
- A'K' hashed with C'D' leads to the root, H(A'K' + C'D')
- Compare the value of H(A'K' + C'D') with the published merkle root hash
Hence, we can prove whether the uer input data of (hashed UID, Balance) is present or not in our merkle tree, wihtout having to reveal any other customer's user id or balance.
Install dependencies
npm install
Install build tool
npm install -g browserify watchify
Create bundle.js to make it runanble in browser
browserify app.js -o bundle.js
To achieve auto build of bundle.js, use watchify as shown below, or use nohup to make watchify command running at background
nohup watchify app.js -o bundle.js -v > nohup.out 2>&1 </dev/null &
- Open
generator.html
in browser, import file with UID and user balances to build Merkle tree - Open
verifier.html
in browser, to validate UID and balance combination
app.js
core logic to build Merkle tree and perform validation- The js function at the top handles interaction between js code and HTML. It receives user actions via HTML events, such as uploading raw user balance, creating Merkle tree, uploading Merkle tree and verifying user balance, and process the user input, then dispatch it to corresponding functions for further processing.
- Function
bufferToString()
handles String conversion to hex format. Since the Merkle tree node values, while retrieved from the buffer, are all in binary format. - Function
createMerkle()
does four things below:- Reads the hashed user id and user balance provided in a plain text file, which was exported from Gate's database
- Then each pair of hashed user id and corresponding balance will be hashed (SHA256 algorithm used in our code) respectively, concatenated and then hashed again to form the leaf nodes of the Merkle tree. In order to reduce the space usage as well as the size of output file, only the first 16 bits of the hashed values will be kept. Then, each pair of the leaf nodes will be hashed and concatenated to form their parent node. This process continues until only one parent node (the root node) left.
- Calculate the total user balance and root hash of the Merkle tree, and display in generator.html.
- Save all leave node values of the Merkle tree in a plain text file for future verification by individuals and auditors.
- Function
verifyMerkle()
does two validations as below:- Validate if the provided hashed user id and user balance can be found in the leave nodes of the Merkle tree. This validation first computes the hashed value of provided hashed user id and user balance, and look up the hashed value in the leaves nodes that saved from createMerkle() function.
- Only after step 1 succeed, then verify the hashed value is within the Merkle tree that generated in createMerkle() function. Verification was processed by the library api verify(), provided in merkletreejs.
FileSaver.js
plugin to save filesgenerator.html
html page to build Merkle tree and calculate merkle root hashverifier.html
html page to validate hashed user id and user balancepackage.json
holds various metadata relevant to the project and handle the project's dependencies
Copyright 2020 © Gate Technology Inc.. All rights reserved.
Licensed under the GPLv3 license.