gorm/clause/where.go

191 lines
4.0 KiB
Go
Raw Normal View History

2020-02-02 09:40:44 +03:00
package clause
import (
"strings"
)
const (
AndWithSpace = " AND "
OrWithSpace = " OR "
)
2020-02-02 09:40:44 +03:00
// Where where clause
type Where struct {
2020-02-07 18:45:35 +03:00
Exprs []Expression
2020-02-02 09:40:44 +03:00
}
// Name where clause name
func (where Where) Name() string {
return "WHERE"
}
// Build build where clause
func (where Where) Build(builder Builder) {
2020-02-07 18:45:35 +03:00
// Switch position if the first query expression is a single Or condition
for idx, expr := range where.Exprs {
2020-06-06 17:52:08 +03:00
if v, ok := expr.(OrConditions); !ok || len(v.Exprs) > 1 {
2020-02-07 18:45:35 +03:00
if idx != 0 {
where.Exprs[0], where.Exprs[idx] = where.Exprs[idx], where.Exprs[0]
2020-02-02 09:40:44 +03:00
}
2020-02-07 18:45:35 +03:00
break
2020-02-02 09:40:44 +03:00
}
}
buildExprs(where.Exprs, builder, AndWithSpace)
}
func buildExprs(exprs []Expression, builder Builder, joinCond string) {
wrapInParentheses := false
for idx, expr := range exprs {
2020-06-06 17:52:08 +03:00
if idx > 0 {
if v, ok := expr.(OrConditions); ok && len(v.Exprs) == 1 {
builder.WriteString(OrWithSpace)
2020-06-06 17:52:08 +03:00
} else {
builder.WriteString(joinCond)
2020-02-02 09:40:44 +03:00
}
}
2020-06-06 17:52:08 +03:00
if len(exprs) > 1 {
switch v := expr.(type) {
case OrConditions:
if len(v.Exprs) == 1 {
if e, ok := v.Exprs[0].(Expr); ok {
sql := strings.ToUpper(e.SQL)
wrapInParentheses = strings.Contains(sql, AndWithSpace) || strings.Contains(sql, OrWithSpace)
}
}
case AndConditions:
if len(v.Exprs) == 1 {
if e, ok := v.Exprs[0].(Expr); ok {
sql := strings.ToUpper(e.SQL)
wrapInParentheses = strings.Contains(sql, AndWithSpace) || strings.Contains(sql, OrWithSpace)
}
}
case Expr:
sql := strings.ToUpper(v.SQL)
wrapInParentheses = strings.Contains(sql, AndWithSpace) || strings.Contains(sql, OrWithSpace)
case NamedExpr:
sql := strings.ToUpper(v.SQL)
wrapInParentheses = strings.Contains(sql, AndWithSpace) || strings.Contains(sql, OrWithSpace)
}
}
if wrapInParentheses {
builder.WriteByte('(')
expr.Build(builder)
builder.WriteByte(')')
wrapInParentheses = false
} else {
expr.Build(builder)
}
2020-02-02 09:40:44 +03:00
}
2020-02-07 18:45:35 +03:00
}
// MergeClause merge where clauses
func (where Where) MergeClause(clause *Clause) {
if w, ok := clause.Expression.(Where); ok {
2020-06-08 04:10:27 +03:00
exprs := make([]Expression, len(w.Exprs)+len(where.Exprs))
copy(exprs, w.Exprs)
copy(exprs[len(w.Exprs):], where.Exprs)
where.Exprs = exprs
2020-02-07 18:45:35 +03:00
}
clause.Expression = where
}
func And(exprs ...Expression) Expression {
if len(exprs) == 0 {
return nil
}
if len(exprs) == 1 {
if _, ok := exprs[0].(OrConditions); !ok {
return exprs[0]
}
}
2020-02-07 18:45:35 +03:00
return AndConditions{Exprs: exprs}
}
type AndConditions struct {
Exprs []Expression
}
func (and AndConditions) Build(builder Builder) {
if len(and.Exprs) > 1 {
2020-02-13 19:09:44 +03:00
builder.WriteByte('(')
buildExprs(and.Exprs, builder, AndWithSpace)
2020-02-13 19:09:44 +03:00
builder.WriteByte(')')
} else {
buildExprs(and.Exprs, builder, AndWithSpace)
2020-02-07 18:45:35 +03:00
}
}
2020-02-02 09:40:44 +03:00
2020-02-07 18:45:35 +03:00
func Or(exprs ...Expression) Expression {
if len(exprs) == 0 {
return nil
2020-02-02 09:40:44 +03:00
}
2020-02-07 18:45:35 +03:00
return OrConditions{Exprs: exprs}
}
2020-02-02 09:40:44 +03:00
2020-02-07 18:45:35 +03:00
type OrConditions struct {
Exprs []Expression
}
func (or OrConditions) Build(builder Builder) {
if len(or.Exprs) > 1 {
2020-02-13 19:09:44 +03:00
builder.WriteByte('(')
buildExprs(or.Exprs, builder, OrWithSpace)
2020-02-13 19:09:44 +03:00
builder.WriteByte(')')
} else {
buildExprs(or.Exprs, builder, OrWithSpace)
2020-02-07 18:45:35 +03:00
}
}
func Not(exprs ...Expression) Expression {
if len(exprs) == 0 {
return nil
}
return NotConditions{Exprs: exprs}
}
type NotConditions struct {
Exprs []Expression
2020-02-02 09:40:44 +03:00
}
2020-02-07 18:45:35 +03:00
func (not NotConditions) Build(builder Builder) {
if len(not.Exprs) > 1 {
2020-02-13 19:09:44 +03:00
builder.WriteByte('(')
2020-02-07 18:45:35 +03:00
}
2020-02-07 18:45:35 +03:00
for idx, c := range not.Exprs {
if idx > 0 {
builder.WriteString(AndWithSpace)
2020-02-07 18:45:35 +03:00
}
if negationBuilder, ok := c.(NegationExpressionBuilder); ok {
negationBuilder.NegationBuild(builder)
} else {
2020-07-05 06:53:10 +03:00
builder.WriteString("NOT ")
e, wrapInParentheses := c.(Expr)
if wrapInParentheses {
sql := strings.ToUpper(e.SQL)
if wrapInParentheses = strings.Contains(sql, AndWithSpace) || strings.Contains(sql, OrWithSpace); wrapInParentheses {
builder.WriteByte('(')
}
}
2020-02-07 18:45:35 +03:00
c.Build(builder)
if wrapInParentheses {
builder.WriteByte(')')
}
2020-02-07 18:45:35 +03:00
}
}
2020-02-07 18:45:35 +03:00
if len(not.Exprs) > 1 {
2020-02-13 19:09:44 +03:00
builder.WriteByte(')')
2020-02-02 09:40:44 +03:00
}
}