| OLD | NEW |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 package memory | 5 package memory |
| 6 | 6 |
| 7 import ( | 7 import ( |
| 8 "fmt" | 8 "fmt" |
| 9 "testing" | 9 "testing" |
| 10 "time" | 10 "time" |
| 11 | 11 |
| 12 dsS "github.com/luci/gae/service/datastore" | 12 dsS "github.com/luci/gae/service/datastore" |
| 13 "github.com/luci/gae/service/datastore/dskey" | |
| 14 "github.com/luci/gae/service/datastore/serialize" | 13 "github.com/luci/gae/service/datastore/serialize" |
| 15 infoS "github.com/luci/gae/service/info" | 14 infoS "github.com/luci/gae/service/info" |
| 16 . "github.com/smartystreets/goconvey/convey" | 15 . "github.com/smartystreets/goconvey/convey" |
| 17 "golang.org/x/net/context" | 16 "golang.org/x/net/context" |
| 18 ) | 17 ) |
| 19 | 18 |
| 20 func TestDatastoreKinder(t *testing.T) { | |
| 21 t.Parallel() | |
| 22 | |
| 23 Convey("Datastore keys", t, func() { | |
| 24 c := Use(context.Background()) | |
| 25 ds := dsS.Get(c) | |
| 26 So(ds, ShouldNotBeNil) | |
| 27 | |
| 28 Convey("implements DSNewKeyer", func() { | |
| 29 Convey("NewKey", func() { | |
| 30 key := ds.NewKey("nerd", "stringID", 0, nil) | |
| 31 So(key, ShouldNotBeNil) | |
| 32 So(key.Kind(), ShouldEqual, "nerd") | |
| 33 So(key.StringID(), ShouldEqual, "stringID") | |
| 34 So(key.IntID(), ShouldEqual, 0) | |
| 35 So(key.Parent(), ShouldBeNil) | |
| 36 So(key.AppID(), ShouldEqual, "dev~app") | |
| 37 So(key.Namespace(), ShouldEqual, "") | |
| 38 So(key.String(), ShouldEqual, "/nerd,stringID") | |
| 39 So(key.Incomplete(), ShouldBeFalse) | |
| 40 So(key.Valid(false, "dev~app", ""), ShouldBeTrue
) | |
| 41 }) | |
| 42 }) | |
| 43 | |
| 44 }) | |
| 45 } | |
| 46 | |
| 47 type MetaGroup struct { | 19 type MetaGroup struct { |
| 48 » _id int64 `gae:"$id,1"` | 20 » _id int64 `gae:"$id,1"` |
| 49 » _kind string `gae:"$kind,__entity_group__"` | 21 » _kind string `gae:"$kind,__entity_group__"` |
| 50 » Parent dsS.Key `gae:"$parent"` | 22 » Parent *dsS.Key `gae:"$parent"` |
| 51 | 23 |
| 52 Version int64 `gae:"__version__"` | 24 Version int64 `gae:"__version__"` |
| 53 } | 25 } |
| 54 | 26 |
| 55 func testGetMeta(c context.Context, k dsS.Key) int64 { | 27 func testGetMeta(c context.Context, k *dsS.Key) int64 { |
| 56 ds := dsS.Get(c) | 28 ds := dsS.Get(c) |
| 57 » mg := &MetaGroup{Parent: dskey.Root(k)} | 29 » mg := &MetaGroup{Parent: k.Root()} |
| 58 if err := ds.Get(mg); err != nil { | 30 if err := ds.Get(mg); err != nil { |
| 59 panic(err) | 31 panic(err) |
| 60 } | 32 } |
| 61 return mg.Version | 33 return mg.Version |
| 62 } | 34 } |
| 63 | 35 |
| 64 var pls = dsS.GetPLS | 36 var pls = dsS.GetPLS |
| 65 | 37 |
| 66 type Foo struct { | 38 type Foo struct { |
| 67 » Id int64 `gae:"$id"` | 39 » ID int64 `gae:"$id"` |
| 68 » Parent dsS.Key `gae:"$parent"` | 40 » Parent *dsS.Key `gae:"$parent"` |
| 69 | 41 |
| 70 Val int | 42 Val int |
| 71 } | 43 } |
| 72 | 44 |
| 73 func TestDatastoreSingleReadWriter(t *testing.T) { | 45 func TestDatastoreSingleReadWriter(t *testing.T) { |
| 74 t.Parallel() | 46 t.Parallel() |
| 75 | 47 |
| 76 Convey("Datastore single reads and writes", t, func() { | 48 Convey("Datastore single reads and writes", t, func() { |
| 77 c := Use(context.Background()) | 49 c := Use(context.Background()) |
| 78 ds := dsS.Get(c) | 50 ds := dsS.Get(c) |
| 79 So(ds, ShouldNotBeNil) | 51 So(ds, ShouldNotBeNil) |
| 80 | 52 |
| 81 Convey("getting objects that DNE is an error", func() { | 53 Convey("getting objects that DNE is an error", func() { |
| 82 » » » So(ds.Get(&Foo{Id: 1}), ShouldEqual, dsS.ErrNoSuchEntity
) | 54 » » » So(ds.Get(&Foo{ID: 1}), ShouldEqual, dsS.ErrNoSuchEntity
) |
| 83 }) | 55 }) |
| 84 | 56 |
| 85 Convey("bad namespaces fail", func() { | 57 Convey("bad namespaces fail", func() { |
| 86 _, err := infoS.Get(c).Namespace("$$blzyall") | 58 _, err := infoS.Get(c).Namespace("$$blzyall") |
| 87 So(err.Error(), ShouldContainSubstring, "namespace \"$$b
lzyall\" does not match") | 59 So(err.Error(), ShouldContainSubstring, "namespace \"$$b
lzyall\" does not match") |
| 88 }) | 60 }) |
| 89 | 61 |
| 90 Convey("Can Put stuff", func() { | 62 Convey("Can Put stuff", func() { |
| 91 // with an incomplete key! | 63 // with an incomplete key! |
| 92 f := &Foo{Val: 10} | 64 f := &Foo{Val: 10} |
| 93 So(ds.Put(f), ShouldBeNil) | 65 So(ds.Put(f), ShouldBeNil) |
| 94 k := ds.KeyForObj(f) | 66 k := ds.KeyForObj(f) |
| 95 » » » So(k.String(), ShouldEqual, "/Foo,1") | 67 » » » So(k.String(), ShouldEqual, "dev~app::/Foo,1") |
| 96 | 68 |
| 97 Convey("and Get it back", func() { | 69 Convey("and Get it back", func() { |
| 98 » » » » newFoo := &Foo{Id: 1} | 70 » » » » newFoo := &Foo{ID: 1} |
| 99 So(ds.Get(newFoo), ShouldBeNil) | 71 So(ds.Get(newFoo), ShouldBeNil) |
| 100 So(newFoo, ShouldResemble, f) | 72 So(newFoo, ShouldResemble, f) |
| 101 | 73 |
| 102 Convey("but it's hidden from a different namespa
ce", func() { | 74 Convey("but it's hidden from a different namespa
ce", func() { |
| 103 c, err := infoS.Get(c).Namespace("whomba
t") | 75 c, err := infoS.Get(c).Namespace("whomba
t") |
| 104 So(err, ShouldBeNil) | 76 So(err, ShouldBeNil) |
| 105 ds = dsS.Get(c) | 77 ds = dsS.Get(c) |
| 106 So(ds.Get(f), ShouldEqual, dsS.ErrNoSuch
Entity) | 78 So(ds.Get(f), ShouldEqual, dsS.ErrNoSuch
Entity) |
| 107 }) | 79 }) |
| 108 | 80 |
| (...skipping 14 matching lines...) Expand all Loading... |
| 123 So(testGetMeta(c, k), ShouldEqual, 1) | 95 So(testGetMeta(c, k), ShouldEqual, 1) |
| 124 | 96 |
| 125 foos := make([]Foo, 10) | 97 foos := make([]Foo, 10) |
| 126 for i := range foos { | 98 for i := range foos { |
| 127 foos[i].Val = 10 | 99 foos[i].Val = 10 |
| 128 foos[i].Parent = k | 100 foos[i].Parent = k |
| 129 } | 101 } |
| 130 So(ds.PutMulti(foos), ShouldBeNil) | 102 So(ds.PutMulti(foos), ShouldBeNil) |
| 131 So(testGetMeta(c, k), ShouldEqual, 11) | 103 So(testGetMeta(c, k), ShouldEqual, 11) |
| 132 | 104 |
| 133 » » » » keys := make([]dsS.Key, len(foos)) | 105 » » » » keys := make([]*dsS.Key, len(foos)) |
| 134 for i, f := range foos { | 106 for i, f := range foos { |
| 135 keys[i] = ds.KeyForObj(&f) | 107 keys[i] = ds.KeyForObj(&f) |
| 136 } | 108 } |
| 137 | 109 |
| 138 Convey("ensure that group versions persist acros
s deletes", func() { | 110 Convey("ensure that group versions persist acros
s deletes", func() { |
| 139 So(ds.DeleteMulti(append(keys, k)), Shou
ldBeNil) | 111 So(ds.DeleteMulti(append(keys, k)), Shou
ldBeNil) |
| 140 | 112 |
| 141 // TODO(riannucci): replace with a Count
query instead of this cast | 113 // TODO(riannucci): replace with a Count
query instead of this cast |
| 142 /* | 114 /* |
| 143 ents := ds.(*dsImpl).data.head.G
etCollection("ents:") | 115 ents := ds.(*dsImpl).data.head.G
etCollection("ents:") |
| 144 num, _ := ents.GetTotals() | 116 num, _ := ents.GetTotals() |
| 145 // /__entity_root_ids__,Foo | 117 // /__entity_root_ids__,Foo |
| 146 // /Foo,1/__entity_group__,1 | 118 // /Foo,1/__entity_group__,1 |
| 147 // /Foo,1/__entity_group_ids__,1 | 119 // /Foo,1/__entity_group_ids__,1 |
| 148 So(num, ShouldEqual, 3) | 120 So(num, ShouldEqual, 3) |
| 149 */ | 121 */ |
| 150 | 122 |
| 151 So(testGetMeta(c, k), ShouldEqual, 22) | 123 So(testGetMeta(c, k), ShouldEqual, 22) |
| 152 | 124 |
| 153 » » » » » So(ds.Put(&Foo{Id: 1}), ShouldBeNil) | 125 » » » » » So(ds.Put(&Foo{ID: 1}), ShouldBeNil) |
| 154 So(testGetMeta(c, k), ShouldEqual, 23) | 126 So(testGetMeta(c, k), ShouldEqual, 23) |
| 155 }) | 127 }) |
| 156 | 128 |
| 157 Convey("can Get", func() { | 129 Convey("can Get", func() { |
| 158 vals := make([]dsS.PropertyMap, len(keys
)) | 130 vals := make([]dsS.PropertyMap, len(keys
)) |
| 159 for i := range vals { | 131 for i := range vals { |
| 160 vals[i] = dsS.PropertyMap{} | 132 vals[i] = dsS.PropertyMap{} |
| 161 » » » » » » vals[i].SetMeta("key", keys[i]) | 133 » » » » » » So(vals[i].SetMeta("key", keys[i
]), ShouldBeNil) |
| 162 } | 134 } |
| 163 So(ds.GetMulti(vals), ShouldBeNil) | 135 So(ds.GetMulti(vals), ShouldBeNil) |
| 164 | 136 |
| 165 for i, val := range vals { | 137 for i, val := range vals { |
| 166 So(val, ShouldResemble, dsS.Prop
ertyMap{ | 138 So(val, ShouldResemble, dsS.Prop
ertyMap{ |
| 167 "Val": {dsS.MkProperty(
10)}, | 139 "Val": {dsS.MkProperty(
10)}, |
| 168 "$key": {dsS.MkPropertyN
I(keys[i])}, | 140 "$key": {dsS.MkPropertyN
I(keys[i])}, |
| 169 }) | 141 }) |
| 170 } | 142 } |
| 171 }) | 143 }) |
| 172 | 144 |
| 173 }) | 145 }) |
| 174 }) | 146 }) |
| 175 | 147 |
| 176 Convey("implements DSTransactioner", func() { | 148 Convey("implements DSTransactioner", func() { |
| 177 Convey("Put", func() { | 149 Convey("Put", func() { |
| 178 f := &Foo{Val: 10} | 150 f := &Foo{Val: 10} |
| 179 So(ds.Put(f), ShouldBeNil) | 151 So(ds.Put(f), ShouldBeNil) |
| 180 k := ds.KeyForObj(f) | 152 k := ds.KeyForObj(f) |
| 181 » » » » So(k.String(), ShouldEqual, "/Foo,1") | 153 » » » » So(k.String(), ShouldEqual, "dev~app::/Foo,1") |
| 182 | 154 |
| 183 Convey("can Put new entity groups", func() { | 155 Convey("can Put new entity groups", func() { |
| 184 err := ds.RunInTransaction(func(c contex
t.Context) error { | 156 err := ds.RunInTransaction(func(c contex
t.Context) error { |
| 185 ds := dsS.Get(c) | 157 ds := dsS.Get(c) |
| 186 | 158 |
| 187 f := &Foo{Val: 100} | 159 f := &Foo{Val: 100} |
| 188 So(ds.Put(f), ShouldBeNil) | 160 So(ds.Put(f), ShouldBeNil) |
| 189 » » » » » » So(f.Id, ShouldEqual, 2) | 161 » » » » » » So(f.ID, ShouldEqual, 2) |
| 190 | 162 |
| 191 » » » » » » f.Id = 0 | 163 » » » » » » f.ID = 0 |
| 192 f.Val = 200 | 164 f.Val = 200 |
| 193 So(ds.Put(f), ShouldBeNil) | 165 So(ds.Put(f), ShouldBeNil) |
| 194 » » » » » » So(f.Id, ShouldEqual, 3) | 166 » » » » » » So(f.ID, ShouldEqual, 3) |
| 195 | 167 |
| 196 return nil | 168 return nil |
| 197 }, &dsS.TransactionOptions{XG: true}) | 169 }, &dsS.TransactionOptions{XG: true}) |
| 198 So(err, ShouldBeNil) | 170 So(err, ShouldBeNil) |
| 199 | 171 |
| 200 » » » » » f := &Foo{Id: 2} | 172 » » » » » f := &Foo{ID: 2} |
| 201 So(ds.Get(f), ShouldBeNil) | 173 So(ds.Get(f), ShouldBeNil) |
| 202 So(f.Val, ShouldEqual, 100) | 174 So(f.Val, ShouldEqual, 100) |
| 203 | 175 |
| 204 » » » » » f.Id = 3 | 176 » » » » » f.ID = 3 |
| 205 So(ds.Get(f), ShouldBeNil) | 177 So(ds.Get(f), ShouldBeNil) |
| 206 So(f.Val, ShouldEqual, 200) | 178 So(f.Val, ShouldEqual, 200) |
| 207 }) | 179 }) |
| 208 | 180 |
| 209 Convey("can Put new entities in a current group"
, func() { | 181 Convey("can Put new entities in a current group"
, func() { |
| 210 err := ds.RunInTransaction(func(c contex
t.Context) error { | 182 err := ds.RunInTransaction(func(c contex
t.Context) error { |
| 211 ds := dsS.Get(c) | 183 ds := dsS.Get(c) |
| 212 | 184 |
| 213 f := &Foo{Val: 100, Parent: k} | 185 f := &Foo{Val: 100, Parent: k} |
| 214 So(ds.Put(f), ShouldBeNil) | 186 So(ds.Put(f), ShouldBeNil) |
| 215 » » » » » » So(ds.KeyForObj(f).String(), Sho
uldEqual, "/Foo,1/Foo,1") | 187 » » » » » » So(ds.KeyForObj(f).String(), Sho
uldEqual, "dev~app::/Foo,1/Foo,1") |
| 216 | 188 |
| 217 » » » » » » f.Id = 0 | 189 » » » » » » f.ID = 0 |
| 218 f.Val = 200 | 190 f.Val = 200 |
| 219 So(ds.Put(f), ShouldBeNil) | 191 So(ds.Put(f), ShouldBeNil) |
| 220 » » » » » » So(ds.KeyForObj(f).String(), Sho
uldEqual, "/Foo,1/Foo,2") | 192 » » » » » » So(ds.KeyForObj(f).String(), Sho
uldEqual, "dev~app::/Foo,1/Foo,2") |
| 221 | 193 |
| 222 return nil | 194 return nil |
| 223 }, nil) | 195 }, nil) |
| 224 So(err, ShouldBeNil) | 196 So(err, ShouldBeNil) |
| 225 | 197 |
| 226 » » » » » f := &Foo{Id: 1, Parent: k} | 198 » » » » » f := &Foo{ID: 1, Parent: k} |
| 227 So(ds.Get(f), ShouldBeNil) | 199 So(ds.Get(f), ShouldBeNil) |
| 228 So(f.Val, ShouldEqual, 100) | 200 So(f.Val, ShouldEqual, 100) |
| 229 | 201 |
| 230 » » » » » f.Id = 2 | 202 » » » » » f.ID = 2 |
| 231 So(ds.Get(f), ShouldBeNil) | 203 So(ds.Get(f), ShouldBeNil) |
| 232 So(f.Val, ShouldEqual, 200) | 204 So(f.Val, ShouldEqual, 200) |
| 233 }) | 205 }) |
| 234 | 206 |
| 235 Convey("Deletes work too", func() { | 207 Convey("Deletes work too", func() { |
| 236 err := ds.RunInTransaction(func(c contex
t.Context) error { | 208 err := ds.RunInTransaction(func(c contex
t.Context) error { |
| 237 return dsS.Get(c).Delete(k) | 209 return dsS.Get(c).Delete(k) |
| 238 }, nil) | 210 }, nil) |
| 239 So(err, ShouldBeNil) | 211 So(err, ShouldBeNil) |
| 240 » » » » » So(ds.Get(&Foo{Id: 1}), ShouldEqual, dsS
.ErrNoSuchEntity) | 212 » » » » » So(ds.Get(&Foo{ID: 1}), ShouldEqual, dsS
.ErrNoSuchEntity) |
| 241 }) | 213 }) |
| 242 | 214 |
| 243 Convey("A Get counts against your group count",
func() { | 215 Convey("A Get counts against your group count",
func() { |
| 244 err := ds.RunInTransaction(func(c contex
t.Context) error { | 216 err := ds.RunInTransaction(func(c contex
t.Context) error { |
| 245 ds := dsS.Get(c) | 217 ds := dsS.Get(c) |
| 246 | 218 |
| 247 pm := dsS.PropertyMap{} | 219 pm := dsS.PropertyMap{} |
| 248 » » » » » » pm.SetMeta("key", ds.NewKey("Foo
", "", 20, nil)) | 220 » » » » » » So(pm.SetMeta("key", ds.NewKey("
Foo", "", 20, nil)), ShouldBeNil) |
| 249 So(ds.Get(pm), ShouldEqual, dsS.
ErrNoSuchEntity) | 221 So(ds.Get(pm), ShouldEqual, dsS.
ErrNoSuchEntity) |
| 250 | 222 |
| 251 » » » » » » pm.SetMeta("key", k) | 223 » » » » » » So(pm.SetMeta("key", k), ShouldB
eNil) |
| 252 So(ds.Get(pm).Error(), ShouldCon
tainSubstring, "cross-group") | 224 So(ds.Get(pm).Error(), ShouldCon
tainSubstring, "cross-group") |
| 253 return nil | 225 return nil |
| 254 }, nil) | 226 }, nil) |
| 255 So(err, ShouldBeNil) | 227 So(err, ShouldBeNil) |
| 256 }) | 228 }) |
| 257 | 229 |
| 258 Convey("Get takes a snapshot", func() { | 230 Convey("Get takes a snapshot", func() { |
| 259 err := ds.RunInTransaction(func(c contex
t.Context) error { | 231 err := ds.RunInTransaction(func(c contex
t.Context) error { |
| 260 txnDS := dsS.Get(c) | 232 txnDS := dsS.Get(c) |
| 261 | 233 |
| 262 So(txnDS.Get(f), ShouldBeNil) | 234 So(txnDS.Get(f), ShouldBeNil) |
| 263 So(f.Val, ShouldEqual, 10) | 235 So(f.Val, ShouldEqual, 10) |
| 264 | 236 |
| 265 // Don't ever do this in a real
program unless you want to guarantee | 237 // Don't ever do this in a real
program unless you want to guarantee |
| 266 // a failed transaction :) | 238 // a failed transaction :) |
| 267 f.Val = 11 | 239 f.Val = 11 |
| 268 So(ds.Put(f), ShouldBeNil) | 240 So(ds.Put(f), ShouldBeNil) |
| 269 | 241 |
| 270 So(txnDS.Get(f), ShouldBeNil) | 242 So(txnDS.Get(f), ShouldBeNil) |
| 271 So(f.Val, ShouldEqual, 10) | 243 So(f.Val, ShouldEqual, 10) |
| 272 | 244 |
| 273 return nil | 245 return nil |
| 274 }, nil) | 246 }, nil) |
| 275 So(err, ShouldBeNil) | 247 So(err, ShouldBeNil) |
| 276 | 248 |
| 277 » » » » » f := &Foo{Id: 1} | 249 » » » » » f := &Foo{ID: 1} |
| 278 So(ds.Get(f), ShouldBeNil) | 250 So(ds.Get(f), ShouldBeNil) |
| 279 So(f.Val, ShouldEqual, 11) | 251 So(f.Val, ShouldEqual, 11) |
| 280 }) | 252 }) |
| 281 | 253 |
| 282 Convey("and snapshots are consistent even after
Puts", func() { | 254 Convey("and snapshots are consistent even after
Puts", func() { |
| 283 err := ds.RunInTransaction(func(c contex
t.Context) error { | 255 err := ds.RunInTransaction(func(c contex
t.Context) error { |
| 284 txnDS := dsS.Get(c) | 256 txnDS := dsS.Get(c) |
| 285 | 257 |
| 286 » » » » » » f := &Foo{Id: 1} | 258 » » » » » » f := &Foo{ID: 1} |
| 287 So(txnDS.Get(f), ShouldBeNil) | 259 So(txnDS.Get(f), ShouldBeNil) |
| 288 So(f.Val, ShouldEqual, 10) | 260 So(f.Val, ShouldEqual, 10) |
| 289 | 261 |
| 290 // Don't ever do this in a real
program unless you want to guarantee | 262 // Don't ever do this in a real
program unless you want to guarantee |
| 291 // a failed transaction :) | 263 // a failed transaction :) |
| 292 f.Val = 11 | 264 f.Val = 11 |
| 293 So(ds.Put(f), ShouldBeNil) | 265 So(ds.Put(f), ShouldBeNil) |
| 294 | 266 |
| 295 So(txnDS.Get(f), ShouldBeNil) | 267 So(txnDS.Get(f), ShouldBeNil) |
| 296 So(f.Val, ShouldEqual, 10) | 268 So(f.Val, ShouldEqual, 10) |
| 297 | 269 |
| 298 f.Val = 20 | 270 f.Val = 20 |
| 299 So(txnDS.Put(f), ShouldBeNil) | 271 So(txnDS.Put(f), ShouldBeNil) |
| 300 | 272 |
| 301 So(txnDS.Get(f), ShouldBeNil) | 273 So(txnDS.Get(f), ShouldBeNil) |
| 302 So(f.Val, ShouldEqual, 10) // st
ill gets 10 | 274 So(f.Val, ShouldEqual, 10) // st
ill gets 10 |
| 303 | 275 |
| 304 return nil | 276 return nil |
| 305 }, &dsS.TransactionOptions{Attempts: 1}) | 277 }, &dsS.TransactionOptions{Attempts: 1}) |
| 306 So(err.Error(), ShouldContainSubstring,
"concurrent") | 278 So(err.Error(), ShouldContainSubstring,
"concurrent") |
| 307 | 279 |
| 308 » » » » » f := &Foo{Id: 1} | 280 » » » » » f := &Foo{ID: 1} |
| 309 So(ds.Get(f), ShouldBeNil) | 281 So(ds.Get(f), ShouldBeNil) |
| 310 So(f.Val, ShouldEqual, 11) | 282 So(f.Val, ShouldEqual, 11) |
| 311 }) | 283 }) |
| 312 | 284 |
| 313 Convey("Reusing a transaction context is bad new
s", func() { | 285 Convey("Reusing a transaction context is bad new
s", func() { |
| 314 txnDS := dsS.Interface(nil) | 286 txnDS := dsS.Interface(nil) |
| 315 err := ds.RunInTransaction(func(c contex
t.Context) error { | 287 err := ds.RunInTransaction(func(c contex
t.Context) error { |
| 316 txnDS = dsS.Get(c) | 288 txnDS = dsS.Get(c) |
| 317 So(txnDS.Get(f), ShouldBeNil) | 289 So(txnDS.Get(f), ShouldBeNil) |
| 318 return nil | 290 return nil |
| (...skipping 15 matching lines...) Expand all Loading... |
| 334 | 306 |
| 335 Convey("Concurrent transactions only accept one
set of changes", func() { | 307 Convey("Concurrent transactions only accept one
set of changes", func() { |
| 336 // Note: I think this implementation is
actually /slightly/ wrong. | 308 // Note: I think this implementation is
actually /slightly/ wrong. |
| 337 // According to my read of the docs for
appengine, when you open a | 309 // According to my read of the docs for
appengine, when you open a |
| 338 // transaction it actually (essentially)
holds a reference to the | 310 // transaction it actually (essentially)
holds a reference to the |
| 339 // entire datastore. Our implementation
takes a snapshot of the | 311 // entire datastore. Our implementation
takes a snapshot of the |
| 340 // entity group as soon as something obs
erves/affects it. | 312 // entity group as soon as something obs
erves/affects it. |
| 341 // | 313 // |
| 342 // That said... I'm not sure if there's
really a semantic difference. | 314 // That said... I'm not sure if there's
really a semantic difference. |
| 343 err := ds.RunInTransaction(func(c contex
t.Context) error { | 315 err := ds.RunInTransaction(func(c contex
t.Context) error { |
| 344 » » » » » » So(dsS.Get(c).Put(&Foo{Id: 1, Va
l: 21}), ShouldBeNil) | 316 » » » » » » So(dsS.Get(c).Put(&Foo{ID: 1, Va
l: 21}), ShouldBeNil) |
| 345 | 317 |
| 346 err := ds.RunInTransaction(func(
c context.Context) error { | 318 err := ds.RunInTransaction(func(
c context.Context) error { |
| 347 » » » » » » » So(dsS.Get(c).Put(&Foo{I
d: 1, Val: 27}), ShouldBeNil) | 319 » » » » » » » So(dsS.Get(c).Put(&Foo{I
D: 1, Val: 27}), ShouldBeNil) |
| 348 return nil | 320 return nil |
| 349 }, nil) | 321 }, nil) |
| 350 So(err, ShouldBeNil) | 322 So(err, ShouldBeNil) |
| 351 | 323 |
| 352 return nil | 324 return nil |
| 353 }, nil) | 325 }, nil) |
| 354 So(err.Error(), ShouldContainSubstring,
"concurrent") | 326 So(err.Error(), ShouldContainSubstring,
"concurrent") |
| 355 | 327 |
| 356 » » » » » f := &Foo{Id: 1} | 328 » » » » » f := &Foo{ID: 1} |
| 357 So(ds.Get(f), ShouldBeNil) | 329 So(ds.Get(f), ShouldBeNil) |
| 358 So(f.Val, ShouldEqual, 27) | 330 So(f.Val, ShouldEqual, 27) |
| 359 }) | 331 }) |
| 360 | 332 |
| 361 Convey("XG", func() { | 333 Convey("XG", func() { |
| 362 Convey("Modifying two groups with XG=fal
se is invalid", func() { | 334 Convey("Modifying two groups with XG=fal
se is invalid", func() { |
| 363 err := ds.RunInTransaction(func(
c context.Context) error { | 335 err := ds.RunInTransaction(func(
c context.Context) error { |
| 364 ds := dsS.Get(c) | 336 ds := dsS.Get(c) |
| 365 » » » » » » » f := &Foo{Id: 1, Val: 20
0} | 337 » » » » » » » f := &Foo{ID: 1, Val: 20
0} |
| 366 So(ds.Put(f), ShouldBeNi
l) | 338 So(ds.Put(f), ShouldBeNi
l) |
| 367 | 339 |
| 368 » » » » » » » f.Id = 2 | 340 » » » » » » » f.ID = 2 |
| 369 err := ds.Put(f) | 341 err := ds.Put(f) |
| 370 So(err.Error(), ShouldCo
ntainSubstring, "cross-group") | 342 So(err.Error(), ShouldCo
ntainSubstring, "cross-group") |
| 371 return err | 343 return err |
| 372 }, nil) | 344 }, nil) |
| 373 So(err.Error(), ShouldContainSub
string, "cross-group") | 345 So(err.Error(), ShouldContainSub
string, "cross-group") |
| 374 }) | 346 }) |
| 375 | 347 |
| 376 Convey("Modifying >25 groups with XG=tru
e is invald", func() { | 348 Convey("Modifying >25 groups with XG=tru
e is invald", func() { |
| 377 err := ds.RunInTransaction(func(
c context.Context) error { | 349 err := ds.RunInTransaction(func(
c context.Context) error { |
| 378 ds := dsS.Get(c) | 350 ds := dsS.Get(c) |
| 379 foos := make([]Foo, 25) | 351 foos := make([]Foo, 25) |
| 380 for i := int64(1); i < 2
6; i++ { | 352 for i := int64(1); i < 2
6; i++ { |
| 381 » » » » » » » » foos[i-1].Id = i | 353 » » » » » » » » foos[i-1].ID = i |
| 382 foos[i-1].Val =
200 | 354 foos[i-1].Val =
200 |
| 383 } | 355 } |
| 384 So(ds.PutMulti(foos), Sh
ouldBeNil) | 356 So(ds.PutMulti(foos), Sh
ouldBeNil) |
| 385 » » » » » » » err := ds.Put(&Foo{Id: 2
6}) | 357 » » » » » » » err := ds.Put(&Foo{ID: 2
6}) |
| 386 So(err.Error(), ShouldCo
ntainSubstring, "too many entity groups") | 358 So(err.Error(), ShouldCo
ntainSubstring, "too many entity groups") |
| 387 return err | 359 return err |
| 388 }, &dsS.TransactionOptions{XG: t
rue}) | 360 }, &dsS.TransactionOptions{XG: t
rue}) |
| 389 So(err.Error(), ShouldContainSub
string, "too many entity groups") | 361 So(err.Error(), ShouldContainSub
string, "too many entity groups") |
| 390 }) | 362 }) |
| 391 }) | 363 }) |
| 392 | 364 |
| 393 Convey("Errors and panics", func() { | 365 Convey("Errors and panics", func() { |
| 394 Convey("returning an error aborts", func
() { | 366 Convey("returning an error aborts", func
() { |
| 395 err := ds.RunInTransaction(func(
c context.Context) error { | 367 err := ds.RunInTransaction(func(
c context.Context) error { |
| 396 ds := dsS.Get(c) | 368 ds := dsS.Get(c) |
| 397 » » » » » » » So(ds.Put(&Foo{Id: 1, Va
l: 200}), ShouldBeNil) | 369 » » » » » » » So(ds.Put(&Foo{ID: 1, Va
l: 200}), ShouldBeNil) |
| 398 return fmt.Errorf("thing
y") | 370 return fmt.Errorf("thing
y") |
| 399 }, nil) | 371 }, nil) |
| 400 So(err.Error(), ShouldEqual, "th
ingy") | 372 So(err.Error(), ShouldEqual, "th
ingy") |
| 401 | 373 |
| 402 » » » » » » f := &Foo{Id: 1} | 374 » » » » » » f := &Foo{ID: 1} |
| 403 So(ds.Get(f), ShouldBeNil) | 375 So(ds.Get(f), ShouldBeNil) |
| 404 So(f.Val, ShouldEqual, 10) | 376 So(f.Val, ShouldEqual, 10) |
| 405 }) | 377 }) |
| 406 | 378 |
| 407 Convey("panicing aborts", func() { | 379 Convey("panicing aborts", func() { |
| 408 So(func() { | 380 So(func() { |
| 409 » » » » » » » ds.RunInTransaction(func
(c context.Context) error { | 381 » » » » » » » So(ds.RunInTransaction(f
unc(c context.Context) error { |
| 410 ds := dsS.Get(c) | 382 ds := dsS.Get(c) |
| 411 So(ds.Put(&Foo{V
al: 200}), ShouldBeNil) | 383 So(ds.Put(&Foo{V
al: 200}), ShouldBeNil) |
| 412 panic("wheeeeee"
) | 384 panic("wheeeeee"
) |
| 413 » » » » » » » }, nil) | 385 » » » » » » » }, nil), ShouldBeNil) |
| 414 }, ShouldPanic) | 386 }, ShouldPanic) |
| 415 | 387 |
| 416 » » » » » » f := &Foo{Id: 1} | 388 » » » » » » f := &Foo{ID: 1} |
| 417 So(ds.Get(f), ShouldBeNil) | 389 So(ds.Get(f), ShouldBeNil) |
| 418 So(f.Val, ShouldEqual, 10) | 390 So(f.Val, ShouldEqual, 10) |
| 419 }) | 391 }) |
| 420 }) | 392 }) |
| 421 | 393 |
| 422 Convey("Transaction retries", func() { | 394 Convey("Transaction retries", func() { |
| 423 tst := ds.Testable() | 395 tst := ds.Testable() |
| 424 Reset(func() { tst.SetTransactionRetryCo
unt(0) }) | 396 Reset(func() { tst.SetTransactionRetryCo
unt(0) }) |
| 425 | 397 |
| 426 Convey("SetTransactionRetryCount set to
zere", func() { | 398 Convey("SetTransactionRetryCount set to
zere", func() { |
| (...skipping 118 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 545 ds := dsS.Get(Use(context.Background())) | 517 ds := dsS.Get(Use(context.Background())) |
| 546 m := Model{ID: 1} | 518 m := Model{ID: 1} |
| 547 So(ds.Put(&m), ShouldBeNil) | 519 So(ds.Put(&m), ShouldBeNil) |
| 548 | 520 |
| 549 // Reset to something non zero to ensure zero is fetched. | 521 // Reset to something non zero to ensure zero is fetched. |
| 550 m.Time = time.Now().UTC() | 522 m.Time = time.Now().UTC() |
| 551 So(ds.Get(&m), ShouldBeNil) | 523 So(ds.Get(&m), ShouldBeNil) |
| 552 So(m.Time.IsZero(), ShouldBeTrue) | 524 So(m.Time.IsZero(), ShouldBeTrue) |
| 553 }) | 525 }) |
| 554 } | 526 } |
| OLD | NEW |