activerecord 3.2.6 → 6.0.0
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.
Potentially problematic release.
This version of activerecord might be problematic. Click here for more details.
- checksums.yaml +7 -0
- data/CHANGELOG.md +611 -6417
- data/MIT-LICENSE +4 -2
- data/README.rdoc +44 -47
- data/examples/performance.rb +79 -71
- data/examples/simple.rb +6 -5
- data/lib/active_record/aggregations.rb +268 -238
- 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 +173 -81
- data/lib/active_record/associations/association_scope.rb +124 -92
- data/lib/active_record/associations/belongs_to_association.rb +83 -38
- data/lib/active_record/associations/belongs_to_polymorphic_association.rb +11 -9
- data/lib/active_record/associations/builder/association.rb +113 -32
- data/lib/active_record/associations/builder/belongs_to.rb +105 -60
- data/lib/active_record/associations/builder/collection_association.rb +53 -56
- data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +98 -41
- data/lib/active_record/associations/builder/has_many.rb +11 -63
- data/lib/active_record/associations/builder/has_one.rb +47 -45
- data/lib/active_record/associations/builder/singular_association.rb +30 -18
- data/lib/active_record/associations/collection_association.rb +217 -295
- data/lib/active_record/associations/collection_proxy.rb +1074 -77
- data/lib/active_record/associations/foreign_association.rb +20 -0
- data/lib/active_record/associations/has_many_association.rb +78 -50
- data/lib/active_record/associations/has_many_through_association.rb +99 -61
- data/lib/active_record/associations/has_one_association.rb +75 -30
- data/lib/active_record/associations/has_one_through_association.rb +20 -11
- data/lib/active_record/associations/join_dependency/join_association.rb +45 -119
- 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 +208 -164
- data/lib/active_record/associations/preloader/association.rb +93 -87
- data/lib/active_record/associations/preloader/through_association.rb +87 -38
- data/lib/active_record/associations/preloader.rb +134 -110
- data/lib/active_record/associations/singular_association.rb +19 -24
- data/lib/active_record/associations/through_association.rb +61 -27
- data/lib/active_record/associations.rb +1766 -1505
- data/lib/active_record/attribute_assignment.rb +57 -193
- data/lib/active_record/attribute_decorators.rb +90 -0
- data/lib/active_record/attribute_methods/before_type_cast.rb +58 -8
- data/lib/active_record/attribute_methods/dirty.rb +187 -67
- data/lib/active_record/attribute_methods/primary_key.rb +100 -78
- data/lib/active_record/attribute_methods/query.rb +10 -8
- data/lib/active_record/attribute_methods/read.rb +29 -118
- data/lib/active_record/attribute_methods/serialization.rb +60 -72
- data/lib/active_record/attribute_methods/time_zone_conversion.rb +69 -42
- data/lib/active_record/attribute_methods/write.rb +36 -44
- data/lib/active_record/attribute_methods.rb +306 -161
- data/lib/active_record/attributes.rb +279 -0
- data/lib/active_record/autosave_association.rb +324 -238
- data/lib/active_record/base.rb +114 -507
- data/lib/active_record/callbacks.rb +147 -83
- data/lib/active_record/coders/json.rb +15 -0
- data/lib/active_record/coders/yaml_column.rb +32 -23
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +962 -279
- data/lib/active_record/connection_adapters/abstract/database_limits.rb +32 -5
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +331 -209
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +95 -23
- data/lib/active_record/connection_adapters/abstract/quoting.rb +201 -65
- data/lib/active_record/connection_adapters/abstract/savepoints.rb +23 -0
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +153 -0
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +510 -289
- data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +93 -0
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +1182 -313
- data/lib/active_record/connection_adapters/abstract/transaction.rb +323 -0
- data/lib/active_record/connection_adapters/abstract_adapter.rb +585 -120
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +610 -463
- data/lib/active_record/connection_adapters/column.rb +58 -233
- data/lib/active_record/connection_adapters/connection_specification.rb +297 -0
- data/lib/active_record/connection_adapters/determine_if_preparable_visitor.rb +29 -0
- data/lib/active_record/connection_adapters/mysql/column.rb +27 -0
- data/lib/active_record/connection_adapters/mysql/database_statements.rb +200 -0
- data/lib/active_record/connection_adapters/mysql/explain_pretty_printer.rb +72 -0
- data/lib/active_record/connection_adapters/mysql/quoting.rb +81 -0
- data/lib/active_record/connection_adapters/mysql/schema_creation.rb +72 -0
- data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +95 -0
- data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +88 -0
- data/lib/active_record/connection_adapters/mysql/schema_statements.rb +264 -0
- data/lib/active_record/connection_adapters/mysql/type_metadata.rb +31 -0
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +75 -207
- data/lib/active_record/connection_adapters/postgresql/column.rb +30 -0
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +182 -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 +53 -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 +113 -0
- data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +26 -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 +205 -0
- data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +43 -0
- data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +76 -0
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +222 -0
- data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +50 -0
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +776 -0
- data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +36 -0
- data/lib/active_record/connection_adapters/postgresql/utils.rb +81 -0
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +695 -1052
- data/lib/active_record/connection_adapters/schema_cache.rb +115 -24
- data/lib/active_record/connection_adapters/sql_type_metadata.rb +37 -0
- data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +118 -0
- data/lib/active_record/connection_adapters/sqlite3/explain_pretty_printer.rb +21 -0
- data/lib/active_record/connection_adapters/sqlite3/quoting.rb +103 -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 +137 -0
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +528 -26
- data/lib/active_record/connection_adapters/statement_pool.rb +34 -13
- data/lib/active_record/connection_handling.rb +267 -0
- data/lib/active_record/core.rb +599 -0
- data/lib/active_record/counter_cache.rb +177 -103
- data/lib/active_record/database_configurations/database_config.rb +37 -0
- data/lib/active_record/database_configurations/hash_config.rb +50 -0
- data/lib/active_record/database_configurations/url_config.rb +79 -0
- data/lib/active_record/database_configurations.rb +233 -0
- data/lib/active_record/define_callbacks.rb +22 -0
- data/lib/active_record/dynamic_matchers.rb +107 -64
- data/lib/active_record/enum.rb +274 -0
- data/lib/active_record/errors.rb +254 -61
- data/lib/active_record/explain.rb +35 -70
- data/lib/active_record/explain_registry.rb +32 -0
- data/lib/active_record/explain_subscriber.rb +18 -8
- data/lib/active_record/fixture_set/file.rb +82 -0
- data/lib/active_record/fixture_set/model_metadata.rb +33 -0
- data/lib/active_record/fixture_set/render_context.rb +17 -0
- data/lib/active_record/fixture_set/table_row.rb +153 -0
- data/lib/active_record/fixture_set/table_rows.rb +47 -0
- data/lib/active_record/fixtures.rb +291 -475
- data/lib/active_record/gem_version.rb +17 -0
- data/lib/active_record/inheritance.rb +219 -100
- data/lib/active_record/insert_all.rb +179 -0
- data/lib/active_record/integration.rb +175 -17
- data/lib/active_record/internal_metadata.rb +53 -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 +106 -92
- data/lib/active_record/locking/pessimistic.rb +23 -11
- data/lib/active_record/log_subscriber.rb +80 -30
- data/lib/active_record/middleware/database_selector/resolver/session.rb +45 -0
- data/lib/active_record/middleware/database_selector/resolver.rb +92 -0
- data/lib/active_record/middleware/database_selector.rb +75 -0
- data/lib/active_record/migration/command_recorder.rb +235 -56
- data/lib/active_record/migration/compatibility.rb +244 -0
- data/lib/active_record/migration/join_table.rb +17 -0
- data/lib/active_record/migration.rb +917 -301
- data/lib/active_record/model_schema.rb +351 -175
- data/lib/active_record/nested_attributes.rb +366 -235
- data/lib/active_record/no_touching.rb +65 -0
- data/lib/active_record/null_relation.rb +68 -0
- data/lib/active_record/persistence.rb +761 -166
- data/lib/active_record/query_cache.rb +22 -44
- data/lib/active_record/querying.rb +55 -31
- data/lib/active_record/railtie.rb +185 -47
- data/lib/active_record/railties/collection_cache_association_loading.rb +34 -0
- data/lib/active_record/railties/console_sandbox.rb +5 -4
- data/lib/active_record/railties/controller_runtime.rb +35 -33
- data/lib/active_record/railties/databases.rake +366 -463
- data/lib/active_record/readonly_attributes.rb +4 -6
- data/lib/active_record/reflection.rb +736 -228
- data/lib/active_record/relation/batches/batch_enumerator.rb +69 -0
- data/lib/active_record/relation/batches.rb +252 -52
- data/lib/active_record/relation/calculations.rb +340 -270
- data/lib/active_record/relation/delegation.rb +117 -36
- data/lib/active_record/relation/finder_methods.rb +439 -286
- data/lib/active_record/relation/from_clause.rb +26 -0
- data/lib/active_record/relation/merger.rb +184 -0
- data/lib/active_record/relation/predicate_builder/array_handler.rb +49 -0
- data/lib/active_record/relation/predicate_builder/association_query_value.rb +43 -0
- data/lib/active_record/relation/predicate_builder/base_handler.rb +18 -0
- data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +19 -0
- data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +53 -0
- data/lib/active_record/relation/predicate_builder/range_handler.rb +22 -0
- data/lib/active_record/relation/predicate_builder/relation_handler.rb +19 -0
- data/lib/active_record/relation/predicate_builder.rb +131 -39
- data/lib/active_record/relation/query_attribute.rb +50 -0
- data/lib/active_record/relation/query_methods.rb +1163 -221
- data/lib/active_record/relation/record_fetch_warning.rb +51 -0
- data/lib/active_record/relation/spawn_methods.rb +49 -120
- data/lib/active_record/relation/where_clause.rb +190 -0
- data/lib/active_record/relation/where_clause_factory.rb +33 -0
- data/lib/active_record/relation.rb +671 -349
- data/lib/active_record/result.rb +149 -15
- data/lib/active_record/runtime_registry.rb +24 -0
- data/lib/active_record/sanitization.rb +153 -133
- data/lib/active_record/schema.rb +22 -19
- data/lib/active_record/schema_dumper.rb +178 -112
- data/lib/active_record/schema_migration.rb +60 -0
- data/lib/active_record/scoping/default.rb +107 -98
- data/lib/active_record/scoping/named.rb +130 -115
- 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 +148 -0
- data/lib/active_record/store.rb +256 -16
- data/lib/active_record/suppressor.rb +61 -0
- data/lib/active_record/table_metadata.rb +75 -0
- data/lib/active_record/tasks/database_tasks.rb +506 -0
- data/lib/active_record/tasks/mysql_database_tasks.rb +115 -0
- data/lib/active_record/tasks/postgresql_database_tasks.rb +141 -0
- data/lib/active_record/tasks/sqlite_database_tasks.rb +77 -0
- data/lib/active_record/test_databases.rb +23 -0
- data/lib/active_record/test_fixtures.rb +224 -0
- data/lib/active_record/timestamp.rb +93 -39
- data/lib/active_record/touch_later.rb +66 -0
- data/lib/active_record/transactions.rb +260 -129
- data/lib/active_record/translation.rb +3 -1
- data/lib/active_record/type/adapter_specific_registry.rb +129 -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 +78 -0
- data/lib/active_record/type_caster/connection.rb +34 -0
- data/lib/active_record/type_caster/map.rb +20 -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 +123 -77
- data/lib/active_record/validations.rb +54 -43
- data/lib/active_record/version.rb +7 -7
- data/lib/active_record.rb +97 -49
- data/lib/arel/alias_predication.rb +9 -0
- data/lib/arel/attributes/attribute.rb +37 -0
- data/lib/arel/attributes.rb +22 -0
- data/lib/arel/collectors/bind.rb +24 -0
- data/lib/arel/collectors/composite.rb +31 -0
- data/lib/arel/collectors/plain_string.rb +20 -0
- data/lib/arel/collectors/sql_string.rb +20 -0
- data/lib/arel/collectors/substitute_binds.rb +28 -0
- data/lib/arel/crud.rb +42 -0
- data/lib/arel/delete_manager.rb +18 -0
- data/lib/arel/errors.rb +9 -0
- data/lib/arel/expressions.rb +29 -0
- data/lib/arel/factory_methods.rb +49 -0
- data/lib/arel/insert_manager.rb +49 -0
- data/lib/arel/math.rb +45 -0
- data/lib/arel/nodes/and.rb +32 -0
- data/lib/arel/nodes/ascending.rb +23 -0
- data/lib/arel/nodes/binary.rb +52 -0
- data/lib/arel/nodes/bind_param.rb +36 -0
- data/lib/arel/nodes/case.rb +55 -0
- data/lib/arel/nodes/casted.rb +50 -0
- data/lib/arel/nodes/comment.rb +29 -0
- data/lib/arel/nodes/count.rb +12 -0
- data/lib/arel/nodes/delete_statement.rb +45 -0
- data/lib/arel/nodes/descending.rb +23 -0
- data/lib/arel/nodes/equality.rb +18 -0
- data/lib/arel/nodes/extract.rb +24 -0
- data/lib/arel/nodes/false.rb +16 -0
- data/lib/arel/nodes/full_outer_join.rb +8 -0
- data/lib/arel/nodes/function.rb +44 -0
- data/lib/arel/nodes/grouping.rb +8 -0
- data/lib/arel/nodes/in.rb +8 -0
- data/lib/arel/nodes/infix_operation.rb +80 -0
- data/lib/arel/nodes/inner_join.rb +8 -0
- data/lib/arel/nodes/insert_statement.rb +37 -0
- data/lib/arel/nodes/join_source.rb +20 -0
- data/lib/arel/nodes/matches.rb +18 -0
- data/lib/arel/nodes/named_function.rb +23 -0
- data/lib/arel/nodes/node.rb +50 -0
- data/lib/arel/nodes/node_expression.rb +13 -0
- data/lib/arel/nodes/outer_join.rb +8 -0
- data/lib/arel/nodes/over.rb +15 -0
- data/lib/arel/nodes/regexp.rb +16 -0
- data/lib/arel/nodes/right_outer_join.rb +8 -0
- data/lib/arel/nodes/select_core.rb +67 -0
- data/lib/arel/nodes/select_statement.rb +41 -0
- data/lib/arel/nodes/sql_literal.rb +16 -0
- data/lib/arel/nodes/string_join.rb +11 -0
- data/lib/arel/nodes/table_alias.rb +27 -0
- data/lib/arel/nodes/terminal.rb +16 -0
- data/lib/arel/nodes/true.rb +16 -0
- data/lib/arel/nodes/unary.rb +45 -0
- data/lib/arel/nodes/unary_operation.rb +20 -0
- data/lib/arel/nodes/unqualified_column.rb +22 -0
- data/lib/arel/nodes/update_statement.rb +41 -0
- data/lib/arel/nodes/values_list.rb +9 -0
- data/lib/arel/nodes/window.rb +126 -0
- data/lib/arel/nodes/with.rb +11 -0
- data/lib/arel/nodes.rb +68 -0
- data/lib/arel/order_predications.rb +13 -0
- data/lib/arel/predications.rb +257 -0
- data/lib/arel/select_manager.rb +271 -0
- data/lib/arel/table.rb +110 -0
- data/lib/arel/tree_manager.rb +72 -0
- data/lib/arel/update_manager.rb +34 -0
- data/lib/arel/visitors/depth_first.rb +204 -0
- data/lib/arel/visitors/dot.rb +297 -0
- data/lib/arel/visitors/ibm_db.rb +34 -0
- data/lib/arel/visitors/informix.rb +62 -0
- data/lib/arel/visitors/mssql.rb +157 -0
- data/lib/arel/visitors/mysql.rb +83 -0
- data/lib/arel/visitors/oracle.rb +159 -0
- data/lib/arel/visitors/oracle12.rb +66 -0
- data/lib/arel/visitors/postgresql.rb +110 -0
- data/lib/arel/visitors/sqlite.rb +39 -0
- data/lib/arel/visitors/to_sql.rb +889 -0
- data/lib/arel/visitors/visitor.rb +46 -0
- data/lib/arel/visitors/where_sql.rb +23 -0
- data/lib/arel/visitors.rb +20 -0
- data/lib/arel/window_predications.rb +9 -0
- data/lib/arel.rb +51 -0
- 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 +59 -9
- 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 +48 -0
- data/lib/rails/generators/active_record/migration.rb +41 -8
- data/lib/rails/generators/active_record/model/model_generator.rb +24 -22
- data/lib/rails/generators/active_record/model/templates/model.rb.tt +22 -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 +285 -149
- 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 -188
- data/lib/active_record/connection_adapters/mysql_adapter.rb +0 -426
- data/lib/active_record/connection_adapters/sqlite_adapter.rb +0 -579
- 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 -358
- 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,182 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module DatabaseStatements
         | 
