// Copyright (c) 2024 Celestino Amoroso (celestino.amoroso@gmail.com). // All rights reserved. // func-builtins.go package expr import ( "math" "strconv" ) func isNilFunc(ctx ExprContext, name string, args []any) (result any, err error) { if len(args) == 1 { result = args[0] == nil } else { err = errOneParam(name) } return } func isIntFunc(ctx ExprContext, name string, args []any) (result any, err error) { result = IsInteger(args[0]) return } func isFloatFunc(ctx ExprContext, name string, args []any) (result any, err error) { result = IsFloat(args[0]) return } func isBoolFunc(ctx ExprContext, name string, args []any) (result any, err error) { result = IsBool(args[0]) return } func isStringFunc(ctx ExprContext, name string, args []any) (result any, err error) { result = IsString(args[0]) return } func isFractionFunc(ctx ExprContext, name string, args []any) (result any, err error) { result = IsFract(args[0]) return } func isRationalFunc(ctx ExprContext, name string, args []any) (result any, err error) { result = IsRational(args[0]) return } func isListFunc(ctx ExprContext, name string, args []any) (result any, err error) { result = IsList(args[0]) return } func isDictionaryFunc(ctx ExprContext, name string, args []any) (result any, err error) { result = IsDict(args[0]) return } func intFunc(ctx ExprContext, name string, args []any) (result any, err error) { switch v := args[0].(type) { case int64: result = v case float64: result = int64(math.Trunc(v)) case bool: if v { result = int64(1) } else { result = int64(0) } case string: var i int if i, err = strconv.Atoi(v); err == nil { result = int64(i) } default: err = errCantConvert(name, v, "int") } return } func decFunc(ctx ExprContext, name string, args []any) (result any, err error) { switch v := args[0].(type) { case int64: result = float64(v) case float64: result = v case bool: if v { result = float64(1) } else { result = float64(0) } case string: var f float64 if f, err = strconv.ParseFloat(v, 64); err == nil { result = f } case *fraction: result = v.toFloat() default: err = errCantConvert(name, v, "float") } return } func fractFunc(ctx ExprContext, name string, args []any) (result any, err error) { switch v := args[0].(type) { case int64: var den int64 = 1 if len(args) > 1 { var ok bool if den, ok = args[1].(int64); !ok { err = errExpectedGot(name, "integer", args[1]) } else if den == 0 { err = errDivisionByZero(name) } } if err == nil { result = newFraction(v, den) } case float64: result, err = float64ToFraction(v) // var n, d int64 // if n, d, err = float64ToFraction(v); err == nil { // result = newFraction(n, d) // } case bool: if v { result = newFraction(1, 1) } else { result = newFraction(0, 1) } case string: result, err = makeGeneratingFraction(v) // var f float64 // // TODO temporary implementation // if f, err = strconv.ParseFloat(v, 64); err == nil { // var n, d int64 // if n, d, err = float64ToFraction(f); err == nil { // result = newFraction(n, d) // } // } else { // errors.New("convertion from string to float is ongoing") // } case *fraction: result = v default: err = errCantConvert(name, v, "float") } return } func iteratorFunc(ctx ExprContext, name string, args []any) (result any, err error) { return } func ImportBuiltinsFuncs(ctx ExprContext) { ctx.RegisterFunc("isNil", &simpleFunctor{f: isNilFunc}, 1, 1) ctx.RegisterFunc("isInt", &simpleFunctor{f: isIntFunc}, 1, 1) ctx.RegisterFunc("isFloat", &simpleFunctor{f: isFloatFunc}, 1, 1) ctx.RegisterFunc("isBool", &simpleFunctor{f: isBoolFunc}, 1, 1) ctx.RegisterFunc("isString", &simpleFunctor{f: isStringFunc}, 1, 1) ctx.RegisterFunc("isFraction", &simpleFunctor{f: isFractionFunc}, 1, 1) ctx.RegisterFunc("isFract", &simpleFunctor{f: isFractionFunc}, 1, 1) ctx.RegisterFunc("isRational", &simpleFunctor{f: isRationalFunc}, 1, 1) ctx.RegisterFunc("isList", &simpleFunctor{f: isListFunc}, 1, 1) ctx.RegisterFunc("isDictionary", &simpleFunctor{f: isDictionaryFunc}, 1, 1) ctx.RegisterFunc("isDict", &simpleFunctor{f: isDictionaryFunc}, 1, 1) ctx.RegisterFunc("int", &simpleFunctor{f: intFunc}, 1, 1) ctx.RegisterFunc("dec", &simpleFunctor{f: decFunc}, 1, 1) ctx.RegisterFunc("fract", &simpleFunctor{f: fractFunc}, 1, 2) } func init() { registerImport("base", ImportBuiltinsFuncs, "Base expression tools like isNil(), int(), etc.") }