2018-06-19 15:30:26 +02:00
|
|
|
// Copyright 2017 Google LLC
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package firestore
|
|
|
|
|
|
|
|
import (
|
2019-03-17 20:19:56 +01:00
|
|
|
"context"
|
2018-06-19 15:30:26 +02:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
2019-03-17 20:19:56 +01:00
|
|
|
"os"
|
2018-06-19 15:30:26 +02:00
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
2019-03-17 20:19:56 +01:00
|
|
|
vkit "cloud.google.com/go/firestore/apiv1"
|
2018-06-19 15:30:26 +02:00
|
|
|
"cloud.google.com/go/internal/version"
|
|
|
|
"github.com/golang/protobuf/ptypes"
|
2019-03-17 20:19:56 +01:00
|
|
|
gax "github.com/googleapis/gax-go/v2"
|
|
|
|
"google.golang.org/api/iterator"
|
2018-06-19 15:30:26 +02:00
|
|
|
"google.golang.org/api/option"
|
2019-03-17 20:19:56 +01:00
|
|
|
pb "google.golang.org/genproto/googleapis/firestore/v1"
|
|
|
|
"google.golang.org/grpc"
|
2018-06-19 15:30:26 +02:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/metadata"
|
|
|
|
"google.golang.org/grpc/status"
|
|
|
|
)
|
|
|
|
|
|
|
|
// resourcePrefixHeader is the name of the metadata header used to indicate
|
|
|
|
// the resource being operated on.
|
|
|
|
const resourcePrefixHeader = "google-cloud-resource-prefix"
|
|
|
|
|
|
|
|
// A Client provides access to the Firestore service.
|
|
|
|
type Client struct {
|
|
|
|
c *vkit.Client
|
|
|
|
projectID string
|
|
|
|
databaseID string // A client is tied to a single database.
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewClient creates a new Firestore client that uses the given project.
|
|
|
|
func NewClient(ctx context.Context, projectID string, opts ...option.ClientOption) (*Client, error) {
|
2019-03-17 20:19:56 +01:00
|
|
|
var o []option.ClientOption
|
|
|
|
// Environment variables for gcloud emulator:
|
|
|
|
// https://cloud.google.com/sdk/gcloud/reference/beta/emulators/firestore/
|
|
|
|
if addr := os.Getenv("FIRESTORE_EMULATOR_HOST"); addr != "" {
|
|
|
|
conn, err := grpc.Dial(addr, grpc.WithInsecure())
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("firestore: dialing address from env var FIRESTORE_EMULATOR_HOST: %v", err)
|
|
|
|
}
|
|
|
|
o = []option.ClientOption{option.WithGRPCConn(conn)}
|
|
|
|
}
|
|
|
|
o = append(o, opts...)
|
|
|
|
vc, err := vkit.NewClient(ctx, o...)
|
2018-06-19 15:30:26 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
vc.SetGoogleClientInfo("gccl", version.Repo)
|
|
|
|
c := &Client{
|
|
|
|
c: vc,
|
|
|
|
projectID: projectID,
|
|
|
|
databaseID: "(default)", // always "(default)", for now
|
|
|
|
}
|
|
|
|
return c, nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close closes any resources held by the client.
|
|
|
|
//
|
|
|
|
// Close need not be called at program exit.
|
|
|
|
func (c *Client) Close() error {
|
|
|
|
return c.c.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) path() string {
|
|
|
|
return fmt.Sprintf("projects/%s/databases/%s", c.projectID, c.databaseID)
|
|
|
|
}
|
|
|
|
|
|
|
|
func withResourceHeader(ctx context.Context, resource string) context.Context {
|
|
|
|
md, _ := metadata.FromOutgoingContext(ctx)
|
|
|
|
md = md.Copy()
|
|
|
|
md[resourcePrefixHeader] = []string{resource}
|
|
|
|
return metadata.NewOutgoingContext(ctx, md)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Collection creates a reference to a collection with the given path.
|
|
|
|
// A path is a sequence of IDs separated by slashes.
|
|
|
|
//
|
|
|
|
// Collection returns nil if path contains an even number of IDs or any ID is empty.
|
|
|
|
func (c *Client) Collection(path string) *CollectionRef {
|
|
|
|
coll, _ := c.idsToRef(strings.Split(path, "/"), c.path())
|
|
|
|
return coll
|
|
|
|
}
|
|
|
|
|
|
|
|
// Doc creates a reference to a document with the given path.
|
|
|
|
// A path is a sequence of IDs separated by slashes.
|
|
|
|
//
|
|
|
|
// Doc returns nil if path contains an odd number of IDs or any ID is empty.
|
|
|
|
func (c *Client) Doc(path string) *DocumentRef {
|
|
|
|
_, doc := c.idsToRef(strings.Split(path, "/"), c.path())
|
|
|
|
return doc
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) idsToRef(IDs []string, dbPath string) (*CollectionRef, *DocumentRef) {
|
|
|
|
if len(IDs) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
for _, id := range IDs {
|
|
|
|
if id == "" {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
coll := newTopLevelCollRef(c, dbPath, IDs[0])
|
|
|
|
i := 1
|
|
|
|
for i < len(IDs) {
|
|
|
|
doc := newDocRef(coll, IDs[i])
|
|
|
|
i++
|
|
|
|
if i == len(IDs) {
|
|
|
|
return nil, doc
|
|
|
|
}
|
|
|
|
coll = newCollRefWithParent(c, doc, IDs[i])
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
return coll, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetAll retrieves multiple documents with a single call. The DocumentSnapshots are
|
|
|
|
// returned in the order of the given DocumentRefs.
|
|
|
|
//
|
|
|
|
// If a document is not present, the corresponding DocumentSnapshot's Exists method will return false.
|
|
|
|
func (c *Client) GetAll(ctx context.Context, docRefs []*DocumentRef) ([]*DocumentSnapshot, error) {
|
|
|
|
return c.getAll(ctx, docRefs, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) getAll(ctx context.Context, docRefs []*DocumentRef, tid []byte) ([]*DocumentSnapshot, error) {
|
|
|
|
var docNames []string
|
|
|
|
docIndex := map[string]int{} // doc name to position in docRefs
|
|
|
|
for i, dr := range docRefs {
|
|
|
|
if dr == nil {
|
|
|
|
return nil, errNilDocRef
|
|
|
|
}
|
|
|
|
docNames = append(docNames, dr.Path)
|
|
|
|
docIndex[dr.Path] = i
|
|
|
|
}
|
|
|
|
req := &pb.BatchGetDocumentsRequest{
|
|
|
|
Database: c.path(),
|
|
|
|
Documents: docNames,
|
|
|
|
}
|
|
|
|
if tid != nil {
|
|
|
|
req.ConsistencySelector = &pb.BatchGetDocumentsRequest_Transaction{tid}
|
|
|
|
}
|
|
|
|
streamClient, err := c.c.BatchGetDocuments(withResourceHeader(ctx, req.Database), req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read and remember all results from the stream.
|
|
|
|
var resps []*pb.BatchGetDocumentsResponse
|
|
|
|
for {
|
|
|
|
resp, err := streamClient.Recv()
|
|
|
|
if err == io.EOF {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
resps = append(resps, resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Results may arrive out of order. Put each at the right index.
|
|
|
|
docs := make([]*DocumentSnapshot, len(docNames))
|
|
|
|
for _, resp := range resps {
|
|
|
|
var (
|
|
|
|
i int
|
|
|
|
doc *pb.Document
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
switch r := resp.Result.(type) {
|
|
|
|
case *pb.BatchGetDocumentsResponse_Found:
|
|
|
|
i = docIndex[r.Found.Name]
|
|
|
|
doc = r.Found
|
|
|
|
case *pb.BatchGetDocumentsResponse_Missing:
|
|
|
|
i = docIndex[r.Missing]
|
|
|
|
doc = nil
|
|
|
|
default:
|
|
|
|
return nil, errors.New("firestore: unknown BatchGetDocumentsResponse result type")
|
|
|
|
}
|
|
|
|
if docs[i] != nil {
|
|
|
|
return nil, fmt.Errorf("firestore: %q seen twice", docRefs[i].Path)
|
|
|
|
}
|
|
|
|
docs[i], err = newDocumentSnapshot(docRefs[i], doc, c, resp.ReadTime)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return docs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Collections returns an interator over the top-level collections.
|
|
|
|
func (c *Client) Collections(ctx context.Context) *CollectionIterator {
|
|
|
|
it := &CollectionIterator{
|
|
|
|
client: c,
|
|
|
|
it: c.c.ListCollectionIds(
|
|
|
|
withResourceHeader(ctx, c.path()),
|
2019-03-17 20:19:56 +01:00
|
|
|
&pb.ListCollectionIdsRequest{Parent: c.path() + "/documents"}),
|
2018-06-19 15:30:26 +02:00
|
|
|
}
|
|
|
|
it.pageInfo, it.nextFunc = iterator.NewPageInfo(
|
|
|
|
it.fetch,
|
|
|
|
func() int { return len(it.items) },
|
|
|
|
func() interface{} { b := it.items; it.items = nil; return b })
|
|
|
|
return it
|
|
|
|
}
|
|
|
|
|
|
|
|
// Batch returns a WriteBatch.
|
|
|
|
func (c *Client) Batch() *WriteBatch {
|
|
|
|
return &WriteBatch{c: c}
|
|
|
|
}
|
|
|
|
|
|
|
|
// commit calls the Commit RPC outside of a transaction.
|
|
|
|
func (c *Client) commit(ctx context.Context, ws []*pb.Write) ([]*WriteResult, error) {
|
|
|
|
req := &pb.CommitRequest{
|
|
|
|
Database: c.path(),
|
|
|
|
Writes: ws,
|
|
|
|
}
|
|
|
|
res, err := c.c.Commit(withResourceHeader(ctx, req.Database), req)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if len(res.WriteResults) == 0 {
|
|
|
|
return nil, errors.New("firestore: missing WriteResult")
|
|
|
|
}
|
|
|
|
var wrs []*WriteResult
|
|
|
|
for _, pwr := range res.WriteResults {
|
|
|
|
wr, err := writeResultFromProto(pwr)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
wrs = append(wrs, wr)
|
|
|
|
}
|
|
|
|
return wrs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Client) commit1(ctx context.Context, ws []*pb.Write) (*WriteResult, error) {
|
|
|
|
wrs, err := c.commit(ctx, ws)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return wrs[0], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// A WriteResult is returned by methods that write documents.
|
|
|
|
type WriteResult struct {
|
|
|
|
// The time at which the document was updated, or created if it did not
|
|
|
|
// previously exist. Writes that do not actually change the document do
|
|
|
|
// not change the update time.
|
|
|
|
UpdateTime time.Time
|
|
|
|
}
|
|
|
|
|
|
|
|
func writeResultFromProto(wr *pb.WriteResult) (*WriteResult, error) {
|
|
|
|
t, err := ptypes.Timestamp(wr.UpdateTime)
|
|
|
|
if err != nil {
|
|
|
|
t = time.Time{}
|
|
|
|
// TODO(jba): Follow up if Delete is supposed to return a nil timestamp.
|
|
|
|
}
|
|
|
|
return &WriteResult{UpdateTime: t}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func sleep(ctx context.Context, dur time.Duration) error {
|
|
|
|
switch err := gax.Sleep(ctx, dur); err {
|
|
|
|
case context.Canceled:
|
|
|
|
return status.Error(codes.Canceled, "context canceled")
|
|
|
|
case context.DeadlineExceeded:
|
|
|
|
return status.Error(codes.DeadlineExceeded, "context deadline exceeded")
|
|
|
|
default:
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|