activerecord 3.2.22.5 → 5.2.8
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.
- checksums.yaml +5 -5
- data/CHANGELOG.md +657 -621
- data/MIT-LICENSE +2 -2
- data/README.rdoc +41 -46
- data/examples/performance.rb +55 -42
- data/examples/simple.rb +6 -5
- data/lib/active_record/aggregations.rb +264 -236
- data/lib/active_record/association_relation.rb +40 -0
- data/lib/active_record/associations/alias_tracker.rb +47 -42
- data/lib/active_record/associations/association.rb +127 -75
- data/lib/active_record/associations/association_scope.rb +126 -92
- data/lib/active_record/associations/belongs_to_association.rb +78 -27
- data/lib/active_record/associations/belongs_to_polymorphic_association.rb +9 -4
- data/lib/active_record/associations/builder/association.rb +117 -32
- data/lib/active_record/associations/builder/belongs_to.rb +135 -60
- data/lib/active_record/associations/builder/collection_association.rb +61 -54
- data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +120 -42
- data/lib/active_record/associations/builder/has_many.rb +10 -64
- data/lib/active_record/associations/builder/has_one.rb +19 -51
- data/lib/active_record/associations/builder/singular_association.rb +28 -18
- data/lib/active_record/associations/collection_association.rb +226 -293
- data/lib/active_record/associations/collection_proxy.rb +1067 -69
- data/lib/active_record/associations/foreign_association.rb +13 -0
- data/lib/active_record/associations/has_many_association.rb +83 -47
- data/lib/active_record/associations/has_many_through_association.rb +98 -65
- data/lib/active_record/associations/has_one_association.rb +57 -20
- data/lib/active_record/associations/has_one_through_association.rb +18 -9
- data/lib/active_record/associations/join_dependency/join_association.rb +48 -126
- data/lib/active_record/associations/join_dependency/join_base.rb +11 -12
- data/lib/active_record/associations/join_dependency/join_part.rb +35 -42
- data/lib/active_record/associations/join_dependency.rb +212 -164
- data/lib/active_record/associations/preloader/association.rb +95 -89
- data/lib/active_record/associations/preloader/through_association.rb +84 -44
- data/lib/active_record/associations/preloader.rb +123 -111
- data/lib/active_record/associations/singular_association.rb +33 -24
- data/lib/active_record/associations/through_association.rb +60 -26
- data/lib/active_record/associations.rb +1759 -1506
- data/lib/active_record/attribute_assignment.rb +60 -193
- data/lib/active_record/attribute_decorators.rb +90 -0
- data/lib/active_record/attribute_methods/before_type_cast.rb +55 -8
- data/lib/active_record/attribute_methods/dirty.rb +113 -74
- data/lib/active_record/attribute_methods/primary_key.rb +106 -77
- data/lib/active_record/attribute_methods/query.rb +8 -5
- data/lib/active_record/attribute_methods/read.rb +63 -114
- data/lib/active_record/attribute_methods/serialization.rb +60 -90
- data/lib/active_record/attribute_methods/time_zone_conversion.rb +69 -43
- data/lib/active_record/attribute_methods/write.rb +43 -45
- data/lib/active_record/attribute_methods.rb +366 -149
- data/lib/active_record/attributes.rb +266 -0
- data/lib/active_record/autosave_association.rb +312 -225
- data/lib/active_record/base.rb +114 -505
- data/lib/active_record/callbacks.rb +145 -67
- data/lib/active_record/coders/json.rb +15 -0
- data/lib/active_record/coders/yaml_column.rb +32 -23
- data/lib/active_record/collection_cache_key.rb +53 -0
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +883 -284
- data/lib/active_record/connection_adapters/abstract/database_limits.rb +16 -2
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +350 -200
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +82 -27
- data/lib/active_record/connection_adapters/abstract/quoting.rb +150 -65
- data/lib/active_record/connection_adapters/abstract/savepoints.rb +23 -0
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +146 -0
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +477 -284
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +95 -0
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +1100 -310
- data/lib/active_record/connection_adapters/abstract/transaction.rb +283 -0
- data/lib/active_record/connection_adapters/abstract_adapter.rb +450 -118
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +657 -446
- data/lib/active_record/connection_adapters/column.rb +50 -255
- data/lib/active_record/connection_adapters/connection_specification.rb +287 -0
- data/lib/active_record/connection_adapters/determine_if_preparable_visitor.rb +33 -0
- data/lib/active_record/connection_adapters/mysql/column.rb +27 -0
- data/lib/active_record/connection_adapters/mysql/database_statements.rb +140 -0
- data/lib/active_record/connection_adapters/mysql/explain_pretty_printer.rb +72 -0
- data/lib/active_record/connection_adapters/mysql/quoting.rb +44 -0
- data/lib/active_record/connection_adapters/mysql/schema_creation.rb +73 -0
- data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +87 -0
- data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +80 -0
- data/lib/active_record/connection_adapters/mysql/schema_statements.rb +148 -0
- data/lib/active_record/connection_adapters/mysql/type_metadata.rb +35 -0
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +59 -210
- data/lib/active_record/connection_adapters/postgresql/column.rb +44 -0
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +163 -0
- data/lib/active_record/connection_adapters/postgresql/explain_pretty_printer.rb +44 -0
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +92 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +56 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bit_varying.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/bytea.rb +17 -0
- data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +50 -0
- data/lib/active_record/connection_adapters/postgresql/oid/date.rb +23 -0
- data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +23 -0
- data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +21 -0
- data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +71 -0
- data/lib/active_record/connection_adapters/postgresql/oid/inet.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +45 -0
- data/lib/active_record/connection_adapters/postgresql/oid/money.rb +41 -0
- data/lib/active_record/connection_adapters/postgresql/oid/oid.rb +15 -0
- data/lib/active_record/connection_adapters/postgresql/oid/point.rb +65 -0
- data/lib/active_record/connection_adapters/postgresql/oid/range.rb +97 -0
- data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +18 -0
- data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +111 -0
- data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +23 -0
- data/lib/active_record/connection_adapters/postgresql/oid/vector.rb +28 -0
- data/lib/active_record/connection_adapters/postgresql/oid/xml.rb +30 -0
- data/lib/active_record/connection_adapters/postgresql/oid.rb +34 -0
- data/lib/active_record/connection_adapters/postgresql/quoting.rb +168 -0
- data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +43 -0
- data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +65 -0
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +206 -0
- data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +50 -0
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +774 -0
- data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +39 -0
- data/lib/active_record/connection_adapters/postgresql/utils.rb +81 -0
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +620 -1080
- data/lib/active_record/connection_adapters/schema_cache.rb +85 -36
- data/lib/active_record/connection_adapters/sql_type_metadata.rb +34 -0
- data/lib/active_record/connection_adapters/sqlite3/explain_pretty_printer.rb +21 -0
- data/lib/active_record/connection_adapters/sqlite3/quoting.rb +67 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +17 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_definitions.rb +19 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_dumper.rb +18 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +106 -0
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +545 -27
- data/lib/active_record/connection_adapters/statement_pool.rb +34 -13
- data/lib/active_record/connection_handling.rb +145 -0
- data/lib/active_record/core.rb +559 -0
- data/lib/active_record/counter_cache.rb +200 -105
- data/lib/active_record/define_callbacks.rb +22 -0
- data/lib/active_record/dynamic_matchers.rb +107 -69
- data/lib/active_record/enum.rb +244 -0
- data/lib/active_record/errors.rb +245 -60
- data/lib/active_record/explain.rb +35 -71
- data/lib/active_record/explain_registry.rb +32 -0
- data/lib/active_record/explain_subscriber.rb +18 -9
- data/lib/active_record/fixture_set/file.rb +82 -0
- data/lib/active_record/fixtures.rb +418 -275
- data/lib/active_record/gem_version.rb +17 -0
- data/lib/active_record/inheritance.rb +209 -100
- data/lib/active_record/integration.rb +116 -21
- data/lib/active_record/internal_metadata.rb +45 -0
- data/lib/active_record/legacy_yaml_adapter.rb +48 -0
- data/lib/active_record/locale/en.yml +9 -1
- data/lib/active_record/locking/optimistic.rb +107 -94
- data/lib/active_record/locking/pessimistic.rb +20 -8
- data/lib/active_record/log_subscriber.rb +99 -34
- data/lib/active_record/migration/command_recorder.rb +199 -64
- data/lib/active_record/migration/compatibility.rb +217 -0
- data/lib/active_record/migration/join_table.rb +17 -0
- data/lib/active_record/migration.rb +893 -296
- data/lib/active_record/model_schema.rb +328 -175
- data/lib/active_record/nested_attributes.rb +338 -242
- data/lib/active_record/no_touching.rb +58 -0
- data/lib/active_record/null_relation.rb +68 -0
- data/lib/active_record/persistence.rb +557 -170
- data/lib/active_record/query_cache.rb +14 -43
- data/lib/active_record/querying.rb +36 -24
- data/lib/active_record/railtie.rb +147 -52
- data/lib/active_record/railties/console_sandbox.rb +5 -4
- data/lib/active_record/railties/controller_runtime.rb +13 -6
- data/lib/active_record/railties/databases.rake +206 -488
- data/lib/active_record/readonly_attributes.rb +4 -6
- data/lib/active_record/reflection.rb +734 -228
- data/lib/active_record/relation/batches/batch_enumerator.rb +69 -0
- data/lib/active_record/relation/batches.rb +249 -52
- data/lib/active_record/relation/calculations.rb +330 -284
- data/lib/active_record/relation/delegation.rb +135 -37
- data/lib/active_record/relation/finder_methods.rb +450 -287
- data/lib/active_record/relation/from_clause.rb +26 -0
- data/lib/active_record/relation/merger.rb +193 -0
- data/lib/active_record/relation/predicate_builder/array_handler.rb +48 -0
- data/lib/active_record/relation/predicate_builder/association_query_value.rb +46 -0
- data/lib/active_record/relation/predicate_builder/base_handler.rb +19 -0
- data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +20 -0
- data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +56 -0
- data/lib/active_record/relation/predicate_builder/range_handler.rb +42 -0
- data/lib/active_record/relation/predicate_builder/relation_handler.rb +19 -0
- data/lib/active_record/relation/predicate_builder.rb +132 -43
- data/lib/active_record/relation/query_attribute.rb +45 -0
- data/lib/active_record/relation/query_methods.rb +1037 -221
- data/lib/active_record/relation/record_fetch_warning.rb +51 -0
- data/lib/active_record/relation/spawn_methods.rb +48 -151
- data/lib/active_record/relation/where_clause.rb +186 -0
- data/lib/active_record/relation/where_clause_factory.rb +34 -0
- data/lib/active_record/relation.rb +451 -359
- data/lib/active_record/result.rb +129 -20
- data/lib/active_record/runtime_registry.rb +24 -0
- data/lib/active_record/sanitization.rb +164 -136
- data/lib/active_record/schema.rb +31 -19
- data/lib/active_record/schema_dumper.rb +154 -107
- data/lib/active_record/schema_migration.rb +56 -0
- data/lib/active_record/scoping/default.rb +108 -98
- data/lib/active_record/scoping/named.rb +125 -112
- data/lib/active_record/scoping.rb +77 -123
- data/lib/active_record/secure_token.rb +40 -0
- data/lib/active_record/serialization.rb +10 -6
- data/lib/active_record/statement_cache.rb +121 -0
- data/lib/active_record/store.rb +175 -16
- data/lib/active_record/suppressor.rb +61 -0
- data/lib/active_record/table_metadata.rb +82 -0
- data/lib/active_record/tasks/database_tasks.rb +337 -0
- data/lib/active_record/tasks/mysql_database_tasks.rb +115 -0
- data/lib/active_record/tasks/postgresql_database_tasks.rb +143 -0
- data/lib/active_record/tasks/sqlite_database_tasks.rb +83 -0
- data/lib/active_record/timestamp.rb +80 -41
- data/lib/active_record/touch_later.rb +64 -0
- data/lib/active_record/transactions.rb +240 -119
- data/lib/active_record/translation.rb +2 -0
- data/lib/active_record/type/adapter_specific_registry.rb +136 -0
- data/lib/active_record/type/date.rb +9 -0
- data/lib/active_record/type/date_time.rb +9 -0
- data/lib/active_record/type/decimal_without_scale.rb +15 -0
- data/lib/active_record/type/hash_lookup_type_map.rb +25 -0
- data/lib/active_record/type/internal/timezone.rb +17 -0
- data/lib/active_record/type/json.rb +30 -0
- data/lib/active_record/type/serialized.rb +71 -0
- data/lib/active_record/type/text.rb +11 -0
- data/lib/active_record/type/time.rb +21 -0
- data/lib/active_record/type/type_map.rb +62 -0
- data/lib/active_record/type/unsigned_integer.rb +17 -0
- data/lib/active_record/type.rb +79 -0
- data/lib/active_record/type_caster/connection.rb +33 -0
- data/lib/active_record/type_caster/map.rb +23 -0
- data/lib/active_record/type_caster.rb +9 -0
- data/lib/active_record/validations/absence.rb +25 -0
- data/lib/active_record/validations/associated.rb +35 -18
- data/lib/active_record/validations/length.rb +26 -0
- data/lib/active_record/validations/presence.rb +68 -0
- data/lib/active_record/validations/uniqueness.rb +133 -75
- data/lib/active_record/validations.rb +53 -43
- data/lib/active_record/version.rb +7 -7
- data/lib/active_record.rb +89 -57
- data/lib/rails/generators/active_record/application_record/application_record_generator.rb +27 -0
- data/lib/rails/generators/active_record/application_record/templates/application_record.rb.tt +5 -0
- data/lib/rails/generators/active_record/migration/migration_generator.rb +61 -8
- data/lib/rails/generators/active_record/migration/templates/create_table_migration.rb.tt +24 -0
- data/lib/rails/generators/active_record/migration/templates/migration.rb.tt +46 -0
- data/lib/rails/generators/active_record/migration.rb +28 -8
- data/lib/rails/generators/active_record/model/model_generator.rb +23 -22
- data/lib/rails/generators/active_record/model/templates/model.rb.tt +13 -0
- data/lib/rails/generators/active_record/model/templates/{module.rb → module.rb.tt} +1 -1
- data/lib/rails/generators/active_record.rb +10 -16
- metadata +141 -62
- data/examples/associations.png +0 -0
- data/lib/active_record/associations/has_and_belongs_to_many_association.rb +0 -63
- data/lib/active_record/associations/join_helper.rb +0 -55
- data/lib/active_record/associations/preloader/belongs_to.rb +0 -17
- data/lib/active_record/associations/preloader/collection_association.rb +0 -24
- data/lib/active_record/associations/preloader/has_and_belongs_to_many.rb +0 -60
- data/lib/active_record/associations/preloader/has_many.rb +0 -17
- data/lib/active_record/associations/preloader/has_many_through.rb +0 -15
- data/lib/active_record/associations/preloader/has_one.rb +0 -23
- data/lib/active_record/associations/preloader/has_one_through.rb +0 -9
- data/lib/active_record/associations/preloader/singular_association.rb +0 -21
- data/lib/active_record/attribute_methods/deprecated_underscore_read.rb +0 -32
- data/lib/active_record/connection_adapters/abstract/connection_specification.rb +0 -191
- data/lib/active_record/connection_adapters/mysql_adapter.rb +0 -441
- data/lib/active_record/connection_adapters/sqlite_adapter.rb +0 -583
- data/lib/active_record/dynamic_finder_match.rb +0 -68
- data/lib/active_record/dynamic_scope_match.rb +0 -23
- data/lib/active_record/fixtures/file.rb +0 -65
- data/lib/active_record/identity_map.rb +0 -162
- data/lib/active_record/observer.rb +0 -121
- data/lib/active_record/railties/jdbcmysql_error.rb +0 -16
- data/lib/active_record/serializers/xml_serializer.rb +0 -203
- data/lib/active_record/session_store.rb +0 -360
- data/lib/active_record/test_case.rb +0 -73
- data/lib/rails/generators/active_record/migration/templates/migration.rb +0 -34
- data/lib/rails/generators/active_record/model/templates/migration.rb +0 -15
- data/lib/rails/generators/active_record/model/templates/model.rb +0 -12
- data/lib/rails/generators/active_record/observer/observer_generator.rb +0 -15
- data/lib/rails/generators/active_record/observer/templates/observer.rb +0 -4
- data/lib/rails/generators/active_record/session_migration/session_migration_generator.rb +0 -25
- data/lib/rails/generators/active_record/session_migration/templates/migration.rb +0 -12
@@ -0,0 +1,121 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActiveRecord
|
4
|
+
# Statement cache is used to cache a single statement in order to avoid creating the AST again.
|
5
|
+
# Initializing the cache is done by passing the statement in the create block:
|
6
|
+
#
|
7
|
+
# cache = StatementCache.create(Book.connection) do |params|
|
8
|
+
# Book.where(name: "my book").where("author_id > 3")
|
9
|
+
# end
|
10
|
+
#
|
11
|
+
# The cached statement is executed by using the
|
12
|
+
# {connection.execute}[rdoc-ref:ConnectionAdapters::DatabaseStatements#execute] method:
|
13
|
+
#
|
14
|
+
# cache.execute([], Book.connection)
|
15
|
+
#
|
16
|
+
# The relation returned by the block is cached, and for each
|
17
|
+
# {execute}[rdoc-ref:ConnectionAdapters::DatabaseStatements#execute]
|
18
|
+
# call the cached relation gets duped. Database is queried when +to_a+ is called on the relation.
|
19
|
+
#
|
20
|
+
# If you want to cache the statement without the values you can use the +bind+ method of the
|
21
|
+
# block parameter.
|
22
|
+
#
|
23
|
+
# cache = StatementCache.create(Book.connection) do |params|
|
24
|
+
# Book.where(name: params.bind)
|
25
|
+
# end
|
26
|
+
#
|
27
|
+
# And pass the bind values as the first argument of +execute+ call.
|
28
|
+
#
|
29
|
+
# cache.execute(["my book"], Book.connection)
|
30
|
+
class StatementCache # :nodoc:
|
31
|
+
class Substitute; end # :nodoc:
|
32
|
+
|
33
|
+
class Query # :nodoc:
|
34
|
+
def initialize(sql)
|
35
|
+
@sql = sql
|
36
|
+
end
|
37
|
+
|
38
|
+
def sql_for(binds, connection)
|
39
|
+
@sql
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
class PartialQuery < Query # :nodoc:
|
44
|
+
def initialize(values)
|
45
|
+
@values = values
|
46
|
+
@indexes = values.each_with_index.find_all { |thing, i|
|
47
|
+
Arel::Nodes::BindParam === thing
|
48
|
+
}.map(&:last)
|
49
|
+
end
|
50
|
+
|
51
|
+
def sql_for(binds, connection)
|
52
|
+
val = @values.dup
|
53
|
+
casted_binds = binds.map(&:value_for_database)
|
54
|
+
@indexes.each { |i| val[i] = connection.quote(casted_binds.shift) }
|
55
|
+
val.join
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
def self.query(sql)
|
60
|
+
Query.new(sql)
|
61
|
+
end
|
62
|
+
|
63
|
+
def self.partial_query(values)
|
64
|
+
PartialQuery.new(values)
|
65
|
+
end
|
66
|
+
|
67
|
+
class Params # :nodoc:
|
68
|
+
def bind; Substitute.new; end
|
69
|
+
end
|
70
|
+
|
71
|
+
class BindMap # :nodoc:
|
72
|
+
def initialize(bound_attributes)
|
73
|
+
@indexes = []
|
74
|
+
@bound_attributes = bound_attributes
|
75
|
+
|
76
|
+
bound_attributes.each_with_index do |attr, i|
|
77
|
+
if Substitute === attr.value
|
78
|
+
@indexes << i
|
79
|
+
end
|
80
|
+
end
|
81
|
+
end
|
82
|
+
|
83
|
+
def bind(values)
|
84
|
+
bas = @bound_attributes.dup
|
85
|
+
@indexes.each_with_index { |offset, i| bas[offset] = bas[offset].with_cast_value(values[i]) }
|
86
|
+
bas
|
87
|
+
end
|
88
|
+
end
|
89
|
+
|
90
|
+
def self.create(connection, callable = nil, &block)
|
91
|
+
relation = (callable || block).call Params.new
|
92
|
+
query_builder, binds = connection.cacheable_query(self, relation.arel)
|
93
|
+
bind_map = BindMap.new(binds)
|
94
|
+
new(query_builder, bind_map, relation.klass)
|
95
|
+
end
|
96
|
+
|
97
|
+
def initialize(query_builder, bind_map, klass)
|
98
|
+
@query_builder = query_builder
|
99
|
+
@bind_map = bind_map
|
100
|
+
@klass = klass
|
101
|
+
end
|
102
|
+
|
103
|
+
def execute(params, connection, &block)
|
104
|
+
bind_values = bind_map.bind params
|
105
|
+
|
106
|
+
sql = query_builder.sql_for bind_values, connection
|
107
|
+
|
108
|
+
klass.find_by_sql(sql, bind_values, preparable: true, &block)
|
109
|
+
end
|
110
|
+
|
111
|
+
def self.unsupported_value?(value)
|
112
|
+
case value
|
113
|
+
when NilClass, Array, Range, Hash, Relation, Base then true
|
114
|
+
end
|
115
|
+
end
|
116
|
+
|
117
|
+
protected
|
118
|
+
|
119
|
+
attr_reader :query_builder, :bind_map, :klass
|
120
|
+
end
|
121
|
+
end
|
data/lib/active_record/store.rb
CHANGED
@@ -1,6 +1,10 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "active_support/core_ext/hash/indifferent_access"
|
4
|
+
|
1
5
|
module ActiveRecord
|
2
6
|
# Store gives you a thin wrapper around serialize for the purpose of storing hashes in a single column.
|
3
|
-
# It's like a simple key/value store
|
7
|
+
# It's like a simple key/value store baked into your record when you don't care about being able to
|
4
8
|
# query that store outside the context of a single record.
|
5
9
|
#
|
6
10
|
# You can then declare accessors to this store that are then accessible just like any other attribute
|
@@ -10,43 +14,198 @@ module ActiveRecord
|
|
10
14
|
# Make sure that you declare the database column used for the serialized store as a text, so there's
|
11
15
|
# plenty of room.
|
12
16
|
#
|
17
|
+
# You can set custom coder to encode/decode your serialized attributes to/from different formats.
|
18
|
+
# JSON, YAML, Marshal are supported out of the box. Generally it can be any wrapper that provides +load+ and +dump+.
|
19
|
+
#
|
20
|
+
# NOTE: If you are using PostgreSQL specific columns like +hstore+ or +json+ there is no need for
|
21
|
+
# the serialization provided by {.store}[rdoc-ref:rdoc-ref:ClassMethods#store].
|
22
|
+
# Simply use {.store_accessor}[rdoc-ref:ClassMethods#store_accessor] instead to generate
|
23
|
+
# the accessor methods. Be aware that these columns use a string keyed hash and do not allow access
|
24
|
+
# using a symbol.
|
25
|
+
#
|
26
|
+
# NOTE: The default validations with the exception of +uniqueness+ will work.
|
27
|
+
# For example, if you want to check for +uniqueness+ with +hstore+ you will
|
28
|
+
# need to use a custom validation to handle it.
|
29
|
+
#
|
13
30
|
# Examples:
|
14
31
|
#
|
15
32
|
# class User < ActiveRecord::Base
|
16
|
-
# store :settings, accessors: [ :color, :homepage ]
|
33
|
+
# store :settings, accessors: [ :color, :homepage ], coder: JSON
|
17
34
|
# end
|
18
|
-
#
|
35
|
+
#
|
19
36
|
# u = User.new(color: 'black', homepage: '37signals.com')
|
20
37
|
# u.color # Accessor stored attribute
|
21
38
|
# u.settings[:country] = 'Denmark' # Any attribute, even if not specified with an accessor
|
22
39
|
#
|
40
|
+
# # There is no difference between strings and symbols for accessing custom attributes
|
41
|
+
# u.settings[:country] # => 'Denmark'
|
42
|
+
# u.settings['country'] # => 'Denmark'
|
43
|
+
#
|
23
44
|
# # Add additional accessors to an existing store through store_accessor
|
24
45
|
# class SuperUser < User
|
25
46
|
# store_accessor :settings, :privileges, :servants
|
26
47
|
# end
|
48
|
+
#
|
49
|
+
# The stored attribute names can be retrieved using {.stored_attributes}[rdoc-ref:rdoc-ref:ClassMethods#stored_attributes].
|
50
|
+
#
|
51
|
+
# User.stored_attributes[:settings] # [:color, :homepage]
|
52
|
+
#
|
53
|
+
# == Overwriting default accessors
|
54
|
+
#
|
55
|
+
# All stored values are automatically available through accessors on the Active Record
|
56
|
+
# object, but sometimes you want to specialize this behavior. This can be done by overwriting
|
57
|
+
# the default accessors (using the same name as the attribute) and calling <tt>super</tt>
|
58
|
+
# to actually change things.
|
59
|
+
#
|
60
|
+
# class Song < ActiveRecord::Base
|
61
|
+
# # Uses a stored integer to hold the volume adjustment of the song
|
62
|
+
# store :settings, accessors: [:volume_adjustment]
|
63
|
+
#
|
64
|
+
# def volume_adjustment=(decibels)
|
65
|
+
# super(decibels.to_i)
|
66
|
+
# end
|
67
|
+
#
|
68
|
+
# def volume_adjustment
|
69
|
+
# super.to_i
|
70
|
+
# end
|
71
|
+
# end
|
27
72
|
module Store
|
28
73
|
extend ActiveSupport::Concern
|
29
|
-
|
74
|
+
|
75
|
+
included do
|
76
|
+
class << self
|
77
|
+
attr_accessor :local_stored_attributes
|
78
|
+
end
|
79
|
+
end
|
80
|
+
|
30
81
|
module ClassMethods
|
31
82
|
def store(store_attribute, options = {})
|
32
|
-
serialize store_attribute,
|
83
|
+
serialize store_attribute, IndifferentCoder.new(store_attribute, options[:coder])
|
33
84
|
store_accessor(store_attribute, options[:accessors]) if options.has_key? :accessors
|
34
85
|
end
|
35
86
|
|
36
87
|
def store_accessor(store_attribute, *keys)
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
88
|
+
keys = keys.flatten
|
89
|
+
|
90
|
+
_store_accessors_module.module_eval do
|
91
|
+
keys.each do |key|
|
92
|
+
define_method("#{key}=") do |value|
|
93
|
+
write_store_attribute(store_attribute, key, value)
|
94
|
+
end
|
95
|
+
|
96
|
+
define_method(key) do
|
97
|
+
read_store_attribute(store_attribute, key)
|
98
|
+
end
|
47
99
|
end
|
48
100
|
end
|
101
|
+
|
102
|
+
# assign new store attribute and create new hash to ensure that each class in the hierarchy
|
103
|
+
# has its own hash of stored attributes.
|
104
|
+
self.local_stored_attributes ||= {}
|
105
|
+
self.local_stored_attributes[store_attribute] ||= []
|
106
|
+
self.local_stored_attributes[store_attribute] |= keys
|
107
|
+
end
|
108
|
+
|
109
|
+
def _store_accessors_module # :nodoc:
|
110
|
+
@_store_accessors_module ||= begin
|
111
|
+
mod = Module.new
|
112
|
+
include mod
|
113
|
+
mod
|
114
|
+
end
|
115
|
+
end
|
116
|
+
|
117
|
+
def stored_attributes
|
118
|
+
parent = superclass.respond_to?(:stored_attributes) ? superclass.stored_attributes : {}
|
119
|
+
if local_stored_attributes
|
120
|
+
parent.merge!(local_stored_attributes) { |k, a, b| a | b }
|
121
|
+
end
|
122
|
+
parent
|
49
123
|
end
|
50
124
|
end
|
125
|
+
|
126
|
+
private
|
127
|
+
def read_store_attribute(store_attribute, key) # :doc:
|
128
|
+
accessor = store_accessor_for(store_attribute)
|
129
|
+
accessor.read(self, store_attribute, key)
|
130
|
+
end
|
131
|
+
|
132
|
+
def write_store_attribute(store_attribute, key, value) # :doc:
|
133
|
+
accessor = store_accessor_for(store_attribute)
|
134
|
+
accessor.write(self, store_attribute, key, value)
|
135
|
+
end
|
136
|
+
|
137
|
+
def store_accessor_for(store_attribute)
|
138
|
+
type_for_attribute(store_attribute).accessor
|
139
|
+
end
|
140
|
+
|
141
|
+
class HashAccessor # :nodoc:
|
142
|
+
def self.read(object, attribute, key)
|
143
|
+
prepare(object, attribute)
|
144
|
+
object.public_send(attribute)[key]
|
145
|
+
end
|
146
|
+
|
147
|
+
def self.write(object, attribute, key, value)
|
148
|
+
prepare(object, attribute)
|
149
|
+
if value != read(object, attribute, key)
|
150
|
+
object.public_send :"#{attribute}_will_change!"
|
151
|
+
object.public_send(attribute)[key] = value
|
152
|
+
end
|
153
|
+
end
|
154
|
+
|
155
|
+
def self.prepare(object, attribute)
|
156
|
+
object.public_send :"#{attribute}=", {} unless object.send(attribute)
|
157
|
+
end
|
158
|
+
end
|
159
|
+
|
160
|
+
class StringKeyedHashAccessor < HashAccessor # :nodoc:
|
161
|
+
def self.read(object, attribute, key)
|
162
|
+
super object, attribute, key.to_s
|
163
|
+
end
|
164
|
+
|
165
|
+
def self.write(object, attribute, key, value)
|
166
|
+
super object, attribute, key.to_s, value
|
167
|
+
end
|
168
|
+
end
|
169
|
+
|
170
|
+
class IndifferentHashAccessor < ActiveRecord::Store::HashAccessor # :nodoc:
|
171
|
+
def self.prepare(object, store_attribute)
|
172
|
+
attribute = object.send(store_attribute)
|
173
|
+
unless attribute.is_a?(ActiveSupport::HashWithIndifferentAccess)
|
174
|
+
attribute = IndifferentCoder.as_indifferent_hash(attribute)
|
175
|
+
object.send :"#{store_attribute}=", attribute
|
176
|
+
end
|
177
|
+
attribute
|
178
|
+
end
|
179
|
+
end
|
180
|
+
|
181
|
+
class IndifferentCoder # :nodoc:
|
182
|
+
def initialize(attr_name, coder_or_class_name)
|
183
|
+
@coder =
|
184
|
+
if coder_or_class_name.respond_to?(:load) && coder_or_class_name.respond_to?(:dump)
|
185
|
+
coder_or_class_name
|
186
|
+
else
|
187
|
+
ActiveRecord::Coders::YAMLColumn.new(attr_name, coder_or_class_name || Object)
|
188
|
+
end
|
189
|
+
end
|
190
|
+
|
191
|
+
def dump(obj)
|
192
|
+
@coder.dump self.class.as_indifferent_hash(obj)
|
193
|
+
end
|
194
|
+
|
195
|
+
def load(yaml)
|
196
|
+
self.class.as_indifferent_hash(@coder.load(yaml || ""))
|
197
|
+
end
|
198
|
+
|
199
|
+
def self.as_indifferent_hash(obj)
|
200
|
+
case obj
|
201
|
+
when ActiveSupport::HashWithIndifferentAccess
|
202
|
+
obj
|
203
|
+
when Hash
|
204
|
+
obj.with_indifferent_access
|
205
|
+
else
|
206
|
+
ActiveSupport::HashWithIndifferentAccess.new
|
207
|
+
end
|
208
|
+
end
|
209
|
+
end
|
51
210
|
end
|
52
|
-
end
|
211
|
+
end
|
@@ -0,0 +1,61 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActiveRecord
|
4
|
+
# ActiveRecord::Suppressor prevents the receiver from being saved during
|
5
|
+
# a given block.
|
6
|
+
#
|
7
|
+
# For example, here's a pattern of creating notifications when new comments
|
8
|
+
# are posted. (The notification may in turn trigger an email, a push
|
9
|
+
# notification, or just appear in the UI somewhere):
|
10
|
+
#
|
11
|
+
# class Comment < ActiveRecord::Base
|
12
|
+
# belongs_to :commentable, polymorphic: true
|
13
|
+
# after_create -> { Notification.create! comment: self,
|
14
|
+
# recipients: commentable.recipients }
|
15
|
+
# end
|
16
|
+
#
|
17
|
+
# That's what you want the bulk of the time. New comment creates a new
|
18
|
+
# Notification. But there may well be off cases, like copying a commentable
|
19
|
+
# and its comments, where you don't want that. So you'd have a concern
|
20
|
+
# something like this:
|
21
|
+
#
|
22
|
+
# module Copyable
|
23
|
+
# def copy_to(destination)
|
24
|
+
# Notification.suppress do
|
25
|
+
# # Copy logic that creates new comments that we do not want
|
26
|
+
# # triggering notifications.
|
27
|
+
# end
|
28
|
+
# end
|
29
|
+
# end
|
30
|
+
module Suppressor
|
31
|
+
extend ActiveSupport::Concern
|
32
|
+
|
33
|
+
module ClassMethods
|
34
|
+
def suppress(&block)
|
35
|
+
previous_state = SuppressorRegistry.suppressed[name]
|
36
|
+
SuppressorRegistry.suppressed[name] = true
|
37
|
+
yield
|
38
|
+
ensure
|
39
|
+
SuppressorRegistry.suppressed[name] = previous_state
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
def save(*) # :nodoc:
|
44
|
+
SuppressorRegistry.suppressed[self.class.name] ? true : super
|
45
|
+
end
|
46
|
+
|
47
|
+
def save!(*) # :nodoc:
|
48
|
+
SuppressorRegistry.suppressed[self.class.name] ? true : super
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
class SuppressorRegistry # :nodoc:
|
53
|
+
extend ActiveSupport::PerThreadRegistry
|
54
|
+
|
55
|
+
attr_reader :suppressed
|
56
|
+
|
57
|
+
def initialize
|
58
|
+
@suppressed = {}
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
@@ -0,0 +1,82 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActiveRecord
|
4
|
+
class TableMetadata # :nodoc:
|
5
|
+
delegate :foreign_type, :foreign_key, :join_primary_key, :join_foreign_key, to: :association, prefix: true
|
6
|
+
|
7
|
+
def initialize(klass, arel_table, association = nil)
|
8
|
+
@klass = klass
|
9
|
+
@arel_table = arel_table
|
10
|
+
@association = association
|
11
|
+
end
|
12
|
+
|
13
|
+
def resolve_column_aliases(hash)
|
14
|
+
new_hash = hash.dup
|
15
|
+
hash.each do |key, _|
|
16
|
+
if (key.is_a?(Symbol)) && klass.attribute_alias?(key)
|
17
|
+
new_hash[klass.attribute_alias(key)] = new_hash.delete(key)
|
18
|
+
end
|
19
|
+
end
|
20
|
+
new_hash
|
21
|
+
end
|
22
|
+
|
23
|
+
def arel_attribute(column_name)
|
24
|
+
if klass
|
25
|
+
klass.arel_attribute(column_name, arel_table)
|
26
|
+
else
|
27
|
+
arel_table[column_name]
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
def type(column_name)
|
32
|
+
if klass
|
33
|
+
klass.type_for_attribute(column_name)
|
34
|
+
else
|
35
|
+
Type.default_value
|
36
|
+
end
|
37
|
+
end
|
38
|
+
|
39
|
+
def has_column?(column_name)
|
40
|
+
klass && klass.columns_hash.key?(column_name.to_s)
|
41
|
+
end
|
42
|
+
|
43
|
+
def associated_with?(association_name)
|
44
|
+
klass && klass._reflect_on_association(association_name)
|
45
|
+
end
|
46
|
+
|
47
|
+
def associated_table(table_name)
|
48
|
+
association = klass._reflect_on_association(table_name) || klass._reflect_on_association(table_name.to_s.singularize)
|
49
|
+
|
50
|
+
if !association && table_name == arel_table.name
|
51
|
+
return self
|
52
|
+
elsif association && !association.polymorphic?
|
53
|
+
association_klass = association.klass
|
54
|
+
arel_table = association_klass.arel_table.alias(table_name)
|
55
|
+
else
|
56
|
+
type_caster = TypeCaster::Connection.new(klass, table_name)
|
57
|
+
association_klass = nil
|
58
|
+
arel_table = Arel::Table.new(table_name, type_caster: type_caster)
|
59
|
+
end
|
60
|
+
|
61
|
+
TableMetadata.new(association_klass, arel_table, association)
|
62
|
+
end
|
63
|
+
|
64
|
+
def polymorphic_association?
|
65
|
+
association && association.polymorphic?
|
66
|
+
end
|
67
|
+
|
68
|
+
def aggregated_with?(aggregation_name)
|
69
|
+
klass && reflect_on_aggregation(aggregation_name)
|
70
|
+
end
|
71
|
+
|
72
|
+
def reflect_on_aggregation(aggregation_name)
|
73
|
+
klass.reflect_on_aggregation(aggregation_name)
|
74
|
+
end
|
75
|
+
|
76
|
+
# TODO Change this to private once we've dropped Ruby 2.2 support.
|
77
|
+
# Workaround for Ruby 2.2 "private attribute?" warning.
|
78
|
+
protected
|
79
|
+
|
80
|
+
attr_reader :klass, :arel_table, :association
|
81
|
+
end
|
82
|
+
end
|