一个基于web服务器的PoW案例

简介: 一个基于web服务器的PoW案例一、安装第三方库go get github.com/davecgh/go-spew/spew这个库的功能是在命令行格式化输出内容。go get github.com/gorilla/mux这个开发包是用来编写Web处理程序的。go get github.com/joho/godotenv这个工具包是读取.env后缀名的文件中的数据,如果是Linux环境,.env文件放置在项目的根目录即可,如果是Windows和Mac OS,.env文件需要放在GOPATH/src目录下。

一个基于web服务器的PoW案例

一、安装第三方库

go get github.com/davecgh/go-spew/spew

这个库的功能是在命令行格式化输出内容。

go get github.com/gorilla/mux

这个开发包是用来编写Web处理程序的。

go get github.com/joho/godotenv

这个工具包是读取.env后缀名的文件中的数据,如果是Linux环境,.env文件放置在项目的根目录即可,如果是Windows和Mac OS,.env文件需要放在GOPATH/src目录下。

二、定义区块信息、难度系数

constdifficulty=4

 

typeBlockstruct {

    Indexint

    Timestampstring

    BMPint

    HashCodestring

    PreHashstring

    Diffint

    Nonceint

}

 

varBlockchain []Block

typeMessagestruct {

    BlockPostMessageint

}

varmutex=&sync.Mutex{}

这里我们定义一下挖矿生成区块的难度值,然后定义区块,包含区块高度、时间戳、交易信息、当前的Hash值和上一个的Hash值,还有难度和随机值。

然后定义区块链,用区块数组。

然后我们这里要根据Get请求和Post请求来生成区块,所以定义一个消息结构体用于存储Post信息。

最后定义一个互斥锁。

三、生成区块

funcgenerateBlock(oldBlockBlock, BlockPostMessageint) Block {

    varnewBlockBlock

    newBlock.PreHash=oldBlock.HashCode

   

    newBlock.Index=oldBlock.Index+1

   

    t :=time.Now()

    newBlock.Timestamp=t.String()

   

    newBlock.BlockPostMessage=BlockPostMessage

   

    newBlock.Diff=difficulty

 

    fori :=0; ; i++ {

        newBlock.Nonce++

        hash :=calculateHash(newBlock)

        fmt.Println(hash)

       

        ifisHashValid(hash, newBlock.Diff) {

            fmt.Println("挖矿成功")

            newBlock.HashCode=hash

            returnnewBlock

        }

    }

}

每次生成新的区块前,先获取先前区块的Hash值放置在这个区块的上一个区块Hash值,然后获取当前时间,通过String()方法转换成为时间戳后放入区块的Timestamp。然后将Post传递的消息放入区块,将我们固定不变的困难值放入区块。

然后循环挖矿,每次挖矿将随机数加一,然后先不管这个区块能不能成功并入区块链,得先计算它的哈希值才能知道,然后校验哈希值的前导0,如果成功就输出挖矿成功。

四、生成哈希值

funccalculateHash(blockBlock) string {

    hashed :=strconv.Itoa(block.Index) +block.Timestamp+

        strconv.Itoa(block.Nonce) +strconv.Itoa(block.BlockPostMessage) +

        block.PreHash

    sha :=sha256.New()

    sha.Write([]byte(hashed))

    hash :=sha.Sum(nil)

    returnhex.EncodeToString(hash)

}

很简单的逻辑,将区块的数据拼接后用sha256进行加密,得到hash值。

五、区块校验

funcisHashValid(hashstring, difficultyint) bool {

    prefix :=strings.Repeat("0", difficulty)

    returnstrings.HasPrefix(hash, prefix)

}

这个我们本专栏之前的文章介绍了,在此简单说一下,这里我们就校验一下哈希值前面的零的数量是不是和难度值一致。

六、启动HTTP服务器

