/Users/qinjianquan/publicChain/Part78-Net-Conn
使用goroutine并行处理请求
/Users/qinjianquan/publicChain/Part78-Net-Conn/src/Server.go
func StartServer(nodeID string, mineAddress string) {
--
for {
//receive data from client
conn, err1 := ln.Accept()
if err != nil {
log.Panic(err1)
}
go HandleConnection(conn)
}
}
func HandleConnection(conn net.Conn) {
//read data from client
request, err := ioutil.ReadAll(conn)
if err != nil {
log.Panic(err)
}
fmt.Printf("received a message:%s\n", request[:COMMANDLENGTH])
}
配置handle方法
func handleAddr(request []byte, bc *Blockchain) {}
func handleVersion(request []byte, bc *Blockchain) {}
func handleBlock(request []byte, bc *Blockchain) {}
func handleInv(request []byte, bc *Blockchain) {}
func handleGetblocks(request []byte, bc *Blockchain) {}
func handleTx(request []byte, bc *Blockchain) {}
func handleGetData(request []byte, bc *Blockchain) {}
配置常量
const COMMAND_VERSION = "version"
const COMMAND_ADDR = "addr"
const COMMAND_BLOCK = "block"
const COMMAND_INV = "inv"
const COMMAND_GETBLOCKS = "getblocks"
const COMMAND_GETDATA = "getdata"
const COMMAND_TX = "tx"
编写case
func HandleConnection(conn net.Conn) {
//read data from client
request, err := ioutil.ReadAll(conn)
if err != nil {
log.Panic(err)
}
command := bytesToCommand(request[:COMMANDLENGTH])
fmt.Printf("received %s command\n", command)
bc := BlockChainObject("test")
switch command {
case COMMAND_VERSION:
handleVersion(request, bc)
case COMMAND_ADDR:
handleAddr(request, bc)
case COMMAND_BLOCK:
handleBlock(request, bc)
case COMMAND_GETBLOCKS:
handleGetblocks(request, bc)
case COMMAND_GETDATA:
handleGetData(request, bc)
case COMMAND_INV:
handleInv(request, bc)
case COMMAND_TX:
handleTx(request, bc)
default:
fmt.Println("Unknown command!")
}
defer conn.Close()
}
func handleVersion(request []byte, bc *Blockchain) {
var buff bytes.Buffer
var payload Version
dataBytes := request[COMMANDLENGTH:]
buff.Write(dataBytes)
dec := gob.NewDecoder(&buff)
err := dec.Decode(&payload)
if err != nil {
log.Panic(err)
}
bestHeight := bc.GetBestHeight()
foreignerBestHeight := payload.BestHeight
if bestHeight > foreignerBestHeight {
SendVersion(payload.AddrFrom, bc)
} else if bestHeight < foreignerBestHeight {
SendGetBlocks(payload.AddrFrom)
}
}
/Users/qinjianquan/publicChain/Part80-Net-sendGetBlocks/src/Server_getBlocks.go
type GetBlocks struct {
AddFrom string
}
这里涉及多case的处理,代码过程请参考github本项目:https://github.com/jianquanqin/publicChain
本项目中,master node 端口号设置为了3000;wallet node 端口号为 3001;miner node 端口号为 3002
1.master node
创建wallet -> 创建创世区块 ->备份数据库为创世区块数据库
2.wallet node
继续备份数据库wallet node数据库 -> 创建4个钱包地址 ->切换到master node向不同地址转两笔账 -> wallet node 查询余额
此时wallet node数据库中只有1个区块,我们先启动master code,接着启动wallet node,让wallet node同步主节点的数据,同步结束后wallet node数据库中应该有三个区块,余额也应该能够正常查询
3.miner node
在miner node中创建wallet address ->拷贝创世区块数据库->启动miner node ->数据同步 ->切换到wallet node 进行转两笔账(tx会发送给master node)->master node 将inv发送给miner node ->miner node若无该数据,则会存储,只有数据量达到规定数目的时候,miner node才会打包区块
创建主节点
cd /Users/qinjianquan/publicChain/Part84-Net-validation
go build -o bc main.go
export NODE_ID=3000
./bc createWallet
./bc createBlockChain -address
备份数据库
cp blockChain_3000.db blockChain_genesis.db
创建wallet node
打开一个新的终端
cd /Users/qinjianquan/publicChain/Part84-Net-validation
export NODE_ID=3001
备份数据库
cp blockChain_genesis.db blockChain_3001.db
创建四个钱包
./bc createWallet
./bc createWallet
./bc createWallet
./bc createWallet
查看钱包
./bc getAddressList
NODE_ID:3001
Address list:
19nkshsC71LkTWkqXHKSm6NsdzCwTaCy2r
122sTFRy9kxUFbvhHMkzENcfs52odyvHny
1G6bg8U2wiqXhyY3K9tKeJrTKGJz6UwisZ
1NTeiYo2b4J9wD6Wg6aujKoz1v4XeBprvj
在主节点中向wallet node中前两个钱包转账
./bc transfer -from '["18UmevN8CW6Fm2h5dt6DixbrXTH42VWDxE"]' -to '["19nkshsC71LkTWkqXHKSm6NsdzCwTaCy2r"]' -amount '["8"]' -mine
./bc transfer -from '["18UmevN8CW6Fm2h5dt6DixbrXTH42VWDxE"]' -to '["122sTFRy9kxUFbvhHMkzENcfs52odyvHny"]' -amount '["6"]' -mine
查看余额
./bc getBalance -address 18UmevN8CW6Fm2h5dt6DixbrXTH42VWDxE
./bc getBalance -address 19nkshsC71LkTWkqXHKSm6NsdzCwTaCy2r
./bc getBalance -address 122sTFRy9kxUFbvhHMkzENcfs52odyvHny
启动主节点
./bc startNode
启动wallet node
./bc startNode
./bc startNode
NODE_ID:3000
start the server, localhost:3000
received version command
version
the client sends message to the server...
received getBlocks command
getBlocks
the client sends message to the server...
received getData command
getData
the client sends message to the server...
received getData command
getData
the client sends message to the server...
received getData command
getData
the client sends message to the server...
./bc startNode
NODE_ID:3001
start the server, localhost:3001
the client sends message to the server...
received version command
version
the client sends message to the server...
received inv command
inv
the client sends message to the server...
received block command
block
the client sends message to the server...
received block command
block
此时关闭进程,重制数据库就可以查到余额了,这说明节点数据已经同步好了
122sTFRy9kxUFbvhHMkzENcfs52odyvHny has 6 token
qinjianquan@MacBook-Pro-10 Part84-Net-validation % ./bc getBalance -address 19nkshsC71LkTWkqXHKSm6NsdzCwTaCy2r
NODE_ID:3001
19nkshsC71LkTWkqXHKSm6NsdzCwTaCy2r has 8 token
qinjianquan@MacBook-Pro-10 Part84-Net-validation % ./bc getBalance -address 18UmevN8CW6Fm2h5dt6DixbrXTH42VWDxE
NODE_ID:3001
18UmevN8CW6Fm2h5dt6DixbrXTH42VWDxE has 16 token
我正在尝试使用ruby和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我
我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..
如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby
我在我的Rails项目中使用Pow和powifygem。现在我尝试升级我的ruby版本(从1.9.3到2.0.0,我使用RVM)当我切换ruby版本、安装所有gem依赖项时,我通过运行railss并访问localhost:3000确保该应用程序正常运行以前,我通过使用pow访问http://my_app.dev来浏览我的应用程序。升级后,由于错误Bundler::RubyVersionMismatch:YourRubyversionis1.9.3,butyourGemfilespecified2.0.0,此url不起作用我尝试过的:重新创建pow应用程序重启pow服务器更新战俘
我已经像这样安装了一个新的Rails项目:$railsnewsite它执行并到达:bundleinstall但是当它似乎尝试安装依赖项时我得到了这个错误Gem::Ext::BuildError:ERROR:Failedtobuildgemnativeextension./System/Library/Frameworks/Ruby.framework/Versions/2.0/usr/bin/rubyextconf.rbcheckingforlibkern/OSAtomic.h...yescreatingMakefilemake"DESTDIR="cleanmake"DESTDIR="
最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru
在Rails4.0.2中,我使用s3_direct_upload和aws-sdkgems直接为s3存储桶上传文件。在开发环境中它工作正常,但在生产环境中它会抛出如下错误,ActionView::Template::Error(noimplicitconversionofnilintoString)在View中,create_cv_url,:id=>"s3_uploader",:key=>"cv_uploads/{unique_id}/${filename}",:key_starts_with=>"cv_uploads/",:callback_param=>"cv[direct_uplo
我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b
假设我有这个范围:("aaaaa".."zzzzz")如何在不事先/每次生成整个项目的情况下从范围中获取第N个项目? 最佳答案 一种快速简便的方法:("aaaaa".."zzzzz").first(42).last#==>"aaabp"如果出于某种原因你不得不一遍又一遍地这样做,或者如果你需要避免为前N个元素构建中间数组,你可以这样写:moduleEnumerabledefskip(n)returnto_enum:skip,nunlessblock_given?each_with_indexdo|item,index|yieldit
您如何在Rails中的实时服务器上进行有效调试,无论是在测试版/生产服务器上?我试过直接在服务器上修改文件,然后重启应用,但是修改好像没有生效,或者需要很长时间(缓存?)我也试过在本地做“脚本/服务器生产”,但是那很慢另一种选择是编码和部署,但效率很低。有人对他们如何有效地做到这一点有任何见解吗? 最佳答案 我会回答你的问题,即使我不同意这种热修补服务器代码的方式:)首先,你真的确定你已经重启了服务器吗?您可以通过跟踪日志文件来检查它。您更改的代码显示的View可能会被缓存。缓存页面位于tmp/cache文件夹下。您可以尝试手动删除