mirror of
https://github.com/minio/minio.git
synced 2025-01-03 19:13:22 -05:00
1f262daf6f
This is to ensure that there are no projects that try to import `minio/minio/pkg` into their own repo. Any such common packages should go to `https://github.com/minio/pkg`
688 lines
12 KiB
Go
688 lines
12 KiB
Go
// Copyright (c) 2015-2021 MinIO, Inc.
|
|
//
|
|
// This file is part of MinIO Object Storage stack
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package sql
|
|
|
|
import (
|
|
"fmt"
|
|
"math"
|
|
"strconv"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
// valueBuilders contains one constructor for each value type.
|
|
// Values should match if type is the same.
|
|
var valueBuilders = []func() *Value{
|
|
func() *Value {
|
|
return FromNull()
|
|
},
|
|
func() *Value {
|
|
return FromBool(true)
|
|
},
|
|
func() *Value {
|
|
return FromBytes([]byte("byte contents"))
|
|
},
|
|
func() *Value {
|
|
return FromFloat(math.Pi)
|
|
},
|
|
func() *Value {
|
|
return FromInt(0x1337)
|
|
},
|
|
func() *Value {
|
|
t, err := time.Parse(time.RFC3339, "2006-01-02T15:04:05Z")
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return FromTimestamp(t)
|
|
},
|
|
func() *Value {
|
|
return FromString("string contents")
|
|
},
|
|
}
|
|
|
|
// altValueBuilders contains one constructor for each value type.
|
|
// Values are zero values and should NOT match the values in valueBuilders, except Null type.
|
|
var altValueBuilders = []func() *Value{
|
|
func() *Value {
|
|
return FromNull()
|
|
},
|
|
func() *Value {
|
|
return FromBool(false)
|
|
},
|
|
func() *Value {
|
|
return FromBytes(nil)
|
|
},
|
|
func() *Value {
|
|
return FromFloat(0)
|
|
},
|
|
func() *Value {
|
|
return FromInt(0)
|
|
},
|
|
func() *Value {
|
|
return FromTimestamp(time.Time{})
|
|
},
|
|
func() *Value {
|
|
return FromString("")
|
|
},
|
|
}
|
|
|
|
func TestValue_SameTypeAs(t *testing.T) {
|
|
type fields struct {
|
|
a, b Value
|
|
}
|
|
type test struct {
|
|
name string
|
|
fields fields
|
|
wantOk bool
|
|
}
|
|
var tests []test
|
|
for i := range valueBuilders {
|
|
a := valueBuilders[i]()
|
|
for j := range valueBuilders {
|
|
b := valueBuilders[j]()
|
|
tests = append(tests, test{
|
|
name: fmt.Sprint(a.GetTypeString(), "==", b.GetTypeString()),
|
|
fields: fields{
|
|
a: *a, b: *b,
|
|
},
|
|
wantOk: i == j,
|
|
})
|
|
}
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
if gotOk := tt.fields.a.SameTypeAs(tt.fields.b); gotOk != tt.wantOk {
|
|
t.Errorf("SameTypeAs() = %v, want %v", gotOk, tt.wantOk)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestValue_Equals(t *testing.T) {
|
|
type fields struct {
|
|
a, b Value
|
|
}
|
|
type test struct {
|
|
name string
|
|
fields fields
|
|
wantOk bool
|
|
}
|
|
var tests []test
|
|
for i := range valueBuilders {
|
|
a := valueBuilders[i]()
|
|
for j := range valueBuilders {
|
|
b := valueBuilders[j]()
|
|
tests = append(tests, test{
|
|
name: fmt.Sprint(a.GetTypeString(), "==", b.GetTypeString()),
|
|
fields: fields{
|
|
a: *a, b: *b,
|
|
},
|
|
wantOk: i == j,
|
|
})
|
|
}
|
|
}
|
|
for i := range valueBuilders {
|
|
a := valueBuilders[i]()
|
|
for j := range altValueBuilders {
|
|
b := altValueBuilders[j]()
|
|
tests = append(tests, test{
|
|
name: fmt.Sprint(a.GetTypeString(), "!=", b.GetTypeString()),
|
|
fields: fields{
|
|
a: *a, b: *b,
|
|
},
|
|
// Only Null == Null
|
|
wantOk: a.IsNull() && b.IsNull() && i == 0 && j == 0,
|
|
})
|
|
}
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
if gotOk := tt.fields.a.Equals(tt.fields.b); gotOk != tt.wantOk {
|
|
t.Errorf("Equals() = %v, want %v", gotOk, tt.wantOk)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestValue_CSVString(t *testing.T) {
|
|
type test struct {
|
|
name string
|
|
want string
|
|
wantAlt string
|
|
}
|
|
|
|
tests := []test{
|
|
{
|
|
name: valueBuilders[0]().String(),
|
|
want: "",
|
|
wantAlt: "",
|
|
},
|
|
{
|
|
name: valueBuilders[1]().String(),
|
|
want: "true",
|
|
wantAlt: "false",
|
|
},
|
|
{
|
|
name: valueBuilders[2]().String(),
|
|
want: "byte contents",
|
|
wantAlt: "",
|
|
},
|
|
{
|
|
name: valueBuilders[3]().String(),
|
|
want: "3.141592653589793",
|
|
wantAlt: "0",
|
|
},
|
|
{
|
|
name: valueBuilders[4]().String(),
|
|
want: "4919",
|
|
wantAlt: "0",
|
|
},
|
|
{
|
|
name: valueBuilders[5]().String(),
|
|
want: "2006-01-02T15:04:05Z",
|
|
wantAlt: "0001T",
|
|
},
|
|
{
|
|
name: valueBuilders[6]().String(),
|
|
want: "string contents",
|
|
wantAlt: "",
|
|
},
|
|
}
|
|
|
|
for i, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
v := valueBuilders[i]()
|
|
vAlt := altValueBuilders[i]()
|
|
if got := v.CSVString(); got != tt.want {
|
|
t.Errorf("CSVString() = %v, want %v", got, tt.want)
|
|
}
|
|
if got := vAlt.CSVString(); got != tt.wantAlt {
|
|
t.Errorf("CSVString() = %v, want %v", got, tt.wantAlt)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestValue_bytesToInt(t *testing.T) {
|
|
type fields struct {
|
|
value interface{}
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
want int64
|
|
wantOK bool
|
|
}{
|
|
{
|
|
name: "zero",
|
|
fields: fields{
|
|
value: []byte("0"),
|
|
},
|
|
want: 0,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "minuszero",
|
|
fields: fields{
|
|
value: []byte("-0"),
|
|
},
|
|
want: 0,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "one",
|
|
fields: fields{
|
|
value: []byte("1"),
|
|
},
|
|
want: 1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "minusone",
|
|
fields: fields{
|
|
value: []byte("-1"),
|
|
},
|
|
want: -1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "plusone",
|
|
fields: fields{
|
|
value: []byte("+1"),
|
|
},
|
|
want: 1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "max",
|
|
fields: fields{
|
|
value: []byte(strconv.FormatInt(math.MaxInt64, 10)),
|
|
},
|
|
want: math.MaxInt64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "min",
|
|
fields: fields{
|
|
value: []byte(strconv.FormatInt(math.MinInt64, 10)),
|
|
},
|
|
want: math.MinInt64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "max-overflow",
|
|
fields: fields{
|
|
value: []byte("9223372036854775808"),
|
|
},
|
|
// Seems to be what strconv.ParseInt returns
|
|
want: math.MaxInt64,
|
|
wantOK: false,
|
|
},
|
|
{
|
|
name: "min-underflow",
|
|
fields: fields{
|
|
value: []byte("-9223372036854775809"),
|
|
},
|
|
// Seems to be what strconv.ParseInt returns
|
|
want: math.MinInt64,
|
|
wantOK: false,
|
|
},
|
|
{
|
|
name: "zerospace",
|
|
fields: fields{
|
|
value: []byte(" 0"),
|
|
},
|
|
want: 0,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "onespace",
|
|
fields: fields{
|
|
value: []byte("1 "),
|
|
},
|
|
want: 1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "minusonespace",
|
|
fields: fields{
|
|
value: []byte(" -1 "),
|
|
},
|
|
want: -1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "plusonespace",
|
|
fields: fields{
|
|
value: []byte("\t+1\t"),
|
|
},
|
|
want: 1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "scientific",
|
|
fields: fields{
|
|
value: []byte("3e5"),
|
|
},
|
|
want: 0,
|
|
wantOK: false,
|
|
},
|
|
{
|
|
// No support for prefixes
|
|
name: "hex",
|
|
fields: fields{
|
|
value: []byte("0xff"),
|
|
},
|
|
want: 0,
|
|
wantOK: false,
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
v := &Value{
|
|
value: tt.fields.value,
|
|
}
|
|
got, got1 := v.bytesToInt()
|
|
if got != tt.want {
|
|
t.Errorf("bytesToInt() got = %v, want %v", got, tt.want)
|
|
}
|
|
if got1 != tt.wantOK {
|
|
t.Errorf("bytesToInt() got1 = %v, want %v", got1, tt.wantOK)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestValue_bytesToFloat(t *testing.T) {
|
|
type fields struct {
|
|
value interface{}
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
want float64
|
|
wantOK bool
|
|
}{
|
|
// Copied from TestValue_bytesToInt.
|
|
{
|
|
name: "zero",
|
|
fields: fields{
|
|
value: []byte("0"),
|
|
},
|
|
want: 0,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "minuszero",
|
|
fields: fields{
|
|
value: []byte("-0"),
|
|
},
|
|
want: 0,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "one",
|
|
fields: fields{
|
|
value: []byte("1"),
|
|
},
|
|
want: 1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "minusone",
|
|
fields: fields{
|
|
value: []byte("-1"),
|
|
},
|
|
want: -1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "plusone",
|
|
fields: fields{
|
|
value: []byte("+1"),
|
|
},
|
|
want: 1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "maxint",
|
|
fields: fields{
|
|
value: []byte(strconv.FormatInt(math.MaxInt64, 10)),
|
|
},
|
|
want: math.MaxInt64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "minint",
|
|
fields: fields{
|
|
value: []byte(strconv.FormatInt(math.MinInt64, 10)),
|
|
},
|
|
want: math.MinInt64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "max-overflow-int",
|
|
fields: fields{
|
|
value: []byte("9223372036854775808"),
|
|
},
|
|
// Seems to be what strconv.ParseInt returns
|
|
want: math.MaxInt64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "min-underflow-int",
|
|
fields: fields{
|
|
value: []byte("-9223372036854775809"),
|
|
},
|
|
// Seems to be what strconv.ParseInt returns
|
|
want: math.MinInt64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "max",
|
|
fields: fields{
|
|
value: []byte(strconv.FormatFloat(math.MaxFloat64, 'g', -1, 64)),
|
|
},
|
|
want: math.MaxFloat64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "min",
|
|
fields: fields{
|
|
value: []byte(strconv.FormatFloat(-math.MaxFloat64, 'g', -1, 64)),
|
|
},
|
|
want: -math.MaxFloat64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "max-overflow",
|
|
fields: fields{
|
|
value: []byte("1.797693134862315708145274237317043567981e+309"),
|
|
},
|
|
// Seems to be what strconv.ParseInt returns
|
|
want: math.Inf(1),
|
|
wantOK: false,
|
|
},
|
|
{
|
|
name: "min-underflow",
|
|
fields: fields{
|
|
value: []byte("-1.797693134862315708145274237317043567981e+309"),
|
|
},
|
|
// Seems to be what strconv.ParseInt returns
|
|
want: math.Inf(-1),
|
|
wantOK: false,
|
|
},
|
|
{
|
|
name: "smallest-pos",
|
|
fields: fields{
|
|
value: []byte(strconv.FormatFloat(math.SmallestNonzeroFloat64, 'g', -1, 64)),
|
|
},
|
|
want: math.SmallestNonzeroFloat64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "smallest-pos",
|
|
fields: fields{
|
|
value: []byte(strconv.FormatFloat(-math.SmallestNonzeroFloat64, 'g', -1, 64)),
|
|
},
|
|
want: -math.SmallestNonzeroFloat64,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "zerospace",
|
|
fields: fields{
|
|
value: []byte(" 0"),
|
|
},
|
|
want: 0,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "onespace",
|
|
fields: fields{
|
|
value: []byte("1 "),
|
|
},
|
|
want: 1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "minusonespace",
|
|
fields: fields{
|
|
value: []byte(" -1 "),
|
|
},
|
|
want: -1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "plusonespace",
|
|
fields: fields{
|
|
value: []byte("\t+1\t"),
|
|
},
|
|
want: 1,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
name: "scientific",
|
|
fields: fields{
|
|
value: []byte("3e5"),
|
|
},
|
|
want: 300000,
|
|
wantOK: true,
|
|
},
|
|
{
|
|
// No support for prefixes
|
|
name: "hex",
|
|
fields: fields{
|
|
value: []byte("0xff"),
|
|
},
|
|
want: 0,
|
|
wantOK: false,
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
v := Value{
|
|
value: tt.fields.value,
|
|
}
|
|
got, got1 := v.bytesToFloat()
|
|
diff := math.Abs(got - tt.want)
|
|
if diff > floatCmpTolerance {
|
|
t.Errorf("bytesToFloat() got = %v, want %v", got, tt.want)
|
|
}
|
|
if got1 != tt.wantOK {
|
|
t.Errorf("bytesToFloat() got1 = %v, want %v", got1, tt.wantOK)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestValue_bytesToBool(t *testing.T) {
|
|
type fields struct {
|
|
value interface{}
|
|
}
|
|
tests := []struct {
|
|
name string
|
|
fields fields
|
|
wantVal bool
|
|
wantOk bool
|
|
}{
|
|
{
|
|
name: "true",
|
|
fields: fields{
|
|
value: []byte("true"),
|
|
},
|
|
wantVal: true,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "false",
|
|
fields: fields{
|
|
value: []byte("false"),
|
|
},
|
|
wantVal: false,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "t",
|
|
fields: fields{
|
|
value: []byte("t"),
|
|
},
|
|
wantVal: true,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "f",
|
|
fields: fields{
|
|
value: []byte("f"),
|
|
},
|
|
wantVal: false,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "1",
|
|
fields: fields{
|
|
value: []byte("1"),
|
|
},
|
|
wantVal: true,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "0",
|
|
fields: fields{
|
|
value: []byte("0"),
|
|
},
|
|
wantVal: false,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "truespace",
|
|
fields: fields{
|
|
value: []byte(" true "),
|
|
},
|
|
wantVal: true,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "truetabs",
|
|
fields: fields{
|
|
value: []byte("\ttrue\t"),
|
|
},
|
|
wantVal: true,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "TRUE",
|
|
fields: fields{
|
|
value: []byte("TRUE"),
|
|
},
|
|
wantVal: true,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "FALSE",
|
|
fields: fields{
|
|
value: []byte("FALSE"),
|
|
},
|
|
wantVal: false,
|
|
wantOk: true,
|
|
},
|
|
{
|
|
name: "invalid",
|
|
fields: fields{
|
|
value: []byte("no"),
|
|
},
|
|
wantVal: false,
|
|
wantOk: false,
|
|
},
|
|
}
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
v := Value{
|
|
value: tt.fields.value,
|
|
}
|
|
gotVal, gotOk := v.bytesToBool()
|
|
if gotVal != tt.wantVal {
|
|
t.Errorf("bytesToBool() gotVal = %v, want %v", gotVal, tt.wantVal)
|
|
}
|
|
if gotOk != tt.wantOk {
|
|
t.Errorf("bytesToBool() gotOk = %v, want %v", gotOk, tt.wantOk)
|
|
}
|
|
})
|
|
}
|
|
}
|