343 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			343 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			Go
		
	
	
	
/*
 | 
						|
 * Copyright 2025 coze-dev Authors
 | 
						|
 *
 | 
						|
 * Licensed 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 selector
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"reflect"
 | 
						|
	"strings"
 | 
						|
)
 | 
						|
 | 
						|
type Predicate interface {
 | 
						|
	Resolve() (bool, error)
 | 
						|
}
 | 
						|
 | 
						|
type Clause struct {
 | 
						|
	LeftOperant  any
 | 
						|
	Op           Operator
 | 
						|
	RightOperant any
 | 
						|
}
 | 
						|
 | 
						|
type MultiClause struct {
 | 
						|
	Clauses  []*Clause
 | 
						|
	Relation ClauseRelation
 | 
						|
}
 | 
						|
 | 
						|
func (c *Clause) Resolve() (bool, error) {
 | 
						|
	leftV := c.LeftOperant
 | 
						|
	rightV := c.RightOperant
 | 
						|
 | 
						|
	leftT := reflect.TypeOf(leftV)
 | 
						|
	rightT := reflect.TypeOf(rightV)
 | 
						|
 | 
						|
	if err := c.Op.WillAccept(leftT, rightT); err != nil {
 | 
						|
		return false, err
 | 
						|
	}
 | 
						|
 | 
						|
	switch c.Op {
 | 
						|
	case OperatorEqual:
 | 
						|
		if leftV == nil && rightV == nil {
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftV == nil || rightV == nil {
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		leftV, rightV = alignNumberTypes(leftV, rightV, leftT, rightT)
 | 
						|
		return leftV == rightV, nil
 | 
						|
	case OperatorNotEqual:
 | 
						|
		if leftV == nil && rightV == nil {
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftV == nil || rightV == nil {
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		leftV, rightV = alignNumberTypes(leftV, rightV, leftT, rightT)
 | 
						|
		return leftV != rightV, nil
 | 
						|
	case OperatorEmpty:
 | 
						|
		if leftV == nil {
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftArray, ok := leftV.([]any); ok {
 | 
						|
			return len(leftArray) == 0, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftObj, ok := leftV.(map[string]any); ok {
 | 
						|
			return len(leftObj) == 0, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftStr, ok := leftV.(string); ok {
 | 
						|
			return len(leftStr) == 0 || leftStr == "None", nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftInt, ok := leftV.(int64); ok {
 | 
						|
			return leftInt == 0, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftFloat, ok := leftV.(float64); ok {
 | 
						|
			return leftFloat == 0, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftBool, ok := leftV.(bool); ok {
 | 
						|
			return !leftBool, nil
 | 
						|
		}
 | 
						|
 | 
						|
		return false, nil
 | 
						|
	case OperatorNotEmpty:
 | 
						|
		empty, err := (&Clause{LeftOperant: leftV, Op: OperatorEmpty}).Resolve()
 | 
						|
		return !empty, err
 | 
						|
	case OperatorGreater:
 | 
						|
		if leftV == nil {
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if rightV == nil {
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		leftV, rightV = alignNumberTypes(leftV, rightV, leftT, rightT)
 | 
						|
		if reflect.TypeOf(leftV).Kind() == reflect.Float64 {
 | 
						|
			return leftV.(float64) > rightV.(float64), nil
 | 
						|
		}
 | 
						|
		return leftV.(int64) > rightV.(int64), nil
 | 
						|
	case OperatorGreaterOrEqual:
 | 
						|
		if leftV == nil {
 | 
						|
			if rightV == nil {
 | 
						|
				return true, nil
 | 
						|
			}
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if rightV == nil {
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		leftV, rightV = alignNumberTypes(leftV, rightV, leftT, rightT)
 | 
						|
		if reflect.TypeOf(leftV).Kind() == reflect.Float64 {
 | 
						|
			return leftV.(float64) >= rightV.(float64), nil
 | 
						|
		}
 | 
						|
		return leftV.(int64) >= rightV.(int64), nil
 | 
						|
	case OperatorLesser:
 | 
						|
		if leftV == nil {
 | 
						|
			if rightV == nil {
 | 
						|
				return false, nil
 | 
						|
			}
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if rightV == nil {
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		leftV, rightV = alignNumberTypes(leftV, rightV, leftT, rightT)
 | 
						|
		if reflect.TypeOf(leftV).Kind() == reflect.Float64 {
 | 
						|
			return leftV.(float64) < rightV.(float64), nil
 | 
						|
		}
 | 
						|
		return leftV.(int64) < rightV.(int64), nil
 | 
						|
	case OperatorLesserOrEqual:
 | 
						|
		if leftV == nil {
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if rightV == nil {
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		leftV, rightV = alignNumberTypes(leftV, rightV, leftT, rightT)
 | 
						|
		if reflect.TypeOf(leftV).Kind() == reflect.Float64 {
 | 
						|
			return leftV.(float64) <= rightV.(float64), nil
 | 
						|
		}
 | 
						|
		return leftV.(int64) <= rightV.(int64), nil
 | 
						|
	case OperatorIsTrue:
 | 
						|
		if leftV == nil {
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		return leftV.(bool), nil
 | 
						|
	case OperatorIsFalse:
 | 
						|
		if leftV == nil {
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		return !leftV.(bool), nil
 | 
						|
	case OperatorLengthGreater:
 | 
						|
		if leftV == nil {
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		return int64(reflect.ValueOf(leftV).Len()) > rightV.(int64), nil
 | 
						|
	case OperatorLengthGreaterOrEqual:
 | 
						|
		if leftV == nil {
 | 
						|
			if rightV.(int64) == 0 {
 | 
						|
				return true, nil
 | 
						|
			}
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		return int64(reflect.ValueOf(leftV).Len()) >= rightV.(int64), nil
 | 
						|
	case OperatorLengthLesser:
 | 
						|
		if leftV == nil {
 | 
						|
			if rightV.(int64) == 0 {
 | 
						|
				return false, nil
 | 
						|
			}
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		return int64(reflect.ValueOf(leftV).Len()) < rightV.(int64), nil
 | 
						|
	case OperatorLengthLesserOrEqual:
 | 
						|
		if leftV == nil {
 | 
						|
			return true, nil
 | 
						|
		}
 | 
						|
 | 
						|
		return int64(reflect.ValueOf(leftV).Len()) <= rightV.(int64), nil
 | 
						|
	case OperatorContain:
 | 
						|
		if leftV == nil { // treat it as empty slice
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftT.Kind() == reflect.String {
 | 
						|
			return strings.Contains(fmt.Sprintf("%v", leftV), rightV.(string)), nil
 | 
						|
		}
 | 
						|
 | 
						|
		leftValue := reflect.ValueOf(leftV)
 | 
						|
		for i := 0; i < leftValue.Len(); i++ {
 | 
						|
			elem := leftValue.Index(i).Interface()
 | 
						|
			if elem == rightV {
 | 
						|
				return true, nil
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		return false, nil
 | 
						|
	case OperatorNotContain:
 | 
						|
		if leftV == nil { // treat it as empty slice
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftT.Kind() == reflect.String {
 | 
						|
			return !strings.Contains(fmt.Sprintf("%v", leftV), rightV.(string)), nil
 | 
						|
		}
 | 
						|
 | 
						|
		leftValue := reflect.ValueOf(leftV)
 | 
						|
		for i := 0; i < leftValue.Len(); i++ {
 | 
						|
			elem := leftValue.Index(i).Interface()
 | 
						|
			if elem == rightV {
 | 
						|
				return false, nil
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		return true, nil
 | 
						|
	case OperatorContainKey:
 | 
						|
		if leftV == nil { // treat it as empty map
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftT.Kind() == reflect.Map {
 | 
						|
			leftValue := reflect.ValueOf(leftV)
 | 
						|
			for _, key := range leftValue.MapKeys() {
 | 
						|
				if key.Interface() == rightV {
 | 
						|
					return true, nil
 | 
						|
				}
 | 
						|
			}
 | 
						|
		} else { // struct, unreachable now
 | 
						|
			for i := 0; i < leftT.NumField(); i++ {
 | 
						|
				field := leftT.Field(i)
 | 
						|
				if field.IsExported() {
 | 
						|
					tag := field.Tag.Get("json")
 | 
						|
					if tag == rightV {
 | 
						|
						return true, nil
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		return false, nil
 | 
						|
	case OperatorNotContainKey:
 | 
						|
		if leftV == nil { // treat it as empty map
 | 
						|
			return false, nil
 | 
						|
		}
 | 
						|
 | 
						|
		if leftT.Kind() == reflect.Map {
 | 
						|
			leftValue := reflect.ValueOf(leftV)
 | 
						|
			for _, key := range leftValue.MapKeys() {
 | 
						|
				if key.Interface() == rightV {
 | 
						|
					return false, nil
 | 
						|
				}
 | 
						|
			}
 | 
						|
		} else { // struct, unreachable now
 | 
						|
			for i := 0; i < leftT.NumField(); i++ {
 | 
						|
				field := leftT.Field(i)
 | 
						|
				if field.IsExported() {
 | 
						|
					tag := field.Tag.Get("json")
 | 
						|
					if tag == rightV {
 | 
						|
						return false, nil
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		return true, nil
 | 
						|
	default:
 | 
						|
		return false, fmt.Errorf("unknown operator: %v", c.Op)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (mc *MultiClause) Resolve() (bool, error) {
 | 
						|
	if mc.Relation == ClauseRelationAND {
 | 
						|
		for _, clause := range mc.Clauses {
 | 
						|
			isTrue, err := clause.Resolve()
 | 
						|
			if err != nil {
 | 
						|
				return false, err
 | 
						|
			}
 | 
						|
			if !isTrue {
 | 
						|
				return false, nil
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return true, nil
 | 
						|
	} else if mc.Relation == ClauseRelationOR {
 | 
						|
		for _, clause := range mc.Clauses {
 | 
						|
			isTrue, err := clause.Resolve()
 | 
						|
			if err != nil {
 | 
						|
				return false, err
 | 
						|
			}
 | 
						|
			if isTrue {
 | 
						|
				return true, nil
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return false, nil
 | 
						|
	} else {
 | 
						|
		return false, fmt.Errorf("unknown relation: %v", mc.Relation)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func alignNumberTypes(leftV, rightV any, leftT, rightT reflect.Type) (any, any) {
 | 
						|
	if leftT == reflect.TypeOf(int64(0)) {
 | 
						|
		if rightT == reflect.TypeOf(float64(0)) {
 | 
						|
			leftV = float64(leftV.(int64))
 | 
						|
		}
 | 
						|
	} else if leftT == reflect.TypeOf(float64(0)) {
 | 
						|
		if rightT == reflect.TypeOf(int64(0)) {
 | 
						|
			rightV = float64(rightV.(int64))
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	return leftV, rightV
 | 
						|
}
 |