You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

316 lines
8.1 KiB

8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
  1. package dummy
  2. import (
  3. "bytes"
  4. "io/ioutil"
  5. "sort"
  6. "testing"
  7. "github.com/stretchr/testify/require"
  8. abcicli "github.com/tendermint/abci/client"
  9. "github.com/tendermint/abci/server"
  10. "github.com/tendermint/abci/types"
  11. crypto "github.com/tendermint/go-crypto"
  12. "github.com/tendermint/merkleeyes/iavl"
  13. cmn "github.com/tendermint/tmlibs/common"
  14. "github.com/tendermint/tmlibs/log"
  15. )
  16. func testDummy(t *testing.T, app types.Application, tx []byte, key, value string) {
  17. ar := app.DeliverTx(tx)
  18. require.False(t, ar.IsErr(), ar)
  19. // repeating tx doesn't raise error
  20. ar = app.DeliverTx(tx)
  21. require.False(t, ar.IsErr(), ar)
  22. // make sure query is fine
  23. resQuery := app.Query(types.RequestQuery{
  24. Path: "/store",
  25. Data: []byte(key),
  26. })
  27. require.Equal(t, types.CodeType_OK, resQuery.Code)
  28. require.Equal(t, value, string(resQuery.Value))
  29. // make sure proof is fine
  30. resQuery = app.Query(types.RequestQuery{
  31. Path: "/store",
  32. Data: []byte(key),
  33. Prove: true,
  34. })
  35. require.Equal(t, types.CodeType_OK, resQuery.Code)
  36. require.Equal(t, value, string(resQuery.Value))
  37. proof, err := iavl.ReadProof(resQuery.Proof)
  38. require.Nil(t, err)
  39. require.True(t, proof.Verify([]byte(key), resQuery.Value, proof.RootHash)) // NOTE: we have no way to verify the RootHash
  40. }
  41. func TestDummyKV(t *testing.T) {
  42. dummy := NewDummyApplication()
  43. key := "abc"
  44. value := key
  45. tx := []byte(key)
  46. testDummy(t, dummy, tx, key, value)
  47. value = "def"
  48. tx = []byte(key + "=" + value)
  49. testDummy(t, dummy, tx, key, value)
  50. }
  51. func TestPersistentDummyKV(t *testing.T) {
  52. dir, err := ioutil.TempDir("/tmp", "abci-dummy-test") // TODO
  53. if err != nil {
  54. t.Fatal(err)
  55. }
  56. dummy := NewPersistentDummyApplication(dir)
  57. key := "abc"
  58. value := key
  59. tx := []byte(key)
  60. testDummy(t, dummy, tx, key, value)
  61. value = "def"
  62. tx = []byte(key + "=" + value)
  63. testDummy(t, dummy, tx, key, value)
  64. }
  65. func TestPersistentDummyInfo(t *testing.T) {
  66. dir, err := ioutil.TempDir("/tmp", "abci-dummy-test") // TODO
  67. if err != nil {
  68. t.Fatal(err)
  69. }
  70. dummy := NewPersistentDummyApplication(dir)
  71. height := uint64(0)
  72. resInfo := dummy.Info()
  73. if resInfo.LastBlockHeight != height {
  74. t.Fatalf("expected height of %d, got %d", height, resInfo.LastBlockHeight)
  75. }
  76. // make and apply block
  77. height = uint64(1)
  78. hash := []byte("foo")
  79. header := &types.Header{
  80. Height: uint64(height),
  81. }
  82. dummy.BeginBlock(hash, header)
  83. dummy.EndBlock(height)
  84. dummy.Commit()
  85. resInfo = dummy.Info()
  86. if resInfo.LastBlockHeight != height {
  87. t.Fatalf("expected height of %d, got %d", height, resInfo.LastBlockHeight)
  88. }
  89. }
  90. // add a validator, remove a validator, update a validator
  91. func TestValSetChanges(t *testing.T) {
  92. dir, err := ioutil.TempDir("/tmp", "abci-dummy-test") // TODO
  93. if err != nil {
  94. t.Fatal(err)
  95. }
  96. dummy := NewPersistentDummyApplication(dir)
  97. // init with some validators
  98. total := 10
  99. nInit := 5
  100. vals := make([]*types.Validator, total)
  101. for i := 0; i < total; i++ {
  102. pubkey := crypto.GenPrivKeyEd25519FromSecret([]byte(cmn.Fmt("test%d", i))).PubKey().Bytes()
  103. power := cmn.RandInt()
  104. vals[i] = &types.Validator{pubkey, uint64(power)}
  105. }
  106. // iniitalize with the first nInit
  107. dummy.InitChain(vals[:nInit])
  108. vals1, vals2 := vals[:nInit], dummy.Validators()
  109. valsEqual(t, vals1, vals2)
  110. var v1, v2, v3 *types.Validator
  111. // add some validators
  112. v1, v2 = vals[nInit], vals[nInit+1]
  113. diff := []*types.Validator{v1, v2}
  114. tx1 := MakeValSetChangeTx(v1.PubKey, v1.Power)
  115. tx2 := MakeValSetChangeTx(v2.PubKey, v2.Power)
  116. makeApplyBlock(t, dummy, 1, diff, tx1, tx2)
  117. vals1, vals2 = vals[:nInit+2], dummy.Validators()
  118. valsEqual(t, vals1, vals2)
  119. // remove some validators
  120. v1, v2, v3 = vals[nInit-2], vals[nInit-1], vals[nInit]
  121. v1.Power = 0
  122. v2.Power = 0
  123. v3.Power = 0
  124. diff = []*types.Validator{v1, v2, v3}
  125. tx1 = MakeValSetChangeTx(v1.PubKey, v1.Power)
  126. tx2 = MakeValSetChangeTx(v2.PubKey, v2.Power)
  127. tx3 := MakeValSetChangeTx(v3.PubKey, v3.Power)
  128. makeApplyBlock(t, dummy, 2, diff, tx1, tx2, tx3)
  129. vals1 = append(vals[:nInit-2], vals[nInit+1])
  130. vals2 = dummy.Validators()
  131. valsEqual(t, vals1, vals2)
  132. // update some validators
  133. v1 = vals[0]
  134. if v1.Power == 5 {
  135. v1.Power = 6
  136. } else {
  137. v1.Power = 5
  138. }
  139. diff = []*types.Validator{v1}
  140. tx1 = MakeValSetChangeTx(v1.PubKey, v1.Power)
  141. makeApplyBlock(t, dummy, 3, diff, tx1)
  142. vals1 = append([]*types.Validator{v1}, vals1[1:len(vals1)]...)
  143. vals2 = dummy.Validators()
  144. valsEqual(t, vals1, vals2)
  145. }
  146. func makeApplyBlock(t *testing.T, dummy types.Application, heightInt int, diff []*types.Validator, txs ...[]byte) {
  147. // make and apply block
  148. height := uint64(heightInt)
  149. hash := []byte("foo")
  150. header := &types.Header{
  151. Height: height,
  152. }
  153. dummy.BeginBlock(hash, header)
  154. for _, tx := range txs {
  155. if r := dummy.DeliverTx(tx); r.IsErr() {
  156. t.Fatal(r)
  157. }
  158. }
  159. resEndBlock := dummy.EndBlock(height)
  160. dummy.Commit()
  161. valsEqual(t, diff, resEndBlock.Diffs)
  162. }
  163. // order doesn't matter
  164. func valsEqual(t *testing.T, vals1, vals2 []*types.Validator) {
  165. if len(vals1) != len(vals2) {
  166. t.Fatalf("vals dont match in len. got %d, expected %d", len(vals2), len(vals1))
  167. }
  168. sort.Sort(types.Validators(vals1))
  169. sort.Sort(types.Validators(vals2))
  170. for i, v1 := range vals1 {
  171. v2 := vals2[i]
  172. if !bytes.Equal(v1.PubKey, v2.PubKey) ||
  173. v1.Power != v2.Power {
  174. t.Fatalf("vals dont match at index %d. got %X/%d , expected %X/%d", i, v2.PubKey, v2.Power, v1.PubKey, v1.Power)
  175. }
  176. }
  177. }
  178. func makeSocketClientServer(app types.Application, name string) (abcicli.Client, cmn.Service, error) {
  179. // Start the listener
  180. socket := cmn.Fmt("unix://%s.sock", name)
  181. logger := log.TestingLogger()
  182. server, err := server.NewSocketServer(socket, app)
  183. if err != nil {
  184. return nil, nil, err
  185. }
  186. server.SetLogger(logger.With("module", "abci-server"))
  187. // Connect to the socket
  188. client, err := abcicli.NewSocketClient(socket, false)
  189. if err != nil {
  190. server.Stop()
  191. return nil, nil, err
  192. }
  193. client.SetLogger(logger.With("module", "abci-client"))
  194. client.Start()
  195. return client, server, err
  196. }
  197. func makeGRPCClientServer(app types.Application, name string) (abcicli.Client, cmn.Service, error) {
  198. // Start the listener
  199. socket := cmn.Fmt("unix://%s.sock", name)
  200. logger := log.TestingLogger()
  201. gapp := types.NewGRPCApplication(app)
  202. server, err := server.NewGRPCServer(socket, gapp)
  203. if err != nil {
  204. return nil, nil, err
  205. }
  206. server.SetLogger(logger.With("module", "abci-server"))
  207. client, err := abcicli.NewGRPCClient(socket, true)
  208. if err != nil {
  209. server.Stop()
  210. return nil, nil, err
  211. }
  212. client.SetLogger(logger.With("module", "abci-client"))
  213. return client, server, err
  214. }
  215. func TestClientServer(t *testing.T) {
  216. // set up socket app
  217. dummy := NewDummyApplication()
  218. client, server, err := makeSocketClientServer(dummy, "dummy-socket")
  219. require.Nil(t, err)
  220. defer server.Stop()
  221. defer client.Stop()
  222. runClientTests(t, client)
  223. // set up grpc app
  224. dummy = NewDummyApplication()
  225. gclient, gserver, err := makeGRPCClientServer(dummy, "dummy-grpc")
  226. require.Nil(t, err)
  227. defer gserver.Stop()
  228. defer gclient.Stop()
  229. runClientTests(t, gclient)
  230. }
  231. func runClientTests(t *testing.T, client abcicli.Client) {
  232. // run some tests....
  233. key := "abc"
  234. value := key
  235. tx := []byte(key)
  236. testClient(t, client, tx, key, value)
  237. value = "def"
  238. tx = []byte(key + "=" + value)
  239. testClient(t, client, tx, key, value)
  240. }
  241. func testClient(t *testing.T, app abcicli.Client, tx []byte, key, value string) {
  242. ar := app.DeliverTxSync(tx)
  243. require.False(t, ar.IsErr(), ar)
  244. // repeating tx doesn't raise error
  245. ar = app.DeliverTxSync(tx)
  246. require.False(t, ar.IsErr(), ar)
  247. // make sure query is fine
  248. resQuery, err := app.QuerySync(types.RequestQuery{
  249. Path: "/store",
  250. Data: []byte(key),
  251. })
  252. require.Nil(t, err)
  253. require.Equal(t, types.CodeType_OK, resQuery.Code)
  254. require.Equal(t, value, string(resQuery.Value))
  255. // make sure proof is fine
  256. resQuery, err = app.QuerySync(types.RequestQuery{
  257. Path: "/store",
  258. Data: []byte(key),
  259. Prove: true,
  260. })
  261. require.Nil(t, err)
  262. require.Equal(t, types.CodeType_OK, resQuery.Code)
  263. require.Equal(t, value, string(resQuery.Value))
  264. proof, err := iavl.ReadProof(resQuery.Proof)
  265. require.Nil(t, err)
  266. require.True(t, proof.Verify([]byte(key), resQuery.Value, proof.RootHash)) // NOTE: we have no way to verify the RootHash
  267. }