/vendor/github.com/influxdata/influxdb/coordinator/shard_mapper.go

https://bitbucket.org/egym-com/dns-tools · Go · 255 lines · 203 code · 33 blank · 19 comment · 55 complexity · 4b2231c8f561dbc159e9a6c257e05c04 MD5 · raw file

  1. package coordinator
  2. import (
  3. "context"
  4. "io"
  5. "time"
  6. "github.com/influxdata/influxdb/query"
  7. "github.com/influxdata/influxdb/services/meta"
  8. "github.com/influxdata/influxdb/tsdb"
  9. "github.com/influxdata/influxql"
  10. )
  11. // IteratorCreator is an interface that combines mapping fields and creating iterators.
  12. type IteratorCreator interface {
  13. query.IteratorCreator
  14. influxql.FieldMapper
  15. io.Closer
  16. }
  17. // LocalShardMapper implements a ShardMapper for local shards.
  18. type LocalShardMapper struct {
  19. MetaClient interface {
  20. ShardGroupsByTimeRange(database, policy string, min, max time.Time) (a []meta.ShardGroupInfo, err error)
  21. }
  22. TSDBStore interface {
  23. ShardGroup(ids []uint64) tsdb.ShardGroup
  24. }
  25. }
  26. // MapShards maps the sources to the appropriate shards into an IteratorCreator.
  27. func (e *LocalShardMapper) MapShards(sources influxql.Sources, t influxql.TimeRange, opt query.SelectOptions) (query.ShardGroup, error) {
  28. a := &LocalShardMapping{
  29. ShardMap: make(map[Source]tsdb.ShardGroup),
  30. }
  31. tmin := time.Unix(0, t.MinTimeNano())
  32. tmax := time.Unix(0, t.MaxTimeNano())
  33. if err := e.mapShards(a, sources, tmin, tmax); err != nil {
  34. return nil, err
  35. }
  36. a.MinTime, a.MaxTime = tmin, tmax
  37. return a, nil
  38. }
  39. func (e *LocalShardMapper) mapShards(a *LocalShardMapping, sources influxql.Sources, tmin, tmax time.Time) error {
  40. for _, s := range sources {
  41. switch s := s.(type) {
  42. case *influxql.Measurement:
  43. source := Source{
  44. Database: s.Database,
  45. RetentionPolicy: s.RetentionPolicy,
  46. }
  47. // Retrieve the list of shards for this database. This list of
  48. // shards is always the same regardless of which measurement we are
  49. // using.
  50. if _, ok := a.ShardMap[source]; !ok {
  51. groups, err := e.MetaClient.ShardGroupsByTimeRange(s.Database, s.RetentionPolicy, tmin, tmax)
  52. if err != nil {
  53. return err
  54. }
  55. if len(groups) == 0 {
  56. a.ShardMap[source] = nil
  57. continue
  58. }
  59. shardIDs := make([]uint64, 0, len(groups[0].Shards)*len(groups))
  60. for _, g := range groups {
  61. for _, si := range g.Shards {
  62. shardIDs = append(shardIDs, si.ID)
  63. }
  64. }
  65. a.ShardMap[source] = e.TSDBStore.ShardGroup(shardIDs)
  66. }
  67. case *influxql.SubQuery:
  68. if err := e.mapShards(a, s.Statement.Sources, tmin, tmax); err != nil {
  69. return err
  70. }
  71. }
  72. }
  73. return nil
  74. }
  75. // ShardMapper maps data sources to a list of shard information.
  76. type LocalShardMapping struct {
  77. ShardMap map[Source]tsdb.ShardGroup
  78. // MinTime is the minimum time that this shard mapper will allow.
  79. // Any attempt to use a time before this one will automatically result in using
  80. // this time instead.
  81. MinTime time.Time
  82. // MaxTime is the maximum time that this shard mapper will allow.
  83. // Any attempt to use a time after this one will automatically result in using
  84. // this time instead.
  85. MaxTime time.Time
  86. }
  87. func (a *LocalShardMapping) FieldDimensions(m *influxql.Measurement) (fields map[string]influxql.DataType, dimensions map[string]struct{}, err error) {
  88. source := Source{
  89. Database: m.Database,
  90. RetentionPolicy: m.RetentionPolicy,
  91. }
  92. sg := a.ShardMap[source]
  93. if sg == nil {
  94. return
  95. }
  96. fields = make(map[string]influxql.DataType)
  97. dimensions = make(map[string]struct{})
  98. var measurements []string
  99. if m.Regex != nil {
  100. measurements = sg.MeasurementsByRegex(m.Regex.Val)
  101. } else {
  102. measurements = []string{m.Name}
  103. }
  104. f, d, err := sg.FieldDimensions(measurements)
  105. if err != nil {
  106. return nil, nil, err
  107. }
  108. for k, typ := range f {
  109. fields[k] = typ
  110. }
  111. for k := range d {
  112. dimensions[k] = struct{}{}
  113. }
  114. return
  115. }
  116. func (a *LocalShardMapping) MapType(m *influxql.Measurement, field string) influxql.DataType {
  117. source := Source{
  118. Database: m.Database,
  119. RetentionPolicy: m.RetentionPolicy,
  120. }
  121. sg := a.ShardMap[source]
  122. if sg == nil {
  123. return influxql.Unknown
  124. }
  125. var names []string
  126. if m.Regex != nil {
  127. names = sg.MeasurementsByRegex(m.Regex.Val)
  128. } else {
  129. names = []string{m.Name}
  130. }
  131. var typ influxql.DataType
  132. for _, name := range names {
  133. if m.SystemIterator != "" {
  134. name = m.SystemIterator
  135. }
  136. t := sg.MapType(name, field)
  137. if typ.LessThan(t) {
  138. typ = t
  139. }
  140. }
  141. return typ
  142. }
  143. func (a *LocalShardMapping) CreateIterator(ctx context.Context, m *influxql.Measurement, opt query.IteratorOptions) (query.Iterator, error) {
  144. source := Source{
  145. Database: m.Database,
  146. RetentionPolicy: m.RetentionPolicy,
  147. }
  148. sg := a.ShardMap[source]
  149. if sg == nil {
  150. return nil, nil
  151. }
  152. // Override the time constraints if they don't match each other.
  153. if !a.MinTime.IsZero() && opt.StartTime < a.MinTime.UnixNano() {
  154. opt.StartTime = a.MinTime.UnixNano()
  155. }
  156. if !a.MaxTime.IsZero() && opt.EndTime > a.MaxTime.UnixNano() {
  157. opt.EndTime = a.MaxTime.UnixNano()
  158. }
  159. if m.Regex != nil {
  160. measurements := sg.MeasurementsByRegex(m.Regex.Val)
  161. inputs := make([]query.Iterator, 0, len(measurements))
  162. if err := func() error {
  163. // Create a Measurement for each returned matching measurement value
  164. // from the regex.
  165. for _, measurement := range measurements {
  166. mm := m.Clone()
  167. mm.Name = measurement // Set the name to this matching regex value.
  168. input, err := sg.CreateIterator(ctx, mm, opt)
  169. if err != nil {
  170. return err
  171. }
  172. inputs = append(inputs, input)
  173. }
  174. return nil
  175. }(); err != nil {
  176. query.Iterators(inputs).Close()
  177. return nil, err
  178. }
  179. return query.Iterators(inputs).Merge(opt)
  180. }
  181. return sg.CreateIterator(ctx, m, opt)
  182. }
  183. func (a *LocalShardMapping) IteratorCost(m *influxql.Measurement, opt query.IteratorOptions) (query.IteratorCost, error) {
  184. source := Source{
  185. Database: m.Database,
  186. RetentionPolicy: m.RetentionPolicy,
  187. }
  188. sg := a.ShardMap[source]
  189. if sg == nil {
  190. return query.IteratorCost{}, nil
  191. }
  192. // Override the time constraints if they don't match each other.
  193. if !a.MinTime.IsZero() && opt.StartTime < a.MinTime.UnixNano() {
  194. opt.StartTime = a.MinTime.UnixNano()
  195. }
  196. if !a.MaxTime.IsZero() && opt.EndTime > a.MaxTime.UnixNano() {
  197. opt.EndTime = a.MaxTime.UnixNano()
  198. }
  199. if m.Regex != nil {
  200. var costs query.IteratorCost
  201. measurements := sg.MeasurementsByRegex(m.Regex.Val)
  202. for _, measurement := range measurements {
  203. cost, err := sg.IteratorCost(measurement, opt)
  204. if err != nil {
  205. return query.IteratorCost{}, err
  206. }
  207. costs = costs.Combine(cost)
  208. }
  209. return costs, nil
  210. }
  211. return sg.IteratorCost(m.Name, opt)
  212. }
  213. // Close clears out the list of mapped shards.
  214. func (a *LocalShardMapping) Close() error {
  215. a.ShardMap = nil
  216. return nil
  217. }
  218. // Source contains the database and retention policy source for data.
  219. type Source struct {
  220. Database string
  221. RetentionPolicy string
  222. }