2019-04-16 14:58:56 +02:00
|
|
|
// 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"
|
|
|
|
|
|
|
|
"go.mongodb.org/mongo-driver/bson"
|
|
|
|
"go.mongodb.org/mongo-driver/bson/bsoncodec"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ErrNoDocuments is returned by Decode when an operation that returns a
|
|
|
|
// SingleResult doesn't return any documents.
|
|
|
|
var ErrNoDocuments = errors.New("mongo: no documents in result")
|
|
|
|
|
|
|
|
// SingleResult represents a single document returned from an operation. If
|
|
|
|
// the operation returned an error, the Err method of SingleResult will
|
|
|
|
// return that error.
|
|
|
|
type SingleResult struct {
|
|
|
|
err error
|
|
|
|
cur *Cursor
|
|
|
|
rdr bson.Raw
|
|
|
|
reg *bsoncodec.Registry
|
|
|
|
}
|
|
|
|
|
|
|
|
// Decode will attempt to decode the first document into v. If there was an
|
|
|
|
// error from the operation that created this SingleResult then the error
|
|
|
|
// will be returned. If there were no returned documents, ErrNoDocuments is
|
2019-11-27 15:29:34 +01:00
|
|
|
// returned. If v is nil or is a typed nil, an error will be returned.
|
2019-04-16 14:58:56 +02:00
|
|
|
func (sr *SingleResult) Decode(v interface{}) error {
|
|
|
|
if sr.err != nil {
|
|
|
|
return sr.err
|
|
|
|
}
|
|
|
|
if sr.reg == nil {
|
|
|
|
return bson.ErrNilRegistry
|
|
|
|
}
|
|
|
|
|
2019-11-27 15:29:34 +01:00
|
|
|
if sr.err = sr.setRdrContents(); sr.err != nil {
|
|
|
|
return sr.err
|
|
|
|
}
|
|
|
|
return bson.UnmarshalWithRegistry(sr.reg, sr.rdr, v)
|
2019-04-16 14:58:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// DecodeBytes will return a copy of the document as a bson.Raw. If there was an
|
|
|
|
// error from the operation that created this SingleResult then the error
|
2019-11-27 15:29:34 +01:00
|
|
|
// will be returned along with the result. If there were no returned documents, ErrNoDocuments is
|
2019-04-16 14:58:56 +02:00
|
|
|
// returned.
|
|
|
|
func (sr *SingleResult) DecodeBytes() (bson.Raw, error) {
|
2019-11-27 15:29:34 +01:00
|
|
|
if sr.err != nil {
|
|
|
|
return sr.rdr, sr.err
|
|
|
|
}
|
|
|
|
|
|
|
|
if sr.err = sr.setRdrContents(); sr.err != nil {
|
|
|
|
return nil, sr.err
|
|
|
|
}
|
|
|
|
return sr.rdr, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// setRdrContents will set the contents of rdr by iterating the underlying cursor if necessary.
|
|
|
|
func (sr *SingleResult) setRdrContents() error {
|
2019-04-16 14:58:56 +02:00
|
|
|
switch {
|
|
|
|
case sr.err != nil:
|
2019-11-27 15:29:34 +01:00
|
|
|
return sr.err
|
2019-04-16 14:58:56 +02:00
|
|
|
case sr.rdr != nil:
|
2019-11-27 15:29:34 +01:00
|
|
|
return nil
|
2019-04-16 14:58:56 +02:00
|
|
|
case sr.cur != nil:
|
|
|
|
defer sr.cur.Close(context.TODO())
|
|
|
|
if !sr.cur.Next(context.TODO()) {
|
|
|
|
if err := sr.cur.Err(); err != nil {
|
2019-11-27 15:29:34 +01:00
|
|
|
return err
|
2019-04-16 14:58:56 +02:00
|
|
|
}
|
2019-11-27 15:29:34 +01:00
|
|
|
|
|
|
|
return ErrNoDocuments
|
2019-04-16 14:58:56 +02:00
|
|
|
}
|
2019-11-27 15:29:34 +01:00
|
|
|
sr.rdr = sr.cur.Current
|
|
|
|
return nil
|
2019-04-16 14:58:56 +02:00
|
|
|
}
|
|
|
|
|
2019-11-27 15:29:34 +01:00
|
|
|
return ErrNoDocuments
|
2019-04-16 14:58:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Err will return the error from the operation that created this SingleResult.
|
|
|
|
// If there was no error, nil is returned.
|
|
|
|
func (sr *SingleResult) Err() error {
|
2019-11-27 15:29:34 +01:00
|
|
|
sr.err = sr.setRdrContents()
|
|
|
|
|
2019-04-16 14:58:56 +02:00
|
|
|
return sr.err
|
|
|
|
}
|