2017-07-09 13:41:28 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2016-08-06 12:26:20 -04:00
|
|
|
require "config"
|
2009-02-03 21:25:37 -05:00
|
|
|
|
2016-08-06 12:26:20 -04:00
|
|
|
require "stringio"
|
2008-01-21 12:20:51 -05:00
|
|
|
|
2016-08-06 12:26:20 -04:00
|
|
|
require "active_record"
|
|
|
|
require "cases/test_case"
|
|
|
|
require "active_support/dependencies"
|
|
|
|
require "active_support/logger"
|
2020-05-09 14:43:05 -04:00
|
|
|
require "active_support/core_ext/kernel/singleton_class"
|
2011-06-04 18:19:17 -04:00
|
|
|
|
2016-08-06 12:26:20 -04:00
|
|
|
require "support/config"
|
|
|
|
require "support/connection"
|
2011-06-04 18:19:17 -04:00
|
|
|
|
|
|
|
# TODO: Move all these random hacks into the ARTest namespace and into the support/ dir
|
2008-01-21 12:20:51 -05:00
|
|
|
|
2013-01-24 05:10:58 -05:00
|
|
|
Thread.abort_on_exception = true
|
|
|
|
|
2008-01-21 12:20:51 -05:00
|
|
|
# Show backtraces for deprecated behavior for quicker cleanup.
|
|
|
|
ActiveSupport::Deprecation.debug = true
|
|
|
|
|
2013-12-16 13:04:07 -05:00
|
|
|
# Disable available locale checks to avoid warnings running the test suite.
|
|
|
|
I18n.enforce_available_locales = false
|
|
|
|
|
2011-06-09 15:15:34 -04:00
|
|
|
# Connect to the database
|
|
|
|
ARTest.connect
|
|
|
|
|
|
|
|
# Quote "type" if it's a reserved word for the current connection.
|
2016-08-06 12:26:20 -04:00
|
|
|
QUOTED_TYPE = ActiveRecord::Base.connection.quote_column_name("type")
|
2011-06-09 15:15:34 -04:00
|
|
|
|
2008-01-21 12:20:51 -05:00
|
|
|
def current_adapter?(*types)
|
|
|
|
types.any? do |type|
|
|
|
|
ActiveRecord::ConnectionAdapters.const_defined?(type) &&
|
|
|
|
ActiveRecord::Base.connection.is_a?(ActiveRecord::ConnectionAdapters.const_get(type))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2011-01-08 13:33:33 -05:00
|
|
|
def in_memory_db?
|
2012-04-27 02:31:37 -04:00
|
|
|
current_adapter?(:SQLite3Adapter) &&
|
2019-09-18 06:26:59 -04:00
|
|
|
ActiveRecord::Base.connection_pool.db_config.database == ":memory:"
|
2011-01-08 13:33:33 -05:00
|
|
|
end
|
|
|
|
|
2014-09-02 01:35:56 -04:00
|
|
|
def mysql_enforcing_gtid_consistency?
|
2016-08-06 12:26:20 -04:00
|
|
|
current_adapter?(:Mysql2Adapter) && "ON" == ActiveRecord::Base.connection.show_variable("enforce_gtid_consistency")
|
2014-09-02 01:35:56 -04:00
|
|
|
end
|
|
|
|
|
2018-10-24 21:11:31 -04:00
|
|
|
def supports_default_expression?
|
|
|
|
if current_adapter?(:PostgreSQLAdapter)
|
|
|
|
true
|
|
|
|
elsif current_adapter?(:Mysql2Adapter)
|
|
|
|
conn = ActiveRecord::Base.connection
|
2019-03-29 11:18:48 -04:00
|
|
|
!conn.mariadb? && conn.database_version >= "8.0.13"
|
2018-10-24 21:11:31 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
%w[
|
|
|
|
supports_savepoints?
|
|
|
|
supports_partial_index?
|
2019-10-29 07:54:03 -04:00
|
|
|
supports_partitioned_indexes?
|
2020-06-02 18:18:27 -04:00
|
|
|
supports_expression_index?
|
2019-03-05 14:16:44 -05:00
|
|
|
supports_insert_returning?
|
|
|
|
supports_insert_on_duplicate_skip?
|
|
|
|
supports_insert_on_duplicate_update?
|
|
|
|
supports_insert_conflict_target?
|
2019-03-12 20:29:00 -04:00
|
|
|
supports_optimizer_hints?
|
2020-04-24 13:56:53 -04:00
|
|
|
supports_datetime_with_precision?
|
2019-03-05 14:16:44 -05:00
|
|
|
].each do |method_name|
|
|
|
|
define_method method_name do
|
|
|
|
ActiveRecord::Base.connection.public_send(method_name)
|
|
|
|
end
|
2011-01-09 10:52:41 -05:00
|
|
|
end
|
|
|
|
|
2016-08-06 12:26:20 -04:00
|
|
|
def with_env_tz(new_tz = "US/Eastern")
|
|
|
|
old_tz, ENV["TZ"] = ENV["TZ"], new_tz
|
2010-09-20 15:15:18 -04:00
|
|
|
yield
|
|
|
|
ensure
|
2016-08-06 12:26:20 -04:00
|
|
|
old_tz ? ENV["TZ"] = old_tz : ENV.delete("TZ")
|
2010-09-20 15:15:18 -04:00
|
|
|
end
|
|
|
|
|
2013-10-24 15:26:23 -04:00
|
|
|
def with_timezone_config(cfg)
|
|
|
|
verify_default_timezone_config
|
|
|
|
|
|
|
|
old_default_zone = ActiveRecord::Base.default_timezone
|
|
|
|
old_awareness = ActiveRecord::Base.time_zone_aware_attributes
|
|
|
|
old_zone = Time.zone
|
|
|
|
|
|
|
|
if cfg.has_key?(:default)
|
|
|
|
ActiveRecord::Base.default_timezone = cfg[:default]
|
|
|
|
end
|
|
|
|
if cfg.has_key?(:aware_attributes)
|
|
|
|
ActiveRecord::Base.time_zone_aware_attributes = cfg[:aware_attributes]
|
|
|
|
end
|
|
|
|
if cfg.has_key?(:zone)
|
|
|
|
Time.zone = cfg[:zone]
|
|
|
|
end
|
2010-09-20 15:15:18 -04:00
|
|
|
yield
|
|
|
|
ensure
|
2013-10-24 15:26:23 -04:00
|
|
|
ActiveRecord::Base.default_timezone = old_default_zone
|
|
|
|
ActiveRecord::Base.time_zone_aware_attributes = old_awareness
|
|
|
|
Time.zone = old_zone
|
|
|
|
end
|
|
|
|
|
|
|
|
# This method makes sure that tests don't leak global state related to time zones.
|
|
|
|
EXPECTED_ZONE = nil
|
|
|
|
EXPECTED_DEFAULT_TIMEZONE = :utc
|
|
|
|
EXPECTED_TIME_ZONE_AWARE_ATTRIBUTES = false
|
|
|
|
def verify_default_timezone_config
|
|
|
|
if Time.zone != EXPECTED_ZONE
|
2013-10-25 09:59:04 -04:00
|
|
|
$stderr.puts <<-MSG
|
2014-10-29 21:18:48 -04:00
|
|
|
\n#{self}
|
2013-10-24 15:26:23 -04:00
|
|
|
Global state `Time.zone` was leaked.
|
|
|
|
Expected: #{EXPECTED_ZONE}
|
|
|
|
Got: #{Time.zone}
|
|
|
|
MSG
|
|
|
|
end
|
|
|
|
if ActiveRecord::Base.default_timezone != EXPECTED_DEFAULT_TIMEZONE
|
2013-10-25 09:59:04 -04:00
|
|
|
$stderr.puts <<-MSG
|
2014-10-29 21:18:48 -04:00
|
|
|
\n#{self}
|
2013-10-24 15:26:23 -04:00
|
|
|
Global state `ActiveRecord::Base.default_timezone` was leaked.
|
|
|
|
Expected: #{EXPECTED_DEFAULT_TIMEZONE}
|
|
|
|
Got: #{ActiveRecord::Base.default_timezone}
|
|
|
|
MSG
|
|
|
|
end
|
|
|
|
if ActiveRecord::Base.time_zone_aware_attributes != EXPECTED_TIME_ZONE_AWARE_ATTRIBUTES
|
2013-10-25 09:59:04 -04:00
|
|
|
$stderr.puts <<-MSG
|
2014-10-29 21:18:48 -04:00
|
|
|
\n#{self}
|
2013-10-24 15:26:23 -04:00
|
|
|
Global state `ActiveRecord::Base.time_zone_aware_attributes` was leaked.
|
|
|
|
Expected: #{EXPECTED_TIME_ZONE_AWARE_ATTRIBUTES}
|
|
|
|
Got: #{ActiveRecord::Base.time_zone_aware_attributes}
|
|
|
|
MSG
|
|
|
|
end
|
2010-09-20 15:15:18 -04:00
|
|
|
end
|
|
|
|
|
2014-07-09 17:10:49 -04:00
|
|
|
def enable_extension!(extension, connection)
|
2013-08-25 05:22:36 -04:00
|
|
|
return false unless connection.supports_extensions?
|
2014-07-09 17:10:49 -04:00
|
|
|
return connection.reconnect! if connection.extension_enabled?(extension)
|
2013-08-25 05:22:36 -04:00
|
|
|
|
2014-07-09 17:10:49 -04:00
|
|
|
connection.enable_extension extension
|
2015-02-12 08:44:47 -05:00
|
|
|
connection.commit_db_transaction if connection.transaction_open?
|
2013-08-25 05:22:36 -04:00
|
|
|
connection.reconnect!
|
|
|
|
end
|
|
|
|
|
2014-07-09 17:10:49 -04:00
|
|
|
def disable_extension!(extension, connection)
|
|
|
|
return false unless connection.supports_extensions?
|
|
|
|
return true unless connection.extension_enabled?(extension)
|
|
|
|
|
|
|
|
connection.disable_extension extension
|
|
|
|
connection.reconnect!
|
|
|
|
end
|
|
|
|
|
2020-08-10 09:51:43 -04:00
|
|
|
def clean_up_connection_handler
|
|
|
|
ActiveRecord::Base.connection_handlers = { ActiveRecord::Base.writing_role => ActiveRecord::Base.default_connection_handler }
|
|
|
|
end
|
|
|
|
|
2011-01-08 13:33:33 -05:00
|
|
|
def load_schema
|
|
|
|
# silence verbose schema loading
|
|
|
|
original_stdout = $stdout
|
|
|
|
$stdout = StringIO.new
|
2009-05-17 14:51:35 -04:00
|
|
|
|
|
|
|
adapter_name = ActiveRecord::Base.connection.adapter_name.downcase
|
|
|
|
adapter_specific_schema_file = SCHEMA_ROOT + "/#{adapter_name}_specific_schema.rb"
|
|
|
|
|
|
|
|
load SCHEMA_ROOT + "/schema.rb"
|
|
|
|
|
2013-11-01 04:49:57 -04:00
|
|
|
if File.exist?(adapter_specific_schema_file)
|
2009-05-17 14:51:35 -04:00
|
|
|
load adapter_specific_schema_file
|
|
|
|
end
|
2017-06-03 02:58:54 -04:00
|
|
|
|
|
|
|
ActiveRecord::FixtureSet.reset_cache
|
2009-05-17 14:51:35 -04:00
|
|
|
ensure
|
|
|
|
$stdout = original_stdout
|
|
|
|
end
|
2010-07-26 08:06:14 -04:00
|
|
|
|
2011-01-08 13:33:33 -05:00
|
|
|
load_schema
|
|
|
|
|
2013-10-04 18:16:49 -04:00
|
|
|
class SQLSubscriber
|
|
|
|
attr_reader :logged
|
2013-10-04 19:51:35 -04:00
|
|
|
attr_reader :payloads
|
2013-10-04 18:16:49 -04:00
|
|
|
|
|
|
|
def initialize
|
|
|
|
@logged = []
|
2013-10-04 19:51:35 -04:00
|
|
|
@payloads = []
|
2013-10-04 18:16:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def start(name, id, payload)
|
2013-10-04 19:51:35 -04:00
|
|
|
@payloads << payload
|
2014-03-24 23:38:10 -04:00
|
|
|
@logged << [payload[:sql].squish, payload[:name], payload[:binds]]
|
2013-10-04 18:16:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def finish(name, id, payload); end
|
|
|
|
end
|
|
|
|
|
2012-11-17 11:01:35 -05:00
|
|
|
module InTimeZone
|
|
|
|
private
|
2016-08-06 13:55:02 -04:00
|
|
|
def in_time_zone(zone)
|
|
|
|
old_zone = Time.zone
|
|
|
|
old_tz = ActiveRecord::Base.time_zone_aware_attributes
|
|
|
|
|
|
|
|
Time.zone = zone ? ActiveSupport::TimeZone[zone] : nil
|
|
|
|
ActiveRecord::Base.time_zone_aware_attributes = !zone.nil?
|
|
|
|
yield
|
|
|
|
ensure
|
|
|
|
Time.zone = old_zone
|
|
|
|
ActiveRecord::Base.time_zone_aware_attributes = old_tz
|
|
|
|
end
|
2012-11-17 10:35:39 -05:00
|
|
|
end
|