智能合約本身沒有訪問區塊鏈外部數據的能力。而外部的數據對于大多數智能合約應用場景來說都是至關重要的,所以這一功能的缺失限制了智能合約的更進一步的發展。比如涉及金融,供應鏈,保險,安全等諸多領域的智能合約都依賴于外部事件。智能合約無法獲取關鍵的鏈下事件信息,比如價格變動,物流日期,以及支付能力。沒有這些外部的信息,大多數智能合約的應用都是沒有實際應用價值的。
為什么智能合約無法自主獲取外部數據?
因為區塊鏈網絡是確定性的。智能合約在區塊鏈這種去中心化的,自我調節的基礎設施上運行,其中的任何信息都是確定的,可驗證的。區塊鏈可以正常運行,必須在各個參與方之間達成共識。為了實現這個目標,人們設計了[各種]*共識機制*,比如工作量證明(Proof of Work),權益證明(Proof of Stake),行動證明(Proof of Activity)。這些共識機制使得區塊鏈這一分布式的系統形成一個統一的狀態。
有了這些共識機制,就可以驗證網絡上的交易,確定統一公開賬本的狀態。這種設計允許區塊鏈以公平和安全的方式運行,而無需使用集中式身份驗證。因此,區塊鏈整體上是*確定性狀態機*。
但是區塊鏈外部的數據是非確定性的,因為從某種意義上說,它是通過區塊鏈的歷史無法驗證的值。外部數據會受各種因素的影響動態變化。價格的頻繁變化,公司實時更新物流信息,物流變化的更新,等等。因為這些信息是不確定的,智能合約沒有一種方式可以驗證這些數據進而達成共識。因此,無法確認為真實的數據對區塊鏈沒有任何意義。
如何把外部數據提供給智能合約?
通過區塊鏈中間件,特別是安全可靠的預言機可以實現。預言機扮演者數據代理人的角色,連接外部數據與智能合約。它充當區塊鏈數據API之間的中間層,將數據轉換為區塊鏈可以讀取的格式。此外,預言機還負責驗證外部數據的正確性,因此可信賴的來源(信任最小化)至關重要。
但是,在中心化的預言機服務中,預言機會有被攻擊的可能性(被黑客攻擊,服務停機,數據篡改等),這導致智能合約丟失了確定性和可靠性這一最關鍵的特性,從而使大多數基于現實場景的智能合約用例的不可用。如何解決這一問題呢,答案是去中心化的預言機網絡。或者說是Chainlink。
Chainlink通過提供與智能合約開發者的安全性和可靠性相匹配的去中心化的預言機網絡來解決聯通性問題。通過外部適配器(也被稱為chainlinks),區塊鏈可以安全地與chainlinked API連接。開發人員可以方便地將他們自己的智能合約與預先編寫的Chainlink API套件連進行連接,從而建立一個鏈下的預言機連接。
例如,假設您開發了一個智能合約,可以把代幣發送到一個地址。Chainlink(輸出預言機)通過PayPal發送離線支付。然后,預言機可以基于離線支付在鏈上提供收據,從而完成區塊鏈系統中的交易循環。
有了Chainlink,智能合約現在能夠通過一個去中心化的預言機網絡在大多數現實世界的應用場景中正常運行。Chainlink通過安全可靠得方式滿足智能合約的預設條件,因此所有相關方都可以從智能合約生態系統的巨大潛力中受益。
代碼層面,預言機是如何工作的?
使用預言機需要由足夠數量的LINK代幣,以及一些基本的Solididy知識,Solidity是編寫智能合約的語言。請參考Chainlink的[Solidity接口文檔]來了解Chainlink的所有方法。最后,wield能從Chainlink的預言機請求數據,你需要首先在你的合約中繼承ChainlinkClient合約。你可以通過[這里]的例子作為指導來創建合約,也可以參考[文檔]。
預言機可以幫助智能合約請求和獲取區塊鏈的外部數據。我們通過jobs來執行預言機任務來完成請求。這些jobs有與預言機地址相對應的JobID。這些Job由一系列任務,或稱為[適配器],所組成,在指定JobID發送請求時, 這些任務或適配器定義了要完成的工作。
為了更好地展示預言機如何在代碼層面運行,我們通過一個請求以太網價格的示例智能合約來解釋:
contract MyContract is ChainlinkClient {
address owner;
constructor() public {
// Set the address for the LINK token on the public network
// 設置公共網絡的LINK代幣發行地址
setPublicChainlinkToken();
owner = msg.sender;
}
// Additional functions here.。.
// 其他的函數。..
}
首先,為了能使用Chainlink網絡,你需要在你的合約中繼承ChainlinkClient合約。這是一個測試網和正式網通用的構造函數體。這是因為我們使用了setPublicChainlinkToken()方法,這個方法會根據合約部署的網絡環境,自動的獲取LINK代幣的發行地址。
所有當前的預言機和LINK代幣地址都可以在[這里]。存儲LINK代幣地址后,您可以指定預言機合約地址及其相應的JobID來創建請求。
// Creates a Chainlink request to the specified oracle with a given Job ID
// 通過給定預言機地址和JobID來創建Chainlink請求
function requestEthereumPrice(address _oracle, bytes32 _jobId)
public
onlyOwner
{
// newRequest takes a JobID, a callback address, and callback function as input
// 新的請求需要JobID,回調地址和回調函數作為輸入
Chainlink.Request memory req = buildChainlinkRequest(_jobId, this, this.fulfill.selector);
// Adds a URL with the key “get” to the request parameters
// 添加一個URL設置“get”作為key來請求參數
req.add(“get”, “https://min-api.cryptocompare.com/data/price?fsym=ETH&tsyms=USD”);
// Uses input param (dot-delimited string) as the “path” in the request parameters
// 使用點分隔的字符串作為請求參數中的path
req.add(“path”, “USD”);
// Adds an integer with the key “times” to the request parameters
// 為請求參數設置倍數
req.addInt(“times”, 100);
// Sends the request with 1 LINK to the oracle contract
// 向預言機還有發送1 LINK
sendChainlinkRequestTo(_oracle, req, ORACLE_PAYMENT);
}
請求通過buildChainlinkRequest()方法創建,接受相應的參數填寫到Chainlink.Request結構體中作為負載。你可以使用req.add()向請求添加參數,比如URL。一旦你準備好了所有的必須參數,可以通過sendChainlinkRequestTo()方法發送到特定的預言機合約地址,并支付1 LINK的代幣,作為給節點運營方的獎勵。請注意,在主網上,支付金額是各不相同的,但是為了方便大家理解,我們目前設置了為每次請求花費1 LINK。由于測試網絡上這些代幣沒有任何價值,所以我們可以通過[水龍頭]來獲取。
uint256 constant private EXPECTED_RESPONSES = 3;
uint256[] private prices;
uint256 public avgPrice;
function fulfillEthereumPrice(bytes32 _requestId, uint256 _price)
public
recordChainlinkFulfillment(_requestId)
{
if(prices.push(_price) == EXPECTED_RESPONSES) {
uint256 sum;
for(uint i = 0; i 《 prices.length; i++) {
sum = sum.add(prices[i]);
delete prices[i];
}
avgPrice = sum.div(EXPECTED_RESPONSES);
}
}
當Chainlink節點從指定的端點取回結果后,預言機合約會調用回填方法(fulfillment method)。回填方法應該通過recordChainlinkFulfillment()修改器或validateChainlinkCallback()方法保護起來。這樣可以防止無關的人調用該方法,并且只能根據你的請求填寫相應的結果。
將所有這些組合到一起,就可以完成一個可以在以太坊測試網絡上可以獲取外部數據的預言機合約了。完整代碼見[這里]。
我如何開始使用Chainlink?
想要快速上手,你可以通過[這里]的幫助,通過Truffle部署智能合約。如果你已熟悉智能合約開發,歡迎您隨時查看我們最新的博客文章“[44種通過Chainlink增強您的智能合約的方法]”。
評論
查看更多