funcrun() error {

    mux :=makeMuxRouter()

    httpAddr :=os.Getenv("PORT")

    log.Println("Listening on ", httpAddr)

   

    s :=&http.Server{

        Addr: ":"+httpAddr,

        Handler: mux,

        ReadTimeout:  10*time.Second,

        WriteTimeout: 10*time.Second,

        MaxHeaderBytes: 1<<20,

    }

    iferr :=s.ListenAndServe(); err!=nil {

        returnerr

    }

    returnnil

}

我们先从.env文件中获取PORT的值。然后监听获取的端口号。http.Server是设置http服务器的参数,其中Addr是地址,ReadTimeout、WriteTimeout分别是读写超时时间,然后是设置请求头的数据大小的最大值,1 << 20是位运算,算出来就是1MB。!!!最重要的就是回调函数了,这里需要我们自己编写来处理Get和Post请求。

然后我们就来监听事件并且根据监听到的事件来服务。

七、回调函数的编写

funcmakeMuxRouter() http.Handler {

    muxRouter :=mux.NewRouter()

    muxRouter.HandleFunc("/",

        handGetBlockchain).Methods("GET")

    muxRouter.HandleFunc("/",

        handWriteBlock).Methods("POST")

    returnmuxRouter

}

mux.NewRouter()是用来创建路由,muxRouter.HandleFunc("/",handGetBlockchain).Methods("GET")是根据你访问的目录和请求类型来调用指定的方法。这里是使用Get方法访问根目录就调用handGetBlockchain方法。同样的,muxRouter.HandleFunc("/",handWriteBlock).Methods("POST")就是使用Post请求访问根目录时就调用handWriteBlock方法。

八、处理Get请求

funchandGetBlockchain(whttp.ResponseWriter, r*http.Request) {

    bytes, err :=json.MarshalIndent(Blockchain, "", "\t")

    iferr!=nil {

        http.Error(w, err.Error(), http.StatusInternalServerError)

        return

    }

    io.WriteString(w, string(bytes))

}

我们需要将数据转换为json格式,便于与前端进行交互。

同样我们的参数分别是响应和请求。然后处理错误,当出现500错误时,也就是http.StatusInternalServerError,我们将err.Error()写入w:

image.png

如果没出错,就将json数据写入w。

九、处理POST请求

funchandWriteBlock(writerhttp.ResponseWriter, request*http.Request) {

    writer.Header().Set("Content-Type", "application/json")

    varmessageMessage

    decoder :=json.NewDecoder(request.Body)

    iferr :=decoder.Decode(&message); err!=nil {

        responseWithJSON(writer, request, http.StatusNotFound, request.Body)

    }

 

    deferrequest.Body.Close()

 

    mutex.Lock()

    newBlock :=generateBlock(Blockchain[len(Blockchain)-1], message.BPM)

    mutex.Unlock()

 

    ifisBlockValid(newBlock, Blockchain[len(Blockchain)-1]) {

        Blockchain=append(Blockchain, newBlock)

        spew.Dump(Blockchain)

    }

    //返回响应信息

    responseWithJSON(writer, request, http.StatusCreated, newBlock)

}

 

因为需要服务器响应结果为json,先设置响应头的"Content-Type"为"application/json"。然后从request中读取JSON数据,将JSON数据转成Message。如果转换失败,就交给下一步处理异常,如果成功就创建新的区块。

这里使用defer,说明我们要记得关闭请求哦~

然后添加区块时要记得上锁,可以防止同个时间点多个POST请求生成区块。

接下来就要校验生成的区块是否正确,如果正确就加入区块链中。

十、处理异常

func responseWithJSON(writer http.ResponseWriter, request *http.Request,

code int, inter interface{}) {

   

writer.Header().Set("Content-Type", "application/json")

response, err := json.MarshalIndent(inter, "", "\t")

if err != nil {

writer.WriteHeader(http.StatusInternalServerError)

writer.Write([]byte("HTTP 500:Server Error"))

return

}

writer.WriteHeader(code)

writer.Write(response)

}

