2023-07-20 14:55:00 +08:00
|
|
|
package condition
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
2024-05-08 22:32:51 +08:00
|
|
|
"git.fsdpf.net/go/condition/contracts"
|
2023-07-20 14:55:00 +08:00
|
|
|
"git.fsdpf.net/go/contracts/support"
|
|
|
|
"git.fsdpf.net/go/db"
|
|
|
|
"github.com/samber/lo"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Condition struct {
|
|
|
|
parent contracts.Condition
|
|
|
|
typ contracts.ConditionType // 分组类型
|
|
|
|
describe string // 描述
|
|
|
|
exprs []contracts.ConditionExpr // 分组表达式成员
|
|
|
|
childrens []contracts.Condition // 分组子集
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *Condition) AppendTo(c contracts.Condition) {
|
|
|
|
this.parent = c
|
|
|
|
}
|
|
|
|
|
|
|
|
// 条件类型
|
|
|
|
func (this Condition) Type() contracts.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 {
|
2024-05-08 22:32:51 +08:00
|
|
|
if expr.GetOperator() != contracts.EQ {
|
2023-07-20 14:55:00 +08:00
|
|
|
flag = append(flag, false)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-05-08 22:32:51 +08:00
|
|
|
if expr.GetTokenType() != contracts.SQL {
|
2023-07-20 14:55:00 +08:00
|
|
|
flag = append(flag, false)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
eRight := fmt.Sprintf("%s.%s", expr.GetFieldResource(), expr.GetField()) == strings.ReplaceAll(expr.GetTokenName(), "`", "")
|
|
|
|
|
2024-05-08 22:32:51 +08:00
|
|
|
if this.typ == contracts.OR && eRight {
|
2023-07-20 14:55:00 +08:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
flag = append(flag, eRight)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, cond := range this.childrens {
|
|
|
|
cRight := cond.IsAlwaysRight()
|
|
|
|
|
2024-05-08 22:32:51 +08:00
|
|
|
if this.typ == contracts.OR && cRight {
|
2023-07-20 14:55:00 +08:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
flag = append(flag, cRight)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(flag) == 0 {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return !lo.Some(flag, []bool{false})
|
|
|
|
}
|
|
|
|
|
|
|
|
// 生成 SQL 语句
|
2024-05-08 22:32:51 +08:00
|
|
|
func (this Condition) ToSql(m contracts.ConditionTokenValue) db.Expression {
|
2023-07-20 14:55:00 +08:00
|
|
|
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 == "" {
|
2024-05-08 22:32:51 +08:00
|
|
|
if this.parent == nil && this.Type() == contracts.OR {
|
2023-07-20 14:55:00 +08:00
|
|
|
return db.Raw("false")
|
2024-05-08 22:32:51 +08:00
|
|
|
} else if this.parent == nil && this.Type() == contracts.AND {
|
2023-07-20 14:55:00 +08:00
|
|
|
return db.Raw("true")
|
|
|
|
}
|
|
|
|
return db.Raw("")
|
|
|
|
}
|
|
|
|
|
|
|
|
// 包裹 SQL, 避免语法表达错误
|
2024-05-08 22:32:51 +08:00
|
|
|
if this.parent == nil || this.Type() == contracts.OR {
|
2023-07-20 14:55:00 +08:00
|
|
|
sql = "(" + sql + ")"
|
|
|
|
}
|
|
|
|
|
|
|
|
if this.describe != "" {
|
|
|
|
sql += "/*" + this.describe + "*/"
|
|
|
|
}
|
|
|
|
|
|
|
|
return db.Raw(sql)
|
|
|
|
}
|
|
|
|
|
|
|
|
// 设置条件表达式
|
|
|
|
func (this *Condition) SetExpr(expr contracts.ConditionExpr) contracts.Condition {
|
|
|
|
expr.AppendTo(this)
|
|
|
|
this.exprs = append(this.exprs, expr)
|
|
|
|
return this
|
|
|
|
}
|
|
|
|
|
|
|
|
// 设置条件子集
|
|
|
|
func (this *Condition) SetCondition(c contracts.Condition) contracts.Condition {
|
|
|
|
c.AppendTo(this)
|
|
|
|
this.childrens = append(this.childrens, c)
|
|
|
|
return this
|
|
|
|
}
|
|
|
|
|
|
|
|
// 设置 Token 匹配前缀
|
|
|
|
func (this *Condition) SetMatchPrefix(prefix string) contracts.Condition {
|
|
|
|
for _, expr := range this.exprs {
|
|
|
|
expr.SetMatchPrefix(prefix)
|
|
|
|
}
|
|
|
|
for _, cond := range this.childrens {
|
|
|
|
cond.SetMatchPrefix(prefix)
|
|
|
|
}
|
|
|
|
return this
|
|
|
|
}
|
|
|
|
|
2024-05-08 22:32:51 +08:00
|
|
|
func (this Condition) GetFieldsValue(m contracts.ConditionTokenValue, isWithResource bool) (result map[string]any) {
|
2023-07-20 14:55:00 +08:00
|
|
|
if this.IsEmpty() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// 表达式
|
|
|
|
for _, item := range this.exprs {
|
2024-05-08 22:32:51 +08:00
|
|
|
if item.GetOperator() != contracts.EQ {
|
2023-07-20 14:55:00 +08:00
|
|
|
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 contracts.ConditionOperator, types ...contracts.ConditionTokenType) 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 New(typ contracts.ConditionType, describe string) contracts.Condition {
|
|
|
|
return &Condition{typ: typ, describe: describe}
|
|
|
|
}
|