From 20cc13e5bf271010fdc641cf622dd1474134c169 Mon Sep 17 00:00:00 2001 From: ValarDragon Date: Tue, 16 Oct 2018 09:29:32 -0700 Subject: [PATCH 1/4] types: Dec.MarshalJSON now marshals as a normal decimal string Closes #2475 --- PENDING.md | 2 ++ types/decimal.go | 33 +++++++++++++++++++++++++++++---- types/decimal_test.go | 40 ++++++++++++++++++++++++++++++++++++++++ 3 files changed, 71 insertions(+), 4 deletions(-) diff --git a/PENDING.md b/PENDING.md index eb61a5468..2f0f9c2d7 100644 --- a/PENDING.md +++ b/PENDING.md @@ -72,6 +72,8 @@ BREAKING CHANGES * [x/stake] \#2412 Added an unbonding validator queue to EndBlock to automatically update validator.Status when finished Unbonding * [x/stake] \#2500 Block conflicting redelegations until we add an index * [x/params] Global Paramstore refactored + * [types] \#2506 sdk.Dec MarshalJSON now marshals as a normal Decimal, with 10 digits of decimal precision + * Tendermint * Update tendermint version from v0.23.0 to v0.25.0, notable changes diff --git a/types/decimal.go b/types/decimal.go index e9623995f..a3cd7cc0e 100644 --- a/types/decimal.go +++ b/types/decimal.go @@ -407,17 +407,36 @@ func (d *Dec) UnmarshalAmino(text string) (err error) { return nil } -// MarshalJSON defines custom encoding scheme +// MarshalJSON marshals the decimal func (d Dec) MarshalJSON() ([]byte, error) { if d.Int == nil { return nilJSON, nil } - bz, err := d.Int.MarshalText() if err != nil { return nil, err } - return json.Marshal(string(bz)) + var bzWDec []byte + // TODO: Remove trailing zeros + // case 1, pure decimal + if len(bz) <= 10 { + bzWDec = make([]byte, 12) + // 0. prefix + bzWDec[0] = byte('0') + bzWDec[1] = byte('.') + // set relevant digits to 0 + for i := 0; i < 10-len(bz); i++ { + bzWDec[i+2] = byte('0') + } + // set last few digits + copy(bzWDec[2+(10-len(bz)):], bz) + } else { + bzWDec = make([]byte, len(bz)+1) + copy(bzWDec, bz[:len(bz)-10]) + bzWDec[len(bz)-10] = byte('.') + copy(bzWDec[len(bz)-9:], bz[len(bz)-10:]) + } + return json.Marshal(string(bzWDec)) } // UnmarshalJSON defines custom decoding scheme @@ -431,7 +450,13 @@ func (d *Dec) UnmarshalJSON(bz []byte) error { if err != nil { return err } - return d.Int.UnmarshalText([]byte(text)) + // TODO: Reuse dec allocation + newDec, err := NewDecFromStr(text) + if err != nil { + return err + } + d.Int = newDec.Int + return nil } //___________________________________________________________________________________ diff --git a/types/decimal_test.go b/types/decimal_test.go index a6ec0740e..73a00f60a 100644 --- a/types/decimal_test.go +++ b/types/decimal_test.go @@ -4,6 +4,8 @@ import ( "math/big" "testing" + "github.com/stretchr/testify/assert" + "github.com/cosmos/cosmos-sdk/codec" "github.com/stretchr/testify/require" ) @@ -248,6 +250,44 @@ func TestToLeftPadded(t *testing.T) { var cdc = codec.New() +func TestDecMarshalJSON(t *testing.T) { + decimal := func(i int64) Dec { + d := NewDec(0) + d.Int = new(big.Int).SetInt64(i) + return d + } + tests := []struct { + name string + d Dec + want string + wantErr bool // if wantErr = false, will also attempt unmarshaling + }{ + {"zero", decimal(0), "\"0.0000000000\"", false}, + {"one", decimal(1), "\"0.0000000001\"", false}, + {"ten", decimal(10), "\"0.0000000010\"", false}, + {"12340", decimal(12340), "\"0.0000012340\"", false}, + {"zeroInt", NewDec(0), "\"0.0000000000\"", false}, + {"oneInt", NewDec(1), "\"1.0000000000\"", false}, + {"tenInt", NewDec(10), "\"10.0000000000\"", false}, + {"12340Int", NewDec(12340), "\"12340.0000000000\"", false}, + } + for _, tt := range tests { + t.Run(tt.name, func(t *testing.T) { + got, err := tt.d.MarshalJSON() + if (err != nil) != tt.wantErr { + t.Errorf("Dec.MarshalJSON() error = %v, wantErr %v", err, tt.wantErr) + return + } + if !tt.wantErr { + assert.Equal(t, tt.want, string(got), "incorrect marshalled value") + unmarshalledDec := NewDec(0) + unmarshalledDec.UnmarshalJSON(got) + assert.Equal(t, tt.d, unmarshalledDec, "incorrect unmarshalled value") + } + }) + } +} + func TestZeroDeserializationJSON(t *testing.T) { d := Dec{new(big.Int)} err := cdc.UnmarshalJSON([]byte(`"0"`), &d) From 8d59b51dfe94664641746cb6734a0f6fcd528310 Mon Sep 17 00:00:00 2001 From: ValarDragon Date: Tue, 16 Oct 2018 21:49:29 -0700 Subject: [PATCH 2/4] fix the weird non-alphanumeric key issue --- x/distribution/keeper/key.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/x/distribution/keeper/key.go b/x/distribution/keeper/key.go index 2e5989081..d139f02b8 100644 --- a/x/distribution/keeper/key.go +++ b/x/distribution/keeper/key.go @@ -13,9 +13,9 @@ var ( ProposerKey = []byte{0x04} // key for storing the proposer operator address // params store - ParamStoreKeyCommunityTax = []byte("community-tax") - ParamStoreKeyBaseProposerReward = []byte("base-proposer-reward") - ParamStoreKeyBonusProposerReward = []byte("bonus-proposer-reward") + ParamStoreKeyCommunityTax = []byte("CommunityTax") + ParamStoreKeyBaseProposerReward = []byte("BaseProposerReward") + ParamStoreKeyBonusProposerReward = []byte("BonusProposerReward") ) const ( From c2d68928e7f7f9032f68282d118656d9507cd8d0 Mon Sep 17 00:00:00 2001 From: ValarDragon Date: Sat, 20 Oct 2018 13:32:43 -0700 Subject: [PATCH 3/4] Switch to new Decimal.String() implementation --- types/decimal.go | 74 ++++++++++++++++--------------------------- types/decimal_test.go | 18 ----------- 2 files changed, 27 insertions(+), 65 deletions(-) diff --git a/types/decimal.go b/types/decimal.go index a3cd7cc0e..b621acd51 100644 --- a/types/decimal.go +++ b/types/decimal.go @@ -248,29 +248,32 @@ func (d Dec) QuoInt(i Int) Dec { } func (d Dec) String() string { - str := d.ToLeftPaddedWithDecimals(Precision) - placement := len(str) - Precision - if placement < 0 { - panic("too few decimal digits") + bz, err := d.Int.MarshalText() + if err != nil { + return "" } - return str[:placement] + "." + str[placement:] -} - -// TODO panic if negative or if totalDigits < len(initStr)??? -// evaluate as an integer and return left padded string -func (d Dec) ToLeftPaddedWithDecimals(totalDigits int8) string { - intStr := d.Int.String() - fcode := `%0` + strconv.Itoa(int(totalDigits)) + `s` - return fmt.Sprintf(fcode, intStr) -} - -// TODO panic if negative or if totalDigits < len(initStr)??? -// evaluate as an integer and return left padded string -func (d Dec) ToLeftPadded(totalDigits int8) string { - chopped := chopPrecisionAndRoundNonMutative(d.Int) - intStr := chopped.String() - fcode := `%0` + strconv.Itoa(int(totalDigits)) + `s` - return fmt.Sprintf(fcode, intStr) + var bzWDec []byte + // TODO: Remove trailing zeros + // case 1, purely decimal + if len(bz) <= 10 { + bzWDec = make([]byte, 12) + // 0. prefix + bzWDec[0] = byte('0') + bzWDec[1] = byte('.') + // set relevant digits to 0 + for i := 0; i < 10-len(bz); i++ { + bzWDec[i+2] = byte('0') + } + // set last few digits + copy(bzWDec[2+(10-len(bz)):], bz) + } else { + // len(bz) + 1 to account for the decimal point that is being added + bzWDec = make([]byte, len(bz)+1) + copy(bzWDec, bz[:len(bz)-10]) + bzWDec[len(bz)-10] = byte('.') + copy(bzWDec[len(bz)-9:], bz[len(bz)-10:]) + } + return string(bzWDec) } // ____ @@ -412,31 +415,8 @@ func (d Dec) MarshalJSON() ([]byte, error) { if d.Int == nil { return nilJSON, nil } - bz, err := d.Int.MarshalText() - if err != nil { - return nil, err - } - var bzWDec []byte - // TODO: Remove trailing zeros - // case 1, pure decimal - if len(bz) <= 10 { - bzWDec = make([]byte, 12) - // 0. prefix - bzWDec[0] = byte('0') - bzWDec[1] = byte('.') - // set relevant digits to 0 - for i := 0; i < 10-len(bz); i++ { - bzWDec[i+2] = byte('0') - } - // set last few digits - copy(bzWDec[2+(10-len(bz)):], bz) - } else { - bzWDec = make([]byte, len(bz)+1) - copy(bzWDec, bz[:len(bz)-10]) - bzWDec[len(bz)-10] = byte('.') - copy(bzWDec[len(bz)-9:], bz[len(bz)-10:]) - } - return json.Marshal(string(bzWDec)) + + return json.Marshal(d.String()) } // UnmarshalJSON defines custom decoding scheme diff --git a/types/decimal_test.go b/types/decimal_test.go index 73a00f60a..07329c7dc 100644 --- a/types/decimal_test.go +++ b/types/decimal_test.go @@ -230,24 +230,6 @@ func TestTruncate(t *testing.T) { } } -func TestToLeftPadded(t *testing.T) { - tests := []struct { - dec Dec - digits int8 - exp string - }{ - {mustNewDecFromStr(t, "33.3"), 8, "00000033"}, - {mustNewDecFromStr(t, "50"), 8, "00000050"}, - {mustNewDecFromStr(t, "333"), 8, "00000333"}, - {mustNewDecFromStr(t, "333"), 12, "000000000333"}, - {mustNewDecFromStr(t, "0.3333"), 8, "00000000"}, - } - for tcIndex, tc := range tests { - res := tc.dec.ToLeftPadded(tc.digits) - require.Equal(t, tc.exp, res, "incorrect left padding, tc %d", tcIndex) - } -} - var cdc = codec.New() func TestDecMarshalJSON(t *testing.T) { From 1a463eb056e6aaa566c94a4f16de7094891abd55 Mon Sep 17 00:00:00 2001 From: ValarDragon Date: Sat, 20 Oct 2018 18:02:17 -0700 Subject: [PATCH 4/4] make more clear --- types/decimal.go | 17 +++++++++-------- 1 file changed, 9 insertions(+), 8 deletions(-) diff --git a/types/decimal.go b/types/decimal.go index b621acd51..c981f6dca 100644 --- a/types/decimal.go +++ b/types/decimal.go @@ -253,25 +253,26 @@ func (d Dec) String() string { return "" } var bzWDec []byte + inputSize := len(bz) // TODO: Remove trailing zeros // case 1, purely decimal - if len(bz) <= 10 { + if inputSize <= 10 { bzWDec = make([]byte, 12) // 0. prefix bzWDec[0] = byte('0') bzWDec[1] = byte('.') // set relevant digits to 0 - for i := 0; i < 10-len(bz); i++ { + for i := 0; i < 10-inputSize; i++ { bzWDec[i+2] = byte('0') } // set last few digits - copy(bzWDec[2+(10-len(bz)):], bz) + copy(bzWDec[2+(10-inputSize):], bz) } else { - // len(bz) + 1 to account for the decimal point that is being added - bzWDec = make([]byte, len(bz)+1) - copy(bzWDec, bz[:len(bz)-10]) - bzWDec[len(bz)-10] = byte('.') - copy(bzWDec[len(bz)-9:], bz[len(bz)-10:]) + // inputSize + 1 to account for the decimal point that is being added + bzWDec = make([]byte, inputSize+1) + copy(bzWDec, bz[:inputSize-10]) + bzWDec[inputSize-10] = byte('.') + copy(bzWDec[inputSize-9:], bz[inputSize-10:]) } return string(bzWDec) }