// 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 array

import (
	
	
	
	

	
	
	
	
	
	
)

type baseDecimal[ interface {
	decimal.DecimalTypes
	decimal.Num[]
}] struct {
	array

	values []
}

func newDecimalData[ interface {
	decimal.DecimalTypes
	decimal.Num[]
}]( arrow.ArrayData) *baseDecimal[] {
	 := &baseDecimal[]{}
	.refCount.Add(1)
	.setData(.(*Data))
	return 
}

func ( *baseDecimal[]) ( int)  { return .values[] }

func ( *baseDecimal[]) ( int) string {
	if .IsNull() {
		return NullValueStr
	}
	return .GetOneForMarshal().(string)
}

func ( *baseDecimal[]) () [] { return .values }

func ( *baseDecimal[]) () string {
	 := new(strings.Builder)
	.WriteString("[")
	for  := 0;  < .Len(); ++ {
		if  > 0 {
			fmt.Fprintf(, " ")
		}
		switch {
		case .IsNull():
			.WriteString(NullValueStr)
		default:
			fmt.Fprintf(, "%v", .Value())
		}
	}
	.WriteString("]")
	return .String()
}

func ( *baseDecimal[]) ( *Data) {
	.array.setData()
	 := .buffers[1]
	if  != nil {
		.values = arrow.GetData[](.Bytes())
		 := .data.offset
		 :=  + .data.length
		.values = .values[:]
	}
}

func ( *baseDecimal[]) ( int) any {
	if .IsNull() {
		return nil
	}

	 := .DataType().(arrow.DecimalType)
	,  := .Value(), .GetScale()
	return .ToBigFloat().Text('g', int(.GetPrecision()))
}

func ( *baseDecimal[]) () ([]byte, error) {
	 := make([]any, .Len())
	for  := 0;  < .Len(); ++ {
		[] = .GetOneForMarshal()
	}
	return json.Marshal()
}

func arrayEqualDecimal[ interface {
	decimal.DecimalTypes
	decimal.Num[]
}](,  *baseDecimal[]) bool {
	for  := 0;  < .Len(); ++ {
		if .IsNull() {
			continue
		}

		if .Value() != .Value() {
			return false
		}
	}
	return true
}

type Decimal32 = baseDecimal[decimal.Decimal32]

func ( arrow.ArrayData) *Decimal32 {
	return newDecimalData[decimal.Decimal32]()
}

type Decimal64 = baseDecimal[decimal.Decimal64]

func ( arrow.ArrayData) *Decimal64 {
	return newDecimalData[decimal.Decimal64]()
}

type Decimal128 = baseDecimal[decimal.Decimal128]

func ( arrow.ArrayData) *Decimal128 {
	return newDecimalData[decimal.Decimal128]()
}

type Decimal256 = baseDecimal[decimal.Decimal256]

func ( arrow.ArrayData) *Decimal256 {
	return newDecimalData[decimal.Decimal256]()
}

type (
	Decimal32Builder  = baseDecimalBuilder[decimal.Decimal32]
	Decimal64Builder  = baseDecimalBuilder[decimal.Decimal64]
	Decimal128Builder struct {
		*baseDecimalBuilder[decimal.Decimal128]
	}
)

func ( *Decimal128Builder) () *Decimal128 {
	return .NewDecimalArray()
}

type Decimal256Builder struct {
	*baseDecimalBuilder[decimal.Decimal256]
}

func ( *Decimal256Builder) () *Decimal256 {
	return .NewDecimalArray()
}

type baseDecimalBuilder[ interface {
	decimal.DecimalTypes
	decimal.Num[]
}] struct {
	builder
	traits decimal.Traits[]

	dtype   arrow.DecimalType
	data    *memory.Buffer
	rawData []
}

func newDecimalBuilder[ interface {
	decimal.DecimalTypes
	decimal.Num[]
},  arrow.DecimalType]( memory.Allocator,  ) *baseDecimalBuilder[] {
	 := &baseDecimalBuilder[]{
		builder: builder{mem: },
		dtype:   ,
	}
	.refCount.Add(1)
	return 
}

func ( *baseDecimalBuilder[]) () arrow.DataType { return .dtype }

func ( *baseDecimalBuilder[]) () {
	debug.Assert(.refCount.Load() > 0, "too many releases")

	if .refCount.Add(-1) == 0 {
		if .nullBitmap != nil {
			.nullBitmap.Release()
			.nullBitmap = nil
		}
		if .data != nil {
			.data.Release()
			.data, .rawData = nil, nil
		}
	}
}

func ( *baseDecimalBuilder[]) ( ) {
	.Reserve(1)
	.UnsafeAppend()
}

func ( *baseDecimalBuilder[]) ( ) {
	bitutil.SetBit(.nullBitmap.Bytes(), .length)
	.rawData[.length] = 
	.length++
}

func ( *baseDecimalBuilder[]) () {
	.Reserve(1)
	.UnsafeAppendBoolToBitmap(false)
}

func ( *baseDecimalBuilder[]) ( int) {
	for  := 0;  < ; ++ {
		.AppendNull()
	}
}

func ( *baseDecimalBuilder[]) () {
	var  
	.Append()
}

func ( *baseDecimalBuilder[]) ( int) {
	for  := 0;  < ; ++ {
		.AppendEmptyValue()
	}
}

func ( *baseDecimalBuilder[]) ( bool) {
	if  {
		bitutil.SetBit(.nullBitmap.Bytes(), .length)
	} else {
		.nulls++
	}
	.length++
}

