sequel 5.45.0 → 5.77.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.
Files changed (218) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG +434 -0
  3. data/MIT-LICENSE +1 -1
  4. data/README.rdoc +59 -27
  5. data/bin/sequel +11 -3
  6. data/doc/advanced_associations.rdoc +16 -14
  7. data/doc/association_basics.rdoc +119 -24
  8. data/doc/cheat_sheet.rdoc +11 -3
  9. data/doc/mass_assignment.rdoc +1 -1
  10. data/doc/migration.rdoc +27 -6
  11. data/doc/model_hooks.rdoc +1 -1
  12. data/doc/object_model.rdoc +8 -8
  13. data/doc/opening_databases.rdoc +28 -12
  14. data/doc/postgresql.rdoc +16 -8
  15. data/doc/querying.rdoc +5 -3
  16. data/doc/release_notes/5.46.0.txt +87 -0
  17. data/doc/release_notes/5.47.0.txt +59 -0
  18. data/doc/release_notes/5.48.0.txt +14 -0
  19. data/doc/release_notes/5.49.0.txt +59 -0
  20. data/doc/release_notes/5.50.0.txt +78 -0
  21. data/doc/release_notes/5.51.0.txt +47 -0
  22. data/doc/release_notes/5.52.0.txt +87 -0
  23. data/doc/release_notes/5.53.0.txt +23 -0
  24. data/doc/release_notes/5.54.0.txt +27 -0
  25. data/doc/release_notes/5.55.0.txt +21 -0
  26. data/doc/release_notes/5.56.0.txt +51 -0
  27. data/doc/release_notes/5.57.0.txt +23 -0
  28. data/doc/release_notes/5.58.0.txt +31 -0
  29. data/doc/release_notes/5.59.0.txt +73 -0
  30. data/doc/release_notes/5.60.0.txt +22 -0
  31. data/doc/release_notes/5.61.0.txt +43 -0
  32. data/doc/release_notes/5.62.0.txt +132 -0
  33. data/doc/release_notes/5.63.0.txt +33 -0
  34. data/doc/release_notes/5.64.0.txt +50 -0
  35. data/doc/release_notes/5.65.0.txt +21 -0
  36. data/doc/release_notes/5.66.0.txt +24 -0
  37. data/doc/release_notes/5.67.0.txt +32 -0
  38. data/doc/release_notes/5.68.0.txt +61 -0
  39. data/doc/release_notes/5.69.0.txt +26 -0
  40. data/doc/release_notes/5.70.0.txt +35 -0
  41. data/doc/release_notes/5.71.0.txt +21 -0
  42. data/doc/release_notes/5.72.0.txt +33 -0
  43. data/doc/release_notes/5.73.0.txt +66 -0
  44. data/doc/release_notes/5.74.0.txt +45 -0
  45. data/doc/release_notes/5.75.0.txt +35 -0
  46. data/doc/release_notes/5.76.0.txt +86 -0
  47. data/doc/release_notes/5.77.0.txt +63 -0
  48. data/doc/schema_modification.rdoc +1 -1
  49. data/doc/security.rdoc +9 -9
  50. data/doc/sharding.rdoc +3 -1
  51. data/doc/sql.rdoc +27 -15
  52. data/doc/testing.rdoc +23 -13
  53. data/doc/transactions.rdoc +6 -6
  54. data/doc/virtual_rows.rdoc +1 -1
  55. data/lib/sequel/adapters/ado/access.rb +1 -1
  56. data/lib/sequel/adapters/ado.rb +1 -1
  57. data/lib/sequel/adapters/amalgalite.rb +3 -5
  58. data/lib/sequel/adapters/ibmdb.rb +3 -3
  59. data/lib/sequel/adapters/jdbc/derby.rb +8 -0
  60. data/lib/sequel/adapters/jdbc/h2.rb +63 -10
  61. data/lib/sequel/adapters/jdbc/hsqldb.rb +8 -0
  62. data/lib/sequel/adapters/jdbc/postgresql.rb +7 -4
  63. data/lib/sequel/adapters/jdbc/sqlanywhere.rb +15 -0
  64. data/lib/sequel/adapters/jdbc/sqlserver.rb +4 -0
  65. data/lib/sequel/adapters/jdbc.rb +24 -22
  66. data/lib/sequel/adapters/mysql.rb +92 -67
  67. data/lib/sequel/adapters/mysql2.rb +56 -51
  68. data/lib/sequel/adapters/odbc/mssql.rb +1 -1
  69. data/lib/sequel/adapters/odbc.rb +1 -1
  70. data/lib/sequel/adapters/oracle.rb +4 -3
  71. data/lib/sequel/adapters/postgres.rb +89 -45
  72. data/lib/sequel/adapters/shared/access.rb +11 -1
  73. data/lib/sequel/adapters/shared/db2.rb +42 -0
  74. data/lib/sequel/adapters/shared/mssql.rb +91 -10
  75. data/lib/sequel/adapters/shared/mysql.rb +78 -3
  76. data/lib/sequel/adapters/shared/oracle.rb +86 -7
  77. data/lib/sequel/adapters/shared/postgres.rb +576 -171
  78. data/lib/sequel/adapters/shared/sqlanywhere.rb +21 -5
  79. data/lib/sequel/adapters/shared/sqlite.rb +92 -8
  80. data/lib/sequel/adapters/sqlanywhere.rb +1 -1
  81. data/lib/sequel/adapters/sqlite.rb +99 -18
  82. data/lib/sequel/adapters/tinytds.rb +1 -1
  83. data/lib/sequel/adapters/trilogy.rb +117 -0
  84. data/lib/sequel/adapters/utils/columns_limit_1.rb +22 -0
  85. data/lib/sequel/adapters/utils/mysql_mysql2.rb +1 -1
  86. data/lib/sequel/ast_transformer.rb +6 -0
  87. data/lib/sequel/connection_pool/sharded_single.rb +5 -7
  88. data/lib/sequel/connection_pool/sharded_threaded.rb +16 -11
  89. data/lib/sequel/connection_pool/sharded_timed_queue.rb +374 -0
  90. data/lib/sequel/connection_pool/single.rb +6 -8
  91. data/lib/sequel/connection_pool/threaded.rb +14 -8
  92. data/lib/sequel/connection_pool/timed_queue.rb +270 -0
  93. data/lib/sequel/connection_pool.rb +57 -31
  94. data/lib/sequel/core.rb +17 -18
  95. data/lib/sequel/database/connecting.rb +27 -3
  96. data/lib/sequel/database/dataset.rb +16 -6
  97. data/lib/sequel/database/misc.rb +70 -14
  98. data/lib/sequel/database/query.rb +73 -2
  99. data/lib/sequel/database/schema_generator.rb +11 -6
  100. data/lib/sequel/database/schema_methods.rb +23 -4
  101. data/lib/sequel/database/transactions.rb +6 -0
  102. data/lib/sequel/dataset/actions.rb +111 -15
  103. data/lib/sequel/dataset/deprecated_singleton_class_methods.rb +42 -0
  104. data/lib/sequel/dataset/features.rb +20 -1
  105. data/lib/sequel/dataset/misc.rb +12 -2
  106. data/lib/sequel/dataset/placeholder_literalizer.rb +20 -9
  107. data/lib/sequel/dataset/query.rb +170 -41
  108. data/lib/sequel/dataset/sql.rb +190 -71
  109. data/lib/sequel/dataset.rb +4 -0
  110. data/lib/sequel/extensions/_model_pg_row.rb +0 -12
  111. data/lib/sequel/extensions/_pretty_table.rb +1 -1
  112. data/lib/sequel/extensions/any_not_empty.rb +2 -2
  113. data/lib/sequel/extensions/async_thread_pool.rb +14 -13
  114. data/lib/sequel/extensions/auto_cast_date_and_time.rb +94 -0
  115. data/lib/sequel/extensions/auto_literal_strings.rb +1 -1
  116. data/lib/sequel/extensions/connection_expiration.rb +15 -9
  117. data/lib/sequel/extensions/connection_validator.rb +16 -11
  118. data/lib/sequel/extensions/constraint_validations.rb +1 -1
  119. data/lib/sequel/extensions/core_refinements.rb +36 -11
  120. data/lib/sequel/extensions/date_arithmetic.rb +36 -8
  121. data/lib/sequel/extensions/date_parse_input_handler.rb +67 -0
  122. data/lib/sequel/extensions/datetime_parse_to_time.rb +5 -1
  123. data/lib/sequel/extensions/duplicate_columns_handler.rb +11 -10
  124. data/lib/sequel/extensions/index_caching.rb +5 -1
  125. data/lib/sequel/extensions/inflector.rb +1 -1
  126. data/lib/sequel/extensions/is_distinct_from.rb +141 -0
  127. data/lib/sequel/extensions/looser_typecasting.rb +3 -0
  128. data/lib/sequel/extensions/migration.rb +57 -15
  129. data/lib/sequel/extensions/named_timezones.rb +22 -6
  130. data/lib/sequel/extensions/pagination.rb +1 -1
  131. data/lib/sequel/extensions/pg_array.rb +33 -4
  132. data/lib/sequel/extensions/pg_array_ops.rb +2 -2
  133. data/lib/sequel/extensions/pg_auto_parameterize.rb +509 -0
  134. data/lib/sequel/extensions/pg_auto_parameterize_in_array.rb +110 -0
  135. data/lib/sequel/extensions/pg_enum.rb +1 -2
  136. data/lib/sequel/extensions/pg_extended_date_support.rb +39 -28
  137. data/lib/sequel/extensions/pg_extended_integer_support.rb +116 -0
  138. data/lib/sequel/extensions/pg_hstore.rb +6 -1
  139. data/lib/sequel/extensions/pg_hstore_ops.rb +53 -3
  140. data/lib/sequel/extensions/pg_inet.rb +10 -11
  141. data/lib/sequel/extensions/pg_inet_ops.rb +1 -1
  142. data/lib/sequel/extensions/pg_interval.rb +11 -11
  143. data/lib/sequel/extensions/pg_json.rb +13 -15
  144. data/lib/sequel/extensions/pg_json_ops.rb +125 -2
  145. data/lib/sequel/extensions/pg_multirange.rb +367 -0
  146. data/lib/sequel/extensions/pg_range.rb +13 -26
  147. data/lib/sequel/extensions/pg_range_ops.rb +37 -9
  148. data/lib/sequel/extensions/pg_row.rb +20 -19
  149. data/lib/sequel/extensions/pg_row_ops.rb +1 -1
  150. data/lib/sequel/extensions/pg_timestamptz.rb +27 -3
  151. data/lib/sequel/extensions/round_timestamps.rb +1 -1
  152. data/lib/sequel/extensions/s.rb +2 -1
  153. data/lib/sequel/extensions/schema_caching.rb +1 -1
  154. data/lib/sequel/extensions/schema_dumper.rb +45 -11
  155. data/lib/sequel/extensions/server_block.rb +10 -13
  156. data/lib/sequel/extensions/set_literalizer.rb +58 -0
  157. data/lib/sequel/extensions/sql_comments.rb +110 -3
  158. data/lib/sequel/extensions/sql_log_normalizer.rb +108 -0
  159. data/lib/sequel/extensions/sqlite_json_ops.rb +255 -0
  160. data/lib/sequel/extensions/string_agg.rb +1 -1
  161. data/lib/sequel/extensions/string_date_time.rb +19 -23
  162. data/lib/sequel/extensions/symbol_aref.rb +2 -0
  163. data/lib/sequel/extensions/transaction_connection_validator.rb +78 -0
  164. data/lib/sequel/model/associations.rb +286 -92
  165. data/lib/sequel/model/base.rb +53 -33
  166. data/lib/sequel/model/dataset_module.rb +3 -0
  167. data/lib/sequel/model/errors.rb +10 -1
  168. data/lib/sequel/model/exceptions.rb +15 -3
  169. data/lib/sequel/model/inflections.rb +1 -1
  170. data/lib/sequel/plugins/auto_restrict_eager_graph.rb +62 -0
  171. data/lib/sequel/plugins/auto_validations.rb +74 -16
  172. data/lib/sequel/plugins/class_table_inheritance.rb +2 -2
  173. data/lib/sequel/plugins/column_encryption.rb +29 -8
  174. data/lib/sequel/plugins/composition.rb +3 -2
  175. data/lib/sequel/plugins/concurrent_eager_loading.rb +4 -4
  176. data/lib/sequel/plugins/constraint_validations.rb +8 -5
  177. data/lib/sequel/plugins/defaults_setter.rb +16 -0
  178. data/lib/sequel/plugins/dirty.rb +1 -1
  179. data/lib/sequel/plugins/enum.rb +124 -0
  180. data/lib/sequel/plugins/finder.rb +4 -2
  181. data/lib/sequel/plugins/insert_conflict.rb +4 -0
  182. data/lib/sequel/plugins/instance_specific_default.rb +1 -1
  183. data/lib/sequel/plugins/json_serializer.rb +2 -2
  184. data/lib/sequel/plugins/lazy_attributes.rb +3 -0
  185. data/lib/sequel/plugins/list.rb +8 -3
  186. data/lib/sequel/plugins/many_through_many.rb +109 -10
  187. data/lib/sequel/plugins/mssql_optimistic_locking.rb +8 -38
  188. data/lib/sequel/plugins/nested_attributes.rb +4 -4
  189. data/lib/sequel/plugins/optimistic_locking.rb +9 -42
  190. data/lib/sequel/plugins/optimistic_locking_base.rb +55 -0
  191. data/lib/sequel/plugins/paged_operations.rb +181 -0
  192. data/lib/sequel/plugins/pg_array_associations.rb +46 -34
  193. data/lib/sequel/plugins/pg_auto_constraint_validations.rb +9 -3
  194. data/lib/sequel/plugins/pg_xmin_optimistic_locking.rb +109 -0
  195. data/lib/sequel/plugins/prepared_statements.rb +12 -2
  196. data/lib/sequel/plugins/prepared_statements_safe.rb +2 -1
  197. data/lib/sequel/plugins/primary_key_lookup_check_values.rb +154 -0
  198. data/lib/sequel/plugins/rcte_tree.rb +7 -4
  199. data/lib/sequel/plugins/require_valid_schema.rb +67 -0
  200. data/lib/sequel/plugins/serialization.rb +1 -0
  201. data/lib/sequel/plugins/serialization_modification_detection.rb +1 -0
  202. data/lib/sequel/plugins/single_table_inheritance.rb +8 -0
  203. data/lib/sequel/plugins/sql_comments.rb +189 -0
  204. data/lib/sequel/plugins/static_cache.rb +39 -1
  205. data/lib/sequel/plugins/static_cache_cache.rb +5 -1
  206. data/lib/sequel/plugins/subclasses.rb +28 -11
  207. data/lib/sequel/plugins/tactical_eager_loading.rb +23 -10
  208. data/lib/sequel/plugins/timestamps.rb +1 -1
  209. data/lib/sequel/plugins/unused_associations.rb +521 -0
  210. data/lib/sequel/plugins/update_or_create.rb +1 -1
  211. data/lib/sequel/plugins/validate_associated.rb +22 -12
  212. data/lib/sequel/plugins/validation_helpers.rb +41 -11
  213. data/lib/sequel/plugins/validation_helpers_generic_type_messages.rb +73 -0
  214. data/lib/sequel/plugins/xml_serializer.rb +1 -1
  215. data/lib/sequel/sql.rb +1 -1
  216. data/lib/sequel/timezones.rb +12 -14
  217. data/lib/sequel/version.rb +1 -1
  218. metadata +109 -19