| 7 | 
            +
                    def explain(arel, binds = [])
         | 
| 8 | 
            +
                      sql = "EXPLAIN #{to_sql(arel, binds)}"
         | 
| 9 | 
            +
                      PostgreSQL::ExplainPrettyPrinter.new.pp(exec_query(sql, "EXPLAIN", binds))
         | 
| 10 | 
            +
                    end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                    # The internal PostgreSQL identifier of the money data type.
         | 
| 13 | 
            +
                    MONEY_COLUMN_TYPE_OID = 790 #:nodoc:
         | 
| 14 | 
            +
                    # The internal PostgreSQL identifier of the BYTEA data type.
         | 
| 15 | 
            +
                    BYTEA_COLUMN_TYPE_OID = 17 #:nodoc:
         | 
| 16 | 
            +
             | 
| 17 | 
            +
                    # create a 2D array representing the result set
         | 
| 18 | 
            +
                    def result_as_array(res) #:nodoc:
         | 
| 19 | 
            +
                      # check if we have any binary column and if they need escaping
         | 
| 20 | 
            +
                      ftypes = Array.new(res.nfields) do |i|
         | 
| 21 | 
            +
                        [i, res.ftype(i)]
         | 
| 22 | 
            +
                      end
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                      rows = res.values
         | 
