You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							218 lines
						
					
					
						
							6.2 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							218 lines
						
					
					
						
							6.2 KiB
						
					
					
				
								package engine
							 | 
						|
								
							 | 
						|
								import (
							 | 
						|
									"fmt"
							 | 
						|
									"math"
							 | 
						|
								
							 | 
						|
									"github.com/seaweedfs/seaweedfs/weed/pb/schema_pb"
							 | 
						|
								)
							 | 
						|
								
							 | 
						|
								// ===============================
							 | 
						|
								// ARITHMETIC OPERATORS
							 | 
						|
								// ===============================
							 | 
						|
								
							 | 
						|
								// ArithmeticOperator represents basic arithmetic operations
							 | 
						|
								type ArithmeticOperator string
							 | 
						|
								
							 | 
						|
								const (
							 | 
						|
									OpAdd    ArithmeticOperator = "+"
							 | 
						|
									OpSub    ArithmeticOperator = "-"
							 | 
						|
									OpMul    ArithmeticOperator = "*"
							 | 
						|
									OpDiv    ArithmeticOperator = "/"
							 | 
						|
									OpMod    ArithmeticOperator = "%"
							 | 
						|
								)
							 | 
						|
								
							 | 
						|
								// EvaluateArithmeticExpression evaluates basic arithmetic operations between two values
							 | 
						|
								func (e *SQLEngine) EvaluateArithmeticExpression(left, right *schema_pb.Value, operator ArithmeticOperator) (*schema_pb.Value, error) {
							 | 
						|
									if left == nil || right == nil {
							 | 
						|
										return nil, fmt.Errorf("arithmetic operation requires non-null operands")
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Convert values to numeric types for calculation
							 | 
						|
									leftNum, err := e.valueToFloat64(left)
							 | 
						|
									if err != nil {
							 | 
						|
										return nil, fmt.Errorf("left operand conversion error: %v", err)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									rightNum, err := e.valueToFloat64(right)
							 | 
						|
									if err != nil {
							 | 
						|
										return nil, fmt.Errorf("right operand conversion error: %v", err)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									var result float64
							 | 
						|
									var resultErr error
							 | 
						|
								
							 | 
						|
									switch operator {
							 | 
						|
									case OpAdd:
							 | 
						|
										result = leftNum + rightNum
							 | 
						|
									case OpSub:
							 | 
						|
										result = leftNum - rightNum
							 | 
						|
									case OpMul:
							 | 
						|
										result = leftNum * rightNum
							 | 
						|
									case OpDiv:
							 | 
						|
										if rightNum == 0 {
							 | 
						|
											return nil, fmt.Errorf("division by zero")
							 | 
						|
										}
							 | 
						|
										result = leftNum / rightNum
							 | 
						|
									case OpMod:
							 | 
						|
										if rightNum == 0 {
							 | 
						|
											return nil, fmt.Errorf("modulo by zero")
							 | 
						|
										}
							 | 
						|
										result = math.Mod(leftNum, rightNum)
							 | 
						|
									default:
							 | 
						|
										return nil, fmt.Errorf("unsupported arithmetic operator: %s", operator)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									if resultErr != nil {
							 | 
						|
										return nil, resultErr
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Convert result back to appropriate schema value type
							 | 
						|
									// If both operands were integers and operation doesn't produce decimal, return integer
							 | 
						|
									if e.isIntegerValue(left) && e.isIntegerValue(right) && 
							 | 
						|
										(operator == OpAdd || operator == OpSub || operator == OpMul || operator == OpMod) {
							 | 
						|
										return &schema_pb.Value{
							 | 
						|
											Kind: &schema_pb.Value_Int64Value{Int64Value: int64(result)},
							 | 
						|
										}, nil
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Otherwise return as double/float
							 | 
						|
									return &schema_pb.Value{
							 | 
						|
										Kind: &schema_pb.Value_DoubleValue{DoubleValue: result},
							 | 
						|
									}, nil
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Add evaluates addition (left + right)
							 | 
						|
								func (e *SQLEngine) Add(left, right *schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									return e.EvaluateArithmeticExpression(left, right, OpAdd)
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Subtract evaluates subtraction (left - right)
							 | 
						|
								func (e *SQLEngine) Subtract(left, right *schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									return e.EvaluateArithmeticExpression(left, right, OpSub)
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Multiply evaluates multiplication (left * right)
							 | 
						|
								func (e *SQLEngine) Multiply(left, right *schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									return e.EvaluateArithmeticExpression(left, right, OpMul)
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Divide evaluates division (left / right)
							 | 
						|
								func (e *SQLEngine) Divide(left, right *schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									return e.EvaluateArithmeticExpression(left, right, OpDiv)
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Modulo evaluates modulo operation (left % right)
							 | 
						|
								func (e *SQLEngine) Modulo(left, right *schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									return e.EvaluateArithmeticExpression(left, right, OpMod)
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// ===============================
							 | 
						|
								// MATHEMATICAL FUNCTIONS
							 | 
						|
								// ===============================
							 | 
						|
								
							 | 
						|
								// Round rounds a numeric value to the nearest integer or specified decimal places
							 | 
						|
								func (e *SQLEngine) Round(value *schema_pb.Value, precision ...*schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									if value == nil {
							 | 
						|
										return nil, fmt.Errorf("ROUND function requires non-null value")
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									num, err := e.valueToFloat64(value)
							 | 
						|
									if err != nil {
							 | 
						|
										return nil, fmt.Errorf("ROUND function conversion error: %v", err)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Default precision is 0 (round to integer)
							 | 
						|
									precisionValue := 0
							 | 
						|
									if len(precision) > 0 && precision[0] != nil {
							 | 
						|
										precFloat, err := e.valueToFloat64(precision[0])
							 | 
						|
										if err != nil {
							 | 
						|
											return nil, fmt.Errorf("ROUND precision conversion error: %v", err)
							 | 
						|
										}
							 | 
						|
										precisionValue = int(precFloat)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Apply rounding
							 | 
						|
									multiplier := math.Pow(10, float64(precisionValue))
							 | 
						|
									rounded := math.Round(num*multiplier) / multiplier
							 | 
						|
								
							 | 
						|
									// Return as integer if precision is 0 and original was integer, otherwise as double
							 | 
						|
									if precisionValue == 0 && e.isIntegerValue(value) {
							 | 
						|
										return &schema_pb.Value{
							 | 
						|
											Kind: &schema_pb.Value_Int64Value{Int64Value: int64(rounded)},
							 | 
						|
										}, nil
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									return &schema_pb.Value{
							 | 
						|
										Kind: &schema_pb.Value_DoubleValue{DoubleValue: rounded},
							 | 
						|
									}, nil
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Ceil returns the smallest integer greater than or equal to the value
							 | 
						|
								func (e *SQLEngine) Ceil(value *schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									if value == nil {
							 | 
						|
										return nil, fmt.Errorf("CEIL function requires non-null value")
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									num, err := e.valueToFloat64(value)
							 | 
						|
									if err != nil {
							 | 
						|
										return nil, fmt.Errorf("CEIL function conversion error: %v", err)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									result := math.Ceil(num)
							 | 
						|
								
							 | 
						|
									return &schema_pb.Value{
							 | 
						|
										Kind: &schema_pb.Value_Int64Value{Int64Value: int64(result)},
							 | 
						|
									}, nil
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Floor returns the largest integer less than or equal to the value
							 | 
						|
								func (e *SQLEngine) Floor(value *schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									if value == nil {
							 | 
						|
										return nil, fmt.Errorf("FLOOR function requires non-null value")
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									num, err := e.valueToFloat64(value)
							 | 
						|
									if err != nil {
							 | 
						|
										return nil, fmt.Errorf("FLOOR function conversion error: %v", err)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									result := math.Floor(num)
							 | 
						|
								
							 | 
						|
									return &schema_pb.Value{
							 | 
						|
										Kind: &schema_pb.Value_Int64Value{Int64Value: int64(result)},
							 | 
						|
									}, nil
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Abs returns the absolute value of a number
							 | 
						|
								func (e *SQLEngine) Abs(value *schema_pb.Value) (*schema_pb.Value, error) {
							 | 
						|
									if value == nil {
							 | 
						|
										return nil, fmt.Errorf("ABS function requires non-null value")
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									num, err := e.valueToFloat64(value)
							 | 
						|
									if err != nil {
							 | 
						|
										return nil, fmt.Errorf("ABS function conversion error: %v", err)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									result := math.Abs(num)
							 | 
						|
								
							 | 
						|
									// Return same type as input if possible
							 | 
						|
									if e.isIntegerValue(value) {
							 | 
						|
										return &schema_pb.Value{
							 | 
						|
											Kind: &schema_pb.Value_Int64Value{Int64Value: int64(result)},
							 | 
						|
										}, nil
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Check if original was float32
							 | 
						|
									if _, ok := value.Kind.(*schema_pb.Value_FloatValue); ok {
							 | 
						|
										return &schema_pb.Value{
							 | 
						|
											Kind: &schema_pb.Value_FloatValue{FloatValue: float32(result)},
							 | 
						|
										}, nil
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Default to double
							 | 
						|
									return &schema_pb.Value{
							 | 
						|
										Kind: &schema_pb.Value_DoubleValue{DoubleValue: result},
							 | 
						|
									}, nil
							 | 
						|
								}
							 |