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,446 +1,446 @@
|
|
1
|
-
require "cases/helper"
|
2
|
-
require 'models/topic'
|
3
|
-
require 'models/task'
|
4
|
-
require 'models/category'
|
5
|
-
require 'models/post'
|
6
|
-
require 'rack'
|
7
|
-
|
8
|
-
class QueryCacheTest < ActiveRecord::TestCase
|
9
|
-
fixtures :tasks, :topics, :categories, :posts, :categories_posts
|
10
|
-
|
11
|
-
teardown do
|
12
|
-
Task.connection.clear_query_cache
|
13
|
-
ActiveRecord::Base.connection.disable_query_cache!
|
14
|
-
end
|
15
|
-
|
16
|
-
def test_exceptional_middleware_clears_and_disables_cache_on_error
|
17
|
-
assert_cache :off
|
18
|
-
|
19
|
-
mw = middleware { |env|
|
20
|
-
Task.find 1
|
21
|
-
Task.find 1
|
22
|
-
assert_equal 1, ActiveRecord::Base.connection.query_cache.length
|
23
|
-
raise "lol borked"
|
24
|
-
}
|
25
|
-
assert_raises(RuntimeError) { mw.call({}) }
|
26
|
-
|
27
|
-
assert_cache :off
|
28
|
-
end
|
29
|
-
|
30
|
-
private def with_temporary_connection_pool
|
31
|
-
old_pool = ActiveRecord::Base.connection_handler.retrieve_connection_pool(ActiveRecord::Base.connection_specification_name)
|
32
|
-
new_pool = ActiveRecord::ConnectionAdapters::ConnectionPool.new ActiveRecord::Base.connection_pool.spec
|
33
|
-
ActiveRecord::Base.connection_handler.send(:owner_to_pool)["primary"] = new_pool
|
34
|
-
|
35
|
-
yield
|
36
|
-
ensure
|
37
|
-
ActiveRecord::Base.connection_handler.send(:owner_to_pool)["primary"] = old_pool
|
38
|
-
end
|
39
|
-
|
40
|
-
def test_query_cache_across_threads
|
41
|
-
with_temporary_connection_pool do
|
42
|
-
begin
|
43
|
-
if in_memory_db?
|
44
|
-
# Separate connections to an in-memory database create an entirely new database,
|
45
|
-
# with an empty schema etc, so we just stub out this schema on the fly.
|
46
|
-
ActiveRecord::Base.connection_pool.with_connection do |connection|
|
47
|
-
connection.create_table :tasks do |t|
|
48
|
-
t.datetime :starting
|
49
|
-
t.datetime :ending
|
50
|
-
end
|
51
|
-
end
|
52
|
-
ActiveRecord::FixtureSet.create_fixtures(self.class.fixture_path, ["tasks"], {}, ActiveRecord::Base)
|
53
|
-
end
|
54
|
-
|
55
|
-
ActiveRecord::Base.connection_pool.connections.each do |conn|
|
56
|
-
assert_cache :off, conn
|
57
|
-
end
|
58
|
-
|
59
|
-
assert !ActiveRecord::Base.connection.nil?
|
60
|
-
assert_cache :off
|
61
|
-
|
62
|
-
middleware {
|
63
|
-
assert_cache :clean
|
64
|
-
|
65
|
-
Task.find 1
|
66
|
-
assert_cache :dirty
|
67
|
-
|
68
|
-
thread_1_connection = ActiveRecord::Base.connection
|
69
|
-
ActiveRecord::Base.clear_active_connections!
|
70
|
-
assert_cache :off, thread_1_connection
|
71
|
-
|
72
|
-
started = Concurrent::Event.new
|
73
|
-
checked = Concurrent::Event.new
|
74
|
-
|
75
|
-
thread_2_connection = nil
|
76
|
-
thread = Thread.new {
|
77
|
-
thread_2_connection = ActiveRecord::Base.connection
|
78
|
-
|
79
|
-
assert_equal thread_2_connection, thread_1_connection
|
80
|
-
assert_cache :off
|
81
|
-
|
82
|
-
middleware {
|
83
|
-
assert_cache :clean
|
84
|
-
|
85
|
-
Task.find 1
|
86
|
-
assert_cache :dirty
|
87
|
-
|
88
|
-
started.set
|
89
|
-
checked.wait
|
90
|
-
|
91
|
-
ActiveRecord::Base.clear_active_connections!
|
92
|
-
}.call({})
|
93
|
-
}
|
94
|
-
|
95
|
-
started.wait
|
96
|
-
|
97
|
-
thread_1_connection = ActiveRecord::Base.connection
|
98
|
-
assert_not_equal thread_1_connection, thread_2_connection
|
99
|
-
assert_cache :dirty, thread_2_connection
|
100
|
-
checked.set
|
101
|
-
thread.join
|
102
|
-
|
103
|
-
assert_cache :off, thread_2_connection
|
104
|
-
}.call({})
|
105
|
-
|
106
|
-
ActiveRecord::Base.connection_pool.connections.each do |conn|
|
107
|
-
assert_cache :off, conn
|
108
|
-
end
|
109
|
-
ensure
|
110
|
-
ActiveRecord::Base.clear_all_connections!
|
111
|
-
end
|
112
|
-
end
|
113
|
-
end
|
114
|
-
|
115
|
-
def test_exceptional_middleware_assigns_original_connection_id_on_error
|
116
|
-
connection_id = ActiveRecord::Base.connection_id
|
117
|
-
|
118
|
-
mw = middleware { |env|
|
119
|
-
ActiveRecord::Base.connection_id = self.object_id
|
120
|
-
raise "lol borked"
|
121
|
-
}
|
122
|
-
assert_raises(RuntimeError) { mw.call({}) }
|
123
|
-
|
124
|
-
assert_equal connection_id, ActiveRecord::Base.connection_id
|
125
|
-
end
|
126
|
-
|
127
|
-
def test_middleware_delegates
|
128
|
-
called = false
|
129
|
-
mw = middleware { |env|
|
130
|
-
called = true
|
131
|
-
[200, {}, nil]
|
132
|
-
}
|
133
|
-
mw.call({})
|
134
|
-
assert called, 'middleware should delegate'
|
135
|
-
end
|
136
|
-
|
137
|
-
def test_middleware_caches
|
138
|
-
mw = middleware { |env|
|
139
|
-
Task.find 1
|
140
|
-
Task.find 1
|
141
|
-
assert_equal 1, ActiveRecord::Base.connection.query_cache.length
|
142
|
-
[200, {}, nil]
|
143
|
-
}
|
144
|
-
mw.call({})
|
145
|
-
end
|
146
|
-
|
147
|
-
def test_cache_enabled_during_call
|
148
|
-
assert_cache :off
|
149
|
-
|
150
|
-
mw = middleware { |env|
|
151
|
-
assert_cache :clean
|
152
|
-
[200, {}, nil]
|
153
|
-
}
|
154
|
-
mw.call({})
|
155
|
-
end
|
156
|
-
|
157
|
-
def test_cache_passing_a_relation
|
158
|
-
post = Post.first
|
159
|
-
Post.cache do
|
160
|
-
query = post.categories.select(:post_id)
|
161
|
-
assert Post.connection.select_all(query).is_a?(ActiveRecord::Result)
|
162
|
-
end
|
163
|
-
end
|
164
|
-
|
165
|
-
def test_find_queries
|
166
|
-
assert_queries(2) { Task.find(1); Task.find(1) }
|
167
|
-
end
|
168
|
-
|
169
|
-
def test_find_queries_with_cache
|
170
|
-
Task.cache do
|
171
|
-
assert_queries(1) { Task.find(1); Task.find(1) }
|
172
|
-
end
|
173
|
-
end
|
174
|
-
|
175
|
-
def test_find_queries_with_cache_multi_record
|
176
|
-
Task.cache do
|
177
|
-
assert_queries(2) { Task.find(1); Task.find(1); Task.find(2) }
|
178
|
-
end
|
179
|
-
end
|
180
|
-
|
181
|
-
def test_find_queries_with_multi_cache_blocks
|
182
|
-
Task.cache do
|
183
|
-
Task.cache do
|
184
|
-
assert_queries(2) { Task.find(1); Task.find(2) }
|
185
|
-
end
|
186
|
-
assert_queries(0) { Task.find(1); Task.find(1); Task.find(2) }
|
187
|
-
end
|
188
|
-
end
|
189
|
-
|
190
|
-
def test_count_queries_with_cache
|
191
|
-
Task.cache do
|
192
|
-
assert_queries(1) { Task.count; Task.count }
|
193
|
-
end
|
194
|
-
end
|
195
|
-
|
196
|
-
def test_query_cache_dups_results_correctly
|
197
|
-
Task.cache do
|
198
|
-
now = Time.now.utc
|
199
|
-
task = Task.find 1
|
200
|
-
assert_not_equal now, task.starting
|
201
|
-
task.starting = now
|
202
|
-
task.reload
|
203
|
-
assert_not_equal now, task.starting
|
204
|
-
end
|
205
|
-
end
|
206
|
-
|
207
|
-
def test_cache_is_flat
|
208
|
-
Task.cache do
|
209
|
-
Topic.columns # don't count this query
|
210
|
-
assert_queries(1) { Topic.find(1); Topic.find(1); }
|
211
|
-
end
|
212
|
-
|
213
|
-
ActiveRecord::Base.cache do
|
214
|
-
assert_queries(1) { Task.find(1); Task.find(1) }
|
215
|
-
end
|
216
|
-
end
|
217
|
-
|
218
|
-
def test_cache_does_not_wrap_string_results_in_arrays
|
219
|
-
Task.cache do
|
220
|
-
# Oracle adapter returns count() as Integer or Float
|
221
|
-
if current_adapter?(:OracleAdapter)
|
222
|
-
assert_kind_of Numeric, Task.connection.select_value("SELECT count(*) AS count_all FROM tasks")
|
223
|
-
elsif current_adapter?(:SQLite3Adapter, :Mysql2Adapter, :PostgreSQLAdapter)
|
224
|
-
# Future versions of the sqlite3 adapter will return numeric
|
225
|
-
assert_instance_of 0.class, Task.connection.select_value("SELECT count(*) AS count_all FROM tasks")
|
226
|
-
else
|
227
|
-
assert_instance_of String, Task.connection.select_value("SELECT count(*) AS count_all FROM tasks")
|
228
|
-
end
|
229
|
-
end
|
230
|
-
end
|
231
|
-
|
232
|
-
def test_cache_is_ignored_for_locked_relations
|
233
|
-
task = Task.find 1
|
234
|
-
|
235
|
-
Task.cache do
|
236
|
-
assert_queries(2) { task.lock!; task.lock! }
|
237
|
-
end
|
238
|
-
end
|
239
|
-
|
240
|
-
def test_cache_is_available_when_connection_is_connected
|
241
|
-
conf = ActiveRecord::Base.configurations
|
242
|
-
|
243
|
-
ActiveRecord::Base.configurations = {}
|
244
|
-
Task.cache do
|
245
|
-
assert_queries(1) { Task.find(1); Task.find(1) }
|
246
|
-
end
|
247
|
-
ensure
|
248
|
-
ActiveRecord::Base.configurations = conf
|
249
|
-
end
|
250
|
-
|
251
|
-
def test_cache_is_available_when_using_a_not_connected_connection
|
252
|
-
skip "In-Memory DB can't test for using a not connected connection" if in_memory_db?
|
253
|
-
with_temporary_connection_pool do
|
254
|
-
spec_name = Task.connection_specification_name
|
255
|
-
resolver = ActiveRecord::ConnectionAdapters::ConnectionSpecification::Resolver.new(ActiveRecord::Base.configurations)
|
256
|
-
ActiveRecord::Base.connection_handler.establish_connection(resolver.spec(:arunit, "test2"))
|
257
|
-
Task.connection_specification_name = "test2"
|
258
|
-
refute Task.connected?
|
259
|
-
|
260
|
-
Task.cache do
|
261
|
-
begin
|
262
|
-
assert_queries(1) { Task.find(1); Task.find(1) }
|
263
|
-
ensure
|
264
|
-
ActiveRecord::Base.connection_handler.remove_connection(Task.connection_specification_name)
|
265
|
-
Task.connection_specification_name = spec_name
|
266
|
-
end
|
267
|
-
end
|
268
|
-
end
|
269
|
-
end
|
270
|
-
|
271
|
-
def test_query_cache_doesnt_leak_cached_results_of_rolled_back_queries
|
272
|
-
ActiveRecord::Base.connection.enable_query_cache!
|
273
|
-
post = Post.first
|
274
|
-
|
275
|
-
Post.transaction do
|
276
|
-
post.update_attributes(title: 'rollback')
|
277
|
-
assert_equal 1, Post.where(title: 'rollback').to_a.count
|
278
|
-
raise ActiveRecord::Rollback
|
279
|
-
end
|
280
|
-
|
281
|
-
assert_equal 0, Post.where(title: 'rollback').to_a.count
|
282
|
-
|
283
|
-
ActiveRecord::Base.connection.uncached do
|
284
|
-
assert_equal 0, Post.where(title: 'rollback').to_a.count
|
285
|
-
end
|
286
|
-
|
287
|
-
begin
|
288
|
-
Post.transaction do
|
289
|
-
post.update_attributes(title: 'rollback')
|
290
|
-
assert_equal 1, Post.where(title: 'rollback').to_a.count
|
291
|
-
raise 'broken'
|
292
|
-
end
|
293
|
-
rescue Exception
|
294
|
-
end
|
295
|
-
|
296
|
-
assert_equal 0, Post.where(title: 'rollback').to_a.count
|
297
|
-
|
298
|
-
ActiveRecord::Base.connection.uncached do
|
299
|
-
assert_equal 0, Post.where(title: 'rollback').to_a.count
|
300
|
-
end
|
301
|
-
end
|
302
|
-
|
303
|
-
def test_query_cache_does_not_establish_connection_if_unconnected
|
304
|
-
with_temporary_connection_pool do
|
305
|
-
ActiveRecord::Base.clear_active_connections!
|
306
|
-
refute ActiveRecord::Base.connection_handler.active_connections? # sanity check
|
307
|
-
|
308
|
-
middleware {
|
309
|
-
refute ActiveRecord::Base.connection_handler.active_connections?, "QueryCache forced ActiveRecord::Base to establish a connection in setup"
|
310
|
-
}.call({})
|
311
|
-
|
312
|
-
refute ActiveRecord::Base.connection_handler.active_connections?, "QueryCache forced ActiveRecord::Base to establish a connection in cleanup"
|
313
|
-
end
|
314
|
-
end
|
315
|
-
|
316
|
-
def test_query_cache_is_enabled_on_connections_established_after_middleware_runs
|
317
|
-
with_temporary_connection_pool do
|
318
|
-
ActiveRecord::Base.clear_active_connections!
|
319
|
-
refute ActiveRecord::Base.connection_handler.active_connections? # sanity check
|
320
|
-
|
321
|
-
middleware {
|
322
|
-
assert ActiveRecord::Base.connection.query_cache_enabled, "QueryCache did not get lazily enabled"
|
323
|
-
}.call({})
|
324
|
-
end
|
325
|
-
end
|
326
|
-
|
327
|
-
def test_query_caching_is_local_to_the_current_thread
|
328
|
-
with_temporary_connection_pool do
|
329
|
-
ActiveRecord::Base.clear_active_connections!
|
330
|
-
|
331
|
-
middleware {
|
332
|
-
assert ActiveRecord::Base.connection_pool.query_cache_enabled
|
333
|
-
assert ActiveRecord::Base.connection.query_cache_enabled
|
334
|
-
|
335
|
-
Thread.new {
|
336
|
-
refute ActiveRecord::Base.connection_pool.query_cache_enabled
|
337
|
-
refute ActiveRecord::Base.connection.query_cache_enabled
|
338
|
-
}.join
|
339
|
-
}.call({})
|
340
|
-
|
341
|
-
end
|
342
|
-
end
|
343
|
-
|
344
|
-
private
|
345
|
-
def middleware(&app)
|
346
|
-
executor = Class.new(ActiveSupport::Executor)
|
347
|
-
ActiveRecord::QueryCache.install_executor_hooks executor
|
348
|
-
lambda { |env| executor.wrap { app.call(env) } }
|
349
|
-
end
|
350
|
-
|
351
|
-
def assert_cache(state, connection = ActiveRecord::Base.connection)
|
352
|
-
case state
|
353
|
-
when :off
|
354
|
-
assert !connection.query_cache_enabled, "cache should be off"
|
355
|
-
assert connection.query_cache.empty?, "cache should be empty"
|
356
|
-
when :clean
|
357
|
-
assert connection.query_cache_enabled, "cache should be on"
|
358
|
-
assert connection.query_cache.empty?, "cache should be empty"
|
359
|
-
when :dirty
|
360
|
-
assert connection.query_cache_enabled, "cache should be on"
|
361
|
-
assert !connection.query_cache.empty?, "cache should be dirty"
|
362
|
-
else
|
363
|
-
raise "unknown state"
|
364
|
-
end
|
365
|
-
end
|
366
|
-
end
|
367
|
-
|
368
|
-
class QueryCacheExpiryTest < ActiveRecord::TestCase
|
369
|
-
fixtures :tasks, :posts, :categories, :categories_posts
|
370
|
-
|
371
|
-
def test_cache_gets_cleared_after_migration
|
372
|
-
# warm the cache
|
373
|
-
Post.find(1)
|
374
|
-
|
375
|
-
# change the column definition
|
376
|
-
Post.connection.change_column :posts, :title, :string, limit: 80
|
377
|
-
assert_nothing_raised { Post.find(1) }
|
378
|
-
|
379
|
-
# restore the old definition
|
380
|
-
Post.connection.change_column :posts, :title, :string
|
381
|
-
end
|
382
|
-
|
383
|
-
def test_find
|
384
|
-
assert_called(Task.connection, :clear_query_cache) do
|
385
|
-
assert !Task.connection.query_cache_enabled
|
386
|
-
Task.cache do
|
387
|
-
assert Task.connection.query_cache_enabled
|
388
|
-
Task.find(1)
|
389
|
-
|
390
|
-
Task.uncached do
|
391
|
-
assert !Task.connection.query_cache_enabled
|
392
|
-
Task.find(1)
|
393
|
-
end
|
394
|
-
|
395
|
-
assert Task.connection.query_cache_enabled
|
396
|
-
end
|
397
|
-
assert !Task.connection.query_cache_enabled
|
398
|
-
end
|
399
|
-
end
|
400
|
-
|
401
|
-
def test_update
|
402
|
-
assert_called(Task.connection, :clear_query_cache, times: 2) do
|
403
|
-
Task.cache do
|
404
|
-
task = Task.find(1)
|
405
|
-
task.starting = Time.now.utc
|
406
|
-
task.save!
|
407
|
-
end
|
408
|
-
end
|
409
|
-
end
|
410
|
-
|
411
|
-
def test_destroy
|
412
|
-
assert_called(Task.connection, :clear_query_cache, times: 2) do
|
413
|
-
Task.cache do
|
414
|
-
Task.find(1).destroy
|
415
|
-
end
|
416
|
-
end
|
417
|
-
end
|
418
|
-
|
419
|
-
def test_insert
|
420
|
-
assert_called(ActiveRecord::Base.connection, :clear_query_cache, times: 2) do
|
421
|
-
Task.cache do
|
422
|
-
Task.create!
|
423
|
-
end
|
424
|
-
end
|
425
|
-
end
|
426
|
-
|
427
|
-
def test_cache_is_expired_by_habtm_update
|
428
|
-
assert_called(ActiveRecord::Base.connection, :clear_query_cache, times: 2) do
|
429
|
-
ActiveRecord::Base.cache do
|
430
|
-
c = Category.first
|
431
|
-
p = Post.first
|
432
|
-
p.categories << c
|
433
|
-
end
|
434
|
-
end
|
435
|
-
end
|
436
|
-
|
437
|
-
def test_cache_is_expired_by_habtm_delete
|
438
|
-
assert_called(ActiveRecord::Base.connection, :clear_query_cache, times: 2) do
|
439
|
-
ActiveRecord::Base.cache do
|
440
|
-
p = Post.find(1)
|
441
|
-
assert p.categories.any?
|
442
|
-
p.categories.delete_all
|
443
|
-
end
|
444
|
-
end
|
445
|
-
end
|
446
|
-
end
|
1
|
+
require "cases/helper"
|
2
|
+
require 'models/topic'
|
3
|
+
require 'models/task'
|
4
|
+
require 'models/category'
|
5
|
+
require 'models/post'
|
6
|
+
require 'rack'
|
7
|
+
|
8
|
+
class QueryCacheTest < ActiveRecord::TestCase
|
9
|
+
fixtures :tasks, :topics, :categories, :posts, :categories_posts
|
10
|
+
|
11
|
+
teardown do
|
12
|
+
Task.connection.clear_query_cache
|
13
|
+
ActiveRecord::Base.connection.disable_query_cache!
|
14
|
+
end
|
15
|
+
|
16
|
+
def test_exceptional_middleware_clears_and_disables_cache_on_error
|
17
|
+
assert_cache :off
|
18
|
+
|
19
|
+
mw = middleware { |env|
|
20
|
+
Task.find 1
|
21
|
+
Task.find 1
|
22
|
+
assert_equal 1, ActiveRecord::Base.connection.query_cache.length
|
23
|
+
raise "lol borked"
|
24
|
+
}
|
25
|
+
assert_raises(RuntimeError) { mw.call({}) }
|
26
|
+
|
27
|
+
assert_cache :off
|
28
|
+
end
|
29
|
+
|
30
|
+
private def with_temporary_connection_pool
|
31
|
+
old_pool = ActiveRecord::Base.connection_handler.retrieve_connection_pool(ActiveRecord::Base.connection_specification_name)
|
32
|
+
new_pool = ActiveRecord::ConnectionAdapters::ConnectionPool.new ActiveRecord::Base.connection_pool.spec
|
33
|
+
ActiveRecord::Base.connection_handler.send(:owner_to_pool)["primary"] = new_pool
|
34
|
+
|
35
|
+
yield
|
36
|
+
ensure
|
37
|
+
ActiveRecord::Base.connection_handler.send(:owner_to_pool)["primary"] = old_pool
|
38
|
+
end
|
39
|
+
|
40
|
+
def test_query_cache_across_threads
|
41
|
+
with_temporary_connection_pool do
|
42
|
+
begin
|
43
|
+
if in_memory_db?
|
44
|
+
# Separate connections to an in-memory database create an entirely new database,
|
45
|
+
# with an empty schema etc, so we just stub out this schema on the fly.
|
46
|
+
ActiveRecord::Base.connection_pool.with_connection do |connection|
|
47
|
+
connection.create_table :tasks do |t|
|
48
|
+
t.datetime :starting
|
49
|
+
t.datetime :ending
|
50
|
+
end
|
51
|
+
end
|
52
|
+
ActiveRecord::FixtureSet.create_fixtures(self.class.fixture_path, ["tasks"], {}, ActiveRecord::Base)
|
53
|
+
end
|
54
|
+
|
55
|
+
ActiveRecord::Base.connection_pool.connections.each do |conn|
|
56
|
+
assert_cache :off, conn
|
57
|
+
end
|
58
|
+
|
59
|
+
assert !ActiveRecord::Base.connection.nil?
|
60
|
+
assert_cache :off
|
61
|
+
|
62
|
+
middleware {
|
63
|
+
assert_cache :clean
|
64
|
+
|
65
|
+
Task.find 1
|
66
|
+
assert_cache :dirty
|
67
|
+
|
68
|
+
thread_1_connection = ActiveRecord::Base.connection
|
69
|
+
ActiveRecord::Base.clear_active_connections!
|
70
|
+
assert_cache :off, thread_1_connection
|
71
|
+
|
72
|
+
started = Concurrent::Event.new
|
73
|
+
checked = Concurrent::Event.new
|
74
|
+
|
75
|
+
thread_2_connection = nil
|
76
|
+
thread = Thread.new {
|
77
|
+
thread_2_connection = ActiveRecord::Base.connection
|
78
|
+
|
79
|
+
assert_equal thread_2_connection, thread_1_connection
|
80
|
+
assert_cache :off
|
81
|
+
|
82
|
+
middleware {
|
83
|
+
assert_cache :clean
|
84
|
+
|
85
|
+
Task.find 1
|
86
|
+
assert_cache :dirty
|
87
|
+
|
88
|
+
started.set
|
89
|
+
checked.wait
|
90
|
+
|
91
|
+
ActiveRecord::Base.clear_active_connections!
|
92
|
+
}.call({})
|
93
|
+
}
|
94
|
+
|
95
|
+
started.wait
|
96
|
+
|
97
|
+
thread_1_connection = ActiveRecord::Base.connection
|
98
|
+
assert_not_equal thread_1_connection, thread_2_connection
|
99
|
+
assert_cache :dirty, thread_2_connection
|
100
|
+
checked.set
|
101
|
+
thread.join
|
102
|
+
|
103
|
+
assert_cache :off, thread_2_connection
|
104
|
+
}.call({})
|
105
|
+
|
106
|
+
ActiveRecord::Base.connection_pool.connections.each do |conn|
|
107
|
+
assert_cache :off, conn
|
108
|
+
end
|
109
|
+
ensure
|
110
|
+
ActiveRecord::Base.clear_all_connections!
|
111
|
+
end
|
112
|
+
end
|
113
|
+
end
|
114
|
+
|
115
|
+
def test_exceptional_middleware_assigns_original_connection_id_on_error
|
116
|
+
connection_id = ActiveRecord::Base.connection_id
|
117
|
+
|
118
|
+
mw = middleware { |env|
|
119
|
+
ActiveRecord::Base.connection_id = self.object_id
|
120
|
+
raise "lol borked"
|
121
|
+
}
|
122
|
+
assert_raises(RuntimeError) { mw.call({}) }
|
123
|
+
|
124
|
+
assert_equal connection_id, ActiveRecord::Base.connection_id
|
125
|
+
end
|
126
|
+
|
127
|
+
def test_middleware_delegates
|
128
|
+
called = false
|
129
|
+
mw = middleware { |env|
|
130
|
+
called = true
|
131
|
+
[200, {}, nil]
|
132
|
+
}
|
133
|
+
mw.call({})
|
134
|
+
assert called, 'middleware should delegate'
|
135
|
+
end
|
136
|
+
|
137
|
+
def test_middleware_caches
|
138
|
+
mw = middleware { |env|
|
139
|
+
Task.find 1
|
140
|
+
Task.find 1
|
141
|
+
assert_equal 1, ActiveRecord::Base.connection.query_cache.length
|
142
|
+
[200, {}, nil]
|
143
|
+
}
|
144
|
+
mw.call({})
|
145
|
+
end
|
146
|
+
|
147
|
+
def test_cache_enabled_during_call
|
148
|
+
assert_cache :off
|
149
|
+
|
150
|
+
mw = middleware { |env|
|
151
|
+
assert_cache :clean
|
152
|
+
[200, {}, nil]
|
153
|
+
}
|
154
|
+
mw.call({})
|
155
|
+
end
|
156
|
+
|
157
|
+
def test_cache_passing_a_relation
|
158
|
+
post = Post.first
|
159
|
+
Post.cache do
|
160
|
+
query = post.categories.select(:post_id)
|
161
|
+
assert Post.connection.select_all(query).is_a?(ActiveRecord::Result)
|
162
|
+
end
|
163
|
+
end
|
164
|
+
|
165
|
+
def test_find_queries
|
166
|
+
assert_queries(2) { Task.find(1); Task.find(1) }
|
167
|
+
end
|
168
|
+
|
169
|
+
def test_find_queries_with_cache
|
170
|
+
Task.cache do
|
171
|
+
assert_queries(1) { Task.find(1); Task.find(1) }
|
172
|
+
end
|
173
|
+
end
|
174
|
+
|
175
|
+
def test_find_queries_with_cache_multi_record
|
176
|
+
Task.cache do
|
177
|
+
assert_queries(2) { Task.find(1); Task.find(1); Task.find(2) }
|
178
|
+
end
|
179
|
+
end
|
180
|
+
|
181
|
+
def test_find_queries_with_multi_cache_blocks
|
182
|
+
Task.cache do
|
183
|
+
Task.cache do
|
184
|
+
assert_queries(2) { Task.find(1); Task.find(2) }
|
185
|
+
end
|
186
|
+
assert_queries(0) { Task.find(1); Task.find(1); Task.find(2) }
|
187
|
+
end
|
188
|
+
end
|
189
|
+
|
190
|
+
def test_count_queries_with_cache
|
191
|
+
Task.cache do
|
192
|
+
assert_queries(1) { Task.count; Task.count }
|
193
|
+
end
|
194
|
+
end
|
195
|
+
|
196
|
+
def test_query_cache_dups_results_correctly
|
197
|
+
Task.cache do
|
198
|
+
now = Time.now.utc
|
199
|
+
task = Task.find 1
|
200
|
+
assert_not_equal now, task.starting
|
201
|
+
task.starting = now
|
202
|
+
task.reload
|
203
|
+
assert_not_equal now, task.starting
|
204
|
+
end
|
205
|
+
end
|
206
|
+
|
207
|
+
def test_cache_is_flat
|
208
|
+
Task.cache do
|
209
|
+
Topic.columns # don't count this query
|
210
|
+
assert_queries(1) { Topic.find(1); Topic.find(1); }
|
211
|
+
end
|
212
|
+
|
213
|
+
ActiveRecord::Base.cache do
|
214
|
+
assert_queries(1) { Task.find(1); Task.find(1) }
|
215
|
+
end
|
216
|
+
end
|
217
|
+
|
218
|
+
def test_cache_does_not_wrap_string_results_in_arrays
|
219
|
+
Task.cache do
|
220
|
+
# Oracle adapter returns count() as Integer or Float
|
221
|
+
if current_adapter?(:OracleAdapter)
|
222
|
+
assert_kind_of Numeric, Task.connection.select_value("SELECT count(*) AS count_all FROM tasks")
|
223
|
+
elsif current_adapter?(:SQLite3Adapter, :Mysql2Adapter, :PostgreSQLAdapter)
|
224
|
+
# Future versions of the sqlite3 adapter will return numeric
|
225
|
+
assert_instance_of 0.class, Task.connection.select_value("SELECT count(*) AS count_all FROM tasks")
|
226
|
+
else
|
227
|
+
assert_instance_of String, Task.connection.select_value("SELECT count(*) AS count_all FROM tasks")
|
228
|
+
end
|
229
|
+
end
|
230
|
+
end
|
231
|
+
|
232
|
+
def test_cache_is_ignored_for_locked_relations
|
233
|
+
task = Task.find 1
|
234
|
+
|
235
|
+
Task.cache do
|
236
|
+
assert_queries(2) { task.lock!; task.lock! }
|
237
|
+
end
|
238
|
+
end
|
239
|
+
|
240
|
+
def test_cache_is_available_when_connection_is_connected
|
241
|
+
conf = ActiveRecord::Base.configurations
|
242
|
+
|
243
|
+
ActiveRecord::Base.configurations = {}
|
244
|
+
Task.cache do
|
245
|
+
assert_queries(1) { Task.find(1); Task.find(1) }
|
246
|
+
end
|
247
|
+
ensure
|
248
|
+
ActiveRecord::Base.configurations = conf
|
249
|
+
end
|
250
|
+
|
251
|
+
def test_cache_is_available_when_using_a_not_connected_connection
|
252
|
+
skip "In-Memory DB can't test for using a not connected connection" if in_memory_db?
|
253
|
+
with_temporary_connection_pool do
|
254
|
+
spec_name = Task.connection_specification_name
|
255
|
+
resolver = ActiveRecord::ConnectionAdapters::ConnectionSpecification::Resolver.new(ActiveRecord::Base.configurations)
|
256
|
+
ActiveRecord::Base.connection_handler.establish_connection(resolver.spec(:arunit, "test2"))
|
257
|
+
Task.connection_specification_name = "test2"
|
258
|
+
refute Task.connected?
|
259
|
+
|
260
|
+
Task.cache do
|
261
|
+
begin
|
262
|
+
assert_queries(1) { Task.find(1); Task.find(1) }
|
263
|
+
ensure
|
264
|
+
ActiveRecord::Base.connection_handler.remove_connection(Task.connection_specification_name)
|
265
|
+
Task.connection_specification_name = spec_name
|
266
|
+
end
|
267
|
+
end
|
268
|
+
end
|
269
|
+
end
|
270
|
+
|
271
|
+
def test_query_cache_doesnt_leak_cached_results_of_rolled_back_queries
|
272
|
+
ActiveRecord::Base.connection.enable_query_cache!
|
273
|
+
post = Post.first
|
274
|
+
|
275
|
+
Post.transaction do
|
276
|
+
post.update_attributes(title: 'rollback')
|
277
|
+
assert_equal 1, Post.where(title: 'rollback').to_a.count
|
278
|
+
raise ActiveRecord::Rollback
|
279
|
+
end
|
280
|
+
|
281
|
+
assert_equal 0, Post.where(title: 'rollback').to_a.count
|
282
|
+
|
283
|
+
ActiveRecord::Base.connection.uncached do
|
284
|
+
assert_equal 0, Post.where(title: 'rollback').to_a.count
|
285
|
+
end
|
286
|
+
|
287
|
+
begin
|
288
|
+
Post.transaction do
|
289
|
+
post.update_attributes(title: 'rollback')
|
290
|
+
assert_equal 1, Post.where(title: 'rollback').to_a.count
|
291
|
+
raise 'broken'
|
292
|
+
end
|
293
|
+
rescue Exception
|
294
|
+
end
|
295
|
+
|
296
|
+
assert_equal 0, Post.where(title: 'rollback').to_a.count
|
297
|
+
|
298
|
+
ActiveRecord::Base.connection.uncached do
|
299
|
+
assert_equal 0, Post.where(title: 'rollback').to_a.count
|
300
|
+
end
|
301
|
+
end
|
302
|
+
|
303
|
+
def test_query_cache_does_not_establish_connection_if_unconnected
|
304
|
+
with_temporary_connection_pool do
|
305
|
+
ActiveRecord::Base.clear_active_connections!
|
306
|
+
refute ActiveRecord::Base.connection_handler.active_connections? # sanity check
|
307
|
+
|
308
|
+
middleware {
|
309
|
+
refute ActiveRecord::Base.connection_handler.active_connections?, "QueryCache forced ActiveRecord::Base to establish a connection in setup"
|
310
|
+
}.call({})
|
311
|
+
|
312
|
+
refute ActiveRecord::Base.connection_handler.active_connections?, "QueryCache forced ActiveRecord::Base to establish a connection in cleanup"
|
313
|
+
end
|
314
|
+
end
|
315
|
+
|
316
|
+
def test_query_cache_is_enabled_on_connections_established_after_middleware_runs
|
317
|
+
with_temporary_connection_pool do
|
318
|
+
ActiveRecord::Base.clear_active_connections!
|
319
|
+
refute ActiveRecord::Base.connection_handler.active_connections? # sanity check
|
320
|
+
|
321
|
+
middleware {
|
322
|
+
assert ActiveRecord::Base.connection.query_cache_enabled, "QueryCache did not get lazily enabled"
|
323
|
+
}.call({})
|
324
|
+
end
|
325
|
+
end
|
326
|
+
|
327
|
+
def test_query_caching_is_local_to_the_current_thread
|
328
|
+
with_temporary_connection_pool do
|
329
|
+
ActiveRecord::Base.clear_active_connections!
|
330
|
+
|
331
|
+
middleware {
|
332
|
+
assert ActiveRecord::Base.connection_pool.query_cache_enabled
|
333
|
+
assert ActiveRecord::Base.connection.query_cache_enabled
|
334
|
+
|
335
|
+
Thread.new {
|
336
|
+
refute ActiveRecord::Base.connection_pool.query_cache_enabled
|
337
|
+
refute ActiveRecord::Base.connection.query_cache_enabled
|
338
|
+
}.join
|
339
|
+
}.call({})
|
340
|
+
|
341
|
+
end
|
342
|
+
end
|
343
|
+
|
344
|
+
private
|
345
|
+
def middleware(&app)
|
346
|
+
executor = Class.new(ActiveSupport::Executor)
|
347
|
+
ActiveRecord::QueryCache.install_executor_hooks executor
|
348
|
+
lambda { |env| executor.wrap { app.call(env) } }
|
349
|
+
end
|
350
|
+
|
351
|
+
def assert_cache(state, connection = ActiveRecord::Base.connection)
|
352
|
+
case state
|
353
|
+
when :off
|
354
|
+
assert !connection.query_cache_enabled, "cache should be off"
|
355
|
+
assert connection.query_cache.empty?, "cache should be empty"
|
356
|
+
when :clean
|
357
|
+
assert connection.query_cache_enabled, "cache should be on"
|
358
|
+
assert connection.query_cache.empty?, "cache should be empty"
|
359
|
+
when :dirty
|
360
|
+
assert connection.query_cache_enabled, "cache should be on"
|
361
|
+
assert !connection.query_cache.empty?, "cache should be dirty"
|
362
|
+
else
|
363
|
+
raise "unknown state"
|
364
|
+
end
|
365
|
+
end
|
366
|
+
end
|
367
|
+
|
368
|
+
class QueryCacheExpiryTest < ActiveRecord::TestCase
|
369
|
+
fixtures :tasks, :posts, :categories, :categories_posts
|
370
|
+
|
371
|
+
def test_cache_gets_cleared_after_migration
|
372
|
+
# warm the cache
|
373
|
+
Post.find(1)
|
374
|
+
|
375
|
+
# change the column definition
|
376
|
+
Post.connection.change_column :posts, :title, :string, limit: 80
|
377
|
+
assert_nothing_raised { Post.find(1) }
|
378
|
+
|
379
|
+
# restore the old definition
|
380
|
+
Post.connection.change_column :posts, :title, :string
|
381
|
+
end
|
382
|
+
|
383
|
+
def test_find
|
384
|
+
assert_called(Task.connection, :clear_query_cache) do
|
385
|
+
assert !Task.connection.query_cache_enabled
|
386
|
+
Task.cache do
|
387
|
+
assert Task.connection.query_cache_enabled
|
388
|
+
Task.find(1)
|
389
|
+
|
390
|
+
Task.uncached do
|
391
|
+
assert !Task.connection.query_cache_enabled
|
392
|
+
Task.find(1)
|
393
|
+
end
|
394
|
+
|
395
|
+
assert Task.connection.query_cache_enabled
|
396
|
+
end
|
397
|
+
assert !Task.connection.query_cache_enabled
|
398
|
+
end
|
399
|
+
end
|
400
|
+
|
401
|
+
def test_update
|
402
|
+
assert_called(Task.connection, :clear_query_cache, times: 2) do
|
403
|
+
Task.cache do
|
404
|
+
task = Task.find(1)
|
405
|
+
task.starting = Time.now.utc
|
406
|
+
task.save!
|
407
|
+
end
|
408
|
+
end
|
409
|
+
end
|
410
|
+
|
411
|
+
def test_destroy
|
412
|
+
assert_called(Task.connection, :clear_query_cache, times: 2) do
|
413
|
+
Task.cache do
|
414
|
+
Task.find(1).destroy
|
415
|
+
end
|
416
|
+
end
|
417
|
+
end
|
418
|
+
|
419
|
+
def test_insert
|
420
|
+
assert_called(ActiveRecord::Base.connection, :clear_query_cache, times: 2) do
|
421
|
+
Task.cache do
|
422
|
+
Task.create!
|
423
|
+
end
|
424
|
+
end
|
425
|
+
end
|
426
|
+
|
427
|
+
def test_cache_is_expired_by_habtm_update
|
428
|
+
assert_called(ActiveRecord::Base.connection, :clear_query_cache, times: 2) do
|
429
|
+
ActiveRecord::Base.cache do
|
430
|
+
c = Category.first
|
431
|
+
p = Post.first
|
432
|
+
p.categories << c
|
433
|
+
end
|
434
|
+
end
|
435
|
+
end
|
436
|
+
|
437
|
+
def test_cache_is_expired_by_habtm_delete
|
438
|
+
assert_called(ActiveRecord::Base.connection, :clear_query_cache, times: 2) do
|
439
|
+
ActiveRecord::Base.cache do
|
440
|
+
p = Post.find(1)
|
441
|
+
assert p.categories.any?
|
442
|
+
p.categories.delete_all
|
443
|
+
end
|
444
|
+
end
|
445
|
+
end
|
446
|
+
end
|