| 25 | 
            +
                      return rows unless ftypes.any? { |_, x|
         | 
| 26 | 
            +
                        x == BYTEA_COLUMN_TYPE_OID || x == MONEY_COLUMN_TYPE_OID
         | 
| 27 | 
            +
                      }
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                      typehash = ftypes.group_by { |_, type| type }
         | 
| 30 | 
            +
                      binaries = typehash[BYTEA_COLUMN_TYPE_OID] || []
         | 
| 31 | 
            +
                      monies   = typehash[MONEY_COLUMN_TYPE_OID] || []
         | 
| 32 | 
            +
             | 
| 33 | 
            +
                      rows.each do |row|
         | 
| 34 | 
            +
                        # unescape string passed BYTEA field (OID == 17)
         | 
| 35 | 
            +
                        binaries.each do |index, _|
         | 
| 36 | 
            +
                          row[index] = unescape_bytea(row[index])
         | 
| 37 | 
            +
                        end
         | 
| 38 | 
            +
             | 
| 39 | 
            +
                        # If this is a money type column and there are any currency symbols,
         | 
| 40 | 
            +
                        # then strip them off. Indeed it would be prettier to do this in
         | 
| 41 | 
            +
                        # PostgreSQLColumn.string_to_decimal but would break form input
         | 
