Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(520)

Unified Diff: impl/prod/raw_datastore.go

Issue 1253263002: Make rawdatastore API safer for writing filters. (Closed) Base URL: https://github.com/luci/gae.git@master
Patch Set: avoid dummy callbacks Created 5 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: impl/prod/raw_datastore.go
diff --git a/impl/prod/raw_datastore.go b/impl/prod/raw_datastore.go
index b3a71a46fd19f7db5723e3eabeb924d5f8da0eda..6f90734a03bdaa014b7ace784ee65e173a6a1234 100644
--- a/impl/prod/raw_datastore.go
+++ b/impl/prod/raw_datastore.go
@@ -6,8 +6,8 @@ package prod
import (
rds "github.com/luci/gae/service/rawdatastore"
- "github.com/luci/luci-go/common/errors"
"golang.org/x/net/context"
+ "google.golang.org/appengine"
"google.golang.org/appengine/datastore"
)
@@ -15,7 +15,9 @@ import (
// by gae.GetDS(c)
func useRDS(c context.Context) context.Context {
return rds.SetFactory(c, func(ci context.Context) rds.Interface {
- return rdsImpl{ci}
+ // TODO(riannucci): Track namespace in a better way
+ k := datastore.NewKey(ci, "kind", "", 1, nil) // get current namespace.
+ return rdsImpl{ci, k.Namespace()}
})
}
@@ -57,88 +59,99 @@ func (q queryImpl) Filter(filterStr string, value interface{}) rds.Query {
return queryImpl{q.Query.Filter(filterStr, value)}
}
-////////// Iterator
-
-type iteratorImpl struct{ *datastore.Iterator }
-
-var _ rds.Iterator = iteratorImpl{}
+////////// Datastore
-func (i iteratorImpl) Cursor() (rds.Cursor, error) {
- return i.Iterator.Cursor()
-}
+type rdsImpl struct {
+ context.Context
-func (i iteratorImpl) Next(pls rds.PropertyLoadSaver) (rds.Key, error) {
- return dsR2FErr(i.Iterator.Next(&typeFilter{pls}))
+ ns string
}
-////////// Datastore
-
-type rdsImpl struct{ context.Context }
-
-// NewKeyer
func (d rdsImpl) NewKey(kind, stringID string, intID int64, parent rds.Key) rds.Key {
return dsR2F(datastore.NewKey(d, kind, stringID, intID, dsF2R(parent)))
}
func (rdsImpl) DecodeKey(encoded string) (rds.Key, error) {
- return dsR2FErr(datastore.DecodeKey(encoded))
+ k, err := datastore.DecodeKey(encoded)
+ return dsR2F(k), err
}
-func multiWrap(os []rds.PropertyLoadSaver) []datastore.PropertyLoadSaver {
- ret := make([]datastore.PropertyLoadSaver, len(os))
- for i, pls := range os {
- ret[i] = &typeFilter{pls}
+func idxCallbacker(err error, amt int, cb func(idx int, err error)) error {
+ if err == nil {
+ for i := 0; i < amt; i++ {
+ cb(i, nil)
+ }
+ } else {
dnj (Google) 2015/07/29 01:44:59 Remove "else" clause, return nil at the end of the
iannucci 2015/07/29 07:35:48 Done.
+ if me, ok := err.(appengine.MultiError); ok {
+ for i, err := range me {
+ cb(i, err)
+ }
+ } else {
+ return err
+ }
}
- return ret
+ return nil
}
-func (d rdsImpl) Delete(k rds.Key) error { return datastore.Delete(d, dsF2R(k)) }
-func (d rdsImpl) Get(key rds.Key, dst rds.PropertyLoadSaver) error {
- return datastore.Get(d, dsF2R(key), &typeFilter{dst})
-}
-func (d rdsImpl) Put(key rds.Key, src rds.PropertyLoadSaver) (rds.Key, error) {
- return dsR2FErr(datastore.Put(d, dsF2R(key), &typeFilter{src}))
+func (d rdsImpl) DeleteMulti(ks []rds.Key, cb rds.DeleteMultiCB) error {
+ err := datastore.DeleteMulti(d, dsMF2R(ks))
+ return idxCallbacker(err, len(ks), func(_ int, err error) {
+ cb(err)
+ })
}
-func (d rdsImpl) DeleteMulti(ks []rds.Key) error {
- return errors.Fix(datastore.DeleteMulti(d, dsMF2R(ks)))
+func (d rdsImpl) GetMulti(keys []rds.Key, cb rds.GetMultiCB) error {
+ rkeys := dsMF2R(keys)
+ vals := make([]datastore.PropertyLoadSaver, len(keys))
+ for i := range keys {
+ vals[i] = &typeFilter{rds.PropertyMap{}}
+ }
+ err := datastore.GetMulti(d, rkeys, vals)
+ return idxCallbacker(err, len(keys), func(idx int, err error) {
+ cb(vals[idx].(*typeFilter).pm, err)
+ })
}
-func (d rdsImpl) GetMulti(ks []rds.Key, plss []rds.PropertyLoadSaver) error {
- return errors.Fix(datastore.GetMulti(d, dsMF2R(ks), multiWrap(plss)))
-}
-func (d rdsImpl) PutMulti(key []rds.Key, plss []rds.PropertyLoadSaver) ([]rds.Key, error) {
- ks, err := datastore.PutMulti(d, dsMF2R(key), multiWrap(plss))
- return dsMR2F(ks), errors.Fix(err)
+func (d rdsImpl) PutMulti(keys []rds.Key, vals []rds.PropertyLoadSaver, cb rds.PutMultiCB) error {
+ rkeys := dsMF2R(keys)
+ rvals := make([]datastore.PropertyLoadSaver, len(vals))
+ for i, val := range vals {
+ rvals[i] = &typeFilter{val.(rds.PropertyMap)}
+ }
+ rkeys, err := datastore.PutMulti(d, rkeys, vals)
+ return idxCallbacker(err, len(keys), func(idx int, err error) {
+ k := rds.Key(nil)
+ if err == nil {
+ k = dsR2F(rkeys[idx])
+ }
+ cb(k, err)
+ })
}
-// DSQueryer
func (d rdsImpl) NewQuery(kind string) rds.Query {
return queryImpl{datastore.NewQuery(kind)}
}
-func (d rdsImpl) Run(q rds.Query) rds.Iterator {
- return iteratorImpl{q.(queryImpl).Query.Run(d)}
-}
-func (d rdsImpl) Count(q rds.Query) (int, error) {
- return q.(queryImpl).Query.Count(d)
-}
-func (d rdsImpl) GetAll(q rds.Query, dst *[]rds.PropertyMap) ([]rds.Key, error) {
- fakeDst := []datastore.PropertyList(nil)
- ks, err := q.(queryImpl).GetAll(d, &fakeDst)
- if err != nil {
- return nil, err
+
+func (d rdsImpl) Run(q rds.Query, cb rds.RunCB) error {
+ keepGoing := true
+ tf := typeFilter{}
+ t := q.(queryImpl).Query.Run(d)
+ cfunc := func() (rds.Cursor, error) {
+ return t.Cursor()
}
- *dst = make([]rds.PropertyMap, len(fakeDst))
- for i, pl := range fakeDst {
- (*dst)[i] = rds.PropertyMap{}
- if err := (&typeFilter{(*dst)[i]}).Load(pl); err != nil {
- return nil, err
+ for keepGoing {
+ k, err := t.Next(&tf)
+ if err == datastore.Done {
+ return nil
+ }
+ if err != nil {
+ return err
}
+ keepGoing = cb(dsR2F(k), tf.pm, cfunc)
dnj (Google) 2015/07/29 01:44:59 Consider just: for { ... if !cb(...) { re
iannucci 2015/07/29 07:35:48 Done.
}
- return dsMR2F(ks), err
+ return nil
}
-// Transactioner
func (d rdsImpl) RunInTransaction(f func(c context.Context) error, opts *rds.TransactionOptions) error {
ropts := (*datastore.TransactionOptions)(opts)
return datastore.RunInTransaction(d, f, ropts)

Powered by Google App Engine
This is Rietveld 408576698