Convert stream loader to a controlled loader

v2
Nikita Tokarchuk 5 years ago
parent 35ffcc3f98
commit e8cd680cde
  1. 65
      mongox/common/loadmany.go
  2. 67
      mongox/common/loadstream.go

@ -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
}
Loading…
Cancel
Save