| 42 | 
            +
                        # fields that call value_before_type_cast.
         | 
| 43 | 
            +
                        monies.each do |index, _|
         | 
| 44 | 
            +
                          data = row[index]
         | 
| 45 | 
            +
                          # Because money output is formatted according to the locale, there are two
         | 
| 46 | 
            +
                          # cases to consider (note the decimal separators):
         | 
| 47 | 
            +
                          #  (1) $12,345,678.12
         | 
| 48 | 
            +
                          #  (2) $12.345.678,12
         | 
| 49 | 
            +
                          case data
         | 
| 50 | 
            +
                          when /^-?\D+[\d,]+\.\d{2}$/  # (1)
         | 
| 51 | 
            +
                            data.gsub!(/[^-\d.]/, "")
         | 
| 52 | 
            +
                          when /^-?\D+[\d.]+,\d{2}$/  # (2)
         | 
| 53 | 
            +
                            data.gsub!(/[^-\d,]/, "").sub!(/,/, ".")
         | 
| 54 | 
            +
                          end
         | 
| 55 | 
            +
                        end
         | 
| 56 | 
            +
                      end
         | 
| 57 | 
            +
                    end
         | 
| 58 | 
            +
             | 
| 59 | 
            +
                    # Queries the database and returns the results in an Array-like object
         | 
| 60 | 
            +
                    def query(sql, name = nil) #:nodoc:
         | 
| 61 | 
            +
                      materialize_transactions
         | 
| 62 | 
            +
             | 
| 63 | 
            +
                      log(sql, name) do
         | 
| 64 | 
            +
                        ActiveSupport::Dependencies.interlock.permit_concurrent_loads do
         | 
| 65 | 
            +
                          result_as_array @connection.async_exec(sql)
         | 
| 66 | 
            +
                        end
         | 
| 67 | 
            +
                      end
         | 
| 68 | 
            +
                    end
         | 
| 69 | 
            +
             | 
| 70 | 
            +
                    READ_QUERY = ActiveRecord::ConnectionAdapters::AbstractAdapter.build_read_query_regexp(:begin, :commit, :explain, :select, :set, :show, :release, :savepoint, :rollback) # :nodoc:
         | 
| 71 | 
            +
                    private_constant :READ_QUERY
         | 
| 72 | 
            +
             | 
| 73 | 
            +
                    def write_query?(sql) # :nodoc:
         | 
| 74 | 
            +
                      !READ_QUERY.match?(sql)
         | 
| 75 | 
            +
                    end
         | 
| 76 | 
            +
             | 
| 77 | 
            +
                    # Executes an SQL statement, returning a PG::Result object on success
         | 
| 78 | 
            +
                    # or raising a PG::Error exception otherwise.
         | 
| 79 | 
            +
                    # Note: the PG::Result object is manually memory managed; if you don't
         | 
| 80 | 
            +
                    # need it specifically, you may want consider the <tt>exec_query</tt> wrapper.
         | 
| 81 | 
            +
                    def execute(sql, name = nil)
         | 
| 82 | 
            +
                      if preventing_writes? && write_query?(sql)
         | 
| 83 | 
            +
                        raise ActiveRecord::ReadOnlyError, "Write query attempted while in readonly mode: #{sql}"
         | 
| 84 | 
            +
                      end
         | 
| 85 | 
            +
             | 
| 86 | 
            +
                      materialize_transactions
         | 
| 87 | 
            +
             | 
| 88 | 
            +
                      log(sql, name) do
         | 
| 89 | 
            +
                        ActiveSupport::Dependencies.interlock.permit_concurrent_loads do
         | 
| 90 | 
            +
                          @connection.async_exec(sql)
         | 
| 91 | 
            +
                        end
         | 
| 92 | 
            +
                      end
         | 
| 93 | 
            +
                    end
         | 
| 94 | 
            +
             | 
| 95 | 
            +
                    def exec_query(sql, name = "SQL", binds = [], prepare: false)
         | 
| 96 | 
            +
                      execute_and_clear(sql, name, binds, prepare: prepare) do |result|
         | 
| 97 | 
            +
                        types = {}
         | 
| 98 | 
            +
                        fields = result.fields
         | 
| 99 | 
            +
                        fields.each_with_index do |fname, i|
         | 
| 100 | 
            +
                          ftype = result.ftype i
         | 
| 101 | 
            +
                          fmod  = result.fmod i
         | 
| 102 | 
            +
                          types[fname] = get_oid_type(ftype, fmod, fname)
         | 
| 103 | 
            +
                        end
         | 
| 104 | 
            +
                        ActiveRecord::Result.new(fields, result.values, types)
         | 
| 105 | 
            +
                      end
         | 
| 106 | 
            +
                    end
         | 
| 107 | 
            +
             | 
| 108 | 
            +
                    def exec_delete(sql, name = nil, binds = [])
         | 
| 109 | 
            +
                      execute_and_clear(sql, name, binds) { |result| result.cmd_tuples }
         | 
| 110 | 
            +
                    end
         | 
| 111 | 
            +
                    alias :exec_update :exec_delete
         | 
| 112 | 
            +
             | 
| 113 | 
            +
                    def sql_for_insert(sql, pk, binds) # :nodoc:
         | 
| 114 | 
            +
                      if pk.nil?
         | 
| 115 | 
            +
                        # Extract the table from the insert sql. Yuck.
         | 
| 116 | 
            +
                        table_ref = extract_table_ref_from_insert_sql(sql)
         | 
| 117 | 
            +
                        pk = primary_key(table_ref) if table_ref
         | 
| 118 | 
            +
                      end
         | 
| 119 | 
            +
             | 
| 120 | 
            +
                      if pk = suppress_composite_primary_key(pk)
         | 
| 121 | 
            +
                        sql = "#{sql} RETURNING #{quote_column_name(pk)}"
         | 
| 122 | 
            +
                      end
         | 
| 123 | 
            +
             | 
| 124 | 
            +
                      super
         | 
| 125 | 
            +
                    end
         | 
| 126 | 
            +
                    private :sql_for_insert
         | 
| 127 | 
            +
             | 
| 128 | 
            +
                    def exec_insert(sql, name = nil, binds = [], pk = nil, sequence_name = nil)
         | 
| 129 | 
            +
                      if use_insert_returning? || pk == false
         | 
| 130 | 
            +
                        super
         | 
| 131 | 
            +
                      else
         | 
| 132 | 
            +
                        result = exec_query(sql, name, binds)
         | 