@@ -109,6 +109,13 @@ module Sequel
109
109
  # to eagerly set the associated objects, and having separate threads
110
110
  # modify the same model instance is not thread-safe.
111
111
  #
112
+ # Because this plugin will automatically use eager loading for
113
+ # performance, it can break code that defines associations that
114
+ # do not support eager loading, without marking that they do not
115
+ # support eager loading via the <tt>allow_eager: false</tt> option.
116
+ # Make sure to set <tt>allow_eager: false</tt> on any association
117
+ # used with this plugin if the association doesn't support eager loading.
118
+ #
112
119
  # Usage:
113
120
  #
114
121
  # # Make all model subclass instances use tactical eager loading (called before loading subclasses)
@@ -143,19 +150,25 @@ module Sequel
143
150
  def load_associated_objects(opts, dynamic_opts=OPTS, &block)
144
151
  dynamic_opts = load_association_objects_options(dynamic_opts, &block)
145
152
  name = opts[:name]
146
- if (!associations.include?(name) || dynamic_opts[:eager_reload]) && opts[:allow_eager] != false && retrieved_by && !frozen? && !dynamic_opts[:callback] && !dynamic_opts[:reload]
147
- begin
148
- retrieved_by.send(:eager_load, retrieved_with.reject(&:frozen?), name=>dynamic_opts[:eager] || OPTS)
149
- rescue Sequel::UndefinedAssociation
150
- # This can happen if class table inheritance is used and the association
151
- # is only defined in a subclass. This particular instance can use the
152
- # association, but it can't be eagerly loaded as the parent class doesn't
153
- # have access to the association, and that's the class doing the eager loading.
154
- nil
155
- end
153
+ eager_reload = dynamic_opts[:eager_reload]
154
+ if (!associations.include?(name) || eager_reload) && opts[:allow_eager] != false && retrieved_by && !frozen? && !dynamic_opts[:callback] && !dynamic_opts[:reload]
155
+ retrieved_by.send(:eager_load, _filter_tactical_eager_load_objects(:eager_reload=>eager_reload, :name=>name), {name=>dynamic_opts[:eager] || OPTS}, model)
156
156
  end
