package common import ( "context" "go.mongodb.org/mongo-driver/mongo" "github.com/mainnika/mongox-go-driver/mongox" "github.com/mainnika/mongox-go-driver/mongox/base" "github.com/mainnika/mongox-go-driver/mongox/errors" "github.com/mainnika/mongox-go-driver/mongox/query" ) // StreamLoader is a controller for a database cursor type StreamLoader struct { *mongo.Cursor ctx context.Context target interface{} } // DecodeNext loads next documents to a target or returns an error func (l *StreamLoader) DecodeNext() error { hasNext := l.Cursor.Next(l.ctx) if !hasNext { return errors.NotFoundErrorf("%s", mongo.ErrNoDocuments) } base.Reset(l.target) err := l.Decode(l.target) if err != nil { return errors.InternalErrorf("can't decode desult: %s", err) } return nil } // Next loads next documents but doesn't perform decoding func (l *StreamLoader) Next() error { hasNext := l.Cursor.Next(l.ctx) if !hasNext { return errors.NotFoundErrorf("%s", mongo.ErrNoDocuments) } return nil } // Close cursor func (l *StreamLoader) Close() error { return l.Cursor.Close(l.ctx) } // LoadStream function loads documents one by one into a target channel func LoadStream(db mongox.Database, target interface{}, filters ...interface{}) (*StreamLoader, error) { var cursor *mongo.Cursor var err error composed := query.Compose(filters...) hasPreloader, _ := composed.Preloader() if hasPreloader { cursor, err = createAggregateLoad(db, target, composed) } else { cursor, err = createSimpleLoad(db, target, composed) } if err != nil { return nil, errors.InternalErrorf("can't create find result: %s", err) } l := &StreamLoader{Cursor: cursor, ctx: db.Context(), target: target} return l, nil }