Class: AnnotateRb::ModelAnnotator::ModelWrapper
- Inherits:
-
Object
- Object
- AnnotateRb::ModelAnnotator::ModelWrapper
- Defined in:
- lib/annotate_rb/model_annotator/model_wrapper.rb
Constant Summary collapse
- DEFAULT_TIMESTAMP_COLUMNS =
Should be the wrapper for an ActiveRecord model that serves as the source of truth of the model of the model that we’re annotating
%w[created_at updated_at]
Instance Method Summary collapse
- #_retrieve_indexes_from_table ⇒ Object
- #built_attributes ⇒ Object
- #classified_sort(cols, grouped_polymorphic) ⇒ Object
- #column_defaults ⇒ Object
-
#columns ⇒ Object
Gets the columns of the ActiveRecord model, processes them, and then returns them.
- #connection ⇒ Object
- #has_table_comments? ⇒ Boolean
-
#ignored_translation_table_columns ⇒ Object
These are the columns that the globalize gem needs to work but are not necessary for the models to be displayed as annotations.
-
#initialize(klass, options) ⇒ ModelWrapper
constructor
A new instance of ModelWrapper.
-
#is_column_primary_key?(column_name) ⇒ Boolean
TODO: Simplify this conditional.
-
#max_schema_info_width ⇒ Object
Calculates the max width of the schema for the model by looking at the columns, schema comments, with respect to the options.
- #migration_version ⇒ Object
- #model_name ⇒ Object
- #position_of_column_comment ⇒ Object
- #primary_key ⇒ Object
-
#raw_columns ⇒ Object
Returns the unmodified model columns.
- #retrieve_indexes_from_table ⇒ Object
- #table_comments ⇒ Object
- #table_exists? ⇒ Boolean
- #table_name ⇒ Object
-
#translated_columns ⇒ Object
Add columns managed by the globalize gem if this gem is being used.
- #with_comments? ⇒ Boolean
Constructor Details
#initialize(klass, options) ⇒ ModelWrapper
Returns a new instance of ModelWrapper.
11 12 13 14 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 11 def initialize(klass, ) @klass = klass @options = end |
Instance Method Details
#_retrieve_indexes_from_table ⇒ Object
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 145 def _retrieve_indexes_from_table table_name = @klass.table_name return [] unless table_name indexes = @klass.connection.indexes(table_name) return indexes if indexes.any? || !@klass.table_name_prefix # Try to search the table without prefix table_name_without_prefix = table_name.to_s.sub(@klass.table_name_prefix.to_s, "") begin @klass.connection.indexes(table_name_without_prefix) rescue ActiveRecord::StatementInvalid => _e # Mysql2 adapter behaves differently than Sqlite3 and Postgres adapter. # If `table_name_without_prefix` does not exist, Mysql2 will raise, # the other adapters will return an empty array. # # See: https://github.com/rails/rails/issues/51205 [] end end |
#built_attributes ⇒ Object
129 130 131 132 133 134 135 136 137 138 139 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 129 def built_attributes @built_attributes ||= begin table_indices = retrieve_indexes_from_table columns.map do |column| is_primary_key = is_column_primary_key?(column.name) column_indices = table_indices.select { |ind| ind.columns.include?(column.name) } built = ColumnAnnotation::AttributesBuilder.new(column, @options, is_primary_key, column_indices, column_defaults).build [column.name, built] end.to_h end end |
#classified_sort(cols, grouped_polymorphic) ⇒ Object
180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 180 def classified_sort(cols, grouped_polymorphic) rest_cols = [] = [] associations = [] id = nil # specs don't load defaults, so ensure we have defaults here = @options[:timestamp_columns] || DEFAULT_TIMESTAMP_COLUMNS col_names = cols.map(&:name) cols.each do |c| if c.name.eql?("id") id = c elsif .include?(c.name) << c elsif c.name[-3, 3].eql?("_id") associations << c elsif c.name[-5, 5].eql?("_type") && col_names.include?(c.name.sub(/_type$/, "_id")) && grouped_polymorphic # This is a polymorphic association's type column associations << c else rest_cols << c end end = .each_with_index.to_h .sort_by! { |col| [col.name] } [rest_cols, associations].each { |a| a.sort_by!(&:name) } ([id] << rest_cols << << associations).flatten.compact end |
#column_defaults ⇒ Object
63 64 65 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 63 def column_defaults @klass.column_defaults end |
#columns ⇒ Object
Gets the columns of the ActiveRecord model, processes them, and then returns them.
17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 17 def columns @columns ||= begin cols = raw_columns cols += translated_columns ignore_columns = @options[:ignore_columns] if ignore_columns cols = cols.reject do |col| col.name.match?(/#{ignore_columns}/) end end cols = cols.sort_by(&:name) if @options[:sort] cols = classified_sort(cols, @options[:grouped_polymorphic]) if @options[:classified_sort] cols end end |
#connection ⇒ Object
37 38 39 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 37 def connection @klass.connection end |
#has_table_comments? ⇒ Boolean
58 59 60 61 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 58 def has_table_comments? @klass.connection.respond_to?(:table_comment) && @klass.connection.table_comment(@klass.table_name).present? end |
#ignored_translation_table_columns ⇒ Object
These are the columns that the globalize gem needs to work but are not necessary for the models to be displayed as annotations.
215 216 217 218 219 220 221 222 223 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 215 def ignored_translation_table_columns [ :id, :created_at, :updated_at, :locale, @klass.name.foreign_key.to_sym ] end |
#is_column_primary_key?(column_name) ⇒ Boolean
TODO: Simplify this conditional
113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 113 def is_column_primary_key?(column_name) if primary_key if primary_key.is_a?(Array) # If the model has multiple primary keys, check if this column is one of them if primary_key.collect(&:to_sym).include?(column_name.to_sym) return true end elsif column_name.to_sym == primary_key.to_sym # If model has 1 primary key, check if this column is it return true end end false end |
#max_schema_info_width ⇒ Object
Calculates the max width of the schema for the model by looking at the columns, schema comments, with respect to the options.
89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 89 def max_schema_info_width @max_schema_info_width ||= begin cols = columns position_of_column_comment = @options.with_default_fallback(:position_of_column_comment) if with_comments? && position_of_column_comment == :with_name column_widths = cols.map do |column| column.name.size + (column.comment ? Helper.width(column.comment) : 0) end max_size = column_widths.max || 0 max_size += 2 else max_size = cols.map(&:name).map(&:size).max end max_size += @options[:format_rdoc] ? 5 : 1 max_size end end |
#migration_version ⇒ Object
225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 225 def migration_version return 0 unless @options[:include_version] # Multi-database support: Cache migration versions per database connection to handle # different schema versions across primary/secondary databases correctly. # Example: primary → "current_version_primary", secondary → "current_version_secondary" connection_pool_name = connection.pool.db_config.name cache_key = "current_version_#{connection_pool_name}".to_sym if @options.get_state(cache_key).nil? migration_version = begin connection.migration_context.current_version rescue 0 end @options.set_state(cache_key, migration_version) end @options.get_state(cache_key) end |
#model_name ⇒ Object
83 84 85 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 83 def model_name @klass.name.underscore end |
#position_of_column_comment ⇒ Object
176 177 178 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 176 def position_of_column_comment @position_of_column_comment ||= @options[:position_of_column_comment] end |
#primary_key ⇒ Object
46 47 48 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 46 def primary_key @klass.primary_key end |
#raw_columns ⇒ Object
Returns the unmodified model columns
42 43 44 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 42 def raw_columns @raw_columns ||= @klass.columns end |
#retrieve_indexes_from_table ⇒ Object
141 142 143 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 141 def retrieve_indexes_from_table @indexes_from_table ||= _retrieve_indexes_from_table end |
#table_comments ⇒ Object
54 55 56 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 54 def table_comments @klass.connection.table_comment(@klass.table_name) end |
#table_exists? ⇒ Boolean
50 51 52 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 50 def table_exists? @klass.table_exists? end |
#table_name ⇒ Object
79 80 81 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 79 def table_name @klass.table_name end |
#translated_columns ⇒ Object
Add columns managed by the globalize gem if this gem is being used. TODO: Audit if this is still needed, it seems like Globalize gem is no longer maintained
69 70 71 72 73 74 75 76 77 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 69 def translated_columns return [] unless @klass.respond_to?(:translation_class) ignored_cols = ignored_translation_table_columns @klass.translation_class.columns.reject do |col| ignored_cols.include? col.name.to_sym end end |
#with_comments? ⇒ Boolean
166 167 168 169 170 171 172 173 174 |
# File 'lib/annotate_rb/model_annotator/model_wrapper.rb', line 166 def with_comments? return @with_comments if instance_variable_defined?(:@with_comments) @with_comments = @options[:with_comment] && @options[:with_column_comments] && raw_columns.first.respond_to?(:comment) && raw_columns.map(&:comment).any? { |comment| !comment.nil? } end |