元宇宙织梦网(区块链大数据 芝麻卡NFR:吃饭赚钱 睡觉赚钱 做梦赚钱)

 找回密码
 立即注册

QQ登录

只需一步,快速开始

社区广播台

查看: 3163|回复: 0

IPFS开发简介与入门实战

[复制链接]
发表于 2021-4-22 23:39:31 | 显示全部楼层 |阅读模式
本帖最后由 赤膊医数 于 2021-4-22 23:41 编辑
% X( u& x5 L& V: x, \
& C" Z; A2 w- r  O! `0 m7 W2018年,最火爆且值得期待的区块链项目莫过于EOS和IPFS。1、IPFS简介
  • IPFS是什么
    * c% }) m  l& T' N" S星际文件系统IPFS(InterPlanetary File System)是一个面向全球的、点对点的分布式版本文件系统,目标是为了补充(甚至是取代)目前统治互联网的超文本传输协议(HTTP),将所有具有相同文件系统的计算设备连接在一起。原理用基于内容的地址替代基于域名的地址,也就是用户寻找的不是某个地址而是储存在某个地方的内容,不需要验证发送者的身份,而只需要验证内容的哈希,通过这样可以让网页的速度更快、更安全、更健壮、更持久。
      `0 c0 @6 d# }1 Q以上内容来自巴比特官网介绍
    ! \5 ^! Q6 S, T3 x9 @8 r, C+ a  a4 ZIPFS是基于区块链技术的去中心化存储网络,实现了永久性存储。
    . g. D- C# L/ Z* M+ }5 S
  • IPFS的火热- j* k( h2 L! @3 R4 E! f. v2 r- \
    Filecoin ICO是迄今为止涉及金额最大的ICO,仅一小时就募集了接近2亿美元,打破ICO记录,成为2017年最火爆的区块链项目。IPFS项目吸引了世界各地数字货币投资者和互联网从业者的密切关注,2018年比较值得期待的是IPFS主网的上线。然而就在前段时间,官方公布说上线时间延期到年底。
    9 i6 m7 @+ Z+ j" U" L9 F注:Filecoin是IPFS激励层的加密数字货币(即代币),有点类似于以太坊平台上的以太币。
    # g. }+ e( _) w, C- i: OIPFS官网:https://ipfs.io/
    $ O5 A# ~2 Q$ _) \5 N* DFilecoin官网:https://filecoin.io/$ W- V/ c' K& @+ F! e& o
  • IPFS的应用案例
    $ Q& R# l0 M) m2 ^+ wGitHub有两款开源项目,且有对应网址,分别是音乐播放器和视频播放器。
    & F4 d! V$ {3 P5 u9 I% g) D- J) Y2 Y0 t! T7 ^
    • IPFS音乐播放器
      - t6 P: l: b7 K+ k8 }
      7 U" e, s. C1 U0 \
      3 k( z6 D* m& N4 } 5521305-ae7a09ee6683f37a.jpg ( o/ B* {; |4 q
      # T9 U9 A6 M( g0 g9 W9 y9 {
      IPFS音乐播放器" l' E* Z: m0 P* v7 t. `7 l8 ^, _
      * w0 D5 g$ q1 s0 j6 ^5 U) Z
      IPFS音乐播放器网址:https://diffuse.sh/! h  s9 w" A" H( v% U
      GitHub地址:https://github.com/icidasset/diffuse
      $ L% K: y& r9 x) k
    • IPFS视频在线播放器
      4 D+ @" c8 F* Z5 n" Q9 h# O5 _: a' y, ^' v, |; e3 [9 R! L& g

      # o- v% a( d# O2 A/ ~# x 5521305-6bb3339c3bf60f73.png ) l6 l% ^7 P5 D5 t! \+ |, {

      $ n' U- S, {1 uIPFS视频在线播放器首页
      , @! `7 ?: S; ]* e! l1 Z- X' m9 F

      8 c- K  B3 S/ a3 i! m
      q( D8 a0 I" M, k  e( I
    5521305-f10de97d94961616.png
    ' J2 w* w- q$ F" C; J' m$ z; v4 Z# @) ^- t( b
    IPFS视频在线播放器播放电影
    + Y1 A! F& {0 T2 i6 ^$ g6 X
    " P, i4 v' a' v" H. H1 J$ Z2 N2 j号称是国内第一个IPFS应用
    ( W: D" y5 [9 oIPFS视频在线播放器网址:http://www.ipfs.guide/
    , N6 p* o% `: E9 F# i& Y6 z0 hGitHub地址:https://github.com/download13/ipfstube
    + M0 J2 [% Y7 A' _6 k可用于测试的电影视频Hash(这里只列举两部):
    % v) v7 R" M' S! U# ^0 Q1 Y$ D3 H神秘巨星:QmWBbKvLhVnkryKG6F5YdkcnoVahwD7Qi3CeJeZgM6Tq68
    ; {, ?- C  K  r% C盗梦空间:QmQATmpxXvSiQgt9c9idz9k3S3gQnh7wYj4DbdMQ9VGyLh. D9 J. |* X# }" Z
2、IPFS的安装既然IPFS这么牛,有必要了解一下IPFS的开发,先从IPFS的安装开始。
% _4 C0 d! n# |* b
  • IPFS Desktop
    0 B& D7 p5 _, v# b" w  u当然,可以直接安装IPFS节点桌面管理软件来体验一下。该软件可以方便地运行和管理自己的节点,查看IPFS节点资源信息,支持二次开发。该项目是Node.js编写的,已开源。- U6 q" ]& w3 Q! g5 g! a; @- d& k2 W

    & m% _) ?! b* j/ J0 y2 y/ V& c- [- @/ i
    5521305-6a23898f658aaa52.png 4 I0 G8 v" c% A5 @3 O

    . H" f4 D3 g, H  G! aIPFS节点桌面管理软件9 z! q( M. _% B9 k
    ' Z7 N( k5 A$ o% v: ]9 [
    GitHub地址:; m  B3 X$ Q% H8 i
    https://github.com/ipfs-shipyard/ipfs-desktop
    3 L$ S0 U3 W1 m7 R3 {
  • Go-IPFS
    7 Q$ A) n3 I4 R进入IPFS的官网,找到并切换到"Install"页面,点击"Download IPFS for your platform",会跳转到如下网址(需要翻墙):
    - @+ W" i2 n- a* dhttps://dist.ipfs.io/#go-ipfs( r. j: ^& z# j4 X

    5 w0 e  c# u3 j9 k/ w( V1 N 5521305-4fc9604f1c9a3f8a.png 0 z8 r6 J5 J& L8 n( [  ]

    ! d/ x! g: Y2 M6 y6 w& j下载Go-IPFS! r! u9 l6 r0 J0 t
    ! [1 \0 d& C! ?8 E5 k
    如果打不开,可以试着去Github查看安装方法7 I9 \4 X* k  b  w8 a
    GitHub地址:https://github.com/ipfs/go-ipfs
    ; q7 Z& r5 K) N4 m3 ~将网页下载好的文件解压出来,下载的是"go-ipfs_v0.4.13_darwin-amd64.tar.gz",解压出来的是go-ipfs文件夹,/ M1 Q  q. {- f( K- d
    然后打开终端,进入到go-ipfs文件夹根目录,复制ipfs文件到系统的bin目录,然后可以通过"ipfs version"来检查,操作如下:3 e% E0 J6 [0 `4 h: d7 U/ n
    wenzildeiMac: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.13- r0 x' u7 {; t0 Y+ a) ^) X" ]" N
3、IPFS节点的创建、查看和启动以下是安装官网的Go-IPFS步骤下实验的' B; K; u% Z% p- l0 H0 t' S( J9 [
  • 创建节点8 h% W( y6 x* e
    进入到当前用户的根目录,并使用"ipfs init"命令创建节点,可以用"open ./"打开创建节点生成的".ipfs"目录。% \9 Q! _: B# s8 m
    wenzildeiMac: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 ./, H7 u! l" n4 I# A9 ?

9 M$ }2 n* h9 h, @# k 5521305-5a5a65e04aadc891.png ' r# A& z4 [8 E5 e" R! L# S$ O

" f* ~8 k6 D% W  v7 ^IPFS节点的创建; s9 ?' r: v  x; `! p% o

" [* ]  r: m" z9 {
  • 查看节点id
    4 Z  Y# T0 T2 ]$ r7 CwenzildeiMac:.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默认存储空间
    3 s7 s# \; B' V/ F存储节点默认存储大小为10G,如果你想修改存储空间,可以修改config配置文件,找到StorageMax并修改) p3 I9 B- l0 z& v8 q8 w3 r7 _# a
    wenzildeiMac:~ wenzil$ export EDITOR=/usr/bin/vimwenzildeiMac:~ wenzil$ ipfs config edit  
    ' ~0 n/ F% P" H! q- O6 F 5521305-9a3e346dfd368715.png
    4 g3 d' `% S1 i% K/ c; Y, \# c& h% F. J" e: r
    修改IPFS默认存储空间
    ; ^2 T8 V; ]( v5 l$ o9 S/ c: R) v3 T: S1 E6 b) W% S+ Q( E0 Y
  • 启动和同步节点服务器
    , l) d. x& k$ Z0 `执行"ipfs daemon"命令,可以同步节点数据到IPFS网络。
    ; v1 d. \6 U( C; v; w5 XwenzildeiMac:.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 ready6 J) a$ Y9 e& E/ u3 Q' p
4、IPFS运行体验
  • IPFS查看ReadMe% a; ]) X/ }! x5 x, L
    可以回去查看执行"ipfs init"命令的时候,返回的内容。打开一个新的终端,操作如下:' G* x/ l! `; F* P( d4 q) {
    wenzildeiMac:~ 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管理界面
    ! f0 b* V0 E( @2 KIPFS节点服务器启动后,可以用浏览器访问:http://localhost:5001/webui
    9 @; e' }. [2 D+ V: w+ b有本地配置、节点连接、本地节点文件等信息,如图:  F1 w  x7 z  u

    . D9 x8 {4 J  n# R" t8 n8 b/ z& K3 @9 E5 i/ g! m( e+ [
    5521305-e7b2945efee17b07.png 2 R6 J# n/ ]( r; J0 [+ z: c1 b. X
    6 n. [% K3 ^) D2 M5 O" E; R
    IPFS的Web管理界面8 S1 \1 R7 D0 u0 O1 p5 ?
    ' Q* N& W! M2 M1 \

    6 @0 c6 b( f4 Q0 v
    % z, g9 X6 x7 d$ }
5、设置跨域资源共享当我们在前端通过js接口操作ipfs时,会遇到跨域资源访问问题,可以在终端执行以下配置来解决:
9 l6 @5 b8 |* Q7 K: m) p  jipfs 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节点
  • 新建文件
    4 z6 p% n, b& QwenzildeiMac:ifps_test wenzil$ vi test.txtwenzildeiMac:ifps_test wenzil$ cat test.txt IPFS测试文件
  • 添加文件到IPFS节点/ \% t( J: H1 q' x* x. X
    wenzildeiMac:ifps_test wenzil$ ipfs add test.txt added QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab test.txt添加文件到IPFS节点后,返回了文件的哈希值$ f( i! ?& b- G( v# `/ @! @- k
  • 查看IPFS节点的文件" C& Z( Q; Y3 X' U6 R
    wenzildeiMac:ifps_test wenzil$ ipfs cat QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2AbIPFS测试文件注意:此时的文件只是添加到了本地的IPFS节点,读取的是本地数据,可以通过如下网址查看。, ]: \$ s) y' ]& F
    http://localhost:8080/ipfs/QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab
    % N; V. D2 m* Q: o# k( d通过"ipfs daemon"命令启动节点服务器,会将本地节点文件同步到外网,同步成功后,这时就可以访问如下网址来查看(如果访问失败,试着翻墙看看)。
    & h6 T9 Y1 f( i# Qhttps://ipfs.io/ipfs/QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab$ f: F, D# H& b3 W' K
    6 r6 \& d, s% }1 B) I1 Q
    5521305-47b7c5838326410f.png / C7 ?; f7 P. V6 l0 I
      [6 W$ f# Y! E6 t
    查看IPFS文件内容
    $ |/ j1 O! D$ a0 z1 i3 \+ `7 Z& m" K8 f% o1 P' _/ C
    如果访问成功的话,说明已经存储到IPFS网络中。目前IPFS网络暂未加入代币Filecoin机制,所以存储读取文件免费,速度也比较慢。
    ' A, _; A4 _* |5 z1 C, c% x
  • 下载IPFS节点的文件
    ! {$ U3 [" r/ c8 q# x# G2 x, rwenzildeiMac:ifps_test wenzil$ ipfs get QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2AbSaving file(s) to QmSVKEwPBTzw5QLzGUE8oN8J1r4cadMeieSw4Co1ozm2Ab 25 B / 25 B [========================================================] 100.00% 0s查看当前目录,发现多了一个"QmSV..."的文件"; x% ]0 G: \- E- n; r# A/ V2 P, d( }

    1 V. e0 Y9 w8 f1 ], K: w
    0 l9 k8 h! \7 J1 {, U% _) ^* u 5521305-eb55bb51ae418085.png 3 x9 T5 g/ f  E/ b) `8 M3 n
    : M+ N* q. |) g, c
    下载IPFS节点的文件
    : G6 b6 X9 F( J; Z! ^/ W
    * A3 A4 Y  ?& P8 ~5 U" g) ~& X6 V. C
6.2 创建文件夹存储文件
  • ipfs命令新建文件夹
    0 |* d( W1 r: i+ q$ DwenzildeiMac: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 /文件夹名"可以删除文件夹。! m9 D) `0 ~  S! d1 F$ h

    & A" w7 P1 D. ?
6.3 IPFS上传文件夹
  • ipfs命令添加文件夹4 {& O: O" {& ?  r, s0 r. C
    先在本地创建一个文件夹和对应文件,如图:& q# O4 \2 M2 ]$ t1 l7 z/ ?
    ) x7 [. Q1 J2 F- }
    ! \* e; z+ i' N. g; ~& j: |& [
    5521305-3f47f331d7d63fb6.png 0 Y6 o! \! L: r2 w+ j

    ' W& w# R3 t0 V5 W/ C本地创建文件夹及文件1 a" O. j" |* {  E$ A7 x
      ?1 D4 \$ f  X  l7 Q$ q/ G
    然后通过"ipfs add -r 文件夹"命令添加整个文件夹
    / f- R0 f5 n: e* ?. ? 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$
  • 查看文件夹的文件内容
      V8 G: \! s! twenzildeiMac: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对应文件夹3 O' N# {  u! [* u8 w( E( p1 ]
    访问文件夹对应的IPFS地址:0 S5 m! g" {  Y& u9 t
    https://ipfs.io/ipfs/QmdSd3xxKcwuMugyLCiLWzaeZKDkuM39R2tzyF3kBFDoaj
      ]( u- y- r0 H+ t5 A; Y第一次访问的时候,需要同步数据到IPFS网络,等待时间有点久,慢到让你以为同步失败了。6 B+ ?6 z; i1 Z- H' ^' U1 u9 I

    ) V4 D& {  ^, N7 B- l 5521305-72d506bca193abec.png $ @7 H- f9 W( v& [1 [9 U1 k- k

    ; d1 R4 H$ R, Q" Q! k2 W查看IPFS对应文件夹
    0 u& r/ ~: G3 M* U/ o8 o4 J( Y( o- T- @* H% J, a' q9 O4 ^' c
    访问文件夹对应的文件,如看到图片文件,访问ipfs网关:6 _1 U% }1 v* B3 e2 u" M
    https://ipfs.io/ipfs/QmdSd3xxKcwuMugyLCiLWzaeZKDkuM39R2tzyF3kBFDoaj/portrait.jpeg1 C; P- g- W) L% s6 g

    " K* u/ D1 g, S0 {4 l  S4 n 5521305-b2de93582c9d0dac.png
    3 B2 s& [6 }; n& A. t
    ' [0 {. m" M: z5 H查看IPFS文件夹对应文件
    - U5 S! g* t6 v) o" u' H
    $ D7 L" k* V" y7 y# z+ a4 v* z4 J1 a或者通过文件的Hash来访问,如访问txt文件的地址:6 U+ P+ p8 _. M9 v$ b$ `/ n
    https://ipfs.io/ipfs/QmPEWgwQEHD6qxm5cUivTEZ252T123DvQ5L6HyUDgViQvT
    ! e# l9 ^/ f2 ]! l. n* m9 z* m5 Q8 d/ c) o" ?/ p* W; N
    5521305-9f5f8194546ee968.png ; |, s% ?7 n' B3 |2 h

    6 T7 \9 M0 V9 Z9 k! b3 _! i通过文件Hash访问查看0 B' p$ T$ X/ {$ @& J$ f/ {( [

    , U% x' m$ j7 d3 J& l" z$ l
    - L; v7 F1 |' r: A& F3 U
6.4 添加简单网页到IPFS节点
  • 网页效果和项目结构
    ) \9 Q4 }4 i2 x实现一个简单的实时显示系统当前时间的网页,效果如图:
    ! [2 `2 h6 y* \! J. y
    ' b/ `/ Q- H# P" T$ E8 U" L$ z5 E' M( Y  {
    5521305-13caff069d686e6e.png # a8 h/ \1 a/ V& l  ]$ ~4 D
    1 b! d* T( \, s0 ?, {
    IPFS添加简单网页6 o: O) m& P* a! y/ E% ~' L

    5 f, R/ c" {/ i) h工程结构如图:# ~& ^& ]7 ?: j0 C5 v. o+ u1 B( Q# z3 G6 }

    3 L1 V1 q- B7 r9 k7 n' y- l" t" _2 b8 O) U  Y6 X
    5521305-1f43ed03e72738db.png
    , C: s' I- e' a2 H0 i1 o' V) }- d; i
    9 U& R" p! A# O) P8 _网页工程结构
    8 J) I+ Z" B! z* q5 e8 J5 U& P; @9 g
    . u0 h: \  |; u9 g; Q
实现代码比较简单,这里就不贴代码了。
9 ^. B" N( p' i- Y) E) V  b' P  _
  • ipfs命令添加文件夹
    , u( O: W' i, Q( NwenzildeiMac: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"命令在新的终端窗口打开运行。( f' }# c" w4 G+ b! B$ q
    wenzildeiMac:~ 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对应文件0 @8 t# W& f9 p& N4 x1 m
    浏览器访问ipfs网关:' j' [! H: ]% t5 N0 D( V. {
    https://ipfs.io/ipfs/QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD/
    8 N% R* H4 Q3 ]! x
    ! u) q4 C$ n% o! @# o 5521305-f507c2fadf344358.png
    1 O5 D5 s4 }0 M  M
      X% W4 q$ E- Y查看IPFS对应网页* u8 m" y4 \8 T0 ~* ~7 c) d& ~4 V

    $ f- z0 t3 n( `7 _
    / D. \4 @3 t! U+ s' R
6.5 IPNS绑定节点名
  • IPNS绑定节点操作' J6 D& f1 a" z7 }
    每次修改文件内容后,文件的哈希值就会发生变化。对于网站而言,发布后可能还需要做修改,这时需要IPNS绑定节点名。每次更新网站内容后,重新publish一次更新发布到IPNS即可。
    , p4 ~5 G3 i& g( J$ b2 ~& e7 {刚才的html根目录的哈希值是QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD,操作如下:8 d5 @% I: a' u& ]) H$ c# n8 X
    wenzildeiMac:study wenzil$ ipfs name publish QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDDPublished to QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz: /ipfs/QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD
  • 查看当前节点
    2 z# }4 U. A6 f- 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"
    ; U" ]* P5 _$ F$ u7 B* {
  • 验证节点4 {6 [2 {0 s, {  P& {
    命令为:"ipfs name resolve 节点ID“
    ' C! q  ]* L9 e, M$ N9 [wenzildeiMac:study wenzil$ ipfs name resolve QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz/ipfs/QmeG6LKmEfF5s62cySE9ZnU8EifmU46CQTixipfzsZspDD
  • 访问IPNS对应文件
    + i1 w. D+ \5 ~! E- y; \浏览器访问ipfs网关:+ c4 `/ K  P! \, \
    https://ipfs.io/ipns/QmcXWJqPjCgr1wrReWu2zUT41E5FrrXp2uyyNZRr7jXXrz/& {* Z# c. o: m$ Q! n/ {

    " c2 U: k4 [! m$ d2 P

/ c) G% t# f+ s7 Y 5521305-34805df9bb0347e0.png . P2 o+ f& u5 A$ K3 W4 O' O
) B2 g* O0 Q  {: o) ~0 {; p
查看IPNS对应网页. {' d2 ?9 `: }% Z0 ]
0 ~) W8 z% e3 U' z8 [& u
7、IPFS与以太坊DApp开发实战
  • IPFS与以太坊DApp结合的好处
    ; I, f& @' n( K2 z& j/ m4 Y* k在以太坊平台上,往区块链写入数据需要花费以太币,调用智能合约执行每一行代码的时候,都需要一定量的gas交易费。区块链存储大数据成本很高,而且不合理。
    ) B+ h! O2 n3 X0 p1 n# N可以先将文件存储到IPFS,然后得到文件的Hash存储到以太坊区块链中。读取文件的时候,从以太坊区块链中获取文件的Hash,再通过Hash来读取IPFS网络上的文件。
    - h* K6 k# Q% M& w( f使用官方提供的ipfs-api,可以很方便地在代码中操作节点将数据上传到IPFS,或者通过Hash从IPFS读取数据。+ W$ G' T, X5 {
    Truffle框架提供了快速搭建包含以太坊智能合约的React项目,可以通过"truffle unbox react"创建工程,然后安装ipfs-api依赖库。于是,可以在前端方便地调用IPFS的接口来读取、上传文件。
    + p" R# r; b7 x9 x4 b6 i' otruffle unbox react官网:
    $ L7 L6 L0 x# u& Bhttp://truffleframework.com/boxes/react
    ( S" j$ `% f/ M3 x) G' x" ~( w
  • IPFS与DApp开发实战
    % ~3 y0 U$ H% b' {- ]3 I& M4 c话不多说,开始进入实战开发; h, u/ W$ s% ~
    7.1 安装Truffle unbox react新建一个空目录"ipfs_dapp",然后执行安装命令
    / y5 ]9 {" e6 V  t; g  f" ~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",修改后完整代码:9 \" n* Q: C4 H4 j' s7 o% z) \) `
    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"
    8 E" d9 j# V: O  dfunction 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"进行智能合约的部署
    , c# k& f$ C6 D8 Z
    , N( O7 V6 `  S$ K 5521305-e9cbe2a2b1d0327b.png ( D# R: F6 M; F

    2 I, \8 x- Q9 L! {5 eRemix-IDE部署智能合约
    # ^& ?0 ]% I+ }4 V4 C5 I6 p
    ' @8 E* `3 G# C$ u
    ' l9 H& R4 U$ Y2 C9 J  \8 O 5521305-1c0743601b91144b.png & B% e1 x8 c( s
    ! H# N6 q- M( m: K& L9 `* z$ y
    Remix-IDE部署智能合约-确认
    1 j, p5 l$ c3 M2 H- e# [
    . E; m; c0 D" |/ j  K+ Z) g然后复制智能合约地址,App.js文件中会用到合约地址为:"0xb177d6cf6916f128c9931e610c63208d9c5a40f3"
    / L$ h! p# ]3 Z7 z' @5 a7.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"命令,结果如图:  v; |* L0 |; q8 l- _2 F# \! o' v
    5521305-4dba93598490e042.png ) J) N- Q3 n0 l
    " w6 M6 M" C* R& v) G
    npm_start1 ~. D- F8 V& I$ v6 h5 L
    4 F  _/ ?9 n. Q) C4 @$ R' I
    会自动打开网页,然后上传一张本地图片,之后点击"图片哈希写入区块链"按钮& v! [. z, z3 J+ y" t

    2 k7 q! f; ?! Z- z! M; f 5521305-c5e820e466316868.png + z8 \- t: W, N) h1 F9 B( {

    " m1 Z, D* T: a% i运行结果11 _7 b' b6 k; L, Z
    ! J) X8 h+ V! R- s
    最终效果如图:
    - e* t- H% c1 K
    & _. Z: [5 T- i
    . F% b5 T5 c6 X; D! j9 n 5521305-a13f100335acb2e5.png . U7 S) D2 c# a8 G  T2 x) X

    * v+ A# ]: n& C: h  q/ I运行结果
    4 W' W# _7 f2 o/ S  e* g( |0 H3 A  t
    访问IPFS网关:
    ) j& m, E) a  ~# n) C% ?# zhttps://ipfs.io/ipfs/QmbHptfJfyuGAZxstFYgAVfz33cytR1seTD3ZabBSDd899
    % p  d/ x8 X) p1 b- W0 w发现图片已经成功写入IPFS,如图:
    8 q# A% k0 F5 M% o) X* N1 u2 q3 S9 W5 H
    ) H% m7 W+ J2 x& L7 q, ?6 q
    5521305-583743843cc663c0.png $ z( _3 W7 M, d9 y+ R9 J/ z
    : x8 M+ w; ^) r7 U" J
    运行结果3" n) I, Q+ Y) q( Q7 |

    4 I+ o1 M/ G( i5 |2 ]3 Q备注:本地IPFS网关端口默认为8080,我的电脑上端口冲突了,系统自动修改为8082。8 A  g0 E! ~7 u" a0 U0 _
    搞定,收工。* M' E0 q: t8 f2 I+ M5 K. e! _- _
    PS:刚入坑的小白,很多不懂,还请各位大佬多赐教,谢谢!
    ( P$ L. A* G3 B9 x; s0 n/ ^. X$ `7 O/ J, ]. k" Q* c3 N
5521305-6bb3339c3bf60f73.png
5521305-f10de97d94961616.png
5521305-6a23898f658aaa52.png
5521305-4fc9604f1c9a3f8a.png
5521305-5a5a65e04aadc891.png
5521305-9a3e346dfd368715.png
5521305-e7b2945efee17b07.png
5521305-47b7c5838326410f.png
5521305-eb55bb51ae418085.png
5521305-3f47f331d7d63fb6.png
5521305-72d506bca193abec.png
5521305-b2de93582c9d0dac.png
5521305-9f5f8194546ee968.png
5521305-13caff069d686e6e.png
5521305-1f43ed03e72738db.png
5521305-f507c2fadf344358.png
5521305-34805df9bb0347e0.png
5521305-e9cbe2a2b1d0327b.png
5521305-1c0743601b91144b.png
5521305-4dba93598490e042.png
5521305-c5e820e466316868.png
5521305-a13f100335acb2e5.png
5521305-583743843cc663c0.png
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

快速回复 返回顶部 返回列表