代码拉取完成,页面将自动刷新
/*
Copyright IBM Corp. 2016 All Rights Reserved.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package jsonledger
import (
"bytes"
"fmt"
"os"
"path/filepath"
"sync"
"github.com/hyperledger/fabric/common/flogging"
"github.com/hyperledger/fabric/common/ledger/blockledger"
cb "github.com/hyperledger/fabric/protos/common"
ab "github.com/hyperledger/fabric/protos/orderer"
"github.com/op/go-logging"
"github.com/golang/protobuf/jsonpb"
)
const pkgLogID = "orderer/ledger/jsonledger"
var logger *logging.Logger
var closedChan chan struct{}
var fileLock sync.Mutex
func init() {
logger = flogging.MustGetLogger(pkgLogID)
closedChan = make(chan struct{})
close(closedChan)
}
const (
blockFileFormatString = "block_%020d.json"
chainDirectoryFormatString = "chain_%s"
)
type cursor struct {
jl *jsonLedger
blockNumber uint64
}
type jsonLedger struct {
directory string
height uint64
lastHash []byte
marshaler *jsonpb.Marshaler
mutex sync.Mutex
signal chan struct{}
}
// readBlock returns the block or nil, and whether the block was found or not, (nil,true) generally indicates an irrecoverable problem
func (jl *jsonLedger) readBlock(number uint64) (*cb.Block, bool) {
name := jl.blockFilename(number)
// In case of ongoing write, reading the block file may result in `unexpected EOF` error.
// Therefore, we use file mutex here to prevent this race condition.
fileLock.Lock()
defer fileLock.Unlock()
file, err := os.Open(name)
if err == nil {
defer file.Close()
block := &cb.Block{}
err = jsonpb.Unmarshal(file, block)
if err != nil {
return nil, true
}
logger.Debugf("Read block %d", block.Header.Number)
return block, true
}
return nil, false
}
// Next blocks until there is a new block available, or returns an error if the
// next block is no longer retrievable
func (cu *cursor) Next() (*cb.Block, cb.Status) {
// This only loops once, as signal reading
// indicates the new block has been written
for {
block, found := cu.jl.readBlock(cu.blockNumber)
if found {
if block == nil {
return nil, cb.Status_SERVICE_UNAVAILABLE
}
cu.blockNumber++
return block, cb.Status_SUCCESS
}
// copy the signal channel under lock to avoid race
// with new signal channel in append
cu.jl.mutex.Lock()
signal := cu.jl.signal
cu.jl.mutex.Unlock()
<-signal
}
}
// ReadyChan supplies a channel which will block until Next will not block
func (cu *cursor) ReadyChan() <-chan struct{} {
cu.jl.mutex.Lock()
signal := cu.jl.signal
cu.jl.mutex.Unlock()
if _, err := os.Stat(cu.jl.blockFilename(cu.blockNumber)); os.IsNotExist(err) {
return signal
}
return closedChan
}
func (cu *cursor) Close() {}
// Iterator returns an Iterator, as specified by a ab.SeekInfo message, and its
// starting block number
func (jl *jsonLedger) Iterator(startPosition *ab.SeekPosition) (blockledger.Iterator, uint64) {
switch start := startPosition.Type.(type) {
case *ab.SeekPosition_Oldest:
return &cursor{jl: jl, blockNumber: 0}, 0
case *ab.SeekPosition_Newest:
high := jl.height - 1
return &cursor{jl: jl, blockNumber: high}, high
case *ab.SeekPosition_Specified:
if start.Specified.Number > jl.height {
return &blockledger.NotFoundErrorIterator{}, 0
}
return &cursor{jl: jl, blockNumber: start.Specified.Number}, start.Specified.Number
default:
return &blockledger.NotFoundErrorIterator{}, 0
}
}
// Height returns the number of blocks on the ledger
func (jl *jsonLedger) Height() uint64 {
return jl.height
}
// Append appends a new block to the ledger
func (jl *jsonLedger) Append(block *cb.Block) error {
if block.Header.Number != jl.height {
return fmt.Errorf("Block number should have been %d but was %d", jl.height, block.Header.Number)
}
if !bytes.Equal(block.Header.PreviousHash, jl.lastHash) {
return fmt.Errorf("Block should have had previous hash of %x but was %x", jl.lastHash, block.Header.PreviousHash)
}
jl.writeBlock(block)
jl.lastHash = block.Header.Hash()
jl.height++
// Manage the signal channel under lock to avoid race with read in Next
jl.mutex.Lock()
close(jl.signal)
jl.signal = make(chan struct{})
jl.mutex.Unlock()
return nil
}
// writeBlock commits a block to disk
func (jl *jsonLedger) writeBlock(block *cb.Block) {
name := jl.blockFilename(block.Header.Number)
fileLock.Lock()
defer fileLock.Unlock()
file, err := os.Create(name)
if err != nil {
panic(err)
}
defer file.Close()
err = jl.marshaler.Marshal(file, block)
logger.Debugf("Wrote block %d", block.Header.Number)
if err != nil {
logger.Panic(err)
}
}
// blockFilename returns the fully qualified path to where a block
// of a given number should be stored on disk
func (jl *jsonLedger) blockFilename(number uint64) string {
return filepath.Join(jl.directory, fmt.Sprintf(blockFileFormatString, number))
}
此处可能存在不合适展示的内容,页面不予展示。您可通过相关编辑功能自查并修改。
如您确认内容无涉及 不当用语 / 纯广告导流 / 暴力 / 低俗色情 / 侵权 / 盗版 / 虚假 / 无价值内容或违法国家有关法律法规的内容,可点击提交进行申诉,我们将尽快为您处理。