| Index: go/src/infra/gae/libs/gae/memory/raw_datstore.go
|
| diff --git a/go/src/infra/gae/libs/gae/memory/raw_datstore.go b/go/src/infra/gae/libs/gae/memory/raw_datstore.go
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..a9a10183005125d7a304e933677b7bfa2bce6c5a
|
| --- /dev/null
|
| +++ b/go/src/infra/gae/libs/gae/memory/raw_datstore.go
|
| @@ -0,0 +1,155 @@
|
| +// Copyright 2015 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +package memory
|
| +
|
| +import (
|
| + "errors"
|
| + "fmt"
|
| + "golang.org/x/net/context"
|
| +
|
| + "infra/gae/libs/gae"
|
| + "infra/gae/libs/gae/helper"
|
| +)
|
| +
|
| +//////////////////////////////////// public ////////////////////////////////////
|
| +
|
| +// useDS adds a gae.Datastore implementation to context, accessible
|
| +// by gae.GetDS(c)
|
| +func useDS(c context.Context) context.Context {
|
| + return gae.SetRDSFactory(c, func(ic context.Context) gae.RawDatastore {
|
| + dsd := cur(ic).Get(memContextDSIdx)
|
| +
|
| + switch x := dsd.(type) {
|
| + case *dataStoreData:
|
| + return &dsImpl{gae.DummyRDS(), x, curGID(ic).namespace, ic}
|
| + case *txnDataStoreData:
|
| + return &txnDsImpl{gae.DummyRDS(), x, curGID(ic).namespace}
|
| + default:
|
| + panic(fmt.Errorf("DS: bad type: %v in context %v", dsd, ic))
|
| + }
|
| + })
|
| +}
|
| +
|
| +//////////////////////////////////// dsImpl ////////////////////////////////////
|
| +
|
| +// dsImpl exists solely to bind the current c to the datastore data.
|
| +type dsImpl struct {
|
| + gae.RawDatastore
|
| +
|
| + data *dataStoreData
|
| + ns string
|
| + c context.Context
|
| +}
|
| +
|
| +var _ interface {
|
| + gae.RawDatastore
|
| + gae.Testable
|
| +} = (*dsImpl)(nil)
|
| +
|
| +func (d *dsImpl) BreakFeatures(err error, features ...string) {
|
| + d.data.BreakFeatures(err, features...)
|
| +}
|
| +func (d *dsImpl) UnbreakFeatures(features ...string) {
|
| + d.data.UnbreakFeatures(features...)
|
| +}
|
| +
|
| +func (d *dsImpl) NewKey(kind, stringID string, intID int64, parent gae.DSKey) gae.DSKey {
|
| + return helper.NewDSKey(globalAppID, d.ns, kind, stringID, intID, parent)
|
| +}
|
| +
|
| +func (d *dsImpl) Put(key gae.DSKey, src interface{}) (retKey gae.DSKey, err error) {
|
| + err = d.data.RunIfNotBroken(func() (err error) {
|
| + retKey, err = d.data.put(d.ns, key, src)
|
| + return
|
| + })
|
| + return
|
| +}
|
| +
|
| +func (d *dsImpl) Get(key gae.DSKey, dst interface{}) error {
|
| + return d.data.RunIfNotBroken(func() error {
|
| + return d.data.get(d.ns, key, dst)
|
| + })
|
| +}
|
| +
|
| +func (d *dsImpl) Delete(key gae.DSKey) error {
|
| + return d.data.RunIfNotBroken(func() error {
|
| + return d.data.del(d.ns, key)
|
| + })
|
| +}
|
| +
|
| +////////////////////////////////// txnDsImpl ///////////////////////////////////
|
| +
|
| +type txnDsImpl struct {
|
| + gae.RawDatastore
|
| +
|
| + data *txnDataStoreData
|
| + ns string
|
| +}
|
| +
|
| +var (
|
| + _ = gae.RawDatastore((*txnDsImpl)(nil))
|
| + _ = gae.Testable((*txnDsImpl)(nil))
|
| +)
|
| +
|
| +func (d *dsImpl) NewQuery(kind string) gae.DSQuery {
|
| + return &queryImpl{DSQuery: gae.DummyQY(), ns: d.ns, kind: kind}
|
| +}
|
| +
|
| +func (d *dsImpl) Run(q gae.DSQuery) gae.DSIterator {
|
| + rq := q.(*queryImpl)
|
| + rq = rq.normalize().checkCorrectness(d.ns, false)
|
| + return &queryIterImpl{rq}
|
| +}
|
| +
|
| +func (d *dsImpl) GetAll(q gae.DSQuery, dst interface{}) ([]gae.DSKey, error) {
|
| + // TODO(riannucci): assert that dst is a slice of structs
|
| + return nil, nil
|
| +}
|
| +
|
| +func (d *dsImpl) Count(q gae.DSQuery) (ret int, err error) {
|
| + itr := d.Run(q.KeysOnly())
|
| + for _, err = itr.Next(nil); err != nil; _, err = itr.Next(nil) {
|
| + ret++
|
| + }
|
| + if err == gae.ErrDSQueryDone {
|
| + err = nil
|
| + }
|
| + return
|
| +}
|
| +
|
| +func (d *txnDsImpl) BreakFeatures(err error, features ...string) {
|
| + d.data.BreakFeatures(err, features...)
|
| +}
|
| +func (d *txnDsImpl) UnbreakFeatures(features ...string) {
|
| + d.data.UnbreakFeatures(features...)
|
| +}
|
| +
|
| +func (d *txnDsImpl) NewKey(kind, stringID string, intID int64, parent gae.DSKey) gae.DSKey {
|
| + return helper.NewDSKey(globalAppID, d.ns, kind, stringID, intID, parent)
|
| +}
|
| +
|
| +func (d *txnDsImpl) Put(key gae.DSKey, src interface{}) (retKey gae.DSKey, err error) {
|
| + err = d.data.RunIfNotBroken(func() (err error) {
|
| + retKey, err = d.data.put(d.ns, key, src)
|
| + return
|
| + })
|
| + return
|
| +}
|
| +
|
| +func (d *txnDsImpl) Get(key gae.DSKey, dst interface{}) error {
|
| + return d.data.RunIfNotBroken(func() error {
|
| + return d.data.get(d.ns, key, dst)
|
| + })
|
| +}
|
| +
|
| +func (d *txnDsImpl) Delete(key gae.DSKey) error {
|
| + return d.data.RunIfNotBroken(func() error {
|
| + return d.data.del(d.ns, key)
|
| + })
|
| +}
|
| +
|
| +func (*txnDsImpl) RunInTransaction(func(c context.Context) error, *gae.DSTransactionOptions) error {
|
| + return errors.New("datastore: nested transactions are not supported")
|
| +}
|
|
|