mirror of
https://github.com/mainnika/mongox-go-driver.git
synced 2026-05-22 15:53:36 +00:00
Convert stream loader to a controlled loader
This commit is contained in:
@@ -0,0 +1,65 @@
|
||||
package common
|
||||
|
||||
import (
|
||||
"context"
|
||||
|
||||
"github.com/mainnika/mongox-go-driver/mongox"
|
||||
"github.com/mainnika/mongox-go-driver/mongox/errors"
|
||||
"github.com/mainnika/mongox-go-driver/mongox/query"
|
||||
"github.com/mongodb/mongo-go-driver/mongo"
|
||||
"github.com/mongodb/mongo-go-driver/mongo/options"
|
||||
)
|
||||
|
||||
// ManyLoader is a controller for a database cursor
|
||||
type ManyLoader struct {
|
||||
mongo.Cursor
|
||||
ctx context.Context
|
||||
target interface{}
|
||||
}
|
||||
|
||||
// Get loads documents to a target or returns an error
|
||||
func (l *ManyLoader) Get() error {
|
||||
|
||||
hasNext := l.Next(l.ctx)
|
||||
|
||||
if !hasNext {
|
||||
return errors.NotFoundErrorf("%s", mongo.ErrNoDocuments)
|
||||
}
|
||||
|
||||
err := l.Decode(l.target)
|
||||
if err != nil {
|
||||
return errors.InternalErrorf("can't decode desult: %s", err)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// Close cursor
|
||||
func (l *ManyLoader) Close() error {
|
||||
|
||||
return l.Cursor.Close(l.ctx)
|
||||
}
|
||||
|
||||
// LoadMany function loads documents one by one into a target channel
|
||||
func LoadMany(db *mongox.Database, target interface{}, composed *query.Query) (*ManyLoader, error) {
|
||||
|
||||
collection := db.GetCollectionOf(target)
|
||||
opts := &options.FindOptions{}
|
||||
|
||||
if composed.Sorter() != nil {
|
||||
opts.Sort = composed.Sorter().Sort()
|
||||
}
|
||||
if composed.Limiter() != nil {
|
||||
limit := int64(composed.Limiter().Limit())
|
||||
opts.Limit = &limit
|
||||
}
|
||||
|
||||
cursor, err := collection.Find(db.Context(), composed.M(), opts)
|
||||
if err != nil {
|
||||
return nil, errors.InternalErrorf("can't create find result: %s", err)
|
||||
}
|
||||
|
||||
l := &ManyLoader{Cursor: cursor, ctx: db.Context(), target: target}
|
||||
|
||||
return l, nil
|
||||
}
|
||||
@@ -1,67 +0,0 @@
|
||||
package common
|
||||
|
||||
import (
|
||||
"reflect"
|
||||
|
||||
"github.com/mainnika/mongox-go-driver/mongox"
|
||||
"github.com/mainnika/mongox-go-driver/mongox/errors"
|
||||
"github.com/mainnika/mongox-go-driver/mongox/query"
|
||||
"github.com/mongodb/mongo-go-driver/mongo/options"
|
||||
)
|
||||
|
||||
// LoadStream function loads documents one by one into a target channel
|
||||
func LoadStream(db *mongox.Database, target interface{}, composed *query.Query) error {
|
||||
|
||||
targetV := reflect.ValueOf(target)
|
||||
targetT := targetV.Type()
|
||||
|
||||
targetK := targetV.Kind()
|
||||
if targetK != reflect.Chan {
|
||||
panic(errors.InternalErrorf("target is not a chan"))
|
||||
}
|
||||
if targetT.Elem().Kind() != reflect.Ptr {
|
||||
panic(errors.InternalErrorf("chan element should be a document ptr"))
|
||||
}
|
||||
|
||||
dummy := reflect.Zero(targetT.Elem())
|
||||
collection := db.GetCollectionOf(dummy.Interface())
|
||||
opts := &options.FindOptions{}
|
||||
|
||||
if composed.Sorter() != nil {
|
||||
opts.Sort = composed.Sorter().Sort()
|
||||
}
|
||||
if composed.Limiter() != nil {
|
||||
limit := int64(composed.Limiter().Limit())
|
||||
opts.Limit = &limit
|
||||
}
|
||||
|
||||
result, err := collection.Find(db.Context(), composed.M(), opts)
|
||||
if err != nil {
|
||||
return errors.InternalErrorf("can't create find result: %s", err)
|
||||
}
|
||||
|
||||
go func() {
|
||||
defer result.Close(db.Context())
|
||||
|
||||
for {
|
||||
elem, ok := targetV.Recv()
|
||||
if !ok {
|
||||
break
|
||||
}
|
||||
|
||||
if result.Next(db.Context()) != true {
|
||||
targetV.Send(dummy)
|
||||
break
|
||||
}
|
||||
|
||||
if result.Decode(elem.Interface()) != nil {
|
||||
targetV.Send(dummy)
|
||||
break
|
||||
}
|
||||
|
||||
targetV.Send(elem)
|
||||
}
|
||||
}()
|
||||
|
||||
return nil
|
||||
}
|
||||
Reference in New Issue
Block a user