diff --git a/converter.go b/converter.go index f16f9ea4a..728f6bc06 100644 --- a/converter.go +++ b/converter.go @@ -942,6 +942,16 @@ func buildStructuredArray(ctx context.Context, fieldMetadata fieldMetadata, srcV return buildStructuredArrayRecursive[[]byte](ctx, fieldMetadata.Fields[0], srcValue, params) case "date", "time", "timestamp_ltz", "timestamp_ntz", "timestamp_tz": return buildStructuredArrayRecursive[time.Time](ctx, fieldMetadata.Fields[0], srcValue, params) + case "array": + arr := make([]any, len(srcValue)) + for i, v := range srcValue { + structuredArray, err := buildStructuredArray(ctx, fieldMetadata.Fields[0], v.([]any), params) + if err != nil { + return nil, err + } + arr[i] = structuredArray + } + return arr, nil } } return srcValue, nil diff --git a/converter_test.go b/converter_test.go index f94a4e81a..ad9537f74 100644 --- a/converter_test.go +++ b/converter_test.go @@ -363,6 +363,30 @@ func TestStringToValue(t *testing.T) { } else if ts.UnixNano() != 1549491451123456789 { t.Errorf("expected unix timestamp: 1549491451123456789, got %v", ts.UnixNano()) } + + rowType = &execResponseRowType{Type: "array", Fields: []fieldMetadata{{Type: "array", Fields: []fieldMetadata{{Type: "fixed"}}}}} + src = "[[3]]" + if err = stringToValue(context.Background(), &dest, *rowType, &src, nil, nil); err != nil { + t.Errorf("unexpected error: %v", err) + } else if arr, ok := dest.([][]int64); !ok { + t.Errorf("expected type: '[][]int64', got '%v'", reflect.TypeOf(dest)) + } else if arr[0][0] != 3 { + t.Errorf("expected value: 3, got '%v'", arr[0][0]) + } + + rowType = &execResponseRowType{Type: "array", Fields: []fieldMetadata{{Type: "array", Fields: []fieldMetadata{{Type: "array", Fields: []fieldMetadata{{Type: "array", Fields: []fieldMetadata{{Type: "fixed"}}}}}}}}} + src = "[[[[3]]]]" + if err = stringToValue(context.Background(), &dest, *rowType, &src, nil, nil); err != nil { + t.Errorf("unexpected error: %v", err) + } else if arr, ok := dest.([]any); !ok { + t.Errorf("expected type: '[]any', got '%v'", reflect.TypeOf(dest)) + } else if arr1, ok := arr[0].([]any); !ok { + t.Errorf("expected type: '[]any', got '%v'", reflect.TypeOf(arr[0])) + } else if arr2, ok := arr1[0].([][]int64); !ok { + t.Errorf("expected type: '[][]int64', got '%v'", reflect.TypeOf(arr1[0])) + } else if arr2[0][0] != 3 { + t.Errorf("expected value: 3, got '%v'", arr2[0][0]) + } } type tcArrayToString struct { diff --git a/structured_type_read_test.go b/structured_type_read_test.go index efedecbcd..6d0709904 100644 --- a/structured_type_read_test.go +++ b/structured_type_read_test.go @@ -1023,6 +1023,12 @@ func TestArrayOfArrays(t *testing.T) { actual: make([][]time.Time, 2), expected: [][]time.Time{{time.Date(2024, time.January, 5, 11, 22, 33, 0, warsawTz)}, {time.Date(2001, time.November, 12, 11, 22, 33, 0, warsawTz)}}, }, + { + name: "3D_bool", + query: "SELECT ARRAY_CONSTRUCT(ARRAY_CONSTRUCT(ARRAY_CONSTRUCT(true, false), ARRAY_CONSTRUCT(false)), ARRAY_CONSTRUCT(ARRAY_CONSTRUCT(false, true), ARRAY_CONSTRUCT(true, true)))::ARRAY(ARRAY(ARRAY(BOOLEAN)))", + actual: make([][][]bool, 2), + expected: []any{[][]bool{{true, false}, {false}}, [][]bool{{false, true}, {true, true}}}, + }, } runDBTest(t, func(dbt *DBTest) { dbt.mustExec("ALTER SESSION SET TIMEZONE = 'Europe/Warsaw'")