activerecord 5.2.6 → 6.0.0.beta1

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of activerecord might be problematic. Click here for more details.

Files changed (241) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +299 -758
  3. data/MIT-LICENSE +3 -1
  4. data/README.rdoc +1 -1
  5. data/examples/performance.rb +1 -1
  6. data/lib/active_record.rb +2 -1
  7. data/lib/active_record/aggregations.rb +4 -2
  8. data/lib/active_record/associations.rb +16 -12
  9. data/lib/active_record/associations/association.rb +35 -19
  10. data/lib/active_record/associations/association_scope.rb +4 -6
  11. data/lib/active_record/associations/belongs_to_association.rb +36 -42
  12. data/lib/active_record/associations/belongs_to_polymorphic_association.rb +0 -4
  13. data/lib/active_record/associations/builder/belongs_to.rb +14 -50
  14. data/lib/active_record/associations/builder/collection_association.rb +3 -3
  15. data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +17 -38
  16. data/lib/active_record/associations/collection_association.rb +11 -25
  17. data/lib/active_record/associations/collection_proxy.rb +32 -6
  18. data/lib/active_record/associations/foreign_association.rb +7 -0
  19. data/lib/active_record/associations/has_many_association.rb +1 -1
  20. data/lib/active_record/associations/has_many_through_association.rb +25 -18
  21. data/lib/active_record/associations/has_one_association.rb +28 -30
  22. data/lib/active_record/associations/has_one_through_association.rb +5 -5
  23. data/lib/active_record/associations/join_dependency.rb +15 -20
  24. data/lib/active_record/associations/join_dependency/join_association.rb +11 -26
  25. data/lib/active_record/associations/join_dependency/join_part.rb +2 -2
  26. data/lib/active_record/associations/preloader.rb +32 -29
  27. data/lib/active_record/associations/preloader/association.rb +1 -2
  28. data/lib/active_record/associations/singular_association.rb +2 -16
  29. data/lib/active_record/attribute_assignment.rb +7 -10
  30. data/lib/active_record/attribute_methods.rb +34 -56
  31. data/lib/active_record/attribute_methods/dirty.rb +64 -26
  32. data/lib/active_record/attribute_methods/primary_key.rb +8 -7
  33. data/lib/active_record/attribute_methods/read.rb +16 -48
  34. data/lib/active_record/attribute_methods/serialization.rb +1 -1
  35. data/lib/active_record/attribute_methods/time_zone_conversion.rb +1 -1
  36. data/lib/active_record/attribute_methods/write.rb +15 -16
  37. data/lib/active_record/autosave_association.rb +7 -21
  38. data/lib/active_record/base.rb +2 -2
  39. data/lib/active_record/callbacks.rb +3 -17
  40. data/lib/active_record/collection_cache_key.rb +1 -1
  41. data/lib/active_record/connection_adapters/abstract/connection_pool.rb +13 -36
  42. data/lib/active_record/connection_adapters/abstract/database_limits.rb +9 -0
  43. data/lib/active_record/connection_adapters/abstract/database_statements.rb +25 -84
  44. data/lib/active_record/connection_adapters/abstract/query_cache.rb +17 -14
  45. data/lib/active_record/connection_adapters/abstract/quoting.rb +5 -11
  46. data/lib/active_record/connection_adapters/abstract/schema_creation.rb +15 -11
  47. data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +30 -13
  48. data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +0 -2
  49. data/lib/active_record/connection_adapters/abstract/schema_statements.rb +41 -27
  50. data/lib/active_record/connection_adapters/abstract/transaction.rb +81 -52
  51. data/lib/active_record/connection_adapters/abstract_adapter.rb +95 -31
  52. data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +65 -90
  53. data/lib/active_record/connection_adapters/connection_specification.rb +52 -42
  54. data/lib/active_record/connection_adapters/determine_if_preparable_visitor.rb +5 -9
  55. data/lib/active_record/connection_adapters/mysql/database_statements.rb +29 -7
  56. data/lib/active_record/connection_adapters/mysql/quoting.rb +1 -1
  57. data/lib/active_record/connection_adapters/mysql/schema_creation.rb +3 -4
  58. data/lib/active_record/connection_adapters/mysql/schema_statements.rb +65 -10
  59. data/lib/active_record/connection_adapters/mysql2_adapter.rb +8 -4
  60. data/lib/active_record/connection_adapters/postgresql/column.rb +1 -2
  61. data/lib/active_record/connection_adapters/postgresql/database_statements.rb +16 -1
  62. data/lib/active_record/connection_adapters/postgresql/oid/array.rb +1 -1
  63. data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +1 -4
  64. data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +1 -1
  65. data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +1 -1
  66. data/lib/active_record/connection_adapters/postgresql/oid/money.rb +2 -2
  67. data/lib/active_record/connection_adapters/postgresql/oid/point.rb +1 -1
  68. data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +9 -7
  69. data/lib/active_record/connection_adapters/postgresql/quoting.rb +4 -4
  70. data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +11 -36
  71. data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +9 -2
  72. data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +38 -20
  73. data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +2 -1
  74. data/lib/active_record/connection_adapters/postgresql_adapter.rb +75 -56
  75. data/lib/active_record/connection_adapters/schema_cache.rb +5 -0
  76. data/lib/active_record/connection_adapters/sqlite3/quoting.rb +5 -5
  77. data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +14 -9
  78. data/lib/active_record/connection_adapters/sqlite3_adapter.rb +95 -62
  79. data/lib/active_record/connection_handling.rb +132 -26
  80. data/lib/active_record/core.rb +76 -43
  81. data/lib/active_record/counter_cache.rb +4 -29
  82. data/lib/active_record/database_configurations.rb +184 -0
  83. data/lib/active_record/database_configurations/database_config.rb +37 -0
  84. data/lib/active_record/database_configurations/hash_config.rb +50 -0
  85. data/lib/active_record/database_configurations/url_config.rb +74 -0
  86. data/lib/active_record/enum.rb +22 -7
  87. data/lib/active_record/errors.rb +24 -21
  88. data/lib/active_record/explain.rb +1 -1
  89. data/lib/active_record/fixture_set/model_metadata.rb +33 -0
  90. data/lib/active_record/fixture_set/render_context.rb +17 -0
  91. data/lib/active_record/fixture_set/table_row.rb +153 -0
  92. data/lib/active_record/fixture_set/table_rows.rb +47 -0
  93. data/lib/active_record/fixtures.rb +140 -472
  94. data/lib/active_record/gem_version.rb +4 -4
  95. data/lib/active_record/inheritance.rb +12 -2
  96. data/lib/active_record/integration.rb +56 -16
  97. data/lib/active_record/internal_metadata.rb +5 -1
  98. data/lib/active_record/locking/optimistic.rb +2 -2
  99. data/lib/active_record/locking/pessimistic.rb +3 -3
  100. data/lib/active_record/log_subscriber.rb +7 -26
  101. data/lib/active_record/migration.rb +38 -37
  102. data/lib/active_record/migration/command_recorder.rb +35 -5
  103. data/lib/active_record/migration/compatibility.rb +34 -16
  104. data/lib/active_record/model_schema.rb +30 -9
  105. data/lib/active_record/nested_attributes.rb +2 -2
  106. data/lib/active_record/no_touching.rb +7 -0
  107. data/lib/active_record/persistence.rb +18 -7
  108. data/lib/active_record/query_cache.rb +11 -4
  109. data/lib/active_record/querying.rb +19 -11
  110. data/lib/active_record/railtie.rb +71 -42
  111. data/lib/active_record/railties/collection_cache_association_loading.rb +34 -0
  112. data/lib/active_record/railties/controller_runtime.rb +30 -35
  113. data/lib/active_record/railties/databases.rake +94 -43
  114. data/lib/active_record/reflection.rb +60 -44
  115. data/lib/active_record/relation.rb +150 -69
  116. data/lib/active_record/relation/batches.rb +13 -10
  117. data/lib/active_record/relation/calculations.rb +38 -28
  118. data/lib/active_record/relation/delegation.rb +4 -13
  119. data/lib/active_record/relation/finder_methods.rb +12 -25
  120. data/lib/active_record/relation/merger.rb +2 -6
  121. data/lib/active_record/relation/predicate_builder.rb +4 -6
  122. data/lib/active_record/relation/predicate_builder/array_handler.rb +5 -4
  123. data/lib/active_record/relation/predicate_builder/association_query_value.rb +1 -4
  124. data/lib/active_record/relation/predicate_builder/base_handler.rb +1 -2
  125. data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +1 -2
  126. data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +1 -4
  127. data/lib/active_record/relation/predicate_builder/range_handler.rb +3 -23
  128. data/lib/active_record/relation/query_attribute.rb +15 -12
  129. data/lib/active_record/relation/query_methods.rb +29 -52
  130. data/lib/active_record/relation/where_clause.rb +4 -0
  131. data/lib/active_record/relation/where_clause_factory.rb +1 -2
  132. data/lib/active_record/result.rb +30 -11
  133. data/lib/active_record/sanitization.rb +2 -39
  134. data/lib/active_record/schema.rb +1 -10
  135. data/lib/active_record/schema_dumper.rb +12 -6
  136. data/lib/active_record/schema_migration.rb +4 -0
  137. data/lib/active_record/scoping.rb +9 -8
  138. data/lib/active_record/scoping/default.rb +10 -3
  139. data/lib/active_record/scoping/named.rb +10 -14
  140. data/lib/active_record/statement_cache.rb +32 -5
  141. data/lib/active_record/store.rb +39 -8
  142. data/lib/active_record/table_metadata.rb +1 -4
  143. data/lib/active_record/tasks/database_tasks.rb +89 -23
  144. data/lib/active_record/tasks/mysql_database_tasks.rb +2 -4
  145. data/lib/active_record/tasks/postgresql_database_tasks.rb +5 -7
  146. data/lib/active_record/tasks/sqlite_database_tasks.rb +2 -8
  147. data/lib/active_record/test_databases.rb +38 -0
  148. data/lib/active_record/test_fixtures.rb +224 -0
  149. data/lib/active_record/timestamp.rb +4 -6
  150. data/lib/active_record/transactions.rb +3 -22
  151. data/lib/active_record/translation.rb +1 -1
  152. data/lib/active_record/type.rb +3 -4
  153. data/lib/active_record/type/adapter_specific_registry.rb +1 -8
  154. data/lib/active_record/type_caster/connection.rb +1 -6
  155. data/lib/active_record/type_caster/map.rb +1 -4
  156. data/lib/active_record/validations/uniqueness.rb +13 -25
  157. data/lib/arel.rb +44 -0
  158. data/lib/arel/alias_predication.rb +9 -0
  159. data/lib/arel/attributes.rb +22 -0
  160. data/lib/arel/attributes/attribute.rb +37 -0
  161. data/lib/arel/collectors/bind.rb +24 -0
  162. data/lib/arel/collectors/composite.rb +31 -0
  163. data/lib/arel/collectors/plain_string.rb +20 -0
  164. data/lib/arel/collectors/sql_string.rb +20 -0
  165. data/lib/arel/collectors/substitute_binds.rb +28 -0
  166. data/lib/arel/crud.rb +42 -0
  167. data/lib/arel/delete_manager.rb +18 -0
  168. data/lib/arel/errors.rb +9 -0
  169. data/lib/arel/expressions.rb +29 -0
  170. data/lib/arel/factory_methods.rb +49 -0
  171. data/lib/arel/insert_manager.rb +49 -0
  172. data/lib/arel/math.rb +45 -0
  173. data/lib/arel/nodes.rb +67 -0
  174. data/lib/arel/nodes/and.rb +32 -0
  175. data/lib/arel/nodes/ascending.rb +23 -0
  176. data/lib/arel/nodes/binary.rb +52 -0
  177. data/lib/arel/nodes/bind_param.rb +36 -0
  178. data/lib/arel/nodes/case.rb +55 -0
  179. data/lib/arel/nodes/casted.rb +50 -0
  180. data/lib/arel/nodes/count.rb +12 -0
  181. data/lib/arel/nodes/delete_statement.rb +45 -0
  182. data/lib/arel/nodes/descending.rb +23 -0
  183. data/lib/arel/nodes/equality.rb +18 -0
  184. data/lib/arel/nodes/extract.rb +24 -0
  185. data/lib/arel/nodes/false.rb +16 -0
  186. data/lib/arel/nodes/full_outer_join.rb +8 -0
  187. data/lib/arel/nodes/function.rb +44 -0
  188. data/lib/arel/nodes/grouping.rb +8 -0
  189. data/lib/arel/nodes/in.rb +8 -0
  190. data/lib/arel/nodes/infix_operation.rb +80 -0
  191. data/lib/arel/nodes/inner_join.rb +8 -0
  192. data/lib/arel/nodes/insert_statement.rb +37 -0
  193. data/lib/arel/nodes/join_source.rb +20 -0
  194. data/lib/arel/nodes/matches.rb +18 -0
  195. data/lib/arel/nodes/named_function.rb +23 -0
  196. data/lib/arel/nodes/node.rb +50 -0
  197. data/lib/arel/nodes/node_expression.rb +13 -0
  198. data/lib/arel/nodes/outer_join.rb +8 -0
  199. data/lib/arel/nodes/over.rb +15 -0
  200. data/lib/arel/nodes/regexp.rb +16 -0
  201. data/lib/arel/nodes/right_outer_join.rb +8 -0
  202. data/lib/arel/nodes/select_core.rb +63 -0
  203. data/lib/arel/nodes/select_statement.rb +41 -0
  204. data/lib/arel/nodes/sql_literal.rb +16 -0
  205. data/lib/arel/nodes/string_join.rb +11 -0
  206. data/lib/arel/nodes/table_alias.rb +27 -0
  207. data/lib/arel/nodes/terminal.rb +16 -0
  208. data/lib/arel/nodes/true.rb +16 -0
  209. data/lib/arel/nodes/unary.rb +44 -0
  210. data/lib/arel/nodes/unary_operation.rb +20 -0
  211. data/lib/arel/nodes/unqualified_column.rb +22 -0
  212. data/lib/arel/nodes/update_statement.rb +41 -0
  213. data/lib/arel/nodes/values.rb +16 -0
  214. data/lib/arel/nodes/values_list.rb +24 -0
  215. data/lib/arel/nodes/window.rb +126 -0
  216. data/lib/arel/nodes/with.rb +11 -0
  217. data/lib/arel/order_predications.rb +13 -0
  218. data/lib/arel/predications.rb +257 -0
  219. data/lib/arel/select_manager.rb +271 -0
  220. data/lib/arel/table.rb +110 -0
  221. data/lib/arel/tree_manager.rb +72 -0
  222. data/lib/arel/update_manager.rb +34 -0
  223. data/lib/arel/visitors.rb +20 -0
  224. data/lib/arel/visitors/depth_first.rb +199 -0
  225. data/lib/arel/visitors/dot.rb +292 -0
  226. data/lib/arel/visitors/ibm_db.rb +21 -0
  227. data/lib/arel/visitors/informix.rb +56 -0
  228. data/lib/arel/visitors/mssql.rb +143 -0
  229. data/lib/arel/visitors/mysql.rb +83 -0
  230. data/lib/arel/visitors/oracle.rb +159 -0
  231. data/lib/arel/visitors/oracle12.rb +67 -0
  232. data/lib/arel/visitors/postgresql.rb +116 -0
  233. data/lib/arel/visitors/sqlite.rb +39 -0
  234. data/lib/arel/visitors/to_sql.rb +913 -0
  235. data/lib/arel/visitors/visitor.rb +42 -0
  236. data/lib/arel/visitors/where_sql.rb +23 -0
  237. data/lib/arel/window_predications.rb +9 -0
  238. data/lib/rails/generators/active_record/migration.rb +14 -1
  239. data/lib/rails/generators/active_record/migration/migration_generator.rb +2 -5
  240. data/lib/rails/generators/active_record/model/model_generator.rb +1 -0
  241. metadata +104 -26
