ibm_db 4.0.0-x86-mingw32 → 5.0.2-x86-mingw32
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -5
- data/MANIFEST +14 -14
- data/README +208 -208
- data/ext/Makefile +269 -0
- data/ext/Makefile.nt32 +181 -181
- data/ext/Makefile.nt32.191 +212 -212
- data/ext/extconf.rb +322 -291
- data/ext/gil_release_version +3 -0
- data/ext/ibm_db.c +11879 -11887
- data/ext/mkmf.log +110 -0
- data/ext/ruby_ibm_db.h +241 -241
- data/ext/ruby_ibm_db_cli.c +866 -866
- data/ext/ruby_ibm_db_cli.h +500 -500
- data/ext/unicode_support_version +3 -0
- data/init.rb +41 -41
- data/lib/IBM_DB.rb +27 -27
- data/lib/active_record/connection_adapters/ibm_db_adapter.rb +3533 -3452
- data/lib/active_record/connection_adapters/ibmdb_adapter.rb +5 -5
- data/lib/active_record/vendor/db2-i5-zOS.yaml +328 -328
- data/lib/mswin32/ibm_db.rb +90 -90
- data/lib/mswin32/rb2x/i386/ibm_db.so +0 -0
- data/test/active_record/connection_adapters/fake_adapter.rb +49 -49
- data/test/assets/example.log +1 -1
- data/test/assets/test.txt +1 -1
- data/test/cases/adapter_test.rb +351 -351
- data/test/cases/adapters/mysql2/active_schema_test.rb +193 -193
- data/test/cases/adapters/mysql2/bind_parameter_test.rb +50 -50
- data/test/cases/adapters/mysql2/boolean_test.rb +100 -100
- data/test/cases/adapters/mysql2/case_sensitivity_test.rb +63 -63
- data/test/cases/adapters/mysql2/charset_collation_test.rb +54 -54
- data/test/cases/adapters/mysql2/connection_test.rb +210 -210
- data/test/cases/adapters/mysql2/datetime_precision_quoting_test.rb +45 -45
- data/test/cases/adapters/mysql2/enum_test.rb +26 -26
- data/test/cases/adapters/mysql2/explain_test.rb +21 -21
- data/test/cases/adapters/mysql2/json_test.rb +195 -195
- data/test/cases/adapters/mysql2/mysql2_adapter_test.rb +83 -83
- data/test/cases/adapters/mysql2/reserved_word_test.rb +152 -152
- data/test/cases/adapters/mysql2/schema_migrations_test.rb +59 -59
- data/test/cases/adapters/mysql2/schema_test.rb +126 -126
- data/test/cases/adapters/mysql2/sp_test.rb +36 -36
- data/test/cases/adapters/mysql2/sql_types_test.rb +14 -14
- data/test/cases/adapters/mysql2/table_options_test.rb +42 -42
- data/test/cases/adapters/mysql2/unsigned_type_test.rb +66 -66
- data/test/cases/adapters/postgresql/active_schema_test.rb +98 -98
- data/test/cases/adapters/postgresql/array_test.rb +339 -339
- data/test/cases/adapters/postgresql/bit_string_test.rb +82 -82
- data/test/cases/adapters/postgresql/bytea_test.rb +134 -134
- data/test/cases/adapters/postgresql/case_insensitive_test.rb +26 -26
- data/test/cases/adapters/postgresql/change_schema_test.rb +38 -38
- data/test/cases/adapters/postgresql/cidr_test.rb +25 -25
- data/test/cases/adapters/postgresql/citext_test.rb +78 -78
- data/test/cases/adapters/postgresql/collation_test.rb +53 -53
- data/test/cases/adapters/postgresql/composite_test.rb +132 -132
- data/test/cases/adapters/postgresql/connection_test.rb +257 -257
- data/test/cases/adapters/postgresql/datatype_test.rb +92 -92
- data/test/cases/adapters/postgresql/domain_test.rb +47 -47
- data/test/cases/adapters/postgresql/enum_test.rb +91 -91
- data/test/cases/adapters/postgresql/explain_test.rb +20 -20
- data/test/cases/adapters/postgresql/extension_migration_test.rb +63 -63
- data/test/cases/adapters/postgresql/full_text_test.rb +44 -44
- data/test/cases/adapters/postgresql/geometric_test.rb +378 -378
- data/test/cases/adapters/postgresql/hstore_test.rb +382 -382
- data/test/cases/adapters/postgresql/infinity_test.rb +69 -69
- data/test/cases/adapters/postgresql/integer_test.rb +25 -25
- data/test/cases/adapters/postgresql/json_test.rb +237 -237
- data/test/cases/adapters/postgresql/ltree_test.rb +53 -53
- data/test/cases/adapters/postgresql/money_test.rb +96 -96
- data/test/cases/adapters/postgresql/network_test.rb +94 -94
- data/test/cases/adapters/postgresql/numbers_test.rb +49 -49
- data/test/cases/adapters/postgresql/postgresql_adapter_test.rb +405 -405
- data/test/cases/adapters/postgresql/prepared_statements_test.rb +22 -22
- data/test/cases/adapters/postgresql/quoting_test.rb +44 -44
- data/test/cases/adapters/postgresql/range_test.rb +343 -343
- data/test/cases/adapters/postgresql/referential_integrity_test.rb +111 -111
- data/test/cases/adapters/postgresql/rename_table_test.rb +34 -34
- data/test/cases/adapters/postgresql/schema_authorization_test.rb +119 -119
- data/test/cases/adapters/postgresql/schema_test.rb +597 -597
- data/test/cases/adapters/postgresql/serial_test.rb +154 -154
- data/test/cases/adapters/postgresql/statement_pool_test.rb +41 -41
- data/test/cases/adapters/postgresql/timestamp_test.rb +90 -90
- data/test/cases/adapters/postgresql/type_lookup_test.rb +33 -33
- data/test/cases/adapters/postgresql/utils_test.rb +62 -62
- data/test/cases/adapters/postgresql/uuid_test.rb +294 -294
- data/test/cases/adapters/postgresql/xml_test.rb +54 -54
- data/test/cases/adapters/sqlite3/collation_test.rb +53 -53
- data/test/cases/adapters/sqlite3/copy_table_test.rb +98 -98
- data/test/cases/adapters/sqlite3/explain_test.rb +21 -21
- data/test/cases/adapters/sqlite3/quoting_test.rb +101 -101
- data/test/cases/adapters/sqlite3/sqlite3_adapter_test.rb +441 -441
- data/test/cases/adapters/sqlite3/sqlite3_create_folder_test.rb +24 -24
- data/test/cases/adapters/sqlite3/statement_pool_test.rb +20 -20
- data/test/cases/aggregations_test.rb +168 -168
- data/test/cases/ar_schema_test.rb +146 -146
- data/test/cases/associations/association_scope_test.rb +16 -16
- data/test/cases/associations/belongs_to_associations_test.rb +1141 -1141
- data/test/cases/associations/bidirectional_destroy_dependencies_test.rb +41 -41
- data/test/cases/associations/callbacks_test.rb +190 -190
- data/test/cases/associations/cascaded_eager_loading_test.rb +188 -188
- data/test/cases/associations/eager_load_includes_full_sti_class_test.rb +36 -36
- data/test/cases/associations/eager_load_nested_include_test.rb +126 -126
- data/test/cases/associations/eager_singularization_test.rb +148 -148
- data/test/cases/associations/eager_test.rb +1514 -1514
- data/test/cases/associations/extension_test.rb +87 -87
- data/test/cases/associations/has_and_belongs_to_many_associations_test.rb +1004 -1004
- data/test/cases/associations/has_many_associations_test.rb +2501 -2501
- data/test/cases/associations/has_many_through_associations_test.rb +1271 -1271
- data/test/cases/associations/has_one_associations_test.rb +707 -707
- data/test/cases/associations/has_one_through_associations_test.rb +383 -383
- data/test/cases/associations/inner_join_association_test.rb +139 -139
- data/test/cases/associations/inverse_associations_test.rb +733 -733
- data/test/cases/associations/join_model_test.rb +777 -777
- data/test/cases/associations/left_outer_join_association_test.rb +88 -88
- data/test/cases/associations/nested_through_associations_test.rb +579 -579
- data/test/cases/associations/required_test.rb +102 -102
- data/test/cases/associations_test.rb +385 -385
- data/test/cases/attribute_decorators_test.rb +126 -125
- data/test/cases/attribute_methods/read_test.rb +60 -60
- data/test/cases/attribute_methods_test.rb +1009 -1009
- data/test/cases/attribute_set_test.rb +270 -270
- data/test/cases/attribute_test.rb +246 -246
- data/test/cases/attributes_test.rb +253 -253
- data/test/cases/autosave_association_test.rb +1708 -1708
- data/test/cases/base_test.rb +1713 -1713
- data/test/cases/batches_test.rb +489 -489
- data/test/cases/binary_test.rb +44 -44
- data/test/cases/bind_parameter_test.rb +110 -110
- data/test/cases/cache_key_test.rb +26 -25
- data/test/cases/calculations_test.rb +798 -798
- data/test/cases/callbacks_test.rb +636 -636
- data/test/cases/clone_test.rb +40 -40
- data/test/cases/coders/json_test.rb +15 -15
- data/test/cases/coders/yaml_column_test.rb +63 -63
- data/test/cases/collection_cache_key_test.rb +115 -115
- data/test/cases/column_alias_test.rb +17 -17
- data/test/cases/column_definition_test.rb +92 -92
- data/test/cases/comment_test.rb +145 -143
- data/test/cases/connection_adapters/adapter_leasing_test.rb +56 -56
- data/test/cases/connection_adapters/connection_handler_test.rb +160 -160
- data/test/cases/connection_adapters/connection_specification_test.rb +12 -12
- data/test/cases/connection_adapters/merge_and_resolve_default_url_config_test.rb +255 -255
- data/test/cases/connection_adapters/mysql_type_lookup_test.rb +69 -69
- data/test/cases/connection_adapters/quoting_test.rb +13 -13
- data/test/cases/connection_adapters/schema_cache_test.rb +61 -61
- data/test/cases/connection_adapters/type_lookup_test.rb +118 -118
- data/test/cases/connection_management_test.rb +112 -112
- data/test/cases/connection_pool_test.rb +521 -521
- data/test/cases/connection_specification/resolver_test.rb +131 -131
- data/test/cases/core_test.rb +112 -112
- data/test/cases/counter_cache_test.rb +214 -214
- data/test/cases/custom_locking_test.rb +17 -17
- data/test/cases/database_statements_test.rb +34 -34
- data/test/cases/date_test.rb +44 -44
- data/test/cases/date_time_precision_test.rb +107 -106
- data/test/cases/date_time_test.rb +61 -61
- data/test/cases/defaults_test.rb +219 -218
- data/test/cases/dirty_test.rb +763 -763
- data/test/cases/disconnected_test.rb +30 -30
- data/test/cases/dup_test.rb +157 -157
- data/test/cases/enum_test.rb +444 -444
- data/test/cases/errors_test.rb +16 -16
- data/test/cases/explain_subscriber_test.rb +64 -64
- data/test/cases/explain_test.rb +87 -87
- data/test/cases/finder_respond_to_test.rb +60 -60
- data/test/cases/finder_test.rb +1294 -1294
- data/test/cases/fixture_set/file_test.rb +156 -156
- data/test/cases/fixtures_test.rb +988 -988
- data/test/cases/forbidden_attributes_protection_test.rb +165 -165
- data/test/cases/habtm_destroy_order_test.rb +61 -61
- data/test/cases/helper.rb +204 -204
- data/test/cases/hot_compatibility_test.rb +142 -142
- data/test/cases/i18n_test.rb +45 -45
- data/test/cases/inheritance_test.rb +606 -606
- data/test/cases/integration_test.rb +155 -155
- data/test/cases/invalid_connection_test.rb +24 -24
- data/test/cases/invertible_migration_test.rb +387 -387
- data/test/cases/json_serialization_test.rb +311 -311
- data/test/cases/locking_test.rb +493 -493
- data/test/cases/log_subscriber_test.rb +225 -225
- data/test/cases/migration/change_schema_test.rb +458 -458
- data/test/cases/migration/change_table_test.rb +256 -256
- data/test/cases/migration/column_attributes_test.rb +176 -176
- data/test/cases/migration/column_positioning_test.rb +56 -56
- data/test/cases/migration/columns_test.rb +310 -310
- data/test/cases/migration/command_recorder_test.rb +350 -350
- data/test/cases/migration/compatibility_test.rb +118 -118
- data/test/cases/migration/create_join_table_test.rb +157 -157
- data/test/cases/migration/foreign_key_test.rb +362 -360
- data/test/cases/migration/helper.rb +39 -39
- data/test/cases/migration/index_test.rb +218 -218
- data/test/cases/migration/logger_test.rb +36 -36
- data/test/cases/migration/pending_migrations_test.rb +52 -52
- data/test/cases/migration/references_foreign_key_test.rb +221 -216
- data/test/cases/migration/references_index_test.rb +101 -101
- data/test/cases/migration/references_statements_test.rb +136 -136
- data/test/cases/migration/rename_table_test.rb +93 -93
- data/test/cases/migration_test.rb +1157 -1157
- data/test/cases/migrator_test.rb +471 -470
- data/test/cases/mixin_test.rb +68 -68
- data/test/cases/modules_test.rb +172 -172
- data/test/cases/multiparameter_attributes_test.rb +372 -372
- data/test/cases/multiple_db_test.rb +122 -122
- data/test/cases/nested_attributes_test.rb +1098 -1098
- data/test/cases/nested_attributes_with_callbacks_test.rb +144 -144
- data/test/cases/persistence_test.rb +1001 -1001
- data/test/cases/pooled_connections_test.rb +81 -81
- data/test/cases/primary_keys_test.rb +376 -376
- data/test/cases/query_cache_test.rb +446 -446
- data/test/cases/quoting_test.rb +202 -202
- data/test/cases/readonly_test.rb +119 -119
- data/test/cases/reaper_test.rb +85 -85
- data/test/cases/reflection_test.rb +509 -509
- data/test/cases/relation/delegation_test.rb +63 -63
- data/test/cases/relation/merging_test.rb +157 -157
- data/test/cases/relation/mutation_test.rb +183 -183
- data/test/cases/relation/or_test.rb +92 -92
- data/test/cases/relation/predicate_builder_test.rb +16 -16
- data/test/cases/relation/record_fetch_warning_test.rb +40 -40
- data/test/cases/relation/where_chain_test.rb +105 -105
- data/test/cases/relation/where_clause_test.rb +182 -182
- data/test/cases/relation/where_test.rb +322 -322
- data/test/cases/relation_test.rb +328 -328
- data/test/cases/relations_test.rb +2026 -2026
- data/test/cases/reload_models_test.rb +22 -22
- data/test/cases/result_test.rb +90 -90
- data/test/cases/sanitize_test.rb +176 -176
- data/test/cases/schema_dumper_test.rb +457 -457
- data/test/cases/schema_loading_test.rb +52 -52
- data/test/cases/scoping/default_scoping_test.rb +528 -528
- data/test/cases/scoping/named_scoping_test.rb +561 -561
- data/test/cases/scoping/relation_scoping_test.rb +400 -400
- data/test/cases/secure_token_test.rb +32 -32
- data/test/cases/serialization_test.rb +104 -104
- data/test/cases/serialized_attribute_test.rb +364 -364
- data/test/cases/statement_cache_test.rb +136 -136
- data/test/cases/store_test.rb +195 -195
- data/test/cases/suppressor_test.rb +63 -63
- data/test/cases/tasks/database_tasks_test.rb +462 -462
- data/test/cases/tasks/mysql_rake_test.rb +345 -345
- data/test/cases/tasks/postgresql_rake_test.rb +304 -304
- data/test/cases/tasks/sqlite_rake_test.rb +220 -220
- data/test/cases/test_case.rb +131 -131
- data/test/cases/test_fixtures_test.rb +36 -36
- data/test/cases/time_precision_test.rb +103 -102
- data/test/cases/timestamp_test.rb +501 -501
- data/test/cases/touch_later_test.rb +121 -121
- data/test/cases/transaction_callbacks_test.rb +518 -518
- data/test/cases/transaction_isolation_test.rb +106 -106
- data/test/cases/transactions_test.rb +835 -834
- data/test/cases/type/adapter_specific_registry_test.rb +133 -133
- data/test/cases/type/date_time_test.rb +14 -14
- data/test/cases/type/integer_test.rb +27 -27
- data/test/cases/type/string_test.rb +22 -22
- data/test/cases/type/type_map_test.rb +177 -177
- data/test/cases/type_test.rb +39 -39
- data/test/cases/types_test.rb +24 -24
- data/test/cases/unconnected_test.rb +33 -33
- data/test/cases/validations/absence_validation_test.rb +73 -73
- data/test/cases/validations/association_validation_test.rb +97 -97
- data/test/cases/validations/i18n_generate_message_validation_test.rb +84 -84
- data/test/cases/validations/i18n_validation_test.rb +86 -86
- data/test/cases/validations/length_validation_test.rb +79 -79
- data/test/cases/validations/presence_validation_test.rb +103 -103
- data/test/cases/validations/uniqueness_validation_test.rb +548 -548
- data/test/cases/validations_repair_helper.rb +19 -19
- data/test/cases/validations_test.rb +194 -194
- data/test/cases/view_test.rb +216 -216
- data/test/cases/yaml_serialization_test.rb +121 -121
- data/test/config.example.yml +97 -97
- data/test/config.rb +5 -5
- data/test/connections/native_ibm_db/connection.rb +44 -0
- data/test/fixtures/accounts.yml +29 -29
- data/test/fixtures/admin/accounts.yml +2 -2
- data/test/fixtures/admin/users.yml +10 -10
- data/test/fixtures/author_addresses.yml +17 -17
- data/test/fixtures/author_favorites.yml +3 -3
- data/test/fixtures/authors.yml +23 -23
- data/test/fixtures/bad_posts.yml +9 -9
- data/test/fixtures/binaries.yml +133 -133
- data/test/fixtures/books.yml +31 -31
- data/test/fixtures/bulbs.yml +5 -5
- data/test/fixtures/cars.yml +9 -9
- data/test/fixtures/categories.yml +19 -19
- data/test/fixtures/categories/special_categories.yml +9 -9
- data/test/fixtures/categories/subsubdir/arbitrary_filename.yml +4 -4
- data/test/fixtures/categories_ordered.yml +7 -7
- data/test/fixtures/categories_posts.yml +31 -31
- data/test/fixtures/categorizations.yml +23 -23
- data/test/fixtures/clubs.yml +8 -8
- data/test/fixtures/collections.yml +3 -3
- data/test/fixtures/colleges.yml +3 -3
- data/test/fixtures/comments.yml +65 -65
- data/test/fixtures/companies.yml +67 -67
- data/test/fixtures/computers.yml +10 -10
- data/test/fixtures/content.yml +3 -3
- data/test/fixtures/content_positions.yml +3 -3
- data/test/fixtures/courses.yml +8 -8
- data/test/fixtures/customers.yml +25 -25
- data/test/fixtures/dashboards.yml +6 -6
- data/test/fixtures/dead_parrots.yml +5 -5
- data/test/fixtures/developers.yml +22 -22
- data/test/fixtures/developers_projects.yml +16 -16
- data/test/fixtures/dog_lovers.yml +7 -7
- data/test/fixtures/dogs.yml +4 -4
- data/test/fixtures/doubloons.yml +3 -3
- data/test/fixtures/edges.yml +5 -5
- data/test/fixtures/entrants.yml +14 -14
- data/test/fixtures/essays.yml +6 -6
- data/test/fixtures/faces.yml +11 -11
- data/test/fixtures/fk_test_has_fk.yml +3 -3
- data/test/fixtures/fk_test_has_pk.yml +1 -1
- data/test/fixtures/friendships.yml +4 -4
- data/test/fixtures/funny_jokes.yml +10 -10
- data/test/fixtures/interests.yml +33 -33
- data/test/fixtures/items.yml +3 -3
- data/test/fixtures/jobs.yml +7 -7
- data/test/fixtures/legacy_things.yml +3 -3
- data/test/fixtures/live_parrots.yml +4 -4
- data/test/fixtures/mateys.yml +4 -4
- data/test/fixtures/member_details.yml +8 -8
- data/test/fixtures/member_types.yml +6 -6
- data/test/fixtures/members.yml +11 -11
- data/test/fixtures/memberships.yml +34 -34
- data/test/fixtures/men.yml +5 -5
- data/test/fixtures/minimalistics.yml +2 -2
- data/test/fixtures/minivans.yml +5 -5
- data/test/fixtures/mixed_case_monkeys.yml +6 -6
- data/test/fixtures/mixins.yml +29 -29
- data/test/fixtures/movies.yml +7 -7
- data/test/fixtures/naked/yml/accounts.yml +1 -1
- data/test/fixtures/naked/yml/companies.yml +1 -1
- data/test/fixtures/naked/yml/courses.yml +1 -1
- data/test/fixtures/naked/yml/parrots.yml +2 -2
- data/test/fixtures/naked/yml/trees.yml +3 -3
- data/test/fixtures/nodes.yml +29 -29
- data/test/fixtures/organizations.yml +5 -5
- data/test/fixtures/other_comments.yml +6 -6
- data/test/fixtures/other_dogs.yml +2 -2
- data/test/fixtures/other_posts.yml +7 -7
- data/test/fixtures/other_topics.yml +42 -42
- data/test/fixtures/owners.yml +9 -9
- data/test/fixtures/parrots.yml +27 -27
- data/test/fixtures/parrots_pirates.yml +7 -7
- data/test/fixtures/people.yml +24 -24
- data/test/fixtures/peoples_treasures.yml +3 -3
- data/test/fixtures/pets.yml +19 -19
- data/test/fixtures/pirates.yml +12 -15
- data/test/fixtures/posts.yml +80 -80
- data/test/fixtures/price_estimates.yml +16 -16
- data/test/fixtures/products.yml +4 -4
- data/test/fixtures/projects.yml +7 -7
- data/test/fixtures/ratings.yml +14 -14
- data/test/fixtures/readers.yml +11 -11
- data/test/fixtures/references.yml +17 -17
- data/test/fixtures/reserved_words/distinct.yml +5 -5
- data/test/fixtures/reserved_words/distinct_select.yml +11 -11
- data/test/fixtures/reserved_words/group.yml +14 -14
- data/test/fixtures/reserved_words/select.yml +8 -8
- data/test/fixtures/reserved_words/values.yml +7 -7
- data/test/fixtures/ships.yml +6 -6
- data/test/fixtures/speedometers.yml +8 -8
- data/test/fixtures/sponsors.yml +12 -12
- data/test/fixtures/string_key_objects.yml +7 -7
- data/test/fixtures/subscribers.yml +10 -10
- data/test/fixtures/subscriptions.yml +12 -12
- data/test/fixtures/taggings.yml +78 -78
- data/test/fixtures/tags.yml +11 -11
- data/test/fixtures/tasks.yml +7 -7
- data/test/fixtures/teapots.yml +3 -3
- data/test/fixtures/to_be_linked/accounts.yml +2 -2
- data/test/fixtures/to_be_linked/users.yml +10 -10
- data/test/fixtures/topics.yml +49 -49
- data/test/fixtures/toys.yml +14 -14
- data/test/fixtures/traffic_lights.yml +9 -9
- data/test/fixtures/treasures.yml +10 -10
- data/test/fixtures/trees.yml +3 -3
- data/test/fixtures/uuid_children.yml +3 -3
- data/test/fixtures/uuid_parents.yml +2 -2
- data/test/fixtures/variants.yml +4 -4
- data/test/fixtures/vegetables.yml +19 -19
- data/test/fixtures/vertices.yml +3 -3
- data/test/fixtures/warehouse_things.yml +2 -2
- data/test/fixtures/zines.yml +5 -5
- data/test/migrations/10_urban/9_add_expressions.rb +11 -11
- data/test/migrations/decimal/1_give_me_big_numbers.rb +15 -15
- data/test/migrations/magic/1_currencies_have_symbols.rb +12 -12
- data/test/migrations/missing/1000_people_have_middle_names.rb +9 -9
- data/test/migrations/missing/1_people_have_last_names.rb +9 -9
- data/test/migrations/missing/3_we_need_reminders.rb +12 -12
- data/test/migrations/missing/4_innocent_jointable.rb +12 -12
- data/test/migrations/rename/1_we_need_things.rb +11 -11
- data/test/migrations/rename/2_rename_things.rb +9 -9
- data/test/migrations/to_copy/1_people_have_hobbies.rb +9 -9
- data/test/migrations/to_copy/2_people_have_descriptions.rb +9 -9
- data/test/migrations/to_copy2/1_create_articles.rb +7 -7
- data/test/migrations/to_copy2/2_create_comments.rb +7 -7
- data/test/migrations/to_copy_with_name_collision/1_people_have_hobbies.rb +9 -9
- data/test/migrations/to_copy_with_timestamps/20090101010101_people_have_hobbies.rb +9 -9
- data/test/migrations/to_copy_with_timestamps/20090101010202_people_have_descriptions.rb +9 -9
- data/test/migrations/to_copy_with_timestamps2/20090101010101_create_articles.rb +7 -7
- data/test/migrations/to_copy_with_timestamps2/20090101010202_create_comments.rb +7 -7
- data/test/migrations/valid/1_valid_people_have_last_names.rb +9 -9
- data/test/migrations/valid/2_we_need_reminders.rb +12 -12
- data/test/migrations/valid/3_innocent_jointable.rb +12 -12
- data/test/migrations/valid_with_subdirectories/1_valid_people_have_last_names.rb +9 -9
- data/test/migrations/valid_with_subdirectories/sub/2_we_need_reminders.rb +12 -12
- data/test/migrations/valid_with_subdirectories/sub1/3_innocent_jointable.rb +12 -12
- data/test/migrations/valid_with_timestamps/20100101010101_valid_with_timestamps_people_have_last_names.rb +9 -9
- data/test/migrations/valid_with_timestamps/20100201010101_valid_with_timestamps_we_need_reminders.rb +12 -12
- data/test/migrations/valid_with_timestamps/20100301010101_valid_with_timestamps_innocent_jointable.rb +12 -12
- data/test/migrations/version_check/20131219224947_migration_version_check.rb +8 -8
- data/test/models/admin.rb +5 -5
- data/test/models/admin/account.rb +3 -3
- data/test/models/admin/user.rb +40 -40
- data/test/models/aircraft.rb +5 -5
- data/test/models/arunit2_model.rb +3 -3
- data/test/models/author.rb +209 -209
- data/test/models/auto_id.rb +4 -4
- data/test/models/autoloadable/extra_firm.rb +2 -2
- data/test/models/binary.rb +2 -2
- data/test/models/bird.rb +12 -12
- data/test/models/book.rb +23 -23
- data/test/models/boolean.rb +2 -2
- data/test/models/bulb.rb +52 -52
- data/test/models/cake_designer.rb +3 -3
- data/test/models/car.rb +29 -29
- data/test/models/carrier.rb +2 -2
- data/test/models/cat.rb +10 -10
- data/test/models/categorization.rb +19 -19
- data/test/models/category.rb +35 -35
- data/test/models/chef.rb +8 -8
- data/test/models/citation.rb +3 -3
- data/test/models/club.rb +25 -25
- data/test/models/college.rb +10 -10
- data/test/models/column.rb +3 -3
- data/test/models/column_name.rb +3 -3
- data/test/models/comment.rb +76 -76
- data/test/models/company.rb +230 -230
- data/test/models/company_in_module.rb +98 -98
- data/test/models/computer.rb +3 -3
- data/test/models/contact.rb +41 -41
- data/test/models/content.rb +40 -40
- data/test/models/contract.rb +20 -20
- data/test/models/country.rb +7 -7
- data/test/models/course.rb +6 -6
- data/test/models/customer.rb +83 -83
- data/test/models/customer_carrier.rb +14 -14
- data/test/models/dashboard.rb +3 -3
- data/test/models/default.rb +2 -2
- data/test/models/department.rb +4 -4
- data/test/models/developer.rb +274 -274
- data/test/models/dog.rb +5 -5
- data/test/models/dog_lover.rb +5 -5
- data/test/models/doubloon.rb +12 -12
- data/test/models/drink_designer.rb +3 -3
- data/test/models/edge.rb +5 -5
- data/test/models/electron.rb +5 -5
- data/test/models/engine.rb +4 -4
- data/test/models/entrant.rb +3 -3
- data/test/models/essay.rb +5 -5
- data/test/models/event.rb +3 -3
- data/test/models/eye.rb +37 -37
- data/test/models/face.rb +9 -9
- data/test/models/friendship.rb +6 -6
- data/test/models/guid.rb +2 -2
- data/test/models/guitar.rb +4 -4
- data/test/models/hotel.rb +11 -11
- data/test/models/image.rb +3 -3
- data/test/models/interest.rb +5 -5
- data/test/models/invoice.rb +4 -4
- data/test/models/item.rb +7 -7
- data/test/models/job.rb +7 -7
- data/test/models/joke.rb +7 -7
- data/test/models/keyboard.rb +3 -3
- data/test/models/legacy_thing.rb +3 -3
- data/test/models/lesson.rb +11 -11
- data/test/models/line_item.rb +3 -3
- data/test/models/liquid.rb +4 -4
- data/test/models/man.rb +11 -11
- data/test/models/matey.rb +4 -4
- data/test/models/member.rb +42 -42
- data/test/models/member_detail.rb +8 -8
- data/test/models/member_type.rb +3 -3
- data/test/models/membership.rb +35 -35
- data/test/models/mentor.rb +2 -2
- data/test/models/minimalistic.rb +2 -2
- data/test/models/minivan.rb +9 -9
- data/test/models/mixed_case_monkey.rb +3 -3
- data/test/models/mocktail_designer.rb +2 -2
- data/test/models/molecule.rb +6 -6
- data/test/models/movie.rb +5 -5
- data/test/models/node.rb +5 -5
- data/test/models/non_primary_key.rb +2 -2
- data/test/models/notification.rb +3 -3
- data/test/models/order.rb +4 -4
- data/test/models/organization.rb +14 -14
- data/test/models/other_dog.rb +5 -5
- data/test/models/owner.rb +37 -37
- data/test/models/parrot.rb +28 -28
- data/test/models/person.rb +142 -142
- data/test/models/personal_legacy_thing.rb +4 -4
- data/test/models/pet.rb +18 -18
- data/test/models/pet_treasure.rb +6 -6
- data/test/models/pirate.rb +92 -92
- data/test/models/possession.rb +3 -3
- data/test/models/post.rb +273 -273
- data/test/models/price_estimate.rb +4 -4
- data/test/models/professor.rb +5 -5
- data/test/models/project.rb +40 -40
- data/test/models/publisher.rb +2 -2
- data/test/models/publisher/article.rb +4 -4
- data/test/models/publisher/magazine.rb +3 -3
- data/test/models/rating.rb +4 -4
- data/test/models/reader.rb +23 -23
- data/test/models/recipe.rb +3 -3
- data/test/models/record.rb +2 -2
- data/test/models/reference.rb +22 -22
- data/test/models/reply.rb +61 -61
- data/test/models/ship.rb +39 -39
- data/test/models/ship_part.rb +8 -8
- data/test/models/shop.rb +17 -17
- data/test/models/shop_account.rb +6 -6
- data/test/models/speedometer.rb +6 -6
- data/test/models/sponsor.rb +7 -7
- data/test/models/string_key_object.rb +3 -3
- data/test/models/student.rb +4 -4
- data/test/models/subject.rb +16 -16
- data/test/models/subscriber.rb +8 -8
- data/test/models/subscription.rb +4 -4
- data/test/models/tag.rb +13 -13
- data/test/models/tagging.rb +13 -13
- data/test/models/task.rb +5 -5
- data/test/models/topic.rb +118 -118
- data/test/models/toy.rb +6 -6
- data/test/models/traffic_light.rb +4 -4
- data/test/models/treasure.rb +14 -14
- data/test/models/treaty.rb +7 -7
- data/test/models/tree.rb +3 -3
- data/test/models/tuning_peg.rb +4 -4
- data/test/models/tyre.rb +11 -11
- data/test/models/user.rb +14 -14
- data/test/models/uuid_child.rb +3 -3
- data/test/models/uuid_item.rb +6 -6
- data/test/models/uuid_parent.rb +3 -3
- data/test/models/vegetables.rb +24 -24
- data/test/models/vehicle.rb +6 -6
- data/test/models/vertex.rb +9 -9
- data/test/models/warehouse_thing.rb +5 -5
- data/test/models/wheel.rb +3 -3
- data/test/models/without_table.rb +3 -3
- data/test/models/zine.rb +3 -3
- data/test/schema/i5/ibm_db_specific_schema.rb +137 -0
- data/test/schema/ids/ibm_db_specific_schema.rb +140 -0
- data/test/schema/luw/ibm_db_specific_schema.rb +137 -0
- data/test/schema/mysql2_specific_schema.rb +68 -68
- data/test/schema/oracle_specific_schema.rb +40 -40
- data/test/schema/postgresql_specific_schema.rb +114 -114
- data/test/schema/schema.rb +1057 -1057
- data/test/schema/schema.rb.original +1057 -1057
- data/test/schema/sqlite_specific_schema.rb +18 -18
- data/test/schema/zOS/ibm_db_specific_schema.rb +208 -0
- data/test/support/config.rb +43 -43
- data/test/support/connection.rb +23 -23
- data/test/support/connection_helper.rb +14 -14
- data/test/support/ddl_helper.rb +8 -8
- data/test/support/schema_dumping_helper.rb +20 -20
- data/test/support/yaml_compatibility_fixtures/rails_4_1.yml +22 -22
- data/test/support/yaml_compatibility_fixtures/rails_4_2_0.yml +182 -182
- metadata +24 -13
- data/test/fixtures/author_addresses.original +0 -11
- data/test/fixtures/authors.original +0 -17
@@ -1,42 +1,42 @@
|
|
1
|
-
require "cases/helper"
|
2
|
-
require 'support/schema_dumping_helper'
|
3
|
-
|
4
|
-
class Mysql2TableOptionsTest < ActiveRecord::Mysql2TestCase
|
5
|
-
include SchemaDumpingHelper
|
6
|
-
|
7
|
-
def setup
|
8
|
-
@connection = ActiveRecord::Base.connection
|
9
|
-
end
|
10
|
-
|
11
|
-
def teardown
|
12
|
-
@connection.drop_table "mysql_table_options", if_exists: true
|
13
|
-
end
|
14
|
-
|
15
|
-
test "table options with ENGINE" do
|
16
|
-
@connection.create_table "mysql_table_options", force: true, options: "ENGINE=MyISAM"
|
17
|
-
output = dump_table_schema("mysql_table_options")
|
18
|
-
options = %r{create_table "mysql_table_options", force: :cascade, options: "(?<options>.*)"}.match(output)[:options]
|
19
|
-
assert_match %r{ENGINE=MyISAM}, options
|
20
|
-
end
|
21
|
-
|
22
|
-
test "table options with ROW_FORMAT" do
|
23
|
-
@connection.create_table "mysql_table_options", force: true, options: "ROW_FORMAT=REDUNDANT"
|
24
|
-
output = dump_table_schema("mysql_table_options")
|
25
|
-
options = %r{create_table "mysql_table_options", force: :cascade, options: "(?<options>.*)"}.match(output)[:options]
|
26
|
-
assert_match %r{ROW_FORMAT=REDUNDANT}, options
|
27
|
-
end
|
28
|
-
|
29
|
-
test "table options with CHARSET" do
|
30
|
-
@connection.create_table "mysql_table_options", force: true, options: "CHARSET=utf8mb4"
|
31
|
-
output = dump_table_schema("mysql_table_options")
|
32
|
-
options = %r{create_table "mysql_table_options", force: :cascade, options: "(?<options>.*)"}.match(output)[:options]
|
33
|
-
assert_match %r{CHARSET=utf8mb4}, options
|
34
|
-
end
|
35
|
-
|
36
|
-
test "table options with COLLATE" do
|
37
|
-
@connection.create_table "mysql_table_options", force: true, options: "COLLATE=utf8mb4_bin"
|
38
|
-
output = dump_table_schema("mysql_table_options")
|
39
|
-
options = %r{create_table "mysql_table_options", force: :cascade, options: "(?<options>.*)"}.match(output)[:options]
|
40
|
-
assert_match %r{COLLATE=utf8mb4_bin}, options
|
41
|
-
end
|
42
|
-
end
|
1
|
+
require "cases/helper"
|
2
|
+
require 'support/schema_dumping_helper'
|
3
|
+
|
4
|
+
class Mysql2TableOptionsTest < ActiveRecord::Mysql2TestCase
|
5
|
+
include SchemaDumpingHelper
|
6
|
+
|
7
|
+
def setup
|
8
|
+
@connection = ActiveRecord::Base.connection
|
9
|
+
end
|
10
|
+
|
11
|
+
def teardown
|
12
|
+
@connection.drop_table "mysql_table_options", if_exists: true
|
13
|
+
end
|
14
|
+
|
15
|
+
test "table options with ENGINE" do
|
16
|
+
@connection.create_table "mysql_table_options", force: true, options: "ENGINE=MyISAM"
|
17
|
+
output = dump_table_schema("mysql_table_options")
|
18
|
+
options = %r{create_table "mysql_table_options", force: :cascade, options: "(?<options>.*)"}.match(output)[:options]
|
19
|
+
assert_match %r{ENGINE=MyISAM}, options
|
20
|
+
end
|
21
|
+
|
22
|
+
test "table options with ROW_FORMAT" do
|
23
|
+
@connection.create_table "mysql_table_options", force: true, options: "ROW_FORMAT=REDUNDANT"
|
24
|
+
output = dump_table_schema("mysql_table_options")
|
25
|
+
options = %r{create_table "mysql_table_options", force: :cascade, options: "(?<options>.*)"}.match(output)[:options]
|
26
|
+
assert_match %r{ROW_FORMAT=REDUNDANT}, options
|
27
|
+
end
|
28
|
+
|
29
|
+
test "table options with CHARSET" do
|
30
|
+
@connection.create_table "mysql_table_options", force: true, options: "CHARSET=utf8mb4"
|
31
|
+
output = dump_table_schema("mysql_table_options")
|
32
|
+
options = %r{create_table "mysql_table_options", force: :cascade, options: "(?<options>.*)"}.match(output)[:options]
|
33
|
+
assert_match %r{CHARSET=utf8mb4}, options
|
34
|
+
end
|
35
|
+
|
36
|
+
test "table options with COLLATE" do
|
37
|
+
@connection.create_table "mysql_table_options", force: true, options: "COLLATE=utf8mb4_bin"
|
38
|
+
output = dump_table_schema("mysql_table_options")
|
39
|
+
options = %r{create_table "mysql_table_options", force: :cascade, options: "(?<options>.*)"}.match(output)[:options]
|
40
|
+
assert_match %r{COLLATE=utf8mb4_bin}, options
|
41
|
+
end
|
42
|
+
end
|
@@ -1,66 +1,66 @@
|
|
1
|
-
require "cases/helper"
|
2
|
-
require "support/schema_dumping_helper"
|
3
|
-
|
4
|
-
class Mysql2UnsignedTypeTest < ActiveRecord::Mysql2TestCase
|
5
|
-
include SchemaDumpingHelper
|
6
|
-
self.use_transactional_tests = false
|
7
|
-
|
8
|
-
class UnsignedType < ActiveRecord::Base
|
9
|
-
end
|
10
|
-
|
11
|
-
setup do
|
12
|
-
@connection = ActiveRecord::Base.connection
|
13
|
-
@connection.create_table("unsigned_types", force: true) do |t|
|
14
|
-
t.integer :unsigned_integer, unsigned: true
|
15
|
-
t.bigint :unsigned_bigint, unsigned: true
|
16
|
-
t.float :unsigned_float, unsigned: true
|
17
|
-
t.decimal :unsigned_decimal, unsigned: true, precision: 10, scale: 2
|
18
|
-
end
|
19
|
-
end
|
20
|
-
|
21
|
-
teardown do
|
22
|
-
@connection.drop_table "unsigned_types", if_exists: true
|
23
|
-
end
|
24
|
-
|
25
|
-
test "unsigned int max value is in range" do
|
26
|
-
assert expected = UnsignedType.create(unsigned_integer: 4294967295)
|
27
|
-
assert_equal expected, UnsignedType.find_by(unsigned_integer: 4294967295)
|
28
|
-
end
|
29
|
-
|
30
|
-
test "minus value is out of range" do
|
31
|
-
assert_raise(ActiveModel::RangeError) do
|
32
|
-
UnsignedType.create(unsigned_integer: -10)
|
33
|
-
end
|
34
|
-
assert_raise(ActiveModel::RangeError) do
|
35
|
-
UnsignedType.create(unsigned_bigint: -10)
|
36
|
-
end
|
37
|
-
assert_raise(ActiveRecord::StatementInvalid) do
|
38
|
-
UnsignedType.create(unsigned_float: -10.0)
|
39
|
-
end
|
40
|
-
assert_raise(ActiveRecord::StatementInvalid) do
|
41
|
-
UnsignedType.create(unsigned_decimal: -10.0)
|
42
|
-
end
|
43
|
-
end
|
44
|
-
|
45
|
-
test "schema definition can use unsigned as the type" do
|
46
|
-
@connection.change_table("unsigned_types") do |t|
|
47
|
-
t.unsigned_integer :unsigned_integer_t
|
48
|
-
t.unsigned_bigint :unsigned_bigint_t
|
49
|
-
t.unsigned_float :unsigned_float_t
|
50
|
-
t.unsigned_decimal :unsigned_decimal_t, precision: 10, scale: 2
|
51
|
-
t.column :unsigned_zerofill, "int unsigned zerofill"
|
52
|
-
end
|
53
|
-
|
54
|
-
@connection.columns("unsigned_types").select { |c| /^unsigned_/ === c.name }.each do |column|
|
55
|
-
assert column.unsigned?
|
56
|
-
end
|
57
|
-
end
|
58
|
-
|
59
|
-
test "schema dump includes unsigned option" do
|
60
|
-
schema = dump_table_schema "unsigned_types"
|
61
|
-
assert_match %r{t.integer\s+"unsigned_integer",\s+unsigned: true$}, schema
|
62
|
-
assert_match %r{t.bigint\s+"unsigned_bigint",\s+unsigned: true$}, schema
|
63
|
-
assert_match %r{t.float\s+"unsigned_float",\s+limit: 24,\s+unsigned: true$}, schema
|
64
|
-
assert_match %r{t.decimal\s+"unsigned_decimal",\s+precision: 10,\s+scale: 2,\s+unsigned: true$}, schema
|
65
|
-
end
|
66
|
-
end
|
1
|
+
require "cases/helper"
|
2
|
+
require "support/schema_dumping_helper"
|
3
|
+
|
4
|
+
class Mysql2UnsignedTypeTest < ActiveRecord::Mysql2TestCase
|
5
|
+
include SchemaDumpingHelper
|
6
|
+
self.use_transactional_tests = false
|
7
|
+
|
8
|
+
class UnsignedType < ActiveRecord::Base
|
9
|
+
end
|
10
|
+
|
11
|
+
setup do
|
12
|
+
@connection = ActiveRecord::Base.connection
|
13
|
+
@connection.create_table("unsigned_types", force: true) do |t|
|
14
|
+
t.integer :unsigned_integer, unsigned: true
|
15
|
+
t.bigint :unsigned_bigint, unsigned: true
|
16
|
+
t.float :unsigned_float, unsigned: true
|
17
|
+
t.decimal :unsigned_decimal, unsigned: true, precision: 10, scale: 2
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
teardown do
|
22
|
+
@connection.drop_table "unsigned_types", if_exists: true
|
23
|
+
end
|
24
|
+
|
25
|
+
test "unsigned int max value is in range" do
|
26
|
+
assert expected = UnsignedType.create(unsigned_integer: 4294967295)
|
27
|
+
assert_equal expected, UnsignedType.find_by(unsigned_integer: 4294967295)
|
28
|
+
end
|
29
|
+
|
30
|
+
test "minus value is out of range" do
|
31
|
+
assert_raise(ActiveModel::RangeError) do
|
32
|
+
UnsignedType.create(unsigned_integer: -10)
|
33
|
+
end
|
34
|
+
assert_raise(ActiveModel::RangeError) do
|
35
|
+
UnsignedType.create(unsigned_bigint: -10)
|
36
|
+
end
|
37
|
+
assert_raise(ActiveRecord::StatementInvalid) do
|
38
|
+
UnsignedType.create(unsigned_float: -10.0)
|
39
|
+
end
|
40
|
+
assert_raise(ActiveRecord::StatementInvalid) do
|
41
|
+
UnsignedType.create(unsigned_decimal: -10.0)
|
42
|
+
end
|
43
|
+
end
|
44
|
+
|
45
|
+
test "schema definition can use unsigned as the type" do
|
46
|
+
@connection.change_table("unsigned_types") do |t|
|
47
|
+
t.unsigned_integer :unsigned_integer_t
|
48
|
+
t.unsigned_bigint :unsigned_bigint_t
|
49
|
+
t.unsigned_float :unsigned_float_t
|
50
|
+
t.unsigned_decimal :unsigned_decimal_t, precision: 10, scale: 2
|
51
|
+
t.column :unsigned_zerofill, "int unsigned zerofill"
|
52
|
+
end
|
53
|
+
|
54
|
+
@connection.columns("unsigned_types").select { |c| /^unsigned_/ === c.name }.each do |column|
|
55
|
+
assert column.unsigned?
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
test "schema dump includes unsigned option" do
|
60
|
+
schema = dump_table_schema "unsigned_types"
|
61
|
+
assert_match %r{t.integer\s+"unsigned_integer",\s+unsigned: true$}, schema
|
62
|
+
assert_match %r{t.bigint\s+"unsigned_bigint",\s+unsigned: true$}, schema
|
63
|
+
assert_match %r{t.float\s+"unsigned_float",\s+limit: 24,\s+unsigned: true$}, schema
|
64
|
+
assert_match %r{t.decimal\s+"unsigned_decimal",\s+precision: 10,\s+scale: 2,\s+unsigned: true$}, schema
|
65
|
+
end
|
66
|
+
end
|
@@ -1,98 +1,98 @@
|
|
1
|
-
require 'cases/helper'
|
2
|
-
|
3
|
-
class PostgresqlActiveSchemaTest < ActiveRecord::PostgreSQLTestCase
|
4
|
-
def setup
|
5
|
-
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.class_eval do
|
6
|
-
def execute(sql, name = nil) sql end
|
7
|
-
end
|
8
|
-
end
|
9
|
-
|
10
|
-
teardown do
|
11
|
-
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.class_eval do
|
12
|
-
remove_method :execute
|
13
|
-
end
|
14
|
-
end
|
15
|
-
|
16
|
-
def test_create_database_with_encoding
|
17
|
-
assert_equal %(CREATE DATABASE "matt" ENCODING = 'utf8'), create_database(:matt)
|
18
|
-
assert_equal %(CREATE DATABASE "aimonetti" ENCODING = 'latin1'), create_database(:aimonetti, :encoding => :latin1)
|
19
|
-
assert_equal %(CREATE DATABASE "aimonetti" ENCODING = 'latin1'), create_database(:aimonetti, 'encoding' => :latin1)
|
20
|
-
end
|
21
|
-
|
22
|
-
def test_create_database_with_collation_and_ctype
|
23
|
-
assert_equal %(CREATE DATABASE "aimonetti" ENCODING = 'UTF8' LC_COLLATE = 'ja_JP.UTF8' LC_CTYPE = 'ja_JP.UTF8'), create_database(:aimonetti, :encoding => :"UTF8", :collation => :"ja_JP.UTF8", :ctype => :"ja_JP.UTF8")
|
24
|
-
end
|
25
|
-
|
26
|
-
def test_add_index
|
27
|
-
# add_index calls index_name_exists? which can't work since execute is stubbed
|
28
|
-
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.send(:define_method, :index_name_exists?) { |*| false }
|
29
|
-
|
30
|
-
expected = %(CREATE UNIQUE INDEX "index_people_on_last_name" ON "people" ("last_name") WHERE state = 'active')
|
31
|
-
assert_equal expected, add_index(:people, :last_name, unique: true, where: "state = 'active'")
|
32
|
-
|
33
|
-
expected = %(CREATE UNIQUE INDEX "index_people_on_lower_last_name" ON "people" (lower(last_name)))
|
34
|
-
assert_equal expected, add_index(:people, 'lower(last_name)', unique: true)
|
35
|
-
|
36
|
-
expected = %(CREATE UNIQUE INDEX "index_people_on_last_name_varchar_pattern_ops" ON "people" (last_name varchar_pattern_ops))
|
37
|
-
assert_equal expected, add_index(:people, 'last_name varchar_pattern_ops', unique: true)
|
38
|
-
|
39
|
-
expected = %(CREATE INDEX CONCURRENTLY "index_people_on_last_name" ON "people" ("last_name"))
|
40
|
-
assert_equal expected, add_index(:people, :last_name, algorithm: :concurrently)
|
41
|
-
|
42
|
-
expected = %(CREATE INDEX "index_people_on_last_name_and_first_name" ON "people" ("last_name" DESC, "first_name" ASC))
|
43
|
-
assert_equal expected, add_index(:people, [:last_name, :first_name], order: { last_name: :desc, first_name: :asc })
|
44
|
-
assert_equal expected, add_index(:people, ["last_name", :first_name], order: { last_name: :desc, "first_name" => :asc })
|
45
|
-
|
46
|
-
%w(gin gist hash btree).each do |type|
|
47
|
-
expected = %(CREATE INDEX "index_people_on_last_name" ON "people" USING #{type} ("last_name"))
|
48
|
-
assert_equal expected, add_index(:people, :last_name, using: type)
|
49
|
-
|
50
|
-
expected = %(CREATE INDEX CONCURRENTLY "index_people_on_last_name" ON "people" USING #{type} ("last_name"))
|
51
|
-
assert_equal expected, add_index(:people, :last_name, using: type, algorithm: :concurrently)
|
52
|
-
|
53
|
-
expected = %(CREATE UNIQUE INDEX "index_people_on_last_name" ON "people" USING #{type} ("last_name") WHERE state = 'active')
|
54
|
-
assert_equal expected, add_index(:people, :last_name, using: type, unique: true, where: "state = 'active'")
|
55
|
-
|
56
|
-
expected = %(CREATE UNIQUE INDEX "index_people_on_lower_last_name" ON "people" USING #{type} (lower(last_name)))
|
57
|
-
assert_equal expected, add_index(:people, 'lower(last_name)', using: type, unique: true)
|
58
|
-
end
|
59
|
-
|
60
|
-
assert_raise ArgumentError do
|
61
|
-
add_index(:people, :last_name, algorithm: :copy)
|
62
|
-
end
|
63
|
-
|
64
|
-
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.send :remove_method, :index_name_exists?
|
65
|
-
end
|
66
|
-
|
67
|
-
def test_remove_index
|
68
|
-
# remove_index calls index_name_for_remove which can't work since execute is stubbed
|
69
|
-
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.send(:define_method, :index_name_for_remove) do |*|
|
70
|
-
'index_people_on_last_name'
|
71
|
-
end
|
72
|
-
|
73
|
-
expected = %(DROP INDEX CONCURRENTLY "index_people_on_last_name")
|
74
|
-
assert_equal expected, remove_index(:people, name: "index_people_on_last_name", algorithm: :concurrently)
|
75
|
-
|
76
|
-
assert_raise ArgumentError do
|
77
|
-
add_index(:people, :last_name, algorithm: :copy)
|
78
|
-
end
|
79
|
-
|
80
|
-
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.send :remove_method, :index_name_for_remove
|
81
|
-
end
|
82
|
-
|
83
|
-
def test_remove_index_when_name_is_specified
|
84
|
-
expected = %(DROP INDEX CONCURRENTLY "index_people_on_last_name")
|
85
|
-
assert_equal expected, remove_index(:people, name: "index_people_on_last_name", algorithm: :concurrently)
|
86
|
-
end
|
87
|
-
|
88
|
-
def test_remove_index_with_wrong_option
|
89
|
-
assert_raises ArgumentError do
|
90
|
-
remove_index(:people, coulmn: :last_name)
|
91
|
-
end
|
92
|
-
end
|
93
|
-
|
94
|
-
private
|
95
|
-
def method_missing(method_symbol, *arguments)
|
96
|
-
ActiveRecord::Base.connection.send(method_symbol, *arguments)
|
97
|
-
end
|
98
|
-
end
|
1
|
+
require 'cases/helper'
|
2
|
+
|
3
|
+
class PostgresqlActiveSchemaTest < ActiveRecord::PostgreSQLTestCase
|
4
|
+
def setup
|
5
|
+
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.class_eval do
|
6
|
+
def execute(sql, name = nil) sql end
|
7
|
+
end
|
8
|
+
end
|
9
|
+
|
10
|
+
teardown do
|
11
|
+
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.class_eval do
|
12
|
+
remove_method :execute
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
16
|
+
def test_create_database_with_encoding
|
17
|
+
assert_equal %(CREATE DATABASE "matt" ENCODING = 'utf8'), create_database(:matt)
|
18
|
+
assert_equal %(CREATE DATABASE "aimonetti" ENCODING = 'latin1'), create_database(:aimonetti, :encoding => :latin1)
|
19
|
+
assert_equal %(CREATE DATABASE "aimonetti" ENCODING = 'latin1'), create_database(:aimonetti, 'encoding' => :latin1)
|
20
|
+
end
|
21
|
+
|
22
|
+
def test_create_database_with_collation_and_ctype
|
23
|
+
assert_equal %(CREATE DATABASE "aimonetti" ENCODING = 'UTF8' LC_COLLATE = 'ja_JP.UTF8' LC_CTYPE = 'ja_JP.UTF8'), create_database(:aimonetti, :encoding => :"UTF8", :collation => :"ja_JP.UTF8", :ctype => :"ja_JP.UTF8")
|
24
|
+
end
|
25
|
+
|
26
|
+
def test_add_index
|
27
|
+
# add_index calls index_name_exists? which can't work since execute is stubbed
|
28
|
+
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.send(:define_method, :index_name_exists?) { |*| false }
|
29
|
+
|
30
|
+
expected = %(CREATE UNIQUE INDEX "index_people_on_last_name" ON "people" ("last_name") WHERE state = 'active')
|
31
|
+
assert_equal expected, add_index(:people, :last_name, unique: true, where: "state = 'active'")
|
32
|
+
|
33
|
+
expected = %(CREATE UNIQUE INDEX "index_people_on_lower_last_name" ON "people" (lower(last_name)))
|
34
|
+
assert_equal expected, add_index(:people, 'lower(last_name)', unique: true)
|
35
|
+
|
36
|
+
expected = %(CREATE UNIQUE INDEX "index_people_on_last_name_varchar_pattern_ops" ON "people" (last_name varchar_pattern_ops))
|
37
|
+
assert_equal expected, add_index(:people, 'last_name varchar_pattern_ops', unique: true)
|
38
|
+
|
39
|
+
expected = %(CREATE INDEX CONCURRENTLY "index_people_on_last_name" ON "people" ("last_name"))
|
40
|
+
assert_equal expected, add_index(:people, :last_name, algorithm: :concurrently)
|
41
|
+
|
42
|
+
expected = %(CREATE INDEX "index_people_on_last_name_and_first_name" ON "people" ("last_name" DESC, "first_name" ASC))
|
43
|
+
assert_equal expected, add_index(:people, [:last_name, :first_name], order: { last_name: :desc, first_name: :asc })
|
44
|
+
assert_equal expected, add_index(:people, ["last_name", :first_name], order: { last_name: :desc, "first_name" => :asc })
|
45
|
+
|
46
|
+
%w(gin gist hash btree).each do |type|
|
47
|
+
expected = %(CREATE INDEX "index_people_on_last_name" ON "people" USING #{type} ("last_name"))
|
48
|
+
assert_equal expected, add_index(:people, :last_name, using: type)
|
49
|
+
|
50
|
+
expected = %(CREATE INDEX CONCURRENTLY "index_people_on_last_name" ON "people" USING #{type} ("last_name"))
|
51
|
+
assert_equal expected, add_index(:people, :last_name, using: type, algorithm: :concurrently)
|
52
|
+
|
53
|
+
expected = %(CREATE UNIQUE INDEX "index_people_on_last_name" ON "people" USING #{type} ("last_name") WHERE state = 'active')
|
54
|
+
assert_equal expected, add_index(:people, :last_name, using: type, unique: true, where: "state = 'active'")
|
55
|
+
|
56
|
+
expected = %(CREATE UNIQUE INDEX "index_people_on_lower_last_name" ON "people" USING #{type} (lower(last_name)))
|
57
|
+
assert_equal expected, add_index(:people, 'lower(last_name)', using: type, unique: true)
|
58
|
+
end
|
59
|
+
|
60
|
+
assert_raise ArgumentError do
|
61
|
+
add_index(:people, :last_name, algorithm: :copy)
|
62
|
+
end
|
63
|
+
|
64
|
+
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.send :remove_method, :index_name_exists?
|
65
|
+
end
|
66
|
+
|
67
|
+
def test_remove_index
|
68
|
+
# remove_index calls index_name_for_remove which can't work since execute is stubbed
|
69
|
+
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.send(:define_method, :index_name_for_remove) do |*|
|
70
|
+
'index_people_on_last_name'
|
71
|
+
end
|
72
|
+
|
73
|
+
expected = %(DROP INDEX CONCURRENTLY "index_people_on_last_name")
|
74
|
+
assert_equal expected, remove_index(:people, name: "index_people_on_last_name", algorithm: :concurrently)
|
75
|
+
|
76
|
+
assert_raise ArgumentError do
|
77
|
+
add_index(:people, :last_name, algorithm: :copy)
|
78
|
+
end
|
79
|
+
|
80
|
+
ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.send :remove_method, :index_name_for_remove
|
81
|
+
end
|
82
|
+
|
83
|
+
def test_remove_index_when_name_is_specified
|
84
|
+
expected = %(DROP INDEX CONCURRENTLY "index_people_on_last_name")
|
85
|
+
assert_equal expected, remove_index(:people, name: "index_people_on_last_name", algorithm: :concurrently)
|
86
|
+
end
|
87
|
+
|
88
|
+
def test_remove_index_with_wrong_option
|
89
|
+
assert_raises ArgumentError do
|
90
|
+
remove_index(:people, coulmn: :last_name)
|
91
|
+
end
|
92
|
+
end
|
93
|
+
|
94
|
+
private
|
95
|
+
def method_missing(method_symbol, *arguments)
|
96
|
+
ActiveRecord::Base.connection.send(method_symbol, *arguments)
|
97
|
+
end
|
98
|
+
end
|
@@ -1,339 +1,339 @@
|
|
1
|
-
require "cases/helper"
|
2
|
-
require 'support/schema_dumping_helper'
|
3
|
-
|
4
|
-
class PostgresqlArrayTest < ActiveRecord::PostgreSQLTestCase
|
5
|
-
include SchemaDumpingHelper
|
6
|
-
include InTimeZone
|
7
|
-
|
8
|
-
class PgArray < ActiveRecord::Base
|
9
|
-
self.table_name = 'pg_arrays'
|
10
|
-
end
|
11
|
-
|
12
|
-
def setup
|
13
|
-
@connection = ActiveRecord::Base.connection
|
14
|
-
|
15
|
-
enable_extension!('hstore', @connection)
|
16
|
-
|
17
|
-
@connection.transaction do
|
18
|
-
@connection.create_table('pg_arrays') do |t|
|
19
|
-
t.string 'tags', array: true
|
20
|
-
t.integer 'ratings', array: true
|
21
|
-
t.datetime :datetimes, array: true
|
22
|
-
t.hstore :hstores, array: true
|
23
|
-
t.timestamp :timestamps, array: true, default: [], precision: 6
|
24
|
-
end
|
25
|
-
end
|
26
|
-
PgArray.reset_column_information
|
27
|
-
@column = PgArray.columns_hash['tags']
|
28
|
-
@type = PgArray.type_for_attribute("tags")
|
29
|
-
end
|
30
|
-
|
31
|
-
teardown do
|
32
|
-
@connection.drop_table 'pg_arrays', if_exists: true
|
33
|
-
disable_extension!('hstore', @connection)
|
34
|
-
end
|
35
|
-
|
36
|
-
def test_column
|
37
|
-
assert_equal :string, @column.type
|
38
|
-
assert_equal "character varying", @column.sql_type
|
39
|
-
assert @column.array?
|
40
|
-
assert_not @type.binary?
|
41
|
-
|
42
|
-
ratings_column = PgArray.columns_hash['ratings']
|
43
|
-
assert_equal :integer, ratings_column.type
|
44
|
-
assert ratings_column.array?
|
45
|
-
end
|
46
|
-
|
47
|
-
def test_default
|
48
|
-
@connection.add_column 'pg_arrays', 'score', :integer, array: true, default: [4, 4, 2]
|
49
|
-
PgArray.reset_column_information
|
50
|
-
|
51
|
-
assert_equal([4, 4, 2], PgArray.column_defaults['score'])
|
52
|
-
assert_equal([4, 4, 2], PgArray.new.score)
|
53
|
-
ensure
|
54
|
-
PgArray.reset_column_information
|
55
|
-
end
|
56
|
-
|
57
|
-
def test_default_strings
|
58
|
-
@connection.add_column 'pg_arrays', 'names', :string, array: true, default: ["foo", "bar"]
|
59
|
-
PgArray.reset_column_information
|
60
|
-
|
61
|
-
assert_equal(["foo", "bar"], PgArray.column_defaults['names'])
|
62
|
-
assert_equal(["foo", "bar"], PgArray.new.names)
|
63
|
-
ensure
|
64
|
-
PgArray.reset_column_information
|
65
|
-
end
|
66
|
-
|
67
|
-
def test_change_column_with_array
|
68
|
-
@connection.add_column :pg_arrays, :snippets, :string, array: true, default: []
|
69
|
-
@connection.change_column :pg_arrays, :snippets, :text, array: true, default: []
|
70
|
-
|
71
|
-
PgArray.reset_column_information
|
72
|
-
column = PgArray.columns_hash['snippets']
|
73
|
-
|
74
|
-
assert_equal :text, column.type
|
75
|
-
assert_equal [], PgArray.column_defaults['snippets']
|
76
|
-
assert column.array?
|
77
|
-
end
|
78
|
-
|
79
|
-
def test_change_column_cant_make_non_array_column_to_array
|
80
|
-
@connection.add_column :pg_arrays, :a_string, :string
|
81
|
-
assert_raises ActiveRecord::StatementInvalid do
|
82
|
-
@connection.transaction do
|
83
|
-
@connection.change_column :pg_arrays, :a_string, :string, array: true
|
84
|
-
end
|
85
|
-
end
|
86
|
-
end
|
87
|
-
|
88
|
-
def test_change_column_default_with_array
|
89
|
-
@connection.change_column_default :pg_arrays, :tags, []
|
90
|
-
|
91
|
-
PgArray.reset_column_information
|
92
|
-
assert_equal [], PgArray.column_defaults['tags']
|
93
|
-
end
|
94
|
-
|
95
|
-
def test_type_cast_array
|
96
|
-
assert_equal(['1', '2', '3'], @type.deserialize('{1,2,3}'))
|
97
|
-
assert_equal([], @type.deserialize('{}'))
|
98
|
-
assert_equal([nil], @type.deserialize('{NULL}'))
|
99
|
-
end
|
100
|
-
|
101
|
-
def test_type_cast_integers
|
102
|
-
x = PgArray.new(ratings: ['1', '2'])
|
103
|
-
|
104
|
-
assert_equal([1, 2], x.ratings)
|
105
|
-
|
106
|
-
x.save!
|
107
|
-
x.reload
|
108
|
-
|
109
|
-
assert_equal([1, 2], x.ratings)
|
110
|
-
end
|
111
|
-
|
112
|
-
def test_schema_dump_with_shorthand
|
113
|
-
output = dump_table_schema "pg_arrays"
|
114
|
-
assert_match %r[t\.string\s+"tags",\s+array: true], output
|
115
|
-
assert_match %r[t\.integer\s+"ratings",\s+array: true], output
|
116
|
-
end
|
117
|
-
|
118
|
-
def test_select_with_strings
|
119
|
-
@connection.execute "insert into pg_arrays (tags) VALUES ('{1,2,3}')"
|
120
|
-
x = PgArray.first
|
121
|
-
assert_equal(['1','2','3'], x.tags)
|
122
|
-
end
|
123
|
-
|
124
|
-
def test_rewrite_with_strings
|
125
|
-
@connection.execute "insert into pg_arrays (tags) VALUES ('{1,2,3}')"
|
126
|
-
x = PgArray.first
|
127
|
-
x.tags = ['1','2','3','4']
|
128
|
-
x.save!
|
129
|
-
assert_equal ['1','2','3','4'], x.reload.tags
|
130
|
-
end
|
131
|
-
|
132
|
-
def test_select_with_integers
|
133
|
-
@connection.execute "insert into pg_arrays (ratings) VALUES ('{1,2,3}')"
|
134
|
-
x = PgArray.first
|
135
|
-
assert_equal([1, 2, 3], x.ratings)
|
136
|
-
end
|
137
|
-
|
138
|
-
def test_rewrite_with_integers
|
139
|
-
@connection.execute "insert into pg_arrays (ratings) VALUES ('{1,2,3}')"
|
140
|
-
x = PgArray.first
|
141
|
-
x.ratings = [2, '3', 4]
|
142
|
-
x.save!
|
143
|
-
assert_equal [2, 3, 4], x.reload.ratings
|
144
|
-
end
|
145
|
-
|
146
|
-
def test_multi_dimensional_with_strings
|
147
|
-
assert_cycle(:tags, [[['1'], ['2']], [['2'], ['3']]])
|
148
|
-
end
|
149
|
-
|
150
|
-
def test_with_empty_strings
|
151
|
-
assert_cycle(:tags, [ '1', '2', '', '4', '', '5' ])
|
152
|
-
end
|
153
|
-
|
154
|
-
def test_with_multi_dimensional_empty_strings
|
155
|
-
assert_cycle(:tags, [[['1', '2'], ['', '4'], ['', '5']]])
|
156
|
-
end
|
157
|
-
|
158
|
-
def test_with_arbitrary_whitespace
|
159
|
-
assert_cycle(:tags, [[['1', '2'], [' ', '4'], [' ', '5']]])
|
160
|
-
end
|
161
|
-
|
162
|
-
def test_multi_dimensional_with_integers
|
163
|
-
assert_cycle(:ratings, [[[1], [7]], [[8], [10]]])
|
164
|
-
end
|
165
|
-
|
166
|
-
def test_strings_with_quotes
|
167
|
-
assert_cycle(:tags, ['this has','some "s that need to be escaped"'])
|
168
|
-
end
|
169
|
-
|
170
|
-
def test_strings_with_commas
|
171
|
-
assert_cycle(:tags, ['this,has','many,values'])
|
172
|
-
end
|
173
|
-
|
174
|
-
def test_strings_with_array_delimiters
|
175
|
-
assert_cycle(:tags, ['{','}'])
|
176
|
-
end
|
177
|
-
|
178
|
-
def test_strings_with_null_strings
|
179
|
-
assert_cycle(:tags, ['NULL','NULL'])
|
180
|
-
end
|
181
|
-
|
182
|
-
def test_contains_nils
|
183
|
-
assert_cycle(:tags, ['1',nil,nil])
|
184
|
-
end
|
185
|
-
|
186
|
-
def test_insert_fixture
|
187
|
-
tag_values = ["val1", "val2", "val3_with_'_multiple_quote_'_chars"]
|
188
|
-
@connection.insert_fixture({"tags" => tag_values}, "pg_arrays" )
|
189
|
-
assert_equal(PgArray.last.tags, tag_values)
|
190
|
-
end
|
191
|
-
|
192
|
-
def test_attribute_for_inspect_for_array_field
|
193
|
-
record = PgArray.new { |a| a.ratings = (1..10).to_a }
|
194
|
-
assert_equal("[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]", record.attribute_for_inspect(:ratings))
|
195
|
-
end
|
196
|
-
|
197
|
-
def test_attribute_for_inspect_for_array_field_for_large_array
|
198
|
-
record = PgArray.new { |a| a.ratings = (1..11).to_a }
|
199
|
-
assert_equal("[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]", record.attribute_for_inspect(:ratings))
|
200
|
-
end
|
201
|
-
|
202
|
-
def test_escaping
|
203
|
-
unknown = 'foo\\",bar,baz,\\'
|
204
|
-
tags = ["hello_#{unknown}"]
|
205
|
-
ar = PgArray.create!(tags: tags)
|
206
|
-
ar.reload
|
207
|
-
assert_equal tags, ar.tags
|
208
|
-
end
|
209
|
-
|
210
|
-
def test_string_quoting_rules_match_pg_behavior
|
211
|
-
tags = ["", "one{", "two}", %(three"), "four\\", "five ", "six\t", "seven\n", "eight,", "nine", "ten\r", "NULL"]
|
212
|
-
x = PgArray.create!(tags: tags)
|
213
|
-
x.reload
|
214
|
-
|
215
|
-
refute x.changed?
|
216
|
-
end
|
217
|
-
|
218
|
-
def test_quoting_non_standard_delimiters
|
219
|
-
strings = ["hello,", "world;"]
|
220
|
-
oid = ActiveRecord::ConnectionAdapters::PostgreSQL::OID
|
221
|
-
comma_delim = oid::Array.new(ActiveRecord::Type::String.new, ',')
|
222
|
-
semicolon_delim = oid::Array.new(ActiveRecord::Type::String.new, ';')
|
223
|
-
conn = PgArray.connection
|
224
|
-
|
225
|
-
assert_equal %({"hello,",world;}), conn.type_cast(comma_delim.serialize(strings))
|
226
|
-
assert_equal %({hello,;"world;"}), conn.type_cast(semicolon_delim.serialize(strings))
|
227
|
-
end
|
228
|
-
|
229
|
-
def test_mutate_array
|
230
|
-
x = PgArray.create!(tags: %w(one two))
|
231
|
-
|
232
|
-
x.tags << "three"
|
233
|
-
x.save!
|
234
|
-
x.reload
|
235
|
-
|
236
|
-
assert_equal %w(one two three), x.tags
|
237
|
-
assert_not x.changed?
|
238
|
-
end
|
239
|
-
|
240
|
-
def test_mutate_value_in_array
|
241
|
-
x = PgArray.create!(hstores: [{ a: 'a' }, { b: 'b' }])
|
242
|
-
|
243
|
-
x.hstores.first['a'] = 'c'
|
244
|
-
x.save!
|
245
|
-
x.reload
|
246
|
-
|
247
|
-
assert_equal [{ 'a' => 'c' }, { 'b' => 'b' }], x.hstores
|
248
|
-
assert_not x.changed?
|
249
|
-
end
|
250
|
-
|
251
|
-
def test_datetime_with_timezone_awareness
|
252
|
-
tz = "Pacific Time (US & Canada)"
|
253
|
-
|
254
|
-
in_time_zone tz do
|
255
|
-
PgArray.reset_column_information
|
256
|
-
time_string = Time.current.to_s
|
257
|
-
time = Time.zone.parse(time_string)
|
258
|
-
|
259
|
-
record = PgArray.new(datetimes: [time_string])
|
260
|
-
assert_equal [time], record.datetimes
|
261
|
-
assert_equal ActiveSupport::TimeZone[tz], record.datetimes.first.time_zone
|
262
|
-
|
263
|
-
record.save!
|
264
|
-
record.reload
|
265
|
-
|
266
|
-
assert_equal [time], record.datetimes
|
267
|
-
assert_equal ActiveSupport::TimeZone[tz], record.datetimes.first.time_zone
|
268
|
-
end
|
269
|
-
end
|
270
|
-
|
271
|
-
def test_assigning_non_array_value
|
272
|
-
record = PgArray.new(tags: "not-an-array")
|
273
|
-
assert_equal [], record.tags
|
274
|
-
assert_equal "not-an-array", record.tags_before_type_cast
|
275
|
-
assert record.save
|
276
|
-
assert_equal record.tags, record.reload.tags
|
277
|
-
end
|
278
|
-
|
279
|
-
def test_assigning_empty_string
|
280
|
-
record = PgArray.new(tags: "")
|
281
|
-
assert_equal [], record.tags
|
282
|
-
assert_equal "", record.tags_before_type_cast
|
283
|
-
assert record.save
|
284
|
-
assert_equal record.tags, record.reload.tags
|
285
|
-
end
|
286
|
-
|
287
|
-
def test_assigning_valid_pg_array_literal
|
288
|
-
record = PgArray.new(tags: "{1,2,3}")
|
289
|
-
assert_equal ["1", "2", "3"], record.tags
|
290
|
-
assert_equal "{1,2,3}", record.tags_before_type_cast
|
291
|
-
assert record.save
|
292
|
-
assert_equal record.tags, record.reload.tags
|
293
|
-
end
|
294
|
-
|
295
|
-
def test_uniqueness_validation
|
296
|
-
klass = Class.new(PgArray) do
|
297
|
-
validates_uniqueness_of :tags
|
298
|
-
|
299
|
-
def self.model_name; ActiveModel::Name.new(PgArray) end
|
300
|
-
end
|
301
|
-
e1 = klass.create("tags" => ["black", "blue"])
|
302
|
-
assert e1.persisted?, "Saving e1"
|
303
|
-
|
304
|
-
e2 = klass.create("tags" => ["black", "blue"])
|
305
|
-
assert !e2.persisted?, "e2 shouldn't be valid"
|
306
|
-
assert e2.errors[:tags].any?, "Should have errors for tags"
|
307
|
-
assert_equal ["has already been taken"], e2.errors[:tags], "Should have uniqueness message for tags"
|
308
|
-
end
|
309
|
-
|
310
|
-
def test_encoding_arrays_of_utf8_strings
|
311
|
-
arrays_of_utf8_strings = %w(nový ファイル)
|
312
|
-
assert_equal arrays_of_utf8_strings, @type.deserialize(@type.serialize(arrays_of_utf8_strings))
|
313
|
-
assert_equal [arrays_of_utf8_strings], @type.deserialize(@type.serialize([arrays_of_utf8_strings]))
|
314
|
-
end
|
315
|
-
|
316
|
-
def test_precision_is_respected_on_timestamp_columns
|
317
|
-
time = Time.now.change(usec: 123)
|
318
|
-
record = PgArray.create!(timestamps: [time])
|
319
|
-
|
320
|
-
assert_equal 123, record.timestamps.first.usec
|
321
|
-
record.reload
|
322
|
-
assert_equal 123, record.timestamps.first.usec
|
323
|
-
end
|
324
|
-
|
325
|
-
private
|
326
|
-
def assert_cycle field, array
|
327
|
-
# test creation
|
328
|
-
x = PgArray.create!(field => array)
|
329
|
-
x.reload
|
330
|
-
assert_equal(array, x.public_send(field))
|
331
|
-
|
332
|
-
# test updating
|
333
|
-
x = PgArray.create!(field => [])
|
334
|
-
x.public_send("#{field}=", array)
|
335
|
-
x.save!
|
336
|
-
x.reload
|
337
|
-
assert_equal(array, x.public_send(field))
|
338
|
-
end
|
339
|
-
end
|
1
|
+
require "cases/helper"
|
2
|
+
require 'support/schema_dumping_helper'
|
3
|
+
|
4
|
+
class PostgresqlArrayTest < ActiveRecord::PostgreSQLTestCase
|
5
|
+
include SchemaDumpingHelper
|
6
|
+
include InTimeZone
|
7
|
+
|
8
|
+
class PgArray < ActiveRecord::Base
|
9
|
+
self.table_name = 'pg_arrays'
|
10
|
+
end
|
11
|
+
|
12
|
+
def setup
|
13
|
+
@connection = ActiveRecord::Base.connection
|
14
|
+
|
15
|
+
enable_extension!('hstore', @connection)
|
16
|
+
|
17
|
+
@connection.transaction do
|
18
|
+
@connection.create_table('pg_arrays') do |t|
|
19
|
+
t.string 'tags', array: true
|
20
|
+
t.integer 'ratings', array: true
|
21
|
+
t.datetime :datetimes, array: true
|
22
|
+
t.hstore :hstores, array: true
|
23
|
+
t.timestamp :timestamps, array: true, default: [], precision: 6
|
24
|
+
end
|
25
|
+
end
|
26
|
+
PgArray.reset_column_information
|
27
|
+
@column = PgArray.columns_hash['tags']
|
28
|
+
@type = PgArray.type_for_attribute("tags")
|
29
|
+
end
|
30
|
+
|
31
|
+
teardown do
|
32
|
+
@connection.drop_table 'pg_arrays', if_exists: true
|
33
|
+
disable_extension!('hstore', @connection)
|
34
|
+
end
|
35
|
+
|
36
|
+
def test_column
|
37
|
+
assert_equal :string, @column.type
|
38
|
+
assert_equal "character varying", @column.sql_type
|
39
|
+
assert @column.array?
|
40
|
+
assert_not @type.binary?
|
41
|
+
|
42
|
+
ratings_column = PgArray.columns_hash['ratings']
|
43
|
+
assert_equal :integer, ratings_column.type
|
44
|
+
assert ratings_column.array?
|
45
|
+
end
|
46
|
+
|
47
|
+
def test_default
|
48
|
+
@connection.add_column 'pg_arrays', 'score', :integer, array: true, default: [4, 4, 2]
|
49
|
+
PgArray.reset_column_information
|
50
|
+
|
51
|
+
assert_equal([4, 4, 2], PgArray.column_defaults['score'])
|
52
|
+
assert_equal([4, 4, 2], PgArray.new.score)
|
53
|
+
ensure
|
54
|
+
PgArray.reset_column_information
|
55
|
+
end
|
56
|
+
|
57
|
+
def test_default_strings
|
58
|
+
@connection.add_column 'pg_arrays', 'names', :string, array: true, default: ["foo", "bar"]
|
59
|
+
PgArray.reset_column_information
|
60
|
+
|
61
|
+
assert_equal(["foo", "bar"], PgArray.column_defaults['names'])
|
62
|
+
assert_equal(["foo", "bar"], PgArray.new.names)
|
63
|
+
ensure
|
64
|
+
PgArray.reset_column_information
|
65
|
+
end
|
66
|
+
|
67
|
+
def test_change_column_with_array
|
68
|
+
@connection.add_column :pg_arrays, :snippets, :string, array: true, default: []
|
69
|
+
@connection.change_column :pg_arrays, :snippets, :text, array: true, default: []
|
70
|
+
|
71
|
+
PgArray.reset_column_information
|
72
|
+
column = PgArray.columns_hash['snippets']
|
73
|
+
|
74
|
+
assert_equal :text, column.type
|
75
|
+
assert_equal [], PgArray.column_defaults['snippets']
|
76
|
+
assert column.array?
|
77
|
+
end
|
78
|
+
|
79
|
+
def test_change_column_cant_make_non_array_column_to_array
|
80
|
+
@connection.add_column :pg_arrays, :a_string, :string
|
81
|
+
assert_raises ActiveRecord::StatementInvalid do
|
82
|
+
@connection.transaction do
|
83
|
+
@connection.change_column :pg_arrays, :a_string, :string, array: true
|
84
|
+
end
|
85
|
+
end
|
86
|
+
end
|
87
|
+
|
88
|
+
def test_change_column_default_with_array
|
89
|
+
@connection.change_column_default :pg_arrays, :tags, []
|
90
|
+
|
91
|
+
PgArray.reset_column_information
|
92
|
+
assert_equal [], PgArray.column_defaults['tags']
|
93
|
+
end
|
94
|
+
|
95
|
+
def test_type_cast_array
|
96
|
+
assert_equal(['1', '2', '3'], @type.deserialize('{1,2,3}'))
|
97
|
+
assert_equal([], @type.deserialize('{}'))
|
98
|
+
assert_equal([nil], @type.deserialize('{NULL}'))
|
99
|
+
end
|
100
|
+
|
101
|
+
def test_type_cast_integers
|
102
|
+
x = PgArray.new(ratings: ['1', '2'])
|
103
|
+
|
104
|
+
assert_equal([1, 2], x.ratings)
|
105
|
+
|
106
|
+
x.save!
|
107
|
+
x.reload
|
108
|
+
|
109
|
+
assert_equal([1, 2], x.ratings)
|
110
|
+
end
|
111
|
+
|
112
|
+
def test_schema_dump_with_shorthand
|
113
|
+
output = dump_table_schema "pg_arrays"
|
114
|
+
assert_match %r[t\.string\s+"tags",\s+array: true], output
|
115
|
+
assert_match %r[t\.integer\s+"ratings",\s+array: true], output
|
116
|
+
end
|
117
|
+
|
118
|
+
def test_select_with_strings
|
119
|
+
@connection.execute "insert into pg_arrays (tags) VALUES ('{1,2,3}')"
|
120
|
+
x = PgArray.first
|
121
|
+
assert_equal(['1','2','3'], x.tags)
|
122
|
+
end
|
123
|
+
|
124
|
+
def test_rewrite_with_strings
|
125
|
+
@connection.execute "insert into pg_arrays (tags) VALUES ('{1,2,3}')"
|
126
|
+
x = PgArray.first
|
127
|
+
x.tags = ['1','2','3','4']
|
128
|
+
x.save!
|
129
|
+
assert_equal ['1','2','3','4'], x.reload.tags
|
130
|
+
end
|
131
|
+
|
132
|
+
def test_select_with_integers
|
133
|
+
@connection.execute "insert into pg_arrays (ratings) VALUES ('{1,2,3}')"
|
134
|
+
x = PgArray.first
|
135
|
+
assert_equal([1, 2, 3], x.ratings)
|
136
|
+
end
|
137
|
+
|
138
|
+
def test_rewrite_with_integers
|
139
|
+
@connection.execute "insert into pg_arrays (ratings) VALUES ('{1,2,3}')"
|
140
|
+
x = PgArray.first
|
141
|
+
x.ratings = [2, '3', 4]
|
142
|
+
x.save!
|
143
|
+
assert_equal [2, 3, 4], x.reload.ratings
|
144
|
+
end
|
145
|
+
|
146
|
+
def test_multi_dimensional_with_strings
|
147
|
+
assert_cycle(:tags, [[['1'], ['2']], [['2'], ['3']]])
|
148
|
+
end
|
149
|
+
|
150
|
+
def test_with_empty_strings
|
151
|
+
assert_cycle(:tags, [ '1', '2', '', '4', '', '5' ])
|
152
|
+
end
|
153
|
+
|
154
|
+
def test_with_multi_dimensional_empty_strings
|
155
|
+
assert_cycle(:tags, [[['1', '2'], ['', '4'], ['', '5']]])
|
156
|
+
end
|
157
|
+
|
158
|
+
def test_with_arbitrary_whitespace
|
159
|
+
assert_cycle(:tags, [[['1', '2'], [' ', '4'], [' ', '5']]])
|
160
|
+
end
|
161
|
+
|
162
|
+
def test_multi_dimensional_with_integers
|
163
|
+
assert_cycle(:ratings, [[[1], [7]], [[8], [10]]])
|
164
|
+
end
|
165
|
+
|
166
|
+
def test_strings_with_quotes
|
167
|
+
assert_cycle(:tags, ['this has','some "s that need to be escaped"'])
|
168
|
+
end
|
169
|
+
|
170
|
+
def test_strings_with_commas
|
171
|
+
assert_cycle(:tags, ['this,has','many,values'])
|
172
|
+
end
|
173
|
+
|
174
|
+
def test_strings_with_array_delimiters
|
175
|
+
assert_cycle(:tags, ['{','}'])
|
176
|
+
end
|
177
|
+
|
178
|
+
def test_strings_with_null_strings
|
179
|
+
assert_cycle(:tags, ['NULL','NULL'])
|
180
|
+
end
|
181
|
+
|
182
|
+
def test_contains_nils
|
183
|
+
assert_cycle(:tags, ['1',nil,nil])
|
184
|
+
end
|
185
|
+
|
186
|
+
def test_insert_fixture
|
187
|
+
tag_values = ["val1", "val2", "val3_with_'_multiple_quote_'_chars"]
|
188
|
+
@connection.insert_fixture({"tags" => tag_values}, "pg_arrays" )
|
189
|
+
assert_equal(PgArray.last.tags, tag_values)
|
190
|
+
end
|
191
|
+
|
192
|
+
def test_attribute_for_inspect_for_array_field
|
193
|
+
record = PgArray.new { |a| a.ratings = (1..10).to_a }
|
194
|
+
assert_equal("[1, 2, 3, 4, 5, 6, 7, 8, 9, 10]", record.attribute_for_inspect(:ratings))
|
195
|
+
end
|
196
|
+
|
197
|
+
def test_attribute_for_inspect_for_array_field_for_large_array
|
198
|
+
record = PgArray.new { |a| a.ratings = (1..11).to_a }
|
199
|
+
assert_equal("[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]", record.attribute_for_inspect(:ratings))
|
200
|
+
end
|
201
|
+
|
202
|
+
def test_escaping
|
203
|
+
unknown = 'foo\\",bar,baz,\\'
|
204
|
+
tags = ["hello_#{unknown}"]
|
205
|
+
ar = PgArray.create!(tags: tags)
|
206
|
+
ar.reload
|
207
|
+
assert_equal tags, ar.tags
|
208
|
+
end
|
209
|
+
|
210
|
+
def test_string_quoting_rules_match_pg_behavior
|
211
|
+
tags = ["", "one{", "two}", %(three"), "four\\", "five ", "six\t", "seven\n", "eight,", "nine", "ten\r", "NULL"]
|
212
|
+
x = PgArray.create!(tags: tags)
|
213
|
+
x.reload
|
214
|
+
|
215
|
+
refute x.changed?
|
216
|
+
end
|
217
|
+
|
218
|
+
def test_quoting_non_standard_delimiters
|
219
|
+
strings = ["hello,", "world;"]
|
220
|
+
oid = ActiveRecord::ConnectionAdapters::PostgreSQL::OID
|
221
|
+
comma_delim = oid::Array.new(ActiveRecord::Type::String.new, ',')
|
222
|
+
semicolon_delim = oid::Array.new(ActiveRecord::Type::String.new, ';')
|
223
|
+
conn = PgArray.connection
|
224
|
+
|
225
|
+
assert_equal %({"hello,",world;}), conn.type_cast(comma_delim.serialize(strings))
|
226
|
+
assert_equal %({hello,;"world;"}), conn.type_cast(semicolon_delim.serialize(strings))
|
227
|
+
end
|
228
|
+
|
229
|
+
def test_mutate_array
|
230
|
+
x = PgArray.create!(tags: %w(one two))
|
231
|
+
|
232
|
+
x.tags << "three"
|
233
|
+
x.save!
|
234
|
+
x.reload
|
235
|
+
|
236
|
+
assert_equal %w(one two three), x.tags
|
237
|
+
assert_not x.changed?
|
238
|
+
end
|
239
|
+
|
240
|
+
def test_mutate_value_in_array
|
241
|
+
x = PgArray.create!(hstores: [{ a: 'a' }, { b: 'b' }])
|
242
|
+
|
243
|
+
x.hstores.first['a'] = 'c'
|
244
|
+
x.save!
|
245
|
+
x.reload
|
246
|
+
|
247
|
+
assert_equal [{ 'a' => 'c' }, { 'b' => 'b' }], x.hstores
|
248
|
+
assert_not x.changed?
|
249
|
+
end
|
250
|
+
|
251
|
+
def test_datetime_with_timezone_awareness
|
252
|
+
tz = "Pacific Time (US & Canada)"
|
253
|
+
|
254
|
+
in_time_zone tz do
|
255
|
+
PgArray.reset_column_information
|
256
|
+
time_string = Time.current.to_s
|
257
|
+
time = Time.zone.parse(time_string)
|
258
|
+
|
259
|
+
record = PgArray.new(datetimes: [time_string])
|
260
|
+
assert_equal [time], record.datetimes
|
261
|
+
assert_equal ActiveSupport::TimeZone[tz], record.datetimes.first.time_zone
|
262
|
+
|
263
|
+
record.save!
|
264
|
+
record.reload
|
265
|
+
|
266
|
+
assert_equal [time], record.datetimes
|
267
|
+
assert_equal ActiveSupport::TimeZone[tz], record.datetimes.first.time_zone
|
268
|
+
end
|
269
|
+
end
|
270
|
+
|
271
|
+
def test_assigning_non_array_value
|
272
|
+
record = PgArray.new(tags: "not-an-array")
|
273
|
+
assert_equal [], record.tags
|
274
|
+
assert_equal "not-an-array", record.tags_before_type_cast
|
275
|
+
assert record.save
|
276
|
+
assert_equal record.tags, record.reload.tags
|
277
|
+
end
|
278
|
+
|
279
|
+
def test_assigning_empty_string
|
280
|
+
record = PgArray.new(tags: "")
|
281
|
+
assert_equal [], record.tags
|
282
|
+
assert_equal "", record.tags_before_type_cast
|
283
|
+
assert record.save
|
284
|
+
assert_equal record.tags, record.reload.tags
|
285
|
+
end
|
286
|
+
|
287
|
+
def test_assigning_valid_pg_array_literal
|
288
|
+
record = PgArray.new(tags: "{1,2,3}")
|
289
|
+
assert_equal ["1", "2", "3"], record.tags
|
290
|
+
assert_equal "{1,2,3}", record.tags_before_type_cast
|
291
|
+
assert record.save
|
292
|
+
assert_equal record.tags, record.reload.tags
|
293
|
+
end
|
294
|
+
|
295
|
+
def test_uniqueness_validation
|
296
|
+
klass = Class.new(PgArray) do
|
297
|
+
validates_uniqueness_of :tags
|
298
|
+
|
299
|
+
def self.model_name; ActiveModel::Name.new(PgArray) end
|
300
|
+
end
|
301
|
+
e1 = klass.create("tags" => ["black", "blue"])
|
302
|
+
assert e1.persisted?, "Saving e1"
|
303
|
+
|
304
|
+
e2 = klass.create("tags" => ["black", "blue"])
|
305
|
+
assert !e2.persisted?, "e2 shouldn't be valid"
|
306
|
+
assert e2.errors[:tags].any?, "Should have errors for tags"
|
307
|
+
assert_equal ["has already been taken"], e2.errors[:tags], "Should have uniqueness message for tags"
|
308
|
+
end
|
309
|
+
|
310
|
+
def test_encoding_arrays_of_utf8_strings
|
311
|
+
arrays_of_utf8_strings = %w(nový ファイル)
|
312
|
+
assert_equal arrays_of_utf8_strings, @type.deserialize(@type.serialize(arrays_of_utf8_strings))
|
313
|
+
assert_equal [arrays_of_utf8_strings], @type.deserialize(@type.serialize([arrays_of_utf8_strings]))
|
314
|
+
end
|
315
|
+
|
316
|
+
def test_precision_is_respected_on_timestamp_columns
|
317
|
+
time = Time.now.change(usec: 123)
|
318
|
+
record = PgArray.create!(timestamps: [time])
|
319
|
+
|
320
|
+
assert_equal 123, record.timestamps.first.usec
|
321
|
+
record.reload
|
322
|
+
assert_equal 123, record.timestamps.first.usec
|
323
|
+
end
|
324
|
+
|
325
|
+
private
|
326
|
+
def assert_cycle field, array
|
327
|
+
# test creation
|
328
|
+
x = PgArray.create!(field => array)
|
329
|
+
x.reload
|
330
|
+
assert_equal(array, x.public_send(field))
|
331
|
+
|
332
|
+
# test updating
|
333
|
+
x = PgArray.create!(field => [])
|
334
|
+
x.public_send("#{field}=", array)
|
335
|
+
x.save!
|
336
|
+
x.reload
|
337
|
+
assert_equal(array, x.public_send(field))
|
338
|
+
end
|
339
|
+
end
|