| 133 | 
            +
                        unless sequence_name
         | 
| 134 | 
            +
                          table_ref = extract_table_ref_from_insert_sql(sql)
         | 
| 135 | 
            +
                          if table_ref
         | 
| 136 | 
            +
                            pk = primary_key(table_ref) if pk.nil?
         | 
| 137 | 
            +
                            pk = suppress_composite_primary_key(pk)
         | 
| 138 | 
            +
                            sequence_name = default_sequence_name(table_ref, pk)
         | 
| 139 | 
            +
                          end
         | 
| 140 | 
            +
                          return result unless sequence_name
         | 
| 141 | 
            +
                        end
         | 
| 142 | 
            +
                        last_insert_id_result(sequence_name)
         | 
| 143 | 
            +
                      end
         | 
| 144 | 
            +
                    end
         | 
| 145 | 
            +
             | 
| 146 | 
            +
                    # Begins a transaction.
         | 
| 147 | 
            +
                    def begin_db_transaction
         | 
| 148 | 
            +
                      execute "BEGIN"
         | 
| 149 | 
            +
                    end
         | 
| 150 | 
            +
             | 
| 151 | 
            +
                    def begin_isolated_db_transaction(isolation)
         | 
| 152 | 
            +
                      begin_db_transaction
         | 
| 153 | 
            +
                      execute "SET TRANSACTION ISOLATION LEVEL #{transaction_isolation_levels.fetch(isolation)}"
         | 
| 154 | 
            +
                    end
         | 
| 155 | 
            +
             | 
| 156 | 
            +
                    # Commits a transaction.
         | 
| 157 | 
            +
                    def commit_db_transaction
         | 
| 158 | 
            +
                      execute "COMMIT"
         | 
| 159 | 
            +
                    end
         | 
| 160 | 
            +
             | 
| 161 | 
            +
                    # Aborts a transaction.
         | 
| 162 | 
            +
                    def exec_rollback_db_transaction
         | 
| 163 | 
            +
                      execute "ROLLBACK"
         | 
| 164 | 
            +
                    end
         | 
| 165 | 
            +
             | 
| 166 | 
            +
                    private
         | 
| 167 | 
            +
                      def build_truncate_statements(*table_names)
         | 
| 168 | 
            +
                        "TRUNCATE TABLE #{table_names.map(&method(:quote_table_name)).join(", ")}"
         | 
| 169 | 
            +
                      end
         | 
| 170 | 
            +
             | 
| 171 | 
            +
                      # Returns the current ID of a table's sequence.
         | 
| 172 | 
            +
                      def last_insert_id_result(sequence_name)
         | 
| 173 | 
            +
                        exec_query("SELECT currval(#{quote(sequence_name)})", "SQL")
         | 
| 174 | 
            +
                      end
         | 
| 175 | 
            +
             | 
| 176 | 
            +
                      def suppress_composite_primary_key(pk)
         | 
| 177 | 
            +
                        pk unless pk.is_a?(Array)
         | 
| 178 | 
            +
                      end
         | 
| 179 | 
            +
                  end
         | 
| 180 | 
            +
                end
         | 
| 181 | 
            +
              end
         | 
| 182 | 
            +
            end
         | 
| @@ -0,0 +1,44 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  class ExplainPrettyPrinter # :nodoc:
         | 
| 7 | 
            +
                    # Pretty prints the result of an EXPLAIN in a way that resembles the output of the
         | 
| 8 | 
            +
                    # PostgreSQL shell:
         | 
| 9 | 
            +
                    #
         | 
| 10 | 
            +
                    #                                     QUERY PLAN
         | 
| 11 | 
            +
                    #   ------------------------------------------------------------------------------
         | 
| 12 | 
            +
                    #    Nested Loop Left Join  (cost=0.00..37.24 rows=8 width=0)
         | 
| 13 | 
            +
                    #      Join Filter: (posts.user_id = users.id)
         | 
| 14 | 
            +
                    #      ->  Index Scan using users_pkey on users  (cost=0.00..8.27 rows=1 width=4)
         | 
| 15 | 
            +
                    #            Index Cond: (id = 1)
         | 
| 16 | 
            +
                    #      ->  Seq Scan on posts  (cost=0.00..28.88 rows=8 width=4)
         | 
| 17 | 
            +
                    #            Filter: (posts.user_id = 1)
         | 
| 18 | 
            +
                    #   (6 rows)
         | 
| 19 | 
            +
                    #
         | 
| 20 | 
            +
                    def pp(result)
         | 
| 21 | 
            +
                      header = result.columns.first
         | 
| 22 | 
            +
                      lines  = result.rows.map(&:first)
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                      # We add 2 because there's one char of padding at both sides, note
         | 
| 25 | 
            +
                      # the extra hyphens in the example above.
         | 
| 26 | 
            +
                      width = [header, *lines].map(&:length).max + 2
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                      pp = []
         | 
| 29 | 
            +
             | 
| 30 | 
            +
                      pp << header.center(width).rstrip
         | 
| 31 | 
            +
                      pp << "-" * width
         | 
| 32 | 
            +
             | 
| 33 | 
            +
                      pp += lines.map { |line| " #{line}" }
         | 
| 34 | 
            +
             | 
| 35 | 
            +
                      nrows = result.rows.length
         | 
| 36 | 
            +
                      rows_label = nrows == 1 ? "row" : "rows"
         | 
| 37 | 
            +
                      pp << "(#{nrows} #{rows_label})"
         | 
| 38 | 
            +
             | 
| 39 | 
            +
                      pp.join("\n") + "\n"
         | 
| 40 | 
            +
                    end
         | 
| 41 | 
            +
                  end
         | 
| 42 | 
            +
                end
         | 
| 43 | 
            +
              end
         | 
| 44 | 
            +
            end
         | 
| @@ -0,0 +1,92 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class Array < Type::Value # :nodoc:
         | 
| 8 | 
            +
                      include ActiveModel::Type::Helpers::Mutable
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                      Data = Struct.new(:encoder, :values) # :nodoc:
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                      attr_reader :subtype, :delimiter
         | 
| 13 | 
            +
                      delegate :type, :user_input_in_time_zone, :limit, :precision, :scale, to: :subtype
         | 
| 14 | 
            +
             | 
| 15 | 
            +
                      def initialize(subtype, delimiter = ",")
         | 
| 16 | 
            +
                        @subtype = subtype
         | 
| 17 | 
            +
                        @delimiter = delimiter
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                        @pg_encoder = PG::TextEncoder::Array.new name: "#{type}[]", delimiter: delimiter
         | 
