You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 

1256 lines
39 KiB

  1. /*
  2. Copyright 2015 Google LLC
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. http://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package bigtable
  14. import (
  15. "context"
  16. "fmt"
  17. "math/rand"
  18. "strings"
  19. "sync"
  20. "testing"
  21. "time"
  22. "cloud.google.com/go/internal/testutil"
  23. "github.com/google/go-cmp/cmp"
  24. "google.golang.org/api/option"
  25. btpb "google.golang.org/genproto/googleapis/bigtable/v2"
  26. "google.golang.org/grpc"
  27. )
  28. func TestPrefix(t *testing.T) {
  29. tests := []struct {
  30. prefix, succ string
  31. }{
  32. {"", ""},
  33. {"\xff", ""}, // when used, "" means Infinity
  34. {"x\xff", "y"},
  35. {"\xfe", "\xff"},
  36. }
  37. for _, tc := range tests {
  38. got := prefixSuccessor(tc.prefix)
  39. if got != tc.succ {
  40. t.Errorf("prefixSuccessor(%q) = %q, want %s", tc.prefix, got, tc.succ)
  41. continue
  42. }
  43. r := PrefixRange(tc.prefix)
  44. if tc.succ == "" && r.limit != "" {
  45. t.Errorf("PrefixRange(%q) got limit %q", tc.prefix, r.limit)
  46. }
  47. if tc.succ != "" && r.limit != tc.succ {
  48. t.Errorf("PrefixRange(%q) got limit %q, want %q", tc.prefix, r.limit, tc.succ)
  49. }
  50. }
  51. }
  52. func TestApplyErrors(t *testing.T) {
  53. ctx := context.Background()
  54. table := &Table{
  55. c: &Client{
  56. project: "P",
  57. instance: "I",
  58. },
  59. table: "t",
  60. }
  61. f := ColumnFilter("C")
  62. m := NewMutation()
  63. m.DeleteRow()
  64. // Test nested conditional mutations.
  65. cm := NewCondMutation(f, NewCondMutation(f, m, nil), nil)
  66. if err := table.Apply(ctx, "x", cm); err == nil {
  67. t.Error("got nil, want error")
  68. }
  69. cm = NewCondMutation(f, nil, NewCondMutation(f, m, nil))
  70. if err := table.Apply(ctx, "x", cm); err == nil {
  71. t.Error("got nil, want error")
  72. }
  73. }
  74. func TestGroupEntries(t *testing.T) {
  75. tests := []struct {
  76. desc string
  77. in []*entryErr
  78. size int
  79. want [][]*entryErr
  80. }{
  81. {
  82. desc: "one entry less than max size is one group",
  83. in: []*entryErr{buildEntry(5)},
  84. size: 10,
  85. want: [][]*entryErr{{buildEntry(5)}},
  86. },
  87. {
  88. desc: "one entry equal to max size is one group",
  89. in: []*entryErr{buildEntry(10)},
  90. size: 10,
  91. want: [][]*entryErr{{buildEntry(10)}},
  92. },
  93. {
  94. desc: "one entry greater than max size is one group",
  95. in: []*entryErr{buildEntry(15)},
  96. size: 10,
  97. want: [][]*entryErr{{buildEntry(15)}},
  98. },
  99. {
  100. desc: "all entries fitting within max size are one group",
  101. in: []*entryErr{buildEntry(10), buildEntry(10)},
  102. size: 20,
  103. want: [][]*entryErr{{buildEntry(10), buildEntry(10)}},
  104. },
  105. {
  106. desc: "entries each under max size and together over max size are grouped separately",
  107. in: []*entryErr{buildEntry(10), buildEntry(10)},
  108. size: 15,
  109. want: [][]*entryErr{{buildEntry(10)}, {buildEntry(10)}},
  110. },
  111. {
  112. desc: "entries together over max size are grouped by max size",
  113. in: []*entryErr{buildEntry(5), buildEntry(5), buildEntry(5)},
  114. size: 10,
  115. want: [][]*entryErr{{buildEntry(5), buildEntry(5)}, {buildEntry(5)}},
  116. },
  117. {
  118. desc: "one entry over max size and one entry under max size are two groups",
  119. in: []*entryErr{buildEntry(15), buildEntry(5)},
  120. size: 10,
  121. want: [][]*entryErr{{buildEntry(15)}, {buildEntry(5)}},
  122. },
  123. }
  124. for _, test := range tests {
  125. if got, want := groupEntries(test.in, test.size), test.want; !cmp.Equal(mutationCounts(got), mutationCounts(want)) {
  126. t.Errorf("[%s] want = %v, got = %v", test.desc, mutationCounts(want), mutationCounts(got))
  127. }
  128. }
  129. }
  130. func buildEntry(numMutations int) *entryErr {
  131. var muts []*btpb.Mutation
  132. for i := 0; i < numMutations; i++ {
  133. muts = append(muts, &btpb.Mutation{})
  134. }
  135. return &entryErr{Entry: &btpb.MutateRowsRequest_Entry{Mutations: muts}}
  136. }
  137. func mutationCounts(batched [][]*entryErr) []int {
  138. var res []int
  139. for _, entries := range batched {
  140. var count int
  141. for _, e := range entries {
  142. count += len(e.Entry.Mutations)
  143. }
  144. res = append(res, count)
  145. }
  146. return res
  147. }
  148. func TestClientIntegration(t *testing.T) {
  149. // TODO(jba): go1.9: Use subtests.
  150. start := time.Now()
  151. lastCheckpoint := start
  152. checkpoint := func(s string) {
  153. n := time.Now()
  154. t.Logf("[%s] %v since start, %v since last checkpoint", s, n.Sub(start), n.Sub(lastCheckpoint))
  155. lastCheckpoint = n
  156. }
  157. testEnv, err := NewIntegrationEnv()
  158. if err != nil {
  159. t.Fatalf("IntegrationEnv: %v", err)
  160. }
  161. var timeout time.Duration
  162. if testEnv.Config().UseProd {
  163. timeout = 10 * time.Minute
  164. t.Logf("Running test against production")
  165. } else {
  166. timeout = 5 * time.Minute
  167. t.Logf("bttest.Server running on %s", testEnv.Config().AdminEndpoint)
  168. }
  169. ctx, cancel := context.WithTimeout(context.Background(), timeout)
  170. defer cancel()
  171. client, err := testEnv.NewClient()
  172. if err != nil {
  173. t.Fatalf("Client: %v", err)
  174. }
  175. defer client.Close()
  176. checkpoint("dialed Client")
  177. adminClient, err := testEnv.NewAdminClient()
  178. if err != nil {
  179. t.Fatalf("AdminClient: %v", err)
  180. }
  181. defer adminClient.Close()
  182. checkpoint("dialed AdminClient")
  183. table := testEnv.Config().Table
  184. // Delete the table at the end of the test.
  185. // Do this even before creating the table so that if this is running
  186. // against production and CreateTable fails there's a chance of cleaning it up.
  187. defer adminClient.DeleteTable(ctx, table)
  188. if err := adminClient.CreateTable(ctx, table); err != nil {
  189. t.Fatalf("Creating table: %v", err)
  190. }
  191. checkpoint("created table")
  192. if err := adminClient.CreateColumnFamily(ctx, table, "follows"); err != nil {
  193. t.Fatalf("Creating column family: %v", err)
  194. }
  195. checkpoint(`created "follows" column family`)
  196. tbl := client.Open(table)
  197. // Insert some data.
  198. initialData := map[string][]string{
  199. "wmckinley": {"tjefferson"},
  200. "gwashington": {"jadams"},
  201. "tjefferson": {"gwashington", "jadams"}, // wmckinley set conditionally below
  202. "jadams": {"gwashington", "tjefferson"},
  203. }
  204. for row, ss := range initialData {
  205. mut := NewMutation()
  206. for _, name := range ss {
  207. mut.Set("follows", name, 1000, []byte("1"))
  208. }
  209. if err := tbl.Apply(ctx, row, mut); err != nil {
  210. t.Errorf("Mutating row %q: %v", row, err)
  211. }
  212. }
  213. checkpoint("inserted initial data")
  214. // TODO(igorbernstein): re-enable this when ready
  215. //if err := adminClient.WaitForReplication(ctx, table); err != nil {
  216. // t.Errorf("Waiting for replication for table %q: %v", table, err)
  217. //}
  218. //checkpoint("waited for replication")
  219. // Do a conditional mutation with a complex filter.
  220. mutTrue := NewMutation()
  221. mutTrue.Set("follows", "wmckinley", 1000, []byte("1"))
  222. filter := ChainFilters(ColumnFilter("gwash[iz].*"), ValueFilter("."))
  223. mut := NewCondMutation(filter, mutTrue, nil)
  224. if err := tbl.Apply(ctx, "tjefferson", mut); err != nil {
  225. t.Errorf("Conditionally mutating row: %v", err)
  226. }
  227. // Do a second condition mutation with a filter that does not match,
  228. // and thus no changes should be made.
  229. mutTrue = NewMutation()
  230. mutTrue.DeleteRow()
  231. filter = ColumnFilter("snoop.dogg")
  232. mut = NewCondMutation(filter, mutTrue, nil)
  233. if err := tbl.Apply(ctx, "tjefferson", mut); err != nil {
  234. t.Errorf("Conditionally mutating row: %v", err)
  235. }
  236. checkpoint("did two conditional mutations")
  237. // Fetch a row.
  238. row, err := tbl.ReadRow(ctx, "jadams")
  239. if err != nil {
  240. t.Fatalf("Reading a row: %v", err)
  241. }
  242. wantRow := Row{
  243. "follows": []ReadItem{
  244. {Row: "jadams", Column: "follows:gwashington", Timestamp: 1000, Value: []byte("1")},
  245. {Row: "jadams", Column: "follows:tjefferson", Timestamp: 1000, Value: []byte("1")},
  246. },
  247. }
  248. if !testutil.Equal(row, wantRow) {
  249. t.Errorf("Read row mismatch.\n got %#v\nwant %#v", row, wantRow)
  250. }
  251. checkpoint("tested ReadRow")
  252. // Do a bunch of reads with filters.
  253. readTests := []struct {
  254. desc string
  255. rr RowSet
  256. filter Filter // may be nil
  257. limit ReadOption // may be nil
  258. // We do the read, grab all the cells, turn them into "<row>-<col>-<val>",
  259. // and join with a comma.
  260. want string
  261. }{
  262. {
  263. desc: "read all, unfiltered",
  264. rr: RowRange{},
  265. want: "gwashington-jadams-1,jadams-gwashington-1,jadams-tjefferson-1,tjefferson-gwashington-1,tjefferson-jadams-1,tjefferson-wmckinley-1,wmckinley-tjefferson-1",
  266. },
  267. {
  268. desc: "read with InfiniteRange, unfiltered",
  269. rr: InfiniteRange("tjefferson"),
  270. want: "tjefferson-gwashington-1,tjefferson-jadams-1,tjefferson-wmckinley-1,wmckinley-tjefferson-1",
  271. },
  272. {
  273. desc: "read with NewRange, unfiltered",
  274. rr: NewRange("gargamel", "hubbard"),
  275. want: "gwashington-jadams-1",
  276. },
  277. {
  278. desc: "read with PrefixRange, unfiltered",
  279. rr: PrefixRange("jad"),
  280. want: "jadams-gwashington-1,jadams-tjefferson-1",
  281. },
  282. {
  283. desc: "read with SingleRow, unfiltered",
  284. rr: SingleRow("wmckinley"),
  285. want: "wmckinley-tjefferson-1",
  286. },
  287. {
  288. desc: "read all, with ColumnFilter",
  289. rr: RowRange{},
  290. filter: ColumnFilter(".*j.*"), // matches "jadams" and "tjefferson"
  291. want: "gwashington-jadams-1,jadams-tjefferson-1,tjefferson-jadams-1,wmckinley-tjefferson-1",
  292. },
  293. {
  294. desc: "read all, with ColumnFilter, prefix",
  295. rr: RowRange{},
  296. filter: ColumnFilter("j"), // no matches
  297. want: "",
  298. },
  299. {
  300. desc: "read range, with ColumnRangeFilter",
  301. rr: RowRange{},
  302. filter: ColumnRangeFilter("follows", "h", "k"),
  303. want: "gwashington-jadams-1,tjefferson-jadams-1",
  304. },
  305. {
  306. desc: "read range from empty, with ColumnRangeFilter",
  307. rr: RowRange{},
  308. filter: ColumnRangeFilter("follows", "", "u"),
  309. want: "gwashington-jadams-1,jadams-gwashington-1,jadams-tjefferson-1,tjefferson-gwashington-1,tjefferson-jadams-1,wmckinley-tjefferson-1",
  310. },
  311. {
  312. desc: "read range from start to empty, with ColumnRangeFilter",
  313. rr: RowRange{},
  314. filter: ColumnRangeFilter("follows", "h", ""),
  315. want: "gwashington-jadams-1,jadams-tjefferson-1,tjefferson-jadams-1,tjefferson-wmckinley-1,wmckinley-tjefferson-1",
  316. },
  317. {
  318. desc: "read with RowKeyFilter",
  319. rr: RowRange{},
  320. filter: RowKeyFilter(".*wash.*"),
  321. want: "gwashington-jadams-1",
  322. },
  323. {
  324. desc: "read with RowKeyFilter, prefix",
  325. rr: RowRange{},
  326. filter: RowKeyFilter("gwash"),
  327. want: "",
  328. },
  329. {
  330. desc: "read with RowKeyFilter, no matches",
  331. rr: RowRange{},
  332. filter: RowKeyFilter(".*xxx.*"),
  333. want: "",
  334. },
  335. {
  336. desc: "read with FamilyFilter, no matches",
  337. rr: RowRange{},
  338. filter: FamilyFilter(".*xxx.*"),
  339. want: "",
  340. },
  341. {
  342. desc: "read with ColumnFilter + row limit",
  343. rr: RowRange{},
  344. filter: ColumnFilter(".*j.*"), // matches "jadams" and "tjefferson"
  345. limit: LimitRows(2),
  346. want: "gwashington-jadams-1,jadams-tjefferson-1",
  347. },
  348. {
  349. desc: "read all, strip values",
  350. rr: RowRange{},
  351. filter: StripValueFilter(),
  352. want: "gwashington-jadams-,jadams-gwashington-,jadams-tjefferson-,tjefferson-gwashington-,tjefferson-jadams-,tjefferson-wmckinley-,wmckinley-tjefferson-",
  353. },
  354. {
  355. desc: "read with ColumnFilter + row limit + strip values",
  356. rr: RowRange{},
  357. filter: ChainFilters(ColumnFilter(".*j.*"), StripValueFilter()), // matches "jadams" and "tjefferson"
  358. limit: LimitRows(2),
  359. want: "gwashington-jadams-,jadams-tjefferson-",
  360. },
  361. {
  362. desc: "read with condition, strip values on true",
  363. rr: RowRange{},
  364. filter: ConditionFilter(ColumnFilter(".*j.*"), StripValueFilter(), nil),
  365. want: "gwashington-jadams-,jadams-gwashington-,jadams-tjefferson-,tjefferson-gwashington-,tjefferson-jadams-,tjefferson-wmckinley-,wmckinley-tjefferson-",
  366. },
  367. {
  368. desc: "read with condition, strip values on false",
  369. rr: RowRange{},
  370. filter: ConditionFilter(ColumnFilter(".*xxx.*"), nil, StripValueFilter()),
  371. want: "gwashington-jadams-,jadams-gwashington-,jadams-tjefferson-,tjefferson-gwashington-,tjefferson-jadams-,tjefferson-wmckinley-,wmckinley-tjefferson-",
  372. },
  373. {
  374. desc: "read with ValueRangeFilter + row limit",
  375. rr: RowRange{},
  376. filter: ValueRangeFilter([]byte("1"), []byte("5")), // matches our value of "1"
  377. limit: LimitRows(2),
  378. want: "gwashington-jadams-1,jadams-gwashington-1,jadams-tjefferson-1",
  379. },
  380. {
  381. desc: "read with ValueRangeFilter, no match on exclusive end",
  382. rr: RowRange{},
  383. filter: ValueRangeFilter([]byte("0"), []byte("1")), // no match
  384. want: "",
  385. },
  386. {
  387. desc: "read with ValueRangeFilter, no matches",
  388. rr: RowRange{},
  389. filter: ValueRangeFilter([]byte("3"), []byte("5")), // matches nothing
  390. want: "",
  391. },
  392. {
  393. desc: "read with InterleaveFilter, no matches on all filters",
  394. rr: RowRange{},
  395. filter: InterleaveFilters(ColumnFilter(".*x.*"), ColumnFilter(".*z.*")),
  396. want: "",
  397. },
  398. {
  399. desc: "read with InterleaveFilter, no duplicate cells",
  400. rr: RowRange{},
  401. filter: InterleaveFilters(ColumnFilter(".*g.*"), ColumnFilter(".*j.*")),
  402. want: "gwashington-jadams-1,jadams-gwashington-1,jadams-tjefferson-1,tjefferson-gwashington-1,tjefferson-jadams-1,wmckinley-tjefferson-1",
  403. },
  404. {
  405. desc: "read with InterleaveFilter, with duplicate cells",
  406. rr: RowRange{},
  407. filter: InterleaveFilters(ColumnFilter(".*g.*"), ColumnFilter(".*g.*")),
  408. want: "jadams-gwashington-1,jadams-gwashington-1,tjefferson-gwashington-1,tjefferson-gwashington-1",
  409. },
  410. {
  411. desc: "read with a RowRangeList and no filter",
  412. rr: RowRangeList{NewRange("gargamel", "hubbard"), InfiniteRange("wmckinley")},
  413. want: "gwashington-jadams-1,wmckinley-tjefferson-1",
  414. },
  415. {
  416. desc: "chain that excludes rows and matches nothing, in a condition",
  417. rr: RowRange{},
  418. filter: ConditionFilter(ChainFilters(ColumnFilter(".*j.*"), ColumnFilter(".*mckinley.*")), StripValueFilter(), nil),
  419. want: "",
  420. },
  421. {
  422. desc: "chain that ends with an interleave that has no match. covers #804",
  423. rr: RowRange{},
  424. filter: ConditionFilter(ChainFilters(ColumnFilter(".*j.*"), InterleaveFilters(ColumnFilter(".*x.*"), ColumnFilter(".*z.*"))), StripValueFilter(), nil),
  425. want: "",
  426. },
  427. }
  428. for _, tc := range readTests {
  429. var opts []ReadOption
  430. if tc.filter != nil {
  431. opts = append(opts, RowFilter(tc.filter))
  432. }
  433. if tc.limit != nil {
  434. opts = append(opts, tc.limit)
  435. }
  436. var elt []string
  437. err := tbl.ReadRows(ctx, tc.rr, func(r Row) bool {
  438. for _, ris := range r {
  439. for _, ri := range ris {
  440. elt = append(elt, formatReadItem(ri))
  441. }
  442. }
  443. return true
  444. }, opts...)
  445. if err != nil {
  446. t.Errorf("%s: %v", tc.desc, err)
  447. continue
  448. }
  449. if got := strings.Join(elt, ","); got != tc.want {
  450. t.Errorf("%s: wrong reads.\n got %q\nwant %q", tc.desc, got, tc.want)
  451. }
  452. }
  453. // Read a RowList
  454. var elt []string
  455. keys := RowList{"wmckinley", "gwashington", "jadams"}
  456. want := "gwashington-jadams-1,jadams-gwashington-1,jadams-tjefferson-1,wmckinley-tjefferson-1"
  457. err = tbl.ReadRows(ctx, keys, func(r Row) bool {
  458. for _, ris := range r {
  459. for _, ri := range ris {
  460. elt = append(elt, formatReadItem(ri))
  461. }
  462. }
  463. return true
  464. })
  465. if err != nil {
  466. t.Errorf("read RowList: %v", err)
  467. }
  468. if got := strings.Join(elt, ","); got != want {
  469. t.Errorf("bulk read: wrong reads.\n got %q\nwant %q", got, want)
  470. }
  471. checkpoint("tested ReadRows in a few ways")
  472. // Do a scan and stop part way through.
  473. // Verify that the ReadRows callback doesn't keep running.
  474. stopped := false
  475. err = tbl.ReadRows(ctx, InfiniteRange(""), func(r Row) bool {
  476. if r.Key() < "h" {
  477. return true
  478. }
  479. if !stopped {
  480. stopped = true
  481. return false
  482. }
  483. t.Errorf("ReadRows kept scanning to row %q after being told to stop", r.Key())
  484. return false
  485. })
  486. if err != nil {
  487. t.Errorf("Partial ReadRows: %v", err)
  488. }
  489. checkpoint("did partial ReadRows test")
  490. // Delete a row and check it goes away.
  491. mut = NewMutation()
  492. mut.DeleteRow()
  493. if err := tbl.Apply(ctx, "wmckinley", mut); err != nil {
  494. t.Errorf("Apply DeleteRow: %v", err)
  495. }
  496. row, err = tbl.ReadRow(ctx, "wmckinley")
  497. if err != nil {
  498. t.Fatalf("Reading a row after DeleteRow: %v", err)
  499. }
  500. if len(row) != 0 {
  501. t.Fatalf("Read non-zero row after DeleteRow: %v", row)
  502. }
  503. checkpoint("exercised DeleteRow")
  504. // Check ReadModifyWrite.
  505. if err := adminClient.CreateColumnFamily(ctx, table, "counter"); err != nil {
  506. t.Fatalf("Creating column family: %v", err)
  507. }
  508. appendRMW := func(b []byte) *ReadModifyWrite {
  509. rmw := NewReadModifyWrite()
  510. rmw.AppendValue("counter", "likes", b)
  511. return rmw
  512. }
  513. incRMW := func(n int64) *ReadModifyWrite {
  514. rmw := NewReadModifyWrite()
  515. rmw.Increment("counter", "likes", n)
  516. return rmw
  517. }
  518. rmwSeq := []struct {
  519. desc string
  520. rmw *ReadModifyWrite
  521. want []byte
  522. }{
  523. {
  524. desc: "append #1",
  525. rmw: appendRMW([]byte{0, 0, 0}),
  526. want: []byte{0, 0, 0},
  527. },
  528. {
  529. desc: "append #2",
  530. rmw: appendRMW([]byte{0, 0, 0, 0, 17}), // the remaining 40 bits to make a big-endian 17
  531. want: []byte{0, 0, 0, 0, 0, 0, 0, 17},
  532. },
  533. {
  534. desc: "increment",
  535. rmw: incRMW(8),
  536. want: []byte{0, 0, 0, 0, 0, 0, 0, 25},
  537. },
  538. }
  539. for _, step := range rmwSeq {
  540. row, err := tbl.ApplyReadModifyWrite(ctx, "gwashington", step.rmw)
  541. if err != nil {
  542. t.Fatalf("ApplyReadModifyWrite %+v: %v", step.rmw, err)
  543. }
  544. // Make sure the modified cell returned by the RMW operation has a timestamp.
  545. if row["counter"][0].Timestamp == 0 {
  546. t.Errorf("RMW returned cell timestamp: got %v, want > 0", row["counter"][0].Timestamp)
  547. }
  548. clearTimestamps(row)
  549. wantRow := Row{"counter": []ReadItem{{Row: "gwashington", Column: "counter:likes", Value: step.want}}}
  550. if !testutil.Equal(row, wantRow) {
  551. t.Fatalf("After %s,\n got %v\nwant %v", step.desc, row, wantRow)
  552. }
  553. }
  554. // Check for google-cloud-go/issues/723. RMWs that insert new rows should keep row order sorted in the emulator.
  555. _, err = tbl.ApplyReadModifyWrite(ctx, "issue-723-2", appendRMW([]byte{0}))
  556. if err != nil {
  557. t.Fatalf("ApplyReadModifyWrite null string: %v", err)
  558. }
  559. _, err = tbl.ApplyReadModifyWrite(ctx, "issue-723-1", appendRMW([]byte{0}))
  560. if err != nil {
  561. t.Fatalf("ApplyReadModifyWrite null string: %v", err)
  562. }
  563. // Get only the correct row back on read.
  564. r, err := tbl.ReadRow(ctx, "issue-723-1")
  565. if err != nil {
  566. t.Fatalf("Reading row: %v", err)
  567. }
  568. if r.Key() != "issue-723-1" {
  569. t.Errorf("ApplyReadModifyWrite: incorrect read after RMW,\n got %v\nwant %v", r.Key(), "issue-723-1")
  570. }
  571. checkpoint("tested ReadModifyWrite")
  572. // Test arbitrary timestamps more thoroughly.
  573. if err := adminClient.CreateColumnFamily(ctx, table, "ts"); err != nil {
  574. t.Fatalf("Creating column family: %v", err)
  575. }
  576. const numVersions = 4
  577. mut = NewMutation()
  578. for i := 1; i < numVersions; i++ {
  579. // Timestamps are used in thousands because the server
  580. // only permits that granularity.
  581. mut.Set("ts", "col", Timestamp(i*1000), []byte(fmt.Sprintf("val-%d", i)))
  582. mut.Set("ts", "col2", Timestamp(i*1000), []byte(fmt.Sprintf("val-%d", i)))
  583. }
  584. if err := tbl.Apply(ctx, "testrow", mut); err != nil {
  585. t.Fatalf("Mutating row: %v", err)
  586. }
  587. r, err = tbl.ReadRow(ctx, "testrow")
  588. if err != nil {
  589. t.Fatalf("Reading row: %v", err)
  590. }
  591. wantRow = Row{"ts": []ReadItem{
  592. // These should be returned in descending timestamp order.
  593. {Row: "testrow", Column: "ts:col", Timestamp: 3000, Value: []byte("val-3")},
  594. {Row: "testrow", Column: "ts:col", Timestamp: 2000, Value: []byte("val-2")},
  595. {Row: "testrow", Column: "ts:col", Timestamp: 1000, Value: []byte("val-1")},
  596. {Row: "testrow", Column: "ts:col2", Timestamp: 3000, Value: []byte("val-3")},
  597. {Row: "testrow", Column: "ts:col2", Timestamp: 2000, Value: []byte("val-2")},
  598. {Row: "testrow", Column: "ts:col2", Timestamp: 1000, Value: []byte("val-1")},
  599. }}
  600. if !testutil.Equal(r, wantRow) {
  601. t.Errorf("Cell with multiple versions,\n got %v\nwant %v", r, wantRow)
  602. }
  603. // Do the same read, but filter to the latest two versions.
  604. r, err = tbl.ReadRow(ctx, "testrow", RowFilter(LatestNFilter(2)))
  605. if err != nil {
  606. t.Fatalf("Reading row: %v", err)
  607. }
  608. wantRow = Row{"ts": []ReadItem{
  609. {Row: "testrow", Column: "ts:col", Timestamp: 3000, Value: []byte("val-3")},
  610. {Row: "testrow", Column: "ts:col", Timestamp: 2000, Value: []byte("val-2")},
  611. {Row: "testrow", Column: "ts:col2", Timestamp: 3000, Value: []byte("val-3")},
  612. {Row: "testrow", Column: "ts:col2", Timestamp: 2000, Value: []byte("val-2")},
  613. }}
  614. if !testutil.Equal(r, wantRow) {
  615. t.Errorf("Cell with multiple versions and LatestNFilter(2),\n got %v\nwant %v", r, wantRow)
  616. }
  617. // Check cell offset / limit
  618. r, err = tbl.ReadRow(ctx, "testrow", RowFilter(CellsPerRowLimitFilter(3)))
  619. if err != nil {
  620. t.Fatalf("Reading row: %v", err)
  621. }
  622. wantRow = Row{"ts": []ReadItem{
  623. {Row: "testrow", Column: "ts:col", Timestamp: 3000, Value: []byte("val-3")},
  624. {Row: "testrow", Column: "ts:col", Timestamp: 2000, Value: []byte("val-2")},
  625. {Row: "testrow", Column: "ts:col", Timestamp: 1000, Value: []byte("val-1")},
  626. }}
  627. if !testutil.Equal(r, wantRow) {
  628. t.Errorf("Cell with multiple versions and CellsPerRowLimitFilter(3),\n got %v\nwant %v", r, wantRow)
  629. }
  630. r, err = tbl.ReadRow(ctx, "testrow", RowFilter(CellsPerRowOffsetFilter(3)))
  631. if err != nil {
  632. t.Fatalf("Reading row: %v", err)
  633. }
  634. wantRow = Row{"ts": []ReadItem{
  635. {Row: "testrow", Column: "ts:col2", Timestamp: 3000, Value: []byte("val-3")},
  636. {Row: "testrow", Column: "ts:col2", Timestamp: 2000, Value: []byte("val-2")},
  637. {Row: "testrow", Column: "ts:col2", Timestamp: 1000, Value: []byte("val-1")},
  638. }}
  639. if !testutil.Equal(r, wantRow) {
  640. t.Errorf("Cell with multiple versions and CellsPerRowOffsetFilter(3),\n got %v\nwant %v", r, wantRow)
  641. }
  642. // Check timestamp range filtering (with truncation)
  643. r, err = tbl.ReadRow(ctx, "testrow", RowFilter(TimestampRangeFilterMicros(1001, 3000)))
  644. if err != nil {
  645. t.Fatalf("Reading row: %v", err)
  646. }
  647. wantRow = Row{"ts": []ReadItem{
  648. {Row: "testrow", Column: "ts:col", Timestamp: 2000, Value: []byte("val-2")},
  649. {Row: "testrow", Column: "ts:col", Timestamp: 1000, Value: []byte("val-1")},
  650. {Row: "testrow", Column: "ts:col2", Timestamp: 2000, Value: []byte("val-2")},
  651. {Row: "testrow", Column: "ts:col2", Timestamp: 1000, Value: []byte("val-1")},
  652. }}
  653. if !testutil.Equal(r, wantRow) {
  654. t.Errorf("Cell with multiple versions and TimestampRangeFilter(1000, 3000),\n got %v\nwant %v", r, wantRow)
  655. }
  656. r, err = tbl.ReadRow(ctx, "testrow", RowFilter(TimestampRangeFilterMicros(1000, 0)))
  657. if err != nil {
  658. t.Fatalf("Reading row: %v", err)
  659. }
  660. wantRow = Row{"ts": []ReadItem{
  661. {Row: "testrow", Column: "ts:col", Timestamp: 3000, Value: []byte("val-3")},
  662. {Row: "testrow", Column: "ts:col", Timestamp: 2000, Value: []byte("val-2")},
  663. {Row: "testrow", Column: "ts:col", Timestamp: 1000, Value: []byte("val-1")},
  664. {Row: "testrow", Column: "ts:col2", Timestamp: 3000, Value: []byte("val-3")},
  665. {Row: "testrow", Column: "ts:col2", Timestamp: 2000, Value: []byte("val-2")},
  666. {Row: "testrow", Column: "ts:col2", Timestamp: 1000, Value: []byte("val-1")},
  667. }}
  668. if !testutil.Equal(r, wantRow) {
  669. t.Errorf("Cell with multiple versions and TimestampRangeFilter(1000, 0),\n got %v\nwant %v", r, wantRow)
  670. }
  671. // Delete non-existing cells, no such column family in this row
  672. // Should not delete anything
  673. if err := adminClient.CreateColumnFamily(ctx, table, "non-existing"); err != nil {
  674. t.Fatalf("Creating column family: %v", err)
  675. }
  676. mut = NewMutation()
  677. mut.DeleteTimestampRange("non-existing", "col", 2000, 3000) // half-open interval
  678. if err := tbl.Apply(ctx, "testrow", mut); err != nil {
  679. t.Fatalf("Mutating row: %v", err)
  680. }
  681. r, err = tbl.ReadRow(ctx, "testrow", RowFilter(LatestNFilter(3)))
  682. if err != nil {
  683. t.Fatalf("Reading row: %v", err)
  684. }
  685. if !testutil.Equal(r, wantRow) {
  686. t.Errorf("Cell was deleted unexpectly,\n got %v\nwant %v", r, wantRow)
  687. }
  688. // Delete non-existing cells, no such column in this column family
  689. // Should not delete anything
  690. mut = NewMutation()
  691. mut.DeleteTimestampRange("ts", "non-existing", 2000, 3000) // half-open interval
  692. if err := tbl.Apply(ctx, "testrow", mut); err != nil {
  693. t.Fatalf("Mutating row: %v", err)
  694. }
  695. r, err = tbl.ReadRow(ctx, "testrow", RowFilter(LatestNFilter(3)))
  696. if err != nil {
  697. t.Fatalf("Reading row: %v", err)
  698. }
  699. if !testutil.Equal(r, wantRow) {
  700. t.Errorf("Cell was deleted unexpectly,\n got %v\nwant %v", r, wantRow)
  701. }
  702. // Delete the cell with timestamp 2000 and repeat the last read,
  703. // checking that we get ts 3000 and ts 1000.
  704. mut = NewMutation()
  705. mut.DeleteTimestampRange("ts", "col", 2001, 3000) // half-open interval
  706. if err := tbl.Apply(ctx, "testrow", mut); err != nil {
  707. t.Fatalf("Mutating row: %v", err)
  708. }
  709. r, err = tbl.ReadRow(ctx, "testrow", RowFilter(LatestNFilter(2)))
  710. if err != nil {
  711. t.Fatalf("Reading row: %v", err)
  712. }
  713. wantRow = Row{"ts": []ReadItem{
  714. {Row: "testrow", Column: "ts:col", Timestamp: 3000, Value: []byte("val-3")},
  715. {Row: "testrow", Column: "ts:col", Timestamp: 1000, Value: []byte("val-1")},
  716. {Row: "testrow", Column: "ts:col2", Timestamp: 3000, Value: []byte("val-3")},
  717. {Row: "testrow", Column: "ts:col2", Timestamp: 2000, Value: []byte("val-2")},
  718. }}
  719. if !testutil.Equal(r, wantRow) {
  720. t.Errorf("Cell with multiple versions and LatestNFilter(2), after deleting timestamp 2000,\n got %v\nwant %v", r, wantRow)
  721. }
  722. checkpoint("tested multiple versions in a cell")
  723. // Check DeleteCellsInFamily
  724. if err := adminClient.CreateColumnFamily(ctx, table, "status"); err != nil {
  725. t.Fatalf("Creating column family: %v", err)
  726. }
  727. mut = NewMutation()
  728. mut.Set("status", "start", 2000, []byte("2"))
  729. mut.Set("status", "end", 3000, []byte("3"))
  730. mut.Set("ts", "col", 1000, []byte("1"))
  731. if err := tbl.Apply(ctx, "row1", mut); err != nil {
  732. t.Errorf("Mutating row: %v", err)
  733. }
  734. if err := tbl.Apply(ctx, "row2", mut); err != nil {
  735. t.Errorf("Mutating row: %v", err)
  736. }
  737. mut = NewMutation()
  738. mut.DeleteCellsInFamily("status")
  739. if err := tbl.Apply(ctx, "row1", mut); err != nil {
  740. t.Errorf("Delete cf: %v", err)
  741. }
  742. // ColumnFamily removed
  743. r, err = tbl.ReadRow(ctx, "row1")
  744. if err != nil {
  745. t.Fatalf("Reading row: %v", err)
  746. }
  747. wantRow = Row{"ts": []ReadItem{
  748. {Row: "row1", Column: "ts:col", Timestamp: 1000, Value: []byte("1")},
  749. }}
  750. if !testutil.Equal(r, wantRow) {
  751. t.Errorf("column family was not deleted.\n got %v\n want %v", r, wantRow)
  752. }
  753. // ColumnFamily not removed
  754. r, err = tbl.ReadRow(ctx, "row2")
  755. if err != nil {
  756. t.Fatalf("Reading row: %v", err)
  757. }
  758. wantRow = Row{
  759. "ts": []ReadItem{
  760. {Row: "row2", Column: "ts:col", Timestamp: 1000, Value: []byte("1")},
  761. },
  762. "status": []ReadItem{
  763. {Row: "row2", Column: "status:end", Timestamp: 3000, Value: []byte("3")},
  764. {Row: "row2", Column: "status:start", Timestamp: 2000, Value: []byte("2")},
  765. },
  766. }
  767. if !testutil.Equal(r, wantRow) {
  768. t.Errorf("Column family was deleted unexpectedly.\n got %v\n want %v", r, wantRow)
  769. }
  770. checkpoint("tested family delete")
  771. // Check DeleteCellsInColumn
  772. mut = NewMutation()
  773. mut.Set("status", "start", 2000, []byte("2"))
  774. mut.Set("status", "middle", 3000, []byte("3"))
  775. mut.Set("status", "end", 1000, []byte("1"))
  776. if err := tbl.Apply(ctx, "row3", mut); err != nil {
  777. t.Errorf("Mutating row: %v", err)
  778. }
  779. mut = NewMutation()
  780. mut.DeleteCellsInColumn("status", "middle")
  781. if err := tbl.Apply(ctx, "row3", mut); err != nil {
  782. t.Errorf("Delete column: %v", err)
  783. }
  784. r, err = tbl.ReadRow(ctx, "row3")
  785. if err != nil {
  786. t.Fatalf("Reading row: %v", err)
  787. }
  788. wantRow = Row{
  789. "status": []ReadItem{
  790. {Row: "row3", Column: "status:end", Timestamp: 1000, Value: []byte("1")},
  791. {Row: "row3", Column: "status:start", Timestamp: 2000, Value: []byte("2")},
  792. },
  793. }
  794. if !testutil.Equal(r, wantRow) {
  795. t.Errorf("Column was not deleted.\n got %v\n want %v", r, wantRow)
  796. }
  797. mut = NewMutation()
  798. mut.DeleteCellsInColumn("status", "start")
  799. if err := tbl.Apply(ctx, "row3", mut); err != nil {
  800. t.Errorf("Delete column: %v", err)
  801. }
  802. r, err = tbl.ReadRow(ctx, "row3")
  803. if err != nil {
  804. t.Fatalf("Reading row: %v", err)
  805. }
  806. wantRow = Row{
  807. "status": []ReadItem{
  808. {Row: "row3", Column: "status:end", Timestamp: 1000, Value: []byte("1")},
  809. },
  810. }
  811. if !testutil.Equal(r, wantRow) {
  812. t.Errorf("Column was not deleted.\n got %v\n want %v", r, wantRow)
  813. }
  814. mut = NewMutation()
  815. mut.DeleteCellsInColumn("status", "end")
  816. if err := tbl.Apply(ctx, "row3", mut); err != nil {
  817. t.Errorf("Delete column: %v", err)
  818. }
  819. r, err = tbl.ReadRow(ctx, "row3")
  820. if err != nil {
  821. t.Fatalf("Reading row: %v", err)
  822. }
  823. if len(r) != 0 {
  824. t.Errorf("Delete column: got %v, want empty row", r)
  825. }
  826. // Add same cell after delete
  827. mut = NewMutation()
  828. mut.Set("status", "end", 1000, []byte("1"))
  829. if err := tbl.Apply(ctx, "row3", mut); err != nil {
  830. t.Errorf("Mutating row: %v", err)
  831. }
  832. r, err = tbl.ReadRow(ctx, "row3")
  833. if err != nil {
  834. t.Fatalf("Reading row: %v", err)
  835. }
  836. if !testutil.Equal(r, wantRow) {
  837. t.Errorf("Column was not deleted correctly.\n got %v\n want %v", r, wantRow)
  838. }
  839. checkpoint("tested column delete")
  840. // Do highly concurrent reads/writes.
  841. // TODO(dsymonds): Raise this to 1000 when https://github.com/grpc/grpc-go/issues/205 is resolved.
  842. const maxConcurrency = 100
  843. var wg sync.WaitGroup
  844. for i := 0; i < maxConcurrency; i++ {
  845. wg.Add(1)
  846. go func() {
  847. defer wg.Done()
  848. switch r := rand.Intn(100); { // r ∈ [0,100)
  849. case 0 <= r && r < 30:
  850. // Do a read.
  851. _, err := tbl.ReadRow(ctx, "testrow", RowFilter(LatestNFilter(1)))
  852. if err != nil {
  853. t.Errorf("Concurrent read: %v", err)
  854. }
  855. case 30 <= r && r < 100:
  856. // Do a write.
  857. mut := NewMutation()
  858. mut.Set("ts", "col", 1000, []byte("data"))
  859. if err := tbl.Apply(ctx, "testrow", mut); err != nil {
  860. t.Errorf("Concurrent write: %v", err)
  861. }
  862. }
  863. }()
  864. }
  865. wg.Wait()
  866. checkpoint("tested high concurrency")
  867. // Large reads, writes and scans.
  868. bigBytes := make([]byte, 5<<20) // 5 MB is larger than current default gRPC max of 4 MB, but less than the max we set.
  869. nonsense := []byte("lorem ipsum dolor sit amet, ")
  870. fill(bigBytes, nonsense)
  871. mut = NewMutation()
  872. mut.Set("ts", "col", 1000, bigBytes)
  873. if err := tbl.Apply(ctx, "bigrow", mut); err != nil {
  874. t.Errorf("Big write: %v", err)
  875. }
  876. r, err = tbl.ReadRow(ctx, "bigrow")
  877. if err != nil {
  878. t.Errorf("Big read: %v", err)
  879. }
  880. wantRow = Row{"ts": []ReadItem{
  881. {Row: "bigrow", Column: "ts:col", Timestamp: 1000, Value: bigBytes},
  882. }}
  883. if !testutil.Equal(r, wantRow) {
  884. t.Errorf("Big read returned incorrect bytes: %v", r)
  885. }
  886. // Now write 1000 rows, each with 82 KB values, then scan them all.
  887. medBytes := make([]byte, 82<<10)
  888. fill(medBytes, nonsense)
  889. sem := make(chan int, 50) // do up to 50 mutations at a time.
  890. for i := 0; i < 1000; i++ {
  891. mut := NewMutation()
  892. mut.Set("ts", "big-scan", 1000, medBytes)
  893. row := fmt.Sprintf("row-%d", i)
  894. wg.Add(1)
  895. go func() {
  896. defer wg.Done()
  897. defer func() { <-sem }()
  898. sem <- 1
  899. if err := tbl.Apply(ctx, row, mut); err != nil {
  900. t.Errorf("Preparing large scan: %v", err)
  901. }
  902. }()
  903. }
  904. wg.Wait()
  905. n := 0
  906. err = tbl.ReadRows(ctx, PrefixRange("row-"), func(r Row) bool {
  907. for _, ris := range r {
  908. for _, ri := range ris {
  909. n += len(ri.Value)
  910. }
  911. }
  912. return true
  913. }, RowFilter(ColumnFilter("big-scan")))
  914. if err != nil {
  915. t.Errorf("Doing large scan: %v", err)
  916. }
  917. if want := 1000 * len(medBytes); n != want {
  918. t.Errorf("Large scan returned %d bytes, want %d", n, want)
  919. }
  920. // Scan a subset of the 1000 rows that we just created, using a LimitRows ReadOption.
  921. rc := 0
  922. wantRc := 3
  923. err = tbl.ReadRows(ctx, PrefixRange("row-"), func(r Row) bool {
  924. rc++
  925. return true
  926. }, LimitRows(int64(wantRc)))
  927. if err != nil {
  928. t.Fatal(err)
  929. }
  930. if rc != wantRc {
  931. t.Errorf("Scan with row limit returned %d rows, want %d", rc, wantRc)
  932. }
  933. checkpoint("tested big read/write/scan")
  934. // Test bulk mutations
  935. if err := adminClient.CreateColumnFamily(ctx, table, "bulk"); err != nil {
  936. t.Fatalf("Creating column family: %v", err)
  937. }
  938. bulkData := map[string][]string{
  939. "red sox": {"2004", "2007", "2013"},
  940. "patriots": {"2001", "2003", "2004", "2014"},
  941. "celtics": {"1981", "1984", "1986", "2008"},
  942. }
  943. var rowKeys []string
  944. var muts []*Mutation
  945. for row, ss := range bulkData {
  946. mut := NewMutation()
  947. for _, name := range ss {
  948. mut.Set("bulk", name, 1000, []byte("1"))
  949. }
  950. rowKeys = append(rowKeys, row)
  951. muts = append(muts, mut)
  952. }
  953. status, err := tbl.ApplyBulk(ctx, rowKeys, muts)
  954. if err != nil {
  955. t.Fatalf("Bulk mutating rows %q: %v", rowKeys, err)
  956. }
  957. if status != nil {
  958. t.Errorf("non-nil errors: %v", err)
  959. }
  960. checkpoint("inserted bulk data")
  961. // Read each row back
  962. for rowKey, ss := range bulkData {
  963. row, err := tbl.ReadRow(ctx, rowKey)
  964. if err != nil {
  965. t.Fatalf("Reading a bulk row: %v", err)
  966. }
  967. var wantItems []ReadItem
  968. for _, val := range ss {
  969. wantItems = append(wantItems, ReadItem{Row: rowKey, Column: "bulk:" + val, Timestamp: 1000, Value: []byte("1")})
  970. }
  971. wantRow := Row{"bulk": wantItems}
  972. if !testutil.Equal(row, wantRow) {
  973. t.Errorf("Read row mismatch.\n got %#v\nwant %#v", row, wantRow)
  974. }
  975. }
  976. checkpoint("tested reading from bulk insert")
  977. // Test bulk write errors.
  978. // Note: Setting timestamps as ServerTime makes sure the mutations are not retried on error.
  979. badMut := NewMutation()
  980. badMut.Set("badfamily", "col", ServerTime, nil)
  981. badMut2 := NewMutation()
  982. badMut2.Set("badfamily2", "goodcol", ServerTime, []byte("1"))
  983. status, err = tbl.ApplyBulk(ctx, []string{"badrow", "badrow2"}, []*Mutation{badMut, badMut2})
  984. if err != nil {
  985. t.Fatalf("Bulk mutating rows %q: %v", rowKeys, err)
  986. }
  987. if status == nil {
  988. t.Errorf("No errors for bad bulk mutation")
  989. } else if status[0] == nil || status[1] == nil {
  990. t.Errorf("No error for bad bulk mutation")
  991. }
  992. }
  993. type requestCountingInterceptor struct {
  994. grpc.ClientStream
  995. requestCallback func()
  996. }
  997. func (i *requestCountingInterceptor) SendMsg(m interface{}) error {
  998. i.requestCallback()
  999. return i.ClientStream.SendMsg(m)
  1000. }
  1001. func (i *requestCountingInterceptor) RecvMsg(m interface{}) error {
  1002. return i.ClientStream.RecvMsg(m)
  1003. }
  1004. func requestCallback(callback func()) func(ctx context.Context, desc *grpc.StreamDesc, cc *grpc.ClientConn, method string, streamer grpc.Streamer, opts ...grpc.CallOption) (grpc.ClientStream, error) {
  1005. return func(ctx context.Context, desc *grpc.StreamDesc, cc *grpc.ClientConn, method string, streamer grpc.Streamer, opts ...grpc.CallOption) (grpc.ClientStream, error) {
  1006. clientStream, err := streamer(ctx, desc, cc, method, opts...)
  1007. return &requestCountingInterceptor{
  1008. ClientStream: clientStream,
  1009. requestCallback: callback,
  1010. }, err
  1011. }
  1012. }
  1013. // TestReadRowsInvalidRowSet verifies that the client doesn't send ReadRows() requests with invalid RowSets.
  1014. func TestReadRowsInvalidRowSet(t *testing.T) {
  1015. testEnv, err := NewEmulatedEnv(IntegrationTestConfig{})
  1016. if err != nil {
  1017. t.Fatalf("NewEmulatedEnv failed: %v", err)
  1018. }
  1019. var requestCount int
  1020. incrementRequestCount := func() { requestCount++ }
  1021. conn, err := grpc.Dial(testEnv.server.Addr, grpc.WithInsecure(), grpc.WithBlock(),
  1022. grpc.WithDefaultCallOptions(grpc.MaxCallSendMsgSize(100<<20), grpc.MaxCallRecvMsgSize(100<<20)),
  1023. grpc.WithStreamInterceptor(requestCallback(incrementRequestCount)),
  1024. )
  1025. if err != nil {
  1026. t.Fatalf("grpc.Dial failed: %v", err)
  1027. }
  1028. ctx, cancel := context.WithTimeout(context.Background(), 20*time.Second)
  1029. defer cancel()
  1030. adminClient, err := NewAdminClient(ctx, testEnv.config.Project, testEnv.config.Instance, option.WithGRPCConn(conn))
  1031. if err != nil {
  1032. t.Fatalf("NewClient failed: %v", err)
  1033. }
  1034. defer adminClient.Close()
  1035. if err := adminClient.CreateTable(ctx, testEnv.config.Table); err != nil {
  1036. t.Fatalf("CreateTable(%v) failed: %v", testEnv.config.Table, err)
  1037. }
  1038. client, err := NewClient(ctx, testEnv.config.Project, testEnv.config.Instance, option.WithGRPCConn(conn))
  1039. if err != nil {
  1040. t.Fatalf("NewClient failed: %v", err)
  1041. }
  1042. defer client.Close()
  1043. table := client.Open(testEnv.config.Table)
  1044. tests := []struct {
  1045. rr RowSet
  1046. valid bool
  1047. }{
  1048. {
  1049. rr: RowRange{},
  1050. valid: true,
  1051. },
  1052. {
  1053. rr: RowRange{start: "b"},
  1054. valid: true,
  1055. },
  1056. {
  1057. rr: RowRange{start: "b", limit: "c"},
  1058. valid: true,
  1059. },
  1060. {
  1061. rr: RowRange{start: "b", limit: "a"},
  1062. valid: false,
  1063. },
  1064. {
  1065. rr: RowList{"a"},
  1066. valid: true,
  1067. },
  1068. {
  1069. rr: RowList{},
  1070. valid: false,
  1071. },
  1072. }
  1073. for _, test := range tests {
  1074. requestCount = 0
  1075. err = table.ReadRows(ctx, test.rr, func(r Row) bool { return true })
  1076. if err != nil {
  1077. t.Fatalf("ReadRows(%v) failed: %v", test.rr, err)
  1078. }
  1079. requestValid := requestCount != 0
  1080. if requestValid != test.valid {
  1081. t.Errorf("%s: got %v, want %v", test.rr, requestValid, test.valid)
  1082. }
  1083. }
  1084. }
  1085. func formatReadItem(ri ReadItem) string {
  1086. // Use the column qualifier only to make the test data briefer.
  1087. col := ri.Column[strings.Index(ri.Column, ":")+1:]
  1088. return fmt.Sprintf("%s-%s-%s", ri.Row, col, ri.Value)
  1089. }
  1090. func fill(b, sub []byte) {
  1091. for len(b) > len(sub) {
  1092. n := copy(b, sub)
  1093. b = b[n:]
  1094. }
  1095. }
  1096. func clearTimestamps(r Row) {
  1097. for _, ris := range r {
  1098. for i := range ris {
  1099. ris[i].Timestamp = 0
  1100. }
  1101. }
  1102. }
  1103. func TestSampleRowKeys(t *testing.T) {
  1104. start := time.Now()
  1105. lastCheckpoint := start
  1106. checkpoint := func(s string) {
  1107. n := time.Now()
  1108. t.Logf("[%s] %v since start, %v since last checkpoint", s, n.Sub(start), n.Sub(lastCheckpoint))
  1109. lastCheckpoint = n
  1110. }
  1111. ctx := context.Background()
  1112. client, adminClient, table, err := doSetup(ctx)
  1113. if err != nil {
  1114. t.Fatalf("%v", err)
  1115. }
  1116. defer client.Close()
  1117. defer adminClient.Close()
  1118. tbl := client.Open(table)
  1119. // Delete the table at the end of the test.
  1120. // Do this even before creating the table so that if this is running
  1121. // against production and CreateTable fails there's a chance of cleaning it up.
  1122. defer adminClient.DeleteTable(ctx, table)
  1123. // Insert some data.
  1124. initialData := map[string][]string{
  1125. "wmckinley11": {"tjefferson11"},
  1126. "gwashington77": {"jadams77"},
  1127. "tjefferson0": {"gwashington0", "jadams0"},
  1128. }
  1129. for row, ss := range initialData {
  1130. mut := NewMutation()
  1131. for _, name := range ss {
  1132. mut.Set("follows", name, 1000, []byte("1"))
  1133. }
  1134. if err := tbl.Apply(ctx, row, mut); err != nil {
  1135. t.Errorf("Mutating row %q: %v", row, err)
  1136. }
  1137. }
  1138. checkpoint("inserted initial data")
  1139. sampleKeys, err := tbl.SampleRowKeys(context.Background())
  1140. if err != nil {
  1141. t.Errorf("%s: %v", "SampleRowKeys:", err)
  1142. }
  1143. if len(sampleKeys) == 0 {
  1144. t.Error("SampleRowKeys length 0")
  1145. }
  1146. checkpoint("tested SampleRowKeys.")
  1147. }
  1148. func doSetup(ctx context.Context) (*Client, *AdminClient, string, error) {
  1149. start := time.Now()
  1150. lastCheckpoint := start
  1151. checkpoint := func(s string) {
  1152. n := time.Now()
  1153. fmt.Printf("[%s] %v since start, %v since last checkpoint", s, n.Sub(start), n.Sub(lastCheckpoint))
  1154. lastCheckpoint = n
  1155. }
  1156. testEnv, err := NewIntegrationEnv()
  1157. if err != nil {
  1158. return nil, nil, "", fmt.Errorf("IntegrationEnv: %v", err)
  1159. }
  1160. var timeout time.Duration
  1161. if testEnv.Config().UseProd {
  1162. timeout = 10 * time.Minute
  1163. fmt.Printf("Running test against production")
  1164. } else {
  1165. timeout = 1 * time.Minute
  1166. fmt.Printf("bttest.Server running on %s", testEnv.Config().AdminEndpoint)
  1167. }
  1168. ctx, cancel := context.WithTimeout(ctx, timeout)
  1169. defer cancel()
  1170. client, err := testEnv.NewClient()
  1171. if err != nil {
  1172. return nil, nil, "", fmt.Errorf("Client: %v", err)
  1173. }
  1174. checkpoint("dialed Client")
  1175. adminClient, err := testEnv.NewAdminClient()
  1176. if err != nil {
  1177. return nil, nil, "", fmt.Errorf("AdminClient: %v", err)
  1178. }
  1179. checkpoint("dialed AdminClient")
  1180. table := testEnv.Config().Table
  1181. if err := adminClient.CreateTable(ctx, table); err != nil {
  1182. return nil, nil, "", fmt.Errorf("Creating table: %v", err)
  1183. }
  1184. checkpoint("created table")
  1185. if err := adminClient.CreateColumnFamily(ctx, table, "follows"); err != nil {
  1186. return nil, nil, "", fmt.Errorf("Creating column family: %v", err)
  1187. }
  1188. checkpoint(`created "follows" column family`)
  1189. return client, adminClient, table, nil
  1190. }