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