| 20 | 
            +
                        @pg_decoder = PG::TextDecoder::Array.new name: "#{type}[]", delimiter: delimiter
         | 
| 21 | 
            +
                      end
         | 
| 22 | 
            +
             | 
| 23 | 
            +
                      def deserialize(value)
         | 
| 24 | 
            +
                        case value
         | 
| 25 | 
            +
                        when ::String
         | 
| 26 | 
            +
                          type_cast_array(@pg_decoder.decode(value), :deserialize)
         | 
| 27 | 
            +
                        when Data
         | 
| 28 | 
            +
                          type_cast_array(value.values, :deserialize)
         | 
| 29 | 
            +
                        else
         | 
| 30 | 
            +
                          super
         | 
| 31 | 
            +
                        end
         | 
| 32 | 
            +
                      end
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                      def cast(value)
         | 
| 35 | 
            +
                        if value.is_a?(::String)
         | 
| 36 | 
            +
                          value = begin
         | 
| 37 | 
            +
                            @pg_decoder.decode(value)
         | 
| 38 | 
            +
                          rescue TypeError
         | 
| 39 | 
            +
                            # malformed array string is treated as [], will raise in PG 2.0 gem
         | 
| 40 | 
            +
                            # this keeps a consistent implementation
         | 
| 41 | 
            +
                            []
         | 
| 42 | 
            +
                          end
         | 
| 43 | 
            +
                        end
         | 
| 44 | 
            +
                        type_cast_array(value, :cast)
         | 
| 45 | 
            +
                      end
         | 
| 46 | 
            +
             | 
| 47 | 
            +
                      def serialize(value)
         | 
| 48 | 
            +
                        if value.is_a?(::Array)
         | 
| 49 | 
            +
                          casted_values = type_cast_array(value, :serialize)
         | 
| 50 | 
            +
                          Data.new(@pg_encoder, casted_values)
         | 
| 51 | 
            +
                        else
         | 
| 52 | 
            +
                          super
         | 
| 53 | 
            +
                        end
         | 
| 54 | 
            +
                      end
         | 
| 55 | 
            +
             | 
| 56 | 
            +
                      def ==(other)
         | 
| 57 | 
            +
                        other.is_a?(Array) &&
         | 
| 58 | 
            +
                          subtype == other.subtype &&
         | 
| 59 | 
            +
                          delimiter == other.delimiter
         | 
| 60 | 
            +
                      end
         | 
| 61 | 
            +
             | 
| 62 | 
            +
                      def type_cast_for_schema(value)
         | 
| 63 | 
            +
                        return super unless value.is_a?(::Array)
         | 
| 64 | 
            +
                        "[" + value.map { |v| subtype.type_cast_for_schema(v) }.join(", ") + "]"
         | 
| 65 | 
            +
                      end
         | 
| 66 | 
            +
             | 
| 67 | 
            +
                      def map(value, &block)
         | 
| 68 | 
            +
                        value.map(&block)
         | 
| 69 | 
            +
                      end
         | 
| 70 | 
            +
             | 
| 71 | 
            +
                      def changed_in_place?(raw_old_value, new_value)
         | 
| 72 | 
            +
                        deserialize(raw_old_value) != new_value
         | 
| 73 | 
            +
                      end
         | 
| 74 | 
            +
             | 
| 75 | 
            +
                      def force_equality?(value)
         | 
| 76 | 
            +
                        value.is_a?(::Array)
         | 
| 77 | 
            +
                      end
         | 
| 78 | 
            +
             | 
| 79 | 
            +
                      private
         | 
| 80 | 
            +
             | 
| 81 | 
            +
                        def type_cast_array(value, method)
         | 
| 82 | 
            +
                          if value.is_a?(::Array)
         | 
| 83 | 
            +
                            value.map { |item| type_cast_array(item, method) }
         | 
| 84 | 
            +
                          else
         | 
| 85 | 
            +
                            @subtype.public_send(method, value)
         | 
| 86 | 
            +
                          end
         | 
| 87 | 
            +
                        end
         | 
| 88 | 
            +
                    end
         | 
| 89 | 
            +
                  end
         | 
| 90 | 
            +
                end
         | 
| 91 | 
            +
              end
         | 
| 92 | 
            +
            end
         | 
| @@ -0,0 +1,53 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class Bit < Type::Value # :nodoc:
         | 
| 8 | 
            +
                      def type
         | 
| 9 | 
            +
                        :bit
         | 
| 10 | 
            +
                      end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                      def cast_value(value)
         | 
| 13 | 
            +
                        if ::String === value
         | 
| 14 | 
            +
                          case value
         | 
| 15 | 
            +
                          when /^0x/i
         | 
| 16 | 
            +
                            value[2..-1].hex.to_s(2) # Hexadecimal notation
         | 
| 17 | 
            +
                          else
         | 
| 18 | 
            +
                            value                    # Bit-string notation
         | 
| 19 | 
            +
                          end
         | 
| 20 | 
            +
                        else
         | 
| 21 | 
            +
                          value.to_s
         | 
| 22 | 
            +
                        end
         | 
| 23 | 
            +
                      end
         | 
| 24 | 
            +
             | 
| 25 | 
            +
                      def serialize(value)
         | 
| 26 | 
            +
                        Data.new(super) if value
         | 
| 27 | 
            +
                      end
         | 
| 28 | 
            +
             | 
| 29 | 
            +
                      class Data
         | 
| 30 | 
            +
                        def initialize(value)
         | 
| 31 | 
            +
                          @value = value
         | 
| 32 | 
            +
                        end
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                        def to_s
         | 
| 35 | 
            +
                          value
         | 
| 36 | 
            +
                        end
         | 
| 37 | 
            +
             | 
| 38 | 
            +
                        def binary?
         | 
| 39 | 
            +
                          /\A[01]*\Z/.match?(value)
         | 
| 40 | 
            +
                        end
         | 
| 41 | 
            +
             | 
| 42 | 
            +
                        def hex?
         | 
| 43 | 
            +
                          /\A[0-9A-F]*\Z/i.match?(value)
         | 
| 44 | 
            +
                        end
         | 
| 45 | 
            +
             | 
| 46 | 
            +
                        private
         | 
| 47 | 
            +
                          attr_reader :value
         | 
| 48 | 
            +
                      end
         | 
| 49 | 
            +
                    end
         | 
| 50 | 
            +
                  end
         | 
| 51 | 
            +
                end
         | 
| 52 | 
            +
              end
         | 
| 53 | 
            +
            end
         | 
