core: validate manifest before incrementing ID in Contract.Create

Also add more tests.
This commit is contained in:
Evgenii Stratonikov 2020-12-02 14:49:22 +03:00
parent b203c23515
commit dcc58b7c44
2 changed files with 46 additions and 11 deletions

View file

@ -115,13 +115,13 @@ func contractCreate(ic *interop.Context) error {
if contract != nil && err == nil { if contract != nil && err == nil {
return errors.New("contract already exists") return errors.New("contract already exists")
} }
if !manif.IsValid(h) {
return errors.New("failed to check contract script hash against manifest")
}
id, err := ic.DAO.GetAndUpdateNextContractID() id, err := ic.DAO.GetAndUpdateNextContractID()
if err != nil { if err != nil {
return err return err
} }
if !manif.IsValid(h) {
return errors.New("failed to check contract script hash against manifest")
}
newcontract := &state.Contract{ newcontract := &state.Contract{
ID: id, ID: id,
Hash: h, Hash: h,

View file

@ -589,14 +589,24 @@ func TestContractCreate(t *testing.T) {
// nef.NewFile() cares about version a lot. // nef.NewFile() cares about version a lot.
config.Version = "0.90.0-test" config.Version = "0.90.0-test"
ne, err := nef.NewFile(cs.Script)
require.NoError(t, err)
neb, err := ne.Bytes()
require.NoError(t, err)
priv, err := keys.NewPrivateKey()
require.NoError(t, err)
sender := util.Uint160{1, 2, 3}
h := state.CreateContractHash(sender, ne.Script)
sig := priv.Sign(h.BytesBE())
cs.Manifest.Groups = []manifest.Group{{
PublicKey: priv.PublicKey(),
Signature: sig,
}}
m, err := json.Marshal(cs.Manifest)
require.NoError(t, err)
putArgsOnStack := func() { putArgsOnStack := func() {
manifest, err := json.Marshal(cs.Manifest) v.Estack().PushVal(m)
require.NoError(t, err)
ne, err := nef.NewFile(cs.Script)
require.NoError(t, err)
neb, err := ne.Bytes()
require.NoError(t, err)
v.Estack().PushVal(manifest)
v.Estack().PushVal(neb) v.Estack().PushVal(neb)
} }
@ -607,11 +617,36 @@ func TestContractCreate(t *testing.T) {
}) })
ic.Tx = transaction.New(netmode.UnitTestNet, []byte{1}, 0) ic.Tx = transaction.New(netmode.UnitTestNet, []byte{1}, 0)
var sender = util.Uint160{1, 2, 3}
ic.Tx.Signers = append(ic.Tx.Signers, transaction.Signer{Account: sender}) ic.Tx.Signers = append(ic.Tx.Signers, transaction.Signer{Account: sender})
cs.ID = 0 cs.ID = 0
cs.Hash = state.CreateContractHash(sender, cs.Script) cs.Hash = state.CreateContractHash(sender, cs.Script)
t.Run("missing NEF", func(t *testing.T) {
v.Estack().PushVal(m)
v.Estack().PushVal(stackitem.Null{})
require.Error(t, contractCreate(ic))
})
t.Run("missing manifest", func(t *testing.T) {
v.Estack().PushVal(stackitem.Null{})
v.Estack().PushVal(neb)
require.Error(t, contractCreate(ic))
})
t.Run("invalid manifest (empty)", func(t *testing.T) {
v.Estack().PushVal([]byte{})
v.Estack().PushVal(neb)
require.Error(t, contractCreate(ic))
})
t.Run("invalid manifest (group signature)", func(t *testing.T) {
cs.Manifest.Groups[0].Signature = make([]byte, 11)
rawManif, err := json.Marshal(cs.Manifest)
require.NoError(t, err)
v.Estack().PushVal(rawManif)
v.Estack().PushVal(neb)
require.Error(t, contractCreate(ic))
})
cs.Manifest.Groups[0].Signature = sig
t.Run("positive", func(t *testing.T) { t.Run("positive", func(t *testing.T) {
putArgsOnStack() putArgsOnStack()