|
|
本帖最后由 赤膊医数 于 2021-4-22 23:41 编辑 : R$ X- x) R! r* @( E7 m
5 @$ k2 s7 A, A3 p
2018年,最火爆且值得期待的区块链项目莫过于EOS和IPFS。1、IPFS简介- IPFS是什么
. ~& C m) R0 s9 A9 {星际文件系统IPFS(InterPlanetary File System)是一个面向全球的、点对点的分布式版本文件系统,目标是为了补充(甚至是取代)目前统治互联网的超文本传输协议(HTTP),将所有具有相同文件系统的计算设备连接在一起。原理用基于内容的地址替代基于域名的地址,也就是用户寻找的不是某个地址而是储存在某个地方的内容,不需要验证发送者的身份,而只需要验证内容的哈希,通过这样可以让网页的速度更快、更安全、更健壮、更持久。: l5 M0 Q4 p) L+ g: {3 X
以上内容来自巴比特官网介绍
3 i6 Y) k1 j& b) i9 lIPFS是基于区块链技术的去中心化存储网络,实现了永久性存储。9 ~3 o$ i6 p. q, _
- IPFS的火热
- }" p& L* W, r+ Y- z& w; SFilecoin ICO是迄今为止涉及金额最大的ICO,仅一小时就募集了接近2亿美元,打破ICO记录,成为2017年最火爆的区块链项目。IPFS项目吸引了世界各地数字货币投资者和互联网从业者的密切关注,2018年比较值得期待的是IPFS主网的上线。然而就在前段时间,官方公布说上线时间延期到年底。
" ?! e" h, J: ^9 n, b注:Filecoin是IPFS激励层的加密数字货币(即代币),有点类似于以太坊平台上的以太币。
0 I" ?1 {/ ?: c/ p$ L# d nIPFS官网:https://ipfs.io/
+ L! X" q# e4 a1 |3 CFilecoin官网:https://filecoin.io/0 N; G& a% ?5 {* J& k9 h
- IPFS的应用案例2 x0 F/ M+ X( G8 ^3 d2 @
GitHub有两款开源项目,且有对应网址,分别是音乐播放器和视频播放器。
# M( K" }+ a& l1 i* @; z4 q* l# G% j. k4 l
- IPFS音乐播放器* T7 l. _: u, j+ f) X" ]
5 m" i$ k& ]1 G, v( m+ l1 n. F' I( r- e
" \/ r) [8 l- n, c8 n* q
- B, w: O! w5 W+ b+ G7 G" x t& [) q; i+ B5 L" G/ X# w
IPFS音乐播放器' b0 u0 E. }+ T9 L; N' t7 ?
5 O7 k( P6 i7 D
IPFS音乐播放器网址:https://diffuse.sh/, u) l) M4 e& q4 ], e
GitHub地址:https://github.com/icidasset/diffuse$ \3 r) u: S: p% j7 i
- IPFS视频在线播放器
$ f" D( P: x; @0 d' l% e
7 e3 t" o0 a" e" s6 p$ S1 G5 @1 I6 Z- u" H) W5 _7 ]
0 q3 `3 B+ W3 L |
) O% ]1 ?. \; _. F0 TIPFS视频在线播放器首页
7 V9 e$ E' a" C) G d' u, H9 O4 m+ v& p7 d
" x p: I# x8 K6 V1 ]5 k+ C$ ^( w
: o, r$ ]# x: @5 @' o0 H
) T t, C4 H: C8 s/ ]1 {
% }+ T8 \. m v) t# L$ c( fIPFS视频在线播放器播放电影+ g+ d) q: g% [4 S. I0 Y
: D2 m8 o$ @( c+ Y/ l( p' a2 X号称是国内第一个IPFS应用8 z+ j7 O% K+ d6 z3 C2 x5 a
IPFS视频在线播放器网址:http://www.ipfs.guide/8 E7 S# i" B# b" o9 q5 a
GitHub地址:https://github.com/download13/ipfstube
/ O! R) O" G0 L: i" d2 W2 j4 V可用于测试的电影视频Hash(这里只列举两部):
5 C7 P" J: U; {( N \神秘巨星:QmWBbKvLhVnkryKG6F5YdkcnoVahwD7Qi3CeJeZgM6Tq68
8 F; n `6 C/ R5 m盗梦空间:QmQATmpxXvSiQgt9c9idz9k3S3gQnh7wYj4DbdMQ9VGyLh
( D. t1 {4 _5 F% p! k, ? 2、IPFS的安装既然IPFS这么牛,有必要了解一下IPFS的开发,先从IPFS的安装开始。
+ p. A f2 I! \, y- IPFS Desktop. m% ~, q$ f( Q1 g" @
当然,可以直接安装IPFS节点桌面管理软件来体验一下。该软件可以方便地运行和管理自己的节点,查看IPFS节点资源信息,支持二次开发。该项目是Node.js编写的,已开源。1 {3 q3 l- F) Y* y* A6 G
, [$ H* a4 {9 b' H& Y/ T
* m# L; e) l4 g6 o. M; X
9 G1 J9 {$ u: v3 m9 }6 m, U, l
0 X1 @! O2 ~5 v4 tIPFS节点桌面管理软件) p) O6 G2 G* Z, m) J( b8 j$ C
4 e. l% a% J. S# H: x3 c
GitHub地址:5 a! s; x; ~% K0 D
https://github.com/ipfs-shipyard/ipfs-desktop# R) j1 X7 G2 Q& @3 `5 {) Z. {6 }
- Go-IPFS. e5 N; S% T$ Y2 w
进入IPFS的官网,找到并切换到"Install"页面,点击"Download IPFS for your platform",会跳转到如下网址(需要翻墙):' L" C6 j1 G4 L1 z% u( w5 ~
https://dist.ipfs.io/#go-ipfs
+ f5 K u, s) Y) l, f
$ c; P' h: e/ z$ s' o
' f4 Q x3 K6 ], T$ U9 T3 _( K
+ \0 w' _! J9 z下载Go-IPFS
2 w# _! h; F. X0 ^/ L8 J/ ^8 Z# z7 s8 ]: ~* Z
如果打不开,可以试着去Github查看安装方法
$ e$ _8 n& W/ l6 kGitHub地址:https://github.com/ipfs/go-ipfs
( e7 `0 _8 l& Q9 z5 C8 V" M [0 t& ~7 v将网页下载好的文件解压出来,下载的是"go-ipfs_v0.4.13_darwin-amd64.tar.gz",解压出来的是go-ipfs文件夹,) |5 H L) N; ~$ J6 { a
然后打开终端,进入到go-ipfs文件夹根目录,复制ipfs文件到系统的bin目录,然后可以通过"ipfs version"来检查,操作如下:
* w6 q+ H3 t; d* uwenzildeiMac:go-ipfs wenzil$ pwd/Users/wenzil/Desktop/study/go-ipfswenzildeiMac:go-ipfs wenzil$ lsLICENSE README.md build-log install.sh ipfswenzildeiMac:go-ipfs wenzil$ cp ipfs /usr/local/bin/ipfswenzildeiMac:go-ipfs wenzil$ ipfs versionipfs version 0.4.130 y- D. ]& ^* V; B8 D" _9 e
3、IPFS节点的创建、查看和启动以下是安装官网的Go-IPFS步骤下实验的
, [8 U X- M+ t- 创建节点
# |6 Y9 J$ C& j. W8 v1 g2 X* o/ z进入到当前用户的根目录,并使用"ipfs init"命令创建节点,可以用"open ./"打开创建节点生成的".ipfs"目录。
. k4 q. d' m# u# W( Z* wwenzildeiMac:go-ipfs wenzil$ cd ~/wenzildeiMac:~ wenzil$ pwd/Users/wenzilwenzildeiMac:~ wenzil$ ipfs initinitializing IPFS node at /Users/wenzil/.ipfsgenerating 2048-bit RSA keypair...donepeer identity: QmWH4xmGBznY9CJ4SjxpxWqGwMwrFtPdjgeweLCfJxW8j9to get started, enter: ipfs cat /ipfs/QmS4ustL54uo8FzR9455qaxZwuMiUhyvMcX9Ba8nUH4uVv/readmewenzildeiMac:~ wenzil$ cd .ipfswenzildeiMac:.ipfs wenzil$ open ./8 t% E7 }5 F: F
; X/ a8 S8 }+ h# k
+ l# q W7 {7 h: x" O
+ l; L3 B4 y4 L) \; f; lIPFS节点的创建
4 i9 z R# p4 g& W* L6 K" a4 Z- p* D% N8 Q6 X0 g' G
- 查看节点id6 W" } u9 u i b
wenzildeiMac:.ipfs wenzil$ ipfs id{ "ID": "QmWH4xmGBznY9CJ4SjxpxWqGwMwrFtPdjgeweLCfJxW8j9", "PublicKey": "CAASpgIwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQCt98O5zNptBtmSF6xYdOHAMunQsXE9rAmgqBXVPLJk+AoaBjNiAipB+sTXKwaj8stp9c3iuSzz10+dyYI38bq4TPDCsHnQ3PjKSgZwEaT0M6pJhGqJcSbLY57CjQOEwgFe+tw3b4hcoDvvNJMG0dZZg1r9xnfevBz0DNewlj2vtviSTQL6r/ZXlD04GsytxTSQ2pzMJTG3QmYyokeZ37DT7Vwa+IJjDCiDJC5BlZ204zunZBx3tqWUn2Hr2NtEX+4YDdX4SBHa0ZDNqZmzLRe5YlAXjVI0ONHs+BVsC1v1L5we52iSYCSgVHoxRP/pa9EZdHMvgs9QeAa5jO90yxgTAgMBAAE=", "Addresses": null, "AgentVersion": "go-ipfs/0.4.13/", "ProtocolVersion": "ipfs/0.1.0"} - 修改IPFS默认存储空间
+ j+ |9 h2 f/ U8 o1 _) P/ v存储节点默认存储大小为10G,如果你想修改存储空间,可以修改config配置文件,找到StorageMax并修改: c% A# t; T2 P0 j( H: Y5 f- u2 P
wenzildeiMac:~ wenzil$ export EDITOR=/usr/bin/vimwenzildeiMac:~ wenzil$ ipfs config edit ' e0 V0 j' Z0 k& g l
% f) R5 ? i/ @. v
, |8 Z0 v- ^8 s5 j修改IPFS默认存储空间: T& X. }2 z) m+ H5 P& w1 d: u: y
6 G+ }; ~9 {, {. `
- 启动和同步节点服务器
5 x& H0 g$ }( S e3 P, F2 d执行"ipfs daemon"命令,可以同步节点数据到IPFS网络。
" _3 k+ m% o" n1 q; ?wenzildeiMac:.ipfs wenzil$ ipfs daemonInitializing daemon...Adjusting current ulimit to 2048...Successfully raised file descriptor limit to 2048.Swarm listening on /ip4/127.0.0.1/tcp/4001Swarm listening on /ip4/192.168.1.100/tcp/4001Swarm listening on /ip6/::1/tcp/4001Swarm listening on /p2p-circuit/ipfs/QmWH4xmGBznY9CJ4SjxpxWqGwMwrFtPdjgeweLCfJxW8j9Swarm announcing /ip4/100.64.9.179/tcp/55898Swarm announcing /ip4/127.0.0.1/tcp/4001Swarm announcing /ip4/192.168.1.100/tcp/4001Swarm announcing /ip6/::1/tcp/4001API server listening on /ip4/127.0.0.1/tcp/5001Gateway (readonly) server listening on /ip4/127.0.0.1/tcp/8082Daemon is ready
+ R l! R6 V* F$ S" U 4、IPFS运行体验- IPFS查看ReadMe, N, q$ X; N& m% L p
可以回去查看执行"ipfs init"命令的时候,返回的内容。打开一个新的终端,操作如下:
% o! X, Y$ c: d$ o W( _" ZwenzildeiMac:~ wenzil$ ipfs cat /ipfs/QmYwAPJzv5CZsnA625s3Xf2nemtYgPpHdWEz79ojWnPbdG/readmeHello and Welcome to IPFS!██╗██████╗ ███████╗███████╗██║██╔══██╗██╔════╝██╔════╝██║██████╔╝█████╗ ███████╗██║██╔═══╝ ██╔══╝ ╚════██║██║██║ ██║ ███████║╚═╝╚═╝ ╚═╝ ╚══════╝If you're seeing this, you have successfully installedIPFS and are now interfacing with the ipfs merkledag! -------------------------------------------------------| Warning: || This is alpha software. Use at your own discretion! || Much is missing or lacking polish. There are bugs. || Not yet secure. Read the security notes for more. | -------------------------------------------------------Check out some of the other files in this directory: ./about ./help ./quick-start <-- usage examples ./readme <-- this file ./security-notes - Web管理界面6 F h& W( c* k
IPFS节点服务器启动后,可以用浏览器访问:http://localhost:5001/webui8 d/ P) q9 F8 m6 h
有本地配置、节点连接、本地节点文件等信息,如图:6 D2 S+ q8 c* K" A9 W- L; \# a
4 r6 ^. L1 j* C) X
" @" f3 _% B) K" |! s
8 t" G; C0 k) E1 p
! V: k. g+ m. b* Y. Q* l
IPFS的Web管理界面5 d, y3 a3 B5 l9 r0 Q+ W
# j7 R+ L8 @; K! q0 p+ ~) L
; K9 n2 W* A9 _& m5 Q
( w' V* n* z- y
5、设置跨域资源共享当我们在前端通过js接口操作ipfs时,会遇到跨域资源访问问题,可以在终端执行以下配置来解决:
0 _& g3 E2 ^) b6 [) I4 ?ipfs config --json API.HTTPHeaders.Access-Control-Allow-Methods '["PUT","GET", "POST", "OPTIONS"]'ipfs config --json API.HTTPHeaders.Access-Control-Allow-Origin '["*"]'ipfs config --json API.HTTPHeaders.Access-Control-Allow-Credentials '["true"]'ipfs config --json API.HTTPHeaders.Access-Control-Allow-Headers '["Authorization"]'ipfs config --json API.HTTPHeaders.Access-Control-Expose-Headers '["Location"]'6、IPFS的基本操作6.1 添加单个文件到IPFS节点- 新建文件
1 w6 o3 Y5 j. R" vwenzildeiMac:ifps_test wenzil$ vi test.txtwenzildeiMac:ifps_test wenzil$ cat test.txt IPFS测试文件 - 添加文件到IPFS节点
- A) u: [) C' cwenzildeiMac:ifps_test wenzil$ ipfs add test.txt added QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab test.txt添加文件到IPFS节点后,返回了文件的哈希值7 `) k6 F1 H$ |1 r) _8 G
- 查看IPFS节点的文件) R, d: \# v* V6 i0 V9 F
wenzildeiMac:ifps_test wenzil$ ipfs cat QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2AbIPFS测试文件注意:此时的文件只是添加到了本地的IPFS节点,读取的是本地数据,可以通过如下网址查看。9 s/ u$ n a3 N) J5 Y S, c
http://localhost:8080/ipfs/QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab% b0 u2 y3 v7 I7 _$ ]7 @2 [
通过"ipfs daemon"命令启动节点服务器,会将本地节点文件同步到外网,同步成功后,这时就可以访问如下网址来查看(如果访问失败,试着翻墙看看)。
. `, G+ o6 I7 h# A$ \' Vhttps://ipfs.io/ipfs/QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab
, j- P) m1 e7 a3 q6 I
8 z9 w9 {6 B* B6 D6 |) M. ^; S
1 y6 w, D0 q6 c$ Z3 ?* l: d' t- H% R; p# k- O
查看IPFS文件内容
# I, G8 e7 i9 X: L4 R' D Q* {& z
5 j+ f9 I4 i( z: ^* V5 E, a* w. X, c O如果访问成功的话,说明已经存储到IPFS网络中。目前IPFS网络暂未加入代币Filecoin机制,所以存储读取文件免费,速度也比较慢。
7 f+ o1 `3 I9 s+ x2 p) F' Z1 y - 下载IPFS节点的文件; Q+ }# X1 K/ y# K5 ~) E- I
wenzildeiMac:ifps_test wenzil$ ipfs get QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2AbSaving file(s) to QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab 25 B / 25 B [========================================================] 100.00% 0s查看当前目录,发现多了一个"QmSV..."的文件"
2 k6 }8 ^, q, G7 _+ \4 w& v' z0 H7 A1 O) c4 Y8 T' R7 z; X
* L* n# `! `3 g# _( z
3 _$ j) A* Q7 b& _
4 |: C8 c; w9 U% ], J
下载IPFS节点的文件% ?7 i1 a0 b: q
0 M) ^" h( o# y+ O$ P
) g2 S/ t! B" S. B! y) U/ a
6.2 创建文件夹存储文件- ipfs命令新建文件夹$ G& O W9 J9 A& ^) F9 v
wenzildeiMac:ipfs_test wenzil$ pwd/Users/wenzil/Desktop/study/ipfs_testwenzildeiMac:ipfs_test wenzil$ ipfs files mkdir /wenzilwenzildeiMac:ipfs_test wenzil$ ipfs files cp /ipfs/QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab /wenzil/test.txtwenzildeiMac:ipfs_test wenzil$ ipfs files lstestwenzilwenzildeiMac:ipfs_test wenzil$ ipfs files ls /testwenzilwenzildeiMac:ipfs_test wenzil$ ipfs files rm -rf /testwenzildeiMac:ipfs_test wenzil$ ipfs files lswenzilwenzildeiMac:ipfs_test wenzil$ ipfs files ls /wenziltest.txtwenzildeiMac:ipfs_test wenzil$ ipfs files read /wenzil/test.txtIPFS测试文件注:"ipfs files rm -rf /文件夹名"可以删除文件夹。
0 J7 p$ Q) U y
( O8 Y1 l$ X% h0 U8 g 6.3 IPFS上传文件夹- ipfs命令添加文件夹
# X! p2 R X$ v& P先在本地创建一个文件夹和对应文件,如图:+ n) d' V4 E5 A$ F0 Q' D: R
1 _: P$ [" z; ]7 [( W+ l5 I4 @
1 i- m1 U: a* L3 z9 g
) z) p! B& [# O
; C+ ?9 M3 |) H- a本地创建文件夹及文件
. i) I m3 h* l, I
! B) f! M) `0 [: p6 u然后通过"ipfs add -r 文件夹"命令添加整个文件夹, [& G* D/ z2 a$ q3 S6 U6 [
wenzildeiMac:ipfs_folder wenzil$ pwd /Users/wenzil/Desktop/study/ipfs_folder wenzildeiMac:ipfs_folder wenzil$ ls portrait.jpeg readme.txt wenzildeiMac:ipfs_folder wenzil$ cd .. wenzildeiMac:study wenzil$ pwd /Users/wenzil/Desktop/study wenzildeiMac:study wenzil$ ipfs add -r ipfs_folder/ added QmbHptfJfyuGAZxstFYgAVfz33cytR1seTD3ZabBSDd899 ipfs_folder/portrait.jpeg added QmPEWgwQEHD6qxm5cUivTEZ252T123DvQ5L6HyUDgViQv T ipfs_folder/readme.txt added QmdSd3xxKcwuMugyLCiLWzaeZKDkuM39R2tzyF3kBFDoaj ipfs_folder wenzildeiMac:study wenzil$ - 查看文件夹的文件内容3 {3 r% g9 J" R3 D7 T8 d
wenzildeiMac:study wenzil$ ipfs cat QmPEWgwQEHD6qxm5cUivTEZ252T123DvQ5L6HyUDgViQvT 我是区块链开发小白,请大家多多赐教,谢谢。欢迎阅读我的区块链系列文章,简书地址:https://www.jianshu.com/u/5f809ab3698cwenzildeiMac:study wenzil$ ipfs cat /ipfs/QmPEWgwQEHD6qxm5cUivTEZ252T123DvQ5L6HyUDgViQvT我是区块链开发小白,请大家多多赐教,谢谢。欢迎阅读我的区块链系列文章,简书地址:https://www.jianshu.com/u/5f809ab3698cwenzildeiMac:study wenzil$ ipfs cat /ipfs/QmdSd3xxKcwuMugyLCiLWzaeZKDkuM39R2tzyF3kBFDoaj/readme.txt我是区块链开发小白,请大家多多赐教,谢谢。欢迎阅读我的区块链系列文章,简书地址:https://www.jianshu.com/u/5f809ab3698c - 查看IPFS对应文件夹- b) c# v8 {/ e& E/ g) Y
访问文件夹对应的IPFS地址:
- _1 p; c1 T3 a+ [2 thttps://ipfs.io/ipfs/QmdSd3xxKcwuMugyLCiLWzaeZKDkuM39R2tzyF3kBFDoaj
% m3 h0 x; [! y- C2 J& } o第一次访问的时候,需要同步数据到IPFS网络,等待时间有点久,慢到让你以为同步失败了。# {1 f+ M% x9 l
}/ |/ E, ^+ w1 e" u$ S0 A
4 L5 G$ R6 o" R7 i9 Y
1 ?0 j! q9 H+ B) f( G4 ?: R查看IPFS对应文件夹" F% G! C' x9 d1 ?
, R" D) }( w# D. x _
访问文件夹对应的文件,如看到图片文件,访问ipfs网关:
4 i% N. B& s/ f5 k7 X) ~5 J: yhttps://ipfs.io/ipfs/QmdSd3xxKcwuMugyLCiLWzaeZKDkuM39R2tzyF3kBFDoaj/portrait.jpeg- a. g( t7 R; V0 z2 v
" r* _0 N2 M$ G
: p' P- a& O' m* f2 g
$ H" \8 t" p9 m; D查看IPFS文件夹对应文件: f5 R5 E/ g8 {) C$ T. J8 O5 e
" M5 { e8 M' u4 }& I/ g
或者通过文件的Hash来访问,如访问txt文件的地址:$ S/ p9 j1 c( y* {
https://ipfs.io/ipfs/QmPEWgwQEHD6qxm5cUivTEZ252T123DvQ5L6HyUDgViQvT) l3 m9 s! K. [3 X$ g
2 c) J% X& R" @ e
2 i' J7 R) ~, P" A3 }/ ?6 {9 M8 P9 x5 f
通过文件Hash访问查看
* Q, m" Q: E, l; b8 w
E3 {' u O; ?' j% A. U/ f& D% U9 }* j7 Q
6.4 添加简单网页到IPFS节点- 网页效果和项目结构
b) S8 V, W, Y5 D! {实现一个简单的实时显示系统当前时间的网页,效果如图:
" g' s- n, P: P4 J$ F# u8 i1 [# y( ^2 m% }" S1 W" W3 G
+ H, ? F) m! w5 |$ c# K
" Q a0 o0 i- L% x
6 ~, I3 K6 Z, U' v2 F* M: H
IPFS添加简单网页
?! F* i/ S, |' j; N
) U) h: e! e1 I2 S. D3 B工程结构如图:
6 |$ N4 z/ f$ W
; _+ l. r0 ~9 A+ _ s& g5 Z& g/ f- L! q+ |
" e" w5 Z& L, X& T# R( y" k, s
+ C6 U) \6 d, ~# ^. @) s0 P; O网页工程结构
( S9 A0 }: `9 P1 W1 M* F# d* z7 s6 I p- o2 L
! |- g# ^5 X" U: x
实现代码比较简单,这里就不贴代码了。8 R# D. |. i0 s+ A* k
- ipfs命令添加文件夹* H% l# c: W( O1 N( E+ `
wenzildeiMac:study wenzil$ ipfs add -r ipfs_htmladded QmTj49dgFgFhfNRTUYyUSzk3puj7vvDoVsAynFrL7GAj1d ipfs_html/index.htmladded QmeHFBSyN9KqeZWKyKTWKYzDRU192YWDRjf6sLWFWhwk2z ipfs_html/styles.cssadded QmWkEfgL2pjo7n8b9YBCv4qwE77YWePkSscxiHQfPpwpkp ipfs_html/time.jsadded QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD ipfs_html记得要开启IPFS节点服务器进行数据的同步,可以使用"ipfs daemon"命令在新的终端窗口打开运行。
1 i7 d: M. @$ Y- P$ M: g8 rwenzildeiMac:~ wenzil$ ipfs daemonInitializing daemon...Adjusting current ulimit to 2048...Successfully raised file descriptor limit to 2048.Swarm listening on /ip4/127.0.0.1/tcp/4001Swarm listening on /ip4/192.168.6.31/tcp/4001Swarm listening on /ip6/::1/tcp/4001Swarm listening on /p2p-circuit/ipfs/QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrzSwarm announcing /ip4/127.0.0.1/tcp/4001Swarm announcing /ip4/192.168.6.31/tcp/4001Swarm announcing /ip6/::1/tcp/4001API server listening on /ip4/127.0.0.1/tcp/5001Gateway (readonly) server listening on /ip4/127.0.0.1/tcp/8082Daemon is ready - 访问ipfs对应文件# g, ^) K. Y$ G" e9 f
浏览器访问ipfs网关:
) M. w3 D% |+ thttps://ipfs.io/ipfs/QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD/
3 p( k+ `+ S( a ^) }; T! U. z, ?: n8 r1 y: y0 U$ c1 {, @
3 g0 R) r$ l, _! s, }1 j
5 u1 V- [' X% [2 J" d查看IPFS对应网页8 H2 M% R3 c$ e7 n$ U* [8 L
- f, ]3 h/ J P
5 d; Z: q' ^4 N3 ^; Z
6.5 IPNS绑定节点名- IPNS绑定节点操作
a* r# E$ L: W: A每次修改文件内容后,文件的哈希值就会发生变化。对于网站而言,发布后可能还需要做修改,这时需要IPNS绑定节点名。每次更新网站内容后,重新publish一次更新发布到IPNS即可。" _4 ?+ H! \# l
刚才的html根目录的哈希值是QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD,操作如下:3 J+ ~: M4 G, Q
wenzildeiMac:study wenzil$ ipfs name publish QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDDPublished to QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz: /ipfs/QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD - 查看当前节点
+ z; C- R9 _. X2 ^" ^1 GwenzildeiMac:study wenzil$ ipfs id{ "ID": "QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz", "PublicKey": "CAASpg......", "Addresses": [ "/ip4/127.0.0.1/tcp/4001/ipfs/QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz", "/ip4/192.168.6.31/tcp/4001/ipfs/QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz", "/ip6/::1/tcp/4001/ipfs/QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz", "/ip4/14.xxx.xxx.xxx/tcp/4001/ipfs/QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz", "/ip4/61.xxx.xxx.xxx/tcp/20472/ipfs/QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz" ], "AgentVersion": "go-ipfs/0.4.13/", "ProtocolVersion": "ipfs/0.1.0"} 发现"ID"和上面"Published to"返回的节点一致,都是"QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz"
5 i' _9 j2 Q, ]$ ?3 B - 验证节点8 j- L# s7 k8 L/ ?
命令为:"ipfs name resolve 节点ID“
- Q: @2 y: s$ I! d- ~wenzildeiMac:study wenzil$ ipfs name resolve QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz/ipfs/QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD - 访问IPNS对应文件3 J3 u/ c/ z! j- V% t0 J1 V8 o
浏览器访问ipfs网关:
" L3 C3 _1 I4 O6 m5 U9 Z3 z7 M: Jhttps://ipfs.io/ipns/QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz/
) S1 W; X6 W4 j
+ w* e, d# k+ D
# v% M: G( \9 F) t! B( ~1 P/ r, ?0 \
/ D! K* d' D/ j2 {) S( g
9 W( |2 G; @7 I4 E8 D查看IPNS对应网页
% U+ P- J/ h7 p2 K! @% n! c! b6 f4 q* p- k! f
7、IPFS与以太坊DApp开发实战- IPFS与以太坊DApp结合的好处8 }2 n2 a7 t: J$ N! U9 z1 q0 _$ c
在以太坊平台上,往区块链写入数据需要花费以太币,调用智能合约执行每一行代码的时候,都需要一定量的gas交易费。区块链存储大数据成本很高,而且不合理。4 D% a0 |- _# R! R/ K/ f
可以先将文件存储到IPFS,然后得到文件的Hash存储到以太坊区块链中。读取文件的时候,从以太坊区块链中获取文件的Hash,再通过Hash来读取IPFS网络上的文件。3 v- c; G9 C3 g
使用官方提供的ipfs-api,可以很方便地在代码中操作节点将数据上传到IPFS,或者通过Hash从IPFS读取数据。
6 }) c9 F% Q a8 j% D3 p$ ]1 p$ ATruffle框架提供了快速搭建包含以太坊智能合约的React项目,可以通过"truffle unbox react"创建工程,然后安装ipfs-api依赖库。于是,可以在前端方便地调用IPFS的接口来读取、上传文件。* u; s6 o6 c0 m. F5 R
truffle unbox react官网:
. K: b& b0 F0 ?' y0 z& C' Yhttp://truffleframework.com/boxes/react
+ E+ Q1 ]6 {4 U - IPFS与DApp开发实战: C8 `1 ^) {6 y5 p3 U7 @
话不多说,开始进入实战开发。, P1 c! ^' h; c- {$ @
7.1 安装Truffle unbox react新建一个空目录"ipfs_dapp",然后执行安装命令 g$ ^" j1 a. b R
truffle unbox reactwenzildeiMac:ipfs_dapp wenzil$ pwd/Users/wenzil/Desktop/study/ipfs_dappwenzildeiMac:ipfs_dapp wenzil$ truffle unbox reactDownloading...Unpacking...Setting up...Unbox successful. Sweet! Commands: Compile: truffle compile Migrate: truffle migrate Test contracts: truffle test Test dapp: npm test Run dev server: npm run start Build for production: npm run build7.2 安装ipfs-apinpm install --save ipfs-apiwenzildeiMac:ipfs_dapp wenzil$ npm install --save ipfs-apinpm WARN deprecated bignumber.js@6.0.0: Custom ALPHABET bug fixed in v7.0.2react-box@0.1.0 /Users/wenzil/Desktop/study/ipfs_dapp├—┬ ipfs-api@22.0.0 │ ├—— async@2.6.1 │ ├—— big.js@5.1.1 │ ├—┬ bs58@4.0.1 │ │ └—— base-x@3.0.4 # ...... 省略N多行输出7.3 修改智能合约代码修改contracts目录中的"SimpleStorage.sol",修改后完整代码:) R% s6 d1 X! ]" f
pragma solidity ^0.4.18;contract SimpleStorage { // 用于存储图片的哈希值 string storedData; function set(string x) public { storedData = x; } function get() public view returns (string) { return storedData; }} 修改"Migrations.sol"* y. U' ^. n, R. Y
function Migrations() public { owner = msg.sender;}修改为:constructor() public { owner = msg.sender;}7.4 编译智能合约truffle compilewenzildeiMac:ipfs_dapp wenzil$ truffle compileCompiling ./contracts/Migrations.sol...Compiling ./contracts/SimpleStorage.sol...Writing artifacts to ./build/contracts7.5 部署智能合约进入remix-ide,复制SimpleStorage.sol代码,然后切换到"Run"菜单,选择"Injected Web3"进行智能合约的部署 F0 A' ]2 J& q) [2 D, V1 n
9 E# P+ l& _) B( f5 a
& r. g1 \8 o& ^4 f/ c
I% W+ t8 n, y c# HRemix-IDE部署智能合约
8 j/ Q' D# d4 T" q: U# d$ m3 w# J( M' | B2 A
; W8 Z/ B& \9 h2 m: c! [
9 }) a9 F+ D8 h$ t+ V! l& H F2 D j% p- V ~* n2 d- }
Remix-IDE部署智能合约-确认
) U% l' f4 H: Y' [( X' r& Y" |. n, b8 G. X6 Q1 S
然后复制智能合约地址,App.js文件中会用到合约地址为:"0xb177d6cf6916f128c9931e610c63208d9c5a40f3"
$ Y/ J6 a+ c; ? D7.5 修改App.js文件import React, {Component} from 'react'import SimpleStorageContract from '../build/contracts/SimpleStorage.json'import getWeb3 from './utils/getWeb3'import './css/oswald.css'import './css/open-sans.css'import './css/pure-min.css'import './App.css'const ipfsAPI = require('ipfs-api');const ipfs = ipfsAPI({ host: 'localhost', port: '5001', protocol: 'http'});const contract = require('truffle-contract')const simpleStorage = contract(SimpleStorageContract)let account;// Declaring this for later so we can chain functions on SimpleStorage.let contractInstance;let saveImageToIPFS = (reader) => { return new Promise(function(resolve, reject) { const buffer = Buffer.from(reader.result); ipfs.add(buffer).then((response) => { console.log(response) resolve(response[0].hash); }).catch((err) => { console.error(err) reject(err); }) })}class App extends Component { constructor(props) { super(props) this.state = { blockChainHash: null, web3: null, address: null, imageHash: null, isSuccess: false }}componentWillMount() { ipfs.swarm.peers(function(err, res) { if (err) { console.error(err); } else { // var numPeers = res.Peers === null ? 0 : res.Peers.length; // console.log("IPFS - connected to " + numPeers + " peers"); console.log(res); }});getWeb3.then(results => { this.setState({web3: results.web3}) // Instantiate contract once web3 provided. this.instantiateContract() }).catch(() => { console.log('Error finding web3.') })}instantiateContract = () => { simpleStorage.setProvider(this.state.web3.currentProvider); this.state.web3.eth.getAccounts((error, accounts) => { account = accounts[0]; simpleStorage.at('0xb177d6cf6916f128c9931e610c63208d9c5a40f3').then((contract) => { console.log(contract.address); contractInstance = contract; this.setState({address: contractInstance.address}); return; });})}render() { return ( <div className="App"> <div style={{marginTop:10}}>智能合约地址:</div> <div>{this.state.address}</div> <div style={{marginTop:10}}>上传图片到IPFS:</div> <div> <label id="file">选择图片</label> <input type="file" ref="file" id="file" name="file" multiple="multiple"/> </div> <button style={{marginTop:10}} onClick={() => { var file = this.refs.file.files[0]; var reader = new FileReader(); reader.readAsArrayBuffer(file) reader.onloadend = function(e) { console.log(reader); saveImageToIPFS(reader).then((hash) => { console.log(hash); this.setState({imageHash: hash}) }); }.bind(this); }}>开始上传</button> <div style={{marginTop:10}}>图片哈希值:{this.state.imageHash}</div> <button onClick={() => { contractInstance.set(this.state.imageHash, {from: account}).then(() => { console.log('图片的hash已经写入到区块链!'); this.setState({isSuccess: true}); }) }}>图片哈希写入区块链</button> { this.state.isSuccess ? <div style={{marginTop:10}}> <div>图片哈希成功写入区块链!</div> <button onClick={() => { contractInstance.get({from: account}).then((data) => { console.log(data); this.setState({blockChainHash: data}); }) }}>从区块链读取图片哈希</button> </div> : <div/> } { this.state.blockChainHash ? <div style={{marginTop:10}}> <div>从区块链读取到的哈希值:{this.state.blockChainHash}</div> </div> : <div/> } { this.state.blockChainHash ? <div style={{marginTop:10}}> <div>访问本地文件:</div> <div>{"http://localhost:8082/ipfs/" + this.state.imageHash}</div> <div>访问IPFS网关:</div> <div>{"https://ipfs.io/ipfs/" + this.state.imageHash}</div> <img alt="" style={{width: 100, height: 100 }} src={"https://ipfs.io/ipfs/" + this.state.imageHash}/> </div> : <img alt=""/> } </div>); }}export default App7.6 新建终端,执行"npm start"命令,结果如图:
4 ^! h: T1 {0 Q5 ]- J2 u# ^# a8 T
j0 O# V8 {7 j: v# C
) P! V7 |- M, Y" ^' K) m0 h) T
npm_start
- j; i8 Z2 x+ `
/ U" i: T- v0 M! s) o0 R$ T1 @/ k会自动打开网页,然后上传一张本地图片,之后点击"图片哈希写入区块链"按钮. h- i/ V; ?. c- J# L
8 |; P. M# o2 j! m4 A/ C9 M- o0 u" {
$ Q" r, D9 i. w1 b
- a( L$ y5 \7 q6 r4 W& X2 F运行结果12 d J1 o' D& D3 `9 G
" M2 g' G% D" f7 a1 F
最终效果如图:
' n" h) U4 `8 H# F: E8 Y1 x3 ~4 j- z/ v" l4 k L0 _" i: d
& g0 V6 E! [9 s5 h7 C$ {
# p. `. T' w8 p" u
. L' c" e6 x; P# Q8 c" K
运行结果- u/ n9 V& {( T3 J/ C7 @
6 p: X" \- y0 x% I3 f9 y访问IPFS网关:
- o$ u- ?. Q3 z0 g: P7 B! R6 i7 @https://ipfs.io/ipfs/QmbHptfJfyuGAZxstFYgAVfz33cytR1seTD3ZabBSDd899% v) ?( \( p/ W* s! n8 a# L; r
发现图片已经成功写入IPFS,如图:
6 b9 W% T* N# W+ }: @8 U& l5 ` t; V
' e" a3 `+ A Y5 [
: x8 S) n6 J6 d8 D7 ?# `1 w2 \. u8 c7 d
运行结果3
: u6 I3 @* p" E+ ^: C( K- i7 n5 p* B% n0 O* a. M. b
备注:本地IPFS网关端口默认为8080,我的电脑上端口冲突了,系统自动修改为8082。2 g+ B) \, m2 q J% K3 L- s
搞定,收工。9 E2 F6 v; F! M
PS:刚入坑的小白,很多不懂,还请各位大佬多赐教,谢谢!4 |8 c5 c5 |' R
" u, N1 e! K7 \! q# I
|
|