activecube 0.1.27 → 0.1.32

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: cfca472f67e11b5fd57a351c506d1c5addfe242357a16baf7af7aef0993b87cf
4
- data.tar.gz: b3f6c6681a821767b64970f7d94ad838a8a0f5066686d8d3a3a265e16c146df3
3
+ metadata.gz: bafdfeabff735168eac514c7a3fa3d80c70e74fcb43372d9110eec9dd58d2eb8
4
+ data.tar.gz: 45f11a99a45eafaf84a127b2e5328ae77515d835122c000669c66fe06a0777a1
5
5
  SHA512:
6
- metadata.gz: ddbe93e4099b0f5cacabb7ca3686206d7378bf5f39e5a40575cbfb14684999ea4b1f7a05083772c62d7244332c154c7c424ef4927b478fc260a50d8f1fee6674
7
- data.tar.gz: 10222838ad5a21399fb811ac244982d2e3c5b9dde553675804da1f78503f4ae64a89571d39c2f7bf8a968e253a330bc0b9b9ae3f8e7429434c8f75db01ee03b1
6
+ metadata.gz: b2faf2c565024c87ddd2045eb644ded72158ebeeec8ce200cd2937d428961a36a8f61c1d3ab83a433ab423a3270ec4af58d5b36f8d17bb99a2d7af2f0505c91a
7
+ data.tar.gz: 27da45c79d812fabb3d59a53d2739db1ae7dde7bd6377be3ca00a7733e0ed169719416b49c9bfefc9a6b3c572a5f4ec3d9c8518f8b78a3e9b05f99f015d8eba8
data/Gemfile.lock CHANGED
@@ -9,7 +9,7 @@ GIT
9
9
  PATH
10
10
  remote: .
11
11
  specs:
12
- activecube (0.1.26)
12
+ activecube (0.1.31)
13
13
  activerecord (>= 5.2)
14
14
 
15
15
  GEM
data/README.md CHANGED
@@ -121,6 +121,12 @@ index ['currency_id'], cardinality: 4
121
121
  index ['currency_id','date'], cardinality: 6
122
122
  ```
123
123
 
124
+ You can require using index in some cases. If required: true added, the table will be used **only** in case when this field is used
125
+ in query metric, dimension or selector.
126
+ ```ruby
127
+ index ['currency_id'], cardinality: 4, required: true
128
+ ```
129
+
124
130
  ### Query language
125
131
 
126
132
  You use the cube class to create and execute queries.
@@ -31,13 +31,14 @@ module Activecube
31
31
 
32
32
  include DefinitionMethods
33
33
 
34
- attr_reader :identity, :fields
34
+ attr_reader :identity, :identity_expression, :fields
35
35
 
36
36
  private
37
37
 
38
38
  def identity_column *args
39
39
  raise "Identity already defined as #{identity} for #{self.name}" if @identity
40
40
  @identity = args.first
41
+ @identity_expression = args.second
41
42
  end
42
43
 
43
44
  def field *args
@@ -52,7 +53,7 @@ module Activecube
52
53
 
53
54
  include DefinitionMethods
54
55
 
55
- attr_reader :modifiers
56
+ attr_reader :modifiers, :tuple
56
57
 
57
58
  private
58
59
 
@@ -60,6 +61,9 @@ module Activecube
60
61
  (@modifiers ||= {} )[args.first.to_sym] = Modifier.new( *args)
61
62
  end
62
63
 
64
+ def tuple_fields *args
65
+ @tuple = args
66
+ end
63
67
  end
64
68
 
65
69
  end
@@ -2,16 +2,20 @@ module Activecube
2
2
  module Processor
3
3
  class Index
4
4
 
5
- attr_reader :fields, :cardinality
5
+ attr_reader :fields, :cardinality, :required
6
6
  def initialize name, *args
7
7
  @fields = [name].flatten
8
8
  @cardinality = args.first && args.first[:cardinality]
9
+ @required = args.first && args.first[:required]
9
10
  end
10
11
 
11
12
  def indexes? query, measures
12
13
  (fields - query.selector_column_names(measures)).empty?
13
14
  end
14
15
 
16
+ def matches? query, measures
17
+ !required || (fields - query.column_names(measures)).empty?
18
+ end
15
19
  end
16
20
  end
17
21
  end
@@ -12,7 +12,8 @@ module Activecube::Processor
12
12
  end
13
13
 
14
14
  def matches? query, measures = query.measures
15
- (query.column_names(measures)-model.attribute_types.keys).empty?
15
+ (query.column_names(measures)-model.attribute_types.keys).empty? &&
16
+ !model.activecube_indexes.detect{|index| !index.matches?(query, measures) }
16
17
  end
17
18
 
18
19
  def measures? measure
@@ -1,6 +1,7 @@
1
1
  require 'activecube/query/chain_appender'
2
2
  require 'activecube/query/item'
3
3
  require 'activecube/query/limit'
4
+ require 'activecube/query/limit_by'
4
5
  require 'activecube/query/measure'
5
6
  require 'activecube/query/ordering'
6
7
  require 'activecube/query/option'
@@ -82,6 +83,12 @@ module Activecube::Query
82
83
  self
83
84
  end
84
85
 
86
+ def limit_by *args
87
+ clear_sql
88
+ options << LimitBy.new(args)
89
+ self
90
+ end
91
+
85
92
 
86
93
  def query
87
94
  sql = to_query.to_sql
@@ -22,5 +22,36 @@ module Activecube::Query
22
22
  "#{definition.class.name}(#{key})"
23
23
  end
24
24
 
25
+ def append_with! model, cube_query, table, query
26
+
27
+ if definition.respond_to?(:with_expression) &&
28
+ (with_expression = definition.with_expression(model, cube_query, table, query))
29
+ with_expression.each_pair do |key, expr|
30
+ query = try_append_with(query, key, expr)
31
+ end
32
+ end
33
+ query
34
+ end
35
+
36
+ private
37
+
38
+
39
+
40
+ def try_append_with(query, key, expr)
41
+ expr = Arel.sql(expr) if expr.kind_of?(String)
42
+ if (with = query.ast.with)
43
+ existing = with.expr.detect{|expr| expr.right==key }
44
+ if existing
45
+ raise "Key #{key} defined twice in WITH statement, with different expressions #{expr.to_sql} AND #{existing.left}" if existing.left!=expr.to_s
46
+ query
47
+ else
48
+ query.with(with.expr + [expr.as(key)])
49
+ end
50
+ else
51
+ query.with(expr.as(key))
52
+ end
53
+
54
+ end
55
+
25
56
  end
26
57
  end
@@ -0,0 +1,19 @@
1
+ module Activecube
2
+ module Query
3
+ class LimitBy
4
+
5
+ attr_reader :each, :limit, :offset
6
+ def initialize arguments
7
+ map = Hash[arguments]
8
+ @each = map[:each]
9
+ @limit = map[:limit]
10
+ @offset = map[:offset] || 0
11
+ end
12
+
13
+ def append_query _model, _cube_query, _table, query
14
+ query.limit_by each, limit, offset
15
+ end
16
+
17
+ end
18
+ end
19
+ end
@@ -36,6 +36,7 @@ module Activecube::Query
36
36
  end
37
37
 
38
38
  def append_query model, cube_query, table, query
39
+ query = append_with!(model, cube_query, table, query)
39
40
  attr_alias = "`#{key.to_s}`"
