Skip to content

Commit 876b274

Browse files
committedFeb 16, 2018
[FAB-7777] Suppress couchdb index creation errors
This CR changes the statedb impl for the couchdb such that it does not propagate the errors that are caused by erroneous db artifacts present in the chaincode package archive. This is because a bad index file could bring the peer to a halt until the chaincode package in uninstalled Change-Id: Iecb8c243fcb5c8e01beb31a6ca21a53a1b14a354 Signed-off-by: manish <manish.sethi@gmail.com>
1 parent 1033fcb commit 876b274

File tree

2 files changed

+108
-76
lines changed

2 files changed

+108
-76
lines changed
 

‎core/ledger/kvledger/txmgmt/statedb/statecouchdb/statecouchdb.go

+17-23
Original file line numberDiff line numberDiff line change
@@ -84,65 +84,59 @@ func NewVersionedDBProvider() (*VersionedDBProvider, error) {
8484
}
8585

8686
//HandleChaincodeDeploy initializes database artifacts for the database associated with the namespace
87+
// This function delibrately suppresses the errors that occur during the creation of the indexes on couchdb.
88+
// This is because, in the present code, we do not differentiate between the errors because of couchdb interaction
89+
// and the errors because of bad index files - the later being unfixable by the admin. Note that the error suppression
90+
// is acceptable since peer can continue in the committing role without the indexes. However, executing chaincode queries
91+
// may be affected, until a new chaincode with fixed indexes is installed and instantiated
8792
func (vdb *VersionedDB) HandleChaincodeDeploy(chaincodeDefinition *cceventmgmt.ChaincodeDefinition, dbArtifactsTar []byte) error {
88-
8993
logger.Debugf("Entering HandleChaincodeDeploy")
90-
9194
if chaincodeDefinition == nil {
92-
return fmt.Errorf("chaincodeDefinition must not be nil")
95+
return fmt.Errorf("chaincodeDefinition found nil while creating couchdb index on chain=%s", vdb.chainName)
9396
}
94-
9597
db, err := vdb.getNamespaceDBHandle(chaincodeDefinition.Name)
9698
if err != nil {
9799
return err
98100
}
99-
100101
//initialize a reader for the artifacts tar file
101102
artifactReader := bytes.NewReader(dbArtifactsTar)
102103
tarReader := tar.NewReader(artifactReader)
103-
104104
for {
105-
106105
//read the next header from the tar
107106
tarHeader, err := tarReader.Next()
108-
109107
//if the EOF is detected, then exit
110108
if err == io.EOF {
111109
// end of tar archive
112-
break
110+
return nil
113111
}
114112
if err != nil {
115-
return err
113+
logger.Errorf("Error during reading db artifacts from tar file for chaincode=[%s] on chain=[%s]. Error=%s",
114+
chaincodeDefinition, vdb.chainName, err)
115+
return nil
116116
}
117117

118118
logger.Debugf("Reading artifact from file: %s", tarHeader.Name)
119-
120119
//Ensure that this is not a directory
121120
if !tarHeader.FileInfo().IsDir() {
122-
123121
//split the filename into directory and file name
124122
dir, file := filepath.Split(tarHeader.Name)
125-
126123
if dir == "META-INF/statedb/couchdb/indexes/" {
127-
128-
logger.Debugf("Creating index from file file: %s", file)
129-
124+
logger.Debugf("Creating index from file: %s", file)
130125
//read the tar entry into a byte array
131126
indexData, err := ioutil.ReadAll(tarReader)
132127
if err != nil {
133-
return err
128+
logger.Errorf("Error during extracting db artifacts file=[%s] from tar for chaincode=[%s] on chain=[%s]. Error=%s",
129+
tarHeader.Name, chaincodeDefinition, vdb.chainName, err)
130+
return nil
134131
}
135-
136132
//create the index from the tar entry
137-
err = db.CreateIndex(string(indexData))
138-
if err != nil {
139-
return err
133+
if err := db.CreateIndex(string(indexData)); err != nil {
134+
logger.Errorf("Error during creation of index from file=[%s] for chaincode=[%s] on chain=[%s]. Error=%s",
135+
tarHeader.Name, chaincodeDefinition, vdb.chainName, err)
140136
}
141137
}
142138
}
143139
}
144-
145-
return nil
146140
}
147141

148142
// GetDBHandle gets the handle to a named database

‎core/ledger/kvledger/txmgmt/statedb/statecouchdb/statecouchdb_test.go

+91-53
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,6 @@ import (
2020
"archive/tar"
2121
"bytes"
2222
"fmt"
23-
"log"
2423
"os"
2524
"testing"
2625
"time"
@@ -328,34 +327,13 @@ func TestHandleChaincodeDeploy(t *testing.T) {
328327
savePoint := version.NewHeight(2, 22)
329328
db.ApplyUpdates(batch, savePoint)
330329

331-
//Create a buffer for the tar file
332-
buffer := new(bytes.Buffer)
333-
tarWriter := tar.NewWriter(buffer)
334-
335-
//Add 2 index definitions
336-
var files = []struct {
337-
Name, Body string
338-
}{
339-
{"META-INF/statedb/couchdb/indexes/indexColorSortName.json", "{\"index\":{\"fields\":[{\"color\":\"desc\"}]},\"ddoc\":\"indexColorSortName\",\"name\":\"indexColorSortName\",\"type\":\"json\"}"},
340-
{"META-INF/statedb/couchdb/indexes/indexSizeSortName.json", "{\"index\":{\"fields\":[{\"size\":\"desc\"}]},\"ddoc\":\"indexSizeSortName\",\"name\":\"indexSizeSortName\",\"type\":\"json\"}"},
341-
}
342-
for _, file := range files {
343-
tarHeader := &tar.Header{
344-
Name: file.Name,
345-
Mode: 0600,
346-
Size: int64(len(file.Body)),
347-
}
348-
err := tarWriter.WriteHeader(tarHeader)
349-
testutil.AssertNoError(t, err, "")
350-
351-
_, err = tarWriter.Write([]byte(file.Body))
352-
testutil.AssertNoError(t, err, "")
353-
354-
}
355-
// Make sure to check the error on Close.
356-
if err := tarWriter.Close(); err != nil {
357-
log.Fatalln(err)
358-
}
330+
//Create a tar file for test with 2 index definitions
331+
dbArtifactsTarBytes := createTarBytesForTest(t,
332+
[]*testFile{
333+
{"META-INF/statedb/couchdb/indexes/indexColorSortName.json", `{"index":{"fields":[{"color":"desc"}]},"ddoc":"indexColorSortName","name":"indexColorSortName","type":"json"}`},
334+
{"META-INF/statedb/couchdb/indexes/indexSizeSortName.json", `{"index":{"fields":[{"size":"desc"}]},"ddoc":"indexSizeSortName","name":"indexSizeSortName","type":"json"}`},
335+
},
336+
)
359337

360338
//Create a query
361339
queryString := `{"selector":{"owner":"fred"}}`
@@ -369,38 +347,98 @@ func TestHandleChaincodeDeploy(t *testing.T) {
369347
_, err = db.ExecuteQuery("ns1", queryString)
370348
testutil.AssertError(t, err, "Error should have been thrown for a missing index")
371349

372-
if handleDefinition, ok := db.(cceventmgmt.ChaincodeLifecycleEventListener); ok {
350+
handleDefinition, _ := db.(cceventmgmt.ChaincodeLifecycleEventListener)
373351

374-
chaincodeDef := &cceventmgmt.ChaincodeDefinition{Name: "ns1", Hash: nil, Version: ""}
352+
chaincodeDef := &cceventmgmt.ChaincodeDefinition{Name: "ns1", Hash: nil, Version: ""}
375353

376-
//Test HandleChaincodeDefinition with a valid tar file
377-
err := handleDefinition.HandleChaincodeDeploy(chaincodeDef, buffer.Bytes())
378-
testutil.AssertNoError(t, err, "")
354+
//Test HandleChaincodeDefinition with a valid tar file
355+
err = handleDefinition.HandleChaincodeDeploy(chaincodeDef, dbArtifactsTarBytes)
356+
testutil.AssertNoError(t, err, "")
379357

380-
//Test HandleChaincodeDefinition with a nil tar file
381-
err = handleDefinition.HandleChaincodeDeploy(chaincodeDef, nil)
382-
testutil.AssertNoError(t, err, "")
358+
//Sleep to allow time for index creation
359+
time.Sleep(100 * time.Millisecond)
360+
//Create a query with a sort
361+
queryString = `{"selector":{"owner":"fred"}, "sort": [{"size": "desc"}]}`
362+
363+
//Query should complete without error
364+
_, err = db.ExecuteQuery("ns1", queryString)
365+
testutil.AssertNoError(t, err, "")
366+
367+
//Query namespace "ns2", index is only created in "ns1". This should return an error.
368+
_, err = db.ExecuteQuery("ns2", queryString)
369+
testutil.AssertError(t, err, "Error should have been thrown for a missing index")
383370

384-
//Test HandleChaincodeDefinition with a bad tar file
385-
err = handleDefinition.HandleChaincodeDeploy(chaincodeDef, []byte(`This is a really bad tar file`))
386-
testutil.AssertError(t, err, "Error should have been thrown for a bad tar file")
371+
//Test HandleChaincodeDefinition with a nil tar file
372+
err = handleDefinition.HandleChaincodeDeploy(chaincodeDef, nil)
373+
testutil.AssertNoError(t, err, "")
387374

388-
//Test HandleChaincodeDefinition with a nil chaincodeDef
389-
err = handleDefinition.HandleChaincodeDeploy(nil, []byte(`This is a really bad tar file`))
390-
testutil.AssertError(t, err, "Error should have been thrown for a nil chaincodeDefinition")
375+
//Test HandleChaincodeDefinition with a bad tar file
376+
err = handleDefinition.HandleChaincodeDeploy(chaincodeDef, []byte(`This is a really bad tar file`))
377+
testutil.AssertNoError(t, err, "Error should not have been thrown for a bad tar file")
391378

392-
//Sleep to allow time for index creation
393-
time.Sleep(100 * time.Millisecond)
394-
//Create a query with a sort
395-
queryString = `{"selector":{"owner":"fred"}, "sort": [{"size": "desc"}]}`
379+
//Test HandleChaincodeDefinition with a nil chaincodeDef
380+
err = handleDefinition.HandleChaincodeDeploy(nil, dbArtifactsTarBytes)
381+
testutil.AssertError(t, err, "Error should have been thrown for a nil chaincodeDefinition")
382+
}
396383

397-
//Query should complete without error
398-
_, err = db.ExecuteQuery("ns1", queryString)
399-
testutil.AssertNoError(t, err, "")
384+
func TestHandleChaincodeDeployErroneousIndexFile(t *testing.T) {
385+
channelName := "ch1"
386+
env := NewTestVDBEnv(t)
387+
env.Cleanup(channelName)
388+
defer env.Cleanup(channelName)
389+
db, err := env.DBProvider.GetDBHandle(channelName)
390+
testutil.AssertNoError(t, err, "")
391+
db.Open()
392+
defer db.Close()
393+
394+
batch := statedb.NewUpdateBatch()
395+
batch.Put("ns1", "key1", []byte(`{"asset_name": "marble1","color": "blue","size": 1,"owner": "tom"}`), version.NewHeight(1, 1))
396+
batch.Put("ns1", "key2", []byte(`{"asset_name": "marble2","color": "blue","size": 2,"owner": "jerry"}`), version.NewHeight(1, 2))
397+
ccEventListener, _ := db.(cceventmgmt.ChaincodeLifecycleEventListener)
398+
399+
// Create a tar file for test with 2 index definitions - one of them being errorneous
400+
badSyntaxFileContent := `{"index":{"fields": This is a bad json}`
401+
dbArtifactsTarBytes := createTarBytesForTest(t,
402+
[]*testFile{
403+
{"META-INF/statedb/couchdb/indexes/indexSizeSortName.json", `{"index":{"fields":[{"size":"desc"}]},"ddoc":"indexSizeSortName","name":"indexSizeSortName","type":"json"}`},
404+
{"META-INF/statedb/couchdb/indexes/badSyntax.json", badSyntaxFileContent},
405+
},
406+
)
407+
// Test HandleChaincodeDefinition with a bad tar file
408+
chaincodeName := "ns1"
409+
chaincodeVer := "1.0"
410+
chaincodeDef := &cceventmgmt.ChaincodeDefinition{Name: chaincodeName, Hash: []byte("Hash for test chaincode"), Version: chaincodeVer}
411+
err = ccEventListener.HandleChaincodeDeploy(chaincodeDef, dbArtifactsTarBytes)
412+
testutil.AssertNoError(t, err, "A tar with a bad syntax file should not cause an error")
413+
//Sleep to allow time for index creation
414+
time.Sleep(100 * time.Millisecond)
415+
//Query should complete without error
416+
_, err = db.ExecuteQuery("ns1", `{"selector":{"owner":"fred"}, "sort": [{"size": "desc"}]}`)
417+
testutil.AssertNoError(t, err, "")
418+
}
400419

401-
//Query namespace "ns2", index is only created in "ns1". This should return an error.
402-
_, err = db.ExecuteQuery("ns2", queryString)
403-
testutil.AssertError(t, err, "Error should have been thrown for a missing index")
420+
type testFile struct {
421+
name, body string
422+
}
404423

424+
func createTarBytesForTest(t *testing.T, testFiles []*testFile) []byte {
425+
//Create a buffer for the tar file
426+
buffer := new(bytes.Buffer)
427+
tarWriter := tar.NewWriter(buffer)
428+
429+
for _, file := range testFiles {
430+
tarHeader := &tar.Header{
431+
Name: file.name,
432+
Mode: 0600,
433+
Size: int64(len(file.body)),
434+
}
435+
err := tarWriter.WriteHeader(tarHeader)
436+
testutil.AssertNoError(t, err, "")
437+
438+
_, err = tarWriter.Write([]byte(file.body))
439+
testutil.AssertNoError(t, err, "")
405440
}
441+
// Make sure to check the error on Close.
442+
testutil.AssertNoError(t, tarWriter.Close(), "")
443+
return buffer.Bytes()
406444
}

0 commit comments

Comments
 (0)