157
157
  super
158
158
  end
159
+
160
+ # Filter the objects used when tactical eager loading.
161
+ # By default, this removes frozen objects and objects that alreayd have the association loaded
162
+ def _filter_tactical_eager_load_objects(opts)
163
+ objects = defined?(super) ? super : retrieved_with.dup
164
+ if opts[:eager_reload]
165
+ objects.reject!(&:frozen?)
166
+ else
167
+ name = opts[:name]
168
+ objects.reject!{|x| x.frozen? || x.associations.include?(name)}
169
+ end
170
+ objects
171
+ end
159
172
  end
160
173
 
161
174
  module DatasetMethods
@@ -19,7 +19,7 @@ module Sequel
19
19
  #
20
20
  # # Timestamp Artist instances, forcing an overwrite of the create
21
21
  # # timestamp, and setting the update timestamp when creating
22
- # Album.plugin :timestamps, force: true, update_on_create: true
22
+ # Artist.plugin :timestamps, force: true, update_on_create: true
23
23
  module Timestamps
24
24
  # Configure the plugin by setting the available options. Note that
25
25
  # if this method is run more than once, previous settings are ignored,
@@ -0,0 +1,521 @@
1
+ # frozen-string-literal: true
2
+
3
+ # :nocov:
4
+
5
+ # This entire file is excluded from coverage testing. This is because it
6
+ # requires coverage testing to work, and if you've already loaded Sequel
7
+ # without enabling coverage, then coverage testing won't work correctly
8
+ # for methods defined by Sequel.
9
+ #
10
+ # While automated coverage testing is disabled, manual coverage testing
11
+ # was used during spec development to make sure this code is 100% covered.
12
+
13
+ if RUBY_VERSION < '2.5'
14
+ raise LoadError, "The Sequel unused_associations plugin depends on Ruby 2.5+ method coverage"
15
+ end
16
+
17
+ require 'coverage'
18
+ require 'json'
19
+
20
+ module Sequel
21
+ module Plugins
22
+ # The unused_associations plugin detects which model associations are not
23
+ # used and can be removed, and which model association methods are not used
24
+ # and can skip being defined. The advantage of removing unused associations
25
+ # and unused association methods is decreased memory usage, since each
26
+ # method defined takes memory and adds more work for the garbage collector.
27
+ #
28
+ # In order to detect which associations are used, this relies on the method
29
+ # coverage support added in Ruby 2.5. To allow flexibility to override
30
+ # association methods, the association methods that Sequel defines are
31
+ # defined in a module included in the class instead of directly in the
32
+ # class. Unfortunately, that makes it difficult to directly use the
33
+ # coverage data to find unused associations. The advantage of this plugin
34
+ # is that it is able to figure out from the coverage information whether
35
+ # the association methods Sequel defines are actually used.
36
+ #
37
+ # = Basic Usage
38
+ #
39
+ # The expected usage of the unused_associations plugin is to load it
40
+ # into the base class for models in your application, which will often
41
+ # be Sequel::Model:
42
+ #
43
+ # Sequel::Model.plugin :unused_associations
44
+ #
45
+ # Then you run your test suite with method coverage enabled, passing the
46
+ # coverage result to +update_associations_coverage+.
47
+ # +update_associations_coverage+ returns a data structure containing
48
+ # method coverage information for all subclasses of the base class.
49
+ # You can pass the coverage information to
50
+ # +update_unused_associations_data+, which will return a data structure
51
+ # with information on unused associations.
52
+ #
53
+ # require 'coverage'
54
+ # Coverage.start(methods: true)
55
+ # # load sequel after starting coverage, then run your tests
56
+ # cov_data = Sequel::Model.update_associations_coverage
57
+ # unused_associations_data = Sequel::Model.update_unused_associations_data(coverage_data: cov_data)
58
+ #
59
+ # You can take that unused association data and pass it to the
60
+ # +unused_associations+ method to get a array of information on
61
+ # associations which have not been used. Each entry in the array
62
+ # will contain a class name and association name for each unused
63
+ # association, both as a string:
64
+ #
65
+ # Sequel::Model.unused_associations(unused_associations_data: unused_associations_data)
66
+ # # => [["Class1", "assoc1"], ...]
67
+ #
68
+ # You can use the output of the +unused_associations+ method to determine
69
+ # which associations are not used at all in your application, and can
70
+ # be eliminiated.
71
+ #
72
+ # You can also take that unused association data and pass it to the
73
+ # +unused_association_options+ method, which will return an array of
74
+ # information on associations which are used, but have related methods
75
+ # defined that are not used. The first two entries in each array are
76
+ # the class name and association name as a string, and the third
77
+ # entry is a hash of association options:
78
+ #
79
+ # Sequel::Model.unused_association_options(unused_associations_data: unused_associations_data)
80
+ # # => [["Class2", "assoc2", {:read_only=>true}], ...]
81
+ #
82
+ # You can use the output of the +unused_association_options+ to
83
+ # find out which association options can be provided when defining
84
+ # the association so that the association method will not define
85
+ # methods that are not used.
86
+ #
87
+ # = Combining Coverage Results
88
+ #
89
+ # It is common to want to combine results from multiple separate
90
+ # coverage runs. For example, if you have multiple test suites
91
+ # for your application, one for model or unit tests and one for
92
+ # web or integration tests, you would want to combine the
93
+ # coverage information from all test suites before determining
94
+ # that the associations are not used.
95
+ #
96
+ # The unused_associations plugin supports combining multiple
97
+ # coverage results using the :coverage_file plugin option:
98
+ #
99
+ # Sequel::Model.plugin :unused_associations,
100
+ # coverage_file: 'unused_associations_coverage.json'
101
+ #
102
+ # With the coverage file option, +update_associations_coverage+
103
+ # will look in the given file for existing coverage information,
104
+ # if it exists. If the file exists, the data from it will be
105
+ # merged with the coverage result passed to the method.
106
+ # Before returning, the coverage file will be updated with the
107
+ # merged result. When using the :coverage_file plugin option,
108
+ # you can each of your test suites update the coverage
109
+ # information:
110
+ #
111
+ # require 'coverage'
112
+ # Coverage.start(methods: true)
113
+ # # run this test suite
114
+ # Sequel::Model.update_associations_coverage
115
+ #
116
+ # After all test suites have been run, you can run
117
+ # +update_unused_associations_data+, without an argument:
118
+ #
119
+ # unused_associations_data = Sequel::Model.update_unused_associations_data
120
+ #
121
+ # With no argument, +update_unused_associations_data+ will get
122
+ # the coverage data from the coverage file, and then use that
123
+ # to prepare the information. You can then use the returned
124
+ # value the same as before to get the data on unused associations.
125
+ # To prevent stale coverage information, calling
126
+ # +update_unused_associations_data+ when using the :coverage_file
127
+ # plugin option will remove the coverage file by default (you can
128
+ # use the :keep_coverage option to prevent the deletion of the
129
+ # coverage file).
130
+ #
131
+ # = Automatic Usage of Unused Association Data
132
+ #
133
+ # Since it can be a pain to manually update all of your code
134
+ # to remove unused assocations or add options to prevent the
135
+ # definition of unused associations, the unused_associations
136
+ # plugin comes with support to take previously saved unused
137
+ # association data, and use it to not create unused associations,
138
+ # and to automatically use the appropriate options so that unused
139
+ # association methods are not created.
140
+ #
141
+ # To use this option, you first need to save the unused association
142
+ # data previously prepared. You can do this by passing an
143
+ # :file option when loading the plugin.
144
+ #
145
+ # Sequel::Model.plugin :unused_associations,
146
+ # file: 'unused_associations.json'
147
+ #
148
+ # With the :file option provided, you no longer need to use
149
+ # the return value of +update_unused_associations_data+, as
150
+ # the file will be updated with the information:
151
+ #
152
+ # Sequel::Model.update_unused_associations_data(coverage_data: cov_data)
153
+ #
154
+ # Then, to use the saved unused associations data, add the
155
+ # :modify_associations plugin option:
156
+ #
157
+ # Sequel::Model.plugin :unused_associations,
158
+ # file: 'unused_associations.json',
159
+ # modify_associations: true
160
+ #
161
+ # With the :modify_associations used, and the unused association
162
+ # data file is available, when subclasses attempt to create an
163
+ # unused association, the attempt will be ignored. If the
164
+ # subclasses attempt to create an association where not
165
+ # all association methods are used, the plugin will automatically
166
+ # set the appropriate options so that the unused association
167
+ # methods are not defined.
168
+ #
169
+ # When you are testing which associations are used, make sure
170
+ # not to set the :modify_associations plugin option, or make sure
171
+ # that the unused associations data file does not exist.
172
+ #
173
+ # == Automatic Usage with Combined Coverage Results
174
+ #
175
+ # If you have multiple test suites and want to automatically
176
+ # use the unused association data, you should provide both
177
+ # :file and :coverage_file options when loading the plugin:
178
+ #
179
+ # Sequel::Model.plugin :unused_associations,
180
+ # file: 'unused_associations.json',
181
+ # coverage_file: 'unused_associations_coverage.json'
182
+ #
183
+ # Then each test suite just needs to run
184
+ # +update_associations_coverage+ to update the coverage information:
185
+ #
186
+ # Sequel::Model.update_associations_coverage
187
+ #
188
+ # After all test suites have been run, you can run
189
+ # +update_unused_associations_data+ to update the unused
190
+ # association data file (and remove the coverage file):
191
+ #
192
+ # Sequel::Model.update_unused_associations_data
193
+ #
194
+ # Then you can add the :modify_associations plugin option to
195
+ # automatically use the unused association data.
196
+ #
197
+ # = Caveats
198
+ #
199
+ # Since this plugin is based on coverage information, if you do
200
+ # not have tests that cover all usage of associations in your
201
+ # application, you can end up with coverage that shows the
202
+ # association is not used, when it is used in code that is not
203
+ # covered. The output of plugin can still be useful in such cases,
204
+ # as long as you are manually checking it. However, you should
205
+ # avoid using the :modify_associations unless you have
206
+ # confidence that your tests cover all usage of associations
207
+ # in your application. You can specify the :is_used association
208
+ # option for any association that you know is used. If an
209
+ # association uses the :is_used association option, this plugin
210
+ # will not modify it if the :modify_associations option is used.
211
+ #
212
+ # This plugin does not handle anonymous classes. Any unused
213
+ # associations defined in anonymous classes will not be
214
+ # reported by this plugin.
215
+ #
216
+ # This plugin only considers the public instance methods the
217
+ # association defines, and direct access to the related
218
+ # association reflection via Sequel::Model.association_reflection
219
+ # to determine if the association was used. If the association
220
+ # metadata was accessed another way, it's possible this plugin
221
+ # will show the association as unused.
222
+ #
223
+ # As this relies on the method coverage added in Ruby 2.5, it does
224
+ # not work on older versions of Ruby. It also does not work on
225
+ # JRuby, as JRuby does not implement method coverage.
226
+ module UnusedAssociations
227
+ # Load the subclasses plugin, as the unused associations plugin
228
+ # is designed to handle all subclasses of the class it is loaded
229
+ # into.
230
+ def self.apply(mod, opts=OPTS)
231
+ mod.plugin :subclasses
232
+ end
233
+
234
+ # Plugin options:
235
+ # :coverage_file :: The file to store the coverage information,
236
+ # when combining coverage information from
237
+ # multiple test suites.
238
+ # :file :: The file to store and/or load the unused associations data.
239
+ # :modify_associations :: Whether to use the unused associations data
240
+ # to skip defining associations or association
241
+ # methods.
242
+ # :unused_associations_data :: The unused associations data to use if the
243
+ # :modify_associations is used (by default, the
244
+ # :modify_associations option will use the data from
245
+ # the file specified by the :file option). This is
246
+ # same data returned by the
247
+ # +update_unused_associations_data+ method.
248
+ def self.configure(mod, opts=OPTS)
249
+ mod.instance_exec do
250
+ @unused_associations_coverage_file = opts[:coverage_file]
251
+ @unused_associations_file = opts[:file]
252
+ @unused_associations_data = if opts[:modify_associations]
253
+ if opts[:unused_associations_data]
254
+ opts[:unused_associations_data]
255
+ elsif File.file?(opts[:file])
256
+ Sequel.parse_json(File.binread(opts[:file]))
257
+ end
258
+ end
259
+ end
260
+ end
261
+
262
+ module ClassMethods
263
+ # Only the data is copied to subclasses, to allow the :modify_associations
264
+ # plugin option to affect them. The :file and :coverage_file are not copied
265
+ # to subclasses, as users are expected ot call methods such as
266
+ # unused_associations only on the class that is loading the plugin.
267
+ Plugins.inherited_instance_variables(self, :@unused_associations_data=>nil)
268
+
269
+ # Synchronize access to the used association reflections.
270
+ def used_association_reflections
271
+ Sequel.synchronize{@used_association_reflections ||= {}}
272
+ end
273
+
274
+ # Record access to association reflections to determine which associations are not used.
275
+ def association_reflection(association)
276
+ uar = used_association_reflections
277
+ Sequel.synchronize{uar[association] ||= true}
278
+ super
279
+ end
280
+
281
+ # If modifying associations, and this association is marked as not used,
282
+ # and the association does not include the specific :is_used option,
283
+ # skip defining the association.
284
+ def associate(type, assoc_name, opts=OPTS)
285
+ if !opts[:is_used] && @unused_associations_data && (data = @unused_associations_data[name]) && data[assoc_name.to_s] == 'unused'
286
+ return
287
+ end
288
+
289
+ super
290
+ end
291
+
292
+ # Setup the used_association_reflections storage before freezing
293
+ def freeze
294
+ used_association_reflections
295
+ super
296
+ end
297
+
298
+ # Parse the coverage result, and return the coverage data for the
299
+ # associations for descendants of this class. If the plugin
300
+ # uses the :coverage_file option, the existing coverage file will be loaded
301
+ # if present, and before the method returns, the coverage file will be updated.
302
+ #
303
+ # Options:
304
+ # :coverage_result :: The coverage result to use. This defaults to +Coverage.result+.
305
+ def update_associations_coverage(opts=OPTS)
306
+ coverage_result = opts[:coverage_result] || Coverage.result
307
+ module_mapping = {}
308
+ file = @unused_associations_coverage_file
309
+
310
+ coverage_data = if file && File.file?(file)
311
+ Sequel.parse_json(File.binread(file))
312
+ else
313
+ {}
314
+ end
315
+
316
+ ([self] + descendants).each do |sc|
317
+ next if sc.associations.empty? || !sc.name
318
+ module_mapping[sc.send(:overridable_methods_module)] = sc
319
+ cov_data = coverage_data[sc.name] ||= {''=>[]}
320
+ cov_data[''].concat(sc.used_association_reflections.keys.map(&:to_s).sort).uniq!
321
+ end
322
+
323
+ coverage_result.each do |file, coverage|
324
+ coverage[:methods].each do |(mod, meth), times|
325
+ next unless sc = module_mapping[mod]
326
+ coverage_data[sc.name][meth.to_s] ||= 0
327
+ coverage_data[sc.name][meth.to_s] += times
328
+ end
329
+ end
330
+
331
+ if file
332
+ File.binwrite(file, Sequel.object_to_json(coverage_data))
333
+ end
334
+
335
+ coverage_data
336
+ end
337
+
338
+ # Parse the coverage data returned by #update_associations_coverage,
339
+ # and return data on unused associations and unused association methods.
340
+ #
341
+ # Options:
342
+ # :coverage_data :: The coverage data to use. If not given, it is taken
343
+ # from the file specified by the :coverage_file plugin option.
344
+ # :keep_coverage :: Do not delete the file specified by the :coverage_file plugin
345
+ # option, even if it exists.
346
+ def update_unused_associations_data(options=OPTS)
347
+ coverage_data = options[:coverage_data] || Sequel.parse_json(File.binread(@unused_associations_coverage_file))
348
+
349
+ unused_associations_data = {}
350
+
351
+ ([self] + descendants).each do |sc|
352
+ next unless cov_data = coverage_data[sc.name]
353
+ reflection_data = cov_data[''] || []
354
+
355
+ sc.association_reflections.each do |assoc, ref|
356
+ # Only report associations for the class they are defined in
357
+ next unless ref[:model] == sc
358
+
359
+ # Do not report associations using methods_module option, because this plugin only
360
+ # looks in the class's overridable_methods_module
361
+ next if ref[:methods_module]
362
+
363
+ info = {}
364
+ if reflection_data.include?(assoc.to_s)
365
+ info[:used] = [:reflection]
366
+ end
367
+
368
+ _update_association_coverage_info(info, cov_data, ref.dataset_method, :dataset_method)
369
+ _update_association_coverage_info(info, cov_data, ref.association_method, :association_method)
370
+
371
+ unless ref[:orig_opts][:read_only]
372
+ if ref.returns_array?
373
+ _update_association_coverage_info(info, cov_data, ref[:add_method], :adder)
374
+ _update_association_coverage_info(info, cov_data, ref[:remove_method], :remover)
375
+ _update_association_coverage_info(info, cov_data, ref[:remove_all_method], :clearer)
376
+ else
377
+ _update_association_coverage_info(info, cov_data, ref[:setter_method], :setter)
378
+ end
379
+ end
380
+
381
+ next if info.keys == [:missing]
382
+
383
+ if !info[:used]
384
+ (unused_associations_data[sc.name] ||= {})[assoc.to_s] = 'unused'
385
+ elsif unused = info[:unused]
386
+ if unused.include?(:setter) || [:adder, :remover, :clearer].all?{|k| unused.include?(k)}
387
+ [:setter, :adder, :remover, :clearer].each do |k|
388
+ unused.delete(k)
389
+ end
390
+ unused << :read_only
391
+ end
392
+ (unused_associations_data[sc.name] ||= {})[assoc.to_s] = unused.map(&:to_s)
393
+ end
394
+ end
395
+ end
396
+
397
+ if @unused_associations_file
398
+ File.binwrite(@unused_associations_file, Sequel.object_to_json(unused_associations_data))
399
+ end
400
+ unless options[:keep_coverage]
401
+ _delete_unused_associations_file(@unused_associations_coverage_file)
402
+ end
403
+
404
+ unused_associations_data
405
+ end
406
+
407
+ # Return an array of unused associations. These are associations where none of the
408
+ # association methods are used, according to the coverage information. Each entry
409
+ # in the array is an array of two strings, with the first string being the class name
410
+ # and the second string being the association name.
411
+ #
412
+ # Options:
413
+ # :unused_associations_data :: The data to use for determining which associations
414
+ # are unused, which is returned from
415
+ # +update_unused_associations_data+. If not given,
416
+ # loads the data from the file specified by the :file
417
+ # plugin option.
418
+ def unused_associations(opts=OPTS)
419
+ unused_associations_data = opts[:unused_associations_data] || Sequel.parse_json(File.binread(@unused_associations_file))
420
+
421
+ unused_associations = []
422
+ unused_associations_data.each do |sc, associations|
423
+ associations.each do |assoc, unused|
424
+ if unused == 'unused'
425
+ unused_associations << [sc, assoc]
426
+ end
427
+ end
428
+ end
429
+ unused_associations
430
+ end
431
+
432
+ # Return an array of unused association options. These are associations some but not all
433
+ # of the association methods are used, according to the coverage information. Each entry
434
+ # in the array is an array of three elements. The first element is the class name string,
435
+ # the second element is the association name string, and the third element is a hash of
436
+ # association options that can be used in the association so it does not define methods
437
+ # that are not used.
438
+ #
439
+ # Options:
440
+ # :unused_associations_data :: The data to use for determining which associations
441
+ # are unused, which is returned from
442
+ # +update_unused_associations_data+. If not given,
443
+ # loads the data from the file specified by the :file
444
+ # plugin option.
445
+ def unused_association_options(opts=OPTS)
446
+ unused_associations_data = opts[:unused_associations_data] || Sequel.parse_json(File.binread(@unused_associations_file))
447
+
448
+ unused_association_methods = []
449
+ unused_associations_data.each do |sc, associations|
450
+ associations.each do |assoc, unused|
451
+ unless unused == 'unused'
452
+ unused_association_methods << [sc, assoc, set_unused_options_for_association({}, unused)]
453
+ end
454
+ end
455
+ end
456
+ unused_association_methods
457
+ end
458
+
459
+ # Delete the unused associations coverage file and unused associations data file,
460
+ # if either exist.
461
+ def delete_unused_associations_files
462
+ _delete_unused_associations_file(@unused_associations_coverage_file)
463
+ _delete_unused_associations_file(@unused_associations_file)
464
+ end
465
+
466
+ private
467
+
468
+ # Delete the given file if it exists.
469
+ def _delete_unused_associations_file(file)
470
+ if file && File.file?(file)
471
+ File.unlink(file)
472
+ end
473
+ end
474
+
475
+ # Update the info hash with information on whether the given method was
476
+ # called, according to the coverage information.
477
+ def _update_association_coverage_info(info, coverage_data, meth, key)
478
+ type = case coverage_data[meth.to_s]
479
+ when 0
480
+ :unused
481
+ when Integer
482
+ :used
483
+ else
484
+ # Missing here means there is no coverage information for the
485
+ # the method, which indicates the expected method was never
486
+ # defined. In that case, it can be ignored.
487
+ :missing
488
+ end
489
+
490
+ (info[type] ||= []) << key
491
+ end
492
+
493
+ # Based on the value of the unused, update the opts hash with association
494
+ # options that will prevent unused association methods from being
495
+ # defined.
496
+ def set_unused_options_for_association(opts, unused)
497
+ opts[:read_only] = true if unused.include?('read_only')
498
+ opts[:no_dataset_method] = true if unused.include?('dataset_method')
499
+ opts[:no_association_method] = true if unused.include?('association_method')
500
+ opts[:adder] = nil if unused.include?('adder')
501
+ opts[:remover] = nil if unused.include?('remover')
502
+ opts[:clearer] = nil if unused.include?('clearer')
503
+ opts
504
+ end
505
+
506
+ # If modifying associations, and this association has unused association
507
+ # methods, automatically set the appropriate options so the unused association
508
+ # methods are not defined, unless the association explicitly uses the :is_used
509
+ # options.
510
+ def def_association(opts)
511
+ if !opts[:is_used] && @unused_associations_data && (data = @unused_associations_data[name]) && (unused = data[opts[:name].to_s])
512
+ set_unused_options_for_association(opts, unused)
513
+ end
514
+
515
+ super
516
+ end
517
+ end
518
+ end
519
+ end
520
+ end
521
+ # :nocov:
@@ -55,7 +55,7 @@ module Sequel
55
55
  def find_or_new(attrs, set_attrs=nil)
