# frozen_string_literal: true module ActiveRecord module ConnectionAdapters module MySQL class SchemaDumper < ConnectionAdapters::SchemaDumper # :nodoc: private def prepare_column_options(column) spec = super spec[:unsigned] = "true" if column.unsigned? spec[:auto_increment] = "true" if column.auto_increment? if /\A(?tiny|medium|long)(?:text|blob)/ =~ column.sql_type spec = { size: size.to_sym.inspect }.merge!(spec) end if @connection.supports_virtual_columns? && column.virtual? spec[:as] = extract_expression_for_virtual_column(column) spec[:stored] = "true" if /\b(?:STORED|PERSISTENT)\b/.match?(column.extra) spec = { type: schema_type(column).inspect }.merge!(spec) end spec end def column_spec_for_primary_key(column) spec = super spec.delete(:auto_increment) if column.type == :integer && column.auto_increment? spec end def default_primary_key?(column) super && column.auto_increment? && !column.unsigned? end def explicit_primary_key_default?(column) column.type == :integer && !column.auto_increment? end def schema_type(column) case column.sql_type when /\Atimestamp\b/ :timestamp when /\A(?:enum|set)\b/ column.sql_type else super end end def schema_limit(column) super unless /\A(?:tiny|medium|long)?(?:text|blob)\b/.match?(column.sql_type) end def schema_precision(column) if /\Atime(?:stamp)?\b/.match?(column.sql_type) && column.precision == 0 nil elsif column.type == :datetime column.precision == 0 ? "nil" : super else super end end def schema_collation(column) if column.collation @table_collation_cache ||= {} @table_collation_cache[table_name] ||= @connection.internal_exec_query("SHOW TABLE STATUS LIKE #{@connection.quote(table_name)}", "SCHEMA").first["Collation"] column.collation.inspect if column.collation != @table_collation_cache[table_name] end end def extract_expression_for_virtual_column(column) if @connection.mariadb? && @connection.database_version < "10.2.5" create_table_info = @connection.send(:create_table_info, table_name) column_name = @connection.quote_column_name(column.name) if %r/#{column_name} #{Regexp.quote(column.sql_type)}(?: COLLATE \w+)? AS \((?.+?)\) #{column.extra}/ =~ create_table_info $~[:expression].inspect end else scope = @connection.send(:quoted_scope, table_name) column_name = @connection.quote(column.name) sql = "SELECT generation_expression FROM information_schema.columns" \ " WHERE table_schema = #{scope[:schema]}" \ " AND table_name = #{scope[:name]}" \ " AND column_name = #{column_name}" # Calling .inspect leads into issues with the query result # which already returns escaped quotes. # We remove the escape sequence from the result in order to deal with double escaping issues. @connection.query_value(sql, "SCHEMA").gsub("\\'", "'").inspect end end end end end end