40
41
  expr = definition.expression model, table, self, cube_query
41
42
  query.project expr.as(attr_alias)
@@ -53,6 +54,7 @@ module Activecube::Query
53
54
 
54
55
  private
55
56
 
57
+
56
58
  def modifier_methods!
57
59
  definition.class.modifiers.each_pair do |key, modifier|
58
60
  define_singleton_method key do |*args|
@@ -32,6 +32,11 @@ module Activecube::Query
32
32
  end
33
33
 
34
34
  def append_query model, cube_query, arel_table, query
35
+
36
+ @selectors.each do |s|
37
+ query = s.append_with!(model, cube_query, arel_table, query)
38
+ end
39
+
35
40
  query.where expression(model, arel_table, cube_query)
36
41
  end
37
42
 
@@ -110,6 +115,7 @@ module Activecube::Query
110
115
  end
111
116
 
112
117
  def append_query model, cube_query, table, query
118
+ query = append_with!(model, cube_query, table, query)
113
119
  query.where(expression model, table, cube_query)
114
120
  end
115
121
 
@@ -60,6 +60,8 @@ module Activecube::Query
60
60
 
61
61
  def append_query model, cube_query, table, query
62
62
 
63
+ query = append_with!(model, cube_query, table, query)
64
+
63
65
  attr_alias = "`#{key.to_s}`"
64
66
  expr = (parent || definition.respond_to?(:expression)) ?
65
67
  Arel.sql(definition.expression( model, table, self, cube_query) ) :
@@ -68,11 +70,20 @@ module Activecube::Query
68
70
  query = query.project(expr.as(attr_alias))
69
71
 
70
72
  if dimension.class.identity
73
+ expr = dimension.class.identity_expression
71
74
  group_by_columns.each do |column|
72
- if !query.projections.detect{|p| p.kind_of?(::Arel::Attributes::Attribute) && p.name==column}
73
- query = query.project(table[column])
75
+
76
+ node = if column==dimension.class.identity && expr
77
+ Arel.sql(expr).as(column)
78
+ else
79
+ table[column]
80
+ end
81
+
82
+ unless query.projections.include?(node)
83
+ query = query.project(node)
74
84
  end
75
- query = query.group(table[column])
85
+
86
+ query = query.group( expr ? column : table[column])
76
87
  end
77
88
  else
78
89
  query = query.group(attr_alias)
@@ -1,3 +1,3 @@
1
1
  module Activecube
2
- VERSION = "0.1.27"
2
+ VERSION = "0.1.32"
3
3
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: activecube
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.27
4
+ version: 0.1.32
5
5
  platform: ruby
6
6
  authors:
7
7
  - Aleksey Studnev
8
8
  autorequire:
9
9
  bindir: exe
10
10
  cert_chain: []
11
- date: 2020-08-01 00:00:00.000000000 Z
11
+ date: 2021-03-19 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: activerecord
@@ -108,6 +108,7 @@ files:
108
108
  - lib/activecube/query/cube_query.rb
109
109
  - lib/activecube/query/item.rb
110
110
  - lib/activecube/query/limit.rb
111
+ - lib/activecube/query/limit_by.rb
111
112
  - lib/activecube/query/measure.rb
112
113
  - lib/activecube/query/measure_nothing.rb
113
114
  - lib/activecube/query/modification.rb