Files
oc-lib/dbs/dbs.go

176 lines
3.9 KiB
Go
Raw Normal View History

2024-07-18 11:51:12 +02:00
package dbs
import (
2024-08-01 09:13:10 +02:00
"fmt"
2026-01-26 10:36:15 +01:00
"runtime/debug"
2024-08-01 09:13:10 +02:00
"strings"
2024-07-18 11:51:12 +02:00
"go.mongodb.org/mongo-driver/bson"
)
2024-08-01 09:13:10 +02:00
type Operator int
const (
LIKE Operator = iota
EXISTS
IN
GTE
LTE
LT
GT
EQUAL
NOT
2026-01-26 10:36:15 +01:00
ELEMMATCH
2024-08-01 09:13:10 +02:00
)
var str = [...]string{
"like",
"exists",
"in",
"gte",
"lte",
"lt",
"gt",
"equal",
"not",
2026-01-26 10:36:15 +01:00
"elemMatch",
2024-08-01 09:13:10 +02:00
}
func (m Operator) String() string {
return str[m]
}
2026-01-26 10:36:15 +01:00
func (m Operator) ToMongoOperator(k string, value interface{}) bson.M {
defer func() {
if r := recover(); r != nil {
2026-01-26 10:36:15 +01:00
fmt.Println("Recovered. Error:\n", r, debug.Stack())
}
}()
2026-01-26 10:36:15 +01:00
defaultValue := bson.M{k: bson.M{"$regex": m.ToValueOperator(StringToOperator(m.String()), value)}}
2024-08-01 09:13:10 +02:00
switch m {
case LIKE:
2026-01-26 10:36:15 +01:00
return bson.M{k: bson.M{"$regex": m.ToValueOperator(StringToOperator(m.String()), value)}}
2024-08-01 09:13:10 +02:00
case EXISTS:
2026-01-26 10:36:15 +01:00
return bson.M{k: bson.M{"$exists": m.ToValueOperator(StringToOperator(m.String()), value)}}
2024-08-01 09:13:10 +02:00
case IN:
2026-01-26 10:36:15 +01:00
return bson.M{k: bson.M{"$in": m.ToValueOperator(StringToOperator(m.String()), value)}}
2024-08-01 09:13:10 +02:00
case GTE:
2026-01-26 10:36:15 +01:00
return bson.M{k: bson.M{"$gte": m.ToValueOperator(StringToOperator(m.String()), value)}}
2024-08-01 09:13:10 +02:00
case GT:
2026-01-26 10:36:15 +01:00
return bson.M{k: bson.M{"$gt": m.ToValueOperator(StringToOperator(m.String()), value)}}
2024-08-01 09:13:10 +02:00
case LTE:
2026-01-26 10:36:15 +01:00
return bson.M{k: bson.M{"$lte": m.ToValueOperator(StringToOperator(m.String()), value)}}
2024-08-01 09:13:10 +02:00
case LT:
2026-01-26 10:36:15 +01:00
return bson.M{k: bson.M{"$lt": m.ToValueOperator(StringToOperator(m.String()), value)}}
case ELEMMATCH:
return bson.M{k: bson.M{"$elemMatch": m.ToValueOperator(StringToOperator(m.String()), value)}}
2024-08-01 09:13:10 +02:00
case EQUAL:
2026-01-26 10:36:15 +01:00
return bson.M{k: value}
case NOT:
v := value.(Filters)
orList := bson.A{}
andList := bson.A{}
f := bson.D{}
for k, filter := range v.Or {
for _, ff := range filter {
orList = append(orList, StringToOperator(ff.Operator).ToMongoOperator(k, ff.Value))
}
}
for k, filter := range v.And {
for _, ff := range filter {
andList = append(andList, StringToOperator(ff.Operator).ToMongoOperator(k, ff.Value))
}
}
if len(orList) > 0 && len(andList) == 0 {
f = bson.D{{"$or", orList}}
} else {
if len(orList) > 0 {
andList = append(andList, bson.M{"$or": orList})
}
f = bson.D{{"$and", andList}}
}
2026-01-26 10:36:15 +01:00
return bson.M{"$not": f}
2024-08-01 09:13:10 +02:00
default:
return defaultValue
2024-08-02 13:10:27 +02:00
}
}
2026-01-26 10:36:15 +01:00
func StringToOperator(s string) Operator {
for i, v := range str {
if v == s {
return Operator(i)
}
2026-01-26 10:36:15 +01:00
}
return LIKE
}
func GetBson(filters *Filters) bson.D {
f := bson.D{}
orList := bson.A{}
andList := bson.A{}
if filters != nil {
for k, filter := range filters.Or {
for _, ff := range filter {
orList = append(orList, StringToOperator(ff.Operator).ToMongoOperator(k, ff.Value))
}
}
2026-01-26 10:36:15 +01:00
for k, filter := range filters.And {
for _, ff := range filter {
andList = append(andList, StringToOperator(ff.Operator).ToMongoOperator(k, ff.Value))
}
}
if len(orList) > 0 && len(andList) == 0 {
f = bson.D{{"$or", orList}}
} else {
if len(orList) > 0 {
andList = append(andList, bson.M{"$or": orList})
}
f = bson.D{{"$and", andList}}
}
2024-08-01 09:13:10 +02:00
}
2026-01-26 10:36:15 +01:00
return f
2024-08-01 09:13:10 +02:00
}
2026-01-26 10:36:15 +01:00
func (m Operator) ToValueOperator(operator Operator, value interface{}) interface{} {
switch value.(type) {
case *Filters:
return GetBson(value.(*Filters))
default:
if strings.TrimSpace(fmt.Sprintf("%v", value)) == "*" {
value = ""
}
if operator == LIKE {
return "(?i).*" + strings.TrimSpace(fmt.Sprintf("%v", value)) + ".*"
2024-08-01 09:13:10 +02:00
}
}
return value
}
type Filters struct {
2024-08-02 14:07:43 +02:00
And map[string][]Filter `json:"and"`
Or map[string][]Filter `json:"or"`
2024-08-01 09:13:10 +02:00
}
type Filter struct {
Operator string `json:"operator,omitempty"`
Value interface{} `json:"value,omitempty"`
}
2024-07-18 11:51:12 +02:00
type Input = map[string]interface{}
func InputToBson(i Input, isUpdate bool) bson.D {
input := bson.D{}
for k, v := range i {
if k == "id" {
input = append(input, bson.E{Key: "_id", Value: v})
} else {
input = append(input, bson.E{Key: k, Value: v})
}
}
if isUpdate {
return bson.D{{Key: "$set", Value: input}}
}
return input
}