get.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. // Copyright (C) 2015 The GoHBase Authors. All rights reserved.
  2. // This file is part of GoHBase.
  3. // Use of this source code is governed by the Apache License 2.0
  4. // that can be found in the COPYING file.
  5. package hrpc
  6. import (
  7. "context"
  8. "github.com/golang/protobuf/proto"
  9. "github.com/tsuna/gohbase/pb"
  10. )
  11. // Get represents a Get HBase call.
  12. type Get struct {
  13. base
  14. baseQuery
  15. // Don't return any KeyValue, just say whether the row key exists in the
  16. // table or not.
  17. existsOnly bool
  18. skipbatch bool
  19. }
  20. // baseGet returns a Get struct with default values set.
  21. func baseGet(ctx context.Context, table []byte, key []byte,
  22. options ...func(Call) error) (*Get, error) {
  23. g := &Get{
  24. base: base{
  25. key: key,
  26. table: table,
  27. ctx: ctx,
  28. resultch: make(chan RPCResult, 1),
  29. },
  30. baseQuery: newBaseQuery(),
  31. }
  32. err := applyOptions(g, options...)
  33. if err != nil {
  34. return nil, err
  35. }
  36. return g, nil
  37. }
  38. // NewGet creates a new Get request for the given table and row key.
  39. func NewGet(ctx context.Context, table, key []byte,
  40. options ...func(Call) error) (*Get, error) {
  41. return baseGet(ctx, table, key, options...)
  42. }
  43. // NewGetStr creates a new Get request for the given table and row key.
  44. func NewGetStr(ctx context.Context, table, key string,
  45. options ...func(Call) error) (*Get, error) {
  46. return NewGet(ctx, []byte(table), []byte(key), options...)
  47. }
  48. // Name returns the name of this RPC call.
  49. func (g *Get) Name() string {
  50. return "Get"
  51. }
  52. // SkipBatch returns true if the Get request shouldn't be batched,
  53. // but should be sent to Region Server right away.
  54. func (g *Get) SkipBatch() bool {
  55. return g.skipbatch
  56. }
  57. func (g *Get) setSkipBatch(v bool) {
  58. g.skipbatch = v
  59. }
  60. // ExistsOnly makes this Get request not return any KeyValue, merely whether
  61. // or not the given row key exists in the table.
  62. func (g *Get) ExistsOnly() {
  63. g.existsOnly = true
  64. }
  65. // ToProto converts this RPC into a protobuf message.
  66. func (g *Get) ToProto() proto.Message {
  67. get := &pb.GetRequest{
  68. Region: g.regionSpecifier(),
  69. Get: &pb.Get{
  70. Row: g.key,
  71. Column: familiesToColumn(g.families),
  72. TimeRange: &pb.TimeRange{},
  73. },
  74. }
  75. /* added support for limit number of cells per row */
  76. if g.storeLimit != DefaultMaxResultsPerColumnFamily {
  77. get.Get.StoreLimit = &g.storeLimit
  78. }
  79. if g.storeOffset != 0 {
  80. get.Get.StoreOffset = &g.storeOffset
  81. }
  82. if g.maxVersions != DefaultMaxVersions {
  83. get.Get.MaxVersions = &g.maxVersions
  84. }
  85. if g.fromTimestamp != MinTimestamp {
  86. get.Get.TimeRange.From = &g.fromTimestamp
  87. }
  88. if g.toTimestamp != MaxTimestamp {
  89. get.Get.TimeRange.To = &g.toTimestamp
  90. }
  91. if g.existsOnly {
  92. get.Get.ExistenceOnly = proto.Bool(true)
  93. }
  94. get.Get.Filter = g.filter
  95. return get
  96. }
  97. // NewResponse creates an empty protobuf message to read the response of this
  98. // RPC.
  99. func (g *Get) NewResponse() proto.Message {
  100. return &pb.GetResponse{}
  101. }
  102. // DeserializeCellBlocks deserializes get result from cell blocks
  103. func (g *Get) DeserializeCellBlocks(m proto.Message, b []byte) (uint32, error) {
  104. resp := m.(*pb.GetResponse)
  105. if resp.Result == nil {
  106. // TODO: is this possible?
  107. return 0, nil
  108. }
  109. cells, read, err := deserializeCellBlocks(b, uint32(resp.Result.GetAssociatedCellCount()))
  110. if err != nil {
  111. return 0, err
  112. }
  113. resp.Result.Cell = append(resp.Result.Cell, cells...)
  114. return read, nil
  115. }
  116. // familiesToColumn takes a map from strings to lists of strings, and converts
  117. // them into protobuf Columns
  118. func familiesToColumn(families map[string][]string) []*pb.Column {
  119. cols := make([]*pb.Column, len(families))
  120. counter := 0
  121. for family, qualifiers := range families {
  122. bytequals := make([][]byte, len(qualifiers))
  123. for i, qual := range qualifiers {
  124. bytequals[i] = []byte(qual)
  125. }
  126. cols[counter] = &pb.Column{
  127. Family: []byte(family),
  128. Qualifier: bytequals,
  129. }
  130. counter++
  131. }
  132. return cols
  133. }