如果将传入的inter转换为json格式的数据没有出现错误就往响应头写入响应码,并将数据写入。

十一、校验区块是否正确

func isBlockValid(newBlock, oldBlock Block) bool {

if oldBlock.Index+1 != newBlock.Index {

return false

}

if oldBlock.HashCode != newBlock.PreHash {

return false

}

if calculateHash(newBlock) != newBlock.HashCode {

return false

}

return true

}

这里校验了新区块的index是否等于原来最后一个区块的index加一,新区块的PreHash应该等于之前区块链最后一个区块的HashCode。然后还需要再一次计算区块的哈希值,进行比对。

十二、主逻辑

然后我们现在用Go实现通过http请求来完成区块链。

func main() {

err := godotenv.Load()

if err != nil {

log.Fatal(err)

}


go func() {

t := time.Now()

genessisBlock := Block{}

genessisBlock = Block{0, t.String(),

  0, calculateHash(genessisBlock),

  "", difficulty, 0}

mutex.Lock()

Blockchain = append(Blockchain, genessisBlock)

mutex.Unlock()

spew.Dump(genessisBlock)

}()


//创建http服务器的启动函数

log.Fatal(run())

}

godotenv.Load()加载一个文件,如果不填写参数,就默认是加载.env文件。

这个.env文件我们这里就只需要填写一个端口号。

image.png

这里我们先将创世区块加入区块链。然后用spew.Dump()将其格式化输出到命令行。

最后我们会要用run来启动http服务器。

十三、运行结果

我们可以使用curl来进行get和post请求。

image.png

这是get请求,得到区块链。

image.png

这是进行post请求,新建一个区块加到了区块链。

image.png

可以看到再次get请求,已经有新的区块在区块链中了。