@@ -0,0 +1,126 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Arel # :nodoc: all
4
+ module Nodes
5
+ class Window < Arel::Nodes::Node
6
+ attr_accessor :orders, :framing, :partitions
7
+
8
+ def initialize
9
+ @orders = []
10
+ @partitions = []
11
+ @framing = nil
12
+ end
13
+
14
+ def order(*expr)
15
+ # FIXME: We SHOULD NOT be converting these to SqlLiteral automatically
16
+ @orders.concat expr.map { |x|
17
+ String === x || Symbol === x ? Nodes::SqlLiteral.new(x.to_s) : x
18
+ }
19
+ self
20
+ end
21
+
22
+ def partition(*expr)
23
+ # FIXME: We SHOULD NOT be converting these to SqlLiteral automatically
24
+ @partitions.concat expr.map { |x|
25
+ String === x || Symbol === x ? Nodes::SqlLiteral.new(x.to_s) : x
26
+ }
27
+ self
28
+ end
29
+
30
+ def frame(expr)
31
+ @framing = expr
32
+ end
33
+
34
+ def rows(expr = nil)
35
+ if @framing
36
+ Rows.new(expr)
37
+ else
38
+ frame(Rows.new(expr))
39
+ end
40
+ end
41
+
42
+ def range(expr = nil)
43
+ if @framing
44
+ Range.new(expr)
45
+ else
46
+ frame(Range.new(expr))
47
+ end
48
+ end
49
+
50
+ def initialize_copy(other)
51
+ super
52
+ @orders = @orders.map { |x| x.clone }
53
+ end
54
+
55
+ def hash
56
+ [@orders, @framing].hash
57
+ end
58
+
59
+ def eql?(other)
60
+ self.class == other.class &&
61
+ self.orders == other.orders &&
62
+ self.framing == other.framing &&
63
+ self.partitions == other.partitions
64
+ end
65
+ alias :== :eql?
66
+ end
67
+
68
+ class NamedWindow < Window
69
+ attr_accessor :name
70
+
71
+ def initialize(name)
72
+ super()
73
+ @name = name
74
+ end
75
+
76
+ def initialize_copy(other)
77
+ super
78
+ @name = other.name.clone
79
+ end
80
+
81
+ def hash
82
+ super ^ @name.hash
83
+ end
84
+
85
+ def eql?(other)
86
+ super && self.name == other.name
87
+ end
88
+ alias :== :eql?
89
+ end
90
+
91
+ class Rows < Unary
92
+ def initialize(expr = nil)
93
+ super(expr)
94
+ end
95
+ end
96
+
97
+ class Range < Unary
98
+ def initialize(expr = nil)
99
+ super(expr)
100
+ end
101
+ end
102
+
103
+ class CurrentRow < Node
104
+ def hash
105
+ self.class.hash
106
+ end
107
+
108
+ def eql?(other)
109
+ self.class == other.class
110
+ end
111
+ alias :== :eql?
112
+ end
113
+
114
+ class Preceding < Unary
115
+ def initialize(expr = nil)
116
+ super(expr)
117
+ end
118
+ end
119
+
120
+ class Following < Unary
121
+ def initialize(expr = nil)
122
+ super(expr)
123
+ end
124
+ end
125
+ end
126
+ end
@@ -0,0 +1,11 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Arel # :nodoc: all
4
+ module Nodes
5
+ class With < Arel::Nodes::Unary
6
+ alias children expr
7
+ end
8
+
9
+ class WithRecursive < With; end
10
+ end
11
+ end
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Arel # :nodoc: all
4
+ module OrderPredications
5
+ def asc
6
+ Nodes::Ascending.new self
7
+ end
8
+
9
+ def desc
10
+ Nodes::Descending.new self
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,257 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Arel # :nodoc: all
4
+ module Predications
5
+ def not_eq(other)
6
+ Nodes::NotEqual.new self, quoted_node(other)
7
+ end
8
+
9
+ def not_eq_any(others)
10
+ grouping_any :not_eq, others
11
+ end
12
+
13
+ def not_eq_all(others)
14
+ grouping_all :not_eq, others
15
+ end
16
+
17
+ def eq(other)
18
+ Nodes::Equality.new self, quoted_node(other)
19
+ end
20
+
21
+ def is_not_distinct_from(other)
22
+ Nodes::IsNotDistinctFrom.new self, quoted_node(other)
23
+ end
24
+
25
+ def is_distinct_from(other)
26
+ Nodes::IsDistinctFrom.new self, quoted_node(other)
27
+ end
28
+
29
+ def eq_any(others)
30
+ grouping_any :eq, others
31
+ end
32
+
33
+ def eq_all(others)
34
+ grouping_all :eq, quoted_array(others)
35
+ end
36
+
37
+ def between(other)
38
+ if unboundable?(other.begin) == 1 || unboundable?(other.end) == -1
39
+ self.in([])
40
+ elsif open_ended?(other.begin)
41
+ if other.end.nil? || open_ended?(other.end)
42
+ not_in([])
43
+ elsif other.exclude_end?
44
+ lt(other.end)
45
+ else
46
+ lteq(other.end)
47
+ end
48
+ elsif other.end.nil? || open_ended?(other.end)
49
+ gteq(other.begin)
50
+ elsif other.exclude_end?
51
+ gteq(other.begin).and(lt(other.end))
52
+ else
53
+ left = quoted_node(other.begin)
54
+ right = quoted_node(other.end)
55
+ Nodes::Between.new(self, left.and(right))
56
+ end
57
+ end
58
+
59
+ def in(other)
60
+ case other
61
+ when Arel::SelectManager
62
+ Arel::Nodes::In.new(self, other.ast)
63
+ when Range
64
+ if $VERBOSE
65
+ warn <<-eowarn
66
+ Passing a range to `#in` is deprecated. Call `#between`, instead.
67
+ eowarn
68
+ end
69
+ between(other)
70
+ when Enumerable
71
+ Nodes::In.new self, quoted_array(other)
72
+ else
73
+ Nodes::In.new self, quoted_node(other)
74
+ end
75
+ end
76
+
77
+ def in_any(others)
78
+ grouping_any :in, others
79
+ end
80
+
81
+ def in_all(others)
82
+ grouping_all :in, others
83
+ end
84
+
85
+ def not_between(other)
86
+ if unboundable?(other.begin) == 1 || unboundable?(other.end) == -1
87
+ not_in([])
88
+ elsif open_ended?(other.begin)
89
+ if other.end.nil? || open_ended?(other.end)
90
+ self.in([])
91
+ elsif other.exclude_end?
92
+ gteq(other.end)
93
+ else
94
+ gt(other.end)
95
+ end
96
+ elsif other.end.nil? || open_ended?(other.end)
97
+ lt(other.begin)
98
+ else
99
+ left = lt(other.begin)
100
+ right = if other.exclude_end?
101
+ gteq(other.end)
102
+ else
103
+ gt(other.end)
104
+ end
105
+ left.or(right)
106
+ end
107
+ end
108
+
109
+ def not_in(other)
110
+ case other
111
+ when Arel::SelectManager
112
+ Arel::Nodes::NotIn.new(self, other.ast)
113
+ when Range
114
+ if $VERBOSE
115
+ warn <<-eowarn
116
+ Passing a range to `#not_in` is deprecated. Call `#not_between`, instead.
117
+ eowarn
118
+ end
119
+ not_between(other)
120
+ when Enumerable
121
+ Nodes::NotIn.new self, quoted_array(other)
122
+ else
123
+ Nodes::NotIn.new self, quoted_node(other)
124
+ end
125
+ end
126
+
127
+ def not_in_any(others)
128
+ grouping_any :not_in, others
129
+ end
130
+
131
+ def not_in_all(others)
132
+ grouping_all :not_in, others
133
+ end
134
+
135
+ def matches(other, escape = nil, case_sensitive = false)
136
+ Nodes::Matches.new self, quoted_node(other), escape, case_sensitive
137
+ end
138
+
139
+ def matches_regexp(other, case_sensitive = true)
140
+ Nodes::Regexp.new self, quoted_node(other), case_sensitive
141
+ end
142
+
143
+ def matches_any(others, escape = nil, case_sensitive = false)
144
+ grouping_any :matches, others, escape, case_sensitive
145
+ end
146
+
147
+ def matches_all(others, escape = nil, case_sensitive = false)
148
+ grouping_all :matches, others, escape, case_sensitive
149
+ end
150
+
151
+ def does_not_match(other, escape = nil, case_sensitive = false)
152
+ Nodes::DoesNotMatch.new self, quoted_node(other), escape, case_sensitive
153
+ end
154
+
155
+ def does_not_match_regexp(other, case_sensitive = true)
156
+ Nodes::NotRegexp.new self, quoted_node(other), case_sensitive
157
+ end
158
+
159
+ def does_not_match_any(others, escape = nil)
160
+ grouping_any :does_not_match, others, escape
161
+ end
162
+
163
+ def does_not_match_all(others, escape = nil)
164
+ grouping_all :does_not_match, others, escape
165
+ end
166
+
167
+ def gteq(right)
168
+ Nodes::GreaterThanOrEqual.new self, quoted_node(right)
169
+ end
170
+
171
+ def gteq_any(others)
172
+ grouping_any :gteq, others
173
+ end
174
+
175
+ def gteq_all(others)
176
+ grouping_all :gteq, others
177
+ end
178
+
179
+ def gt(right)
180
+ Nodes::GreaterThan.new self, quoted_node(right)
181
+ end
182
+
183
+ def gt_any(others)
184
+ grouping_any :gt, others
185
+ end
186
+
187
+ def gt_all(others)
188
+ grouping_all :gt, others
189
+ end
190
+
191
+ def lt(right)
192
+ Nodes::LessThan.new self, quoted_node(right)
193
+ end
194
+
195
+ def lt_any(others)
196
+ grouping_any :lt, others
197
+ end
198
+
199
+ def lt_all(others)
200
+ grouping_all :lt, others
201
+ end
202
+
203
+ def lteq(right)
204
+ Nodes::LessThanOrEqual.new self, quoted_node(right)
205
+ end
206
+
207
+ def lteq_any(others)
208
+ grouping_any :lteq, others
209
+ end
210
+
211
+ def lteq_all(others)
212
+ grouping_all :lteq, others
213
+ end
214
+
215
+ def when(right)
216
+ Nodes::Case.new(self).when quoted_node(right)
217
+ end
218
+
219
+ def concat(other)
220
+ Nodes::Concat.new self, other
221
+ end
222
+
223
+ private
224
+
225
+ def grouping_any(method_id, others, *extras)
226
+ nodes = others.map { |expr| send(method_id, expr, *extras) }
227
+ Nodes::Grouping.new nodes.inject { |memo, node|
228
+ Nodes::Or.new(memo, node)
229
+ }
230
+ end
231
+
232
+ def grouping_all(method_id, others, *extras)
233
+ nodes = others.map { |expr| send(method_id, expr, *extras) }
234
+ Nodes::Grouping.new Nodes::And.new(nodes)
235
+ end
236
+
237
+ def quoted_node(other)
238
+ Nodes.build_quoted(other, self)
239
+ end
240
+
241
+ def quoted_array(others)
242
+ others.map { |v| quoted_node(v) }
243
+ end
244
+
245
+ def infinity?(value)
246
+ value.respond_to?(:infinite?) && value.infinite?
247
+ end
248
+
249
+ def unboundable?(value)
250
+ value.respond_to?(:unboundable?) && value.unboundable?
251
+ end
252
+
253
+ def open_ended?(value)
254
+ infinity?(value) || unboundable?(value)
255
+ end
256
+ end
257
+ end
@@ -0,0 +1,271 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Arel # :nodoc: all
4
+ class SelectManager < Arel::TreeManager
5
+ include Arel::Crud
6
+
7
+ STRING_OR_SYMBOL_CLASS = [Symbol, String]
8
+
9
+ def initialize(table = nil)
10
+ super()
11
+ @ast = Nodes::SelectStatement.new
12
+ @ctx = @ast.cores.last
13
+ from table
14
+ end
15
+
16
+ def initialize_copy(other)
17
+ super
18
+ @ctx = @ast.cores.last
19
+ end
20
+
21
+ def limit
22
+ @ast.limit && @ast.limit.expr
23
+ end
24
+ alias :taken :limit
25
+
26
+ def constraints
27
+ @ctx.wheres
28
+ end
29
+
30
+ def offset
31
+ @ast.offset && @ast.offset.expr
32
+ end
33
+
34
+ def skip(amount)
35
+ if amount
36
+ @ast.offset = Nodes::Offset.new(amount)
37
+ else
38
+ @ast.offset = nil
39
+ end
40
+ self
41
+ end
42
+ alias :offset= :skip
43
+
44
+ ###
45
+ # Produces an Arel::Nodes::Exists node
46
+ def exists
47
+ Arel::Nodes::Exists.new @ast
48
+ end
49
+
50
+ def as(other)
51
+ create_table_alias grouping(@ast), Nodes::SqlLiteral.new(other)
52
+ end
53
+
54
+ def lock(locking = Arel.sql("FOR UPDATE"))
55
+ case locking
56
+ when true
57
+ locking = Arel.sql("FOR UPDATE")
58
+ when Arel::Nodes::SqlLiteral
59
+ when String
60
+ locking = Arel.sql locking
61
+ end
62
+
63
+ @ast.lock = Nodes::Lock.new(locking)
64
+ self
65
+ end
66
+
67
+ def locked
68
+ @ast.lock
69
+ end
70
+
71
+ def on(*exprs)
72
+ @ctx.source.right.last.right = Nodes::On.new(collapse(exprs))
73
+ self
74
+ end
75
+
76
+ def group(*columns)
77
+ columns.each do |column|
78
+ # FIXME: backwards compat
79
+ column = Nodes::SqlLiteral.new(column) if String === column
80
+ column = Nodes::SqlLiteral.new(column.to_s) if Symbol === column
81
+
82
+ @ctx.groups.push Nodes::Group.new column
83
+ end
84
+ self
85
+ end
86
+
87
+ def from(table)
88
+ table = Nodes::SqlLiteral.new(table) if String === table
89
+
90
+ case table
91
+ when Nodes::Join
92
+ @ctx.source.right << table
93
+ else
94
+ @ctx.source.left = table
95
+ end
96
+
97
+ self
98
+ end
99
+
100
+ def froms
101
+ @ast.cores.map { |x| x.from }.compact
102
+ end
103
+
104
+ def join(relation, klass = Nodes::InnerJoin)
105
+ return self unless relation
106
+
107
+ case relation
108
+ when String, Nodes::SqlLiteral
109
+ raise EmptyJoinError if relation.empty?
110
+ klass = Nodes::StringJoin
111
+ end
112
+
113
+ @ctx.source.right << create_join(relation, nil, klass)
114
+ self
115
+ end
116
+
117
+ def outer_join(relation)
118
+ join(relation, Nodes::OuterJoin)
119
+ end
120
+
121
+ def having(expr)
122
+ @ctx.havings << expr
123
+ self
124
+ end
125
+
126
+ def window(name)
127
+ window = Nodes::NamedWindow.new(name)
128
+ @ctx.windows.push window
129
+ window
130
+ end
131
+
132
+ def project(*projections)
133
+ # FIXME: converting these to SQLLiterals is probably not good, but
134
+ # rails tests require it.
135
+ @ctx.projections.concat projections.map { |x|
136
+ STRING_OR_SYMBOL_CLASS.include?(x.class) ? Nodes::SqlLiteral.new(x.to_s) : x
137
+ }
138
+ self
139
+ end
140
+
141
+ def projections
142
+ @ctx.projections
143
+ end
144
+
145
+ def projections=(projections)
146
+ @ctx.projections = projections
147
+ end
148
+
149
+ def distinct(value = true)
150
+ if value
151
+ @ctx.set_quantifier = Arel::Nodes::Distinct.new
152
+ else
153
+ @ctx.set_quantifier = nil
154
+ end
155
+ self
156
+ end
157
+
158
+ def distinct_on(value)
159
+ if value
160
+ @ctx.set_quantifier = Arel::Nodes::DistinctOn.new(value)
161
+ else
162
+ @ctx.set_quantifier = nil
163
+ end
164
+ self
165
+ end
166
+
167
+ def order(*expr)
168
+ # FIXME: We SHOULD NOT be converting these to SqlLiteral automatically
169
+ @ast.orders.concat expr.map { |x|
170
+ STRING_OR_SYMBOL_CLASS.include?(x.class) ? Nodes::SqlLiteral.new(x.to_s) : x
171
+ }
172
+ self
173
+ end
174
+
175
+ def orders
176
+ @ast.orders
177
+ end
178
+
179
+ def where_sql(engine = Table.engine)
180
+ return if @ctx.wheres.empty?
181
+
182
+ viz = Visitors::WhereSql.new(engine.connection.visitor, engine.connection)
183
+ Nodes::SqlLiteral.new viz.accept(@ctx, Collectors::SQLString.new).value
184
+ end
185
+
186
+ def union(operation, other = nil)
187
+ if other
188
+ node_class = Nodes.const_get("Union#{operation.to_s.capitalize}")
189
+ else
190
+ other = operation
191
+ node_class = Nodes::Union
192
+ end
193
+
194
+ node_class.new self.ast, other.ast
195
+ end
196
+
197
+ def intersect(other)
198
+ Nodes::Intersect.new ast, other.ast
199
+ end
200
+
201
+ def except(other)
202
+ Nodes::Except.new ast, other.ast
203
+ end
204
+ alias :minus :except
205
+
206
+ def lateral(table_name = nil)
207
+ base = table_name.nil? ? ast : as(table_name)
208
+ Nodes::Lateral.new(base)
209
+ end
210
+
211
+ def with(*subqueries)
212
+ if subqueries.first.is_a? Symbol
213
+ node_class = Nodes.const_get("With#{subqueries.shift.to_s.capitalize}")
214
+ else
215
+ node_class = Nodes::With
216
+ end
217
+ @ast.with = node_class.new(subqueries.flatten)
218
+
219
+ self
220
+ end
221
+
222
+ def take(limit)
223
+ if limit
224
+ @ast.limit = Nodes::Limit.new(limit)
225
+ else
226
+ @ast.limit = nil
227
+ end
228
+ self
229
+ end
230
+ alias limit= take
231
+
232
+ def join_sources
233
+ @ctx.source.right
234
+ end
235
+
236
+ def source
237
+ @ctx.source
238
+ end
239
+
240
+ class Row < Struct.new(:data) # :nodoc:
241
+ def id
242
+ data["id"]
243
+ end
244
+
245
+ def method_missing(name, *args)
246
+ name = name.to_s
247
+ return data[name] if data.key?(name)
248
+ super
249
+ end
250
+ end
251
+
252
+ private
253
+ def collapse(exprs)
254
+ exprs = exprs.compact
255
+ exprs.map! { |expr|
256
+ if String === expr
257
+ # FIXME: Don't do this automatically
258
+ Arel.sql(expr)
259
+ else
260
+ expr
261
+ end
262
+ }
263
+
264
+ if exprs.length == 1
265
+ exprs.first
266
+ else
267
+ create_and exprs
268
+ end
269
+ end
270
+ end
271
+ end