blob: 5e39f8ea619e16e8e43c973b24a75737276bd716 [file] [log] [blame]
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License. You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package arrow_test
import (
"testing"
"time"
"github.com/apache/arrow/go/v12/arrow"
"github.com/stretchr/testify/assert"
)
// TestTimeUnit_String verifies each time unit matches its string representation.
func TestTimeUnit_String(t *testing.T) {
tests := []struct {
u arrow.TimeUnit
exp string
}{
{arrow.Nanosecond, "ns"},
{arrow.Microsecond, "us"},
{arrow.Millisecond, "ms"},
{arrow.Second, "s"},
}
for _, test := range tests {
t.Run(test.exp, func(t *testing.T) {
assert.Equal(t, test.exp, test.u.String())
})
}
}
func TestDecimal128Type(t *testing.T) {
for _, tc := range []struct {
precision int32
scale int32
want string
}{
{1, 10, "decimal(1, 10)"},
{10, 10, "decimal(10, 10)"},
{10, 1, "decimal(10, 1)"},
} {
t.Run(tc.want, func(t *testing.T) {
dt := arrow.Decimal128Type{Precision: tc.precision, Scale: tc.scale}
if got, want := dt.BitWidth(), 128; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.ID(), arrow.DECIMAL128; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), tc.want; got != want {
t.Fatalf("invalid stringer: got=%q, want=%q", got, want)
}
})
}
}
func TestDecimal256Type(t *testing.T) {
for _, tc := range []struct {
precision int32
scale int32
want string
}{
{1, 10, "decimal256(1, 10)"},
{10, 10, "decimal256(10, 10)"},
{10, 1, "decimal256(10, 1)"},
} {
t.Run(tc.want, func(t *testing.T) {
dt := arrow.Decimal256Type{Precision: tc.precision, Scale: tc.scale}
if got, want := dt.BitWidth(), 256; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.ID(), arrow.DECIMAL256; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), tc.want; got != want {
t.Fatalf("invalid stringer: got=%q, want=%q", got, want)
}
})
}
}
func TestFixedSizeBinaryType(t *testing.T) {
for _, tc := range []struct {
byteWidth int
want string
}{
{1, "fixed_size_binary[1]"},
{8, "fixed_size_binary[8]"},
{100, "fixed_size_binary[100]"},
{100000000, "fixed_size_binary[100000000]"},
} {
t.Run(tc.want, func(t *testing.T) {
dt := arrow.FixedSizeBinaryType{tc.byteWidth}
if got, want := dt.BitWidth(), 8*tc.byteWidth; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "fixed_size_binary"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.FIXED_SIZE_BINARY; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), tc.want; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
})
}
}
func TestTimestampType(t *testing.T) {
for _, tc := range []struct {
unit arrow.TimeUnit
timeZone string
want string
}{
{arrow.Nanosecond, "CST", "timestamp[ns, tz=CST]"},
{arrow.Microsecond, "EST", "timestamp[us, tz=EST]"},
{arrow.Millisecond, "UTC", "timestamp[ms, tz=UTC]"},
{arrow.Second, "", "timestamp[s]"},
} {
t.Run(tc.want, func(t *testing.T) {
dt := arrow.TimestampType{Unit: tc.unit, TimeZone: tc.timeZone}
if got, want := dt.BitWidth(), 64; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "timestamp"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.TIMESTAMP; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), tc.want; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
})
}
}
func TestTime32Type(t *testing.T) {
for _, tc := range []struct {
unit arrow.TimeUnit
want string
}{
{arrow.Millisecond, "time32[ms]"},
{arrow.Second, "time32[s]"},
} {
t.Run(tc.want, func(t *testing.T) {
dt := arrow.Time32Type{tc.unit}
if got, want := dt.BitWidth(), 32; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "time32"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.TIME32; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), tc.want; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
})
}
for _, tc := range []struct {
unit arrow.TimeUnit
str string
want arrow.Time32
wantErr bool
}{
{arrow.Second, "12:21", arrow.Time32(12*3600 + 21*60), false},
{arrow.Second, "02:30:45", arrow.Time32(2*3600 + 30*60 + 45), false},
{arrow.Second, "21:21:21.21", arrow.Time32(0), true},
{arrow.Millisecond, "21:21:21.21", arrow.Time32(21*3600000 + 21*60000 + 21*1000 + 210), false},
{arrow.Millisecond, "15:02:04.123", arrow.Time32(15*3600000 + 2*60000 + 4*1000 + 123), false},
{arrow.Millisecond, "12:12:12.1212", arrow.Time32(0), true},
{arrow.Microsecond, "10:10:10", arrow.Time32(0), true},
{arrow.Nanosecond, "10:10:10", arrow.Time32(0), true},
} {
t.Run("FromString", func(t *testing.T) {
v, e := arrow.Time32FromString(tc.str, tc.unit)
assert.Equal(t, tc.want, v)
if tc.wantErr {
assert.Error(t, e)
} else {
assert.NoError(t, e)
}
})
}
}
func TestTime64Type(t *testing.T) {
for _, tc := range []struct {
unit arrow.TimeUnit
want string
}{
{arrow.Nanosecond, "time64[ns]"},
{arrow.Microsecond, "time64[us]"},
} {
t.Run(tc.want, func(t *testing.T) {
dt := arrow.Time64Type{tc.unit}
if got, want := dt.BitWidth(), 64; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "time64"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.TIME64; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), tc.want; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
})
}
const (
h = time.Hour
m = time.Minute
s = time.Second
us = time.Microsecond
ns = time.Nanosecond
)
for _, tc := range []struct {
unit arrow.TimeUnit
str string
want arrow.Time64
wantErr bool
}{
{arrow.Second, "12:21", arrow.Time64(0), true},
{arrow.Millisecond, "21:21:21.21", arrow.Time64(0), true},
{arrow.Microsecond, "10:10:10", arrow.Time64((10*h + 10*m + 10*s).Microseconds()), false},
{arrow.Microsecond, "22:10:15.123456", arrow.Time64((22*h + 10*m + 15*s + 123456*us).Microseconds()), false},
{arrow.Microsecond, "12:34:56.78901234", arrow.Time64(0), true},
{arrow.Nanosecond, "12:34:56.78901234", arrow.Time64(12*h + 34*m + 56*s + 789012340), false},
{arrow.Nanosecond, "12:34:56.123456789 9", arrow.Time64(0), true},
} {
t.Run("FromString", func(t *testing.T) {
v, e := arrow.Time64FromString(tc.str, tc.unit)
assert.Equal(t, tc.want, v)
if tc.wantErr {
assert.Error(t, e)
} else {
assert.NoError(t, e)
}
})
}
}
func TestDurationType(t *testing.T) {
for _, tc := range []struct {
unit arrow.TimeUnit
want string
}{
{arrow.Nanosecond, "duration[ns]"},
{arrow.Microsecond, "duration[us]"},
{arrow.Millisecond, "duration[ms]"},
{arrow.Second, "duration[s]"},
} {
t.Run(tc.want, func(t *testing.T) {
dt := arrow.DurationType{tc.unit}
if got, want := dt.BitWidth(), 64; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "duration"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.DURATION; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), tc.want; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
})
}
}
func TestBooleanType(t *testing.T) {
dt := arrow.BooleanType{}
if got, want := dt.BitWidth(), 1; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "bool"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.BOOL; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), "bool"; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
}
func TestFloat16Type(t *testing.T) {
dt := arrow.Float16Type{}
if got, want := dt.BitWidth(), 16; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "float16"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.FLOAT16; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), "float16"; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
}
func TestDayTimeIntervalType(t *testing.T) {
dt := arrow.DayTimeIntervalType{}
if got, want := dt.BitWidth(), 64; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "day_time_interval"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.INTERVAL_DAY_TIME; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), "day_time_interval"; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
}
func TestMonthIntervalType(t *testing.T) {
dt := arrow.MonthIntervalType{}
if got, want := dt.BitWidth(), 32; got != want {
t.Fatalf("invalid bitwidth: got=%d, want=%d", got, want)
}
if got, want := dt.Name(), "month_interval"; got != want {
t.Fatalf("invalid type name: got=%q, want=%q", got, want)
}
if got, want := dt.ID(), arrow.INTERVAL_MONTHS; got != want {
t.Fatalf("invalid type ID: got=%v, want=%v", got, want)
}
if got, want := dt.String(), "month_interval"; got != want {
t.Fatalf("invalid type stringer: got=%q, want=%q", got, want)
}
}