active_record_adapter.rb 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. module Blazer
  2. module Adapters
  3. class ActiveRecordAdapter
  4. attr_reader :data_source, :connection_model
  5. def initialize(data_source)
  6. @data_source = data_source
  7. @connection_model =
  8. Class.new(Blazer::Connection) do
  9. def self.name
  10. "Blazer::Connection::#{object_id}"
  11. end
  12. establish_connection(data_source.settings["url"]) if data_source.settings["url"]
  13. end
  14. end
  15. def schemas
  16. default_schema = (postgresql? || redshift?) ? "public" : connection_model.connection_config[:database]
  17. settings["schemas"] || [connection_model.connection_config[:schema] || default_schema]
  18. end
  19. def tables
  20. result = data_source.run_statement(connection_model.send(:sanitize_sql_array, ["SELECT table_name FROM information_schema.tables WHERE table_schema IN (?) ORDER BY table_name", schemas]))
  21. result.rows.map(&:first)
  22. end
  23. def run_statement(statement, comment)
  24. columns = []
  25. rows = []
  26. error = nil
  27. start_time = Time.now
  28. begin
  29. in_transaction do
  30. set_timeout(data_source.timeout) if data_source.timeout
  31. result = connection_model.connection.select_all("#{statement} /*#{comment}*/")
  32. columns = result.columns
  33. cast_method = Rails::VERSION::MAJOR < 5 ? :type_cast : :cast_value
  34. result.rows.each do |untyped_row|
  35. rows << (result.column_types.empty? ? untyped_row : columns.each_with_index.map { |c, i| untyped_row[i] ? result.column_types[c].send(cast_method, untyped_row[i]) : nil })
  36. end
  37. end
  38. rescue ActiveRecord::StatementInvalid => e
  39. error = e.message.sub(/.+ERROR: /, "")
  40. error = Blazer::TIMEOUT_MESSAGE if Blazer::TIMEOUT_ERRORS.any? { |e| error.include?(e) }
  41. end
  42. [columns, rows, error]
  43. end
  44. def reconnect
  45. connection_model.establish_connection(settings["url"])
  46. end
  47. def cost(statement)
  48. result = explain(statement)
  49. match = /cost=\d+\.\d+..(\d+\.\d+) /.match(result)
  50. match[1] if match
  51. end
  52. def explain(statement)
  53. if postgresql? || redshift?
  54. connection_model.connection.select_all("EXPLAIN #{statement}").rows.first.first
  55. end
  56. rescue
  57. nil
  58. end
  59. private
  60. def postgresql?
  61. ["PostgreSQL", "PostGIS"].include?(adapter_name)
  62. end
  63. def redshift?
  64. ["Redshift"].include?(adapter_name)
  65. end
  66. def mysql?
  67. ["MySQL", "Mysql2", "Mysql2Spatial"].include?(adapter_name)
  68. end
  69. def adapter_name
  70. connection_model.connection.adapter_name
  71. end
  72. def set_timeout(timeout)
  73. if postgresql? || redshift?
  74. connection_model.connection.execute("SET statement_timeout = #{timeout.to_i * 1000}")
  75. elsif mysql?
  76. connection_model.connection.execute("SET max_execution_time = #{timeout.to_i * 1000}")
  77. else
  78. raise Blazer::TimeoutNotSupported, "Timeout not supported for #{adapter_name} adapter"
  79. end
  80. end
  81. def use_transaction?
  82. settings.key?("use_transaction") ? settings["use_transaction"] : true
  83. end
  84. def in_transaction
  85. if use_transaction?
  86. connection_model.transaction do
  87. yield
  88. raise ActiveRecord::Rollback
  89. end
  90. else
  91. yield
  92. end
  93. end
  94. def settings
  95. @data_source.settings
  96. end
  97. end
  98. end
  99. end