sequel 5.28.0 → 5.33.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG +64 -1922
- data/MIT-LICENSE +1 -1
- data/README.rdoc +1 -1
- data/doc/advanced_associations.rdoc +4 -4
- data/doc/association_basics.rdoc +3 -3
- data/doc/code_order.rdoc +12 -2
- data/doc/model_dataset_method_design.rdoc +1 -1
- data/doc/postgresql.rdoc +71 -0
- data/doc/release_notes/5.29.0.txt +22 -0
- data/doc/release_notes/5.30.0.txt +20 -0
- data/doc/release_notes/5.31.0.txt +148 -0
- data/doc/release_notes/5.32.0.txt +46 -0
- data/doc/release_notes/5.33.0.txt +24 -0
- data/doc/testing.rdoc +1 -1
- data/lib/sequel/adapters/postgres.rb +5 -1
- data/lib/sequel/adapters/shared/access.rb +6 -6
- data/lib/sequel/adapters/shared/mssql.rb +5 -5
- data/lib/sequel/adapters/shared/mysql.rb +10 -10
- data/lib/sequel/adapters/shared/oracle.rb +16 -16
- data/lib/sequel/adapters/shared/postgres.rb +169 -14
- data/lib/sequel/adapters/shared/sqlanywhere.rb +9 -9
- data/lib/sequel/adapters/shared/sqlite.rb +32 -5
- data/lib/sequel/adapters/tinytds.rb +1 -1
- data/lib/sequel/connection_pool/sharded_threaded.rb +2 -2
- data/lib/sequel/connection_pool/threaded.rb +1 -1
- data/lib/sequel/core.rb +318 -314
- data/lib/sequel/database/query.rb +1 -1
- data/lib/sequel/database/schema_generator.rb +1 -1
- data/lib/sequel/database/transactions.rb +8 -12
- data/lib/sequel/dataset/misc.rb +2 -2
- data/lib/sequel/extensions/connection_expiration.rb +2 -2
- data/lib/sequel/extensions/connection_validator.rb +2 -2
- data/lib/sequel/extensions/fiber_concurrency.rb +24 -0
- data/lib/sequel/extensions/index_caching.rb +9 -7
- data/lib/sequel/extensions/integer64.rb +2 -0
- data/lib/sequel/extensions/migration.rb +1 -1
- data/lib/sequel/extensions/pg_enum.rb +5 -2
- data/lib/sequel/extensions/pg_hstore.rb +6 -0
- data/lib/sequel/extensions/pg_inet.rb +13 -5
- data/lib/sequel/extensions/pg_interval.rb +2 -0
- data/lib/sequel/extensions/pg_range.rb +2 -0
- data/lib/sequel/extensions/pg_timestamptz.rb +2 -0
- data/lib/sequel/extensions/run_transaction_hooks.rb +72 -0
- data/lib/sequel/extensions/schema_dumper.rb +10 -4
- data/lib/sequel/extensions/server_block.rb +3 -3
- data/lib/sequel/model/associations.rb +18 -5
- data/lib/sequel/model/base.rb +60 -53
- data/lib/sequel/model/plugins.rb +1 -0
- data/lib/sequel/plugins/association_lazy_eager_option.rb +66 -0
- data/lib/sequel/plugins/association_multi_add_remove.rb +2 -0
- data/lib/sequel/plugins/association_proxies.rb +2 -0
- data/lib/sequel/plugins/boolean_subsets.rb +4 -1
- data/lib/sequel/plugins/class_table_inheritance.rb +26 -26
- data/lib/sequel/plugins/dirty.rb +13 -13
- data/lib/sequel/plugins/empty_failure_backtraces.rb +38 -0
- data/lib/sequel/plugins/forbid_lazy_load.rb +214 -0
- data/lib/sequel/plugins/json_serializer.rb +18 -11
- data/lib/sequel/plugins/single_table_inheritance.rb +15 -15
- data/lib/sequel/plugins/skip_saving_columns.rb +108 -0
- data/lib/sequel/plugins/subclasses.rb +2 -0
- data/lib/sequel/plugins/throw_failures.rb +1 -1
- data/lib/sequel/timezones.rb +6 -4
- data/lib/sequel/version.rb +1 -1
- metadata +18 -367
- data/Rakefile +0 -151
- data/doc/release_notes/4.0.0.txt +0 -262
- data/doc/release_notes/4.1.0.txt +0 -85
- data/doc/release_notes/4.10.0.txt +0 -226
- data/doc/release_notes/4.11.0.txt +0 -147
- data/doc/release_notes/4.12.0.txt +0 -105
- data/doc/release_notes/4.13.0.txt +0 -169
- data/doc/release_notes/4.14.0.txt +0 -68
- data/doc/release_notes/4.15.0.txt +0 -56
- data/doc/release_notes/4.16.0.txt +0 -36
- data/doc/release_notes/4.17.0.txt +0 -38
- data/doc/release_notes/4.18.0.txt +0 -36
- data/doc/release_notes/4.19.0.txt +0 -45
- data/doc/release_notes/4.2.0.txt +0 -129
- data/doc/release_notes/4.20.0.txt +0 -79
- data/doc/release_notes/4.21.0.txt +0 -94
- data/doc/release_notes/4.22.0.txt +0 -72
- data/doc/release_notes/4.23.0.txt +0 -65
- data/doc/release_notes/4.24.0.txt +0 -99
- data/doc/release_notes/4.25.0.txt +0 -181
- data/doc/release_notes/4.26.0.txt +0 -44
- data/doc/release_notes/4.27.0.txt +0 -78
- data/doc/release_notes/4.28.0.txt +0 -57
- data/doc/release_notes/4.29.0.txt +0 -41
- data/doc/release_notes/4.3.0.txt +0 -40
- data/doc/release_notes/4.30.0.txt +0 -37
- data/doc/release_notes/4.31.0.txt +0 -57
- data/doc/release_notes/4.32.0.txt +0 -132
- data/doc/release_notes/4.33.0.txt +0 -88
- data/doc/release_notes/4.34.0.txt +0 -86
- data/doc/release_notes/4.35.0.txt +0 -130
- data/doc/release_notes/4.36.0.txt +0 -116
- data/doc/release_notes/4.37.0.txt +0 -50
- data/doc/release_notes/4.38.0.txt +0 -67
- data/doc/release_notes/4.39.0.txt +0 -127
- data/doc/release_notes/4.4.0.txt +0 -92
- data/doc/release_notes/4.40.0.txt +0 -179
- data/doc/release_notes/4.41.0.txt +0 -77
- data/doc/release_notes/4.42.0.txt +0 -221
- data/doc/release_notes/4.43.0.txt +0 -87
- data/doc/release_notes/4.44.0.txt +0 -125
- data/doc/release_notes/4.45.0.txt +0 -370
- data/doc/release_notes/4.46.0.txt +0 -404
- data/doc/release_notes/4.47.0.txt +0 -56
- data/doc/release_notes/4.48.0.txt +0 -293
- data/doc/release_notes/4.49.0.txt +0 -222
- data/doc/release_notes/4.5.0.txt +0 -34
- data/doc/release_notes/4.6.0.txt +0 -30
- data/doc/release_notes/4.7.0.txt +0 -103
- data/doc/release_notes/4.8.0.txt +0 -175
- data/doc/release_notes/4.9.0.txt +0 -190
- data/spec/adapter_spec.rb +0 -4
- data/spec/adapters/db2_spec.rb +0 -170
- data/spec/adapters/mssql_spec.rb +0 -828
- data/spec/adapters/mysql_spec.rb +0 -1060
- data/spec/adapters/oracle_spec.rb +0 -371
- data/spec/adapters/postgres_spec.rb +0 -4476
- data/spec/adapters/spec_helper.rb +0 -44
- data/spec/adapters/sqlanywhere_spec.rb +0 -97
- data/spec/adapters/sqlite_spec.rb +0 -652
- data/spec/bin_spec.rb +0 -278
- data/spec/core/connection_pool_spec.rb +0 -1250
- data/spec/core/database_spec.rb +0 -2915
- data/spec/core/dataset_spec.rb +0 -5547
- data/spec/core/deprecated_spec.rb +0 -70
- data/spec/core/expression_filters_spec.rb +0 -1498
- data/spec/core/mock_adapter_spec.rb +0 -722
- data/spec/core/object_graph_spec.rb +0 -336
- data/spec/core/placeholder_literalizer_spec.rb +0 -166
- data/spec/core/schema_generator_spec.rb +0 -214
- data/spec/core/schema_spec.rb +0 -1844
- data/spec/core/spec_helper.rb +0 -24
- data/spec/core/version_spec.rb +0 -14
- data/spec/core_extensions_spec.rb +0 -763
- data/spec/core_model_spec.rb +0 -2
- data/spec/core_spec.rb +0 -1
- data/spec/deprecation_helper.rb +0 -30
- data/spec/extensions/accessed_columns_spec.rb +0 -51
- data/spec/extensions/active_model_spec.rb +0 -99
- data/spec/extensions/after_initialize_spec.rb +0 -28
- data/spec/extensions/any_not_empty_spec.rb +0 -23
- data/spec/extensions/arbitrary_servers_spec.rb +0 -109
- data/spec/extensions/association_dependencies_spec.rb +0 -125
- data/spec/extensions/association_multi_add_remove_spec.rb +0 -1041
- data/spec/extensions/association_pks_spec.rb +0 -423
- data/spec/extensions/association_proxies_spec.rb +0 -100
- data/spec/extensions/auto_literal_strings_spec.rb +0 -205
- data/spec/extensions/auto_validations_spec.rb +0 -229
- data/spec/extensions/blacklist_security_spec.rb +0 -95
- data/spec/extensions/blank_spec.rb +0 -69
- data/spec/extensions/boolean_readers_spec.rb +0 -93
- data/spec/extensions/boolean_subsets_spec.rb +0 -47
- data/spec/extensions/caching_spec.rb +0 -273
- data/spec/extensions/caller_logging_spec.rb +0 -52
- data/spec/extensions/class_table_inheritance_spec.rb +0 -750
- data/spec/extensions/column_conflicts_spec.rb +0 -75
- data/spec/extensions/column_select_spec.rb +0 -129
- data/spec/extensions/columns_introspection_spec.rb +0 -90
- data/spec/extensions/columns_updated_spec.rb +0 -35
- data/spec/extensions/composition_spec.rb +0 -248
- data/spec/extensions/connection_expiration_spec.rb +0 -151
- data/spec/extensions/connection_validator_spec.rb +0 -144
- data/spec/extensions/constant_sql_override_spec.rb +0 -24
- data/spec/extensions/constraint_validations_plugin_spec.rb +0 -300
- data/spec/extensions/constraint_validations_spec.rb +0 -439
- data/spec/extensions/core_refinements_spec.rb +0 -528
- data/spec/extensions/csv_serializer_spec.rb +0 -183
- data/spec/extensions/current_datetime_timestamp_spec.rb +0 -27
- data/spec/extensions/dataset_associations_spec.rb +0 -365
- data/spec/extensions/dataset_source_alias_spec.rb +0 -51
- data/spec/extensions/date_arithmetic_spec.rb +0 -181
- data/spec/extensions/datetime_parse_to_time_spec.rb +0 -169
- data/spec/extensions/def_dataset_method_spec.rb +0 -100
- data/spec/extensions/defaults_setter_spec.rb +0 -150
- data/spec/extensions/delay_add_association_spec.rb +0 -73
- data/spec/extensions/dirty_spec.rb +0 -222
- data/spec/extensions/duplicate_columns_handler_spec.rb +0 -104
- data/spec/extensions/eager_each_spec.rb +0 -62
- data/spec/extensions/eager_graph_eager_spec.rb +0 -100
- data/spec/extensions/empty_array_consider_nulls_spec.rb +0 -24
- data/spec/extensions/error_splitter_spec.rb +0 -18
- data/spec/extensions/error_sql_spec.rb +0 -20
- data/spec/extensions/escaped_like_spec.rb +0 -40
- data/spec/extensions/eval_inspect_spec.rb +0 -81
- data/spec/extensions/exclude_or_null_spec.rb +0 -15
- data/spec/extensions/finder_spec.rb +0 -260
- data/spec/extensions/force_encoding_spec.rb +0 -126
- data/spec/extensions/freeze_datasets_spec.rb +0 -31
- data/spec/extensions/graph_each_spec.rb +0 -113
- data/spec/extensions/hook_class_methods_spec.rb +0 -402
- data/spec/extensions/identifier_mangling_spec.rb +0 -201
- data/spec/extensions/implicit_subquery_spec.rb +0 -58
- data/spec/extensions/index_caching_spec.rb +0 -66
- data/spec/extensions/inflector_spec.rb +0 -183
- data/spec/extensions/input_transformer_spec.rb +0 -69
- data/spec/extensions/insert_conflict_spec.rb +0 -103
- data/spec/extensions/insert_returning_select_spec.rb +0 -72
- data/spec/extensions/instance_filters_spec.rb +0 -79
- data/spec/extensions/instance_hooks_spec.rb +0 -246
- data/spec/extensions/integer64_spec.rb +0 -22
- data/spec/extensions/inverted_subsets_spec.rb +0 -33
- data/spec/extensions/json_serializer_spec.rb +0 -336
- data/spec/extensions/lazy_attributes_spec.rb +0 -183
- data/spec/extensions/list_spec.rb +0 -291
- data/spec/extensions/looser_typecasting_spec.rb +0 -43
- data/spec/extensions/many_through_many_spec.rb +0 -2177
- data/spec/extensions/migration_spec.rb +0 -864
- data/spec/extensions/modification_detection_spec.rb +0 -93
- data/spec/extensions/mssql_optimistic_locking_spec.rb +0 -92
- data/spec/extensions/named_timezones_spec.rb +0 -218
- data/spec/extensions/nested_attributes_spec.rb +0 -815
- data/spec/extensions/null_dataset_spec.rb +0 -85
- data/spec/extensions/optimistic_locking_spec.rb +0 -127
- data/spec/extensions/pagination_spec.rb +0 -116
- data/spec/extensions/pg_array_associations_spec.rb +0 -802
- data/spec/extensions/pg_array_ops_spec.rb +0 -144
- data/spec/extensions/pg_array_spec.rb +0 -398
- data/spec/extensions/pg_auto_constraint_validations_spec.rb +0 -209
- data/spec/extensions/pg_enum_spec.rb +0 -118
- data/spec/extensions/pg_extended_date_support_spec.rb +0 -126
- data/spec/extensions/pg_hstore_ops_spec.rb +0 -238
- data/spec/extensions/pg_hstore_spec.rb +0 -219
- data/spec/extensions/pg_inet_ops_spec.rb +0 -102
- data/spec/extensions/pg_inet_spec.rb +0 -72
- data/spec/extensions/pg_interval_spec.rb +0 -103
- data/spec/extensions/pg_json_ops_spec.rb +0 -356
- data/spec/extensions/pg_json_spec.rb +0 -451
- data/spec/extensions/pg_loose_count_spec.rb +0 -23
- data/spec/extensions/pg_range_ops_spec.rb +0 -60
- data/spec/extensions/pg_range_spec.rb +0 -600
- data/spec/extensions/pg_row_ops_spec.rb +0 -61
- data/spec/extensions/pg_row_plugin_spec.rb +0 -60
- data/spec/extensions/pg_row_spec.rb +0 -363
- data/spec/extensions/pg_static_cache_updater_spec.rb +0 -93
- data/spec/extensions/pg_timestamptz_spec.rb +0 -17
- data/spec/extensions/prepared_statements_safe_spec.rb +0 -66
- data/spec/extensions/prepared_statements_spec.rb +0 -177
- data/spec/extensions/pretty_table_spec.rb +0 -123
- data/spec/extensions/query_spec.rb +0 -94
- data/spec/extensions/rcte_tree_spec.rb +0 -387
- data/spec/extensions/round_timestamps_spec.rb +0 -39
- data/spec/extensions/s_spec.rb +0 -60
- data/spec/extensions/schema_caching_spec.rb +0 -64
- data/spec/extensions/schema_dumper_spec.rb +0 -870
- data/spec/extensions/select_remove_spec.rb +0 -38
- data/spec/extensions/sequel_4_dataset_methods_spec.rb +0 -121
- data/spec/extensions/serialization_modification_detection_spec.rb +0 -98
- data/spec/extensions/serialization_spec.rb +0 -365
- data/spec/extensions/server_block_spec.rb +0 -135
- data/spec/extensions/server_logging_spec.rb +0 -45
- data/spec/extensions/sharding_spec.rb +0 -197
- data/spec/extensions/shared_caching_spec.rb +0 -151
- data/spec/extensions/single_table_inheritance_spec.rb +0 -347
- data/spec/extensions/singular_table_names_spec.rb +0 -22
- data/spec/extensions/skip_create_refresh_spec.rb +0 -18
- data/spec/extensions/spec_helper.rb +0 -70
- data/spec/extensions/split_array_nil_spec.rb +0 -24
- data/spec/extensions/split_values_spec.rb +0 -57
- data/spec/extensions/sql_comments_spec.rb +0 -33
- data/spec/extensions/sql_expr_spec.rb +0 -59
- data/spec/extensions/static_cache_cache_spec.rb +0 -35
- data/spec/extensions/static_cache_spec.rb +0 -471
- data/spec/extensions/string_agg_spec.rb +0 -90
- data/spec/extensions/string_date_time_spec.rb +0 -95
- data/spec/extensions/string_stripper_spec.rb +0 -68
- data/spec/extensions/subclasses_spec.rb +0 -79
- data/spec/extensions/subset_conditions_spec.rb +0 -38
- data/spec/extensions/symbol_aref_refinement_spec.rb +0 -28
- data/spec/extensions/symbol_as_refinement_spec.rb +0 -21
- data/spec/extensions/synchronize_sql_spec.rb +0 -124
- data/spec/extensions/table_select_spec.rb +0 -83
- data/spec/extensions/tactical_eager_loading_spec.rb +0 -402
- data/spec/extensions/thread_local_timezones_spec.rb +0 -67
- data/spec/extensions/throw_failures_spec.rb +0 -74
- data/spec/extensions/timestamps_spec.rb +0 -209
- data/spec/extensions/to_dot_spec.rb +0 -153
- data/spec/extensions/touch_spec.rb +0 -226
- data/spec/extensions/tree_spec.rb +0 -334
- data/spec/extensions/typecast_on_load_spec.rb +0 -86
- data/spec/extensions/unlimited_update_spec.rb +0 -21
- data/spec/extensions/update_or_create_spec.rb +0 -83
- data/spec/extensions/update_primary_key_spec.rb +0 -105
- data/spec/extensions/update_refresh_spec.rb +0 -59
- data/spec/extensions/uuid_spec.rb +0 -101
- data/spec/extensions/validate_associated_spec.rb +0 -52
- data/spec/extensions/validation_class_methods_spec.rb +0 -1040
- data/spec/extensions/validation_contexts_spec.rb +0 -31
- data/spec/extensions/validation_helpers_spec.rb +0 -525
- data/spec/extensions/whitelist_security_spec.rb +0 -157
- data/spec/extensions/xml_serializer_spec.rb +0 -213
- data/spec/files/bad_down_migration/001_create_alt_basic.rb +0 -4
- data/spec/files/bad_down_migration/002_create_alt_advanced.rb +0 -4
- data/spec/files/bad_timestamped_migrations/1273253849_create_sessions.rb +0 -9
- data/spec/files/bad_timestamped_migrations/1273253851_create_nodes.rb +0 -9
- data/spec/files/bad_timestamped_migrations/1273253853_3_create_users.rb +0 -3
- data/spec/files/bad_up_migration/001_create_alt_basic.rb +0 -4
- data/spec/files/bad_up_migration/002_create_alt_advanced.rb +0 -3
- data/spec/files/convert_to_timestamp_migrations/001_create_sessions.rb +0 -9
- data/spec/files/convert_to_timestamp_migrations/002_create_nodes.rb +0 -9
- data/spec/files/convert_to_timestamp_migrations/003_3_create_users.rb +0 -4
- data/spec/files/convert_to_timestamp_migrations/1273253850_create_artists.rb +0 -9
- data/spec/files/convert_to_timestamp_migrations/1273253852_create_albums.rb +0 -9
- data/spec/files/double_migration/001_create_sessions.rb +0 -9
- data/spec/files/double_migration/002_create_nodes.rb +0 -19
- data/spec/files/double_migration/003_3_create_users.rb +0 -4
- data/spec/files/duplicate_integer_migrations/001_create_alt_advanced.rb +0 -4
- data/spec/files/duplicate_integer_migrations/001_create_alt_basic.rb +0 -4
- data/spec/files/duplicate_timestamped_migrations/1273253849_create_sessions.rb +0 -9
- data/spec/files/duplicate_timestamped_migrations/1273253853_create_nodes.rb +0 -9
- data/spec/files/duplicate_timestamped_migrations/1273253853_create_users.rb +0 -4
- data/spec/files/empty_migration/001_create_sessions.rb +0 -9
- data/spec/files/empty_migration/002_create_nodes.rb +0 -0
- data/spec/files/empty_migration/003_3_create_users.rb +0 -4
- data/spec/files/integer_migrations/001_create_sessions.rb +0 -9
- data/spec/files/integer_migrations/002_create_nodes.rb +0 -9
- data/spec/files/integer_migrations/003_3_create_users.rb +0 -4
- data/spec/files/interleaved_timestamped_migrations/1273253849_create_sessions.rb +0 -9
- data/spec/files/interleaved_timestamped_migrations/1273253850_create_artists.rb +0 -9
- data/spec/files/interleaved_timestamped_migrations/1273253851_create_nodes.rb +0 -9
- data/spec/files/interleaved_timestamped_migrations/1273253852_create_albums.rb +0 -9
- data/spec/files/interleaved_timestamped_migrations/1273253853_3_create_users.rb +0 -4
- data/spec/files/missing_integer_migrations/001_create_alt_basic.rb +0 -4
- data/spec/files/missing_integer_migrations/003_create_alt_advanced.rb +0 -4
- data/spec/files/missing_timestamped_migrations/1273253849_create_sessions.rb +0 -9
- data/spec/files/missing_timestamped_migrations/1273253853_3_create_users.rb +0 -4
- data/spec/files/reversible_migrations/001_reversible.rb +0 -5
- data/spec/files/reversible_migrations/002_reversible.rb +0 -5
- data/spec/files/reversible_migrations/003_reversible.rb +0 -5
- data/spec/files/reversible_migrations/004_reversible.rb +0 -5
- data/spec/files/reversible_migrations/005_reversible.rb +0 -10
- data/spec/files/reversible_migrations/006_reversible.rb +0 -10
- data/spec/files/reversible_migrations/007_reversible.rb +0 -10
- data/spec/files/timestamped_migrations/1273253849_create_sessions.rb +0 -9
- data/spec/files/timestamped_migrations/1273253851_create_nodes.rb +0 -9
- data/spec/files/timestamped_migrations/1273253853_3_create_users.rb +0 -4
- data/spec/files/transaction_specified_migrations/001_create_alt_basic.rb +0 -4
- data/spec/files/transaction_specified_migrations/002_create_basic.rb +0 -4
- data/spec/files/transaction_unspecified_migrations/001_create_alt_basic.rb +0 -3
- data/spec/files/transaction_unspecified_migrations/002_create_basic.rb +0 -3
- data/spec/files/uppercase_timestamped_migrations/1273253849_CREATE_SESSIONS.RB +0 -9
- data/spec/files/uppercase_timestamped_migrations/1273253851_CREATE_NODES.RB +0 -9
- data/spec/files/uppercase_timestamped_migrations/1273253853_3_CREATE_USERS.RB +0 -4
- data/spec/guards_helper.rb +0 -59
- data/spec/integration/associations_test.rb +0 -2597
- data/spec/integration/database_test.rb +0 -113
- data/spec/integration/dataset_test.rb +0 -2037
- data/spec/integration/eager_loader_test.rb +0 -687
- data/spec/integration/migrator_test.rb +0 -262
- data/spec/integration/model_test.rb +0 -203
- data/spec/integration/plugin_test.rb +0 -2423
- data/spec/integration/prepared_statement_test.rb +0 -405
- data/spec/integration/schema_test.rb +0 -903
- data/spec/integration/spec_helper.rb +0 -71
- data/spec/integration/timezone_test.rb +0 -86
- data/spec/integration/transaction_test.rb +0 -603
- data/spec/integration/type_test.rb +0 -127
- data/spec/model/association_reflection_spec.rb +0 -803
- data/spec/model/associations_spec.rb +0 -4738
- data/spec/model/base_spec.rb +0 -875
- data/spec/model/class_dataset_methods_spec.rb +0 -146
- data/spec/model/dataset_methods_spec.rb +0 -198
- data/spec/model/eager_loading_spec.rb +0 -2377
- data/spec/model/hooks_spec.rb +0 -370
- data/spec/model/inflector_spec.rb +0 -26
- data/spec/model/model_spec.rb +0 -956
- data/spec/model/plugins_spec.rb +0 -429
- data/spec/model/record_spec.rb +0 -2118
- data/spec/model/spec_helper.rb +0 -46
- data/spec/model/validations_spec.rb +0 -220
- data/spec/model_no_assoc_spec.rb +0 -1
- data/spec/model_spec.rb +0 -1
- data/spec/plugin_spec.rb +0 -1
- data/spec/sequel_coverage.rb +0 -15
- data/spec/sequel_warning.rb +0 -4
- data/spec/spec_config.rb +0 -12
data/lib/sequel/model/base.rb
CHANGED
|
@@ -460,47 +460,6 @@ module Sequel
|
|
|
460
460
|
super
|
|
461
461
|
end
|
|
462
462
|
|
|
463
|
-
# If possible, set the dataset for the model subclass as soon as it
|
|
464
|
-
# is created. Also, make sure the inherited class instance variables
|
|
465
|
-
# are copied into the subclass.
|
|
466
|
-
#
|
|
467
|
-
# Sequel queries the database to get schema information as soon as
|
|
468
|
-
# a model class is created:
|
|
469
|
-
#
|
|
470
|
-
# class Artist < Sequel::Model # Causes schema query
|
|
471
|
-
# end
|
|
472
|
-
def inherited(subclass)
|
|
473
|
-
super
|
|
474
|
-
ivs = subclass.instance_variables
|
|
475
|
-
inherited_instance_variables.each do |iv, dup|
|
|
476
|
-
next if ivs.include?(iv)
|
|
477
|
-
if (sup_class_value = instance_variable_get(iv)) && dup
|
|
478
|
-
sup_class_value = case dup
|
|
479
|
-
when :dup
|
|
480
|
-
sup_class_value.dup
|
|
481
|
-
when :hash_dup
|
|
482
|
-
h = {}
|
|
483
|
-
sup_class_value.each{|k,v| h[k] = v.dup}
|
|
484
|
-
h
|
|
485
|
-
when Proc
|
|
486
|
-
dup.call(sup_class_value)
|
|
487
|
-
else
|
|
488
|
-
raise Error, "bad inherited instance variable type: #{dup.inspect}"
|
|
489
|
-
end
|
|
490
|
-
end
|
|
491
|
-
subclass.instance_variable_set(iv, sup_class_value)
|
|
492
|
-
end
|
|
493
|
-
|
|
494
|
-
unless ivs.include?(:@dataset)
|
|
495
|
-
if @dataset && self != Model
|
|
496
|
-
subclass.set_dataset(@dataset.clone, :inherited=>true)
|
|
497
|
-
elsif (n = subclass.name) && !n.to_s.empty?
|
|
498
|
-
db
|
|
499
|
-
subclass.set_dataset(subclass.implicit_table_name)
|
|
500
|
-
end
|
|
501
|
-
end
|
|
502
|
-
end
|
|
503
|
-
|
|
504
463
|
# Returns the implicit table name for the model class, which is the demodulized,
|
|
505
464
|
# underscored, pluralized name of the class.
|
|
506
465
|
#
|
|
@@ -515,12 +474,6 @@ module Sequel
|
|
|
515
474
|
call(values)
|
|
516
475
|
end
|
|
517
476
|
|
|
518
|
-
# Clear the setter_methods cache when a setter method is added.
|
|
519
|
-
def method_added(meth)
|
|
520
|
-
clear_setter_methods_cache if meth.to_s.end_with?('=')
|
|
521
|
-
super
|
|
522
|
-
end
|
|
523
|
-
|
|
524
477
|
# Mark the model as not having a primary key. Not having a primary key
|
|
525
478
|
# can cause issues, among which is that you won't be able to update records.
|
|
526
479
|
#
|
|
@@ -854,6 +807,47 @@ module Sequel
|
|
|
854
807
|
meths
|
|
855
808
|
end
|
|
856
809
|
|
|
810
|
+
# If possible, set the dataset for the model subclass as soon as it
|
|
811
|
+
# is created. Also, make sure the inherited class instance variables
|
|
812
|
+
# are copied into the subclass.
|
|
813
|
+
#
|
|
814
|
+
# Sequel queries the database to get schema information as soon as
|
|
815
|
+
# a model class is created:
|
|
816
|
+
#
|
|
817
|
+
# class Artist < Sequel::Model # Causes schema query
|
|
818
|
+
# end
|
|
819
|
+
def inherited(subclass)
|
|
820
|
+
super
|
|
821
|
+
ivs = subclass.instance_variables
|
|
822
|
+
inherited_instance_variables.each do |iv, dup|
|
|
823
|
+
next if ivs.include?(iv)
|
|
824
|
+
if (sup_class_value = instance_variable_get(iv)) && dup
|
|
825
|
+
sup_class_value = case dup
|
|
826
|
+
when :dup
|
|
827
|
+
sup_class_value.dup
|
|
828
|
+
when :hash_dup
|
|
829
|
+
h = {}
|
|
830
|
+
sup_class_value.each{|k,v| h[k] = v.dup}
|
|
831
|
+
h
|
|
832
|
+
when Proc
|
|
833
|
+
dup.call(sup_class_value)
|
|
834
|
+
else
|
|
835
|
+
raise Error, "bad inherited instance variable type: #{dup.inspect}"
|
|
836
|
+
end
|
|
837
|
+
end
|
|
838
|
+
subclass.instance_variable_set(iv, sup_class_value)
|
|
839
|
+
end
|
|
840
|
+
|
|
841
|
+
unless ivs.include?(:@dataset)
|
|
842
|
+
if @dataset && self != Model
|
|
843
|
+
subclass.set_dataset(@dataset.clone, :inherited=>true)
|
|
844
|
+
elsif (n = subclass.name) && !n.to_s.empty?
|
|
845
|
+
db
|
|
846
|
+
subclass.set_dataset(subclass.implicit_table_name)
|
|
847
|
+
end
|
|
848
|
+
end
|
|
849
|
+
end
|
|
850
|
+
|
|
857
851
|
# A hash of instance variables to automatically set up in subclasses.
|
|
858
852
|
# Keys are instance variable symbols, values should be:
|
|
859
853
|
# nil :: Assign directly from superclass to subclass (frozen objects)
|
|
@@ -908,6 +902,12 @@ module Sequel
|
|
|
908
902
|
opts[:class_name] ||= '::' + ((name || '').split("::")[0..-2] + [camelize(default)]).join('::')
|
|
909
903
|
end
|
|
910
904
|
|
|
905
|
+
# Clear the setter_methods cache when a setter method is added.
|
|
906
|
+
def method_added(meth)
|
|
907
|
+
clear_setter_methods_cache if meth.to_s.end_with?('=')
|
|
908
|
+
super
|
|
909
|
+
end
|
|
910
|
+
|
|
911
911
|
# Module that the class includes that holds methods the class adds for column accessors and
|
|
912
912
|
# associations so that the methods can be overridden with +super+.
|
|
913
913
|
def overridable_methods_module
|
|
@@ -1718,6 +1718,7 @@ module Sequel
|
|
|
1718
1718
|
|
|
1719
1719
|
# The values hash to use when inserting a new record.
|
|
1720
1720
|
alias _insert_values values
|
|
1721
|
+
private :_insert_values
|
|
1721
1722
|
|
|
1722
1723
|
# Refresh using a particular dataset, used inside save to make sure the same server
|
|
1723
1724
|
# is used for reading newly inserted values from the database
|
|
@@ -1772,14 +1773,12 @@ module Sequel
|
|
|
1772
1773
|
before_update
|
|
1773
1774
|
columns = opts[:columns]
|
|
1774
1775
|
if columns.nil?
|
|
1775
|
-
if opts[:changed]
|
|
1776
|
-
|
|
1777
|
-
columns_updated = @values.reject{|k,v| !cc.include?(k)}
|
|
1778
|
-
cc.clear
|
|
1776
|
+
columns_updated = if opts[:changed]
|
|
1777
|
+
_save_update_changed_colums_hash
|
|
1779
1778
|
else
|
|
1780
|
-
|
|
1781
|
-
_clear_changed_columns(:update)
|
|
1779
|
+
_save_update_all_columns_hash
|
|
1782
1780
|
end
|
|
1781
|
+
_clear_changed_columns(:update)
|
|
1783
1782
|
else # update only the specified columns
|
|
1784
1783
|
columns = Array(columns)
|
|
1785
1784
|
columns_updated = @values.reject{|k, v| !columns.include?(k)}
|
|
@@ -1827,6 +1826,14 @@ module Sequel
|
|
|
1827
1826
|
v
|
|
1828
1827
|
end
|
|
1829
1828
|
|
|
1829
|
+
# Return a hash of values used when saving changed columns of an
|
|
1830
|
+
# existing object. Defaults to all of the objects current values
|
|
1831
|
+
# that are recorded as modified.
|
|
1832
|
+
def _save_update_changed_colums_hash
|
|
1833
|
+
cc = changed_columns
|
|
1834
|
+
@values.reject{|k,v| !cc.include?(k)}
|
|
1835
|
+
end
|
|
1836
|
+
|
|
1830
1837
|
# Validate the object if validating on save. Skips validation
|
|
1831
1838
|
# completely (including validation hooks) if
|
|
1832
1839
|
# skip_validation_on_save! has been called on the object,
|
data/lib/sequel/model/plugins.rb
CHANGED
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
# frozen-string-literal: true
|
|
2
|
+
|
|
3
|
+
module Sequel
|
|
4
|
+
module Plugins
|
|
5
|
+
# The association_lazy_eager_option plugin supports passing
|
|
6
|
+
# an +:eager+ option to an association method. If the related
|
|
7
|
+
# association is already cached, the cached version will be
|
|
8
|
+
# returned. If the association is not already cached, it will
|
|
9
|
+
# be loaded, and the value of the +:eager+ option will be used
|
|
10
|
+
# to perform an eager load of the given associations.
|
|
11
|
+
# the plural versions.
|
|
12
|
+
#
|
|
13
|
+
# With Sequel's default behavior, you can already perform an
|
|
14
|
+
# eager load when lazy loading using a block:
|
|
15
|
+
#
|
|
16
|
+
# obj.association{|ds| ds.eager(:nested_association)}
|
|
17
|
+
#
|
|
18
|
+
# However, this will ignore any cached version. In more
|
|
19
|
+
# complex software, the association may already be cached
|
|
20
|
+
# and have the nested association cached inside of it, and
|
|
21
|
+
# using this callback approach then requires 2 unnecessary
|
|
22
|
+
# queries. This plugin will not perform any queries if the
|
|
23
|
+
# association is already cached, preventing duplicate work.
|
|
24
|
+
# However, you should make sure that an already loaded
|
|
25
|
+
# association has the nested association already eagerly
|
|
26
|
+
# loaded.
|
|
27
|
+
#
|
|
28
|
+
# Usage:
|
|
29
|
+
#
|
|
30
|
+
# # Make all model subclasses support the :eager association
|
|
31
|
+
# # method option (called before loading subclasses)
|
|
32
|
+
# Sequel::Model.plugin :association_lazy_eager_option
|
|
33
|
+
#
|
|
34
|
+
# # Make the Album class support the :eager association
|
|
35
|
+
# # method option
|
|
36
|
+
# Album.plugin :association_lazy_eager_option
|
|
37
|
+
module AssociationLazyEagerOption
|
|
38
|
+
module InstanceMethods
|
|
39
|
+
private
|
|
40
|
+
|
|
41
|
+
# Return a dataset for the association after applying any dynamic callback.
|
|
42
|
+
def _associated_dataset(opts, dynamic_opts)
|
|
43
|
+
ds = super
|
|
44
|
+
|
|
45
|
+
if eager = dynamic_opts[:eager]
|
|
46
|
+
ds = ds.eager(eager)
|
|
47
|
+
end
|
|
48
|
+
|
|
49
|
+
ds
|
|
50
|
+
end
|
|
51
|
+
|
|
52
|
+
# A placeholder literalizer that can be used to load the association, or nil to not use one.
|
|
53
|
+
def _associated_object_loader(opts, dynamic_opts)
|
|
54
|
+
return if dynamic_opts[:eager]
|
|
55
|
+
super
|
|
56
|
+
end
|
|
57
|
+
|
|
58
|
+
# Whether to use a simple primary key lookup on the associated class when loading.
|
|
59
|
+
def load_with_primary_key_lookup?(opts, dynamic_opts)
|
|
60
|
+
return false if dynamic_opts[:eager]
|
|
61
|
+
super
|
|
62
|
+
end
|
|
63
|
+
end
|
|
64
|
+
end
|
|
65
|
+
end
|
|
66
|
+
end
|
|
@@ -39,6 +39,8 @@ module Sequel
|
|
|
39
39
|
# Album.plugin :association_multi_add_remove
|
|
40
40
|
module AssociationMultiAddRemove
|
|
41
41
|
module ClassMethods
|
|
42
|
+
private
|
|
43
|
+
|
|
42
44
|
# Define the methods use to add/remove/set multiple associated objects
|
|
43
45
|
# in a single method call.
|
|
44
46
|
def def_association_instance_methods(opts)
|
|
@@ -109,6 +109,8 @@ module Sequel
|
|
|
109
109
|
|
|
110
110
|
Plugins.inherited_instance_variables(self, :@association_proxy_to_dataset=>nil)
|
|
111
111
|
|
|
112
|
+
private
|
|
113
|
+
|
|
112
114
|
# Changes the association method to return a proxy instead of the associated objects
|
|
113
115
|
# directly.
|
|
114
116
|
def def_association_method(opts)
|
|
@@ -31,7 +31,10 @@ module Sequel
|
|
|
31
31
|
# Create boolean subset methods for each boolean column.
|
|
32
32
|
def self.configure(model, &block)
|
|
33
33
|
model.instance_exec do
|
|
34
|
-
|
|
34
|
+
if block
|
|
35
|
+
define_singleton_method(:boolean_subset_args, &block)
|
|
36
|
+
singleton_class.send(:private, :boolean_subset_args)
|
|
37
|
+
end
|
|
35
38
|
create_boolean_subsets if @dataset
|
|
36
39
|
end
|
|
37
40
|
end
|
|
@@ -278,6 +278,27 @@ module Sequel
|
|
|
278
278
|
|
|
279
279
|
Plugins.inherited_instance_variables(self, :@cti_models=>nil, :@cti_tables=>nil, :@cti_table_columns=>nil, :@cti_instance_dataset=>nil, :@cti_table_map=>nil, :@cti_alias=>nil, :@cti_ignore_subclass_columns=>nil, :@cti_qualify_tables=>nil)
|
|
280
280
|
|
|
281
|
+
# The table name for the current model class's main table.
|
|
282
|
+
def table_name
|
|
283
|
+
if cti_tables && cti_tables.length > 1
|
|
284
|
+
@cti_alias
|
|
285
|
+
else
|
|
286
|
+
super
|
|
287
|
+
end
|
|
288
|
+
end
|
|
289
|
+
|
|
290
|
+
# The name of the most recently joined table.
|
|
291
|
+
def cti_table_name
|
|
292
|
+
cti_tables ? cti_tables.last : dataset.first_source_alias
|
|
293
|
+
end
|
|
294
|
+
|
|
295
|
+
# The model class for the given key value.
|
|
296
|
+
def sti_class_from_key(key)
|
|
297
|
+
sti_class(sti_model_map[key])
|
|
298
|
+
end
|
|
299
|
+
|
|
300
|
+
private
|
|
301
|
+
|
|
281
302
|
def inherited(subclass)
|
|
282
303
|
ds = sti_dataset
|
|
283
304
|
|
|
@@ -340,27 +361,6 @@ module Sequel
|
|
|
340
361
|
end
|
|
341
362
|
end
|
|
342
363
|
|
|
343
|
-
# The table name for the current model class's main table.
|
|
344
|
-
def table_name
|
|
345
|
-
if cti_tables && cti_tables.length > 1
|
|
346
|
-
@cti_alias
|
|
347
|
-
else
|
|
348
|
-
super
|
|
349
|
-
end
|
|
350
|
-
end
|
|
351
|
-
|
|
352
|
-
# The name of the most recently joined table.
|
|
353
|
-
def cti_table_name
|
|
354
|
-
cti_tables ? cti_tables.last : dataset.first_source_alias
|
|
355
|
-
end
|
|
356
|
-
|
|
357
|
-
# The model class for the given key value.
|
|
358
|
-
def sti_class_from_key(key)
|
|
359
|
-
sti_class(sti_model_map[key])
|
|
360
|
-
end
|
|
361
|
-
|
|
362
|
-
private
|
|
363
|
-
|
|
364
364
|
# If using a subquery for class table inheritance, also use a subquery
|
|
365
365
|
# when setting subclass dataset.
|
|
366
366
|
def sti_subclass_dataset(key)
|
|
@@ -383,11 +383,6 @@ module Sequel
|
|
|
383
383
|
self
|
|
384
384
|
end
|
|
385
385
|
|
|
386
|
-
# Don't allow use of prepared statements.
|
|
387
|
-
def use_prepared_statements_for?(type)
|
|
388
|
-
false
|
|
389
|
-
end
|
|
390
|
-
|
|
391
386
|
# Set the sti_key column based on the sti_key_map.
|
|
392
387
|
def before_validation
|
|
393
388
|
if new? && (set = self[model.sti_key])
|
|
@@ -438,6 +433,11 @@ module Sequel
|
|
|
438
433
|
end
|
|
439
434
|
end
|
|
440
435
|
end
|
|
436
|
+
|
|
437
|
+
# Don't allow use of prepared statements.
|
|
438
|
+
def use_prepared_statements_for?(type)
|
|
439
|
+
false
|
|
440
|
+
end
|
|
441
441
|
end
|
|
442
442
|
end
|
|
443
443
|
end
|
data/lib/sequel/plugins/dirty.rb
CHANGED
|
@@ -61,6 +61,19 @@ module Sequel
|
|
|
61
61
|
# that were used in the update statement.
|
|
62
62
|
attr_reader :previous_changes
|
|
63
63
|
|
|
64
|
+
# Reset the initial values after saving.
|
|
65
|
+
def after_save
|
|
66
|
+
super
|
|
67
|
+
reset_initial_values
|
|
68
|
+
end
|
|
69
|
+
|
|
70
|
+
# Save the current changes so they are available after updating. This happens
|
|
71
|
+
# before after_save resets them.
|
|
72
|
+
def after_update
|
|
73
|
+
super
|
|
74
|
+
@previous_changes = column_changes
|
|
75
|
+
end
|
|
76
|
+
|
|
64
77
|
# An array with the initial value and the current value
|
|
65
78
|
# of the column, if the column has been changed. If the
|
|
66
79
|
# column has not been changed, returns nil.
|
|
@@ -165,19 +178,6 @@ module Sequel
|
|
|
165
178
|
super
|
|
166
179
|
end
|
|
167
180
|
|
|
168
|
-
# Reset the initial values after saving.
|
|
169
|
-
def after_save
|
|
170
|
-
super
|
|
171
|
-
reset_initial_values
|
|
172
|
-
end
|
|
173
|
-
|
|
174
|
-
# Save the current changes so they are available after updating. This happens
|
|
175
|
-
# before after_save resets them.
|
|
176
|
-
def after_update
|
|
177
|
-
super
|
|
178
|
-
@previous_changes = column_changes
|
|
179
|
-
end
|
|
180
|
-
|
|
181
181
|
# When changing the column value, save the initial column value. If the column
|
|
182
182
|
# value is changed back to the initial value, update changed columns to remove
|
|
183
183
|
# the column.
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
# frozen-string-literal: true
|
|
2
|
+
|
|
3
|
+
module Sequel
|
|
4
|
+
module Plugins
|
|
5
|
+
# The empty_failure_backtraces plugin uses empty backtraces when raising HookFailed and ValidationFailed
|
|
6
|
+
# exceptions. This can be significantly faster, and if you are using these exceptions for
|
|
7
|
+
# flow control, you do not need the backtraces. This plugin is about 10% faster on CRuby
|
|
8
|
+
# and 10-15x faster on JRuby 9.2.7.0+. This does not have an effect on JRuby <9.2.7.0.
|
|
9
|
+
#
|
|
10
|
+
# Usage:
|
|
11
|
+
#
|
|
12
|
+
# # Make all model subclass instances use empty backtraces for HookFailed
|
|
13
|
+
# # and ValidationFailed exceptions (called before loading subclasses)
|
|
14
|
+
# Sequel::Model.plugin :empty_failure_backtraces
|
|
15
|
+
#
|
|
16
|
+
# # Make the Album class use empty backtraces for HookFailed and ValidationFailed exceptions
|
|
17
|
+
# Album.plugin :empty_failure_backtraces
|
|
18
|
+
module EmptyFailureBacktraces
|
|
19
|
+
module InstanceMethods
|
|
20
|
+
private
|
|
21
|
+
|
|
22
|
+
# Use empty backtrace for HookFailed exceptions.
|
|
23
|
+
def hook_failed_error(msg)
|
|
24
|
+
e = super
|
|
25
|
+
e.set_backtrace([])
|
|
26
|
+
e
|
|
27
|
+
end
|
|
28
|
+
|
|
29
|
+
# Use empty backtrace for ValidationFailed exceptions.
|
|
30
|
+
def validation_failed_error
|
|
31
|
+
e = super
|
|
32
|
+
e.set_backtrace([])
|
|
33
|
+
e
|
|
34
|
+
end
|
|
35
|
+
end
|
|
36
|
+
end
|
|
37
|
+
end
|
|
38
|
+
end
|
|
@@ -0,0 +1,214 @@
|
|
|
1
|
+
# frozen-string-literal: true
|
|
2
|
+
|
|
3
|
+
module Sequel
|
|
4
|
+
module Plugins
|
|
5
|
+
# The forbid_lazy_load plugin forbids lazy loading of associations
|
|
6
|
+
# for objects in cases where the object wasn't loaded with a
|
|
7
|
+
# method that only returns a single object.
|
|
8
|
+
#
|
|
9
|
+
# The main reason for doing this is it makes it easier to detect
|
|
10
|
+
# N+1 query issues. Note that Sequel also offers a
|
|
11
|
+
# tactical_eager_loading plugin which will automatically eagerly
|
|
12
|
+
# load associations for all objects retrived in the same query
|
|
13
|
+
# if any object would attempt to lazily load an association. That
|
|
14
|
+
# approach may be simpler if you are trying to prevent N+1 issues,
|
|
15
|
+
# though it does retain more objects in memory.
|
|
16
|
+
#
|
|
17
|
+
# This plugin offers multiple different ways to forbid lazy
|
|
18
|
+
# loading. You can forbid lazy loading associations for individual
|
|
19
|
+
# model instances:
|
|
20
|
+
#
|
|
21
|
+
# obj = Album[1]
|
|
22
|
+
# obj.forbid_lazy_load
|
|
23
|
+
# obj.artist # raises Sequel::Plugins::ForbidLazyLoad::Error
|
|
24
|
+
#
|
|
25
|
+
# +forbid_lazy_load+ is automatically called on instances if the
|
|
26
|
+
# instances are loaded via a method such as Dataset#all,
|
|
27
|
+
# Dataset#each, and other methods that load multiple instances
|
|
28
|
+
# at once. These are the cases where lazily loading associations
|
|
29
|
+
# for such instances can cause N+1 issues.
|
|
30
|
+
#
|
|
31
|
+
# Album.all.first.artist
|
|
32
|
+
# objs.first.artist # raises Sequel::Plugins::ForbidLazyLoad::Error
|
|
33
|
+
#
|
|
34
|
+
# Album.each do |obj|
|
|
35
|
+
# obj.artist # raises Sequel::Plugins::ForbidLazyLoad::Error
|
|
36
|
+
# end
|
|
37
|
+
#
|
|
38
|
+
# Album[1].artist # no error
|
|
39
|
+
#
|
|
40
|
+
# Album.first.artist # no error
|
|
41
|
+
#
|
|
42
|
+
# You can allow lazy loading associations for an instance that it
|
|
43
|
+
# was previously forbidden for:
|
|
44
|
+
#
|
|
45
|
+
# obj = Album.all.first
|
|
46
|
+
# obj.allow_lazy_load
|
|
47
|
+
# obj.artist # no error
|
|
48
|
+
#
|
|
49
|
+
# You can forbid lazy loading associations on a per-call basis,
|
|
50
|
+
# even if lazy loading of associations is allowed for the instance:
|
|
51
|
+
#
|
|
52
|
+
# obj = Album[1]
|
|
53
|
+
# obj.artist(forbid_lazy_load: true)
|
|
54
|
+
# # raises Sequel::Plugins::ForbidLazyLoad::Error
|
|
55
|
+
#
|
|
56
|
+
# This also works for allowing lazy loading associations for a
|
|
57
|
+
# specific association load even if it is forbidden for the instance:
|
|
58
|
+
#
|
|
59
|
+
# obj = Album.all.first
|
|
60
|
+
# obj.artist(forbid_lazy_load: false)
|
|
61
|
+
# # nothing raised
|
|
62
|
+
#
|
|
63
|
+
# You can also forbid lazy loading on a per-association basis using the
|
|
64
|
+
# +:forbid_lazy_load+ association option with a +true+ value:
|
|
65
|
+
#
|
|
66
|
+
# Album.many_to_one :artist, forbid_lazy_load: true
|
|
67
|
+
# Album[1].artist # raises Sequel::Plugins::ForbidLazyLoad::Error
|
|
68
|
+
#
|
|
69
|
+
# However, you probably don't want to do this as it will forbid any
|
|
70
|
+
# lazy loading of the association, even if the loading could not
|
|
71
|
+
# result in an N+1 issue.
|
|
72
|
+
#
|
|
73
|
+
# On the flip side, you can allow lazy loading using the
|
|
74
|
+
# +:forbid_lazy_load+ association option with a +false+ value:
|
|
75
|
+
#
|
|
76
|
+
# Album.many_to_one :artist, forbid_lazy_load: false
|
|
77
|
+
# Album.all.first.artist # no error
|
|
78
|
+
#
|
|
79
|
+
# One reason to do this is when using a plugin like static_cache
|
|
80
|
+
# on the associated model, where a query is not actually issued
|
|
81
|
+
# when doing a lazy association load. To make that particular
|
|
82
|
+
# case easier, this plugin makes Model.finalize_associations
|
|
83
|
+
# automatically set the association option if the associated
|
|
84
|
+
# class uses the static_cache plugin.
|
|
85
|
+
#
|
|
86
|
+
# Note that even with this plugin, there can still be N+1 issues,
|
|
87
|
+
# such as:
|
|
88
|
+
#
|
|
89
|
+
# Album.each do |obj| # 1 query for all albums
|
|
90
|
+
# Artist[obj.artist_id] # 1 query per album for each artist
|
|
91
|
+
# end
|
|
92
|
+
#
|
|
93
|
+
# Usage:
|
|
94
|
+
#
|
|
95
|
+
# # Make all model subclasses support forbidding lazy load
|
|
96
|
+
# # (called before loading subclasses)
|
|
97
|
+
# Sequel::Model.plugin :forbid_lazy_load
|
|
98
|
+
#
|
|
99
|
+
# # Make the Album class support forbidding lazy load
|
|
100
|
+
# Album.plugin :forbid_lazy_load
|
|
101
|
+
module ForbidLazyLoad
|
|
102
|
+
# Error raised when attempting to lazy load an association when
|
|
103
|
+
# lazy loading has been forbidden.
|
|
104
|
+
class Error < StandardError
|
|
105
|
+
end
|
|
106
|
+
|
|
107
|
+
module ClassMethods
|
|
108
|
+
Plugins.def_dataset_methods(self, :forbid_lazy_load)
|
|
109
|
+
|
|
110
|
+
# If the static_cache plugin is used by the associated class for
|
|
111
|
+
# an association, allow lazy loading that association, since the
|
|
112
|
+
# lazy association load will use a hash table lookup and not a query.
|
|
113
|
+
def allow_lazy_load_for_static_cache_associations
|
|
114
|
+
if defined?(::Sequel::Plugins::StaticCache::ClassMethods)
|
|
115
|
+
@association_reflections.each_value do |ref|
|
|
116
|
+
if ref.associated_class.is_a?(::Sequel::Plugins::StaticCache::ClassMethods)
|
|
117
|
+
ref[:forbid_lazy_load] = false
|
|
118
|
+
end
|
|
119
|
+
end
|
|
120
|
+
end
|
|
121
|
+
end
|
|
122
|
+
|
|
123
|
+
# Allow lazy loading for static cache associations before finalizing.
|
|
124
|
+
def finalize_associations
|
|
125
|
+
allow_lazy_load_for_static_cache_associations
|
|
126
|
+
super
|
|
127
|
+
end
|
|
128
|
+
end
|
|
129
|
+
|
|
130
|
+
module InstanceMethods
|
|
131
|
+
# Set this model instance to allow lazy loading of associations.
|
|
132
|
+
def allow_lazy_load
|
|
133
|
+
@forbid_lazy_load = false
|
|
134
|
+
self
|
|
135
|
+
end
|
|
136
|
+
|
|
137
|
+
# Set this model instance to not allow lazy loading of associations.
|
|
138
|
+
def forbid_lazy_load
|
|
139
|
+
@forbid_lazy_load = true
|
|
140
|
+
self
|
|
141
|
+
end
|
|
142
|
+
|
|
143
|
+
private
|
|
144
|
+
|
|
145
|
+
# Allow lazy loading for objects returned by singular associations.
|
|
146
|
+
def _load_associated_object(opts, dynamic_opts)
|
|
147
|
+
# The implementation that loads these associations does
|
|
148
|
+
# .all.first, which would result in the object returned being
|
|
149
|
+
# marked as forbidding lazy load.
|
|
150
|
+
obj = super
|
|
151
|
+
obj.allow_lazy_load if obj.is_a?(InstanceMethods)
|
|
152
|
+
obj
|
|
153
|
+
end
|
|
154
|
+
|
|
155
|
+
# Raise an Error if lazy loading has been forbidden for
|
|
156
|
+
# the instance, association, or call.
|
|
157
|
+
def _load_associated_objects(opts, dynamic_opts=OPTS)
|
|
158
|
+
case dynamic_opts[:forbid_lazy_load]
|
|
159
|
+
when false
|
|
160
|
+
# nothing
|
|
161
|
+
when nil
|
|
162
|
+
unless dynamic_opts[:reload]
|
|
163
|
+
case opts[:forbid_lazy_load]
|
|
164
|
+
when nil
|
|
165
|
+
raise Error, "lazy loading forbidden for this object (association: #{opts.inspect}, object: #{inspect})" if @forbid_lazy_load
|
|
166
|
+
when false
|
|
167
|
+
# nothing
|
|
168
|
+
else
|
|
169
|
+
raise Error, "lazy loading forbidden for this association (#{opts.inspect})"
|
|
170
|
+
end
|
|
171
|
+
end
|
|
172
|
+
else
|
|
173
|
+
raise Error, "lazy loading forbidden for this association method call (association: #{opts.inspect})"
|
|
174
|
+
end
|
|
175
|
+
|
|
176
|
+
super
|
|
177
|
+
end
|
|
178
|
+
end
|
|
179
|
+
|
|
180
|
+
module DatasetMethods
|
|
181
|
+
# Mark model instances retrieved in this call as forbidding lazy loading.
|
|
182
|
+
def each
|
|
183
|
+
if row_proc
|
|
184
|
+
super do |obj|
|
|
185
|
+
obj.forbid_lazy_load if obj.is_a?(InstanceMethods)
|
|
186
|
+
yield obj
|
|
187
|
+
end
|
|
188
|
+
else
|
|
189
|
+
super
|
|
190
|
+
end
|
|
191
|
+
end
|
|
192
|
+
|
|
193
|
+
# Mark model instances retrieved in this call as forbidding lazy loading.
|
|
194
|
+
def with_sql_each(sql)
|
|
195
|
+
if row_proc
|
|
196
|
+
super(sql) do |obj|
|
|
197
|
+
obj.forbid_lazy_load if obj.is_a?(InstanceMethods)
|
|
198
|
+
yield obj
|
|
199
|
+
end
|
|
200
|
+
else
|
|
201
|
+
super
|
|
202
|
+
end
|
|
203
|
+
end
|
|
204
|
+
|
|
205
|
+
# Mark model instances retrieved in this call as allowing lazy loading.
|
|
206
|
+
def with_sql_first(sql)
|
|
207
|
+
obj = super
|
|
208
|
+
obj.allow_lazy_load if obj.is_a?(InstanceMethods)
|
|
209
|
+
obj
|
|
210
|
+
end
|
|
211
|
+
end
|
|
212
|
+
end
|
|
213
|
+
end
|
|
214
|
+
end
|