| @@ -0,0 +1,17 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class Bytea < Type::Binary # :nodoc:
         | 
| 8 | 
            +
                      def deserialize(value)
         | 
| 9 | 
            +
                        return if value.nil?
         | 
| 10 | 
            +
                        return value.to_s if value.is_a?(Type::Binary::Data)
         | 
| 11 | 
            +
                        PG::Connection.unescape_bytea(super)
         | 
| 12 | 
            +
                      end
         | 
| 13 | 
            +
                    end
         | 
| 14 | 
            +
                  end
         | 
| 15 | 
            +
                end
         | 
| 16 | 
            +
              end
         | 
| 17 | 
            +
            end
         | 
| @@ -0,0 +1,50 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            require "ipaddr"
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            module ActiveRecord
         | 
| 6 | 
            +
              module ConnectionAdapters
         | 
| 7 | 
            +
                module PostgreSQL
         | 
| 8 | 
            +
                  module OID # :nodoc:
         | 
| 9 | 
            +
                    class Cidr < Type::Value # :nodoc:
         | 
| 10 | 
            +
                      def type
         | 
| 11 | 
            +
                        :cidr
         | 
| 12 | 
            +
                      end
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                      def type_cast_for_schema(value)
         | 
| 15 | 
            +
                        subnet_mask = value.instance_variable_get(:@mask_addr)
         | 
| 16 | 
            +
             | 
| 17 | 
            +
                        # If the subnet mask is equal to /32, don't output it
         | 
| 18 | 
            +
                        if subnet_mask == (2**32 - 1)
         | 
| 19 | 
            +
                          "\"#{value}\""
         | 
| 20 | 
            +
                        else
         | 
| 21 | 
            +
                          "\"#{value}/#{subnet_mask.to_s(2).count('1')}\""
         | 
| 22 | 
            +
                        end
         | 
| 23 | 
            +
                      end
         | 
| 24 | 
            +
             | 
| 25 | 
            +
                      def serialize(value)
         | 
| 26 | 
            +
                        if IPAddr === value
         | 
| 27 | 
            +
                          "#{value}/#{value.instance_variable_get(:@mask_addr).to_s(2).count('1')}"
         | 
| 28 | 
            +
                        else
         | 
| 29 | 
            +
                          value
         | 
| 30 | 
            +
                        end
         | 
| 31 | 
            +
                      end
         | 
| 32 | 
            +
             | 
| 33 | 
            +
                      def cast_value(value)
         | 
| 34 | 
            +
                        if value.nil?
         | 
| 35 | 
            +
                          nil
         | 
| 36 | 
            +
                        elsif String === value
         | 
| 37 | 
            +
                          begin
         | 
| 38 | 
            +
                            IPAddr.new(value)
         | 
| 39 | 
            +
                          rescue ArgumentError
         | 
| 40 | 
            +
                            nil
         | 
| 41 | 
            +
                          end
         | 
| 42 | 
            +
                        else
         | 
| 43 | 
            +
                          value
         | 
| 44 | 
            +
                        end
         | 
| 45 | 
            +
                      end
         | 
| 46 | 
            +
                    end
         | 
| 47 | 
            +
                  end
         | 
| 48 | 
            +
                end
         | 
| 49 | 
            +
              end
         | 
| 50 | 
            +
            end
         | 
| @@ -0,0 +1,23 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class Date < Type::Date # :nodoc:
         | 
| 8 | 
            +
                      def cast_value(value)
         | 
| 9 | 
            +
                        case value
         | 
| 10 | 
            +
                        when "infinity" then ::Float::INFINITY
         | 
| 11 | 
            +
                        when "-infinity" then -::Float::INFINITY
         | 
| 12 | 
            +
                        when / BC$/
         | 
| 13 | 
            +
                          astronomical_year = format("%04d", -value[/^\d+/].to_i + 1)
         | 
| 14 | 
            +
                          super(value.sub(/ BC$/, "").sub(/^\d+/, astronomical_year))
         | 
| 15 | 
            +
                        else
         | 
| 16 | 
            +
                          super
         | 
| 17 | 
            +
                        end
         | 
| 18 | 
            +
                      end
         | 
| 19 | 
            +
                    end
         | 
| 20 | 
            +
                  end
         | 
| 21 | 
            +
                end
         | 
| 22 | 
            +
              end
         | 
| 23 | 
            +
            end
         | 
| @@ -0,0 +1,23 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class DateTime < Type::DateTime # :nodoc:
         | 
| 8 | 
            +
                      def cast_value(value)
         | 
| 9 | 
            +
                        case value
         | 
| 10 | 
            +
                        when "infinity" then ::Float::INFINITY
         | 
| 11 | 
            +
                        when "-infinity" then -::Float::INFINITY
         | 
| 12 | 
            +
                        when / BC$/
         | 
| 13 | 
            +
                          astronomical_year = format("%04d", -value[/^\d+/].to_i + 1)
         | 
| 14 | 
            +
                          super(value.sub(/ BC$/, "").sub(/^\d+/, astronomical_year))
         | 
| 15 | 
            +
                        else
         | 
| 16 | 
            +
                          super
         | 
| 17 | 
            +
                        end
         | 
| 18 | 
            +
                      end
         | 
| 19 | 
            +
                    end
         | 
| 20 | 
            +
                  end
         | 
| 21 | 
            +
                end
         | 
| 22 | 
            +
              end
         | 
| 23 | 
            +
            end
         | 
| @@ -0,0 +1,15 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class Decimal < Type::Decimal # :nodoc:
         | 
| 8 | 
            +
                      def infinity(options = {})
         | 
| 9 | 
            +
                        BigDecimal("Infinity") * (options[:negative] ? -1 : 1)
         | 
| 10 | 
            +
                      end
         | 
| 11 | 
            +
                    end
         | 
| 12 | 
            +
                  end
         | 
| 13 | 
            +
                end
         | 
| 14 | 
            +
              end
         | 
| 15 | 
            +
            end
         | 
| @@ -0,0 +1,21 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class Enum < Type::Value # :nodoc:
         | 
| 8 | 
            +
                      def type
         | 
| 9 | 
            +
                        :enum
         | 
| 10 | 
            +
                      end
         | 
| 11 | 
            +
             | 
| 12 | 
            +
                      private
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                        def cast_value(value)
         | 
| 15 | 
            +
                          value.to_s
         | 
| 16 | 
            +
                        end
         | 
| 17 | 
            +
                    end
         | 
| 18 | 
            +
                  end
         | 
| 19 | 
            +
                end
         | 
