package pgtype

import (
	
	
	
	
	

	
)

type Date struct {
	Time             time.Time
	Status           Status
	InfinityModifier InfinityModifier
}

const (
	negativeInfinityDayOffset = -2147483648
	infinityDayOffset         = 2147483647
)

func ( *Date) ( interface{}) error {
	if  == nil {
		* = Date{Status: Null}
		return nil
	}

	if ,  := .(interface{ () interface{} });  {
		 := .()
		if  !=  {
			return .()
		}
	}

	switch value := .(type) {
	case time.Time:
		* = Date{Time: , Status: Present}
	case string:
		return .DecodeText(nil, []byte())
	case *time.Time:
		if  == nil {
			* = Date{Status: Null}
		} else {
			return .(*)
		}
	case *string:
		if  == nil {
			* = Date{Status: Null}
		} else {
			return .(*)
		}
	default:
		if ,  := underlyingTimeType();  {
			return .()
		}
		return fmt.Errorf("cannot convert %v to Date", )
	}

	return nil
}

func ( Date) () interface{} {
	switch .Status {
	case Present:
		if .InfinityModifier != None {
			return .InfinityModifier
		}
		return .Time
	case Null:
		return nil
	default:
		return .Status
	}
}

func ( *Date) ( interface{}) error {
	switch .Status {
	case Present:
		switch v := .(type) {
		case *time.Time:
			if .InfinityModifier != None {
				return fmt.Errorf("cannot assign %v to %T", , )
			}
			* = .Time
			return nil
		default:
			if ,  := GetAssignToDstType();  {
				return .()
			}
			return fmt.Errorf("unable to assign to %T", )
		}
	case Null:
		return NullAssignTo()
	}

	return fmt.Errorf("cannot decode %#v into %T", , )
}

func ( *Date) ( *ConnInfo,  []byte) error {
	if  == nil {
		* = Date{Status: Null}
		return nil
	}

	 := string()
	switch  {
	case "infinity":
		* = Date{Status: Present, InfinityModifier: Infinity}
	case "-infinity":
		* = Date{Status: Present, InfinityModifier: -Infinity}
	default:
		,  := time.ParseInLocation("2006-01-02", , time.UTC)
		if  != nil {
			return 
		}

		* = Date{Time: , Status: Present}
	}

	return nil
}

func ( *Date) ( *ConnInfo,  []byte) error {
	if  == nil {
		* = Date{Status: Null}
		return nil
	}

	if len() != 4 {
		return fmt.Errorf("invalid length for date: %v", len())
	}

	 := int32(binary.BigEndian.Uint32())

	switch  {
	case infinityDayOffset:
		* = Date{Status: Present, InfinityModifier: Infinity}
	case negativeInfinityDayOffset:
		* = Date{Status: Present, InfinityModifier: -Infinity}
	default:
		 := time.Date(2000, 1, int(1+), 0, 0, 0, 0, time.UTC)
		* = Date{Time: , Status: Present}
	}

	return nil
}

func ( Date) ( *ConnInfo,  []byte) ([]byte, error) {
	switch .Status {
	case Null:
		return nil, nil
	case Undefined:
		return nil, errUndefined
	}

	var  string

	switch .InfinityModifier {
	case None:
		 = .Time.Format("2006-01-02")
	case Infinity:
		 = "infinity"
	case NegativeInfinity:
		 = "-infinity"
	}

	return append(, ...), nil
}

func ( Date) ( *ConnInfo,  []byte) ([]byte, error) {
	switch .Status {
	case Null:
		return nil, nil
	case Undefined:
		return nil, errUndefined
	}

	var  int32
	switch .InfinityModifier {
	case None:
		 := time.Date(.Time.Year(), .Time.Month(), .Time.Day(), 0, 0, 0, 0, time.UTC).Unix()
		 := time.Date(2000, 1, 1, 0, 0, 0, 0, time.UTC).Unix()

		 :=  - 
		 = int32( / 86400)
	case Infinity:
		 = infinityDayOffset
	case NegativeInfinity:
		 = negativeInfinityDayOffset
	}

	return pgio.AppendInt32(, ), nil
}
Scan implements the database/sql Scanner interface.
func ( *Date) ( interface{}) error {
	if  == nil {
		* = Date{Status: Null}
		return nil
	}

	switch src := .(type) {
	case string:
		return .DecodeText(nil, []byte())
	case []byte:
		 := make([]byte, len())
		copy(, )
		return .DecodeText(nil, )
	case time.Time:
		* = Date{Time: , Status: Present}
		return nil
	}

	return fmt.Errorf("cannot scan %T", )
}
Value implements the database/sql/driver Valuer interface.
func ( Date) () (driver.Value, error) {
	switch .Status {
	case Present:
		if .InfinityModifier != None {
			return .InfinityModifier.String(), nil
		}
		return .Time, nil
	case Null:
		return nil, nil
	default:
		return nil, errUndefined
	}
}

func ( Date) () ([]byte, error) {
	switch .Status {
	case Null:
		return []byte("null"), nil
	case Undefined:
		return nil, errUndefined
	}

	if .Status != Present {
		return nil, errBadStatus
	}

	var  string

	switch .InfinityModifier {
	case None:
		 = .Time.Format("2006-01-02")
	case Infinity:
		 = "infinity"
	case NegativeInfinity:
		 = "-infinity"
	}

	return json.Marshal()
}

func ( *Date) ( []byte) error {
	var  *string
	 := json.Unmarshal(, &)
	if  != nil {
		return 
	}

	if  == nil {
		* = Date{Status: Null}
		return nil
	}

	switch * {
	case "infinity":
		* = Date{Status: Present, InfinityModifier: Infinity}
	case "-infinity":
		* = Date{Status: Present, InfinityModifier: -Infinity}
	default:
		,  := time.ParseInLocation("2006-01-02", *, time.UTC)
		if  != nil {
			return 
		}

		* = Date{Time: , Status: Present}
	}

	return nil