mirror of
https://github.com/arnaucube/hermez-node.git
synced 2026-02-07 03:16:45 +01:00
Compare commits
5 Commits
feature/up
...
feature/fl
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
6a01c0ac14 | ||
|
|
63151a285c | ||
|
|
52d4197330 | ||
|
|
ea63cba62a | ||
|
|
c7e6267189 |
@@ -2916,7 +2916,7 @@ components:
|
|||||||
example: 101
|
example: 101
|
||||||
l1UserTotalBytes:
|
l1UserTotalBytes:
|
||||||
type: integer
|
type: integer
|
||||||
description: Number of bytes that a L1 user transaction has ([20 bytes] fromEthAddr + [32 bytes] fromBjj-compressed + [6 bytes] fromIdx + [2 bytes] depositAmountFloat16 + [2 bytes] amountFloat16 + [4 bytes] tokenId + [6 bytes] toIdx).
|
description: Number of bytes that a L1 user transaction has ([20 bytes] fromEthAddr + [32 bytes] fromBjj-compressed + [6 bytes] fromIdx + [5 bytes] depositAmountFloat40 + [5 bytes] amountFloat40 + [4 bytes] tokenId + [6 bytes] toIdx).
|
||||||
example: 72
|
example: 72
|
||||||
maxL1UserTx:
|
maxL1UserTx:
|
||||||
type: integer
|
type: integer
|
||||||
|
|||||||
@@ -17,6 +17,11 @@ const AccountCreationAuthMsg = "I authorize this babyjubjub key for hermez rollu
|
|||||||
// EthMsgPrefix is the prefix for message signing at the Ethereum ecosystem
|
// EthMsgPrefix is the prefix for message signing at the Ethereum ecosystem
|
||||||
const EthMsgPrefix = "\x19Ethereum Signed Message:\n"
|
const EthMsgPrefix = "\x19Ethereum Signed Message:\n"
|
||||||
|
|
||||||
|
var (
|
||||||
|
// EmptyEthSignature is an ethereum signature of all zeroes
|
||||||
|
EmptyEthSignature = make([]byte, 65)
|
||||||
|
)
|
||||||
|
|
||||||
// AccountCreationAuth authorizations sent by users to the L2DB, to be used for
|
// AccountCreationAuth authorizations sent by users to the L2DB, to be used for
|
||||||
// account creations when necessary
|
// account creations when necessary
|
||||||
type AccountCreationAuth struct {
|
type AccountCreationAuth struct {
|
||||||
|
|||||||
@@ -24,8 +24,8 @@ const (
|
|||||||
// RollupConstL1CoordinatorTotalBytes [4 bytes] token + [32 bytes] babyjub + [65 bytes] compressedSignature
|
// RollupConstL1CoordinatorTotalBytes [4 bytes] token + [32 bytes] babyjub + [65 bytes] compressedSignature
|
||||||
RollupConstL1CoordinatorTotalBytes = 101
|
RollupConstL1CoordinatorTotalBytes = 101
|
||||||
// RollupConstL1UserTotalBytes [20 bytes] fromEthAddr + [32 bytes] fromBjj-compressed + [6 bytes] fromIdx +
|
// RollupConstL1UserTotalBytes [20 bytes] fromEthAddr + [32 bytes] fromBjj-compressed + [6 bytes] fromIdx +
|
||||||
// [2 bytes] depositAmountFloat16 + [2 bytes] amountFloat16 + [4 bytes] tokenId + [6 bytes] toIdx
|
// [5 bytes] depositAmountFloat40 + [5 bytes] amountFloat40 + [4 bytes] tokenId + [6 bytes] toIdx
|
||||||
RollupConstL1UserTotalBytes = 72
|
RollupConstL1UserTotalBytes = 78
|
||||||
// RollupConstMaxL1UserTx Maximum L1-user transactions allowed to be queued in a batch
|
// RollupConstMaxL1UserTx Maximum L1-user transactions allowed to be queued in a batch
|
||||||
RollupConstMaxL1UserTx = 128
|
RollupConstMaxL1UserTx = 128
|
||||||
// RollupConstMaxL1Tx Maximum L1 transactions allowed to be queued in a batch
|
// RollupConstMaxL1Tx Maximum L1 transactions allowed to be queued in a batch
|
||||||
|
|||||||
@@ -30,6 +30,7 @@ func (f16 Float16) Bytes() []byte {
|
|||||||
|
|
||||||
// Float16FromBytes returns a Float16 from a byte array of 2 bytes.
|
// Float16FromBytes returns a Float16 from a byte array of 2 bytes.
|
||||||
func Float16FromBytes(b []byte) *Float16 {
|
func Float16FromBytes(b []byte) *Float16 {
|
||||||
|
// WARNING b[:2] for a b where len(b)<2 can break
|
||||||
f16 := Float16(binary.BigEndian.Uint16(b[:2]))
|
f16 := Float16(binary.BigEndian.Uint16(b[:2]))
|
||||||
return &f16
|
return &f16
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import (
|
|||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestConversions(t *testing.T) {
|
func TestConversionsFloat16(t *testing.T) {
|
||||||
testVector := map[Float16]string{
|
testVector := map[Float16]string{
|
||||||
0x307B: "123000000",
|
0x307B: "123000000",
|
||||||
0x1DC6: "454500",
|
0x1DC6: "454500",
|
||||||
@@ -32,14 +32,14 @@ func TestConversions(t *testing.T) {
|
|||||||
bi.SetString(testVector[test], 10)
|
bi.SetString(testVector[test], 10)
|
||||||
|
|
||||||
fl, err := NewFloat16(bi)
|
fl, err := NewFloat16(bi)
|
||||||
assert.Equal(t, nil, err)
|
assert.NoError(t, err)
|
||||||
|
|
||||||
fx2 := fl.BigInt()
|
fx2 := fl.BigInt()
|
||||||
assert.Equal(t, fx2.String(), testVector[test])
|
assert.Equal(t, fx2.String(), testVector[test])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFloorFix2Float(t *testing.T) {
|
func TestFloorFix2FloatFloat16(t *testing.T) {
|
||||||
testVector := map[string]Float16{
|
testVector := map[string]Float16{
|
||||||
"87999990000000000": 0x776f,
|
"87999990000000000": 0x776f,
|
||||||
"87950000000000001": 0x776f,
|
"87950000000000001": 0x776f,
|
||||||
@@ -57,10 +57,10 @@ func TestFloorFix2Float(t *testing.T) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestConversionLosses(t *testing.T) {
|
func TestConversionLossesFloat16(t *testing.T) {
|
||||||
a := big.NewInt(1000)
|
a := big.NewInt(1000)
|
||||||
b, err := NewFloat16(a)
|
b, err := NewFloat16(a)
|
||||||
assert.Equal(t, nil, err)
|
assert.NoError(t, err)
|
||||||
c := b.BigInt()
|
c := b.BigInt()
|
||||||
assert.Equal(t, c, a)
|
assert.Equal(t, c, a)
|
||||||
|
|
||||||
|
|||||||
105
common/float40.go
Normal file
105
common/float40.go
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
// Package common Float40 provides methods to work with Hermez custom half
|
||||||
|
// float precision, 40 bits, codification internally called Float40 has been
|
||||||
|
// adopted to encode large integers. This is done in order to save bits when L2
|
||||||
|
// transactions are published.
|
||||||
|
//nolint:gomnd
|
||||||
|
package common
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/binary"
|
||||||
|
"errors"
|
||||||
|
"math/big"
|
||||||
|
|
||||||
|
"github.com/hermeznetwork/tracerr"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// maxFloat40Value is the maximum value that the Float40 can have
|
||||||
|
// (40 bits: maxFloat40Value=2**40-1)
|
||||||
|
maxFloat40Value = 0xffffffffff
|
||||||
|
// Float40BytesLength defines the length of the Float40 values
|
||||||
|
// represented as byte arrays
|
||||||
|
Float40BytesLength = 5
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
// ErrFloat40Overflow is used when a given nonce overflows the maximum
|
||||||
|
// capacity of the Float40 (2**40-1)
|
||||||
|
ErrFloat40Overflow = errors.New("Float40 overflow, max value: 2**40 -1")
|
||||||
|
// ErrFloat40E31 is used when the e > 31 when trying to convert a
|
||||||
|
// *big.Int to Float40
|
||||||
|
ErrFloat40E31 = errors.New("Float40 error, e > 31")
|
||||||
|
// ErrFloat40NotEnoughPrecission is used when the given *big.Int can
|
||||||
|
// not be represented as Float40 due not enough precission
|
||||||
|
ErrFloat40NotEnoughPrecission = errors.New("Float40 error, not enough precission")
|
||||||
|
)
|
||||||
|
|
||||||
|
// Float40 represents a float in a 64 bit format
|
||||||
|
type Float40 uint64
|
||||||
|
|
||||||
|
// Bytes return a byte array of length 5 with the Float40 value encoded in
|
||||||
|
// BigEndian
|
||||||
|
func (f40 Float40) Bytes() ([]byte, error) {
|
||||||
|
if f40 > maxFloat40Value {
|
||||||
|
return []byte{}, tracerr.Wrap(ErrFloat40Overflow)
|
||||||
|
}
|
||||||
|
|
||||||
|
var f40Bytes [8]byte
|
||||||
|
binary.BigEndian.PutUint64(f40Bytes[:], uint64(f40))
|
||||||
|
var b [5]byte
|
||||||
|
copy(b[:], f40Bytes[3:])
|
||||||
|
return b[:], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Float40FromBytes returns a Float40 from a byte array of 5 bytes in Bigendian
|
||||||
|
// representation.
|
||||||
|
func Float40FromBytes(b []byte) Float40 {
|
||||||
|
var f40Bytes [8]byte
|
||||||
|
copy(f40Bytes[3:], b[:])
|
||||||
|
f40 := binary.BigEndian.Uint64(f40Bytes[:])
|
||||||
|
return Float40(f40)
|
||||||
|
}
|
||||||
|
|
||||||
|
// BigInt converts the Float40 to a *big.Int v, where v = m * 10^e, being:
|
||||||
|
// [ e | m ]
|
||||||
|
// [ 5 bits | 35 bits ]
|
||||||
|
func (f40 Float40) BigInt() (*big.Int, error) {
|
||||||
|
// take the 5 used bytes (FF * 5)
|
||||||
|
var f40Uint64 uint64 = uint64(f40) & 0x00_00_00_FF_FF_FF_FF_FF
|
||||||
|
f40Bytes, err := f40.Bytes()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
e := f40Bytes[0] & 0xF8 >> 3 // take first 5 bits
|
||||||
|
m := f40Uint64 & 0x07_FF_FF_FF_FF // take the others 35 bits
|
||||||
|
|
||||||
|
exp := new(big.Int).Exp(big.NewInt(10), big.NewInt(int64(e)), nil)
|
||||||
|
r := new(big.Int).Mul(big.NewInt(int64(m)), exp)
|
||||||
|
return r, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewFloat40 encodes a *big.Int integer as a Float40, returning error in case
|
||||||
|
// of loss during the encoding.
|
||||||
|
func NewFloat40(f *big.Int) (Float40, error) {
|
||||||
|
m := f
|
||||||
|
e := big.NewInt(0)
|
||||||
|
zero := big.NewInt(0)
|
||||||
|
ten := big.NewInt(10)
|
||||||
|
thres := big.NewInt(0x08_00_00_00_00)
|
||||||
|
for bytes.Equal(zero.Bytes(), new(big.Int).Mod(m, ten).Bytes()) &&
|
||||||
|
!bytes.Equal(zero.Bytes(), new(big.Int).Div(m, thres).Bytes()) {
|
||||||
|
m = new(big.Int).Div(m, ten)
|
||||||
|
e = new(big.Int).Add(e, big.NewInt(1))
|
||||||
|
}
|
||||||
|
if e.Int64() > 31 {
|
||||||
|
return 0, ErrFloat40E31
|
||||||
|
}
|
||||||
|
if !bytes.Equal(zero.Bytes(), new(big.Int).Div(m, thres).Bytes()) {
|
||||||
|
return 0, ErrFloat40NotEnoughPrecission
|
||||||
|
}
|
||||||
|
r := new(big.Int).Add(m,
|
||||||
|
new(big.Int).Mul(e, thres))
|
||||||
|
return Float40(r.Uint64()), nil
|
||||||
|
}
|
||||||
95
common/float40_test.go
Normal file
95
common/float40_test.go
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
package common
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math/big"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestConversionsFloat40(t *testing.T) {
|
||||||
|
testVector := map[Float40]string{
|
||||||
|
6*0x800000000 + 123: "123000000",
|
||||||
|
2*0x800000000 + 4545: "454500",
|
||||||
|
30*0x800000000 + 10235: "10235000000000000000000000000000000",
|
||||||
|
0x000000000: "0",
|
||||||
|
0x800000000: "0",
|
||||||
|
0x0001: "1",
|
||||||
|
0x0401: "1025",
|
||||||
|
0x800000000 + 1: "10",
|
||||||
|
0xFFFFFFFFFF: "343597383670000000000000000000000000000000",
|
||||||
|
}
|
||||||
|
|
||||||
|
for test := range testVector {
|
||||||
|
fix, err := test.BigInt()
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, fix.String(), testVector[test])
|
||||||
|
|
||||||
|
bi, ok := new(big.Int).SetString(testVector[test], 10)
|
||||||
|
require.True(t, ok)
|
||||||
|
|
||||||
|
fl, err := NewFloat40(bi)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
|
||||||
|
fx2, err := fl.BigInt()
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, fx2.String(), testVector[test])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestExpectError(t *testing.T) {
|
||||||
|
testVector := map[string]error{
|
||||||
|
"9922334455000000000000000000000000000000": nil,
|
||||||
|
"9922334455000000000000000000000000000001": ErrFloat40NotEnoughPrecission,
|
||||||
|
"9922334454999999999999999999999999999999": ErrFloat40NotEnoughPrecission,
|
||||||
|
"42949672950000000000000000000000000000000": nil,
|
||||||
|
"99223344556573838487575": ErrFloat40NotEnoughPrecission,
|
||||||
|
"992233445500000000000000000000000000000000": ErrFloat40E31,
|
||||||
|
"343597383670000000000000000000000000000000": nil,
|
||||||
|
"343597383680000000000000000000000000000000": ErrFloat40NotEnoughPrecission,
|
||||||
|
"343597383690000000000000000000000000000000": ErrFloat40NotEnoughPrecission,
|
||||||
|
"343597383700000000000000000000000000000000": ErrFloat40E31,
|
||||||
|
}
|
||||||
|
for test := range testVector {
|
||||||
|
bi, ok := new(big.Int).SetString(test, 10)
|
||||||
|
require.True(t, ok)
|
||||||
|
_, err := NewFloat40(bi)
|
||||||
|
assert.Equal(t, testVector[test], err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkFloat40(b *testing.B) {
|
||||||
|
newBigInt := func(s string) *big.Int {
|
||||||
|
bigInt, ok := new(big.Int).SetString(s, 10)
|
||||||
|
if !ok {
|
||||||
|
panic("Can not convert string to *big.Int")
|
||||||
|
}
|
||||||
|
return bigInt
|
||||||
|
}
|
||||||
|
type pair struct {
|
||||||
|
Float40 Float40
|
||||||
|
BigInt *big.Int
|
||||||
|
}
|
||||||
|
testVector := []pair{
|
||||||
|
{6*0x800000000 + 123, newBigInt("123000000")},
|
||||||
|
{2*0x800000000 + 4545, newBigInt("454500")},
|
||||||
|
{30*0x800000000 + 10235, newBigInt("10235000000000000000000000000000000")},
|
||||||
|
{0x000000000, newBigInt("0")},
|
||||||
|
{0x800000000, newBigInt("0")},
|
||||||
|
{0x0001, newBigInt("1")},
|
||||||
|
{0x0401, newBigInt("1025")},
|
||||||
|
{0x800000000 + 1, newBigInt("10")},
|
||||||
|
{0xFFFFFFFFFF, newBigInt("343597383670000000000000000000000000000000")},
|
||||||
|
}
|
||||||
|
b.Run("NewFloat40()", func(b *testing.B) {
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
_, _ = NewFloat40(testVector[i%len(testVector)].BigInt)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
b.Run("Float40.BigInt()", func(b *testing.B) {
|
||||||
|
for i := 0; i < b.N; i++ {
|
||||||
|
_, _ = testVector[i%len(testVector)].Float40.BigInt()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -11,18 +11,11 @@ import (
|
|||||||
"github.com/iden3/go-iden3-crypto/babyjub"
|
"github.com/iden3/go-iden3-crypto/babyjub"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
|
||||||
// L1UserTxBytesLen is the length of the byte array that represents the L1Tx
|
|
||||||
L1UserTxBytesLen = 72
|
|
||||||
// L1CoordinatorTxBytesLen is the length of the byte array that represents the L1CoordinatorTx
|
|
||||||
L1CoordinatorTxBytesLen = 101
|
|
||||||
)
|
|
||||||
|
|
||||||
// L1Tx is a struct that represents a L1 tx
|
// L1Tx is a struct that represents a L1 tx
|
||||||
type L1Tx struct {
|
type L1Tx struct {
|
||||||
// Stored in DB: mandatory fileds
|
// Stored in DB: mandatory fileds
|
||||||
|
|
||||||
// TxID (12 bytes) for L1Tx is:
|
// TxID (32 bytes) for L1Tx is the Keccak256 (ethereum) hash of:
|
||||||
// bytes: | 1 | 8 | 2 | 1 |
|
// bytes: | 1 | 8 | 2 | 1 |
|
||||||
// values: | type | ToForgeL1TxsNum | Position | 0 (padding) |
|
// values: | type | ToForgeL1TxsNum | Position | 0 (padding) |
|
||||||
// where type:
|
// where type:
|
||||||
@@ -179,45 +172,38 @@ func (tx L1Tx) Tx() Tx {
|
|||||||
// [ 8 bits ] empty (userFee) // 1 byte
|
// [ 8 bits ] empty (userFee) // 1 byte
|
||||||
// [ 40 bits ] empty (nonce) // 5 bytes
|
// [ 40 bits ] empty (nonce) // 5 bytes
|
||||||
// [ 32 bits ] tokenID // 4 bytes
|
// [ 32 bits ] tokenID // 4 bytes
|
||||||
// [ 16 bits ] amountFloat16 // 2 bytes
|
|
||||||
// [ 48 bits ] toIdx // 6 bytes
|
// [ 48 bits ] toIdx // 6 bytes
|
||||||
// [ 48 bits ] fromIdx // 6 bytes
|
// [ 48 bits ] fromIdx // 6 bytes
|
||||||
// [ 16 bits ] chainId // 2 bytes
|
// [ 16 bits ] chainId // 2 bytes
|
||||||
// [ 32 bits ] empty (signatureConstant) // 4 bytes
|
// [ 32 bits ] empty (signatureConstant) // 4 bytes
|
||||||
// Total bits compressed data: 241 bits // 31 bytes in *big.Int representation
|
// Total bits compressed data: 225 bits // 29 bytes in *big.Int representation
|
||||||
func (tx L1Tx) TxCompressedData(chainID uint16) (*big.Int, error) {
|
func (tx L1Tx) TxCompressedData(chainID uint16) (*big.Int, error) {
|
||||||
amountFloat16, err := NewFloat16(tx.Amount)
|
var b [29]byte
|
||||||
if err != nil {
|
|
||||||
return nil, tracerr.Wrap(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
var b [31]byte
|
|
||||||
// b[0:7] empty: no ToBJJSign, no fee, no nonce
|
// b[0:7] empty: no ToBJJSign, no fee, no nonce
|
||||||
copy(b[7:11], tx.TokenID.Bytes())
|
copy(b[7:11], tx.TokenID.Bytes())
|
||||||
copy(b[11:13], amountFloat16.Bytes())
|
|
||||||
toIdxBytes, err := tx.ToIdx.Bytes()
|
toIdxBytes, err := tx.ToIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[13:19], toIdxBytes[:])
|
copy(b[11:17], toIdxBytes[:])
|
||||||
fromIdxBytes, err := tx.FromIdx.Bytes()
|
fromIdxBytes, err := tx.FromIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[19:25], fromIdxBytes[:])
|
copy(b[17:23], fromIdxBytes[:])
|
||||||
binary.BigEndian.PutUint16(b[25:27], chainID)
|
binary.BigEndian.PutUint16(b[23:25], chainID)
|
||||||
copy(b[27:31], SignatureConstantBytes[:])
|
copy(b[25:29], SignatureConstantBytes[:])
|
||||||
|
|
||||||
bi := new(big.Int).SetBytes(b[:])
|
bi := new(big.Int).SetBytes(b[:])
|
||||||
return bi, nil
|
return bi, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// BytesDataAvailability encodes a L1Tx into []byte for the Data Availability
|
// BytesDataAvailability encodes a L1Tx into []byte for the Data Availability
|
||||||
// [ fromIdx | toIdx | amountFloat16 | Fee ]
|
// [ fromIdx | toIdx | amountFloat40 | Fee ]
|
||||||
func (tx *L1Tx) BytesDataAvailability(nLevels uint32) ([]byte, error) {
|
func (tx *L1Tx) BytesDataAvailability(nLevels uint32) ([]byte, error) {
|
||||||
idxLen := nLevels / 8 //nolint:gomnd
|
idxLen := nLevels / 8 //nolint:gomnd
|
||||||
|
|
||||||
b := make([]byte, ((nLevels*2)+16+8)/8) //nolint:gomnd
|
b := make([]byte, ((nLevels*2)+40+8)/8) //nolint:gomnd
|
||||||
|
|
||||||
fromIdxBytes, err := tx.FromIdx.Bytes()
|
fromIdxBytes, err := tx.FromIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -231,13 +217,17 @@ func (tx *L1Tx) BytesDataAvailability(nLevels uint32) ([]byte, error) {
|
|||||||
copy(b[idxLen:idxLen*2], toIdxBytes[6-idxLen:])
|
copy(b[idxLen:idxLen*2], toIdxBytes[6-idxLen:])
|
||||||
|
|
||||||
if tx.EffectiveAmount != nil {
|
if tx.EffectiveAmount != nil {
|
||||||
amountFloat16, err := NewFloat16(tx.EffectiveAmount)
|
amountFloat40, err := NewFloat40(tx.EffectiveAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[idxLen*2:idxLen*2+2], amountFloat16.Bytes())
|
amountFloat40Bytes, err := amountFloat40.Bytes()
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
// fee = 0 (as is L1Tx) b[10:11]
|
copy(b[idxLen*2:idxLen*2+Float40BytesLength], amountFloat40Bytes)
|
||||||
|
}
|
||||||
|
// fee = 0 (as is L1Tx)
|
||||||
return b[:], nil
|
return b[:], nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -247,7 +237,7 @@ func L1TxFromDataAvailability(b []byte, nLevels uint32) (*L1Tx, error) {
|
|||||||
|
|
||||||
fromIdxBytes := b[0:idxLen]
|
fromIdxBytes := b[0:idxLen]
|
||||||
toIdxBytes := b[idxLen : idxLen*2]
|
toIdxBytes := b[idxLen : idxLen*2]
|
||||||
amountBytes := b[idxLen*2 : idxLen*2+2]
|
amountBytes := b[idxLen*2 : idxLen*2+Float40BytesLength]
|
||||||
|
|
||||||
l1tx := L1Tx{}
|
l1tx := L1Tx{}
|
||||||
fromIdx, err := IdxFromBytes(ethCommon.LeftPadBytes(fromIdxBytes, 6))
|
fromIdx, err := IdxFromBytes(ethCommon.LeftPadBytes(fromIdxBytes, 6))
|
||||||
@@ -260,8 +250,8 @@ func L1TxFromDataAvailability(b []byte, nLevels uint32) (*L1Tx, error) {
|
|||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
l1tx.ToIdx = toIdx
|
l1tx.ToIdx = toIdx
|
||||||
l1tx.EffectiveAmount = Float16FromBytes(amountBytes).BigInt()
|
l1tx.EffectiveAmount, err = Float40FromBytes(amountBytes).BigInt()
|
||||||
return &l1tx, nil
|
return &l1tx, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// BytesGeneric returns the generic representation of a L1Tx. This method is
|
// BytesGeneric returns the generic representation of a L1Tx. This method is
|
||||||
@@ -269,7 +259,7 @@ func L1TxFromDataAvailability(b []byte, nLevels uint32) (*L1Tx, error) {
|
|||||||
// the L1TxData for the ZKInputs (at the HashGlobalInputs), using this method
|
// the L1TxData for the ZKInputs (at the HashGlobalInputs), using this method
|
||||||
// for L1CoordinatorTxs & L1UserTxs (for the ZKInputs case).
|
// for L1CoordinatorTxs & L1UserTxs (for the ZKInputs case).
|
||||||
func (tx *L1Tx) BytesGeneric() ([]byte, error) {
|
func (tx *L1Tx) BytesGeneric() ([]byte, error) {
|
||||||
var b [L1UserTxBytesLen]byte
|
var b [RollupConstL1UserTotalBytes]byte
|
||||||
copy(b[0:20], tx.FromEthAddr.Bytes())
|
copy(b[0:20], tx.FromEthAddr.Bytes())
|
||||||
if tx.FromBJJ != EmptyBJJComp {
|
if tx.FromBJJ != EmptyBJJComp {
|
||||||
pkCompL := tx.FromBJJ
|
pkCompL := tx.FromBJJ
|
||||||
@@ -281,22 +271,33 @@ func (tx *L1Tx) BytesGeneric() ([]byte, error) {
|
|||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[52:58], fromIdxBytes[:])
|
copy(b[52:58], fromIdxBytes[:])
|
||||||
depositAmountFloat16, err := NewFloat16(tx.DepositAmount)
|
|
||||||
|
depositAmountFloat40, err := NewFloat40(tx.DepositAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[58:60], depositAmountFloat16.Bytes())
|
depositAmountFloat40Bytes, err := depositAmountFloat40.Bytes()
|
||||||
amountFloat16, err := NewFloat16(tx.Amount)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[60:62], amountFloat16.Bytes())
|
copy(b[58:63], depositAmountFloat40Bytes)
|
||||||
copy(b[62:66], tx.TokenID.Bytes())
|
|
||||||
|
amountFloat40, err := NewFloat40(tx.Amount)
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
amountFloat40Bytes, err := amountFloat40.Bytes()
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
copy(b[63:68], amountFloat40Bytes)
|
||||||
|
|
||||||
|
copy(b[68:72], tx.TokenID.Bytes())
|
||||||
toIdxBytes, err := tx.ToIdx.Bytes()
|
toIdxBytes, err := tx.ToIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[66:72], toIdxBytes[:])
|
copy(b[72:78], toIdxBytes[:])
|
||||||
return b[:], nil
|
return b[:], nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -313,7 +314,7 @@ func (tx *L1Tx) BytesCoordinatorTx(compressedSignatureBytes []byte) ([]byte, err
|
|||||||
if tx.UserOrigin {
|
if tx.UserOrigin {
|
||||||
return nil, tracerr.Wrap(fmt.Errorf("Can not calculate BytesCoordinatorTx() for a L1UserTx"))
|
return nil, tracerr.Wrap(fmt.Errorf("Can not calculate BytesCoordinatorTx() for a L1UserTx"))
|
||||||
}
|
}
|
||||||
var b [L1CoordinatorTxBytesLen]byte
|
var b [RollupConstL1CoordinatorTotalBytes]byte
|
||||||
v := compressedSignatureBytes[64]
|
v := compressedSignatureBytes[64]
|
||||||
s := compressedSignatureBytes[32:64]
|
s := compressedSignatureBytes[32:64]
|
||||||
r := compressedSignatureBytes[0:32]
|
r := compressedSignatureBytes[0:32]
|
||||||
@@ -329,7 +330,7 @@ func (tx *L1Tx) BytesCoordinatorTx(compressedSignatureBytes []byte) ([]byte, err
|
|||||||
|
|
||||||
// L1UserTxFromBytes decodes a L1Tx from []byte
|
// L1UserTxFromBytes decodes a L1Tx from []byte
|
||||||
func L1UserTxFromBytes(b []byte) (*L1Tx, error) {
|
func L1UserTxFromBytes(b []byte) (*L1Tx, error) {
|
||||||
if len(b) != L1UserTxBytesLen {
|
if len(b) != RollupConstL1UserTotalBytes {
|
||||||
return nil, tracerr.Wrap(fmt.Errorf("Can not parse L1Tx bytes, expected length %d, current: %d", 68, len(b)))
|
return nil, tracerr.Wrap(fmt.Errorf("Can not parse L1Tx bytes, expected length %d, current: %d", 68, len(b)))
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -347,13 +348,19 @@ func L1UserTxFromBytes(b []byte) (*L1Tx, error) {
|
|||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
tx.FromIdx = fromIdx
|
tx.FromIdx = fromIdx
|
||||||
tx.DepositAmount = Float16FromBytes(b[58:60]).BigInt()
|
tx.DepositAmount, err = Float40FromBytes(b[58:63]).BigInt()
|
||||||
tx.Amount = Float16FromBytes(b[60:62]).BigInt()
|
|
||||||
tx.TokenID, err = TokenIDFromBytes(b[62:66])
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
tx.ToIdx, err = IdxFromBytes(b[66:72])
|
tx.Amount, err = Float40FromBytes(b[63:68]).BigInt()
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
tx.TokenID, err = TokenIDFromBytes(b[68:72])
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
tx.ToIdx, err = IdxFromBytes(b[72:78])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
@@ -368,7 +375,7 @@ func signHash(data []byte) []byte {
|
|||||||
|
|
||||||
// L1CoordinatorTxFromBytes decodes a L1Tx from []byte
|
// L1CoordinatorTxFromBytes decodes a L1Tx from []byte
|
||||||
func L1CoordinatorTxFromBytes(b []byte, chainID *big.Int, hermezAddress ethCommon.Address) (*L1Tx, error) {
|
func L1CoordinatorTxFromBytes(b []byte, chainID *big.Int, hermezAddress ethCommon.Address) (*L1Tx, error) {
|
||||||
if len(b) != L1CoordinatorTxBytesLen {
|
if len(b) != RollupConstL1CoordinatorTotalBytes {
|
||||||
return nil, tracerr.Wrap(fmt.Errorf("Can not parse L1CoordinatorTx bytes, expected length %d, current: %d", 101, len(b)))
|
return nil, tracerr.Wrap(fmt.Errorf("Can not parse L1CoordinatorTx bytes, expected length %d, current: %d", 101, len(b)))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -50,64 +50,110 @@ func TestNewL1CoordinatorTx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestL1TxCompressedData(t *testing.T) {
|
func TestL1TxCompressedData(t *testing.T) {
|
||||||
|
// test vectors values generated from javascript implementation (using
|
||||||
|
// PoolL2Tx values)
|
||||||
|
amount, ok := new(big.Int).SetString("343597383670000000000000000000000000000000", 10)
|
||||||
|
require.True(t, ok)
|
||||||
tx := L1Tx{
|
tx := L1Tx{
|
||||||
FromIdx: 2,
|
FromIdx: (1 << 48) - 1,
|
||||||
ToIdx: 3,
|
ToIdx: (1 << 48) - 1,
|
||||||
Amount: big.NewInt(4),
|
Amount: amount,
|
||||||
TokenID: 5,
|
TokenID: (1 << 32) - 1,
|
||||||
}
|
}
|
||||||
chainID := uint16(0)
|
txCompressedData, err := tx.TxCompressedData(uint16((1 << 16) - 1))
|
||||||
txCompressedData, err := tx.TxCompressedData(chainID)
|
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
|
expectedStr := "ffffffffffffffffffffffffffffffffffffc60be60f"
|
||||||
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
|
|
||||||
// test vector value generated from javascript implementation
|
tx = L1Tx{
|
||||||
expectedStr := "7307597389635308713748674793997299267459594577423"
|
FromIdx: 0,
|
||||||
assert.Equal(t, expectedStr, txCompressedData.String())
|
ToIdx: 0,
|
||||||
assert.Equal(t, "0500040000000000030000000000020000c60be60f", hex.EncodeToString(txCompressedData.Bytes()))
|
Amount: big.NewInt(0),
|
||||||
|
TokenID: 0,
|
||||||
|
}
|
||||||
|
txCompressedData, err = tx.TxCompressedData(uint16(0))
|
||||||
|
assert.NoError(t, err)
|
||||||
|
expectedStr = "c60be60f"
|
||||||
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
|
|
||||||
|
amount, ok = new(big.Int).SetString("63000000000000000", 10)
|
||||||
|
require.True(t, ok)
|
||||||
|
tx = L1Tx{
|
||||||
|
FromIdx: 324,
|
||||||
|
ToIdx: 256,
|
||||||
|
Amount: amount,
|
||||||
|
TokenID: 123,
|
||||||
|
}
|
||||||
|
txCompressedData, err = tx.TxCompressedData(uint16(1))
|
||||||
|
assert.NoError(t, err)
|
||||||
|
expectedStr = "7b0000000001000000000001440001c60be60f"
|
||||||
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
|
|
||||||
|
tx = L1Tx{
|
||||||
|
FromIdx: 1,
|
||||||
|
ToIdx: 2,
|
||||||
|
TokenID: 3,
|
||||||
|
}
|
||||||
|
txCompressedData, err = tx.TxCompressedData(uint16(0))
|
||||||
|
assert.NoError(t, err)
|
||||||
|
expectedStr = "030000000000020000000000010000c60be60f"
|
||||||
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestBytesDataAvailability(t *testing.T) {
|
func TestBytesDataAvailability(t *testing.T) {
|
||||||
|
// test vectors values generated from javascript implementation
|
||||||
|
amount, ok := new(big.Int).SetString("343597383670000000000000000000000000000000", 10)
|
||||||
|
require.True(t, ok)
|
||||||
tx := L1Tx{
|
tx := L1Tx{
|
||||||
FromIdx: 2,
|
ToIdx: (1 << 16) - 1,
|
||||||
ToIdx: 3,
|
FromIdx: (1 << 16) - 1,
|
||||||
Amount: big.NewInt(4),
|
EffectiveAmount: amount,
|
||||||
TokenID: 5,
|
|
||||||
}
|
}
|
||||||
txCompressedData, err := tx.BytesDataAvailability(32)
|
txCompressedData, err := tx.BytesDataAvailability(16)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0000000200000003000000", hex.EncodeToString(txCompressedData))
|
assert.Equal(t, "ffffffffffffffffff00", hex.EncodeToString(txCompressedData))
|
||||||
|
l1tx, err := L1TxFromDataAvailability(txCompressedData, 16)
|
||||||
tx = L1Tx{
|
|
||||||
FromIdx: 2,
|
|
||||||
ToIdx: 3,
|
|
||||||
EffectiveAmount: big.NewInt(4),
|
|
||||||
TokenID: 5,
|
|
||||||
}
|
|
||||||
txCompressedData, err = tx.BytesDataAvailability(32)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Equal(t, "0000000200000003000400", hex.EncodeToString(txCompressedData))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestL1TxFromDataAvailability(t *testing.T) {
|
|
||||||
tx := L1Tx{
|
|
||||||
FromIdx: 2,
|
|
||||||
ToIdx: 3,
|
|
||||||
Amount: big.NewInt(4),
|
|
||||||
}
|
|
||||||
txCompressedData, err := tx.BytesDataAvailability(32)
|
|
||||||
assert.NoError(t, err)
|
|
||||||
l1tx, err := L1TxFromDataAvailability(txCompressedData, 32)
|
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, tx.FromIdx, l1tx.FromIdx)
|
assert.Equal(t, tx.FromIdx, l1tx.FromIdx)
|
||||||
assert.Equal(t, tx.ToIdx, l1tx.ToIdx)
|
assert.Equal(t, tx.ToIdx, l1tx.ToIdx)
|
||||||
|
assert.Equal(t, tx.EffectiveAmount, l1tx.EffectiveAmount)
|
||||||
|
|
||||||
tx = L1Tx{
|
tx = L1Tx{
|
||||||
FromIdx: 2,
|
ToIdx: (1 << 32) - 1,
|
||||||
ToIdx: 3,
|
FromIdx: (1 << 32) - 1,
|
||||||
EffectiveAmount: big.NewInt(4),
|
EffectiveAmount: amount,
|
||||||
}
|
}
|
||||||
txCompressedData, err = tx.BytesDataAvailability(32)
|
txCompressedData, err = tx.BytesDataAvailability(32)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, "ffffffffffffffffffffffffff00", hex.EncodeToString(txCompressedData))
|
||||||
|
l1tx, err = L1TxFromDataAvailability(txCompressedData, 32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, tx.FromIdx, l1tx.FromIdx)
|
||||||
|
assert.Equal(t, tx.ToIdx, l1tx.ToIdx)
|
||||||
|
assert.Equal(t, tx.EffectiveAmount, l1tx.EffectiveAmount)
|
||||||
|
|
||||||
|
tx = L1Tx{
|
||||||
|
ToIdx: 0,
|
||||||
|
FromIdx: 0,
|
||||||
|
EffectiveAmount: big.NewInt(0),
|
||||||
|
}
|
||||||
|
txCompressedData, err = tx.BytesDataAvailability(32)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, "0000000000000000000000000000", hex.EncodeToString(txCompressedData))
|
||||||
|
l1tx, err = L1TxFromDataAvailability(txCompressedData, 32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, tx.FromIdx, l1tx.FromIdx)
|
||||||
|
assert.Equal(t, tx.ToIdx, l1tx.ToIdx)
|
||||||
|
assert.Equal(t, tx.EffectiveAmount, l1tx.EffectiveAmount)
|
||||||
|
|
||||||
|
tx = L1Tx{
|
||||||
|
ToIdx: 635,
|
||||||
|
FromIdx: 296,
|
||||||
|
EffectiveAmount: big.NewInt(1000000000000000000),
|
||||||
|
}
|
||||||
|
txCompressedData, err = tx.BytesDataAvailability(32)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, "000001280000027b42540be40000", hex.EncodeToString(txCompressedData))
|
||||||
l1tx, err = L1TxFromDataAvailability(txCompressedData, 32)
|
l1tx, err = L1TxFromDataAvailability(txCompressedData, 32)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, tx.FromIdx, l1tx.FromIdx)
|
assert.Equal(t, tx.FromIdx, l1tx.FromIdx)
|
||||||
@@ -172,12 +218,10 @@ func TestL1TxByteParsersCompatibility(t *testing.T) {
|
|||||||
UserOrigin: true,
|
UserOrigin: true,
|
||||||
}
|
}
|
||||||
|
|
||||||
expected, err := utils.HexDecode("85dab5b9e2e361d0c208d77be90efcc0439b0a530dd02deb2c81068e7a0f7e327df80b4ab79ee1f41a7def613e73a20c32eece5a000001c638db8be880f00020039c0000053cb88d")
|
|
||||||
require.NoError(t, err)
|
|
||||||
|
|
||||||
encodedData, err := l1Tx.BytesUser()
|
encodedData, err := l1Tx.BytesUser()
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, expected, encodedData)
|
expected := "85dab5b9e2e361d0c208d77be90efcc0439b0a530dd02deb2c81068e7a0f7e327df80b4ab79ee1f41a7def613e73a20c32eece5a000001c638db52540be400459682f0000020039c0000053cb88d"
|
||||||
|
assert.Equal(t, expected, hex.EncodeToString(encodedData))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestL1CoordinatorTxByteParsers(t *testing.T) {
|
func TestL1CoordinatorTxByteParsers(t *testing.T) {
|
||||||
|
|||||||
@@ -89,11 +89,15 @@ func (tx L2Tx) CalculateTxID() ([TxIDLen]byte, error) {
|
|||||||
// TokenID
|
// TokenID
|
||||||
b = append(b, tx.TokenID.Bytes()[:]...)
|
b = append(b, tx.TokenID.Bytes()[:]...)
|
||||||
// Amount
|
// Amount
|
||||||
amountFloat16, err := NewFloat16(tx.Amount)
|
amountFloat40, err := NewFloat40(tx.Amount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return txID, tracerr.Wrap(fmt.Errorf("%s: %d", err, tx.Amount))
|
return txID, tracerr.Wrap(fmt.Errorf("%s: %d", err, tx.Amount))
|
||||||
}
|
}
|
||||||
b = append(b, amountFloat16.Bytes()...)
|
amountFloat40Bytes, err := amountFloat40.Bytes()
|
||||||
|
if err != nil {
|
||||||
|
return txID, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
b = append(b, amountFloat40Bytes...)
|
||||||
// Nonce
|
// Nonce
|
||||||
nonceBytes, err := tx.Nonce.Bytes()
|
nonceBytes, err := tx.Nonce.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -170,11 +174,11 @@ func TxIDsFromL2Txs(txs []L2Tx) []TxID {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// BytesDataAvailability encodes a L2Tx into []byte for the Data Availability
|
// BytesDataAvailability encodes a L2Tx into []byte for the Data Availability
|
||||||
// [ fromIdx | toIdx | amountFloat16 | Fee ]
|
// [ fromIdx | toIdx | amountFloat40 | Fee ]
|
||||||
func (tx L2Tx) BytesDataAvailability(nLevels uint32) ([]byte, error) {
|
func (tx L2Tx) BytesDataAvailability(nLevels uint32) ([]byte, error) {
|
||||||
idxLen := nLevels / 8 //nolint:gomnd
|
idxLen := nLevels / 8 //nolint:gomnd
|
||||||
|
|
||||||
b := make([]byte, ((nLevels*2)+16+8)/8) //nolint:gomnd
|
b := make([]byte, ((nLevels*2)+40+8)/8) //nolint:gomnd
|
||||||
|
|
||||||
fromIdxBytes, err := tx.FromIdx.Bytes()
|
fromIdxBytes, err := tx.FromIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -188,13 +192,16 @@ func (tx L2Tx) BytesDataAvailability(nLevels uint32) ([]byte, error) {
|
|||||||
}
|
}
|
||||||
copy(b[idxLen:idxLen*2], toIdxBytes[6-idxLen:])
|
copy(b[idxLen:idxLen*2], toIdxBytes[6-idxLen:])
|
||||||
|
|
||||||
amountFloat16, err := NewFloat16(tx.Amount)
|
amountFloat40, err := NewFloat40(tx.Amount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
|
amountFloat40Bytes, err := amountFloat40.Bytes()
|
||||||
copy(b[idxLen*2:idxLen*2+2], amountFloat16.Bytes())
|
if err != nil {
|
||||||
b[idxLen*2+2] = byte(tx.Fee)
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
copy(b[idxLen*2:idxLen*2+Float40BytesLength], amountFloat40Bytes)
|
||||||
|
b[idxLen*2+Float40BytesLength] = byte(tx.Fee)
|
||||||
|
|
||||||
return b[:], nil
|
return b[:], nil
|
||||||
}
|
}
|
||||||
@@ -219,7 +226,10 @@ func L2TxFromBytesDataAvailability(b []byte, nLevels int) (*L2Tx, error) {
|
|||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
tx.Amount = Float16FromBytes(b[idxLen*2 : idxLen*2+2]).BigInt()
|
tx.Amount, err = Float40FromBytes(b[idxLen*2 : idxLen*2+Float40BytesLength]).BigInt()
|
||||||
tx.Fee = FeeSelector(b[idxLen*2+2])
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
tx.Fee = FeeSelector(b[idxLen*2+Float40BytesLength])
|
||||||
return tx, nil
|
return tx, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ func TestNewL2Tx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
l2Tx, err := NewL2Tx(l2Tx)
|
l2Tx, err := NewL2Tx(l2Tx)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0x02fb52b5d0b9ef2626c11701bb751b2720c76d59946b9a48146ac153bb6e63bf6a", l2Tx.TxID.String())
|
assert.Equal(t, "0x022669acda59b827d20ef5354a3eebd1dffb3972b0a6bf89d18bfd2efa0ab9f41e", l2Tx.TxID.String())
|
||||||
|
|
||||||
l2Tx = &L2Tx{
|
l2Tx = &L2Tx{
|
||||||
FromIdx: 87654,
|
FromIdx: 87654,
|
||||||
@@ -30,7 +30,7 @@ func TestNewL2Tx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
l2Tx, err = NewL2Tx(l2Tx)
|
l2Tx, err = NewL2Tx(l2Tx)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0x0276114a8f666fa1ff7dbf34b4a9da577808dc501e3b2760d01fe3ef5473f5737f", l2Tx.TxID.String())
|
assert.Equal(t, "0x029e7499a830f8f5eb17c07da48cf91415710f1bcbe0169d363ff91e81faf92fc2", l2Tx.TxID.String())
|
||||||
|
|
||||||
l2Tx = &L2Tx{
|
l2Tx = &L2Tx{
|
||||||
FromIdx: 87654,
|
FromIdx: 87654,
|
||||||
@@ -42,7 +42,7 @@ func TestNewL2Tx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
l2Tx, err = NewL2Tx(l2Tx)
|
l2Tx, err = NewL2Tx(l2Tx)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0x025afb63126d3067f61f633d13e5a51da0551af3a4567a9af2db5321ed04214ff4", l2Tx.TxID.String())
|
assert.Equal(t, "0x0255c70ed20e1b8935232e1b9c5884dbcc88a6e1a3454d24f2d77252eb2bb0b64e", l2Tx.TxID.String())
|
||||||
|
|
||||||
l2Tx = &L2Tx{
|
l2Tx = &L2Tx{
|
||||||
FromIdx: 87654,
|
FromIdx: 87654,
|
||||||
@@ -54,7 +54,7 @@ func TestNewL2Tx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
l2Tx, err = NewL2Tx(l2Tx)
|
l2Tx, err = NewL2Tx(l2Tx)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0x02cf390157041c3b1b59f0aaed4da464f0d0d48f1d026e46fd89c7fe1e5aed7fcf", l2Tx.TxID.String())
|
assert.Equal(t, "0x0206b372f967061d1148bbcff679de38120e075141a80a07326d0f514c2efc6ca9", l2Tx.TxID.String())
|
||||||
|
|
||||||
l2Tx = &L2Tx{
|
l2Tx = &L2Tx{
|
||||||
FromIdx: 1,
|
FromIdx: 1,
|
||||||
@@ -66,7 +66,7 @@ func TestNewL2Tx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
l2Tx, err = NewL2Tx(l2Tx)
|
l2Tx, err = NewL2Tx(l2Tx)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0x020ec18eaae67fcd545998841a9c4be09ee3083e12db6ae5e5213a2ecaaa52d5cf", l2Tx.TxID.String())
|
assert.Equal(t, "0x0236f7ea5bccf78ba60baf56c058d235a844f9b09259fd0efa4f5f72a7d4a26618", l2Tx.TxID.String())
|
||||||
|
|
||||||
l2Tx = &L2Tx{
|
l2Tx = &L2Tx{
|
||||||
FromIdx: 999,
|
FromIdx: 999,
|
||||||
@@ -78,7 +78,7 @@ func TestNewL2Tx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
l2Tx, err = NewL2Tx(l2Tx)
|
l2Tx, err = NewL2Tx(l2Tx)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0x02f036223e79fac776de107f50822552cc964ee9fc4caa304613285f6976bcc940", l2Tx.TxID.String())
|
assert.Equal(t, "0x02ac122f5b709ce190129fecbbe35bfd30c70e6433dbd85a8eb743d110906a1dc1", l2Tx.TxID.String())
|
||||||
|
|
||||||
l2Tx = &L2Tx{
|
l2Tx = &L2Tx{
|
||||||
FromIdx: 4444,
|
FromIdx: 4444,
|
||||||
@@ -90,25 +90,85 @@ func TestNewL2Tx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
l2Tx, err = NewL2Tx(l2Tx)
|
l2Tx, err = NewL2Tx(l2Tx)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0x029c8aef9ef24531e4cf84e78cbab1018ba1626a5a10afb6b7c356be1b5c28e92c", l2Tx.TxID.String())
|
assert.Equal(t, "0x02c674951a81881b7bc50db3b9e5efd97ac88550c7426ac548720e5057cfba515a", l2Tx.TxID.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestL2TxByteParsers(t *testing.T) {
|
func TestL2TxByteParsers(t *testing.T) {
|
||||||
amount := new(big.Int)
|
// test vectors values generated from javascript implementation
|
||||||
amount.SetString("79000000", 10)
|
amount, ok := new(big.Int).SetString("343597383670000000000000000000000000000000", 10)
|
||||||
|
require.True(t, ok)
|
||||||
l2Tx := &L2Tx{
|
l2Tx := &L2Tx{
|
||||||
ToIdx: 256,
|
ToIdx: (1 << 16) - 1,
|
||||||
|
FromIdx: (1 << 16) - 1,
|
||||||
Amount: amount,
|
Amount: amount,
|
||||||
FromIdx: 257,
|
Fee: (1 << 8) - 1,
|
||||||
Fee: 201,
|
|
||||||
}
|
}
|
||||||
// Data from the compatibility test
|
expected := "ffffffffffffffffffff"
|
||||||
expected := "00000101000001002b16c9"
|
encodedData, err := l2Tx.BytesDataAvailability(16)
|
||||||
encodedData, err := l2Tx.BytesDataAvailability(32)
|
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, expected, hex.EncodeToString(encodedData))
|
assert.Equal(t, expected, hex.EncodeToString(encodedData))
|
||||||
|
|
||||||
decodedData, err := L2TxFromBytesDataAvailability(encodedData, 32)
|
decodedData, err := L2TxFromBytesDataAvailability(encodedData, 16)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, l2Tx, decodedData)
|
||||||
|
|
||||||
|
l2Tx = &L2Tx{
|
||||||
|
ToIdx: (1 << 32) - 1,
|
||||||
|
FromIdx: (1 << 32) - 1,
|
||||||
|
Amount: amount,
|
||||||
|
Fee: (1 << 8) - 1,
|
||||||
|
}
|
||||||
|
expected = "ffffffffffffffffffffffffffff"
|
||||||
|
encodedData, err = l2Tx.BytesDataAvailability(32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, expected, hex.EncodeToString(encodedData))
|
||||||
|
|
||||||
|
decodedData, err = L2TxFromBytesDataAvailability(encodedData, 32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, l2Tx, decodedData)
|
||||||
|
|
||||||
|
l2Tx = &L2Tx{
|
||||||
|
ToIdx: 0,
|
||||||
|
FromIdx: 0,
|
||||||
|
Amount: big.NewInt(0),
|
||||||
|
Fee: 0,
|
||||||
|
}
|
||||||
|
expected = "0000000000000000000000000000"
|
||||||
|
encodedData, err = l2Tx.BytesDataAvailability(32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, expected, hex.EncodeToString(encodedData))
|
||||||
|
|
||||||
|
decodedData, err = L2TxFromBytesDataAvailability(encodedData, 32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, l2Tx, decodedData)
|
||||||
|
|
||||||
|
l2Tx = &L2Tx{
|
||||||
|
ToIdx: 0,
|
||||||
|
FromIdx: 1061,
|
||||||
|
Amount: big.NewInt(420000000000),
|
||||||
|
Fee: 127,
|
||||||
|
}
|
||||||
|
expected = "000004250000000010fa56ea007f"
|
||||||
|
encodedData, err = l2Tx.BytesDataAvailability(32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, expected, hex.EncodeToString(encodedData))
|
||||||
|
|
||||||
|
decodedData, err = L2TxFromBytesDataAvailability(encodedData, 32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, l2Tx, decodedData)
|
||||||
|
|
||||||
|
l2Tx = &L2Tx{
|
||||||
|
ToIdx: 256,
|
||||||
|
FromIdx: 257,
|
||||||
|
Amount: big.NewInt(79000000),
|
||||||
|
Fee: 201,
|
||||||
|
}
|
||||||
|
expected = "00000101000001000004b571c0c9"
|
||||||
|
encodedData, err = l2Tx.BytesDataAvailability(32)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, expected, hex.EncodeToString(encodedData))
|
||||||
|
|
||||||
|
decodedData, err = L2TxFromBytesDataAvailability(encodedData, 32)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, l2Tx, decodedData)
|
assert.Equal(t, l2Tx, decodedData)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ type PoolL2Tx struct {
|
|||||||
ToEthAddr ethCommon.Address `meddler:"to_eth_addr,zeroisnull"`
|
ToEthAddr ethCommon.Address `meddler:"to_eth_addr,zeroisnull"`
|
||||||
ToBJJ babyjub.PublicKeyComp `meddler:"to_bjj,zeroisnull"`
|
ToBJJ babyjub.PublicKeyComp `meddler:"to_bjj,zeroisnull"`
|
||||||
TokenID TokenID `meddler:"token_id"`
|
TokenID TokenID `meddler:"token_id"`
|
||||||
Amount *big.Int `meddler:"amount,bigint"` // TODO: change to float16
|
Amount *big.Int `meddler:"amount,bigint"` // TODO: change to float40
|
||||||
Fee FeeSelector `meddler:"fee"`
|
Fee FeeSelector `meddler:"fee"`
|
||||||
Nonce Nonce `meddler:"nonce"` // effective 40 bits used
|
Nonce Nonce `meddler:"nonce"` // effective 40 bits used
|
||||||
State PoolL2TxState `meddler:"state"`
|
State PoolL2TxState `meddler:"state"`
|
||||||
@@ -53,7 +53,7 @@ type PoolL2Tx struct {
|
|||||||
RqToEthAddr ethCommon.Address `meddler:"rq_to_eth_addr,zeroisnull"`
|
RqToEthAddr ethCommon.Address `meddler:"rq_to_eth_addr,zeroisnull"`
|
||||||
RqToBJJ babyjub.PublicKeyComp `meddler:"rq_to_bjj,zeroisnull"`
|
RqToBJJ babyjub.PublicKeyComp `meddler:"rq_to_bjj,zeroisnull"`
|
||||||
RqTokenID TokenID `meddler:"rq_token_id,zeroisnull"`
|
RqTokenID TokenID `meddler:"rq_token_id,zeroisnull"`
|
||||||
RqAmount *big.Int `meddler:"rq_amount,bigintnull"` // TODO: change to float16
|
RqAmount *big.Int `meddler:"rq_amount,bigintnull"` // TODO: change to float40
|
||||||
RqFee FeeSelector `meddler:"rq_fee,zeroisnull"`
|
RqFee FeeSelector `meddler:"rq_fee,zeroisnull"`
|
||||||
RqNonce Nonce `meddler:"rq_nonce,zeroisnull"` // effective 48 bits used
|
RqNonce Nonce `meddler:"rq_nonce,zeroisnull"` // effective 48 bits used
|
||||||
AbsoluteFee float64 `meddler:"fee_usd,zeroisnull"`
|
AbsoluteFee float64 `meddler:"fee_usd,zeroisnull"`
|
||||||
@@ -122,18 +122,13 @@ func (tx *PoolL2Tx) SetID() error {
|
|||||||
// [ 8 bits ] userFee // 1 byte
|
// [ 8 bits ] userFee // 1 byte
|
||||||
// [ 40 bits ] nonce // 5 bytes
|
// [ 40 bits ] nonce // 5 bytes
|
||||||
// [ 32 bits ] tokenID // 4 bytes
|
// [ 32 bits ] tokenID // 4 bytes
|
||||||
// [ 16 bits ] amountFloat16 // 2 bytes
|
|
||||||
// [ 48 bits ] toIdx // 6 bytes
|
// [ 48 bits ] toIdx // 6 bytes
|
||||||
// [ 48 bits ] fromIdx // 6 bytes
|
// [ 48 bits ] fromIdx // 6 bytes
|
||||||
// [ 16 bits ] chainId // 2 bytes
|
// [ 16 bits ] chainId // 2 bytes
|
||||||
// [ 32 bits ] signatureConstant // 4 bytes
|
// [ 32 bits ] signatureConstant // 4 bytes
|
||||||
// Total bits compressed data: 241 bits // 31 bytes in *big.Int representation
|
// Total bits compressed data: 241 bits // 31 bytes in *big.Int representation
|
||||||
func (tx *PoolL2Tx) TxCompressedData(chainID uint16) (*big.Int, error) {
|
func (tx *PoolL2Tx) TxCompressedData(chainID uint16) (*big.Int, error) {
|
||||||
amountFloat16, err := NewFloat16(tx.Amount)
|
var b [29]byte
|
||||||
if err != nil {
|
|
||||||
return nil, tracerr.Wrap(err)
|
|
||||||
}
|
|
||||||
var b [31]byte
|
|
||||||
|
|
||||||
toBJJSign := byte(0)
|
toBJJSign := byte(0)
|
||||||
pkSign, _ := babyjub.UnpackSignY(tx.ToBJJ)
|
pkSign, _ := babyjub.UnpackSignY(tx.ToBJJ)
|
||||||
@@ -149,19 +144,18 @@ func (tx *PoolL2Tx) TxCompressedData(chainID uint16) (*big.Int, error) {
|
|||||||
}
|
}
|
||||||
copy(b[2:7], nonceBytes[:])
|
copy(b[2:7], nonceBytes[:])
|
||||||
copy(b[7:11], tx.TokenID.Bytes())
|
copy(b[7:11], tx.TokenID.Bytes())
|
||||||
copy(b[11:13], amountFloat16.Bytes())
|
|
||||||
toIdxBytes, err := tx.ToIdx.Bytes()
|
toIdxBytes, err := tx.ToIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[13:19], toIdxBytes[:])
|
copy(b[11:17], toIdxBytes[:])
|
||||||
fromIdxBytes, err := tx.FromIdx.Bytes()
|
fromIdxBytes, err := tx.FromIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[19:25], fromIdxBytes[:])
|
copy(b[17:23], fromIdxBytes[:])
|
||||||
binary.BigEndian.PutUint16(b[25:27], chainID)
|
binary.BigEndian.PutUint16(b[23:25], chainID)
|
||||||
copy(b[27:31], SignatureConstantBytes[:])
|
copy(b[25:29], SignatureConstantBytes[:])
|
||||||
|
|
||||||
bi := new(big.Int).SetBytes(b[:])
|
bi := new(big.Int).SetBytes(b[:])
|
||||||
return bi, nil
|
return bi, nil
|
||||||
@@ -170,9 +164,9 @@ func (tx *PoolL2Tx) TxCompressedData(chainID uint16) (*big.Int, error) {
|
|||||||
// TxCompressedDataEmpty calculates the TxCompressedData of an empty
|
// TxCompressedDataEmpty calculates the TxCompressedData of an empty
|
||||||
// transaction
|
// transaction
|
||||||
func TxCompressedDataEmpty(chainID uint16) *big.Int {
|
func TxCompressedDataEmpty(chainID uint16) *big.Int {
|
||||||
var b [31]byte
|
var b [29]byte
|
||||||
binary.BigEndian.PutUint16(b[25:27], chainID)
|
binary.BigEndian.PutUint16(b[23:25], chainID)
|
||||||
copy(b[27:31], SignatureConstantBytes[:])
|
copy(b[25:29], SignatureConstantBytes[:])
|
||||||
bi := new(big.Int).SetBytes(b[:])
|
bi := new(big.Int).SetBytes(b[:])
|
||||||
return bi
|
return bi
|
||||||
}
|
}
|
||||||
@@ -182,7 +176,7 @@ func TxCompressedDataEmpty(chainID uint16) *big.Int {
|
|||||||
// [ 8 bits ] userFee // 1 byte
|
// [ 8 bits ] userFee // 1 byte
|
||||||
// [ 40 bits ] nonce // 5 bytes
|
// [ 40 bits ] nonce // 5 bytes
|
||||||
// [ 32 bits ] tokenID // 4 bytes
|
// [ 32 bits ] tokenID // 4 bytes
|
||||||
// [ 16 bits ] amountFloat16 // 2 bytes
|
// [ 40 bits ] amountFloat40 // 5 bytes
|
||||||
// [ 48 bits ] toIdx // 6 bytes
|
// [ 48 bits ] toIdx // 6 bytes
|
||||||
// [ 48 bits ] fromIdx // 6 bytes
|
// [ 48 bits ] fromIdx // 6 bytes
|
||||||
// Total bits compressed data: 193 bits // 25 bytes in *big.Int representation
|
// Total bits compressed data: 193 bits // 25 bytes in *big.Int representation
|
||||||
@@ -190,11 +184,16 @@ func (tx *PoolL2Tx) TxCompressedDataV2() (*big.Int, error) {
|
|||||||
if tx.Amount == nil {
|
if tx.Amount == nil {
|
||||||
tx.Amount = big.NewInt(0)
|
tx.Amount = big.NewInt(0)
|
||||||
}
|
}
|
||||||
amountFloat16, err := NewFloat16(tx.Amount)
|
amountFloat40, err := NewFloat40(tx.Amount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
var b [25]byte
|
amountFloat40Bytes, err := amountFloat40.Bytes()
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var b [28]byte
|
||||||
toBJJSign := byte(0)
|
toBJJSign := byte(0)
|
||||||
if tx.ToBJJ != EmptyBJJComp {
|
if tx.ToBJJ != EmptyBJJComp {
|
||||||
sign, _ := babyjub.UnpackSignY(tx.ToBJJ)
|
sign, _ := babyjub.UnpackSignY(tx.ToBJJ)
|
||||||
@@ -210,17 +209,17 @@ func (tx *PoolL2Tx) TxCompressedDataV2() (*big.Int, error) {
|
|||||||
}
|
}
|
||||||
copy(b[2:7], nonceBytes[:])
|
copy(b[2:7], nonceBytes[:])
|
||||||
copy(b[7:11], tx.TokenID.Bytes())
|
copy(b[7:11], tx.TokenID.Bytes())
|
||||||
copy(b[11:13], amountFloat16.Bytes())
|
copy(b[11:16], amountFloat40Bytes)
|
||||||
toIdxBytes, err := tx.ToIdx.Bytes()
|
toIdxBytes, err := tx.ToIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[13:19], toIdxBytes[:])
|
copy(b[16:22], toIdxBytes[:])
|
||||||
fromIdxBytes, err := tx.FromIdx.Bytes()
|
fromIdxBytes, err := tx.FromIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[19:25], fromIdxBytes[:])
|
copy(b[22:28], fromIdxBytes[:])
|
||||||
|
|
||||||
bi := new(big.Int).SetBytes(b[:])
|
bi := new(big.Int).SetBytes(b[:])
|
||||||
return bi, nil
|
return bi, nil
|
||||||
@@ -236,7 +235,7 @@ func (tx *PoolL2Tx) TxCompressedDataV2() (*big.Int, error) {
|
|||||||
// [ 8 bits ] rqUserFee // 1 byte
|
// [ 8 bits ] rqUserFee // 1 byte
|
||||||
// [ 40 bits ] rqNonce // 5 bytes
|
// [ 40 bits ] rqNonce // 5 bytes
|
||||||
// [ 32 bits ] rqTokenID // 4 bytes
|
// [ 32 bits ] rqTokenID // 4 bytes
|
||||||
// [ 16 bits ] rqAmountFloat16 // 2 bytes
|
// [ 40 bits ] rqAmountFloat40 // 5 bytes
|
||||||
// [ 48 bits ] rqToIdx // 6 bytes
|
// [ 48 bits ] rqToIdx // 6 bytes
|
||||||
// [ 48 bits ] rqFromIdx // 6 bytes
|
// [ 48 bits ] rqFromIdx // 6 bytes
|
||||||
// Total bits compressed data: 193 bits // 25 bytes in *big.Int representation
|
// Total bits compressed data: 193 bits // 25 bytes in *big.Int representation
|
||||||
@@ -244,11 +243,16 @@ func (tx *PoolL2Tx) RqTxCompressedDataV2() (*big.Int, error) {
|
|||||||
if tx.RqAmount == nil {
|
if tx.RqAmount == nil {
|
||||||
tx.RqAmount = big.NewInt(0)
|
tx.RqAmount = big.NewInt(0)
|
||||||
}
|
}
|
||||||
amountFloat16, err := NewFloat16(tx.RqAmount)
|
amountFloat40, err := NewFloat40(tx.RqAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
var b [25]byte
|
amountFloat40Bytes, err := amountFloat40.Bytes()
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var b [28]byte
|
||||||
rqToBJJSign := byte(0)
|
rqToBJJSign := byte(0)
|
||||||
if tx.RqToBJJ != EmptyBJJComp {
|
if tx.RqToBJJ != EmptyBJJComp {
|
||||||
sign, _ := babyjub.UnpackSignY(tx.RqToBJJ)
|
sign, _ := babyjub.UnpackSignY(tx.RqToBJJ)
|
||||||
@@ -264,17 +268,17 @@ func (tx *PoolL2Tx) RqTxCompressedDataV2() (*big.Int, error) {
|
|||||||
}
|
}
|
||||||
copy(b[2:7], nonceBytes[:])
|
copy(b[2:7], nonceBytes[:])
|
||||||
copy(b[7:11], tx.RqTokenID.Bytes())
|
copy(b[7:11], tx.RqTokenID.Bytes())
|
||||||
copy(b[11:13], amountFloat16.Bytes())
|
copy(b[11:16], amountFloat40Bytes)
|
||||||
toIdxBytes, err := tx.RqToIdx.Bytes()
|
toIdxBytes, err := tx.RqToIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[13:19], toIdxBytes[:])
|
copy(b[16:22], toIdxBytes[:])
|
||||||
fromIdxBytes, err := tx.RqFromIdx.Bytes()
|
fromIdxBytes, err := tx.RqFromIdx.Bytes()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
copy(b[19:25], fromIdxBytes[:])
|
copy(b[22:28], fromIdxBytes[:])
|
||||||
|
|
||||||
bi := new(big.Int).SetBytes(b[:])
|
bi := new(big.Int).SetBytes(b[:])
|
||||||
return bi, nil
|
return bi, nil
|
||||||
@@ -287,7 +291,22 @@ func (tx *PoolL2Tx) HashToSign(chainID uint16) (*big.Int, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// e1: [5 bytes AmountFloat40 | 20 bytes ToEthAddr]
|
||||||
|
var e1B [25]byte
|
||||||
|
amountFloat40, err := NewFloat40(tx.Amount)
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
amountFloat40Bytes, err := amountFloat40.Bytes()
|
||||||
|
if err != nil {
|
||||||
|
return nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
copy(e1B[0:5], amountFloat40Bytes)
|
||||||
toEthAddr := EthAddrToBigInt(tx.ToEthAddr)
|
toEthAddr := EthAddrToBigInt(tx.ToEthAddr)
|
||||||
|
copy(e1B[5:25], toEthAddr.Bytes())
|
||||||
|
e1 := new(big.Int).SetBytes(e1B[:])
|
||||||
|
|
||||||
rqToEthAddr := EthAddrToBigInt(tx.RqToEthAddr)
|
rqToEthAddr := EthAddrToBigInt(tx.RqToEthAddr)
|
||||||
|
|
||||||
_, toBJJY := babyjub.UnpackSignY(tx.ToBJJ)
|
_, toBJJY := babyjub.UnpackSignY(tx.ToBJJ)
|
||||||
@@ -299,7 +318,7 @@ func (tx *PoolL2Tx) HashToSign(chainID uint16) (*big.Int, error) {
|
|||||||
|
|
||||||
_, rqToBJJY := babyjub.UnpackSignY(tx.RqToBJJ)
|
_, rqToBJJY := babyjub.UnpackSignY(tx.RqToBJJ)
|
||||||
|
|
||||||
return poseidon.Hash([]*big.Int{toCompressedData, toEthAddr, toBJJY, rqTxCompressedDataV2, rqToEthAddr, rqToBJJY})
|
return poseidon.Hash([]*big.Int{toCompressedData, e1, toBJJY, rqTxCompressedDataV2, rqToEthAddr, rqToBJJY})
|
||||||
}
|
}
|
||||||
|
|
||||||
// VerifySignature returns true if the signature verification is correct for the given PublicKeyComp
|
// VerifySignature returns true if the signature verification is correct for the given PublicKeyComp
|
||||||
|
|||||||
@@ -21,80 +21,104 @@ func TestNewPoolL2Tx(t *testing.T) {
|
|||||||
}
|
}
|
||||||
poolL2Tx, err := NewPoolL2Tx(poolL2Tx)
|
poolL2Tx, err := NewPoolL2Tx(poolL2Tx)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "0x02fb52b5d0b9ef2626c11701bb751b2720c76d59946b9a48146ac153bb6e63bf6a", poolL2Tx.TxID.String())
|
assert.Equal(t, "0x022669acda59b827d20ef5354a3eebd1dffb3972b0a6bf89d18bfd2efa0ab9f41e", poolL2Tx.TxID.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestTxCompressedData(t *testing.T) {
|
func TestTxCompressedDataAndTxCompressedDataV2JSVectors(t *testing.T) {
|
||||||
chainID := uint16(0)
|
// test vectors values generated from javascript implementation
|
||||||
var sk babyjub.PrivateKey
|
var skPositive babyjub.PrivateKey // 'Positive' refers to the sign
|
||||||
_, err := hex.Decode(sk[:], []byte("0001020304050607080900010203040506070809000102030405060708090001"))
|
_, err := hex.Decode(skPositive[:], []byte("0001020304050607080900010203040506070809000102030405060708090001"))
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
|
|
||||||
|
var skNegative babyjub.PrivateKey // 'Negative' refers to the sign
|
||||||
|
_, err = hex.Decode(skNegative[:], []byte("0001020304050607080900010203040506070809000102030405060708090002"))
|
||||||
|
assert.NoError(t, err)
|
||||||
|
|
||||||
|
amount, ok := new(big.Int).SetString("343597383670000000000000000000000000000000", 10)
|
||||||
|
require.True(t, ok)
|
||||||
tx := PoolL2Tx{
|
tx := PoolL2Tx{
|
||||||
FromIdx: 2,
|
FromIdx: (1 << 48) - 1,
|
||||||
ToIdx: 3,
|
ToIdx: (1 << 48) - 1,
|
||||||
Amount: big.NewInt(4),
|
Amount: amount,
|
||||||
TokenID: 5,
|
TokenID: (1 << 32) - 1,
|
||||||
Nonce: 6,
|
Nonce: (1 << 40) - 1,
|
||||||
ToBJJ: sk.Public().Compress(),
|
Fee: (1 << 3) - 1,
|
||||||
|
ToBJJ: skPositive.Public().Compress(),
|
||||||
}
|
}
|
||||||
txCompressedData, err := tx.TxCompressedData(chainID)
|
txCompressedData, err := tx.TxCompressedData(uint16((1 << 16) - 1))
|
||||||
assert.NoError(t, err)
|
require.NoError(t, err)
|
||||||
// test vector value generated from javascript implementation
|
expectedStr := "0107ffffffffffffffffffffffffffffffffffffffffffffffc60be60f"
|
||||||
expectedStr := "1766847064778421992193717128424891165872736891548909569553540445094274575"
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
assert.Equal(t, expectedStr, txCompressedData.String())
|
|
||||||
assert.Equal(t, "010000000000060000000500040000000000030000000000020000c60be60f", hex.EncodeToString(txCompressedData.Bytes()))
|
txCompressedDataV2, err := tx.TxCompressedDataV2()
|
||||||
// using a different chainID
|
require.NoError(t, err)
|
||||||
txCompressedData, err = tx.TxCompressedData(uint16(100))
|
expectedStr = "0107ffffffffffffffffffffffffffffffffffffffffffffffffffff"
|
||||||
assert.NoError(t, err)
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedDataV2.Bytes()))
|
||||||
expectedStr = "1766847064778421992193717128424891165872736891548909569553540874591004175"
|
|
||||||
assert.Equal(t, expectedStr, txCompressedData.String())
|
|
||||||
assert.Equal(t, "010000000000060000000500040000000000030000000000020064c60be60f", hex.EncodeToString(txCompressedData.Bytes()))
|
|
||||||
txCompressedData, err = tx.TxCompressedData(uint16(65535))
|
|
||||||
assert.NoError(t, err)
|
|
||||||
expectedStr = "1766847064778421992193717128424891165872736891548909569553821915776017935"
|
|
||||||
assert.Equal(t, expectedStr, txCompressedData.String())
|
|
||||||
assert.Equal(t, "01000000000006000000050004000000000003000000000002ffffc60be60f", hex.EncodeToString(txCompressedData.Bytes()))
|
|
||||||
|
|
||||||
tx = PoolL2Tx{
|
tx = PoolL2Tx{
|
||||||
RqFromIdx: 7,
|
FromIdx: 0,
|
||||||
RqToIdx: 8,
|
ToIdx: 0,
|
||||||
RqAmount: big.NewInt(9),
|
Amount: big.NewInt(0),
|
||||||
RqTokenID: 10,
|
TokenID: 0,
|
||||||
RqNonce: 11,
|
Nonce: 0,
|
||||||
RqFee: 12,
|
Fee: 0,
|
||||||
RqToBJJ: sk.Public().Compress(),
|
ToBJJ: skNegative.Public().Compress(),
|
||||||
}
|
}
|
||||||
rqTxCompressedData, err := tx.RqTxCompressedDataV2()
|
txCompressedData, err = tx.TxCompressedData(uint16(0))
|
||||||
assert.NoError(t, err)
|
require.NoError(t, err)
|
||||||
// test vector value generated from javascript implementation
|
expectedStr = "c60be60f"
|
||||||
expectedStr = "6571340879233176732837827812956721483162819083004853354503"
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
assert.Equal(t, expectedStr, rqTxCompressedData.String())
|
|
||||||
assert.Equal(t, "010c000000000b0000000a0009000000000008000000000007", hex.EncodeToString(rqTxCompressedData.Bytes()))
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestTxCompressedDataV2(t *testing.T) {
|
txCompressedDataV2, err = tx.TxCompressedDataV2()
|
||||||
var sk babyjub.PrivateKey
|
require.NoError(t, err)
|
||||||
_, err := hex.Decode(sk[:], []byte("0001020304050607080900010203040506070809000102030405060708090001"))
|
assert.Equal(t, "0", txCompressedDataV2.String())
|
||||||
assert.NoError(t, err)
|
|
||||||
tx := PoolL2Tx{
|
amount, ok = new(big.Int).SetString("63000000000000000", 10)
|
||||||
FromIdx: 7,
|
require.True(t, ok)
|
||||||
ToIdx: 8,
|
tx = PoolL2Tx{
|
||||||
Amount: big.NewInt(9),
|
FromIdx: 324,
|
||||||
TokenID: 10,
|
ToIdx: 256,
|
||||||
Nonce: 11,
|
Amount: amount,
|
||||||
Fee: 12,
|
TokenID: 123,
|
||||||
ToBJJ: sk.Public().Compress(),
|
Nonce: 76,
|
||||||
|
Fee: 214,
|
||||||
|
ToBJJ: skNegative.Public().Compress(),
|
||||||
}
|
}
|
||||||
txCompressedData, err := tx.TxCompressedDataV2()
|
txCompressedData, err = tx.TxCompressedData(uint16(1))
|
||||||
assert.NoError(t, err)
|
require.NoError(t, err)
|
||||||
// test vector value generated from javascript implementation
|
expectedStr = "d6000000004c0000007b0000000001000000000001440001c60be60f"
|
||||||
expectedStr := "6571340879233176732837827812956721483162819083004853354503"
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
assert.Equal(t, expectedStr, txCompressedData.String())
|
|
||||||
expected, ok := new(big.Int).SetString(expectedStr, 10)
|
|
||||||
assert.True(t, ok)
|
|
||||||
|
|
||||||
assert.Equal(t, expected.Bytes(), txCompressedData.Bytes())
|
txCompressedDataV2, err = tx.TxCompressedDataV2()
|
||||||
assert.Equal(t, "010c000000000b0000000a0009000000000008000000000007", hex.EncodeToString(txCompressedData.Bytes()))
|
require.NoError(t, err)
|
||||||
|
expectedStr = "d6000000004c0000007b3977825f00000000000100000000000144"
|
||||||
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedDataV2.Bytes()))
|
||||||
|
|
||||||
|
tx = PoolL2Tx{
|
||||||
|
FromIdx: 1,
|
||||||
|
ToIdx: 2,
|
||||||
|
TokenID: 3,
|
||||||
|
Nonce: 4,
|
||||||
|
Fee: 5,
|
||||||
|
ToBJJ: skNegative.Public().Compress(),
|
||||||
|
}
|
||||||
|
txCompressedData, err = tx.TxCompressedData(uint16(0))
|
||||||
|
require.NoError(t, err)
|
||||||
|
expectedStr = "050000000004000000030000000000020000000000010000c60be60f"
|
||||||
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
|
|
||||||
|
tx = PoolL2Tx{
|
||||||
|
FromIdx: 2,
|
||||||
|
ToIdx: 3,
|
||||||
|
TokenID: 4,
|
||||||
|
Nonce: 5,
|
||||||
|
Fee: 6,
|
||||||
|
ToBJJ: skPositive.Public().Compress(),
|
||||||
|
}
|
||||||
|
txCompressedData, err = tx.TxCompressedData(uint16(0))
|
||||||
|
require.NoError(t, err)
|
||||||
|
expectedStr = "01060000000005000000040000000000030000000000020000c60be60f"
|
||||||
|
assert.Equal(t, expectedStr, hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestRqTxCompressedDataV2(t *testing.T) {
|
func TestRqTxCompressedDataV2(t *testing.T) {
|
||||||
@@ -113,19 +137,16 @@ func TestRqTxCompressedDataV2(t *testing.T) {
|
|||||||
txCompressedData, err := tx.RqTxCompressedDataV2()
|
txCompressedData, err := tx.RqTxCompressedDataV2()
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
// test vector value generated from javascript implementation
|
// test vector value generated from javascript implementation
|
||||||
expectedStr := "6571340879233176732837827812956721483162819083004853354503"
|
expectedStr := "110248805340524920412994530176819463725852160917809517418728390663"
|
||||||
assert.Equal(t, expectedStr, txCompressedData.String())
|
assert.Equal(t, expectedStr, txCompressedData.String())
|
||||||
expected, ok := new(big.Int).SetString(expectedStr, 10)
|
expected, ok := new(big.Int).SetString(expectedStr, 10)
|
||||||
assert.True(t, ok)
|
assert.True(t, ok)
|
||||||
assert.Equal(t, expected.Bytes(), txCompressedData.Bytes())
|
assert.Equal(t, expected.Bytes(), txCompressedData.Bytes())
|
||||||
assert.Equal(t, "010c000000000b0000000a0009000000000008000000000007", hex.EncodeToString(txCompressedData.Bytes()))
|
assert.Equal(t, "010c000000000b0000000a0000000009000000000008000000000007", hex.EncodeToString(txCompressedData.Bytes()))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestHashToSign(t *testing.T) {
|
func TestHashToSign(t *testing.T) {
|
||||||
chainID := uint16(0)
|
chainID := uint16(0)
|
||||||
var sk babyjub.PrivateKey
|
|
||||||
_, err := hex.Decode(sk[:], []byte("0001020304050607080900010203040506070809000102030405060708090001"))
|
|
||||||
assert.NoError(t, err)
|
|
||||||
tx := PoolL2Tx{
|
tx := PoolL2Tx{
|
||||||
FromIdx: 2,
|
FromIdx: 2,
|
||||||
ToIdx: 3,
|
ToIdx: 3,
|
||||||
@@ -136,7 +157,7 @@ func TestHashToSign(t *testing.T) {
|
|||||||
}
|
}
|
||||||
toSign, err := tx.HashToSign(chainID)
|
toSign, err := tx.HashToSign(chainID)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "1469900657138253851938022936440971384682713995864967090251961124784132925291", toSign.String())
|
assert.Equal(t, "2d49ce1d4136e06f64e3eb1f79a346e6ee3e93ceeac909a57806a8d87005c263", hex.EncodeToString(toSign.Bytes()))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestVerifyTxSignature(t *testing.T) {
|
func TestVerifyTxSignature(t *testing.T) {
|
||||||
@@ -156,7 +177,7 @@ func TestVerifyTxSignature(t *testing.T) {
|
|||||||
}
|
}
|
||||||
toSign, err := tx.HashToSign(chainID)
|
toSign, err := tx.HashToSign(chainID)
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "18645218094210271622244722988708640202588315450486586312909439859037906375295", toSign.String())
|
assert.Equal(t, "1571327027383224465388301747239444557034990637650927918405777653988509342917", toSign.String())
|
||||||
|
|
||||||
sig := sk.SignPoseidon(toSign)
|
sig := sk.SignPoseidon(toSign)
|
||||||
tx.Signature = sig.Compress()
|
tx.Signature = sig.Compress()
|
||||||
|
|||||||
11
common/zk.go
11
common/zk.go
@@ -102,6 +102,8 @@ type ZKInputs struct {
|
|||||||
ToBJJAy []*big.Int `json:"toBjjAy"` // big.Int, len: [maxTx]
|
ToBJJAy []*big.Int `json:"toBjjAy"` // big.Int, len: [maxTx]
|
||||||
// ToEthAddr
|
// ToEthAddr
|
||||||
ToEthAddr []*big.Int `json:"toEthAddr"` // ethCommon.Address, len: [maxTx]
|
ToEthAddr []*big.Int `json:"toEthAddr"` // ethCommon.Address, len: [maxTx]
|
||||||
|
// AmountF encoded as float40
|
||||||
|
AmountF []*big.Int `json:"amountF"`
|
||||||
|
|
||||||
// OnChain determines if is L1 (1/true) or L2 (0/false)
|
// OnChain determines if is L1 (1/true) or L2 (0/false)
|
||||||
OnChain []*big.Int `json:"onChain"` // bool, len: [maxTx]
|
OnChain []*big.Int `json:"onChain"` // bool, len: [maxTx]
|
||||||
@@ -112,8 +114,8 @@ type ZKInputs struct {
|
|||||||
// NewAccount boolean (0/1) flag set 'true' when L1 tx creates a new
|
// NewAccount boolean (0/1) flag set 'true' when L1 tx creates a new
|
||||||
// account (fromIdx==0)
|
// account (fromIdx==0)
|
||||||
NewAccount []*big.Int `json:"newAccount"` // bool, len: [maxTx]
|
NewAccount []*big.Int `json:"newAccount"` // bool, len: [maxTx]
|
||||||
// DepositAmountF encoded as float16
|
// DepositAmountF encoded as float40
|
||||||
DepositAmountF []*big.Int `json:"loadAmountF"` // uint16, len: [maxTx]
|
DepositAmountF []*big.Int `json:"loadAmountF"` // uint40, len: [maxTx]
|
||||||
// FromEthAddr
|
// FromEthAddr
|
||||||
FromEthAddr []*big.Int `json:"fromEthAddr"` // ethCommon.Address, len: [maxTx]
|
FromEthAddr []*big.Int `json:"fromEthAddr"` // ethCommon.Address, len: [maxTx]
|
||||||
// FromBJJCompressed boolean encoded where each value is a *big.Int
|
// FromBJJCompressed boolean encoded where each value is a *big.Int
|
||||||
@@ -326,6 +328,7 @@ func NewZKInputs(chainID uint16, maxTx, maxL1Tx, maxFeeIdxs, nLevels uint32, cur
|
|||||||
zki.AuxToIdx = newSlice(maxTx)
|
zki.AuxToIdx = newSlice(maxTx)
|
||||||
zki.ToBJJAy = newSlice(maxTx)
|
zki.ToBJJAy = newSlice(maxTx)
|
||||||
zki.ToEthAddr = newSlice(maxTx)
|
zki.ToEthAddr = newSlice(maxTx)
|
||||||
|
zki.AmountF = newSlice(maxTx)
|
||||||
zki.OnChain = newSlice(maxTx)
|
zki.OnChain = newSlice(maxTx)
|
||||||
zki.NewAccount = newSlice(maxTx)
|
zki.NewAccount = newSlice(maxTx)
|
||||||
|
|
||||||
@@ -477,7 +480,7 @@ func (z ZKInputs) ToHashGlobalData() ([]byte, error) {
|
|||||||
b = append(b, newExitRoot...)
|
b = append(b, newExitRoot...)
|
||||||
|
|
||||||
// [MAX_L1_TX * (2 * MAX_NLEVELS + 480) bits] L1TxsData
|
// [MAX_L1_TX * (2 * MAX_NLEVELS + 480) bits] L1TxsData
|
||||||
l1TxDataLen := (2*z.Metadata.MaxLevels + 480)
|
l1TxDataLen := (2*z.Metadata.MaxLevels + 528)
|
||||||
l1TxsDataLen := (z.Metadata.MaxL1Tx * l1TxDataLen)
|
l1TxsDataLen := (z.Metadata.MaxL1Tx * l1TxDataLen)
|
||||||
l1TxsData := make([]byte, l1TxsDataLen/8) //nolint:gomnd
|
l1TxsData := make([]byte, l1TxsDataLen/8) //nolint:gomnd
|
||||||
for i := 0; i < len(z.Metadata.L1TxsData); i++ {
|
for i := 0; i < len(z.Metadata.L1TxsData); i++ {
|
||||||
@@ -496,7 +499,7 @@ func (z ZKInputs) ToHashGlobalData() ([]byte, error) {
|
|||||||
|
|
||||||
// [MAX_TX*(2*NLevels + 24) bits] L2TxsData
|
// [MAX_TX*(2*NLevels + 24) bits] L2TxsData
|
||||||
var l2TxsData []byte
|
var l2TxsData []byte
|
||||||
l2TxDataLen := 2*z.Metadata.NLevels + 24 //nolint:gomnd
|
l2TxDataLen := 2*z.Metadata.NLevels + 48 //nolint:gomnd
|
||||||
l2TxsDataLen := (z.Metadata.MaxTx * l2TxDataLen)
|
l2TxsDataLen := (z.Metadata.MaxTx * l2TxDataLen)
|
||||||
expectedL2TxsDataLen := l2TxsDataLen / 8 //nolint:gomnd
|
expectedL2TxsDataLen := l2TxsDataLen / 8 //nolint:gomnd
|
||||||
for i := 0; i < len(z.Metadata.L2TxsData); i++ {
|
for i := 0; i < len(z.Metadata.L2TxsData); i++ {
|
||||||
|
|||||||
@@ -611,10 +611,10 @@ func TestTxs(t *testing.T) {
|
|||||||
assert.Equal(t, common.TxTypeExit, dbL2Txs[3].Type)
|
assert.Equal(t, common.TxTypeExit, dbL2Txs[3].Type)
|
||||||
|
|
||||||
// Tx ID
|
// Tx ID
|
||||||
assert.Equal(t, "0x02d709307533c4e3c03f20751fc4d72bc18b225d14f9616525540a64342c7c350d", dbL2Txs[0].TxID.String())
|
assert.Equal(t, "0x024e555248100b69a8aabf6d31719b9fe8a60dcc6c3407904a93c8d2d9ade18ee5", dbL2Txs[0].TxID.String())
|
||||||
assert.Equal(t, "0x02e88bc5503f282cca045847668511290e642410a459bb67b1fafcd1b6097c149c", dbL2Txs[1].TxID.String())
|
assert.Equal(t, "0x021ae87ca34d50ff35d98dfc0d7c95f2bf2e4ffeebb82ea71f43a8b0dfa5d36d89", dbL2Txs[1].TxID.String())
|
||||||
assert.Equal(t, "0x027911262b43315c0b24942a02fe228274b6e4d57a476bfcdd7a324b3091362c7d", dbL2Txs[2].TxID.String())
|
assert.Equal(t, "0x024abce7f3f2382dc520ed557593f11dea1ee197e55b60402e664facc27aa19774", dbL2Txs[2].TxID.String())
|
||||||
assert.Equal(t, "0x02f572b63f2a5c302e1b9337ea6944bfbac3d199e4ddd262b5a53759c72ec10ee6", dbL2Txs[3].TxID.String())
|
assert.Equal(t, "0x02f921ad9e7a6e59606570fe12a7dde0e36014197de0363b9b45e5097d6f2b1dd0", dbL2Txs[3].TxID.String())
|
||||||
|
|
||||||
// Tx From and To IDx
|
// Tx From and To IDx
|
||||||
assert.Equal(t, dbL2Txs[0].ToIdx, dbL2Txs[2].FromIdx)
|
assert.Equal(t, dbL2Txs[0].ToIdx, dbL2Txs[2].FromIdx)
|
||||||
|
|||||||
@@ -462,11 +462,11 @@ func (c *RollupClient) RollupL1UserTxERC20ETH(fromBJJ babyjub.PublicKeyComp, fro
|
|||||||
}
|
}
|
||||||
fromIdxBig := big.NewInt(fromIdx)
|
fromIdxBig := big.NewInt(fromIdx)
|
||||||
toIdxBig := big.NewInt(toIdx)
|
toIdxBig := big.NewInt(toIdx)
|
||||||
depositAmountF, err := common.NewFloat16(depositAmount)
|
depositAmountF, err := common.NewFloat40(depositAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
amountF, err := common.NewFloat16(amount)
|
amountF, err := common.NewFloat40(amount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
@@ -497,11 +497,11 @@ func (c *RollupClient) RollupL1UserTxERC20Permit(fromBJJ babyjub.PublicKeyComp,
|
|||||||
}
|
}
|
||||||
fromIdxBig := big.NewInt(fromIdx)
|
fromIdxBig := big.NewInt(fromIdx)
|
||||||
toIdxBig := big.NewInt(toIdx)
|
toIdxBig := big.NewInt(toIdx)
|
||||||
depositAmountF, err := common.NewFloat16(depositAmount)
|
depositAmountF, err := common.NewFloat40(depositAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
amountF, err := common.NewFloat16(amount)
|
amountF, err := common.NewFloat40(amount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
@@ -939,9 +939,9 @@ func (c *RollupClient) RollupForgeBatchArgs(ethTxHash ethCommon.Hash, l1UserTxsL
|
|||||||
FeeIdxCoordinator: []common.Idx{},
|
FeeIdxCoordinator: []common.Idx{},
|
||||||
}
|
}
|
||||||
nLevels := c.consts.Verifiers[rollupForgeBatchArgs.VerifierIdx].NLevels
|
nLevels := c.consts.Verifiers[rollupForgeBatchArgs.VerifierIdx].NLevels
|
||||||
lenL1L2TxsBytes := int((nLevels/8)*2 + 2 + 1)
|
lenL1L2TxsBytes := int((nLevels/8)*2 + common.Float40BytesLength + 1)
|
||||||
numBytesL1TxUser := int(l1UserTxsLen) * lenL1L2TxsBytes
|
numBytesL1TxUser := int(l1UserTxsLen) * lenL1L2TxsBytes
|
||||||
numTxsL1Coord := len(aux.EncodedL1CoordinatorTx) / common.L1CoordinatorTxBytesLen
|
numTxsL1Coord := len(aux.EncodedL1CoordinatorTx) / common.RollupConstL1CoordinatorTotalBytes
|
||||||
numBytesL1TxCoord := numTxsL1Coord * lenL1L2TxsBytes
|
numBytesL1TxCoord := numTxsL1Coord * lenL1L2TxsBytes
|
||||||
numBeginL2Tx := numBytesL1TxCoord + numBytesL1TxUser
|
numBeginL2Tx := numBytesL1TxCoord + numBytesL1TxUser
|
||||||
l1UserTxsData := []byte{}
|
l1UserTxsData := []byte{}
|
||||||
@@ -968,7 +968,7 @@ func (c *RollupClient) RollupForgeBatchArgs(ethTxHash ethCommon.Hash, l1UserTxsL
|
|||||||
rollupForgeBatchArgs.L2TxsData = append(rollupForgeBatchArgs.L2TxsData, *l2Tx)
|
rollupForgeBatchArgs.L2TxsData = append(rollupForgeBatchArgs.L2TxsData, *l2Tx)
|
||||||
}
|
}
|
||||||
for i := 0; i < numTxsL1Coord; i++ {
|
for i := 0; i < numTxsL1Coord; i++ {
|
||||||
bytesL1Coordinator := aux.EncodedL1CoordinatorTx[i*common.L1CoordinatorTxBytesLen : (i+1)*common.L1CoordinatorTxBytesLen]
|
bytesL1Coordinator := aux.EncodedL1CoordinatorTx[i*common.RollupConstL1CoordinatorTotalBytes : (i+1)*common.RollupConstL1CoordinatorTotalBytes]
|
||||||
var signature []byte
|
var signature []byte
|
||||||
v := bytesL1Coordinator[0]
|
v := bytesL1Coordinator[0]
|
||||||
s := bytesL1Coordinator[1:33]
|
s := bytesL1Coordinator[1:33]
|
||||||
|
|||||||
@@ -131,9 +131,9 @@ func TestRollupForgeBatch(t *testing.T) {
|
|||||||
args.FeeIdxCoordinator = []common.Idx{} // When encoded, 64 times the 0 idx means that no idx to collect fees is specified.
|
args.FeeIdxCoordinator = []common.Idx{} // When encoded, 64 times the 0 idx means that no idx to collect fees is specified.
|
||||||
l1CoordinatorBytes, err := hex.DecodeString("1c660323607bb113e586183609964a333d07ebe4bef3be82ec13af453bae9590bd7711cdb6abf42f176eadfbe5506fbef5e092e5543733f91b0061d9a7747fa10694a915a6470fa230de387b51e6f4db0b09787867778687b55197ad6d6a86eac000000001")
|
l1CoordinatorBytes, err := hex.DecodeString("1c660323607bb113e586183609964a333d07ebe4bef3be82ec13af453bae9590bd7711cdb6abf42f176eadfbe5506fbef5e092e5543733f91b0061d9a7747fa10694a915a6470fa230de387b51e6f4db0b09787867778687b55197ad6d6a86eac000000001")
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
numTxsL1 := len(l1CoordinatorBytes) / common.L1CoordinatorTxBytesLen
|
numTxsL1 := len(l1CoordinatorBytes) / common.RollupConstL1CoordinatorTotalBytes
|
||||||
for i := 0; i < numTxsL1; i++ {
|
for i := 0; i < numTxsL1; i++ {
|
||||||
bytesL1Coordinator := l1CoordinatorBytes[i*common.L1CoordinatorTxBytesLen : (i+1)*common.L1CoordinatorTxBytesLen]
|
bytesL1Coordinator := l1CoordinatorBytes[i*common.RollupConstL1CoordinatorTotalBytes : (i+1)*common.RollupConstL1CoordinatorTotalBytes]
|
||||||
var signature []byte
|
var signature []byte
|
||||||
v := bytesL1Coordinator[0]
|
v := bytesL1Coordinator[0]
|
||||||
s := bytesL1Coordinator[1:33]
|
s := bytesL1Coordinator[1:33]
|
||||||
|
|||||||
@@ -797,11 +797,11 @@ func (c *Client) RollupL1UserTxERC20ETH(
|
|||||||
cpy := c.nextBlock().copy()
|
cpy := c.nextBlock().copy()
|
||||||
defer func() { c.revertIfErr(err, cpy) }()
|
defer func() { c.revertIfErr(err, cpy) }()
|
||||||
|
|
||||||
_, err = common.NewFloat16(amount)
|
_, err = common.NewFloat40(amount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
_, err = common.NewFloat16(depositAmount)
|
_, err = common.NewFloat40(depositAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, tracerr.Wrap(err)
|
return nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -142,10 +142,12 @@ func GenerateTxsZKInputs0(t *testing.T, chainID uint16) (users []til.User, coord
|
|||||||
// same values than in the js test
|
// same values than in the js test
|
||||||
users = GenerateJsUsers(t)
|
users = GenerateJsUsers(t)
|
||||||
|
|
||||||
|
depositAmount, err := common.Float40(10400).BigInt()
|
||||||
|
require.Nil(t, err)
|
||||||
l1UserTxs = []common.L1Tx{
|
l1UserTxs = []common.L1Tx{
|
||||||
{
|
{
|
||||||
FromIdx: 0,
|
FromIdx: 0,
|
||||||
DepositAmount: big.NewInt(16000000),
|
DepositAmount: depositAmount,
|
||||||
Amount: big.NewInt(0),
|
Amount: big.NewInt(0),
|
||||||
TokenID: 1,
|
TokenID: 1,
|
||||||
FromBJJ: users[0].BJJ.Public().Compress(),
|
FromBJJ: users[0].BJJ.Public().Compress(),
|
||||||
|
|||||||
@@ -311,7 +311,7 @@ func TestZKInputsExitWithFee0(t *testing.T) {
|
|||||||
assert.Equal(t, "8737171572459172806192626402462788826264011087579491137542380589998149683116", bb.LocalStateDB().MT.Root().BigInt().String())
|
assert.Equal(t, "8737171572459172806192626402462788826264011087579491137542380589998149683116", bb.LocalStateDB().MT.Root().BigInt().String())
|
||||||
h, err := zki.HashGlobalData()
|
h, err := zki.HashGlobalData()
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, "9971598169768987067017223790214537222850903267980994716992761290793474746117", h.String())
|
assert.Equal(t, "18608843755023673022528019960628191162333429206359207449879743919826610006009", h.String())
|
||||||
sendProofAndCheckResp(t, zki)
|
sendProofAndCheckResp(t, zki)
|
||||||
|
|
||||||
// batch3
|
// batch3
|
||||||
@@ -334,7 +334,7 @@ func TestZKInputsExitWithFee0(t *testing.T) {
|
|||||||
assert.Equal(t, "18306761925365215381387147754881756804475668085493847010988306480531520370130", bb.LocalStateDB().MT.Root().BigInt().String())
|
assert.Equal(t, "18306761925365215381387147754881756804475668085493847010988306480531520370130", bb.LocalStateDB().MT.Root().BigInt().String())
|
||||||
h, err = zki.HashGlobalData()
|
h, err = zki.HashGlobalData()
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
assert.Equal(t, "7992262236065691439683036344554725221924027193771770363772735722054938818364", h.String())
|
assert.Equal(t, "6651837443119278772088559395433504719862425648816904171510845286897104469889", h.String())
|
||||||
assert.Equal(t, common.EthAddrToBigInt(tc.Users["Coord"].Addr), zki.EthAddr3[0])
|
assert.Equal(t, common.EthAddrToBigInt(tc.Users["Coord"].Addr), zki.EthAddr3[0])
|
||||||
assert.Equal(t, "0", zki.EthAddr3[1].String())
|
assert.Equal(t, "0", zki.EthAddr3[1].String())
|
||||||
sendProofAndCheckResp(t, zki)
|
sendProofAndCheckResp(t, zki)
|
||||||
|
|||||||
@@ -31,7 +31,7 @@ func TestMain(m *testing.M) {
|
|||||||
os.Exit(exitVal)
|
os.Exit(exitVal)
|
||||||
}
|
}
|
||||||
|
|
||||||
const MaxTx = 376
|
const MaxTx = 352
|
||||||
const NLevels = 32
|
const NLevels = 32
|
||||||
const MaxL1Tx = 256
|
const MaxL1Tx = 256
|
||||||
const MaxFeeTx = 64
|
const MaxFeeTx = 64
|
||||||
@@ -61,6 +61,7 @@ func sendProofAndCheckResp(t *testing.T, zki *common.ZKInputs) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log.Infof("sending proof to %s", proofServerURL)
|
||||||
// Store zkinputs json for debugging purposes
|
// Store zkinputs json for debugging purposes
|
||||||
zkInputsJSON, err := json.Marshal(zki)
|
zkInputsJSON, err := json.Marshal(zki)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|||||||
@@ -501,11 +501,11 @@ func (tp *TxProcessor) ProcessL1Tx(exitTree *merkletree.MerkleTree, tx *common.L
|
|||||||
tp.zki.OnChain[tp.i] = big.NewInt(1)
|
tp.zki.OnChain[tp.i] = big.NewInt(1)
|
||||||
|
|
||||||
// L1Txs
|
// L1Txs
|
||||||
depositAmountF16, err := common.NewFloat16(tx.DepositAmount)
|
depositAmountF40, err := common.NewFloat40(tx.DepositAmount)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, nil, false, nil, tracerr.Wrap(err)
|
return nil, nil, false, nil, tracerr.Wrap(err)
|
||||||
}
|
}
|
||||||
tp.zki.DepositAmountF[tp.i] = big.NewInt(int64(depositAmountF16))
|
tp.zki.DepositAmountF[tp.i] = big.NewInt(int64(depositAmountF40))
|
||||||
tp.zki.FromEthAddr[tp.i] = common.EthAddrToBigInt(tx.FromEthAddr)
|
tp.zki.FromEthAddr[tp.i] = common.EthAddrToBigInt(tx.FromEthAddr)
|
||||||
if tx.FromBJJ != common.EmptyBJJComp {
|
if tx.FromBJJ != common.EmptyBJJComp {
|
||||||
tp.zki.FromBJJCompressed[tp.i] = BJJCompressedTo256BigInts(tx.FromBJJ)
|
tp.zki.FromBJJCompressed[tp.i] = BJJCompressedTo256BigInts(tx.FromBJJ)
|
||||||
@@ -515,6 +515,20 @@ func (tp *TxProcessor) ProcessL1Tx(exitTree *merkletree.MerkleTree, tx *common.L
|
|||||||
if tp.i < len(tp.zki.ISOnChain) { // len(tp.zki.ISOnChain) == nTx
|
if tp.i < len(tp.zki.ISOnChain) { // len(tp.zki.ISOnChain) == nTx
|
||||||
tp.zki.ISOnChain[tp.i] = big.NewInt(1)
|
tp.zki.ISOnChain[tp.i] = big.NewInt(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if tx.Type == common.TxTypeForceTransfer ||
|
||||||
|
tx.Type == common.TxTypeDepositTransfer ||
|
||||||
|
tx.Type == common.TxTypeCreateAccountDepositTransfer ||
|
||||||
|
tx.Type == common.TxTypeForceExit {
|
||||||
|
// in the cases where at L1Tx there is usage of the
|
||||||
|
// Amount parameter, add it at the ZKInputs.AmountF
|
||||||
|
// slot
|
||||||
|
amountF40, err := common.NewFloat40(tx.Amount)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, false, nil, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
tp.zki.AmountF[tp.i] = big.NewInt(int64(amountF40))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
switch tx.Type {
|
switch tx.Type {
|
||||||
@@ -657,6 +671,11 @@ func (tp *TxProcessor) ProcessL2Tx(coordIdxsMap map[common.TokenID]common.Idx,
|
|||||||
tp.zki.ToEthAddr[tp.i] = common.EthAddrToBigInt(tx.ToEthAddr)
|
tp.zki.ToEthAddr[tp.i] = common.EthAddrToBigInt(tx.ToEthAddr)
|
||||||
|
|
||||||
tp.zki.OnChain[tp.i] = big.NewInt(0)
|
tp.zki.OnChain[tp.i] = big.NewInt(0)
|
||||||
|
amountF40, err := common.NewFloat40(tx.Amount)
|
||||||
|
if err != nil {
|
||||||
|
return nil, nil, false, tracerr.Wrap(err)
|
||||||
|
}
|
||||||
|
tp.zki.AmountF[tp.i] = big.NewInt(int64(amountF40))
|
||||||
tp.zki.NewAccount[tp.i] = big.NewInt(0)
|
tp.zki.NewAccount[tp.i] = big.NewInt(0)
|
||||||
|
|
||||||
// L2Txs
|
// L2Txs
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
package txprocessor
|
package txprocessor
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/binary"
|
|
||||||
"encoding/hex"
|
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"math/big"
|
"math/big"
|
||||||
"os"
|
"os"
|
||||||
@@ -642,17 +640,16 @@ func TestCreateAccountDepositMaxValue(t *testing.T) {
|
|||||||
|
|
||||||
users := txsets.GenerateJsUsers(t)
|
users := txsets.GenerateJsUsers(t)
|
||||||
|
|
||||||
daMaxHex, err := hex.DecodeString("FFFF")
|
daMaxF40 := common.Float40(0xFFFFFFFFFF)
|
||||||
|
daMaxBI, err := daMaxF40.BigInt()
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
daMaxF16 := common.Float16(binary.BigEndian.Uint16(daMaxHex))
|
assert.Equal(t, "343597383670000000000000000000000000000000", daMaxBI.String())
|
||||||
daMaxBI := daMaxF16.BigInt()
|
|
||||||
assert.Equal(t, "10235000000000000000000000000000000", daMaxBI.String())
|
|
||||||
|
|
||||||
daMax1Hex, err := hex.DecodeString("FFFE")
|
daMax1F40 := common.Float40(0xFFFFFFFFFE)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
daMax1F16 := common.Float16(binary.BigEndian.Uint16(daMax1Hex))
|
daMax1BI, err := daMax1F40.BigInt()
|
||||||
daMax1BI := daMax1F16.BigInt()
|
require.NoError(t, err)
|
||||||
assert.Equal(t, "10225000000000000000000000000000000", daMax1BI.String())
|
assert.Equal(t, "343597383660000000000000000000000000000000", daMax1BI.String())
|
||||||
|
|
||||||
l1Txs := []common.L1Tx{
|
l1Txs := []common.L1Tx{
|
||||||
{
|
{
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
@@ -3,7 +3,6 @@ package txselector
|
|||||||
// current: very simple version of TxSelector
|
// current: very simple version of TxSelector
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"math/big"
|
"math/big"
|
||||||
"sort"
|
"sort"
|
||||||
@@ -236,7 +235,8 @@ func (txsel *TxSelector) GetL1L2TxSelection(selectionConfig *SelectionConfig,
|
|||||||
if len(l1CoordinatorTxs) >= int(selectionConfig.MaxL1UserTxs)-len(l1UserTxs) {
|
if len(l1CoordinatorTxs) >= int(selectionConfig.MaxL1UserTxs)-len(l1UserTxs) {
|
||||||
// discard L2Tx, and update Info parameter of
|
// discard L2Tx, and update Info parameter of
|
||||||
// the tx, and add it to the discardedTxs array
|
// the tx, and add it to the discardedTxs array
|
||||||
l2Txs0[i].Info = "Tx not selected due the L2Tx depends on a L1CoordinatorTx and there is not enough space for L1Coordinator"
|
l2Txs0[i].Info = "Tx not selected because the L2Tx depends on a " +
|
||||||
|
"L1CoordinatorTx and there is not enough space for L1Coordinator"
|
||||||
discardedL2Txs = append(discardedL2Txs, l2Txs0[i])
|
discardedL2Txs = append(discardedL2Txs, l2Txs0[i])
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@@ -261,7 +261,9 @@ func (txsel *TxSelector) GetL1L2TxSelection(selectionConfig *SelectionConfig,
|
|||||||
// not valid Amount with current Balance. Discard L2Tx,
|
// not valid Amount with current Balance. Discard L2Tx,
|
||||||
// and update Info parameter of the tx, and add it to
|
// and update Info parameter of the tx, and add it to
|
||||||
// the discardedTxs array
|
// the discardedTxs array
|
||||||
l2Txs[i].Info = fmt.Sprintf("Tx not selected due not enough Balance at the sender. Current sender account Balance: %s, Amount+Fee: %s", balance.String(), feeAndAmount.String())
|
l2Txs[i].Info = fmt.Sprintf("Tx not selected due to not enough Balance at the sender. "+
|
||||||
|
"Current sender account Balance: %s, Amount+Fee: %s",
|
||||||
|
balance.String(), feeAndAmount.String())
|
||||||
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@@ -273,7 +275,8 @@ func (txsel *TxSelector) GetL1L2TxSelection(selectionConfig *SelectionConfig,
|
|||||||
// not valid Nonce at tx. Discard L2Tx, and update Info
|
// not valid Nonce at tx. Discard L2Tx, and update Info
|
||||||
// parameter of the tx, and add it to the discardedTxs
|
// parameter of the tx, and add it to the discardedTxs
|
||||||
// array
|
// array
|
||||||
l2Txs[i].Info = fmt.Sprintf("Tx not selected due not current Nonce. Tx.Nonce: %d, Account.Nonce: %d", l2Txs[i].Nonce, nonce)
|
l2Txs[i].Info = fmt.Sprintf("Tx not selected due to not current Nonce. "+
|
||||||
|
"Tx.Nonce: %d, Account.Nonce: %d", l2Txs[i].Nonce, nonce)
|
||||||
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@@ -291,18 +294,31 @@ func (txsel *TxSelector) GetL1L2TxSelection(selectionConfig *SelectionConfig,
|
|||||||
txsel.processTxToEthAddrBJJ(validTxs, selectionConfig,
|
txsel.processTxToEthAddrBJJ(validTxs, selectionConfig,
|
||||||
len(l1UserTxs), l1CoordinatorTxs, positionL1, l2Txs[i])
|
len(l1UserTxs), l1CoordinatorTxs, positionL1, l2Txs[i])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Debug(err)
|
log.Debugw("txsel.processTxToEthAddrBJJ", "err", err)
|
||||||
// Discard L2Tx, and update Info parameter of
|
// Discard L2Tx, and update Info parameter of
|
||||||
// the tx, and add it to the discardedTxs array
|
// the tx, and add it to the discardedTxs array
|
||||||
l2Txs[i].Info = fmt.Sprintf("Tx not selected (in processTxToEthAddrBJJ) due %s", err.Error())
|
l2Txs[i].Info = fmt.Sprintf("Tx not selected (in processTxToEthAddrBJJ) due to %s",
|
||||||
|
err.Error())
|
||||||
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
if accAuth != nil && l1CoordinatorTx != nil {
|
if l1CoordinatorTx != nil {
|
||||||
|
// If ToEthAddr == 0xff.. this means that we
|
||||||
|
// are handling a TransferToBJJ, which doesn't
|
||||||
|
// require an authorization because it doesn't
|
||||||
|
// contain a valid ethereum address.
|
||||||
|
// Otherwise only create the account if we have
|
||||||
|
// the corresponding authorization
|
||||||
|
if validL2Tx.ToEthAddr == common.FFAddr {
|
||||||
|
accAuths = append(accAuths, common.EmptyEthSignature)
|
||||||
|
l1CoordinatorTxs = append(l1CoordinatorTxs, *l1CoordinatorTx)
|
||||||
|
positionL1++
|
||||||
|
} else if accAuth != nil {
|
||||||
accAuths = append(accAuths, accAuth.Signature)
|
accAuths = append(accAuths, accAuth.Signature)
|
||||||
l1CoordinatorTxs = append(l1CoordinatorTxs, *l1CoordinatorTx)
|
l1CoordinatorTxs = append(l1CoordinatorTxs, *l1CoordinatorTx)
|
||||||
positionL1++
|
positionL1++
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if validL2Tx != nil {
|
if validL2Tx != nil {
|
||||||
validTxs = append(validTxs, *validL2Tx)
|
validTxs = append(validTxs, *validL2Tx)
|
||||||
}
|
}
|
||||||
@@ -314,8 +330,8 @@ func (txsel *TxSelector) GetL1L2TxSelection(selectionConfig *SelectionConfig,
|
|||||||
"ToIdx", l2Txs[i].ToIdx)
|
"ToIdx", l2Txs[i].ToIdx)
|
||||||
// Discard L2Tx, and update Info parameter of
|
// Discard L2Tx, and update Info parameter of
|
||||||
// the tx, and add it to the discardedTxs array
|
// the tx, and add it to the discardedTxs array
|
||||||
l2Txs[i].Info = fmt.Sprintf("Tx not selected due tx.ToIdx not found in StateDB. ToIdx: %d",
|
l2Txs[i].Info = fmt.Sprintf("Tx not selected due to tx.ToIdx not found in StateDB. "+
|
||||||
l2Txs[i].ToIdx)
|
"ToIdx: %d", l2Txs[i].ToIdx)
|
||||||
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@@ -327,7 +343,9 @@ func (txsel *TxSelector) GetL1L2TxSelection(selectionConfig *SelectionConfig,
|
|||||||
// Discard L2Tx, and update Info
|
// Discard L2Tx, and update Info
|
||||||
// parameter of the tx, and add it to
|
// parameter of the tx, and add it to
|
||||||
// the discardedTxs array
|
// the discardedTxs array
|
||||||
l2Txs[i].Info = fmt.Sprintf("Tx not selected due ToEthAddr does not correspond to the Account.EthAddr. tx.ToIdx: %d, tx.ToEthAddr: %s, account.EthAddr: %s",
|
l2Txs[i].Info = fmt.Sprintf("Tx not selected because ToEthAddr "+
|
||||||
|
"does not correspond to the Account.EthAddr. "+
|
||||||
|
"tx.ToIdx: %d, tx.ToEthAddr: %s, account.EthAddr: %s",
|
||||||
l2Txs[i].ToIdx, l2Txs[i].ToEthAddr, receiverAcc.EthAddr)
|
l2Txs[i].ToIdx, l2Txs[i].ToEthAddr, receiverAcc.EthAddr)
|
||||||
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
||||||
continue
|
continue
|
||||||
@@ -341,7 +359,9 @@ func (txsel *TxSelector) GetL1L2TxSelection(selectionConfig *SelectionConfig,
|
|||||||
// Discard L2Tx, and update Info
|
// Discard L2Tx, and update Info
|
||||||
// parameter of the tx, and add it to
|
// parameter of the tx, and add it to
|
||||||
// the discardedTxs array
|
// the discardedTxs array
|
||||||
l2Txs[i].Info = fmt.Sprintf("Tx not selected due tx.ToBJJ does not correspond to the Account.BJJ. tx.ToIdx: %d, tx.ToEthAddr: %s, tx.ToBJJ: %s, account.BJJ: %s",
|
l2Txs[i].Info = fmt.Sprintf("Tx not selected because tx.ToBJJ "+
|
||||||
|
"does not correspond to the Account.BJJ. "+
|
||||||
|
"tx.ToIdx: %d, tx.ToEthAddr: %s, tx.ToBJJ: %s, account.BJJ: %s",
|
||||||
l2Txs[i].ToIdx, l2Txs[i].ToEthAddr, l2Txs[i].ToBJJ, receiverAcc.BJJ)
|
l2Txs[i].ToIdx, l2Txs[i].ToEthAddr, l2Txs[i].ToBJJ, receiverAcc.BJJ)
|
||||||
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
discardedL2Txs = append(discardedL2Txs, l2Txs[i])
|
||||||
continue
|
continue
|
||||||
@@ -415,7 +435,7 @@ func (txsel *TxSelector) GetL1L2TxSelection(selectionConfig *SelectionConfig,
|
|||||||
log.Error(err)
|
log.Error(err)
|
||||||
// Discard L2Tx, and update Info parameter of the tx,
|
// Discard L2Tx, and update Info parameter of the tx,
|
||||||
// and add it to the discardedTxs array
|
// and add it to the discardedTxs array
|
||||||
selectedL2Txs[i].Info = fmt.Sprintf("Tx not selected (in ProcessL2Tx) due %s", err.Error())
|
selectedL2Txs[i].Info = fmt.Sprintf("Tx not selected (in ProcessL2Tx) due to %s", err.Error())
|
||||||
discardedL2Txs = append(discardedL2Txs, selectedL2Txs[i])
|
discardedL2Txs = append(discardedL2Txs, selectedL2Txs[i])
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
@@ -471,8 +491,7 @@ func (txsel *TxSelector) processTxToEthAddrBJJ(validTxs []common.PoolL2Tx,
|
|||||||
|
|
||||||
var l1CoordinatorTx *common.L1Tx
|
var l1CoordinatorTx *common.L1Tx
|
||||||
var accAuth *common.AccountCreationAuth
|
var accAuth *common.AccountCreationAuth
|
||||||
if !bytes.Equal(l2Tx.ToEthAddr.Bytes(), common.EmptyAddr.Bytes()) &&
|
if l2Tx.ToEthAddr != common.EmptyAddr && l2Tx.ToEthAddr != common.FFAddr {
|
||||||
!bytes.Equal(l2Tx.ToEthAddr.Bytes(), common.FFAddr.Bytes()) {
|
|
||||||
// case: ToEthAddr != 0x00 neither 0xff
|
// case: ToEthAddr != 0x00 neither 0xff
|
||||||
if l2Tx.ToBJJ != common.EmptyBJJComp {
|
if l2Tx.ToBJJ != common.EmptyBJJComp {
|
||||||
// case: ToBJJ!=0:
|
// case: ToBJJ!=0:
|
||||||
@@ -528,8 +547,7 @@ func (txsel *TxSelector) processTxToEthAddrBJJ(validTxs []common.PoolL2Tx,
|
|||||||
DepositAmount: big.NewInt(0),
|
DepositAmount: big.NewInt(0),
|
||||||
Type: common.TxTypeCreateAccountDeposit,
|
Type: common.TxTypeCreateAccountDeposit,
|
||||||
}
|
}
|
||||||
} else if bytes.Equal(l2Tx.ToEthAddr.Bytes(), common.FFAddr.Bytes()) &&
|
} else if l2Tx.ToEthAddr == common.FFAddr && l2Tx.ToBJJ != common.EmptyBJJComp {
|
||||||
l2Tx.ToBJJ != common.EmptyBJJComp {
|
|
||||||
// if idx exist for EthAddr&BJJ use it
|
// if idx exist for EthAddr&BJJ use it
|
||||||
_, err := txsel.localAccountsDB.GetIdxByEthAddrBJJ(l2Tx.ToEthAddr, l2Tx.ToBJJ,
|
_, err := txsel.localAccountsDB.GetIdxByEthAddrBJJ(l2Tx.ToEthAddr, l2Tx.ToBJJ,
|
||||||
l2Tx.TokenID)
|
l2Tx.TokenID)
|
||||||
@@ -555,7 +573,8 @@ func (txsel *TxSelector) processTxToEthAddrBJJ(validTxs []common.PoolL2Tx,
|
|||||||
}
|
}
|
||||||
if len(l1CoordinatorTxs) >= int(selectionConfig.MaxL1UserTxs)-nL1UserTxs {
|
if len(l1CoordinatorTxs) >= int(selectionConfig.MaxL1UserTxs)-nL1UserTxs {
|
||||||
// L2Tx discarded
|
// L2Tx discarded
|
||||||
return nil, nil, nil, tracerr.Wrap(fmt.Errorf("L2Tx discarded due not slots for L1CoordinatorTx to create a new account for receiver of L2Tx"))
|
return nil, nil, nil, tracerr.Wrap(fmt.Errorf("L2Tx discarded due to no available slots " +
|
||||||
|
"for L1CoordinatorTx to create a new account for receiver of L2Tx"))
|
||||||
}
|
}
|
||||||
|
|
||||||
return &l2Tx, l1CoordinatorTx, accAuth, nil
|
return &l2Tx, l1CoordinatorTx, accAuth, nil
|
||||||
@@ -564,7 +583,7 @@ func (txsel *TxSelector) processTxToEthAddrBJJ(validTxs []common.PoolL2Tx,
|
|||||||
func checkAlreadyPendingToCreate(l1CoordinatorTxs []common.L1Tx, tokenID common.TokenID,
|
func checkAlreadyPendingToCreate(l1CoordinatorTxs []common.L1Tx, tokenID common.TokenID,
|
||||||
addr ethCommon.Address, bjj babyjub.PublicKeyComp) bool {
|
addr ethCommon.Address, bjj babyjub.PublicKeyComp) bool {
|
||||||
for i := 0; i < len(l1CoordinatorTxs); i++ {
|
for i := 0; i < len(l1CoordinatorTxs); i++ {
|
||||||
if bytes.Equal(l1CoordinatorTxs[i].FromEthAddr.Bytes(), addr.Bytes()) &&
|
if l1CoordinatorTxs[i].FromEthAddr == addr &&
|
||||||
l1CoordinatorTxs[i].TokenID == tokenID &&
|
l1CoordinatorTxs[i].TokenID == tokenID &&
|
||||||
l1CoordinatorTxs[i].FromBJJ == bjj {
|
l1CoordinatorTxs[i].FromBJJ == bjj {
|
||||||
return true
|
return true
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ func initTest(t *testing.T, chainID uint16, hermezContractAddr ethCommon.Address
|
|||||||
BJJ: coordUser.BJJ.Public().Compress(),
|
BJJ: coordUser.BJJ.Public().Compress(),
|
||||||
AccountCreationAuth: nil,
|
AccountCreationAuth: nil,
|
||||||
}
|
}
|
||||||
fmt.Printf("%v", coordAccount)
|
// fmt.Printf("%v\n", coordAccount)
|
||||||
auth := common.AccountCreationAuth{
|
auth := common.AccountCreationAuth{
|
||||||
EthAddr: coordUser.Addr,
|
EthAddr: coordUser.Addr,
|
||||||
BJJ: coordUser.BJJ.Public().Compress(),
|
BJJ: coordUser.BJJ.Public().Compress(),
|
||||||
@@ -424,9 +424,9 @@ func TestPoolL2TxsWithoutEnoughBalance(t *testing.T) {
|
|||||||
_, _, _, _, _, _, err = txsel.GetL1L2TxSelection(selectionConfig, l1UserTxs)
|
_, _, _, _, _, _, err = txsel.GetL1L2TxSelection(selectionConfig, l1UserTxs)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
|
||||||
expectedTxID0 := "0x0248bae02b5c8c3847d312bfac3a33ae790616e888f2f711f22aeaff007cde92c2" // 1st TransferToEthAddr
|
expectedTxID0 := "0x028847b86613c0b70be18c8622119ed045b42e4e47d7938fa90bb3d1dc14928965" // 1st TransferToEthAddr
|
||||||
expectedTxID1 := "0x0249af018311a393c337ab9174ca2466cba489e49942b4ca4e5c530903671c4aef" // 1st Exit
|
expectedTxID1 := "0x0200b18773dcf56f770d65870fb02041cb59a088fd35b7c3f3df69f8a250b99a42" // 1st Exit
|
||||||
expectedTxID2 := "0x0228b93a261a0cdc62f35588c03bd179d31a0807c28afffdb6a7aaf0c4f017e4cf" // 2nd TransferToEthAddr
|
expectedTxID2 := "0x029720ff506153f970f120ac638cd7ee759eeff2c2012e7634a78e4fdc05c04a90" // 2nd TransferToEthAddr
|
||||||
|
|
||||||
// batch2
|
// batch2
|
||||||
// prepare the PoolL2Txs
|
// prepare the PoolL2Txs
|
||||||
@@ -497,3 +497,134 @@ func TestPoolL2TxsWithoutEnoughBalance(t *testing.T) {
|
|||||||
err = txsel.l2db.StartForging(common.TxIDsFromPoolL2Txs(oL2Txs), txsel.localAccountsDB.CurrentBatch())
|
err = txsel.l2db.StartForging(common.TxIDsFromPoolL2Txs(oL2Txs), txsel.localAccountsDB.CurrentBatch())
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestTransferToBjj(t *testing.T) {
|
||||||
|
set := `
|
||||||
|
Type: Blockchain
|
||||||
|
AddToken(1)
|
||||||
|
|
||||||
|
CreateAccountDeposit(0) Coord: 0
|
||||||
|
CreateAccountDeposit(0) A: 1000
|
||||||
|
CreateAccountDeposit(0) B: 1000
|
||||||
|
CreateAccountDeposit(1) B: 1000
|
||||||
|
|
||||||
|
> batchL1 // freeze L1User{1}
|
||||||
|
> batchL1 // forge L1User{1}
|
||||||
|
> block
|
||||||
|
`
|
||||||
|
|
||||||
|
chainID := uint16(0)
|
||||||
|
tc := til.NewContext(chainID, common.RollupConstMaxL1UserTx)
|
||||||
|
blocks, err := tc.GenerateBlocks(set)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
|
||||||
|
hermezContractAddr := ethCommon.HexToAddress("0xc344E203a046Da13b0B4467EB7B3629D0C99F6E6")
|
||||||
|
txsel := initTest(t, chainID, hermezContractAddr, tc.Users["Coord"])
|
||||||
|
|
||||||
|
// restart nonces of TilContext, as will be set by generating directly
|
||||||
|
// the PoolL2Txs for each specific batch with tc.GeneratePoolL2Txs
|
||||||
|
tc.RestartNonces()
|
||||||
|
|
||||||
|
addTokens(t, tc, txsel.l2db.DB())
|
||||||
|
|
||||||
|
tpc := txprocessor.Config{
|
||||||
|
NLevels: 16,
|
||||||
|
MaxFeeTx: 10,
|
||||||
|
MaxTx: 20,
|
||||||
|
MaxL1Tx: 10,
|
||||||
|
ChainID: chainID,
|
||||||
|
}
|
||||||
|
selectionConfig := &SelectionConfig{
|
||||||
|
MaxL1UserTxs: 5,
|
||||||
|
TxProcessorConfig: tpc,
|
||||||
|
}
|
||||||
|
// batch1 to create some accounts with positive balance
|
||||||
|
l1UserTxs := []common.L1Tx{}
|
||||||
|
_, _, _, _, _, _, err = txsel.GetL1L2TxSelection(selectionConfig, l1UserTxs)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
// Transfer is ToBJJ to a BJJ-only account that doesn't exist
|
||||||
|
// and the coordinator will create it via L1CoordTx.
|
||||||
|
|
||||||
|
batchPoolL2 := `
|
||||||
|
Type: PoolL2
|
||||||
|
PoolTransferToBJJ(0) A-B: 50 (126)
|
||||||
|
`
|
||||||
|
poolL2Txs, err := tc.GeneratePoolL2Txs(batchPoolL2)
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
// add the PoolL2Txs to the l2DB
|
||||||
|
addL2Txs(t, txsel, poolL2Txs)
|
||||||
|
|
||||||
|
l1UserTxs = til.L1TxsToCommonL1Txs(tc.Queues[*blocks[0].Rollup.Batches[1].Batch.ForgeL1TxsNum])
|
||||||
|
_, _, oL1UserTxs, oL1CoordTxs, oL2Txs, discardedL2Txs, err := txsel.GetL1L2TxSelection(selectionConfig, l1UserTxs)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, 4, len(oL1UserTxs))
|
||||||
|
// We expect the coordinator to add an L1CoordTx to create an account for the recipient of the l2tx
|
||||||
|
require.Equal(t, 1, len(oL1CoordTxs))
|
||||||
|
assert.Equal(t, poolL2Txs[0].ToEthAddr, oL1CoordTxs[0].FromEthAddr)
|
||||||
|
assert.Equal(t, poolL2Txs[0].ToBJJ, oL1CoordTxs[0].FromBJJ)
|
||||||
|
// fmt.Printf("DBG l1CoordTx[0]: %+v\n", oL1CoordTxs[0])
|
||||||
|
assert.Equal(t, 1, len(oL2Txs))
|
||||||
|
assert.Equal(t, 0, len(discardedL2Txs))
|
||||||
|
err = txsel.l2db.StartForging(common.TxIDsFromPoolL2Txs(oL2Txs), txsel.localAccountsDB.CurrentBatch())
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
// Now the BJJ-only account for B is already created, so the transfer
|
||||||
|
// happens without an L1CoordTx that creates the user account.
|
||||||
|
|
||||||
|
batchPoolL2 = `
|
||||||
|
Type: PoolL2
|
||||||
|
PoolTransferToBJJ(0) A-B: 50 (126)
|
||||||
|
`
|
||||||
|
|
||||||
|
poolL2Txs, err = tc.GeneratePoolL2Txs(batchPoolL2)
|
||||||
|
require.NoError(t, err)
|
||||||
|
addL2Txs(t, txsel, poolL2Txs)
|
||||||
|
|
||||||
|
l1UserTxs = []common.L1Tx{}
|
||||||
|
_, _, oL1UserTxs, oL1CoordTxs, oL2Txs, discardedL2Txs, err = txsel.GetL1L2TxSelection(selectionConfig, l1UserTxs)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, 0, len(oL1UserTxs))
|
||||||
|
// Since the BJJ-only account B already exists, the coordinator doesn't add any L1CoordTxs
|
||||||
|
assert.Equal(t, 0, len(oL1CoordTxs))
|
||||||
|
assert.Equal(t, 1, len(oL2Txs))
|
||||||
|
assert.Equal(t, 0, len(discardedL2Txs))
|
||||||
|
err = txsel.l2db.StartForging(common.TxIDsFromPoolL2Txs(oL2Txs), txsel.localAccountsDB.CurrentBatch())
|
||||||
|
require.NoError(t, err)
|
||||||
|
|
||||||
|
// The transfer now is ToBJJ to a BJJ-only account that doesn't exist
|
||||||
|
// and the coordinator will create it via L1CoordTx. Since it's a
|
||||||
|
// transfer of a token for which the coordinator doesn't have a fee
|
||||||
|
// account, another L1CoordTx will be created for the coordinator to
|
||||||
|
// receive the fees.
|
||||||
|
|
||||||
|
batchPoolL2 = `
|
||||||
|
Type: PoolL2
|
||||||
|
PoolTransferToBJJ(1) B-A: 50 (126)
|
||||||
|
`
|
||||||
|
|
||||||
|
poolL2Txs, err = tc.GeneratePoolL2Txs(batchPoolL2)
|
||||||
|
require.NoError(t, err)
|
||||||
|
addL2Txs(t, txsel, poolL2Txs)
|
||||||
|
|
||||||
|
l1UserTxs = []common.L1Tx{}
|
||||||
|
_, _, oL1UserTxs, oL1CoordTxs, oL2Txs, discardedL2Txs, err = txsel.GetL1L2TxSelection(selectionConfig, l1UserTxs)
|
||||||
|
require.NoError(t, err)
|
||||||
|
assert.Equal(t, 0, len(oL1UserTxs))
|
||||||
|
// We expect the coordinator to add an L1CoordTx to create an account
|
||||||
|
// to receive the fees by the coordinator and another one for the
|
||||||
|
// recipient of the l2tx
|
||||||
|
assert.Equal(t, 2, len(oL1CoordTxs))
|
||||||
|
// [0] Coordinator account cration for token 1
|
||||||
|
assert.Equal(t, tc.Users["Coord"].Addr, oL1CoordTxs[0].FromEthAddr)
|
||||||
|
// [1] User A BJJ-only account creation for token 1
|
||||||
|
assert.Equal(t, poolL2Txs[0].ToEthAddr, oL1CoordTxs[1].FromEthAddr)
|
||||||
|
assert.Equal(t, poolL2Txs[0].ToBJJ, oL1CoordTxs[1].FromBJJ)
|
||||||
|
assert.Equal(t, common.TokenID(1), oL1CoordTxs[1].TokenID)
|
||||||
|
|
||||||
|
assert.Equal(t, 1, len(oL2Txs))
|
||||||
|
assert.Equal(t, 0, len(discardedL2Txs))
|
||||||
|
err = txsel.l2db.StartForging(common.TxIDsFromPoolL2Txs(oL2Txs), txsel.localAccountsDB.CurrentBatch())
|
||||||
|
require.NoError(t, err)
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user