| 20 | 
            +
              end
         | 
| 21 | 
            +
            end
         | 
| @@ -0,0 +1,71 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class Hstore < Type::Value # :nodoc:
         | 
| 8 | 
            +
                      include ActiveModel::Type::Helpers::Mutable
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                      def type
         | 
| 11 | 
            +
                        :hstore
         | 
| 12 | 
            +
                      end
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                      def deserialize(value)
         | 
| 15 | 
            +
                        if value.is_a?(::String)
         | 
| 16 | 
            +
                          ::Hash[value.scan(HstorePair).map { |k, v|
         | 
| 17 | 
            +
                            v = v.upcase == "NULL" ? nil : v.gsub(/\A"(.*)"\Z/m, '\1').gsub(/\\(.)/, '\1')
         | 
| 18 | 
            +
                            k = k.gsub(/\A"(.*)"\Z/m, '\1').gsub(/\\(.)/, '\1')
         | 
| 19 | 
            +
                            [k, v]
         | 
| 20 | 
            +
                          }]
         | 
| 21 | 
            +
                        else
         | 
| 22 | 
            +
                          value
         | 
| 23 | 
            +
                        end
         | 
| 24 | 
            +
                      end
         | 
| 25 | 
            +
             | 
| 26 | 
            +
                      def serialize(value)
         | 
| 27 | 
            +
                        if value.is_a?(::Hash)
         | 
| 28 | 
            +
                          value.map { |k, v| "#{escape_hstore(k)}=>#{escape_hstore(v)}" }.join(", ")
         | 
| 29 | 
            +
                        elsif value.respond_to?(:to_unsafe_h)
         | 
| 30 | 
            +
                          serialize(value.to_unsafe_h)
         | 
| 31 | 
            +
                        else
         | 
| 32 | 
            +
                          value
         | 
| 33 | 
            +
                        end
         | 
| 34 | 
            +
                      end
         | 
| 35 | 
            +
             | 
| 36 | 
            +
                      def accessor
         | 
| 37 | 
            +
                        ActiveRecord::Store::StringKeyedHashAccessor
         | 
| 38 | 
            +
                      end
         | 
| 39 | 
            +
             | 
| 40 | 
            +
                      # Will compare the Hash equivalents of +raw_old_value+ and +new_value+.
         | 
| 41 | 
            +
                      # By comparing hashes, this avoids an edge case where the order of
         | 
| 42 | 
            +
                      # the keys change between the two hashes, and they would not be marked
         | 
| 43 | 
            +
                      # as equal.
         | 
| 44 | 
            +
                      def changed_in_place?(raw_old_value, new_value)
         | 
| 45 | 
            +
                        deserialize(raw_old_value) != new_value
         | 
| 46 | 
            +
                      end
         | 
| 47 | 
            +
             | 
| 48 | 
            +
                      private
         | 
| 49 | 
            +
             | 
| 50 | 
            +
                        HstorePair = begin
         | 
| 51 | 
            +
                          quoted_string = /"[^"\\]*(?:\\.[^"\\]*)*"/
         | 
| 52 | 
            +
                          unquoted_string = /(?:\\.|[^\s,])[^\s=,\\]*(?:\\.[^\s=,\\]*|=[^,>])*/
         | 
| 53 | 
            +
                          /(#{quoted_string}|#{unquoted_string})\s*=>\s*(#{quoted_string}|#{unquoted_string})/
         | 
| 54 | 
            +
                        end
         | 
| 55 | 
            +
             | 
| 56 | 
            +
                        def escape_hstore(value)
         | 
| 57 | 
            +
                          if value.nil?
         | 
| 58 | 
            +
                            "NULL"
         | 
| 59 | 
            +
                          else
         | 
| 60 | 
            +
                            if value == ""
         | 
| 61 | 
            +
                              '""'
         | 
| 62 | 
            +
                            else
         | 
| 63 | 
            +
                              '"%s"' % value.to_s.gsub(/(["\\])/, '\\\\\1')
         | 
| 64 | 
            +
                            end
         | 
| 65 | 
            +
                          end
         | 
| 66 | 
            +
                        end
         | 
| 67 | 
            +
                    end
         | 
| 68 | 
            +
                  end
         | 
| 69 | 
            +
                end
         | 
| 70 | 
            +
              end
         | 
| 71 | 
            +
            end
         | 
| @@ -0,0 +1,45 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module ActiveRecord
         | 
| 4 | 
            +
              module ConnectionAdapters
         | 
| 5 | 
            +
                module PostgreSQL
         | 
| 6 | 
            +
                  module OID # :nodoc:
         | 
| 7 | 
            +
                    class LegacyPoint < Type::Value # :nodoc:
         | 
| 8 | 
            +
                      include ActiveModel::Type::Helpers::Mutable
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                      def type
         | 
| 11 | 
            +
                        :point
         | 
| 12 | 
            +
                      end
         | 
| 13 | 
            +
             | 
| 14 | 
            +
                      def cast(value)
         | 
| 15 | 
            +
                        case value
         | 
| 16 | 
            +
                        when ::String
         | 
| 17 | 
            +
                          if value[0] == "(" && value[-1] == ")"
         | 
| 18 | 
            +
                            value = value[1...-1]
         | 
| 19 | 
            +
                          end
         | 
| 20 | 
            +
                          cast(value.split(","))
         | 
| 21 | 
            +
                        when ::Array
         | 
| 22 | 
            +
                          value.map { |v| Float(v) }
         | 
| 23 | 
            +
                        else
         | 
| 24 | 
            +
                          value
         | 
| 25 | 
            +
                        end
         | 
| 26 | 
            +
                      end
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                      def serialize(value)
         | 
| 29 | 
            +
                        if value.is_a?(::Array)
         | 
| 30 | 
            +
                          "(#{number_for_point(value[0])},#{number_for_point(value[1])})"
         | 
| 31 | 
            +
                        else
         | 
| 32 | 
            +
                          super
         | 
| 33 | 
            +
                        end
         | 
| 34 | 
            +
                      end
         | 
| 35 | 
            +
             | 
| 36 | 
            +
                      private
         | 
| 37 | 
            +
             | 
| 38 | 
            +
                        def number_for_point(number)
         | 
| 39 | 
            +
                          number.to_s.gsub(/\.0$/, "")
         | 
| 40 | 
            +
                        end
         | 
| 41 | 
            +
                    end
         | 
| 42 | 
            +
                  end
         | 
| 43 | 
            +
                end
         | 
| 44 | 
            +
              end
         | 
| 45 | 
            +
            end
         |