query.go 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. // Copyright (C) 2017 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. "errors"
  8. "math"
  9. "time"
  10. "github.com/tsuna/gohbase/filter"
  11. "github.com/tsuna/gohbase/pb"
  12. )
  13. // baseQuery bundles common fields that can be provided for quering requests: Scans and Gets
  14. type baseQuery struct {
  15. families map[string][]string
  16. filter *pb.Filter
  17. fromTimestamp uint64
  18. toTimestamp uint64
  19. maxVersions uint32
  20. storeLimit uint32
  21. storeOffset uint32
  22. }
  23. // newBaseQuery return baseQuery with all default values
  24. func newBaseQuery() baseQuery {
  25. return baseQuery{
  26. storeLimit: DefaultMaxResultsPerColumnFamily,
  27. fromTimestamp: MinTimestamp,
  28. toTimestamp: MaxTimestamp,
  29. maxVersions: DefaultMaxVersions,
  30. }
  31. }
  32. func (bq *baseQuery) setFamilies(families map[string][]string) {
  33. bq.families = families
  34. }
  35. func (bq *baseQuery) setFilter(filter *pb.Filter) {
  36. bq.filter = filter
  37. }
  38. func (bq *baseQuery) setTimeRangeUint64(from, to uint64) {
  39. bq.fromTimestamp = from
  40. bq.toTimestamp = to
  41. }
  42. func (bq *baseQuery) setMaxVersions(versions uint32) {
  43. bq.maxVersions = versions
  44. }
  45. func (bq *baseQuery) setMaxResultsPerColumnFamily(maxresults uint32) {
  46. bq.storeLimit = maxresults
  47. }
  48. func (bq *baseQuery) setResultOffset(offset uint32) {
  49. bq.storeOffset = offset
  50. }
  51. // Families option adds families constraint to a Scan or Get request.
  52. func Families(f map[string][]string) func(Call) error {
  53. return func(hc Call) error {
  54. if c, ok := hc.(hasQueryOptions); ok {
  55. c.setFamilies(f)
  56. return nil
  57. }
  58. return errors.New("'Families' option can only be used with Get or Scan request")
  59. }
  60. }
  61. // Filters option adds filters constraint to a Scan or Get request.
  62. func Filters(f filter.Filter) func(Call) error {
  63. return func(hc Call) error {
  64. if c, ok := hc.(hasQueryOptions); ok {
  65. pbF, err := f.ConstructPBFilter()
  66. if err != nil {
  67. return err
  68. }
  69. c.setFilter(pbF)
  70. return nil
  71. }
  72. return errors.New("'Filters' option can only be used with Get or Scan request")
  73. }
  74. }
  75. // TimeRange is used as a parameter for request creation. Adds TimeRange constraint to a request.
  76. // It will get values in range [from, to[ ('to' is exclusive).
  77. func TimeRange(from, to time.Time) func(Call) error {
  78. return TimeRangeUint64(uint64(from.UnixNano()/1e6), uint64(to.UnixNano()/1e6))
  79. }
  80. // TimeRangeUint64 is used as a parameter for request creation.
  81. // Adds TimeRange constraint to a request.
  82. // from and to should be in milliseconds
  83. // // It will get values in range [from, to[ ('to' is exclusive).
  84. func TimeRangeUint64(from, to uint64) func(Call) error {
  85. return func(hc Call) error {
  86. if c, ok := hc.(hasQueryOptions); ok {
  87. if from >= to {
  88. // or equal is becuase 'to' is exclusive
  89. return errors.New("'from' timestamp is greater or equal to 'to' timestamp")
  90. }
  91. c.setTimeRangeUint64(from, to)
  92. return nil
  93. }
  94. return errors.New("'TimeRange' option can only be used with Get or Scan request")
  95. }
  96. }
  97. // MaxVersions is used as a parameter for request creation.
  98. // Adds MaxVersions constraint to a request.
  99. func MaxVersions(versions uint32) func(Call) error {
  100. return func(hc Call) error {
  101. if c, ok := hc.(hasQueryOptions); ok {
  102. if versions > math.MaxInt32 {
  103. return errors.New("'MaxVersions' exceeds supported number of versions")
  104. }
  105. c.setMaxVersions(versions)
  106. return nil
  107. }
  108. return errors.New("'MaxVersions' option can only be used with Get or Scan request")
  109. }
  110. }
  111. // MaxResultsPerColumnFamily is an option for Get or Scan requests that sets the maximum
  112. // number of cells returned per column family in a row.
  113. func MaxResultsPerColumnFamily(maxresults uint32) func(Call) error {
  114. return func(hc Call) error {
  115. if c, ok := hc.(hasQueryOptions); ok {
  116. if maxresults > math.MaxInt32 {
  117. return errors.New(
  118. "'MaxResultsPerColumnFamily' exceeds supported number of value results")
  119. }
  120. c.setMaxResultsPerColumnFamily(maxresults)
  121. return nil
  122. }
  123. return errors.New(
  124. "'MaxResultsPerColumnFamily' option can only be used with Get or Scan request")
  125. }
  126. }
  127. // ResultOffset is a option for Scan or Get requests that sets the offset for cells
  128. // within a column family.
  129. func ResultOffset(offset uint32) func(Call) error {
  130. return func(hc Call) error {
  131. if c, ok := hc.(hasQueryOptions); ok {
  132. if offset > math.MaxInt32 {
  133. return errors.New("'ResultOffset' exceeds supported offset value")
  134. }
  135. c.setResultOffset(offset)
  136. return nil
  137. }
  138. return errors.New("'ResultOffset' option can only be used with Get or Scan request")
  139. }
  140. }