mgocrud/crud.go

299 lines
6.5 KiB
Go
Raw Normal View History

2018-03-20 14:16:01 +01:00
package mgocrud
import (
"fmt"
"reflect"
2020-11-28 20:34:06 +01:00
"runtime/debug"
2018-03-20 14:16:01 +01:00
"strings"
"time"
"gopkg.in/mgo.v2/bson"
)
2021-03-01 13:53:26 +01:00
// ErrorWithStack is error which stored its stack trace
2020-11-28 20:34:06 +01:00
type ErrorWithStack struct {
Message string
StackTrace string
}
2021-03-01 13:53:26 +01:00
// Error returns error message
2020-11-28 20:34:06 +01:00
func (e *ErrorWithStack) Error() string {
return e.Message
}
2021-03-01 13:53:26 +01:00
// Stack returns strack trace to error origin
2020-11-28 20:34:06 +01:00
func (e *ErrorWithStack) Stack() string {
return e.StackTrace
}
2018-03-20 14:16:01 +01:00
// CreateDocument creates a document from specified model
2022-02-09 15:11:15 +01:00
func CreateDocument(db *Database, m ModelInterface) error {
2018-03-20 14:16:01 +01:00
m.PrepareInsert()
c := db.C(GetCollectionName(m))
err := c.Insert(m)
return err
}
// ReadDocument gets one document via its id
2022-02-09 15:11:15 +01:00
func ReadDocument(db *Database, m ModelInterface, selector bson.M) error {
2018-03-20 14:16:01 +01:00
c := db.C(GetCollectionName(m))
q := c.FindId(m.GetID())
if selector != nil {
q = q.Select(selector)
}
err := q.One(m)
return err
}
// PipelineModifierFunction is a function to modify mongodb query
type PipelineModifierFunction func(pipeline []bson.M) []bson.M
2020-11-28 14:51:28 +01:00
func convertIDValue(v reflect.Value) reflect.Value {
v = reflect.ValueOf(v.Interface())
vKind := v.Kind() // mapVal.Kind() does not work, no idea why ;(
2020-11-28 21:21:24 +01:00
//spew.Dump(v.Interface())
//spew.Dump(vKind)
2020-11-28 14:51:28 +01:00
switch vKind {
case reflect.String:
2020-11-28 21:21:24 +01:00
if hex, ok := v.Interface().(string); ok {
return reflect.ValueOf(bson.ObjectIdHex(hex))
}
2020-11-28 14:51:28 +01:00
case reflect.Map:
for _, key := range v.MapKeys() {
v.SetMapIndex(key, convertIDValue(v.MapIndex(key)))
}
case reflect.Slice:
for i := 0; i < v.Len(); i++ {
v.Index(i).Set(convertIDValue(v.Index(i)))
}
}
return v
}
func idToObjectID(filter interface{}) {
2021-08-31 19:41:24 +02:00
if filter == nil {
return
}
2020-11-28 21:21:24 +01:00
//spew.Dump(filter)
2020-11-28 14:51:28 +01:00
val := reflect.ValueOf(filter)
switch reflect.TypeOf(filter).Kind() {
case reflect.Slice:
for i := 0; i < val.Len(); i++ {
idToObjectID(val.Index(i).Interface())
}
case reflect.Ptr:
idToObjectID(reflect.Indirect(val).Interface())
case reflect.Map:
for _, key := range val.MapKeys() {
if k, ok := key.Interface().(string); ok {
mapVal := val.MapIndex(key)
if mapVal.Type().Kind() == reflect.Ptr {
mapVal = reflect.Indirect(mapVal)
}
if k == "_id" {
val.SetMapIndex(key, convertIDValue(mapVal))
} else {
idToObjectID(mapVal.Interface())
}
}
}
}
/*
for key, data := range filter {
if key == "_id" {
if hex, ok := data.(string); ok {
filter[key] = bson.ObjectIdHex(hex)
}
} else {
switch d := data.(type) {
case map[string]interface{}:
idToObjectID(d)
case bson.M:
idToObjectID(d)
case []map[string]interface{}:
for _, s := range d {
idToObjectID(s)
}
case []bson.M:
for _, s := range d {
idToObjectID(s)
}
case []interface{}:
for _, s := range d {
isToObjectID(s)
}
}
}
}
*/
}
2018-03-20 14:16:01 +01:00
// ReadCollection gets the filtered collection of the model
2022-02-09 15:11:15 +01:00
func ReadCollection(db *Database, results interface{}, filter bson.M, selector bson.M, offset int, limit int, sort []string, pipelineModifier PipelineModifierFunction) (err error) {
2018-03-20 14:16:01 +01:00
defer func() {
if r := recover(); r != nil {
2020-11-28 20:34:06 +01:00
err = &ErrorWithStack{Message: fmt.Sprintf("%v", r), StackTrace: string(debug.Stack())}
2018-03-20 14:16:01 +01:00
}
}()
mSlice := reflect.Indirect(reflect.ValueOf(results)) // results is *[]ModelInterface, so we need to indirect
var m ModelInterface
if mSlice.Len() > 0 {
// use existing first element
m = mSlice.Index(0).Interface().(ModelInterface)
} else {
// create new element to get collection name
m = reflect.New(reflect.TypeOf(mSlice.Interface()).Elem()).Interface().(ModelInterface)
}
/*
// get pointer to model (element of slice in results) to get collection name
m := reflect.New(reflect.TypeOf(
reflect.Indirect(reflect.ValueOf(results)).Interface(), // get indirection of slice pointer
).Elem()).Interface().(ModelInterface) // it must be a ModelInterface here
*/
2018-03-20 14:16:01 +01:00
c := db.C(GetCollectionName(m))
2020-11-28 20:34:06 +01:00
var _err error
2018-03-20 14:16:01 +01:00
if pipelineModifier != nil {
// search via pipeline
pipeline := []bson.M{}
if filter != nil {
pipeline = append(pipeline, bson.M{
"$match": filter,
})
}
if len(sort) > 0 {
sortM := bson.M{}
for _, s := range sort {
if strings.HasPrefix(s, "-") {
s = s[1:]
sortM[s] = -1
} else {
sortM[s] = 1
}
}
2021-03-01 13:53:26 +01:00
// spew.Dump(sortM)
2018-03-20 14:16:01 +01:00
pipeline = append(pipeline, bson.M{
"$sort": sortM,
})
}
if offset > 0 {
pipeline = append(pipeline, bson.M{
"$skip": offset,
})
}
if limit > 0 {
pipeline = append(pipeline, bson.M{
"$limit": limit,
})
}
if selector != nil {
pipeline = append(pipeline, bson.M{
"$project": selector,
})
}
if pipelineModifier != nil {
pipeline = pipelineModifier(pipeline)
}
2022-02-09 15:11:15 +01:00
_err = c.Pipe(pipeline).All(results)
2018-03-20 14:16:01 +01:00
} else {
// search without pipe is faster
2020-11-28 14:51:28 +01:00
idToObjectID(filter)
2021-03-01 13:53:26 +01:00
// spew.Dump(filter)
2020-11-28 14:51:28 +01:00
2018-03-20 14:16:01 +01:00
q := c.Find(filter)
if selector != nil {
q = q.Select(selector)
}
if len(sort) > 0 {
q = q.Sort(sort...)
}
if offset > 0 {
q = q.Skip(offset)
}
if limit > 0 {
q = q.Limit(limit)
}
2020-11-28 20:34:06 +01:00
_err = q.All(results)
2018-03-20 14:16:01 +01:00
}
2020-11-28 20:34:06 +01:00
if _err != nil {
return &ErrorWithStack{Message: _err.Error(), StackTrace: string(debug.Stack())}
}
2020-11-28 21:21:24 +01:00
return nil
2018-03-20 14:16:01 +01:00
}
// ReadCollectionCount gets the count of elements in filtered collection
2022-02-09 15:11:15 +01:00
func ReadCollectionCount(db *Database, m ModelInterface, filter bson.M) (count int, err error) {
2018-03-20 14:16:01 +01:00
defer func() {
if r := recover(); r != nil {
err = fmt.Errorf("%v", r)
}
}()
c := db.C(GetCollectionName(m))
return c.Find(filter).Count()
}
// UpdateDocument updates a document from specified model
2022-02-09 15:11:15 +01:00
func UpdateDocument(db *Database, m ModelInterface, changes bson.M) error {
2018-03-20 14:16:01 +01:00
m.PrepareUpdate()
changes["updateTime"] = time.Now()
c := db.C(GetCollectionName(m))
err := c.UpdateId(m.GetID(), bson.M{"$set": changes})
return err
}
// UpsertDocument updates a document from specified model or inserts it, of not found
2022-02-09 15:11:15 +01:00
func UpsertDocument(db *Database, m ModelInterface, changes bson.M) error {
2018-03-20 14:16:01 +01:00
m.PrepareUpdate()
changes["updateTime"] = time.Now()
c := db.C(GetCollectionName(m))
_, err := c.Upsert(m, bson.M{"$set": changes})
return err
}
// DeleteDocument deletes one document via its id
2022-02-09 15:11:15 +01:00
func DeleteDocument(db *Database, m ModelInterface) error {
2018-03-20 14:16:01 +01:00
c := db.C(GetCollectionName(m))
err := c.RemoveId(m.GetID())
return err
}
2021-03-01 13:38:03 +01:00
// DeleteDocuments deletes documents found by filter
2022-02-09 15:11:15 +01:00
func DeleteDocuments(db *Database, m ModelInterface, filter bson.M) (removed int, err error) {
2021-03-01 13:38:03 +01:00
c := db.C(GetCollectionName(m))
info, err := c.RemoveAll(filter)
if info != nil {
2022-02-09 15:11:15 +01:00
removed = info.Removed()
2021-03-01 13:38:03 +01:00
}
return removed, err
}