如果你想要入門閃電網路編程,又不在乎所謂正確、最佳的做法,這篇文很適合你。換言之,這篇指南具有很強的 主觀性,如果你偏好 O’Reilly 式的權威文本,這篇文章可能不對你的胃口,但是,如果你想要了解閃電網路的最小開發環境及其基本內容,不妨讀一讀。我寫本文的目的是提煉出創建閃電網路應用的基礎知識點,讓新手能夠快速構建和實驗。
我相信,只要掌握了基礎知識點,你就能編寫出更加復雜安全的應用,但是,每個人都得有個開始,我個人的感受是萬事開頭難,我認為將這些思想整合到一起可以幫助一些迷途羔羊緩解痛苦,
在深入介紹細節之前,請允許我簡單介紹一下自己的背景:
大約一個月之前,我辭去了原本穩定的數據架構師工作,開始研究日新月異的比特幣生態。你會找到這篇文章,表明你已經邁出了這一步(或者正在考慮這么做),
簡單來說,我認為比特幣是我所見過最有趣的技術創新。我越了解比特幣,就越被它吸引。比特幣技術有希望改變乃至重塑如今的交互方式,使之脫胎換骨,我希望貢獻自己的一份力量,
這就是為什么我會選擇在比特幣上開發。我希望這篇文章能夠讓不熟悉閃電網路生態及其潛力的開發者受益,鼓勵更多開發者為比特幣領域注入創造力,
老實說,關于比特幣生態,我想寫的實在太多了(其中,“由能源支持的貨幣”這一概念排在第一位),但是只能擇日再談了,現在開始進入正題:
閃電網路的架構
-圖源:lnrouter.app-
閃電網路的基礎設計是利用通道實現節點間的一對一連接,開通通道需要存入一定數量的比特幣,該事務會記錄在比特幣區塊鏈上。然后,用戶可以使用特殊的鏈下合約在節點之間轉移通道內的比特幣,無需提交新的事務到區塊鏈上,
實際做法是實時追蹤通道內的余額變化,例如,你在與 Alice 開通的通道內存入 100 satoshi,你有大約 100 satoshi (減去鏈上交易手續費)的轉賬額度,既可以直接轉給 Alice ,也可通過 Alice 間接轉給其他人。這些轉賬幾乎是即時的,因為我們只需根據每筆交易調整通道的余額。這種調整是通過在付款方和收款方之間傳遞加密消息實現的。
閃電應用是在閃電網路(所謂的比特幣二層)上運行的。閃電應用同樣以比特幣作為支付方式,但是結算速度更快、交易費更低,
雖然我只是非常簡略地概括了閃電網路協議,但是知道這些已經足夠入門了,如果你想更詳細地了解閃電網路是如何運作的,這里有一些很好的閱讀材料。小心不要沉溺在知識的海洋里,坐在電腦前讀到很晚,卻一行代碼都沒有敲,也不是說這樣一定不好,只是我們的目的還是在于實踐,另外,保證充足的睡眠也很重要。
我認為,若想了解閃電網路及其節點的運作方式,更好的辦法是自己運行一個節點并弄清楚如何通過節點路由付款。雖然這個辦法需要付出更多時間和金錢,但是既有趣又長知識。1
實現
閃電網路是開放式協議,這就意味著,閃電網路有標準的消息通信和解釋方式,就像 HTTPS 和 TCP/IP 不歸任何人所有那樣,閃電網路協議也是如此。任何人只要遵守標準都可以參與閃電網路。不僅如此,隨著開發者不斷加固協議并增加新功能,這一標準也會持續發展,
目前有四家公司在閃電網路協議開發中處于核心地位:Lightning Labs、Blockstream、ACINQ 和 Square Crypto,而且都有自己的實現:
Lightning Labs — lnd (Go)
Blockstream — c-lighting (C)
ACINQ — eclair (Scala)
Square Crypto — Rust Lightning (Rust)
上述所有實現都能通過閃電網路進行通信,區別在于這些實現的 API,本文只聚焦于 Lightning Labs 的 lnd,因為這是我最先接觸、也是了解最多的。雖然我最終選擇 Ind 有偶然成分在里面,但是不得不夸一下,Lightning Labs 的 Ind 文檔寫得很好,而且開發者能夠通過官方 Slack 頻道獲得很多幫助和支持,
請注意,下文將圍繞 Ind 展開討論,但是基本概念適用于所有閃電網路實現。
你的開發環境
不廢話,使用 Polar 就行了,
好吧,再多說兩句,無論你是開發者還是用戶,在閃電網路上進行構建的最大障礙是如何創建后端,我說的不只是一個錢包,因為有很多簡單快速的托管解決方案可以用,我指的是真正的后端 —— 運行比特幣和閃電網路的節點,
我之所以這么說,是因為你的應用需要擁有節點的某些權限,才能完成一些重要操作,例如,通過 API/gRPC 調用創建發票和監控付款。如果有節點運營者授予你節點的訪問權限,你同樣可以完成這些操作,但還是自己運營一個節點比較好,
能夠用于閃電網路開發的節點必須運行兩個進程:(1)同步到比特幣區塊鏈最新區塊的比特幣客戶端;(2)已開啟通道(可以收發支付)的閃電網路客戶端,
如果你使用的是普通的家用寬頻,光是同步區塊鏈就需要花費至少幾天的時間。你可以試一下 Neutrino(一款輕量級比特幣錢包),但是我聽說用它在閃電網路上開發可能會導致一些問題,我沒有親身嘗試過。不管怎樣,如果你才剛開始接觸這類工具,我想最好去除那些可能會產生負面影響的變量。因此,如果你負擔得起,我建議你同步完整的區塊鏈就好。
如果我們對比特幣的分層進行排序,閃電網路層就是比特幣的二層。正如我上文提到的,我們需要在閃電網路上開啟通道來進行交易。雖然在閃電網路上開啟通道不是很難,但是你需要一些計劃和協調,還要付出比特幣,你當然可以質押比特幣來開啟通道,然后開始交易,但是,你沒必要承擔這種金融風險,2
暫時先別管這些,Polar 是一個很棒的工具,可以讓你在筆記本電腦上的 Docker 容器內模擬運行這些進程,另外,它還提供一個精美的 UI 界面,為你展示網路可視化效果。
-你看!是不是等不及想同步了!(Polar 界面)-
Polar 的優點在于,當你準備好將應用轉移到測試網或主網后端時,你只需要更新一些配置即可。我已經可以做到在本地、測試網和主網后端之間輕松切換,只需在應用中添加或注釋掉幾行代碼即可。(我最后會得到 .env 文件,當然了,測評工具不是本文的目的。)
Polar 也支持 lnd、c-lighting 和 eclair 節點,這就意味著你可以靈活嘗試其它實現,
再重申一遍:使用 Polar 就好。這是一種快速且簡單的入門方法。再者說,萬一你根本不喜歡在閃電網路上開發呢?早日發現這一點,總好過在一個你不一定會使用的系統上浪費過多時間和金錢。話說回來,如果你發現自己沉迷于 Stephan Livera(知名比特幣主播)的播客,每晚收聽不可自拔,可以搭建一個更嚴肅的開發環境,
設置 Polar
點擊 “Create Network(創建網路)”,即可在 Polar 中輕松創建模擬閃電網路環境。你可以隨意命名這個網路(我個人認為 “test(測試)” 是個聰明的選擇)并添加兩個 lnd 節點和一個 Bitcoin Core 節點,實際上,對于剛入門的人來說,這些只是最低配,之后還可以添加更多節點,你創建好網路后就可以啟動它。首次啟動網路時需要的時間可能會長一些,因為你必須下載所有 Docker 鏡像。順帶一提,你需要在自己的設備上運行 Docker。就像我之前說的,這里還是得靠自己摸索,
最后,你會看到一個類似下圖的 Polar 界面:
在上圖所示界面中,我們可以看到 Alice 和 Bob 的 Ind 節點,而且這兩個節點都與 Bitcoin Core 后端進程相連。另外還要注意的是,這時的區塊高度是 1,這是我們的創世塊!不同于實時區塊鏈,Polar 只有在開發和測試過程中提交交易時才會出塊,
下一步是在 Alice 和 Bob 之間創建一條通道,用來進行閃電付款,點擊 Alice 的節點,右側控制面板會顯示我們可以與該節點交互的不同方式。點擊 “Actions(操作)” ,我們就會看到向 Alice 的錢包充值模擬比特幣以及在 Alice 和 Bob 之間開啟通道所需的一切工具。接下來,我們先向 Alice 的錢包充值一些資金,再開啟 Alice 與 Bob 的通道吧。
點擊 “Deposit(充值)” ,接著將 100 萬 satoshi 充值進 Alice 的錢包。
如果一切按計劃進行,你就會看到區塊高度和 Alice 的錢包余額發生了變化。如果沒有,請尋求幫助。這不是你的錯,3
現在,我們可以在 Alice 和 Bob 之間開啟通道。現在,Alice 已經有錢了,我們可以讓她開啟一個與 Bob 的 “Outgoing(轉出)” 通道。所謂的轉出通道,就是 Alice 可以通過該通道轉給 Bob 一定數量的比特幣,Alice 和 Bob 之間可以進行多筆轉賬交易,只要通道內有余額即可。但是,請注意,該通道剛開啟時,只有 Alice 可以向 Bob 轉賬,因為可用余額都在她那里。當然了,等到 Alice 向 Bob 轉過賬之后,Bob 就可以向 Alice 轉賬了,
這種通道內資金管理方式是為了保證雙方都具備交易能力,這就是我們所說的閃電網路的 流動性,實際上,用戶只能使用通道內的余額,更深入一點來講,只有當付款方和收款方之間的通道內有足夠的余額(且資金流向正確)時,付款才能成功,否則付款就會失敗。如何管理閃電網路中有限的流動性以及如何找到變通之法本身就是一個值得探討的話題。
坦白來說,我在使用 Polar 開啟通道時遇到過一些問題,我懷疑這是 UI 和后端之間的狀態差異導致的,但是我并不確定。有時,關閉并重啟節點(或整個網路)會有幫助。完全退出并重啟 Polar 同樣有效,另外,遇到這類情況時,出門晃悠 10 分鐘可以讓你的頭腦和精神得到很好的放松,
不管是什么原因,我發現進行這些節點操作的最佳方式是通過 Polar 提供的 CLI。雖然 UI 不一定會顯示出變化,但是我們可以直接查詢資料庫(可以這么說4)來確定 UI 是否如實反映變化,Polar 讓這一切變得簡單了,我們現在就上手做吧。
首選選中 Alice 的 Ind 節點,進入 “Actions” 界面,點擊 “Terminal(終端)” 下面的 “Launch(啟動)”,接著會出現一個命令提示符,如下圖所示:
現在,我們可以使用該 Ind 節點的 lncli 工具開啟通道、創建發票和付款了。首先,運行下方命令來了解大致情況:
lncli –help
我們可以運行下方命令在 Alice 和 Bob 之間開啟一條余額為 10 萬 satoshi 的通道:
lncli openchannel –node_key
你會看到一個帶有 “funding_txid” 的響應,對應的是 Alice 和 Bob 廣播到我們的模擬比特幣區塊鏈上的充值交易。運行下方命令查看我們新創建的通道:
lncli listchannels
如果該命令返回的列表為空,請嘗試使用( “Actions” 界面下的)比特幣節點挖幾個區塊。這樣做應該有助于確認交易并開啟通道。
現在,我們可以通過 listchannels 響應看到 Alice 和 Bob 之間已經成功開啟通道,且通道余額約為 10 萬 satoshi(減去交易費),這 10 萬不到的 satoshi 就是 Alice 可以通過閃電網路支付給 Bob 的可用余額,
支付流程
迄今為止,閃電網路上最簡單(我相信也是最常見的)支付流程是通過發票(invoice)。發票本質上是一組帶有 “金額” 和 “收款方” 的支付指令,還有其它參數和變體可以幫助支付指令解鎖更多有趣的可能性(例如,hodl 發票和 BOLT12 提案),但是本文只關注最基本的模式,
繼續設置我們的 Polar,我們先創建一個發票,但是這回要用 Bob 的節點。因此,我們要啟動 Bob 的終端并運行以下命令:
lncli addinvoice –amt 100
以上命令創建了一個價值 100 satoshi 的發票(實際上,我在執行這些步驟時遇到了連接錯誤,如果你也遇到同樣的問題,請停止并重啟 Bob 的節點)。我們可以通過返回的響應看到這個發票的資訊:
{ “r_hash”: “7d91cafaba85b6086924142dfd890f350eb53b17b80e2993d0a2ce5ccc7252f1”, “payment_request”: “lnbcrt1u1ps3lu04pp50kgu4746skmqs6fyzsklmzg0x58t2wchhq8zny7s5t89enrj2tcsdqqcqzpgsp55rtlzlf5rt0z5zg34nc2rlcm9mw6nd77x45r85z6zp07qumphr7q9qyyssqzrvxdlsluaeu7esscvv8skcmaly4794j7pg9ytapmn50uukezf4xpqma9758s39wpn4pwk475dztezg4tff8xpylksl4mww57q8hj7cq7s7222”, “add_index”: “1”, “payment_addr”: “a0d7f17d341ade2a0911acf0a1ff1b2edda9b7de356833d05a105fe07361b8fc”}
現在,我們只關注 “payment_request” 部分,因為這部分數據包含 Alice 向 Bob 付款所需的一切資訊,即,付款金額和收款方地址5,
如果我們切回 Alice 的節點終端,就可以得到付款請求并將它作為參數傳遞給下方命令:
lncli sendpayment –pay_req
開始你的冒險征程
這時,你應該已經具備了足夠的基礎知識和工具,可以開始構建應用了,上圖概述了一個簡單的示例應用,使用的正是我們在 Polar 中用來創建并支付發票的 API 調用。這就是搭建一個最基礎的應用架構所需的一切。當然了,這只是一個例子,我們在開發過程中還會遇到許多其它問題(目前還只是開始),但是你會慢慢弄清楚自己需要什么以及如何解決這些問題,
一些額外的建議和參考:
有很多庫可以幫助開發者少寫些樣板代碼、直奔主題。就我本人而言,學習如何使用這些庫帶給我的更多是挫敗感,而非更高的效率。問題主要出在我身上,抽象確實很棒,但前提是你要對被抽象的內容有基本了解,我在起步時還沒有領悟到這一點,我覺得 Ind 的 API 文檔學起來最容易。當我按照這個指南使用 Javascript 編寫 gRPC 客戶端時,我就已經步入正軌了,
如果你想看一個更具體的應用示例,不妨看看 Lightning Labs 構建者指南的教程。如果你熟悉教程中用到的工具 express、mobx 和 React,那么我很推薦這個教程,如果你不熟悉這些工具,你可能不會從這篇教程中得到很大幫助,但還是能夠學到一些東西,我喜歡這個教程的一個原因是,它展示了利用閃電網路(和密碼學證明)構建應用可以實現的一些有趣功能。
腳注
- 如果你想運行節點,Umbrel 很適合初學者。我聽說 MyNode、RaspiBolt 和 RaspiBlitz 也不錯。如果你喜歡修補軟體系統(或 SimCity(模擬城市游戲)),那么運行節點往壞了說是一種有趣的消遣,往好了說是一場高成本且無休止的優化游戲(需要付出真金白銀的那種)。
- 你還可以在測試網上進行實驗,通過比特幣水龍頭獲得一些實驗用比特幣,這些幣一文不值,但是當個守財奴的體驗會很有趣,如果你對自己配置和管理節點不感興趣,可以使用 Voltage 之類的服務,Voltage 是即用即付的云上服務,支持測試網和主網節點,
- 好吧,我們不能百分之百確定,謹慎起見,請查看 lnd Slack 的開發者頻道。我在這里遇到過很多構建并維護這些工具的開發者,你可能會看到我!有人可能經歷過你正在面臨的問題。如果沒有,那就太棒了 —— 大家都能從你的問題中有所收獲。
- 區塊鏈是一個公共資料庫,每個人都有 root 權限,如需了解更多資訊,請查看:https://balajis.com/yes-you-may-need-a-blockchain/
- 關于發票中其它字段的詳細解釋,可以查看這篇總結,如需了解更多關于底層合約的資訊,請閱讀這篇文章,