func ( *baseDecimalBuilder[]) ( [],  []bool) {
	if len() != len() && len() != 0 {
		panic("len(v) != len(valid) && len(valid) != 0")
	}

	if len() == 0 {
		return
	}

	.Reserve(len())
	if len() > 0 {
		copy(.rawData[.length:], )
	}
	.unsafeAppendBoolsToBitmap(, len())
}

func ( *baseDecimalBuilder[]) ( int) {
	.builder.init()

	.data = memory.NewResizableBuffer(.mem)
	 := int(reflect.TypeFor[]().Size()) * 
	.data.Resize()
	.rawData = arrow.GetData[](.data.Bytes())
}

func ( *baseDecimalBuilder[]) ( int) {
	.reserve(, .Resize)
}

func ( *baseDecimalBuilder[]) ( int) {
	 := 
	if  < minBuilderCapacity {
		 = minBuilderCapacity
	}

	if .capacity == 0 {
		.init()
	} else {
		.resize(, .init)
		.data.Resize(.traits.BytesRequired())
		.rawData = arrow.GetData[](.data.Bytes())
	}
}

func ( *baseDecimalBuilder[]) () ( *baseDecimal[]) {
	 := .newData()
	 = newDecimalData[]()
	.Release()
	return
}

func ( *baseDecimalBuilder[]) () arrow.Array {
	return .NewDecimalArray()
}

func ( *baseDecimalBuilder[]) () ( *Data) {
	 := .traits.BytesRequired(.length)
	if  > 0 &&  < .data.Len() {
		// trim buffers
		.data.Resize()
	}
	 = NewData(.dtype, .length, []*memory.Buffer{.nullBitmap, .data}, nil, .nulls, 0)
	.reset()

	if .data != nil {
		.data.Release()
		.data, .rawData = nil, nil
	}

	return
}

func ( *baseDecimalBuilder[]) ( string) error {
	if  == NullValueStr {
		.AppendNull()
		return nil
	}

	,  := .traits.FromString(, .dtype.GetPrecision(), .dtype.GetScale())
	if  != nil {
		.AppendNull()
		return 
	}
	.Append()
	return nil
}

func ( *baseDecimalBuilder[]) ( *json.Decoder) error {
	,  := .Token()
	if  != nil {
		return 
	}

	var  
	switch v := .(type) {
	case float64:
		,  = .traits.FromFloat64(, .dtype.GetPrecision(), .dtype.GetScale())
		if  != nil {
			return 
		}
		.Append()
	case string:
		,  = .traits.FromString(, .dtype.GetPrecision(), .dtype.GetScale())
		if  != nil {
			return 
		}
		.Append()
	case json.Number:
		,  = .traits.FromString(.String(), .dtype.GetPrecision(), .dtype.GetScale())
		if  != nil {
			return 
		}
		.Append()
	case nil:
		.AppendNull()
	default:
		return &json.UnmarshalTypeError{
			Value:  fmt.Sprint(),
			Type:   reflect.TypeFor[](),
			Offset: .InputOffset(),
		}
	}

	return nil
}

func ( *baseDecimalBuilder[]) ( *json.Decoder) error {
	for .More() {
		if  := .UnmarshalOne();  != nil {
			return 
		}
	}
	return nil
}

func ( *baseDecimalBuilder[]) ( []byte) error {
	 := json.NewDecoder(bytes.NewReader())
	,  := .Token()
	if  != nil {
		return 
	}

	if ,  := .(json.Delim); ! ||  != '[' {
		return fmt.Errorf("decimal builder must unpack from json array, found %s", )
	}

	return .Unmarshal()
}

func ( memory.Allocator,  *arrow.Decimal32Type) *Decimal32Builder {
	 := newDecimalBuilder[decimal.Decimal32](, )
	.traits = decimal.Dec32Traits
	return 
}

func ( memory.Allocator,  *arrow.Decimal64Type) *Decimal64Builder {
	 := newDecimalBuilder[decimal.Decimal64](, )
	.traits = decimal.Dec64Traits
	return 
}

func ( memory.Allocator,  *arrow.Decimal128Type) *Decimal128Builder {
	 := newDecimalBuilder[decimal.Decimal128](, )
	.traits = decimal.Dec128Traits
	return &Decimal128Builder{}
}

func ( memory.Allocator,  *arrow.Decimal256Type) *Decimal256Builder {
	 := newDecimalBuilder[decimal.Decimal256](, )
	.traits = decimal.Dec256Traits
	return &Decimal256Builder{}
}

var (
	_ arrow.Array = (*Decimal32)(nil)
	_ arrow.Array = (*Decimal64)(nil)
	_ arrow.Array = (*Decimal128)(nil)
	_ arrow.Array = (*Decimal256)(nil)
	_ Builder     = (*Decimal32Builder)(nil)
	_ Builder     = (*Decimal64Builder)(nil)
	_ Builder     = (*Decimal128Builder)(nil)
	_ Builder     = (*Decimal256Builder)(nil)

	_ arrow.TypedArray[decimal.Decimal32]  = (*Decimal32)(nil)
	_ arrow.TypedArray[decimal.Decimal64]  = (*Decimal64)(nil)
	_ arrow.TypedArray[decimal.Decimal128] = (*Decimal128)(nil)
	_ arrow.TypedArray[decimal.Decimal256] = (*Decimal256)(nil)
)