123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755 |
- // Copyright 2014 Google Inc. All Rights Reserved.
- //
- // 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 datastore
- import (
- "errors"
- "fmt"
- "reflect"
- "sort"
- "strings"
- "testing"
- "time"
- "golang.org/x/net/context"
- "google.golang.org/cloud"
- "google.golang.org/cloud/internal/testutil"
- )
- // TODO(djd): Make test entity clean up more robust: some test entities may
- // be left behind if tests are aborted, the transport fails, etc.
- func newClient(ctx context.Context, t *testing.T) *Client {
- ts := testutil.TokenSource(ctx, ScopeDatastore, ScopeUserEmail)
- if ts == nil {
- t.Skip("Integration tests skipped. See CONTRIBUTING.md for details")
- }
- client, err := NewClient(ctx, testutil.ProjID(), cloud.WithTokenSource(ts))
- if err != nil {
- t.Fatalf("NewClient: %v", err)
- }
- return client
- }
- func TestBasics(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- type X struct {
- I int
- S string
- T time.Time
- }
- x0 := X{66, "99", time.Now().Truncate(time.Millisecond)}
- k, err := client.Put(ctx, NewIncompleteKey(ctx, "BasicsX", nil), &x0)
- if err != nil {
- t.Fatalf("client.Put: %v", err)
- }
- x1 := X{}
- err = client.Get(ctx, k, &x1)
- if err != nil {
- t.Errorf("client.Get: %v", err)
- }
- err = client.Delete(ctx, k)
- if err != nil {
- t.Errorf("client.Delete: %v", err)
- }
- if !reflect.DeepEqual(x0, x1) {
- t.Errorf("compare: x0=%v, x1=%v", x0, x1)
- }
- }
- func TestListValues(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- p0 := PropertyList{
- {Name: "L", Value: int64(12), Multiple: true},
- {Name: "L", Value: "string", Multiple: true},
- {Name: "L", Value: true, Multiple: true},
- }
- k, err := client.Put(ctx, NewIncompleteKey(ctx, "ListValue", nil), &p0)
- if err != nil {
- t.Fatalf("client.Put: %v", err)
- }
- var p1 PropertyList
- if err := client.Get(ctx, k, &p1); err != nil {
- t.Errorf("client.Get: %v", err)
- }
- if !reflect.DeepEqual(p0, p1) {
- t.Errorf("compare:\np0=%v\np1=%#v", p0, p1)
- }
- if err = client.Delete(ctx, k); err != nil {
- t.Errorf("client.Delete: %v", err)
- }
- }
- func TestGetMulti(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- type X struct {
- I int
- }
- p := NewKey(ctx, "X", "", time.Now().Unix(), nil)
- cases := []struct {
- key *Key
- put bool
- }{
- {key: NewKey(ctx, "X", "item1", 0, p), put: true},
- {key: NewKey(ctx, "X", "item2", 0, p), put: false},
- {key: NewKey(ctx, "X", "item3", 0, p), put: false},
- {key: NewKey(ctx, "X", "item4", 0, p), put: true},
- }
- var src, dst []*X
- var srcKeys, dstKeys []*Key
- for _, c := range cases {
- dst = append(dst, &X{})
- dstKeys = append(dstKeys, c.key)
- if c.put {
- src = append(src, &X{})
- srcKeys = append(srcKeys, c.key)
- }
- }
- if _, err := client.PutMulti(ctx, srcKeys, src); err != nil {
- t.Error(err)
- }
- err := client.GetMulti(ctx, dstKeys, dst)
- if err == nil {
- t.Errorf("client.GetMulti got %v, expected error", err)
- }
- e, ok := err.(MultiError)
- if !ok {
- t.Errorf("client.GetMulti got %t, expected MultiError", err)
- }
- for i, err := range e {
- got, want := err, (error)(nil)
- if !cases[i].put {
- got, want = err, ErrNoSuchEntity
- }
- if got != want {
- t.Errorf("MultiError[%d] == %v, want %v", i, got, want)
- }
- }
- }
- type Z struct {
- S string
- T string `datastore:",noindex"`
- P []byte
- K []byte `datastore:",noindex"`
- }
- func (z Z) String() string {
- var lens []string
- v := reflect.ValueOf(z)
- for i := 0; i < v.NumField(); i++ {
- if l := v.Field(i).Len(); l > 0 {
- lens = append(lens, fmt.Sprintf("len(%s)=%d", v.Type().Field(i).Name, l))
- }
- }
- return fmt.Sprintf("Z{ %s }", strings.Join(lens, ","))
- }
- func TestUnindexableValues(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- x1500 := strings.Repeat("x", 1500)
- x1501 := strings.Repeat("x", 1501)
- testCases := []struct {
- in Z
- wantErr bool
- }{
- {in: Z{S: x1500}, wantErr: false},
- {in: Z{S: x1501}, wantErr: true},
- {in: Z{T: x1500}, wantErr: false},
- {in: Z{T: x1501}, wantErr: false},
- {in: Z{P: []byte(x1500)}, wantErr: false},
- {in: Z{P: []byte(x1501)}, wantErr: true},
- {in: Z{K: []byte(x1500)}, wantErr: false},
- {in: Z{K: []byte(x1501)}, wantErr: false},
- }
- for _, tt := range testCases {
- _, err := client.Put(ctx, NewIncompleteKey(ctx, "BasicsZ", nil), &tt.in)
- if (err != nil) != tt.wantErr {
- t.Errorf("client.Put %s got err %v, want err %t", tt.in, err, tt.wantErr)
- }
- }
- }
- type SQChild struct {
- I, J int
- T, U int64
- }
- type SQTestCase struct {
- desc string
- q *Query
- wantCount int
- wantSum int
- }
- func testSmallQueries(t *testing.T, ctx context.Context, client *Client, parent *Key, children []*SQChild,
- testCases []SQTestCase, extraTests ...func()) {
- keys := make([]*Key, len(children))
- for i := range keys {
- keys[i] = NewIncompleteKey(ctx, "SQChild", parent)
- }
- keys, err := client.PutMulti(ctx, keys, children)
- if err != nil {
- t.Fatalf("client.PutMulti: %v", err)
- }
- defer func() {
- err := client.DeleteMulti(ctx, keys)
- if err != nil {
- t.Errorf("client.DeleteMulti: %v", err)
- }
- }()
- for _, tc := range testCases {
- count, err := client.Count(ctx, tc.q)
- if err != nil {
- t.Errorf("Count %q: %v", tc.desc, err)
- continue
- }
- if count != tc.wantCount {
- t.Errorf("Count %q: got %d want %d", tc.desc, count, tc.wantCount)
- continue
- }
- }
- for _, tc := range testCases {
- var got []SQChild
- _, err := client.GetAll(ctx, tc.q, &got)
- if err != nil {
- t.Errorf("client.GetAll %q: %v", tc.desc, err)
- continue
- }
- sum := 0
- for _, c := range got {
- sum += c.I + c.J
- }
- if sum != tc.wantSum {
- t.Errorf("sum %q: got %d want %d", tc.desc, sum, tc.wantSum)
- continue
- }
- }
- for _, x := range extraTests {
- x()
- }
- }
- func TestFilters(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- parent := NewKey(ctx, "SQParent", "TestFilters", 0, nil)
- now := time.Now().Truncate(time.Millisecond).Unix()
- children := []*SQChild{
- {I: 0, T: now, U: now},
- {I: 1, T: now, U: now},
- {I: 2, T: now, U: now},
- {I: 3, T: now, U: now},
- {I: 4, T: now, U: now},
- {I: 5, T: now, U: now},
- {I: 6, T: now, U: now},
- {I: 7, T: now, U: now},
- }
- baseQuery := NewQuery("SQChild").Ancestor(parent).Filter("T=", now)
- testSmallQueries(t, ctx, client, parent, children, []SQTestCase{
- {
- "I>1",
- baseQuery.Filter("I>", 1),
- 6,
- 2 + 3 + 4 + 5 + 6 + 7,
- },
- {
- "I>2 AND I<=5",
- baseQuery.Filter("I>", 2).Filter("I<=", 5),
- 3,
- 3 + 4 + 5,
- },
- {
- "I>=3 AND I<3",
- baseQuery.Filter("I>=", 3).Filter("I<", 3),
- 0,
- 0,
- },
- {
- "I=4",
- baseQuery.Filter("I=", 4),
- 1,
- 4,
- },
- }, func() {
- got := []*SQChild{}
- want := []*SQChild{
- {I: 0, T: now, U: now},
- {I: 1, T: now, U: now},
- {I: 2, T: now, U: now},
- {I: 3, T: now, U: now},
- {I: 4, T: now, U: now},
- {I: 5, T: now, U: now},
- {I: 6, T: now, U: now},
- {I: 7, T: now, U: now},
- }
- _, err := client.GetAll(ctx, baseQuery.Order("I"), &got)
- if err != nil {
- t.Errorf("client.GetAll: %v", err)
- }
- if !reflect.DeepEqual(got, want) {
- t.Errorf("compare: got=%v, want=%v", got, want)
- }
- }, func() {
- got := []*SQChild{}
- want := []*SQChild{
- {I: 7, T: now, U: now},
- {I: 6, T: now, U: now},
- {I: 5, T: now, U: now},
- {I: 4, T: now, U: now},
- {I: 3, T: now, U: now},
- {I: 2, T: now, U: now},
- {I: 1, T: now, U: now},
- {I: 0, T: now, U: now},
- }
- _, err := client.GetAll(ctx, baseQuery.Order("-I"), &got)
- if err != nil {
- t.Errorf("client.GetAll: %v", err)
- }
- if !reflect.DeepEqual(got, want) {
- t.Errorf("compare: got=%v, want=%v", got, want)
- }
- })
- }
- func TestEventualConsistency(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- parent := NewKey(ctx, "SQParent", "TestEventualConsistency", 0, nil)
- now := time.Now().Truncate(time.Millisecond).Unix()
- children := []*SQChild{
- {I: 0, T: now, U: now},
- {I: 1, T: now, U: now},
- {I: 2, T: now, U: now},
- }
- query := NewQuery("SQChild").Ancestor(parent).Filter("T =", now).EventualConsistency()
- testSmallQueries(t, ctx, client, parent, children, nil, func() {
- got, err := client.Count(ctx, query)
- if err != nil {
- t.Fatalf("Count: %v", err)
- }
- if got < 0 || 3 < got {
- t.Errorf("Count: got %d, want [0,3]", got)
- }
- })
- }
- func TestProjection(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- parent := NewKey(ctx, "SQParent", "TestProjection", 0, nil)
- now := time.Now().Truncate(time.Millisecond).Unix()
- children := []*SQChild{
- {I: 1 << 0, J: 100, T: now, U: now},
- {I: 1 << 1, J: 100, T: now, U: now},
- {I: 1 << 2, J: 200, T: now, U: now},
- {I: 1 << 3, J: 300, T: now, U: now},
- {I: 1 << 4, J: 300, T: now, U: now},
- }
- baseQuery := NewQuery("SQChild").Ancestor(parent).Filter("T=", now).Filter("J>", 150)
- testSmallQueries(t, ctx, client, parent, children, []SQTestCase{
- {
- "project",
- baseQuery.Project("J"),
- 3,
- 200 + 300 + 300,
- },
- {
- "distinct",
- baseQuery.Project("J").Distinct(),
- 2,
- 200 + 300,
- },
- {
- "project on meaningful (GD_WHEN) field",
- baseQuery.Project("U"),
- 3,
- 0,
- },
- })
- }
- func TestAllocateIDs(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- keys := make([]*Key, 5)
- for i := range keys {
- keys[i] = NewIncompleteKey(ctx, "AllocID", nil)
- }
- keys, err := client.AllocateIDs(ctx, keys)
- if err != nil {
- t.Errorf("AllocID #0 failed: %v", err)
- }
- if want := len(keys); want != 5 {
- t.Errorf("Expected to allocate 5 keys, %d keys are found", want)
- }
- for _, k := range keys {
- if k.Incomplete() {
- t.Errorf("Unexpeceted incomplete key found: %v", k)
- }
- }
- }
- func TestGetAllWithFieldMismatch(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- type Fat struct {
- X, Y int
- }
- type Thin struct {
- X int
- }
- // Ancestor queries (those within an entity group) are strongly consistent
- // by default, which prevents a test from being flaky.
- // See https://cloud.google.com/appengine/docs/go/datastore/queries#Go_Data_consistency
- // for more information.
- parent := NewKey(ctx, "SQParent", "TestGetAllWithFieldMismatch", 0, nil)
- putKeys := make([]*Key, 3)
- for i := range putKeys {
- putKeys[i] = NewKey(ctx, "GetAllThing", "", int64(10+i), parent)
- _, err := client.Put(ctx, putKeys[i], &Fat{X: 20 + i, Y: 30 + i})
- if err != nil {
- t.Fatalf("client.Put: %v", err)
- }
- }
- var got []Thin
- want := []Thin{
- {X: 20},
- {X: 21},
- {X: 22},
- }
- getKeys, err := client.GetAll(ctx, NewQuery("GetAllThing").Ancestor(parent), &got)
- if len(getKeys) != 3 && !reflect.DeepEqual(getKeys, putKeys) {
- t.Errorf("client.GetAll: keys differ\ngetKeys=%v\nputKeys=%v", getKeys, putKeys)
- }
- if !reflect.DeepEqual(got, want) {
- t.Errorf("client.GetAll: entities differ\ngot =%v\nwant=%v", got, want)
- }
- if _, ok := err.(*ErrFieldMismatch); !ok {
- t.Errorf("client.GetAll: got err=%v, want ErrFieldMismatch", err)
- }
- }
- func TestKindlessQueries(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- type Dee struct {
- I int
- Why string
- }
- type Dum struct {
- I int
- Pling string
- }
- parent := NewKey(ctx, "Tweedle", "tweedle", 0, nil)
- keys := []*Key{
- NewKey(ctx, "Dee", "dee0", 0, parent),
- NewKey(ctx, "Dum", "dum1", 0, parent),
- NewKey(ctx, "Dum", "dum2", 0, parent),
- NewKey(ctx, "Dum", "dum3", 0, parent),
- }
- src := []interface{}{
- &Dee{1, "binary0001"},
- &Dum{2, "binary0010"},
- &Dum{4, "binary0100"},
- &Dum{8, "binary1000"},
- }
- keys, err := client.PutMulti(ctx, keys, src)
- if err != nil {
- t.Fatalf("put: %v", err)
- }
- testCases := []struct {
- desc string
- query *Query
- want []int
- wantErr string
- }{
- {
- desc: "Dee",
- query: NewQuery("Dee"),
- want: []int{1},
- },
- {
- desc: "Doh",
- query: NewQuery("Doh"),
- want: nil},
- {
- desc: "Dum",
- query: NewQuery("Dum"),
- want: []int{2, 4, 8},
- },
- {
- desc: "",
- query: NewQuery(""),
- want: []int{1, 2, 4, 8},
- },
- {
- desc: "Kindless filter",
- query: NewQuery("").Filter("__key__ =", keys[2]),
- want: []int{4},
- },
- {
- desc: "Kindless order",
- query: NewQuery("").Order("__key__"),
- want: []int{1, 2, 4, 8},
- },
- {
- desc: "Kindless bad filter",
- query: NewQuery("").Filter("I =", 4),
- wantErr: "kind is required for filter: I",
- },
- {
- desc: "Kindless bad order",
- query: NewQuery("").Order("-__key__"),
- wantErr: "kind is required for all orders except __key__ ascending",
- },
- }
- loop:
- for _, tc := range testCases {
- q := tc.query.Ancestor(parent)
- gotCount, err := client.Count(ctx, q)
- if err != nil {
- if tc.wantErr == "" || !strings.Contains(err.Error(), tc.wantErr) {
- t.Errorf("count %q: err %v, want err %q", tc.desc, err, tc.wantErr)
- }
- continue
- }
- if tc.wantErr != "" {
- t.Errorf("count %q: want err %q", tc.desc, tc.wantErr)
- continue
- }
- if gotCount != len(tc.want) {
- t.Errorf("count %q: got %d want %d", tc.desc, gotCount, len(tc.want))
- continue
- }
- var got []int
- for iter := client.Run(ctx, q); ; {
- var dst struct {
- I int
- Why, Pling string
- }
- _, err := iter.Next(&dst)
- if err == Done {
- break
- }
- if err != nil {
- t.Errorf("iter.Next %q: %v", tc.desc, err)
- continue loop
- }
- got = append(got, dst.I)
- }
- sort.Ints(got)
- if !reflect.DeepEqual(got, tc.want) {
- t.Errorf("elems %q: got %+v want %+v", tc.desc, got, tc.want)
- continue
- }
- }
- }
- func TestTransaction(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- type Counter struct {
- N int
- T time.Time
- }
- bangErr := errors.New("bang")
- tests := []struct {
- desc string
- causeConflict []bool
- retErr []error
- want int
- wantErr error
- }{
- {
- desc: "no conflicts",
- causeConflict: []bool{false},
- retErr: []error{nil},
- want: 11,
- },
- {
- desc: "user error",
- causeConflict: []bool{false},
- retErr: []error{bangErr},
- wantErr: bangErr,
- },
- {
- desc: "2 conflicts",
- causeConflict: []bool{true, true, false},
- retErr: []error{nil, nil, nil},
- want: 15, // Each conflict increments by 2.
- },
- {
- desc: "3 conflicts",
- causeConflict: []bool{true, true, true},
- retErr: []error{nil, nil, nil},
- wantErr: ErrConcurrentTransaction,
- },
- }
- for _, tt := range tests {
- // Put a new counter.
- c := &Counter{N: 10, T: time.Now()}
- key, err := client.Put(ctx, NewIncompleteKey(ctx, "TransCounter", nil), c)
- if err != nil {
- t.Errorf("%s: client.Put: %v", tt.desc, err)
- continue
- }
- defer client.Delete(ctx, key)
- // Increment the counter in a transaction.
- // The test case can manually cause a conflict or return an
- // error at each attempt.
- var attempts int
- _, err = client.RunInTransaction(ctx, func(tx *Transaction) error {
- attempts++
- if attempts > len(tt.causeConflict) {
- return fmt.Errorf("too many attempts. Got %d, max %d", attempts, len(tt.causeConflict))
- }
- var c Counter
- if err := tx.Get(key, &c); err != nil {
- return err
- }
- c.N++
- if _, err := tx.Put(key, &c); err != nil {
- return err
- }
- if tt.causeConflict[attempts-1] {
- c.N += 1
- if _, err := client.Put(ctx, key, &c); err != nil {
- return err
- }
- }
- return tt.retErr[attempts-1]
- })
- // Check the error returned by RunInTransaction.
- if err != tt.wantErr {
- t.Errorf("%s: got err %v, want %v", tt.desc, err, tt.wantErr)
- continue
- }
- if err != nil {
- continue
- }
- // Check the final value of the counter.
- if err := client.Get(ctx, key, c); err != nil {
- t.Errorf("%s: client.Get: %v", err)
- continue
- }
- if c.N != tt.want {
- t.Errorf("%s: counter N=%d, want N=%d", c.N, tt.want)
- }
- }
- }
- func TestNilPointers(t *testing.T) {
- if testing.Short() {
- t.Skip("Integration tests skipped in short mode")
- }
- ctx := context.Background()
- client := newClient(ctx, t)
- type X struct {
- S string
- }
- src := []*X{{"zero"}, {"one"}}
- keys := []*Key{NewIncompleteKey(ctx, "NilX", nil), NewIncompleteKey(ctx, "NilX", nil)}
- keys, err := client.PutMulti(ctx, keys, src)
- if err != nil {
- t.Fatalf("PutMulti: %v", err)
- }
- // It's okay to store into a slice of nil *X.
- xs := make([]*X, 2)
- if err := client.GetMulti(ctx, keys, xs); err != nil {
- t.Errorf("GetMulti: %v", err)
- } else if !reflect.DeepEqual(xs, src) {
- t.Errorf("GetMulti fetched %v, want %v", xs, src)
- }
- // It isn't okay to store into a single nil *X.
- var x0 *X
- if err, want := client.Get(ctx, keys[0], x0), ErrInvalidEntityType; err != want {
- t.Errorf("Get: err %v; want %v", err, want)
- }
- if err := client.DeleteMulti(ctx, keys); err != nil {
- t.Errorf("Delete: %v", err)
- }
- }
|