how to change rails migration t.timestamps to use `timestamp(0) without timezone` in postgres

后端 未结 2 852
一整个雨季
一整个雨季 2021-01-25 13:54

I\'m trying to figure out how to change the native data type that t.timestamps uses in a rails migration. Default type that ends up in postgres is timestamp w

2条回答
  •  面向向阳花
    2021-01-25 14:14

    Here is a solution. It alters the default timestamp precision in Rails, including for migration, for the two timestamps to one second accuracy in PostgreSQL. It's neither easy nor simple, but works for Rails 5.2 with PostgreSQL.

    I think the initializer should be placed in config/initializers/ (not in environments).
    Write the following file.

    # ./config/initializers/arbitrary.rb
    
    require "active_record/connection_adapters/abstract/schema_definitions.rb"
    require "active_record/connection_adapters/abstract_adapter"
    require "active_record/connection_adapters/abstract/schema_statements"
    require "active_record/connection_adapters/postgresql/schema_statements"
    require "active_record/connection_adapters/postgresql_adapter"
    
    module ActiveRecord
      module ConnectionAdapters
        # Overwrites a method in /abstract/schema_definitions.rb
        class TableDefinition
          def timestamps(**options)
            options[:null] = false if options[:null].nil?
    
            column(:created_at, :datetime0, options)
            column(:updated_at, :datetime0, options)
          end
        end
    
        # Overwrites a method in /abstract/schema_statements.rb
        module SchemaStatements
          def add_timestamps(table_name, options = {})
            options[:null] = false if options[:null].nil?
        
            add_column table_name, :created_at, :datetime0, options
            add_column table_name, :updated_at, :datetime0, options
          end
        end
    
        # Overwrites a method in /postgresql/schema_statements.rb
        module PostgreSQL
          module SchemaStatements
            def add_timestamps_for_alter(table_name, options = {})
              [add_column_for_alter(table_name, :created_at, :datetime0, options), add_column_for_alter(table_name, :updated_at, :datetime0, options)]
            end
          end
        end
    
        # Modifies a constant and methods in /postgresql_adapter.rb
        class PostgreSQLAdapter
          alias_method :initialize_type_map_orig, :initialize_type_map if ! self.method_defined?(:initialize_type_map_orig)
          NATIVE_DATABASE_TYPES[:datetime0] = { name: "timestamp(0)" }
    
          private    
            def initialize_type_map(m = type_map)
              register_class_with_precision_t0 m, "timestamp0", OID::DateTime
              initialize_type_map_orig(m)
            end
    
            def register_class_with_precision_t0(mapping, key, klass)
              mapping.register_type(key) do |*args|
                klass.new(precision: 0)
              end
            end
        end
      end
    end
    

    Here is an example migration file.

    # db/migrate/20181023182238_create_articles.rb
    class CreateArticles < ActiveRecord::Migration[5.2]
      def change
        create_table :articles do |t|
          t.string :title
          t.timestamps
        end
      end
    end
    

    Migration (bin/rails db:migrate) creates a table articles with the two timestamps columns of timestamp(0) (without timezone) in the PostgreSQL database.

    The SQL executed is this:

    CREATE TABLE "articles" (
      "id" bigserial primary key, 
      "title" character varying, 
      "created_at" timestamp(0) NOT NULL,
      "updated_at" timestamp(0) NOT NULL);
    

    Note

    I have confirmed both migration to create a table and data updating works in Rails console. It is meant to work in a migration to update a table, too, but I haven't tested it.

    With a bit more tweaking it would work in other databases as well.

    Basically the code above defines a new Rails type timestamp0, to which timestamps() (which is created_at and updated_at) is assigned. If you want any other columns of timestamp to be the same (i.e., no sub-second precision in the DB), specify timestamp0 in your migration and it should work (although I haven't tested it).

提交回复
热议问题