parent
35ffcc3f98
commit
e8cd680cde
@ -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…
Reference in new issue