VENDOR intensifies
This commit is contained in:
225
vendor/go.mongodb.org/mongo-driver/mongo/cursor.go
generated
vendored
Executable file
225
vendor/go.mongodb.org/mongo-driver/mongo/cursor.go
generated
vendored
Executable file
@@ -0,0 +1,225 @@
|
||||
// Copyright (C) MongoDB, Inc. 2017-present.
|
||||
//
|
||||
// Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||
// not use this file except in compliance with the License. You may obtain
|
||||
// a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
|
||||
|
||||
package mongo
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"io"
|
||||
"reflect"
|
||||
|
||||
"go.mongodb.org/mongo-driver/bson"
|
||||
"go.mongodb.org/mongo-driver/bson/bsoncodec"
|
||||
"go.mongodb.org/mongo-driver/x/bsonx/bsoncore"
|
||||
"go.mongodb.org/mongo-driver/x/mongo/driver/session"
|
||||
"go.mongodb.org/mongo-driver/x/mongo/driverlegacy"
|
||||
)
|
||||
|
||||
// Cursor is used to iterate a stream of documents. Each document is decoded into the result
|
||||
// according to the rules of the bson package.
|
||||
//
|
||||
// A typical usage of the Cursor type would be:
|
||||
//
|
||||
// var cur *Cursor
|
||||
// ctx := context.Background()
|
||||
// defer cur.Close(ctx)
|
||||
//
|
||||
// for cur.Next(ctx) {
|
||||
// elem := &bson.D{}
|
||||
// if err := cur.Decode(elem); err != nil {
|
||||
// log.Fatal(err)
|
||||
// }
|
||||
//
|
||||
// // do something with elem....
|
||||
// }
|
||||
//
|
||||
// if err := cur.Err(); err != nil {
|
||||
// log.Fatal(err)
|
||||
// }
|
||||
//
|
||||
type Cursor struct {
|
||||
// Current is the BSON bytes of the current document. This property is only valid until the next
|
||||
// call to Next or Close. If continued access is required to the bson.Raw, you must make a copy
|
||||
// of it.
|
||||
Current bson.Raw
|
||||
|
||||
bc batchCursor
|
||||
batch *bsoncore.DocumentSequence
|
||||
registry *bsoncodec.Registry
|
||||
clientSession *session.Client
|
||||
|
||||
err error
|
||||
}
|
||||
|
||||
func newCursor(bc batchCursor, registry *bsoncodec.Registry) (*Cursor, error) {
|
||||
return newCursorWithSession(bc, registry, nil)
|
||||
}
|
||||
|
||||
func newCursorWithSession(bc batchCursor, registry *bsoncodec.Registry, clientSession *session.Client) (*Cursor, error) {
|
||||
if registry == nil {
|
||||
registry = bson.DefaultRegistry
|
||||
}
|
||||
if bc == nil {
|
||||
return nil, errors.New("batch cursor must not be nil")
|
||||
}
|
||||
c := &Cursor{
|
||||
bc: bc,
|
||||
registry: registry,
|
||||
clientSession: clientSession,
|
||||
}
|
||||
if bc.ID() == 0 {
|
||||
c.closeImplicitSession()
|
||||
}
|
||||
return c, nil
|
||||
}
|
||||
|
||||
func newEmptyCursor() *Cursor {
|
||||
return &Cursor{bc: driverlegacy.NewEmptyBatchCursor()}
|
||||
}
|
||||
|
||||
// ID returns the ID of this cursor.
|
||||
func (c *Cursor) ID() int64 { return c.bc.ID() }
|
||||
|
||||
// Next gets the next result from this cursor. Returns true if there were no errors and the next
|
||||
// result is available for decoding.
|
||||
func (c *Cursor) Next(ctx context.Context) bool {
|
||||
if ctx == nil {
|
||||
ctx = context.Background()
|
||||
}
|
||||
doc, err := c.batch.Next()
|
||||
switch err {
|
||||
case nil:
|
||||
c.Current = bson.Raw(doc)
|
||||
return true
|
||||
case io.EOF: // Need to do a getMore
|
||||
default:
|
||||
c.err = err
|
||||
return false
|
||||
}
|
||||
|
||||
// call the Next method in a loop until at least one document is returned in the next batch or
|
||||
// the context times out.
|
||||
for {
|
||||
// If we don't have a next batch
|
||||
if !c.bc.Next(ctx) {
|
||||
// Do we have an error? If so we return false.
|
||||
c.err = c.bc.Err()
|
||||
if c.err != nil {
|
||||
return false
|
||||
}
|
||||
// Is the cursor ID zero?
|
||||
if c.bc.ID() == 0 {
|
||||
c.closeImplicitSession()
|
||||
return false
|
||||
}
|
||||
// empty batch, but cursor is still valid, so continue.
|
||||
continue
|
||||
}
|
||||
|
||||
// close the implicit session if this was the last getMore
|
||||
if c.bc.ID() == 0 {
|
||||
c.closeImplicitSession()
|
||||
}
|
||||
|
||||
c.batch = c.bc.Batch()
|
||||
doc, err = c.batch.Next()
|
||||
switch err {
|
||||
case nil:
|
||||
c.Current = bson.Raw(doc)
|
||||
return true
|
||||
case io.EOF: // Empty batch so we continue
|
||||
default:
|
||||
c.err = err
|
||||
return false
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Decode will decode the current document into val. If val is nil or is a typed nil, an error will be returned.
|
||||
func (c *Cursor) Decode(val interface{}) error {
|
||||
return bson.UnmarshalWithRegistry(c.registry, c.Current, val)
|
||||
}
|
||||
|
||||
// Err returns the current error.
|
||||
func (c *Cursor) Err() error { return c.err }
|
||||
|
||||
// Close closes this cursor.
|
||||
func (c *Cursor) Close(ctx context.Context) error {
|
||||
defer c.closeImplicitSession()
|
||||
return c.bc.Close(ctx)
|
||||
}
|
||||
|
||||
// All iterates the cursor and decodes each document into results.
|
||||
// The results parameter must be a pointer to a slice. The slice pointed to by results will be completely overwritten.
|
||||
// If the cursor has been iterated, any previously iterated documents will not be included in results.
|
||||
func (c *Cursor) All(ctx context.Context, results interface{}) error {
|
||||
resultsVal := reflect.ValueOf(results)
|
||||
if resultsVal.Kind() != reflect.Ptr {
|
||||
return errors.New("results argument must be a pointer to a slice")
|
||||
}
|
||||
|
||||
sliceVal := resultsVal.Elem()
|
||||
elementType := sliceVal.Type().Elem()
|
||||
var index int
|
||||
var err error
|
||||
|
||||
batch := c.batch // exhaust the current batch before iterating the batch cursor
|
||||
for {
|
||||
sliceVal, index, err = c.addFromBatch(sliceVal, elementType, batch, index)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if !c.bc.Next(ctx) {
|
||||
break
|
||||
}
|
||||
|
||||
batch = c.bc.Batch()
|
||||
}
|
||||
|
||||
if err = c.bc.Err(); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
resultsVal.Elem().Set(sliceVal.Slice(0, index))
|
||||
return nil
|
||||
}
|
||||
|
||||
// addFromBatch adds all documents from batch to sliceVal starting at the given index. It returns the new slice value,
|
||||
// the next empty index in the slice, and an error if one occurs.
|
||||
func (c *Cursor) addFromBatch(sliceVal reflect.Value, elemType reflect.Type, batch *bsoncore.DocumentSequence,
|
||||
index int) (reflect.Value, int, error) {
|
||||
|
||||
docs, err := batch.Documents()
|
||||
if err != nil {
|
||||
return sliceVal, index, err
|
||||
}
|
||||
|
||||
for _, doc := range docs {
|
||||
if sliceVal.Len() == index {
|
||||
// slice is full
|
||||
newElem := reflect.New(elemType)
|
||||
sliceVal = reflect.Append(sliceVal, newElem.Elem())
|
||||
sliceVal = sliceVal.Slice(0, sliceVal.Cap())
|
||||
}
|
||||
|
||||
currElem := sliceVal.Index(index).Addr().Interface()
|
||||
if err = bson.UnmarshalWithRegistry(c.registry, doc, currElem); err != nil {
|
||||
return sliceVal, index, err
|
||||
}
|
||||
|
||||
index++
|
||||
}
|
||||
|
||||
return sliceVal, index, nil
|
||||
}
|
||||
|
||||
func (c *Cursor) closeImplicitSession() {
|
||||
if c.clientSession != nil && c.clientSession.SessionType == session.Implicit {
|
||||
c.clientSession.EndSession()
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user