aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMeng-Ying Yang <garfield@dexon.org>2019-04-15 12:01:34 +0800
committerJhih-Ming Huang <jm.huang@cobinhood.com>2019-05-06 10:44:05 +0800
commitba915a703cda5172b37d20d93cc46c8d4c1ddc34 (patch)
treeba3a8765e795f2b73c11159240847e0c93c8d6a3
parent65362012ce4028c1438fe85dcaf8f17576423a2f (diff)
downloaddexon-ba915a703cda5172b37d20d93cc46c8d4c1ddc34.tar.gz
dexon-ba915a703cda5172b37d20d93cc46c8d4c1ddc34.tar.zst
dexon-ba915a703cda5172b37d20d93cc46c8d4c1ddc34.zip
core: vm: sqlvm: add built-in function BLOCK_COINBASE()
-rw-r--r--core/vm/sqlvm/runtime/functions.go11
-rw-r--r--core/vm/sqlvm/runtime/functions_test.go53
2 files changed, 64 insertions, 0 deletions
diff --git a/core/vm/sqlvm/runtime/functions.go b/core/vm/sqlvm/runtime/functions.go
index 90b1a93fe..c21d1da19 100644
--- a/core/vm/sqlvm/runtime/functions.go
+++ b/core/vm/sqlvm/runtime/functions.go
@@ -14,6 +14,7 @@ const (
BLOCKHASH = "BLOCK_HASH"
BLOCKNUMBER = "BLOCK_NUMBER"
BLOCKTIMESTAMP = "BLOCK_TIMESTAMP"
+ BLOCKCOINBASE = "BLOCK_COINBASE"
NOW = "NOW"
)
@@ -25,6 +26,7 @@ var (
BLOCKNUMBER: fnBlockNumber,
BLOCKTIMESTAMP: fnBlockTimestamp,
NOW: fnBlockTimestamp,
+ BLOCKCOINBASE: fnBlockCoinBase,
}
)
@@ -99,3 +101,12 @@ func fnBlockTimestamp(ctx *common.Context, ops []*Operand, length uint64) (resul
return
}
+func fnBlockCoinBase(ctx *common.Context, ops []*Operand, length uint64) (result *Operand, err error) {
+ r := &Raw{Bytes: ctx.Coinbase.Bytes()}
+ result = assignFuncResult(
+ []ast.DataType{ast.ComposeDataType(ast.DataTypeMajorAddress, 0)},
+ r.clone, length,
+ )
+ return
+}
+
diff --git a/core/vm/sqlvm/runtime/functions_test.go b/core/vm/sqlvm/runtime/functions_test.go
index e90e9b6c6..cda24fcdf 100644
--- a/core/vm/sqlvm/runtime/functions_test.go
+++ b/core/vm/sqlvm/runtime/functions_test.go
@@ -205,3 +205,56 @@ func (s *FunctionSuite) TestFnBlockTimestamp() {
}
}
+func (s *FunctionSuite) TestFnCoinBase() {
+ type blockCoinBaseCase struct {
+ Name string
+ Address dexCommon.Address
+ Length uint64
+ Res []byte
+ Err error
+ }
+
+ res := []byte{
+ 0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef,
+ 0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef,
+ 0x01, 0x23, 0x45, 0x67}
+ address := dexCommon.BytesToAddress(res)
+
+ testcases := []blockCoinBaseCase{
+ {"address with length 1", address, 1, res, nil},
+ {"address with length 10", address, 10, res, nil},
+ {"address with length 0", address, 0, res, nil},
+ }
+
+ callFn := func(c blockCoinBaseCase) (*Operand, error) {
+ return fnBlockCoinBase(
+ &common.Context{
+ Context: vm.Context{Coinbase: c.Address},
+ },
+ nil,
+ c.Length)
+ }
+
+ meta := []ast.DataType{ast.ComposeDataType(ast.DataTypeMajorAddress, 0)}
+
+ for idx, tCase := range testcases {
+ r, err := callFn(tCase)
+ s.Require().Equal(
+ tCase.Err, err,
+ "Index: %v. Error not expected: %v != %v", idx, tCase.Err, err)
+ s.Require().Equal(
+ meta, r.Meta,
+ "Index: %v. Meta not equal: %v != %v", idx, meta, r.Meta)
+ s.Require().Equal(
+ uint64(len(r.Data)), tCase.Length,
+ "Index: %v. Length not equal: %v != %v", idx, len(r.Data), tCase.Length)
+
+ for i := 0; i < len(r.Data); i++ {
+ s.Require().Equal(
+ tCase.Res, r.Data[i][0].Bytes,
+ "Index: %v. Data Index: %v. Value not equal: %v != %v",
+ idx, i, tCase.Res, r.Data[i][0].Bytes)
+ }
+ }
+}
+