242 lines
4.9 KiB
Go
242 lines
4.9 KiB
Go
package condition
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
"git.fsdpf.net/go/contracts/support"
|
|
"git.fsdpf.net/go/db"
|
|
"github.com/samber/lo"
|
|
)
|
|
|
|
type ConditionType string
|
|
|
|
const (
|
|
OR ConditionType = "OR"
|
|
AND ConditionType = "AND"
|
|
)
|
|
|
|
type Condition struct {
|
|
parent *Condition
|
|
typ ConditionType // 分组类型
|
|
describe string // 描述
|
|
exprs []*ConditionExpr // 分组表达式成员
|
|
childrens []*Condition // 分组子集
|
|
}
|
|
|
|
func (this *Condition) AppendTo(c *Condition) {
|
|
this.parent = c
|
|
}
|
|
|
|
// 条件类型
|
|
func (this Condition) Type() ConditionType {
|
|
return this.typ
|
|
}
|
|
|
|
// 判断条件为空
|
|
func (this Condition) IsEmpty() bool {
|
|
if len(this.exprs) > 0 {
|
|
return false
|
|
}
|
|
for _, cond := range this.childrens {
|
|
if !cond.IsEmpty() {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
// 判断条件不为空
|
|
func (this Condition) IsNotEmpty() bool {
|
|
return !this.IsEmpty()
|
|
}
|
|
|
|
// 判断条件是否恒成立
|
|
func (this Condition) IsAlwaysRight() bool {
|
|
flag := []bool{}
|
|
|
|
for _, expr := range this.exprs {
|
|
if expr.GetOperator() != EQ {
|
|
flag = append(flag, false)
|
|
continue
|
|
}
|
|
|
|
if expr.GetTokenType() != SQL {
|
|
flag = append(flag, false)
|
|
continue
|
|
}
|
|
|
|
eRight := fmt.Sprintf("%s.%s", expr.GetFieldResource(), expr.GetField()) == strings.ReplaceAll(expr.GetTokenName(), "`", "")
|
|
|
|
if this.typ == OR && eRight {
|
|
return true
|
|
}
|
|
|
|
flag = append(flag, eRight)
|
|
}
|
|
|
|
for _, cond := range this.childrens {
|
|
cRight := cond.IsAlwaysRight()
|
|
|
|
if this.typ == OR && cRight {
|
|
return true
|
|
}
|
|
|
|
flag = append(flag, cRight)
|
|
}
|
|
|
|
if len(flag) == 0 {
|
|
return true
|
|
}
|
|
|
|
return !lo.Some(flag, []bool{false})
|
|
}
|
|
|
|
// 生成 SQL 语句
|
|
func (this Condition) ToSql(m TokenValue) db.Expression {
|
|
conditions := []string{}
|
|
// 表达式
|
|
for _, item := range this.exprs {
|
|
if item.IsIgnoreEmptyParma(m) {
|
|
continue
|
|
}
|
|
conditions = append(conditions, string(item.ToSql(m)))
|
|
}
|
|
// 条件子集
|
|
for _, item := range this.childrens {
|
|
conditions = append(conditions, string(item.ToSql(m)))
|
|
}
|
|
// 去除无效的表达式
|
|
conditions = lo.Filter(conditions, func(item string, _ int) bool { return item != "" })
|
|
// 组合 SQL
|
|
sql := strings.Join(conditions, " "+string(this.Type())+" ")
|
|
|
|
if sql == "" {
|
|
if this.parent == nil && this.Type() == OR {
|
|
return db.Raw("false")
|
|
} else if this.parent == nil && this.Type() == AND {
|
|
return db.Raw("true")
|
|
}
|
|
return db.Raw("")
|
|
}
|
|
|
|
// 包裹 SQL, 避免语法表达错误
|
|
if this.parent == nil || this.Type() == OR {
|
|
sql = "(" + sql + ")"
|
|
}
|
|
|
|
if this.describe != "" {
|
|
sql += "/*" + this.describe + "*/"
|
|
}
|
|
|
|
return db.Raw(sql)
|
|
}
|
|
|
|
// 设置条件表达式
|
|
func (this *Condition) SetExpr(expr *ConditionExpr) *Condition {
|
|
expr.AppendTo(this)
|
|
this.exprs = append(this.exprs, expr)
|
|
return this
|
|
}
|
|
|
|
// 设置条件子集
|
|
func (this *Condition) SetCondition(c *Condition) *Condition {
|
|
c.AppendTo(this)
|
|
this.childrens = append(this.childrens, c)
|
|
return this
|
|
}
|
|
|
|
// 设置 Token 匹配前缀
|
|
func (this *Condition) SetMatchPrefix(prefix string) *Condition {
|
|
for _, expr := range this.exprs {
|
|
expr.SetMatchPrefix(prefix)
|
|
}
|
|
for _, cond := range this.childrens {
|
|
cond.SetMatchPrefix(prefix)
|
|
}
|
|
return this
|
|
}
|
|
|
|
func (this Condition) GetFieldsValue(m TokenValue, isWithResource bool) (result map[string]any) {
|
|
if this.IsEmpty() {
|
|
return
|
|
}
|
|
|
|
// 表达式
|
|
for _, item := range this.exprs {
|
|
if item.GetOperator() != EQ {
|
|
continue
|
|
}
|
|
|
|
vField := map[string]any{item.GetField(): item.GetTokenValue(m)}
|
|
|
|
if isWithResource {
|
|
vField = map[string]any{item.GetFieldResource(): vField}
|
|
}
|
|
|
|
result = support.MergeMap(result, vField)
|
|
}
|
|
// 条件子集
|
|
for _, item := range this.childrens {
|
|
result = support.MergeMap(result, item.GetFieldsValue(m, isWithResource))
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
/**
|
|
* @param contracts.ConditionOperator operator
|
|
* @param contracts.ConditionTokenType types
|
|
* @return map[string][string] // 如: {"TestA.field_a": "param"}
|
|
*/
|
|
func (this Condition) GetFields(operator ConditionOperator, types ...TokenType) map[string]string {
|
|
result := make(map[string]string)
|
|
|
|
if this.IsEmpty() {
|
|
return result
|
|
}
|
|
|
|
// 表达式
|
|
for _, item := range this.exprs {
|
|
if item.GetOperator() != operator || !lo.Contains(types, item.GetTokenType()) {
|
|
continue
|
|
}
|
|
|
|
result[fmt.Sprintf("%s.%s", item.GetFieldResource(), item.GetField())] = item.GetTokenName()
|
|
}
|
|
|
|
// 条件子集
|
|
for _, item := range this.childrens {
|
|
result = lo.Assign(result, item.GetFields(operator, types...))
|
|
}
|
|
|
|
return result
|
|
}
|
|
|
|
func (this *Condition) SetOption(opts ...Option) *Condition {
|
|
for i := 0; i < len(opts); i++ {
|
|
opts[i](this)
|
|
}
|
|
return this
|
|
}
|
|
|
|
type Option func(option *Condition)
|
|
|
|
func Type(v ConditionType) Option {
|
|
return func(option *Condition) {
|
|
option.typ = v
|
|
}
|
|
}
|
|
|
|
func Describe(v string) Option {
|
|
return func(option *Condition) {
|
|
option.describe = v
|
|
}
|
|
}
|
|
|
|
func New(opts ...Option) *Condition {
|
|
cond := &Condition{typ: AND}
|
|
cond.SetOption(opts...)
|
|
return cond
|
|
}
|