56
56
  obj = find(attrs) || new(attrs)
57
57
  obj.set(set_attrs) if set_attrs
58
- yield obj if block_given?
58
+ yield obj if defined?(yield)
59
59
  obj
60
60
  end
61
61
  end
@@ -2,7 +2,7 @@
2
2
 
3
3
  module Sequel
4
4
  module Plugins
5
- # The validates_associated plugin allows you to validate associated
5
+ # The validate_associated plugin allows you to validate associated
6
6
  # objects. It also offers the ability to delay the validation of
7
7
  # associated objects until the current object is validated.
8
8
  # If the associated object is invalid, validation error messages
@@ -54,22 +54,32 @@ module Sequel
54
54
  return if reflection[:validate] == false
55
55
  association = reflection[:name]
56
56
  if (reflection[:type] == :one_to_many || reflection[:type] == :one_to_one) && (key = reflection[:key]).is_a?(Symbol) && !(pk_val = obj.values[key])
57
- # There could be a presence validation on the foreign key in the associated model,
58
- # which will fail if we validate before saving the current object. If there is
59
- # no value for the foreign key, set it to the current primary key value, or a dummy
60
- # value of 0 if we haven't saved the current object.
61
57
  p_key = pk unless pk.is_a?(Array)
62
- obj.values[key] = p_key || 0
63
- key = nil if p_key
58
+ if p_key
59
+ obj.values[key] = p_key
60
+ else
61
+ ignore_key_errors = true
62
+ end
64
63
  end
65
- obj.errors.full_messages.each{|m| errors.add(association, m)} unless obj.valid?
66
- if key && !pk_val
67
- # If we used a dummy value of 0, remove it so it doesn't accidently remain.
68
- obj.values.delete(key)
64
+
65
+ unless obj.valid?
66
+ if ignore_key_errors
67
+ # Ignore errors on the key column in the associated object. This column
68
+ # will be set when saving to a presumably valid value using a column
69
+ # in the current object (which may not be available until after the current
70
+ # object is saved).
71
+ obj.errors.delete(key)
72
+ obj.errors.delete_if{|k,| Array === k && k.include?(key)}
73
+ end
74
+
75
+ obj.errors.full_messages.each do |m|
76
+ errors.add(association, m)
77
+ end
69
78
  end
79
+
80
+ nil
70
81
  end
71
82
  end
72
83
  end
73
84
  end
74
85
  end
75
-