目录
相关文章
|
7天前
|
前端开发 机器人 测试技术
【RF案例】Web自动化测试弹窗处理
在进行Web自动化测试时,常会遇到不同类型的弹窗,如ajax、iframe、新窗口及alert/Confirm等。这些弹窗可通过Selenium进行定位与处理。其中,ajax弹窗直接定位处理;iframe需先选中再操作;新窗口类似iframe处理;而alert/Confirm则需特殊方法应对。在Robot Framework中,需先定义并获取窗口后使用特定关键字处理。此外,还有部分div弹窗需在消失前快速定位。希望本文能帮助大家更好地处理各类弹窗。
18 6
【RF案例】Web自动化测试弹窗处理
|
8天前
|
前端开发 数据安全/隐私保护
【前端web入门第二天】03 表单-下拉菜单 文本域 label标签 按钮 【附注册信息综合案例】
本文档详细介绍了HTML表单的多种元素及其用法,包括下拉菜单(`&lt;select&gt;` 和 `&lt;option&gt;`)、文本域(`&lt;textarea&gt;`)、标签解释(`&lt;label&gt;`)、各类按钮(`&lt;button&gt;`)及表单重置功能、无语义布局标签(`&lt;div&gt;` 和 `&lt;span&gt;`)以及字符实体的应用。此外,还提供了一个完整的注册信息表单案例,涵盖个人信息、教育经历和工作经历等部分,展示了如何综合运用上述元素构建实用的表单。
【前端web入门第二天】03 表单-下拉菜单 文本域 label标签 按钮 【附注册信息综合案例】
|
10天前
|
SQL 数据挖掘 数据库
服务器数据恢复—意外断电导致XenServer虚拟机不可用的数据恢复案例
服务器数据恢复环境: 一台服务器中有一组由4块STAT硬盘通过RAID卡组建的RAID10阵列,上层是XenServer虚拟化平台,虚拟机安装Windows Server操作系统,作为Web服务器使用。 服务器故障: 因机房异常断电导致服务器中一台VPS(Xen Server虚拟机)不可用,虚拟磁盘文件丢失。
服务器数据恢复—意外断电导致XenServer虚拟机不可用的数据恢复案例
|
17天前
|
网络协议
keepalived对后端服务器的监测方式实战案例
关于使用keepalived进行后端服务器TCP监测的实战案例,包括配置文件的编辑和keepalived服务的重启,以确保配置生效。
29 1
keepalived对后端服务器的监测方式实战案例
|
7天前
|
前端开发
【前端web入门第五天】03 清除默认样式与外边距问题【附综合案例产品卡片与新闻列表】
本文档详细介绍了CSS中清除默认样式的方法,包括清除内外边距、列表项目符号等;探讨了外边距的合并与塌陷问题及其解决策略;讲解了行内元素垂直边距的处理技巧;并介绍了圆角与盒子阴影效果的实现方法。最后通过产品卡片和新闻列表两个综合案例,展示了所学知识的实际应用。
22 11
|
7天前
|
前端开发
前端web入门第四天】03 显示模式+综合案例热词与banner效果
本文档介绍了HTML中标签的三种显示模式:块级元素、行内元素与行内块元素,并详细解释了各自的特性和应用场景。块级元素独占一行,宽度默认为父级100%,可设置宽高;行内元素在同一行显示,尺寸由内容决定,设置宽高无效;行内块元素在同一行显示,尺寸由内容决定,可设置宽高。此外,还提供了两个综合案例,包括热词展示和banner效果实现,帮助读者更好地理解和应用这些显示模式。
|
10天前
|
JavaScript 前端开发
【前端web入门第一天】03 综合案例 个人简介与vue简介
该网页采用“从上到下,先整体再局部”的制作思路,逐步分析并编写代码实现个人简介页面。内容涵盖尤雨溪的背景、学习经历及主要成就,同时介绍其开发的Vue.js框架特点。代码结构清晰,注重细节处理,如使用快捷键提高效率,预留超链接位置等,确保最终效果符合预期。
|
8天前
|
存储 数据挖掘 Linux
服务器数据恢复—Linux操作系统网站服务器数据恢复案例
服务器数据恢复环境: 一台linux操作系统服务器上跑了几十个网站,服务器上只有一块SATA硬盘。 服务器故障: 服务器突然宕机,尝试再次启动失败。将硬盘拆下检测,发现存在坏扇区
|
18天前
|
存储 运维 小程序
服务器数据恢复—双循环RAID5阵列数据恢复案例
服务器存储数据恢复环境: 一台存储中有一组由7块硬盘组建的RAID5阵列,存储中还有另外3块盘是raid中掉线的硬盘(硬盘掉线了,管理员只是添加一块的新的硬盘做rebuild,并没有将掉线的硬盘拔掉)。整个RAID5阵列的存储空间划分了一个LUN。 服务器存储故障: 硬盘出现故障导致存储中阵列瘫痪。 和管理员沟通,据管理员说是磁盘阵列中某些硬盘出现故障导致存储不可用,初步判断RAID中有硬盘掉线了。
|
19天前
|
API C# 开发框架
WPF与Web服务集成大揭秘:手把手教你调用RESTful API,客户端与服务器端优劣对比全解析!
【8月更文挑战第31天】在现代软件开发中,WPF 和 Web 服务各具特色。WPF 以其出色的界面展示能力受到欢迎,而 Web 服务则凭借跨平台和易维护性在互联网应用中占有一席之地。本文探讨了 WPF 如何通过 HttpClient 类调用 RESTful API,并展示了基于 ASP.NET Core 的 Web 服务如何实现同样的功能。通过对比分析,揭示了两者各自的优缺点:WPF 客户端直接处理数据,减轻服务器负担,但需处理网络异常;Web 服务则能利用服务器端功能如缓存和权限验证,但可能增加服务器负载。希望本文能帮助开发者根据具体需求选择合适的技术方案。
53 0