2018-09-18 22:42:38 +02:00
|
|
|
package values
|
|
|
|
|
|
|
|
import (
|
2018-10-05 21:17:22 +02:00
|
|
|
"encoding/binary"
|
2018-09-18 22:42:38 +02:00
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2018-10-28 07:45:26 +02:00
|
|
|
"github.com/MontFerret/ferret/pkg/runtime/core"
|
2018-10-05 21:17:22 +02:00
|
|
|
"hash/fnv"
|
|
|
|
"math"
|
2018-09-18 22:42:38 +02:00
|
|
|
"strconv"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Float float64
|
|
|
|
|
|
|
|
var ZeroFloat = Float(0.0)
|
|
|
|
|
|
|
|
func NewFloat(input float64) Float {
|
|
|
|
return Float(input)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ParseFloat(input interface{}) (Float, error) {
|
|
|
|
if core.IsNil(input) {
|
|
|
|
return ZeroFloat, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
i, ok := input.(float64)
|
|
|
|
|
|
|
|
if ok {
|
|
|
|
if i == 0 {
|
|
|
|
return ZeroFloat, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return Float(i), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// try to cast
|
|
|
|
str, ok := input.(string)
|
|
|
|
|
|
|
|
if ok {
|
|
|
|
i, err := strconv.Atoi(str)
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
if i == 0 {
|
|
|
|
return ZeroFloat, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return Float(i), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-28 07:45:26 +02:00
|
|
|
return ZeroFloat, core.Error(core.ErrInvalidType, "expected 'float'")
|
2018-09-18 22:42:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func ParseFloatP(input interface{}) Float {
|
|
|
|
res, err := ParseFloat(input)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
2018-10-14 03:07:28 +02:00
|
|
|
func IsNaN(input Float) Boolean {
|
|
|
|
return NewBoolean(math.IsNaN(float64(input)))
|
|
|
|
}
|
|
|
|
|
|
|
|
func IsInf(input Float, sign Int) Boolean {
|
|
|
|
return NewBoolean(math.IsInf(float64(input), int(sign)))
|
|
|
|
}
|
|
|
|
|
2018-09-18 22:42:38 +02:00
|
|
|
func (t Float) MarshalJSON() ([]byte, error) {
|
|
|
|
return json.Marshal(float64(t))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t Float) Type() core.Type {
|
|
|
|
return core.FloatType
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t Float) String() string {
|
|
|
|
return fmt.Sprintf("%f", t)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t Float) Compare(other core.Value) int {
|
|
|
|
raw := float64(t)
|
|
|
|
|
|
|
|
switch other.Type() {
|
|
|
|
case core.FloatType:
|
|
|
|
f := other.Unwrap().(float64)
|
|
|
|
|
|
|
|
if raw == f {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if raw < f {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
return +1
|
|
|
|
case core.IntType:
|
|
|
|
i := other.Unwrap().(int)
|
|
|
|
f := float64(i)
|
|
|
|
|
|
|
|
if raw == f {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if raw < f {
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
|
|
|
|
return +1
|
|
|
|
case core.BooleanType, core.NoneType:
|
|
|
|
return 1
|
|
|
|
default:
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t Float) Unwrap() interface{} {
|
|
|
|
return float64(t)
|
|
|
|
}
|
|
|
|
|
2018-10-05 21:17:22 +02:00
|
|
|
func (t Float) Hash() uint64 {
|
|
|
|
h := fnv.New64a()
|
2018-09-18 22:42:38 +02:00
|
|
|
|
2018-10-05 21:17:22 +02:00
|
|
|
h.Write([]byte(t.Type().String()))
|
|
|
|
h.Write([]byte(":"))
|
2018-09-18 22:42:38 +02:00
|
|
|
|
2018-10-05 21:17:22 +02:00
|
|
|
bytes := make([]byte, 8)
|
|
|
|
binary.LittleEndian.PutUint64(bytes, math.Float64bits(float64(t)))
|
|
|
|
h.Write(bytes)
|
2018-09-18 22:42:38 +02:00
|
|
|
|
2018-10-05 21:17:22 +02:00
|
|
|
return h.Sum64()
|
2018-09-18 22:42:38 +02:00
|
|
|
}
|
2018-09-27 17:53:26 +02:00
|
|
|
|
2018-10-12 17:58:08 +02:00
|
|
|
func (t Float) Copy() core.Value {
|
2018-09-27 17:53:26 +02:00
|
|
|
return t
|
|
|
|
}
|