ibm_db 3.0.4-x86-mingw32 → 3.0.5-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 +4 -4
- data/CHANGES +4 -1
- data/LICENSE +1 -1
- data/MANIFEST +14 -14
- data/README +225 -225
- data/ext/Makefile.nt32 +181 -181
- data/ext/Makefile.nt32.191 +212 -212
- data/ext/extconf.rb +291 -291
- data/ext/ibm_db.c +11887 -11884
- 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/init.rb +41 -41
- data/lib/IBM_DB.rb +27 -27
- data/lib/active_record/connection_adapters/ibm_db_adapter.rb +3177 -3177
- data/lib/active_record/connection_adapters/ibmdb_adapter.rb +1 -1
- data/lib/active_record/vendor/db2-i5-zOS.yaml +328 -328
- data/lib/mswin32/ibm_db.rb +122 -122
- data/lib/mswin32/rb21x/i386/ibm_db.so +0 -0
- data/lib/mswin32/rb22x/i386/ibm_db.so +0 -0
- data/lib/mswin32/rb23x/i386/ibm_db.so +0 -0
- data/test/active_record/connection_adapters/fake_adapter.rb +46 -46
- data/test/assets/example.log +1 -1
- data/test/assets/test.txt +1 -1
- data/test/cases/adapter_test.rb +276 -261
- data/test/cases/aggregations_test.rb +158 -158
- data/test/cases/ar_schema_test.rb +161 -161
- data/test/cases/associations/association_scope_test.rb +21 -21
- data/test/cases/associations/belongs_to_associations_test.rb +1029 -1029
- data/test/cases/associations/callbacks_test.rb +192 -192
- data/test/cases/associations/cascaded_eager_loading_test.rb +188 -188
- data/test/cases/associations/deprecated_counter_cache_on_has_many_through_test.rb +26 -26
- data/test/cases/associations/eager_load_includes_full_sti_class_test.rb +36 -36
- data/test/cases/associations/eager_load_nested_include_test.rb +128 -128
- data/test/cases/associations/eager_singularization_test.rb +148 -148
- data/test/cases/associations/eager_test.rb +1429 -1411
- data/test/cases/associations/extension_test.rb +82 -82
- data/test/cases/associations/has_and_belongs_to_many_associations_test.rb +972 -932
- data/test/cases/associations/has_many_associations_test.rb +2182 -2162
- data/test/cases/associations/has_many_through_associations_test.rb +1204 -1204
- data/test/cases/associations/has_one_associations_test.rb +610 -610
- data/test/cases/associations/has_one_through_associations_test.rb +380 -380
- data/test/cases/associations/inner_join_association_test.rb +139 -139
- data/test/cases/associations/inverse_associations_test.rb +706 -693
- data/test/cases/associations/join_model_test.rb +754 -754
- data/test/cases/associations/nested_through_associations_test.rb +579 -579
- data/test/cases/associations/required_test.rb +82 -82
- data/test/cases/associations_test.rb +380 -380
- data/test/cases/attribute_decorators_test.rb +125 -125
- data/test/cases/attribute_methods/read_test.rb +60 -60
- data/test/cases/attribute_methods/serialization_test.rb +29 -29
- data/test/cases/attribute_methods_test.rb +952 -952
- data/test/cases/attribute_set_test.rb +210 -200
- data/test/cases/attribute_test.rb +180 -180
- data/test/cases/attributes_test.rb +136 -136
- data/test/cases/autosave_association_test.rb +1595 -1595
- data/test/cases/base_test.rb +1664 -1638
- data/test/cases/batches_test.rb +212 -212
- data/test/cases/binary_test.rb +52 -52
- data/test/cases/bind_parameter_test.rb +100 -100
- data/test/cases/calculations_test.rb +646 -646
- data/test/cases/callbacks_test.rb +543 -543
- data/test/cases/clone_test.rb +40 -40
- data/test/cases/coders/yaml_column_test.rb +63 -63
- data/test/cases/column_alias_test.rb +17 -17
- data/test/cases/column_definition_test.rb +123 -123
- data/test/cases/connection_adapters/adapter_leasing_test.rb +54 -54
- data/test/cases/connection_adapters/connection_handler_test.rb +53 -53
- data/test/cases/connection_adapters/connection_specification_test.rb +12 -12
- data/test/cases/connection_adapters/merge_and_resolve_default_url_config_test.rb +293 -293
- data/test/cases/connection_adapters/mysql_type_lookup_test.rb +65 -65
- data/test/cases/connection_adapters/quoting_test.rb +13 -13
- data/test/cases/connection_adapters/schema_cache_test.rb +56 -56
- data/test/cases/connection_adapters/type_lookup_test.rb +110 -110
- data/test/cases/connection_management_test.rb +122 -122
- data/test/cases/connection_pool_test.rb +346 -346
- data/test/cases/connection_specification/resolver_test.rb +116 -116
- data/test/cases/core_test.rb +112 -112
- data/test/cases/counter_cache_test.rb +209 -209
- data/test/cases/custom_locking_test.rb +17 -17
- data/test/cases/database_statements_test.rb +19 -19
- data/test/cases/date_time_test.rb +61 -61
- data/test/cases/defaults_test.rb +223 -223
- data/test/cases/dirty_test.rb +785 -775
- data/test/cases/disconnected_test.rb +28 -28
- data/test/cases/dup_test.rb +157 -157
- data/test/cases/enum_test.rb +290 -290
- data/test/cases/explain_subscriber_test.rb +64 -64
- data/test/cases/explain_test.rb +76 -76
- data/test/cases/finder_respond_to_test.rb +60 -60
- data/test/cases/finder_test.rb +1169 -1166
- data/test/cases/fixture_set/file_test.rb +138 -138
- data/test/cases/fixtures_test.rb +908 -897
- data/test/cases/forbidden_attributes_protection_test.rb +99 -99
- data/test/cases/habtm_destroy_order_test.rb +61 -61
- data/test/cases/helper.rb +210 -210
- data/test/cases/hot_compatibility_test.rb +54 -54
- data/test/cases/i18n_test.rb +45 -45
- data/test/cases/inheritance_test.rb +375 -375
- data/test/cases/integration_test.rb +139 -139
- data/test/cases/invalid_connection_test.rb +22 -22
- data/test/cases/invalid_date_test.rb +32 -32
- data/test/cases/invertible_migration_test.rb +295 -295
- data/test/cases/json_serialization_test.rb +302 -302
- data/test/cases/locking_test.rb +477 -477
- data/test/cases/log_subscriber_test.rb +136 -136
- data/test/cases/migration/change_schema_test - Copy.rb +448 -448
- data/test/cases/migration/change_schema_test.rb +512 -472
- data/test/cases/migration/change_table_test.rb +224 -224
- data/test/cases/migration/column_attributes_test.rb +192 -192
- data/test/cases/migration/column_positioning_test.rb +56 -56
- data/test/cases/migration/columns_test.rb +304 -304
- data/test/cases/migration/command_recorder_test.rb +305 -305
- data/test/cases/migration/create_join_table_test.rb +148 -148
- data/test/cases/migration/foreign_key_test - Changed.rb +325 -325
- data/test/cases/migration/foreign_key_test.rb +328 -360
- data/test/cases/migration/helper.rb +39 -39
- data/test/cases/migration/index_test.rb +216 -216
- data/test/cases/migration/logger_test.rb +36 -36
- data/test/cases/migration/pending_migrations_test.rb +53 -53
- data/test/cases/migration/references_foreign_key_test.rb +169 -214
- data/test/cases/migration/references_index_test.rb +101 -101
- data/test/cases/migration/references_statements_test.rb +116 -116
- data/test/cases/migration/rename_table_test.rb +93 -93
- data/test/cases/migration/table_and_index_test.rb +24 -24
- data/test/cases/migration_test.rb +959 -959
- data/test/cases/migrator_test.rb +388 -388
- data/test/cases/mixin_test.rb +70 -70
- data/test/cases/modules_test.rb +173 -173
- data/test/cases/multiparameter_attributes_test.rb +350 -350
- data/test/cases/multiple_db_test.rb +115 -115
- data/test/cases/nested_attributes_test.rb +1070 -1057
- data/test/cases/nested_attributes_with_callbacks_test.rb +144 -144
- data/test/cases/persistence_test.rb +909 -909
- data/test/cases/pooled_connections_test.rb +81 -81
- data/test/cases/primary_keys_test.rb +237 -237
- data/test/cases/query_cache_test.rb +326 -326
- data/test/cases/quoting_test.rb +156 -156
- data/test/cases/readonly_test.rb +118 -118
- data/test/cases/reaper_test.rb +85 -85
- data/test/cases/reflection_test.rb +463 -454
- data/test/cases/relation/delegation_test.rb +68 -68
- data/test/cases/relation/merging_test.rb +161 -161
- data/test/cases/relation/mutation_test.rb +165 -165
- data/test/cases/relation/predicate_builder_test.rb +14 -14
- data/test/cases/relation/where_chain_test.rb +181 -181
- data/test/cases/relation/where_test.rb +300 -300
- data/test/cases/relation/where_test2.rb +36 -36
- data/test/cases/relation_test.rb +319 -297
- data/test/cases/relations_test.rb +1815 -1815
- data/test/cases/reload_models_test.rb +22 -22
- data/test/cases/result_test.rb +80 -80
- data/test/cases/sanitize_test.rb +83 -83
- data/test/cases/schema_dumper_test.rb +463 -463
- data/test/cases/scoping/default_scoping_test.rb +454 -454
- data/test/cases/scoping/named_scoping_test.rb +524 -524
- data/test/cases/scoping/relation_scoping_test.rb +357 -357
- data/test/cases/serialization_test.rb +104 -104
- data/test/cases/serialized_attribute_test.rb +277 -277
- data/test/cases/statement_cache_test.rb +98 -98
- data/test/cases/store_test.rb +194 -194
- data/test/cases/tasks/database_tasks_test.rb +398 -396
- data/test/cases/tasks/mysql_rake_test.rb +324 -311
- data/test/cases/tasks/postgresql_rake_test.rb +250 -245
- data/test/cases/tasks/sqlite_rake_test.rb +193 -193
- data/test/cases/test_case.rb +123 -123
- data/test/cases/timestamp_test.rb +467 -468
- data/test/cases/transaction_callbacks_test.rb +452 -452
- data/test/cases/transaction_isolation_test.rb +106 -106
- data/test/cases/transactions_test.rb +817 -817
- data/test/cases/type/decimal_test.rb +56 -51
- data/test/cases/type/integer_test.rb +121 -121
- data/test/cases/type/string_test.rb +36 -36
- data/test/cases/type/type_map_test.rb +177 -177
- data/test/cases/type/unsigned_integer_test.rb +18 -18
- data/test/cases/types_test.rb +141 -141
- data/test/cases/unconnected_test.rb +33 -33
- data/test/cases/validations/association_validation_test.rb +86 -86
- data/test/cases/validations/i18n_generate_message_validation_test.rb +84 -84
- data/test/cases/validations/i18n_validation_test.rb +90 -90
- data/test/cases/validations/length_validation_test.rb +47 -47
- data/test/cases/validations/presence_validation_test.rb +68 -68
- data/test/cases/validations/uniqueness_validation_test.rb +457 -434
- data/test/cases/validations_repair_helper.rb +23 -23
- data/test/cases/validations_test.rb +165 -165
- data/test/cases/view_test.rb +119 -113
- data/test/cases/xml_serialization_test.rb +457 -457
- data/test/cases/yaml_serialization_test.rb +126 -86
- data/test/config.rb +5 -5
- data/test/config.yml +154 -154
- data/test/connections/native_ibm_db/connection.rb +43 -43
- data/test/fixtures/accounts.yml +29 -29
- data/test/fixtures/admin/accounts.yml +2 -2
- data/test/fixtures/admin/randomly_named_a9.yml +7 -7
- data/test/fixtures/admin/randomly_named_b0.yml +7 -7
- 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/binaries.yml +133 -133
- data/test/fixtures/books.yml +11 -11
- 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/courses.yml +8 -8
- data/test/fixtures/customers.yml +25 -25
- data/test/fixtures/dashboards.yml +6 -6
- data/test/fixtures/developers.yml +21 -21
- 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/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/csv/accounts.csv +1 -1
- 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/organizations.yml +5 -5
- 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 -12
- data/test/fixtures/posts.yml +80 -80
- data/test/fixtures/price_estimates.yml +7 -7
- data/test/fixtures/products.yml +4 -4
- data/test/fixtures/projects.yml +7 -7
- data/test/fixtures/randomly_named_a9.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/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/ibm_db_test.rb +24 -24
- 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 +8 -8
- data/test/migrations/missing/1_people_have_last_names.rb +8 -8
- data/test/migrations/missing/3_we_need_reminders.rb +11 -11
- data/test/migrations/missing/4_innocent_jointable.rb +11 -11
- data/test/migrations/rename/1_we_need_things.rb +10 -10
- data/test/migrations/rename/2_rename_things.rb +8 -8
- 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 +11 -11
- data/test/migrations/valid/3_innocent_jointable.rb +11 -11
- 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 +11 -11
- data/test/migrations/valid_with_subdirectories/sub1/3_innocent_jointable.rb +11 -11
- 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 +4 -4
- data/test/models/admin/account.rb +2 -2
- data/test/models/admin/randomly_named_c1.rb +3 -3
- data/test/models/admin/user.rb +40 -40
- data/test/models/aircraft.rb +4 -4
- data/test/models/arunit2_model.rb +3 -3
- data/test/models/author.rb +212 -212
- data/test/models/auto_id.rb +4 -4
- data/test/models/autoloadable/extra_firm.rb +2 -2
- data/test/models/binary.rb +1 -1
- data/test/models/bird.rb +12 -12
- data/test/models/book.rb +18 -18
- data/test/models/boolean.rb +2 -2
- data/test/models/bulb.rb +51 -51
- data/test/models/cake_designer.rb +3 -3
- data/test/models/car.rb +26 -26
- data/test/models/carrier.rb +2 -2
- data/test/models/categorization.rb +19 -19
- data/test/models/category.rb +35 -35
- data/test/models/chef.rb +7 -3
- data/test/models/citation.rb +3 -3
- data/test/models/club.rb +23 -23
- 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 +64 -64
- data/test/models/company.rb +228 -225
- 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/contract.rb +20 -20
- data/test/models/country.rb +7 -7
- data/test/models/course.rb +6 -6
- data/test/models/customer.rb +77 -77
- 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 +255 -252
- 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 +2 -2
- 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 +1 -1
- data/test/models/hotel.rb +9 -6
- 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 +41 -41
- data/test/models/member_detail.rb +7 -7
- data/test/models/member_type.rb +3 -3
- data/test/models/membership.rb +35 -35
- 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/molecule.rb +6 -6
- data/test/models/movie.rb +5 -5
- data/test/models/order.rb +4 -4
- data/test/models/organization.rb +14 -14
- data/test/models/owner.rb +34 -34
- data/test/models/parrot.rb +29 -29
- data/test/models/person.rb +143 -143
- data/test/models/personal_legacy_thing.rb +4 -4
- data/test/models/pet.rb +15 -15
- data/test/models/pirate.rb +92 -92
- data/test/models/possession.rb +3 -3
- data/test/models/post.rb +264 -264
- data/test/models/price_estimate.rb +4 -4
- data/test/models/professor.rb +5 -5
- data/test/models/project.rb +31 -29
- 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/randomly_named_c1.rb +3 -3
- data/test/models/rating.rb +4 -4
- data/test/models/reader.rb +23 -23
- 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 +33 -33
- data/test/models/ship_part.rb +7 -7
- 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 +7 -7
- data/test/models/tagging.rb +13 -13
- data/test/models/task.rb +5 -5
- data/test/models/topic.rb +124 -124
- 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/tyre.rb +11 -11
- data/test/models/uuid_child.rb +3 -3
- 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/mysql2_specific_schema.rb +58 -58
- data/test/schema/mysql_specific_schema.rb +70 -70
- data/test/schema/oracle_specific_schema.rb +43 -43
- data/test/schema/postgresql_specific_schema.rb +202 -202
- data/test/schema/schema.rb +952 -938
- data/test/schema/sqlite_specific_schema.rb +21 -21
- data/test/support/config.rb +43 -43
- data/test/support/connection.rb +22 -22
- 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
- metadata +2 -2
@@ -1,452 +1,452 @@
|
|
1
|
-
require "cases/helper"
|
2
|
-
require 'models/owner'
|
3
|
-
require 'models/pet'
|
4
|
-
require 'models/topic'
|
5
|
-
|
6
|
-
class TransactionCallbacksTest < ActiveRecord::TestCase
|
7
|
-
self.use_transactional_fixtures = false
|
8
|
-
fixtures :topics, :owners, :pets
|
9
|
-
|
10
|
-
class ReplyWithCallbacks < ActiveRecord::Base
|
11
|
-
self.table_name = :topics
|
12
|
-
|
13
|
-
belongs_to :topic, foreign_key: "parent_id"
|
14
|
-
|
15
|
-
validates_presence_of :content
|
16
|
-
|
17
|
-
after_commit :do_after_commit, on: :create
|
18
|
-
|
19
|
-
attr_accessor :save_on_after_create
|
20
|
-
after_create do
|
21
|
-
self.save! if save_on_after_create
|
22
|
-
end
|
23
|
-
|
24
|
-
def history
|
25
|
-
@history ||= []
|
26
|
-
end
|
27
|
-
|
28
|
-
def do_after_commit
|
29
|
-
history << :commit_on_create
|
30
|
-
end
|
31
|
-
end
|
32
|
-
|
33
|
-
class TopicWithCallbacks < ActiveRecord::Base
|
34
|
-
self.table_name = :topics
|
35
|
-
|
36
|
-
has_many :replies, class_name: "ReplyWithCallbacks", foreign_key: "parent_id"
|
37
|
-
|
38
|
-
after_commit { |record| record.do_after_commit(nil) }
|
39
|
-
after_commit(on: :create) { |record| record.do_after_commit(:create) }
|
40
|
-
after_commit(on: :update) { |record| record.do_after_commit(:update) }
|
41
|
-
after_commit(on: :destroy) { |record| record.do_after_commit(:destroy) }
|
42
|
-
after_rollback { |record| record.do_after_rollback(nil) }
|
43
|
-
after_rollback(on: :create) { |record| record.do_after_rollback(:create) }
|
44
|
-
after_rollback(on: :update) { |record| record.do_after_rollback(:update) }
|
45
|
-
after_rollback(on: :destroy) { |record| record.do_after_rollback(:destroy) }
|
46
|
-
|
47
|
-
def history
|
48
|
-
@history ||= []
|
49
|
-
end
|
50
|
-
|
51
|
-
def after_commit_block(on = nil, &block)
|
52
|
-
@after_commit ||= {}
|
53
|
-
@after_commit[on] ||= []
|
54
|
-
@after_commit[on] << block
|
55
|
-
end
|
56
|
-
|
57
|
-
def after_rollback_block(on = nil, &block)
|
58
|
-
@after_rollback ||= {}
|
59
|
-
@after_rollback[on] ||= []
|
60
|
-
@after_rollback[on] << block
|
61
|
-
end
|
62
|
-
|
63
|
-
def do_after_commit(on)
|
64
|
-
blocks = @after_commit[on] if defined?(@after_commit)
|
65
|
-
blocks.each{|b| b.call(self)} if blocks
|
66
|
-
end
|
67
|
-
|
68
|
-
def do_after_rollback(on)
|
69
|
-
blocks = @after_rollback[on] if defined?(@after_rollback)
|
70
|
-
blocks.each{|b| b.call(self)} if blocks
|
71
|
-
end
|
72
|
-
end
|
73
|
-
|
74
|
-
def setup
|
75
|
-
@first = TopicWithCallbacks.find(1)
|
76
|
-
end
|
77
|
-
|
78
|
-
def test_call_after_commit_after_transaction_commits
|
79
|
-
@first.after_commit_block{|r| r.history << :after_commit}
|
80
|
-
@first.after_rollback_block{|r| r.history << :after_rollback}
|
81
|
-
|
82
|
-
@first.save!
|
83
|
-
assert_equal [:after_commit], @first.history
|
84
|
-
end
|
85
|
-
|
86
|
-
def test_only_call_after_commit_on_update_after_transaction_commits_for_existing_record
|
87
|
-
add_transaction_execution_blocks @first
|
88
|
-
|
89
|
-
@first.save!
|
90
|
-
assert_equal [:commit_on_update], @first.history
|
91
|
-
end
|
92
|
-
|
93
|
-
def test_only_call_after_commit_on_destroy_after_transaction_commits_for_destroyed_record
|
94
|
-
add_transaction_execution_blocks @first
|
95
|
-
|
96
|
-
@first.destroy
|
97
|
-
assert_equal [:commit_on_destroy], @first.history
|
98
|
-
end
|
99
|
-
|
100
|
-
def test_only_call_after_commit_on_create_after_transaction_commits_for_new_record
|
101
|
-
unless current_adapter?(:IBM_DBAdapter)
|
102
|
-
@new_record = TopicWithCallbacks.new(:title => "New topic", :written_on => Date.today)
|
103
|
-
else
|
104
|
-
@new_record = TopicWithCallbacks.new(:title => "New topic", :written_on => Time.now)
|
105
|
-
end
|
106
|
-
add_transaction_execution_blocks new_record
|
107
|
-
|
108
|
-
new_record.save!
|
109
|
-
assert_equal [:commit_on_create], new_record.history
|
110
|
-
end
|
111
|
-
|
112
|
-
def test_only_call_after_commit_on_create_after_transaction_commits_for_new_record_if_create_succeeds_creating_through_association
|
113
|
-
topic = TopicWithCallbacks.create!(:title => "New topic", :written_on => Date.today)
|
114
|
-
reply = topic.replies.create
|
115
|
-
|
116
|
-
assert_equal [], reply.history
|
117
|
-
end
|
118
|
-
|
119
|
-
def test_only_call_after_commit_on_create_and_doesnt_leaky
|
120
|
-
r = ReplyWithCallbacks.new(content: 'foo')
|
121
|
-
r.save_on_after_create = true
|
122
|
-
r.save!
|
123
|
-
r.content = 'bar'
|
124
|
-
r.save!
|
125
|
-
r.save!
|
126
|
-
assert_equal [:commit_on_create], r.history
|
127
|
-
end
|
128
|
-
|
129
|
-
def test_only_call_after_commit_on_update_after_transaction_commits_for_existing_record_on_touch
|
130
|
-
add_transaction_execution_blocks @first
|
131
|
-
|
132
|
-
@first.touch
|
133
|
-
assert_equal [:commit_on_update], @first.history
|
134
|
-
end
|
135
|
-
|
136
|
-
def test_only_call_after_commit_on_top_level_transactions
|
137
|
-
@first.after_commit_block{|r| r.history << :after_commit}
|
138
|
-
assert @first.history.empty?
|
139
|
-
|
140
|
-
@first.transaction do
|
141
|
-
@first.transaction(requires_new: true) do
|
142
|
-
@first.touch
|
143
|
-
end
|
144
|
-
assert @first.history.empty?
|
145
|
-
end
|
146
|
-
assert_equal [:after_commit], @first.history
|
147
|
-
end
|
148
|
-
|
149
|
-
def test_call_after_rollback_after_transaction_rollsback
|
150
|
-
@first.after_commit_block{|r| r.history << :after_commit}
|
151
|
-
@first.after_rollback_block{|r| r.history << :after_rollback}
|
152
|
-
|
153
|
-
Topic.transaction do
|
154
|
-
@first.save!
|
155
|
-
raise ActiveRecord::Rollback
|
156
|
-
end
|
157
|
-
|
158
|
-
assert_equal [:after_rollback], @first.history
|
159
|
-
end
|
160
|
-
|
161
|
-
def test_only_call_after_rollback_on_update_after_transaction_rollsback_for_existing_record
|
162
|
-
add_transaction_execution_blocks @first
|
163
|
-
|
164
|
-
Topic.transaction do
|
165
|
-
@first.save!
|
166
|
-
raise ActiveRecord::Rollback
|
167
|
-
end
|
168
|
-
|
169
|
-
assert_equal [:rollback_on_update], @first.history
|
170
|
-
end
|
171
|
-
|
172
|
-
def test_only_call_after_rollback_on_update_after_transaction_rollsback_for_existing_record_on_touch
|
173
|
-
add_transaction_execution_blocks @first
|
174
|
-
|
175
|
-
Topic.transaction do
|
176
|
-
@first.touch
|
177
|
-
raise ActiveRecord::Rollback
|
178
|
-
end
|
179
|
-
|
180
|
-
assert_equal [:rollback_on_update], @first.history
|
181
|
-
end
|
182
|
-
|
183
|
-
def test_only_call_after_rollback_on_destroy_after_transaction_rollsback_for_destroyed_record
|
184
|
-
add_transaction_execution_blocks @first
|
185
|
-
|
186
|
-
Topic.transaction do
|
187
|
-
@first.destroy
|
188
|
-
raise ActiveRecord::Rollback
|
189
|
-
end
|
190
|
-
|
191
|
-
assert_equal [:rollback_on_destroy], @first.history
|
192
|
-
end
|
193
|
-
|
194
|
-
def test_only_call_after_rollback_on_create_after_transaction_rollsback_for_new_record
|
195
|
-
unless current_adapter?(:IBM_DBAdapter)
|
196
|
-
@new_record = TopicWithCallbacks.new(:title => "New topic", :written_on => Date.today)
|
197
|
-
else
|
198
|
-
@new_record = TopicWithCallbacks.new(:title => "New topic", :written_on => Time.now)
|
199
|
-
end
|
200
|
-
add_transaction_execution_blocks new_record
|
201
|
-
|
202
|
-
Topic.transaction do
|
203
|
-
new_record.save!
|
204
|
-
raise ActiveRecord::Rollback
|
205
|
-
end
|
206
|
-
|
207
|
-
assert_equal [:rollback_on_create], new_record.history
|
208
|
-
end
|
209
|
-
|
210
|
-
def test_call_after_rollback_when_commit_fails
|
211
|
-
@first.class.connection.singleton_class.send(:alias_method, :real_method_commit_db_transaction, :commit_db_transaction)
|
212
|
-
begin
|
213
|
-
@first.class.connection.singleton_class.class_eval do
|
214
|
-
def commit_db_transaction; raise "boom!"; end
|
215
|
-
end
|
216
|
-
|
217
|
-
@first.after_commit_block{|r| r.history << :after_commit}
|
218
|
-
@first.after_rollback_block{|r| r.history << :after_rollback}
|
219
|
-
|
220
|
-
assert !@first.save rescue nil
|
221
|
-
assert_equal [:after_rollback], @first.history
|
222
|
-
ensure
|
223
|
-
@first.class.connection.singleton_class.send(:remove_method, :commit_db_transaction)
|
224
|
-
@first.class.connection.singleton_class.send(:alias_method, :commit_db_transaction, :real_method_commit_db_transaction)
|
225
|
-
end
|
226
|
-
end
|
227
|
-
|
228
|
-
def test_only_call_after_rollback_on_records_rolled_back_to_a_savepoint
|
229
|
-
def @first.rollbacks(i=0); @rollbacks ||= 0; @rollbacks += i if i; end
|
230
|
-
def @first.commits(i=0); @commits ||= 0; @commits += i if i; end
|
231
|
-
@first.after_rollback_block{|r| r.rollbacks(1)}
|
232
|
-
@first.after_commit_block{|r| r.commits(1)}
|
233
|
-
|
234
|
-
second = TopicWithCallbacks.find(3)
|
235
|
-
def second.rollbacks(i=0); @rollbacks ||= 0; @rollbacks += i if i; end
|
236
|
-
def second.commits(i=0); @commits ||= 0; @commits += i if i; end
|
237
|
-
second.after_rollback_block{|r| r.rollbacks(1)}
|
238
|
-
second.after_commit_block{|r| r.commits(1)}
|
239
|
-
|
240
|
-
Topic.transaction do
|
241
|
-
@first.save!
|
242
|
-
Topic.transaction(:requires_new => true) do
|
243
|
-
second.save!
|
244
|
-
raise ActiveRecord::Rollback
|
245
|
-
end
|
246
|
-
end
|
247
|
-
|
248
|
-
assert_equal 1, @first.commits
|
249
|
-
assert_equal 0, @first.rollbacks
|
250
|
-
assert_equal 0, second.commits
|
251
|
-
assert_equal 1, second.rollbacks
|
252
|
-
end
|
253
|
-
|
254
|
-
def test_only_call_after_rollback_on_records_rolled_back_to_a_savepoint_when_release_savepoint_fails
|
255
|
-
def @first.rollbacks(i=0); @rollbacks ||= 0; @rollbacks += i if i; end
|
256
|
-
def @first.commits(i=0); @commits ||= 0; @commits += i if i; end
|
257
|
-
|
258
|
-
@first.after_rollback_block{|r| r.rollbacks(1)}
|
259
|
-
@first.after_commit_block{|r| r.commits(1)}
|
260
|
-
|
261
|
-
Topic.transaction do
|
262
|
-
@first.save
|
263
|
-
Topic.transaction(:requires_new => true) do
|
264
|
-
@first.save!
|
265
|
-
raise ActiveRecord::Rollback
|
266
|
-
end
|
267
|
-
Topic.transaction(:requires_new => true) do
|
268
|
-
@first.save!
|
269
|
-
raise ActiveRecord::Rollback
|
270
|
-
end
|
271
|
-
end
|
272
|
-
|
273
|
-
assert_equal 1, @first.commits
|
274
|
-
assert_equal 2, @first.rollbacks
|
275
|
-
end
|
276
|
-
|
277
|
-
def test_after_transaction_callbacks_should_prevent_callbacks_from_being_called
|
278
|
-
old_transaction_config = ActiveRecord::Base.raise_in_transactional_callbacks
|
279
|
-
ActiveRecord::Base.raise_in_transactional_callbacks = false
|
280
|
-
|
281
|
-
def @first.last_after_transaction_error=(e); @last_transaction_error = e; end
|
282
|
-
def @first.last_after_transaction_error; @last_transaction_error; end
|
283
|
-
@first.after_commit_block{|r| r.last_after_transaction_error = :commit; raise "fail!";}
|
284
|
-
@first.after_rollback_block{|r| r.last_after_transaction_error = :rollback; raise "fail!";}
|
285
|
-
|
286
|
-
second = TopicWithCallbacks.find(3)
|
287
|
-
second.after_commit_block{|r| r.history << :after_commit}
|
288
|
-
second.after_rollback_block{|r| r.history << :after_rollback}
|
289
|
-
|
290
|
-
Topic.transaction do
|
291
|
-
@first.save!
|
292
|
-
second.save!
|
293
|
-
end
|
294
|
-
assert_equal :commit, @first.last_after_transaction_error
|
295
|
-
assert_equal [:after_commit], second.history
|
296
|
-
|
297
|
-
second.history.clear
|
298
|
-
Topic.transaction do
|
299
|
-
@first.save!
|
300
|
-
second.save!
|
301
|
-
raise ActiveRecord::Rollback
|
302
|
-
end
|
303
|
-
assert_equal :rollback, @first.last_after_transaction_error
|
304
|
-
assert_equal [:after_rollback], second.history
|
305
|
-
ensure
|
306
|
-
ActiveRecord::Base.raise_in_transactional_callbacks = old_transaction_config
|
307
|
-
end
|
308
|
-
|
309
|
-
def test_after_commit_should_not_raise_when_raise_in_transactional_callbacks_false
|
310
|
-
old_transaction_config = ActiveRecord::Base.raise_in_transactional_callbacks
|
311
|
-
ActiveRecord::Base.raise_in_transactional_callbacks = false
|
312
|
-
@first.after_commit_block{ fail "boom" }
|
313
|
-
Topic.transaction { @first.save! }
|
314
|
-
ensure
|
315
|
-
ActiveRecord::Base.raise_in_transactional_callbacks = old_transaction_config
|
316
|
-
end
|
317
|
-
|
318
|
-
def test_after_commit_callback_should_not_swallow_errors
|
319
|
-
@first.after_commit_block{ fail "boom" }
|
320
|
-
assert_raises(RuntimeError) do
|
321
|
-
Topic.transaction do
|
322
|
-
@first.save!
|
323
|
-
end
|
324
|
-
end
|
325
|
-
end
|
326
|
-
|
327
|
-
def test_after_commit_callback_when_raise_should_not_restore_state
|
328
|
-
first = TopicWithCallbacks.new
|
329
|
-
second = TopicWithCallbacks.new
|
330
|
-
first.after_commit_block{ fail "boom" }
|
331
|
-
second.after_commit_block{ fail "boom" }
|
332
|
-
|
333
|
-
begin
|
334
|
-
Topic.transaction do
|
335
|
-
first.save!
|
336
|
-
assert_not_nil first.id
|
337
|
-
second.save!
|
338
|
-
assert_not_nil second.id
|
339
|
-
end
|
340
|
-
rescue
|
341
|
-
end
|
342
|
-
assert_not_nil first.id
|
343
|
-
assert_not_nil second.id
|
344
|
-
assert first.reload
|
345
|
-
end
|
346
|
-
|
347
|
-
def test_after_rollback_callback_should_not_swallow_errors_when_set_to_raise
|
348
|
-
error_class = Class.new(StandardError)
|
349
|
-
@first.after_rollback_block{ raise error_class }
|
350
|
-
assert_raises(error_class) do
|
351
|
-
Topic.transaction do
|
352
|
-
@first.save!
|
353
|
-
raise ActiveRecord::Rollback
|
354
|
-
end
|
355
|
-
end
|
356
|
-
end
|
357
|
-
|
358
|
-
def test_after_rollback_callback_when_raise_should_restore_state
|
359
|
-
error_class = Class.new(StandardError)
|
360
|
-
|
361
|
-
first = TopicWithCallbacks.new
|
362
|
-
second = TopicWithCallbacks.new
|
363
|
-
first.after_rollback_block{ raise error_class }
|
364
|
-
second.after_rollback_block{ raise error_class }
|
365
|
-
|
366
|
-
begin
|
367
|
-
Topic.transaction do
|
368
|
-
first.save!
|
369
|
-
assert_not_nil first.id
|
370
|
-
second.save!
|
371
|
-
assert_not_nil second.id
|
372
|
-
raise ActiveRecord::Rollback
|
373
|
-
end
|
374
|
-
rescue error_class
|
375
|
-
end
|
376
|
-
assert_nil first.id
|
377
|
-
assert_nil second.id
|
378
|
-
end
|
379
|
-
|
380
|
-
def test_after_rollback_callbacks_should_validate_on_condition
|
381
|
-
assert_raise(ArgumentError) { Topic.after_rollback(on: :save) }
|
382
|
-
e = assert_raise(ArgumentError) { Topic.after_rollback(on: 'create') }
|
383
|
-
assert_match(/:on conditions for after_commit and after_rollback callbacks have to be one of \[:create, :destroy, :update\]/, e.message)
|
384
|
-
end
|
385
|
-
|
386
|
-
def test_after_commit_callbacks_should_validate_on_condition
|
387
|
-
assert_raise(ArgumentError) { Topic.after_commit(on: :save) }
|
388
|
-
e = assert_raise(ArgumentError) { Topic.after_commit(on: 'create') }
|
389
|
-
assert_match(/:on conditions for after_commit and after_rollback callbacks have to be one of \[:create, :destroy, :update\]/, e.message)
|
390
|
-
end
|
391
|
-
|
392
|
-
def test_saving_a_record_with_a_belongs_to_that_specifies_touching_the_parent_should_call_callbacks_on_the_parent_object
|
393
|
-
pet = Pet.first
|
394
|
-
owner = pet.owner
|
395
|
-
flag = false
|
396
|
-
|
397
|
-
owner.on_after_commit do
|
398
|
-
flag = true
|
399
|
-
end
|
400
|
-
|
401
|
-
pet.name = "Fluffy the Third"
|
402
|
-
pet.save
|
403
|
-
|
404
|
-
assert flag
|
405
|
-
end
|
406
|
-
|
407
|
-
private
|
408
|
-
|
409
|
-
def add_transaction_execution_blocks(record)
|
410
|
-
record.after_commit_block(:create) { |r| r.history << :commit_on_create }
|
411
|
-
record.after_commit_block(:update) { |r| r.history << :commit_on_update }
|
412
|
-
record.after_commit_block(:destroy) { |r| r.history << :commit_on_destroy }
|
413
|
-
record.after_rollback_block(:create) { |r| r.history << :rollback_on_create }
|
414
|
-
record.after_rollback_block(:update) { |r| r.history << :rollback_on_update }
|
415
|
-
record.after_rollback_block(:destroy) { |r| r.history << :rollback_on_destroy }
|
416
|
-
end
|
417
|
-
end
|
418
|
-
|
419
|
-
class CallbacksOnMultipleActionsTest < ActiveRecord::TestCase
|
420
|
-
self.use_transactional_fixtures = false
|
421
|
-
|
422
|
-
class TopicWithCallbacksOnMultipleActions < ActiveRecord::Base
|
423
|
-
self.table_name = :topics
|
424
|
-
|
425
|
-
after_commit(on: [:create, :destroy]) { |record| record.history << :create_and_destroy }
|
426
|
-
after_commit(on: [:create, :update]) { |record| record.history << :create_and_update }
|
427
|
-
after_commit(on: [:update, :destroy]) { |record| record.history << :update_and_destroy }
|
428
|
-
|
429
|
-
def clear_history
|
430
|
-
@history = []
|
431
|
-
end
|
432
|
-
|
433
|
-
def history
|
434
|
-
@history ||= []
|
435
|
-
end
|
436
|
-
end
|
437
|
-
|
438
|
-
def test_after_commit_on_multiple_actions
|
439
|
-
topic = TopicWithCallbacksOnMultipleActions.new
|
440
|
-
topic.save
|
441
|
-
assert_equal [:create_and_update, :create_and_destroy], topic.history
|
442
|
-
|
443
|
-
topic.clear_history
|
444
|
-
topic.approved = true
|
445
|
-
topic.save
|
446
|
-
assert_equal [:update_and_destroy, :create_and_update], topic.history
|
447
|
-
|
448
|
-
topic.clear_history
|
449
|
-
topic.destroy
|
450
|
-
assert_equal [:update_and_destroy, :create_and_destroy], topic.history
|
451
|
-
end
|
452
|
-
end
|
1
|
+
require "cases/helper"
|
2
|
+
require 'models/owner'
|
3
|
+
require 'models/pet'
|
4
|
+
require 'models/topic'
|
5
|
+
|
6
|
+
class TransactionCallbacksTest < ActiveRecord::TestCase
|
7
|
+
self.use_transactional_fixtures = false
|
8
|
+
fixtures :topics, :owners, :pets
|
9
|
+
|
10
|
+
class ReplyWithCallbacks < ActiveRecord::Base
|
11
|
+
self.table_name = :topics
|
12
|
+
|
13
|
+
belongs_to :topic, foreign_key: "parent_id"
|
14
|
+
|
15
|
+
validates_presence_of :content
|
16
|
+
|
17
|
+
after_commit :do_after_commit, on: :create
|
18
|
+
|
19
|
+
attr_accessor :save_on_after_create
|
20
|
+
after_create do
|
21
|
+
self.save! if save_on_after_create
|
22
|
+
end
|
23
|
+
|
24
|
+
def history
|
25
|
+
@history ||= []
|
26
|
+
end
|
27
|
+
|
28
|
+
def do_after_commit
|
29
|
+
history << :commit_on_create
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
class TopicWithCallbacks < ActiveRecord::Base
|
34
|
+
self.table_name = :topics
|
35
|
+
|
36
|
+
has_many :replies, class_name: "ReplyWithCallbacks", foreign_key: "parent_id"
|
37
|
+
|
38
|
+
after_commit { |record| record.do_after_commit(nil) }
|
39
|
+
after_commit(on: :create) { |record| record.do_after_commit(:create) }
|
40
|
+
after_commit(on: :update) { |record| record.do_after_commit(:update) }
|
41
|
+
after_commit(on: :destroy) { |record| record.do_after_commit(:destroy) }
|
42
|
+
after_rollback { |record| record.do_after_rollback(nil) }
|
43
|
+
after_rollback(on: :create) { |record| record.do_after_rollback(:create) }
|
44
|
+
after_rollback(on: :update) { |record| record.do_after_rollback(:update) }
|
45
|
+
after_rollback(on: :destroy) { |record| record.do_after_rollback(:destroy) }
|
46
|
+
|
47
|
+
def history
|
48
|
+
@history ||= []
|
49
|
+
end
|
50
|
+
|
51
|
+
def after_commit_block(on = nil, &block)
|
52
|
+
@after_commit ||= {}
|
53
|
+
@after_commit[on] ||= []
|
54
|
+
@after_commit[on] << block
|
55
|
+
end
|
56
|
+
|
57
|
+
def after_rollback_block(on = nil, &block)
|
58
|
+
@after_rollback ||= {}
|
59
|
+
@after_rollback[on] ||= []
|
60
|
+
@after_rollback[on] << block
|
61
|
+
end
|
62
|
+
|
63
|
+
def do_after_commit(on)
|
64
|
+
blocks = @after_commit[on] if defined?(@after_commit)
|
65
|
+
blocks.each{|b| b.call(self)} if blocks
|
66
|
+
end
|
67
|
+
|
68
|
+
def do_after_rollback(on)
|
69
|
+
blocks = @after_rollback[on] if defined?(@after_rollback)
|
70
|
+
blocks.each{|b| b.call(self)} if blocks
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
74
|
+
def setup
|
75
|
+
@first = TopicWithCallbacks.find(1)
|
76
|
+
end
|
77
|
+
|
78
|
+
def test_call_after_commit_after_transaction_commits
|
79
|
+
@first.after_commit_block{|r| r.history << :after_commit}
|
80
|
+
@first.after_rollback_block{|r| r.history << :after_rollback}
|
81
|
+
|
82
|
+
@first.save!
|
83
|
+
assert_equal [:after_commit], @first.history
|
84
|
+
end
|
85
|
+
|
86
|
+
def test_only_call_after_commit_on_update_after_transaction_commits_for_existing_record
|
87
|
+
add_transaction_execution_blocks @first
|
88
|
+
|
89
|
+
@first.save!
|
90
|
+
assert_equal [:commit_on_update], @first.history
|
91
|
+
end
|
92
|
+
|
93
|
+
def test_only_call_after_commit_on_destroy_after_transaction_commits_for_destroyed_record
|
94
|
+
add_transaction_execution_blocks @first
|
95
|
+
|
96
|
+
@first.destroy
|
97
|
+
assert_equal [:commit_on_destroy], @first.history
|
98
|
+
end
|
99
|
+
|
100
|
+
def test_only_call_after_commit_on_create_after_transaction_commits_for_new_record
|
101
|
+
unless current_adapter?(:IBM_DBAdapter)
|
102
|
+
@new_record = TopicWithCallbacks.new(:title => "New topic", :written_on => Date.today)
|
103
|
+
else
|
104
|
+
@new_record = TopicWithCallbacks.new(:title => "New topic", :written_on => Time.now)
|
105
|
+
end
|
106
|
+
add_transaction_execution_blocks new_record
|
107
|
+
|
108
|
+
new_record.save!
|
109
|
+
assert_equal [:commit_on_create], new_record.history
|
110
|
+
end
|
111
|
+
|
112
|
+
def test_only_call_after_commit_on_create_after_transaction_commits_for_new_record_if_create_succeeds_creating_through_association
|
113
|
+
topic = TopicWithCallbacks.create!(:title => "New topic", :written_on => Date.today)
|
114
|
+
reply = topic.replies.create
|
115
|
+
|
116
|
+
assert_equal [], reply.history
|
117
|
+
end
|
118
|
+
|
119
|
+
def test_only_call_after_commit_on_create_and_doesnt_leaky
|
120
|
+
r = ReplyWithCallbacks.new(content: 'foo')
|
121
|
+
r.save_on_after_create = true
|
122
|
+
r.save!
|
123
|
+
r.content = 'bar'
|
124
|
+
r.save!
|
125
|
+
r.save!
|
126
|
+
assert_equal [:commit_on_create], r.history
|
127
|
+
end
|
128
|
+
|
129
|
+
def test_only_call_after_commit_on_update_after_transaction_commits_for_existing_record_on_touch
|
130
|
+
add_transaction_execution_blocks @first
|
131
|
+
|
132
|
+
@first.touch
|
133
|
+
assert_equal [:commit_on_update], @first.history
|
134
|
+
end
|
135
|
+
|
136
|
+
def test_only_call_after_commit_on_top_level_transactions
|
137
|
+
@first.after_commit_block{|r| r.history << :after_commit}
|
138
|
+
assert @first.history.empty?
|
139
|
+
|
140
|
+
@first.transaction do
|
141
|
+
@first.transaction(requires_new: true) do
|
142
|
+
@first.touch
|
143
|
+
end
|
144
|
+
assert @first.history.empty?
|
145
|
+
end
|
146
|
+
assert_equal [:after_commit], @first.history
|
147
|
+
end
|
148
|
+
|
149
|
+
def test_call_after_rollback_after_transaction_rollsback
|
150
|
+
@first.after_commit_block{|r| r.history << :after_commit}
|
151
|
+
@first.after_rollback_block{|r| r.history << :after_rollback}
|
152
|
+
|
153
|
+
Topic.transaction do
|
154
|
+
@first.save!
|
155
|
+
raise ActiveRecord::Rollback
|
156
|
+
end
|
157
|
+
|
158
|
+
assert_equal [:after_rollback], @first.history
|
159
|
+
end
|
160
|
+
|
161
|
+
def test_only_call_after_rollback_on_update_after_transaction_rollsback_for_existing_record
|
162
|
+
add_transaction_execution_blocks @first
|
163
|
+
|
164
|
+
Topic.transaction do
|
165
|
+
@first.save!
|
166
|
+
raise ActiveRecord::Rollback
|
167
|
+
end
|
168
|
+
|
169
|
+
assert_equal [:rollback_on_update], @first.history
|
170
|
+
end
|
171
|
+
|
172
|
+
def test_only_call_after_rollback_on_update_after_transaction_rollsback_for_existing_record_on_touch
|
173
|
+
add_transaction_execution_blocks @first
|
174
|
+
|
175
|
+
Topic.transaction do
|
176
|
+
@first.touch
|
177
|
+
raise ActiveRecord::Rollback
|
178
|
+
end
|
179
|
+
|
180
|
+
assert_equal [:rollback_on_update], @first.history
|
181
|
+
end
|
182
|
+
|
183
|
+
def test_only_call_after_rollback_on_destroy_after_transaction_rollsback_for_destroyed_record
|
184
|
+
add_transaction_execution_blocks @first
|
185
|
+
|
186
|
+
Topic.transaction do
|
187
|
+
@first.destroy
|
188
|
+
raise ActiveRecord::Rollback
|
189
|
+
end
|
190
|
+
|
191
|
+
assert_equal [:rollback_on_destroy], @first.history
|
192
|
+
end
|
193
|
+
|
194
|
+
def test_only_call_after_rollback_on_create_after_transaction_rollsback_for_new_record
|
195
|
+
unless current_adapter?(:IBM_DBAdapter)
|
196
|
+
@new_record = TopicWithCallbacks.new(:title => "New topic", :written_on => Date.today)
|
197
|
+
else
|
198
|
+
@new_record = TopicWithCallbacks.new(:title => "New topic", :written_on => Time.now)
|
199
|
+
end
|
200
|
+
add_transaction_execution_blocks new_record
|
201
|
+
|
202
|
+
Topic.transaction do
|
203
|
+
new_record.save!
|
204
|
+
raise ActiveRecord::Rollback
|
205
|
+
end
|
206
|
+
|
207
|
+
assert_equal [:rollback_on_create], new_record.history
|
208
|
+
end
|
209
|
+
|
210
|
+
def test_call_after_rollback_when_commit_fails
|
211
|
+
@first.class.connection.singleton_class.send(:alias_method, :real_method_commit_db_transaction, :commit_db_transaction)
|
212
|
+
begin
|
213
|
+
@first.class.connection.singleton_class.class_eval do
|
214
|
+
def commit_db_transaction; raise "boom!"; end
|
215
|
+
end
|
216
|
+
|
217
|
+
@first.after_commit_block{|r| r.history << :after_commit}
|
218
|
+
@first.after_rollback_block{|r| r.history << :after_rollback}
|
219
|
+
|
220
|
+
assert !@first.save rescue nil
|
221
|
+
assert_equal [:after_rollback], @first.history
|
222
|
+
ensure
|
223
|
+
@first.class.connection.singleton_class.send(:remove_method, :commit_db_transaction)
|
224
|
+
@first.class.connection.singleton_class.send(:alias_method, :commit_db_transaction, :real_method_commit_db_transaction)
|
225
|
+
end
|
226
|
+
end
|
227
|
+
|
228
|
+
def test_only_call_after_rollback_on_records_rolled_back_to_a_savepoint
|
229
|
+
def @first.rollbacks(i=0); @rollbacks ||= 0; @rollbacks += i if i; end
|
230
|
+
def @first.commits(i=0); @commits ||= 0; @commits += i if i; end
|
231
|
+
@first.after_rollback_block{|r| r.rollbacks(1)}
|
232
|
+
@first.after_commit_block{|r| r.commits(1)}
|
233
|
+
|
234
|
+
second = TopicWithCallbacks.find(3)
|
235
|
+
def second.rollbacks(i=0); @rollbacks ||= 0; @rollbacks += i if i; end
|
236
|
+
def second.commits(i=0); @commits ||= 0; @commits += i if i; end
|
237
|
+
second.after_rollback_block{|r| r.rollbacks(1)}
|
238
|
+
second.after_commit_block{|r| r.commits(1)}
|
239
|
+
|
240
|
+
Topic.transaction do
|
241
|
+
@first.save!
|
242
|
+
Topic.transaction(:requires_new => true) do
|
243
|
+
second.save!
|
244
|
+
raise ActiveRecord::Rollback
|
245
|
+
end
|
246
|
+
end
|
247
|
+
|
248
|
+
assert_equal 1, @first.commits
|
249
|
+
assert_equal 0, @first.rollbacks
|
250
|
+
assert_equal 0, second.commits
|
251
|
+
assert_equal 1, second.rollbacks
|
252
|
+
end
|
253
|
+
|
254
|
+
def test_only_call_after_rollback_on_records_rolled_back_to_a_savepoint_when_release_savepoint_fails
|
255
|
+
def @first.rollbacks(i=0); @rollbacks ||= 0; @rollbacks += i if i; end
|
256
|
+
def @first.commits(i=0); @commits ||= 0; @commits += i if i; end
|
257
|
+
|
258
|
+
@first.after_rollback_block{|r| r.rollbacks(1)}
|
259
|
+
@first.after_commit_block{|r| r.commits(1)}
|
260
|
+
|
261
|
+
Topic.transaction do
|
262
|
+
@first.save
|
263
|
+
Topic.transaction(:requires_new => true) do
|
264
|
+
@first.save!
|
265
|
+
raise ActiveRecord::Rollback
|
266
|
+
end
|
267
|
+
Topic.transaction(:requires_new => true) do
|
268
|
+
@first.save!
|
269
|
+
raise ActiveRecord::Rollback
|
270
|
+
end
|
271
|
+
end
|
272
|
+
|
273
|
+
assert_equal 1, @first.commits
|
274
|
+
assert_equal 2, @first.rollbacks
|
275
|
+
end
|
276
|
+
|
277
|
+
def test_after_transaction_callbacks_should_prevent_callbacks_from_being_called
|
278
|
+
old_transaction_config = ActiveRecord::Base.raise_in_transactional_callbacks
|
279
|
+
ActiveRecord::Base.raise_in_transactional_callbacks = false
|
280
|
+
|
281
|
+
def @first.last_after_transaction_error=(e); @last_transaction_error = e; end
|
282
|
+
def @first.last_after_transaction_error; @last_transaction_error; end
|
283
|
+
@first.after_commit_block{|r| r.last_after_transaction_error = :commit; raise "fail!";}
|
284
|
+
@first.after_rollback_block{|r| r.last_after_transaction_error = :rollback; raise "fail!";}
|
285
|
+
|
286
|
+
second = TopicWithCallbacks.find(3)
|
287
|
+
second.after_commit_block{|r| r.history << :after_commit}
|
288
|
+
second.after_rollback_block{|r| r.history << :after_rollback}
|
289
|
+
|
290
|
+
Topic.transaction do
|
291
|
+
@first.save!
|
292
|
+
second.save!
|
293
|
+
end
|
294
|
+
assert_equal :commit, @first.last_after_transaction_error
|
295
|
+
assert_equal [:after_commit], second.history
|
296
|
+
|
297
|
+
second.history.clear
|
298
|
+
Topic.transaction do
|
299
|
+
@first.save!
|
300
|
+
second.save!
|
301
|
+
raise ActiveRecord::Rollback
|
302
|
+
end
|
303
|
+
assert_equal :rollback, @first.last_after_transaction_error
|
304
|
+
assert_equal [:after_rollback], second.history
|
305
|
+
ensure
|
306
|
+
ActiveRecord::Base.raise_in_transactional_callbacks = old_transaction_config
|
307
|
+
end
|
308
|
+
|
309
|
+
def test_after_commit_should_not_raise_when_raise_in_transactional_callbacks_false
|
310
|
+
old_transaction_config = ActiveRecord::Base.raise_in_transactional_callbacks
|
311
|
+
ActiveRecord::Base.raise_in_transactional_callbacks = false
|
312
|
+
@first.after_commit_block{ fail "boom" }
|
313
|
+
Topic.transaction { @first.save! }
|
314
|
+
ensure
|
315
|
+
ActiveRecord::Base.raise_in_transactional_callbacks = old_transaction_config
|
316
|
+
end
|
317
|
+
|
318
|
+
def test_after_commit_callback_should_not_swallow_errors
|
319
|
+
@first.after_commit_block{ fail "boom" }
|
320
|
+
assert_raises(RuntimeError) do
|
321
|
+
Topic.transaction do
|
322
|
+
@first.save!
|
323
|
+
end
|
324
|
+
end
|
325
|
+
end
|
326
|
+
|
327
|
+
def test_after_commit_callback_when_raise_should_not_restore_state
|
328
|
+
first = TopicWithCallbacks.new
|
329
|
+
second = TopicWithCallbacks.new
|
330
|
+
first.after_commit_block{ fail "boom" }
|
331
|
+
second.after_commit_block{ fail "boom" }
|
332
|
+
|
333
|
+
begin
|
334
|
+
Topic.transaction do
|
335
|
+
first.save!
|
336
|
+
assert_not_nil first.id
|
337
|
+
second.save!
|
338
|
+
assert_not_nil second.id
|
339
|
+
end
|
340
|
+
rescue
|
341
|
+
end
|
342
|
+
assert_not_nil first.id
|
343
|
+
assert_not_nil second.id
|
344
|
+
assert first.reload
|
345
|
+
end
|
346
|
+
|
347
|
+
def test_after_rollback_callback_should_not_swallow_errors_when_set_to_raise
|
348
|
+
error_class = Class.new(StandardError)
|
349
|
+
@first.after_rollback_block{ raise error_class }
|
350
|
+
assert_raises(error_class) do
|
351
|
+
Topic.transaction do
|
352
|
+
@first.save!
|
353
|
+
raise ActiveRecord::Rollback
|
354
|
+
end
|
355
|
+
end
|
356
|
+
end
|
357
|
+
|
358
|
+
def test_after_rollback_callback_when_raise_should_restore_state
|
359
|
+
error_class = Class.new(StandardError)
|
360
|
+
|
361
|
+
first = TopicWithCallbacks.new
|
362
|
+
second = TopicWithCallbacks.new
|
363
|
+
first.after_rollback_block{ raise error_class }
|
364
|
+
second.after_rollback_block{ raise error_class }
|
365
|
+
|
366
|
+
begin
|
367
|
+
Topic.transaction do
|
368
|
+
first.save!
|
369
|
+
assert_not_nil first.id
|
370
|
+
second.save!
|
371
|
+
assert_not_nil second.id
|
372
|
+
raise ActiveRecord::Rollback
|
373
|
+
end
|
374
|
+
rescue error_class
|
375
|
+
end
|
376
|
+
assert_nil first.id
|
377
|
+
assert_nil second.id
|
378
|
+
end
|
379
|
+
|
380
|
+
def test_after_rollback_callbacks_should_validate_on_condition
|
381
|
+
assert_raise(ArgumentError) { Topic.after_rollback(on: :save) }
|
382
|
+
e = assert_raise(ArgumentError) { Topic.after_rollback(on: 'create') }
|
383
|
+
assert_match(/:on conditions for after_commit and after_rollback callbacks have to be one of \[:create, :destroy, :update\]/, e.message)
|
384
|
+
end
|
385
|
+
|
386
|
+
def test_after_commit_callbacks_should_validate_on_condition
|
387
|
+
assert_raise(ArgumentError) { Topic.after_commit(on: :save) }
|
388
|
+
e = assert_raise(ArgumentError) { Topic.after_commit(on: 'create') }
|
389
|
+
assert_match(/:on conditions for after_commit and after_rollback callbacks have to be one of \[:create, :destroy, :update\]/, e.message)
|
390
|
+
end
|
391
|
+
|
392
|
+
def test_saving_a_record_with_a_belongs_to_that_specifies_touching_the_parent_should_call_callbacks_on_the_parent_object
|
393
|
+
pet = Pet.first
|
394
|
+
owner = pet.owner
|
395
|
+
flag = false
|
396
|
+
|
397
|
+
owner.on_after_commit do
|
398
|
+
flag = true
|
399
|
+
end
|
400
|
+
|
401
|
+
pet.name = "Fluffy the Third"
|
402
|
+
pet.save
|
403
|
+
|
404
|
+
assert flag
|
405
|
+
end
|
406
|
+
|
407
|
+
private
|
408
|
+
|
409
|
+
def add_transaction_execution_blocks(record)
|
410
|
+
record.after_commit_block(:create) { |r| r.history << :commit_on_create }
|
411
|
+
record.after_commit_block(:update) { |r| r.history << :commit_on_update }
|
412
|
+
record.after_commit_block(:destroy) { |r| r.history << :commit_on_destroy }
|
413
|
+
record.after_rollback_block(:create) { |r| r.history << :rollback_on_create }
|
414
|
+
record.after_rollback_block(:update) { |r| r.history << :rollback_on_update }
|
415
|
+
record.after_rollback_block(:destroy) { |r| r.history << :rollback_on_destroy }
|
416
|
+
end
|
417
|
+
end
|
418
|
+
|
419
|
+
class CallbacksOnMultipleActionsTest < ActiveRecord::TestCase
|
420
|
+
self.use_transactional_fixtures = false
|
421
|
+
|
422
|
+
class TopicWithCallbacksOnMultipleActions < ActiveRecord::Base
|
423
|
+
self.table_name = :topics
|
424
|
+
|
425
|
+
after_commit(on: [:create, :destroy]) { |record| record.history << :create_and_destroy }
|
426
|
+
after_commit(on: [:create, :update]) { |record| record.history << :create_and_update }
|
427
|
+
after_commit(on: [:update, :destroy]) { |record| record.history << :update_and_destroy }
|
428
|
+
|
429
|
+
def clear_history
|
430
|
+
@history = []
|
431
|
+
end
|
432
|
+
|
433
|
+
def history
|
434
|
+
@history ||= []
|
435
|
+
end
|
436
|
+
end
|
437
|
+
|
438
|
+
def test_after_commit_on_multiple_actions
|
439
|
+
topic = TopicWithCallbacksOnMultipleActions.new
|
440
|
+
topic.save
|
441
|
+
assert_equal [:create_and_update, :create_and_destroy], topic.history
|
442
|
+
|
443
|
+
topic.clear_history
|
444
|
+
topic.approved = true
|
445
|
+
topic.save
|
446
|
+
assert_equal [:update_and_destroy, :create_and_update], topic.history
|
447
|
+
|
448
|
+
topic.clear_history
|
449
|
+
topic.destroy
|
450
|
+
assert_equal [:update_and_destroy, :create_and_destroy], topic.history
|
451
|
+
end
|
452
|
+
end
|