# oracle_enhanced_adapter.rb -- ActiveRecord adapter for Oracle 8i, 9i, 10g, 11g # # Authors or original oracle_adapter: Graham Jenkins, Michael Schoen # # Current maintainer: Raimonds Simanovskis (http://blog.rayapps.com) # ######################################################################### # # See History.txt for changes added to original oracle_adapter.rb # ######################################################################### # # From original oracle_adapter.rb: # # Implementation notes: # 1. Redefines (safely) a method in ActiveRecord to make it possible to # implement an autonumbering solution for Oracle. # 2. The OCI8 driver is patched to properly handle values for LONG and # TIMESTAMP columns. The driver-author has indicated that a future # release of the driver will obviate this patch. # 3. LOB support is implemented through an after_save callback. # 4. Oracle does not offer native LIMIT and OFFSET options; this # functionality is mimiced through the use of nested selects. # See http://asktom.oracle.com/pls/ask/f?p=4950:8:::::F4950_P8_DISPLAYID:127412348064 # # Do what you want with this code, at your own peril, but if any # significant portion of my code remains then please acknowledge my # contribution. # portions Copyright 2005 Graham Jenkins require 'active_record/connection_adapters/abstract_adapter' require 'delegate' begin require 'active_record/connection_adapters/oracle_enhanced_tasks' rescue LoadError end if defined?(RAILS_ROOT) begin require_library_or_gem 'oci8' unless self.class.const_defined? :OCI8 # RSI: added mapping for TIMESTAMP / WITH TIME ZONE / LOCAL TIME ZONE types # currently Ruby-OCI8 does not support fractional seconds for timestamps OCI8::BindType::Mapping[OCI8::SQLT_TIMESTAMP] = OCI8::BindType::OraDate OCI8::BindType::Mapping[OCI8::SQLT_TIMESTAMP_TZ] = OCI8::BindType::OraDate OCI8::BindType::Mapping[OCI8::SQLT_TIMESTAMP_LTZ] = OCI8::BindType::OraDate module ActiveRecord class Base def self.oracle_enhanced_connection(config) #:nodoc: # Use OCI8AutoRecover instead of normal OCI8 driver. if config[:emulate_oracle_adapter] == true # allows the enhanced adapter to look like the OracleAdapter. Useful to pick up # conditionals in the rails activerecord test suite require 'active_record/connection_adapters/emulation/oracle_adapter' ConnectionAdapters::OracleAdapter.new OCI8EnhancedAutoRecover.new(config), logger else ConnectionAdapters::OracleEnhancedAdapter.new OCI8EnhancedAutoRecover.new(config), logger end end # RSI: specify table columns which should be ifnored def self.ignore_table_columns(*args) connection.ignore_table_columns(table_name,*args) end # RSI: specify which table columns should be treated as date (without time) def self.set_date_columns(*args) connection.set_type_for_columns(table_name,:date,*args) end # RSI: specify which table columns should be treated as datetime def self.set_datetime_columns(*args) connection.set_type_for_columns(table_name,:datetime,*args) end # RSI: specify which table columns should be treated as booleans def self.set_boolean_columns(*args) connection.set_type_for_columns(table_name,:boolean,*args) end # After setting large objects to empty, select the OCI8::LOB # and write back the data. after_save :enhanced_write_lobs def enhanced_write_lobs #:nodoc: if connection.is_a?(ConnectionAdapters::OracleEnhancedAdapter) && !(self.class.custom_create_method || self.class.custom_update_method) connection.write_lobs(self.class.table_name, self.class, attributes) end end private :enhanced_write_lobs class << self # RSI: patch ORDER BY to work with LOBs def add_order_with_lobs!(sql, order, scope = :auto) if connection.is_a?(ConnectionAdapters::OracleEnhancedAdapter) order = connection.lob_order_by_expression(self, order) if order orig_scope = scope scope = scope(:find) if :auto == scope if scope new_scope_order = connection.lob_order_by_expression(self, scope[:order]) if new_scope_order != scope[:order] scope = scope.merge(:order => new_scope_order) else scope = orig_scope end end end add_order_without_lobs!(sql, order, scope = :auto) end private :add_order_with_lobs! alias_method :add_order_without_lobs!, :add_order! alias_method :add_order!, :add_order_with_lobs! end # RSI: get table comment from schema definition def self.table_comment self.connection.table_comment(self.table_name) end end module ConnectionAdapters #:nodoc: class OracleEnhancedColumn < Column #:nodoc: attr_reader :table_name, :forced_column_type def initialize(name, default, sql_type = nil, null = true, table_name = nil, forced_column_type = nil) @table_name = table_name @forced_column_type = forced_column_type super(name, default, sql_type, null) end def type_cast(value) return guess_date_or_time(value) if type == :datetime && OracleEnhancedAdapter.emulate_dates super end # convert something to a boolean # RSI: added y as boolean value def self.value_to_boolean(value) if value == true || value == false value else %w(true t 1 y +).include?(value.to_s.downcase) end end # RSI: convert Time value to Date for :date columns def self.string_to_date(string) return string.to_date if string.is_a?(Time) super end # RSI: convert Date value to Time for :datetime columns def self.string_to_time(string) return string.to_time if string.is_a?(Date) && !OracleEnhancedAdapter.emulate_dates super end # RSI: get column comment from schema definition # will work only if using default ActiveRecord connection def comment ActiveRecord::Base.connection.column_comment(@table_name, name) end private def simplified_type(field_type) return :boolean if OracleEnhancedAdapter.emulate_booleans && field_type == 'NUMBER(1)' return :boolean if OracleEnhancedAdapter.emulate_booleans_from_strings && (forced_column_type == :boolean || OracleEnhancedAdapter.is_boolean_column?(name, field_type, table_name)) case field_type when /date/i forced_column_type || (:date if OracleEnhancedAdapter.emulate_dates_by_column_name && OracleEnhancedAdapter.is_date_column?(name, table_name)) || :datetime when /timestamp/i then :timestamp when /time/i then :datetime when /decimal|numeric|number/i return :integer if extract_scale(field_type) == 0 # RSI: if column name is ID or ends with _ID return :integer if OracleEnhancedAdapter.emulate_integers_by_column_name && OracleEnhancedAdapter.is_integer_column?(name, table_name) :decimal else super end end def guess_date_or_time(value) value.respond_to?(:hour) && (value.hour == 0 and value.min == 0 and value.sec == 0) ? Date.new(value.year, value.month, value.day) : value end class << self protected def fallback_string_to_date(string) if OracleEnhancedAdapter.string_to_date_format || OracleEnhancedAdapter.string_to_time_format return (string_to_date_or_time_using_format(string).to_date rescue super) end super end def fallback_string_to_time(string) if OracleEnhancedAdapter.string_to_time_format || OracleEnhancedAdapter.string_to_date_format return (string_to_date_or_time_using_format(string).to_time rescue super) end super end def string_to_date_or_time_using_format(string) if OracleEnhancedAdapter.string_to_time_format && dt=Date._strptime(string, OracleEnhancedAdapter.string_to_time_format) return Time.mktime(*dt.values_at(:year, :mon, :mday, :hour, :min, :sec, :zone, :wday)) end DateTime.strptime(string, OracleEnhancedAdapter.string_to_date_format) end end end # This is an Oracle/OCI adapter for the ActiveRecord persistence # framework. It relies upon the OCI8 driver, which works with Oracle 8i # and above. Most recent development has been on Debian Linux against # a 10g database, ActiveRecord 1.12.1 and OCI8 0.1.13. # See: http://rubyforge.org/projects/ruby-oci8/ # # Usage notes: # * Key generation assumes a "${table_name}_seq" sequence is available # for all tables; the sequence name can be changed using # ActiveRecord::Base.set_sequence_name. When using Migrations, these # sequences are created automatically. # * Oracle uses DATE or TIMESTAMP datatypes for both dates and times. # Consequently some hacks are employed to map data back to Date or Time # in Ruby. If the column_name ends in _time it's created as a Ruby Time. # Else if the hours/minutes/seconds are 0, I make it a Ruby Date. Else # it's a Ruby Time. This is a bit nasty - but if you use Duck Typing # you'll probably not care very much. In 9i and up it's tempting to # map DATE to Date and TIMESTAMP to Time, but too many databases use # DATE for both. Timezones and sub-second precision on timestamps are # not supported. # * Default values that are functions (such as "SYSDATE") are not # supported. This is a restriction of the way ActiveRecord supports # default values. # * Support for Oracle8 is limited by Rails' use of ANSI join syntax, which # is supported in Oracle9i and later. You will need to use #finder_sql for # has_and_belongs_to_many associations to run against Oracle8. # # Required parameters: # # * :username # * :password # * :database class OracleEnhancedAdapter < AbstractAdapter @@emulate_booleans = true cattr_accessor :emulate_booleans @@emulate_dates = false cattr_accessor :emulate_dates # RSI: set to true if columns with DATE in their name should be emulated as date @@emulate_dates_by_column_name = false cattr_accessor :emulate_dates_by_column_name def self.is_date_column?(name, table_name = nil) name =~ /(^|_)date(_|$)/i end # RSI: instance method uses at first check if column type defined at class level def is_date_column?(name, table_name = nil) case get_type_for_column(table_name, name) when nil self.class.is_date_column?(name, table_name) when :date true else false end end # RSI: set to true if NUMBER columns with ID at the end of their name should be emulated as integers @@emulate_integers_by_column_name = false cattr_accessor :emulate_integers_by_column_name def self.is_integer_column?(name, table_name = nil) name =~ /(^|_)id$/i end # RSI: set to true if CHAR(1), VARCHAR2(1) columns or VARCHAR2 columns with FLAG or YN at the end of their name # should be emulated as booleans @@emulate_booleans_from_strings = false cattr_accessor :emulate_booleans_from_strings def self.is_boolean_column?(name, field_type, table_name = nil) return true if ["CHAR(1)","VARCHAR2(1)"].include?(field_type) field_type =~ /^VARCHAR2/ && (name =~ /_flag$/i || name =~ /_yn$/i) end def self.boolean_to_string(bool) bool ? "Y" : "N" end # RSI: use to set NLS specific date formats which will be used when assigning string to :date and :datetime columns @@string_to_date_format = @@string_to_time_format = nil cattr_accessor :string_to_date_format, :string_to_time_format def adapter_name #:nodoc: 'OracleEnhanced' end def supports_migrations? #:nodoc: true end def native_database_types #:nodoc: { :primary_key => "NUMBER(38) NOT NULL PRIMARY KEY", :string => { :name => "VARCHAR2", :limit => 255 }, :text => { :name => "CLOB" }, :integer => { :name => "NUMBER", :limit => 38 }, :float => { :name => "NUMBER" }, :decimal => { :name => "DECIMAL" }, :datetime => { :name => "DATE" }, # RSI: changed to native TIMESTAMP type # :timestamp => { :name => "DATE" }, :timestamp => { :name => "TIMESTAMP" }, :time => { :name => "DATE" }, :date => { :name => "DATE" }, :binary => { :name => "BLOB" }, # RSI: if emulate_booleans_from_strings then store booleans in VARCHAR2 :boolean => emulate_booleans_from_strings ? { :name => "VARCHAR2", :limit => 1 } : { :name => "NUMBER", :limit => 1 } } end def table_alias_length 30 end # Returns an array of arrays containing the field values. # Order is the same as that returned by #columns. def select_rows(sql, name = nil) result = select(sql, name) result.map{ |v| v.values} end # QUOTING ================================================== # # see: abstract/quoting.rb # camelCase column names need to be quoted; not that anyone using Oracle # would really do this, but handling this case means we pass the test... def quote_column_name(name) #:nodoc: name.to_s =~ /[A-Z]/ ? "\"#{name}\"" : quote_oracle_reserved_words(name) end # abstract_adapter calls quote_column_name from quote_table_name, so prevent that # but still quote names that have non alphanumeric values def quote_table_name(name) if name.to_s =~ /^[A-Z_0-9\.]+$/i name else "\"#{name}\"" end end def quote_string(s) #:nodoc: s.gsub(/'/, "''") end def quote(value, column = nil) #:nodoc: if value && column case column.type when :text, :binary %Q{empty_#{ column.sql_type.downcase rescue 'blob' }()} # RSI: TIMESTAMP support when :timestamp quote_timestamp_with_to_timestamp(value) # RSI: NLS_DATE_FORMAT independent DATE support when :date, :time, :datetime quote_date_with_to_date(value) else super end elsif value.acts_like?(:date) quote_date_with_to_date(value) elsif value.acts_like?(:time) value.to_i == value.to_f ? quote_date_with_to_date(value) : quote_timestamp_with_to_timestamp(value) else super end end def quoted_true return "'#{self.class.boolean_to_string(true)}'" if emulate_booleans_from_strings "1" end def quoted_false return "'#{self.class.boolean_to_string(false)}'" if emulate_booleans_from_strings "0" end # RSI: should support that composite_primary_keys gem will pass date as string def quote_date_with_to_date(value) value = value.to_s(:db) if value.acts_like?(:date) || value.acts_like?(:time) "TO_DATE('#{value}','YYYY-MM-DD HH24:MI:SS')" end def quote_timestamp_with_to_timestamp(value) # add up to 9 digits of fractional seconds to inserted time value = "#{value.to_s(:db)}.#{("%.6f"%value.to_f).split('.')[1]}" if value.acts_like?(:time) "TO_TIMESTAMP('#{value}','YYYY-MM-DD HH24:MI:SS.FF6')" end # CONNECTION MANAGEMENT ==================================== # # Returns true if the connection is active. def active? # Pings the connection to check if it's still good. Note that an # #active? method is also available, but that simply returns the # last known state, which isn't good enough if the connection has # gone stale since the last use. @connection.ping rescue OCIException false end # Reconnects to the database. def reconnect! @connection.reset! rescue OCIException => e @logger.warn "#{adapter_name} automatic reconnection failed: #{e.message}" end # Disconnects from the database. def disconnect! @connection.logoff rescue nil @connection.active = false end # DATABASE STATEMENTS ====================================== # # see: abstract/database_statements.rb def execute(sql, name = nil) #:nodoc: log(sql, name) { @connection.exec sql } end # Returns the next sequence value from a sequence generator. Not generally # called directly; used by ActiveRecord to get the next primary key value # when inserting a new database record (see #prefetch_primary_key?). def next_sequence_value(sequence_name) id = 0 @connection.exec("select #{sequence_name}.nextval id from dual") { |r| id = r[0].to_i } id end def begin_db_transaction #:nodoc: @connection.autocommit = false end def commit_db_transaction #:nodoc: @connection.commit ensure @connection.autocommit = true end def rollback_db_transaction #:nodoc: @connection.rollback ensure @connection.autocommit = true end def add_limit_offset!(sql, options) #:nodoc: # RSI: added to_i for limit and offset to protect from SQL injection offset = (options[:offset] || 0).to_i if limit = options[:limit] limit = limit.to_i sql.replace "select * from (select raw_sql_.*, rownum raw_rnum_ from (#{sql}) raw_sql_ where rownum <= #{offset+limit}) where raw_rnum_ > #{offset}" elsif offset > 0 sql.replace "select * from (select raw_sql_.*, rownum raw_rnum_ from (#{sql}) raw_sql_) where raw_rnum_ > #{offset}" end end # Returns true for Oracle adapter (since Oracle requires primary key # values to be pre-fetched before insert). See also #next_sequence_value. def prefetch_primary_key?(table_name = nil) true end def default_sequence_name(table, column) #:nodoc: quote_table_name("#{table}_seq") end # Inserts the given fixture into the table. Overridden to properly handle lobs. def insert_fixture(fixture, table_name) super klass = fixture.class_name.constantize rescue nil if klass.respond_to?(:ancestors) && klass.ancestors.include?(ActiveRecord::Base) write_lobs(table_name, klass, fixture) end end # Writes LOB values from attributes, as indicated by the LOB columns of klass. def write_lobs(table_name, klass, attributes) id = quote(attributes[klass.primary_key]) klass.columns.select { |col| col.sql_type =~ /LOB$/i }.each do |col| value = attributes[col.name] # RSI: changed sequence of next two lines - should check if value is nil before converting to yaml next if value.nil? || (value == '') value = value.to_yaml if col.text? && klass.serialized_attributes[col.name] uncached do lob = select_one("SELECT #{col.name} FROM #{table_name} WHERE #{klass.primary_key} = #{id} FOR UPDATE", 'Writable Large Object')[col.name] lob.write value end end end # RSI: change LOB column for ORDER BY clause # just first 100 characters are taken for ordering def lob_order_by_expression(klass, order) return order if order.nil? changed = false new_order = order.to_s.strip.split(/, */).map do |order_by_col| column_name, asc_desc = order_by_col.split(/ +/) if column = klass.columns.detect { |col| col.name == column_name && col.sql_type =~ /LOB$/i} changed = true "DBMS_LOB.SUBSTR(#{column_name},100,1) #{asc_desc}" else order_by_col end end.join(', ') changed ? new_order : order end # SCHEMA STATEMENTS ======================================== # # see: abstract/schema_statements.rb def current_database #:nodoc: select_one("select sys_context('userenv','db_name') db from dual")["db"] end # RSI: changed select from user_tables to all_tables - much faster in large data dictionaries def tables(name = nil) #:nodoc: select_all("select lower(table_name) from all_tables where owner = sys_context('userenv','session_user')").inject([]) do | tabs, t | tabs << t.to_a.first.last end end def indexes(table_name, name = nil) #:nodoc: result = select_all(<<-SQL, name) SELECT lower(i.index_name) as index_name, i.uniqueness, lower(c.column_name) as column_name FROM all_indexes i, user_ind_columns c WHERE i.table_name = '#{table_name.to_s.upcase}' AND c.index_name = i.index_name AND i.index_name NOT IN (SELECT uc.index_name FROM user_constraints uc WHERE uc.constraint_type = 'P') AND i.owner = sys_context('userenv','session_user') ORDER BY i.index_name, c.column_position SQL current_index = nil indexes = [] result.each do |row| if current_index != row['index_name'] indexes << IndexDefinition.new(table_name, row['index_name'], row['uniqueness'] == "UNIQUE", []) current_index = row['index_name'] end indexes.last.columns << row['column_name'] end indexes end # RSI: set ignored columns for table def ignore_table_columns(table_name, *args) @ignore_table_columns ||= {} @ignore_table_columns[table_name] ||= [] @ignore_table_columns[table_name] += args.map{|a| a.to_s.downcase} @ignore_table_columns[table_name].uniq! end def ignored_table_columns(table_name) @ignore_table_columns ||= {} @ignore_table_columns[table_name] end # RSI: set explicit type for specified table columns def set_type_for_columns(table_name, column_type, *args) @table_column_type ||= {} @table_column_type[table_name] ||= {} args.each do |col| @table_column_type[table_name][col.to_s.downcase] = column_type end end def get_type_for_column(table_name, column_name) result = @table_column_type && @table_column_type[table_name] && @table_column_type[table_name][column_name.to_s.downcase] result end def clear_types_for_columns @table_column_type = nil end def columns(table_name, name = nil) #:nodoc: # RSI: get ignored_columns by original table name ignored_columns = ignored_table_columns(table_name) (owner, desc_table_name) = @connection.describe(quote_table_name(table_name)) table_cols = <<-SQL select column_name as name, data_type as sql_type, data_default, nullable, decode(data_type, 'NUMBER', data_precision, 'FLOAT', data_precision, 'VARCHAR2', data_length, 'CHAR', data_length, null) as limit, decode(data_type, 'NUMBER', data_scale, null) as scale from all_tab_columns where owner = '#{owner}' and table_name = '#{desc_table_name}' order by column_id SQL # RSI: added deletion of ignored columns select_all(table_cols, name).delete_if do |row| ignored_columns && ignored_columns.include?(row['name'].downcase) end.map do |row| limit, scale = row['limit'], row['scale'] if limit || scale row['sql_type'] << "(#{(limit || 38).to_i}" + ((scale = scale.to_i) > 0 ? ",#{scale})" : ")") end # clean up odd default spacing from Oracle if row['data_default'] row['data_default'].sub!(/^(.*?)\s*$/, '\1') row['data_default'].sub!(/^'(.*)'$/, '\1') row['data_default'] = nil if row['data_default'] =~ /^(null|empty_[bc]lob\(\))$/i end OracleEnhancedColumn.new(oracle_downcase(row['name']), row['data_default'], row['sql_type'], row['nullable'] == 'Y', # RSI: pass table name for table specific column definitions table_name, # RSI: pass column type if specified in class definition get_type_for_column(table_name, oracle_downcase(row['name']))) end end # RSI: default sequence start with value @@default_sequence_start_value = 10000 cattr_accessor :default_sequence_start_value def create_table(name, options = {}, &block) #:nodoc: create_sequence = options[:id] != false column_comments = {} super(name, options) do |t| # store that primary key was defined in create_table block unless create_sequence class << t attr_accessor :create_sequence def primary_key(*args) self.create_sequence = true super(*args) end end end # store column comments class << t attr_accessor :column_comments def column(name, type, options = {}) if options[:comment] self.column_comments ||= {} self.column_comments[name] = options[:comment] end super(name, type, options) end end result = block.call(t) create_sequence = create_sequence || t.create_sequence column_comments = t.column_comments if t.column_comments end seq_name = options[:sequence_name] || quote_table_name("#{name}_seq") seq_start_value = options[:sequence_start_value] || default_sequence_start_value execute "CREATE SEQUENCE #{seq_name} START WITH #{seq_start_value}" if create_sequence add_table_comment name, options[:comment] column_comments.each do |column_name, comment| add_comment name, column_name, comment end end def rename_table(name, new_name) #:nodoc: execute "RENAME #{name} TO #{new_name}" execute "RENAME #{name}_seq TO #{new_name}_seq" rescue nil end def drop_table(name, options = {}) #:nodoc: super(name) seq_name = options[:sequence_name] || quote_table_name("#{name}_seq") execute "DROP SEQUENCE #{seq_name}" rescue nil end def remove_index(table_name, options = {}) #:nodoc: execute "DROP INDEX #{index_name(table_name, options)}" end def change_column_default(table_name, column_name, default) #:nodoc: execute "ALTER TABLE #{table_name} MODIFY #{quote_column_name(column_name)} DEFAULT #{quote(default)}" end def change_column(table_name, column_name, type, options = {}) #:nodoc: change_column_sql = "ALTER TABLE #{table_name} MODIFY #{quote_column_name(column_name)} #{type_to_sql(type, options[:limit], options[:precision], options[:scale])}" add_column_options!(change_column_sql, options) execute(change_column_sql) end def rename_column(table_name, column_name, new_column_name) #:nodoc: execute "ALTER TABLE #{table_name} RENAME COLUMN #{quote_column_name(column_name)} to #{quote_column_name(new_column_name)}" end def remove_column(table_name, column_name) #:nodoc: execute "ALTER TABLE #{table_name} DROP COLUMN #{quote_column_name(column_name)}" end # RSI: table and column comments def add_comment(table_name, column_name, comment) return if comment.blank? execute "COMMENT ON COLUMN #{quote_table_name(table_name)}.#{column_name} IS '#{comment}'" end def add_table_comment(table_name, comment) return if comment.blank? execute "COMMENT ON TABLE #{quote_table_name(table_name)} IS '#{comment}'" end def table_comment(table_name) (owner, table_name) = @connection.describe(quote_table_name(table_name)) select_value <<-SQL SELECT comments FROM all_tab_comments WHERE owner = '#{owner}' AND table_name = '#{table_name}' SQL end def column_comment(table_name, column_name) (owner, table_name) = @connection.describe(quote_table_name(table_name)) select_value <<-SQL SELECT comments FROM all_col_comments WHERE owner = '#{owner}' AND table_name = '#{table_name}' AND column_name = '#{column_name.upcase}' SQL end # Find a table's primary key and sequence. # *Note*: Only primary key is implemented - sequence will be nil. def pk_and_sequence_for(table_name) (owner, table_name) = @connection.describe(quote_table_name(table_name)) # RSI: changed select from all_constraints to user_constraints - much faster in large data dictionaries pks = select_values(<<-SQL, 'Primary Key') select cc.column_name from user_constraints c, all_cons_columns cc where c.owner = '#{owner}' and c.table_name = '#{table_name}' and c.constraint_type = 'P' and cc.owner = c.owner and cc.constraint_name = c.constraint_name SQL # only support single column keys pks.size == 1 ? [oracle_downcase(pks.first), nil] : nil end def structure_dump #:nodoc: s = select_all("select sequence_name from user_sequences").inject("") do |structure, seq| structure << "create sequence #{seq.to_a.first.last};\n\n" end # RSI: changed select from user_tables to all_tables - much faster in large data dictionaries select_all("select table_name from all_tables where owner = sys_context('userenv','session_user')").inject(s) do |structure, table| ddl = "create table #{table.to_a.first.last} (\n " cols = select_all(%Q{ select column_name, data_type, data_length, char_used, char_length, data_precision, data_scale, data_default, nullable from user_tab_columns where table_name = '#{table.to_a.first.last}' order by column_id }).map do |row| col = "#{row['column_name'].downcase} #{row['data_type'].downcase}" if row['data_type'] =='NUMBER' and !row['data_precision'].nil? col << "(#{row['data_precision'].to_i}" col << ",#{row['data_scale'].to_i}" if !row['data_scale'].nil? col << ')' elsif row['data_type'].include?('CHAR') length = row['char_used'] == 'C' ? row['char_length'].to_i : row['data_length'].to_i col << "(#{length})" end col << " default #{row['data_default']}" if !row['data_default'].nil? col << ' not null' if row['nullable'] == 'N' col end ddl << cols.join(",\n ") ddl << ");\n\n" structure << ddl end end def structure_drop #:nodoc: s = select_all("select sequence_name from user_sequences").inject("") do |drop, seq| drop << "drop sequence #{seq.to_a.first.last};\n\n" end # RSI: changed select from user_tables to all_tables - much faster in large data dictionaries select_all("select table_name from all_tables where owner = sys_context('userenv','session_user')").inject(s) do |drop, table| drop << "drop table #{table.to_a.first.last} cascade constraints;\n\n" end end def add_column_options!(sql, options) #:nodoc: # handle case of defaults for CLOB columns, which would otherwise get "quoted" incorrectly if options_include_default?(options) && (column = options[:column]) && column.type == :text sql << " DEFAULT #{quote(options.delete(:default))}" end super end # SELECT DISTINCT clause for a given set of columns and a given ORDER BY clause. # # Oracle requires the ORDER BY columns to be in the SELECT list for DISTINCT # queries. However, with those columns included in the SELECT DISTINCT list, you # won't actually get a distinct list of the column you want (presuming the column # has duplicates with multiple values for the ordered-by columns. So we use the # FIRST_VALUE function to get a single (first) value for each column, effectively # making every row the same. # # distinct("posts.id", "posts.created_at desc") def distinct(columns, order_by) return "DISTINCT #{columns}" if order_by.blank? # construct a valid DISTINCT clause, ie. one that includes the ORDER BY columns, using # FIRST_VALUE such that the inclusion of these columns doesn't invalidate the DISTINCT order_columns = order_by.split(',').map { |s| s.strip }.reject(&:blank?) order_columns = order_columns.zip((0...order_columns.size).to_a).map do |c, i| "FIRST_VALUE(#{c.split.first}) OVER (PARTITION BY #{columns} ORDER BY #{c}) AS alias_#{i}__" end sql = "DISTINCT #{columns}, " sql << order_columns * ", " end # ORDER BY clause for the passed order option. # # Uses column aliases as defined by #distinct. def add_order_by_for_association_limiting!(sql, options) return sql if options[:order].blank? order = options[:order].split(',').collect { |s| s.strip }.reject(&:blank?) order.map! {|s| $1 if s =~ / (.*)/} order = order.zip((0...order.size).to_a).map { |s,i| "alias_#{i}__ #{s}" }.join(', ') sql << " ORDER BY #{order}" end private def select(sql, name = nil) cursor = execute(sql, name) cols = cursor.get_col_names.map { |x| oracle_downcase(x) } rows = [] while row = cursor.fetch hash = Hash.new cols.each_with_index do |col, i| hash[col] = case row[i] when OCI8::LOB name == 'Writable Large Object' ? row[i]: row[i].read when OraDate d = row[i] # RSI: added emulate_dates_by_column_name functionality # if emulate_dates_by_column_name && self.class.is_date_column?(col) # d.to_date # elsif if emulate_dates && (d.hour == 0 && d.minute == 0 && d.second == 0) d.to_date else # see string_to_time; Time overflowing to DateTime, respecting the default timezone time_array = [d.year, d.month, d.day, d.hour, d.minute, d.second] begin Time.send(Base.default_timezone, *time_array) rescue zone_offset = if Base.default_timezone == :local then DateTime.now.offset else 0 end # Append zero calendar reform start to account for dates skipped by calendar reform DateTime.new(*time_array[0..5] << zone_offset << 0) rescue nil end end # RSI: added emulate_integers_by_column_name functionality when Float n = row[i] if emulate_integers_by_column_name && self.class.is_integer_column?(col) n.to_i else n end else row[i] end unless col == 'raw_rnum_' end rows << hash end rows ensure cursor.close if cursor end # Oracle column names by default are case-insensitive, but treated as upcase; # for neatness, we'll downcase within Rails. EXCEPT that folks CAN quote # their column names when creating Oracle tables, which makes then case-sensitive. # I don't know anybody who does this, but we'll handle the theoretical case of a # camelCase column name. I imagine other dbs handle this different, since there's a # unit test that's currently failing test_oci. def oracle_downcase(column_name) column_name =~ /[a-z]/ ? column_name : column_name.downcase end end end end class OCI8 #:nodoc: # This OCI8 patch may not longer be required with the upcoming # release of version 0.2. class Cursor #:nodoc: alias :enhanced_define_a_column_pre_ar :define_a_column def define_a_column(i) case do_ocicall(@ctx) { @parms[i - 1].attrGet(OCI_ATTR_DATA_TYPE) } when 8; @stmt.defineByPos(i, String, 65535) # Read LONG values when 187; @stmt.defineByPos(i, OraDate) # Read TIMESTAMP values when 108 if @parms[i - 1].attrGet(OCI_ATTR_TYPE_NAME) == 'XMLTYPE' @stmt.defineByPos(i, String, 65535) else raise 'unsupported datatype' end else enhanced_define_a_column_pre_ar i end end end # missing constant from oci8 < 0.1.14 OCI_PTYPE_UNK = 0 unless defined?(OCI_PTYPE_UNK) # Uses the describeAny OCI call to find the target owner and table_name # indicated by +name+, parsing through synonynms as necessary. Returns # an array of [owner, table_name]. def describe(name) @desc ||= @@env.alloc(OCIDescribe) @desc.attrSet(OCI_ATTR_DESC_PUBLIC, -1) if VERSION >= '0.1.14' do_ocicall(@ctx) { @desc.describeAny(@svc, name.to_s, OCI_PTYPE_UNK) } rescue raise %Q{"DESC #{name}" failed; does it exist?} info = @desc.attrGet(OCI_ATTR_PARAM) case info.attrGet(OCI_ATTR_PTYPE) when OCI_PTYPE_TABLE, OCI_PTYPE_VIEW owner = info.attrGet(OCI_ATTR_OBJ_SCHEMA) table_name = info.attrGet(OCI_ATTR_OBJ_NAME) [owner, table_name] when OCI_PTYPE_SYN schema = info.attrGet(OCI_ATTR_SCHEMA_NAME) name = info.attrGet(OCI_ATTR_NAME) describe(schema + '.' + name) else raise %Q{"DESC #{name}" failed; not a table or view.} end end end # The OracleConnectionFactory factors out the code necessary to connect and # configure an Oracle/OCI connection. class OracleEnhancedConnectionFactory #:nodoc: def new_connection(username, password, database, async, prefetch_rows, cursor_sharing, privilege) conn = OCI8.new username, password, database, privilege conn.exec %q{alter session set nls_date_format = 'YYYY-MM-DD HH24:MI:SS'} conn.exec %q{alter session set nls_timestamp_format = 'YYYY-MM-DD HH24:MI:SS'} rescue nil conn.autocommit = true conn.non_blocking = true if async conn.prefetch_rows = prefetch_rows conn.exec "alter session set cursor_sharing = #{cursor_sharing}" rescue nil conn end end # The OCI8AutoRecover class enhances the OCI8 driver with auto-recover and # reset functionality. If a call to #exec fails, and autocommit is turned on # (ie., we're not in the middle of a longer transaction), it will # automatically reconnect and try again. If autocommit is turned off, # this would be dangerous (as the earlier part of the implied transaction # may have failed silently if the connection died) -- so instead the # connection is marked as dead, to be reconnected on it's next use. class OCI8EnhancedAutoRecover < DelegateClass(OCI8) #:nodoc: attr_accessor :active alias :active? :active cattr_accessor :auto_retry class << self alias :auto_retry? :auto_retry end @@auto_retry = false def initialize(config, factory = OracleEnhancedConnectionFactory.new) @active = true @username, @password, @database, = config[:username].to_s, config[:password].to_s, config[:database].to_s @async = config[:allow_concurrency] @prefetch_rows = config[:prefetch_rows] || 100 @cursor_sharing = config[:cursor_sharing] || 'similar' @factory = factory @privilege = config[:privilege] @connection = @factory.new_connection @username, @password, @database, @async, @prefetch_rows, @cursor_sharing, @privilege super @connection end # Checks connection, returns true if active. Note that ping actively # checks the connection, while #active? simply returns the last # known state. def ping @connection.exec("select 1 from dual") { |r| nil } @active = true rescue @active = false raise end # Resets connection, by logging off and creating a new connection. def reset! logoff rescue nil begin @connection = @factory.new_connection @username, @password, @database, @async, @prefetch_rows, @cursor_sharing, @privilege __setobj__ @connection @active = true rescue @active = false raise end end # ORA-00028: your session has been killed # ORA-01012: not logged on # ORA-03113: end-of-file on communication channel # ORA-03114: not connected to ORACLE LOST_CONNECTION_ERROR_CODES = [ 28, 1012, 3113, 3114 ] # Adds auto-recovery functionality. # # See: http://www.jiubao.org/ruby-oci8/api.en.html#label-11 def exec(sql, *bindvars, &block) should_retry = self.class.auto_retry? && autocommit? begin @connection.exec(sql, *bindvars, &block) rescue OCIException => e raise unless LOST_CONNECTION_ERROR_CODES.include?(e.code) @active = false raise unless should_retry should_retry = false reset! rescue nil retry end end end rescue LoadError # OCI8 driver is unavailable. if defined?(RAILS_DEFAULT_LOGGER) RAILS_DEFAULT_LOGGER.error "ERROR: ActiveRecord oracle_enhanced adapter could not load ruby-oci8 library. "+ "Please install ruby-oci8 library or gem." end module ActiveRecord # :nodoc: class Base @@oracle_error_message = "Oracle/OCI libraries could not be loaded: #{$!.to_s}" def self.oracle_enhanced_connection(config) # :nodoc: # Set up a reasonable error message raise LoadError, @@oracle_error_message end end end end # RSI: Added LOB writing callback for sessions stored in database # Otherwise it is not working as Session class is defined before OracleAdapter is loaded in Rails 2.0 if defined?(CGI::Session::ActiveRecordStore::Session) if !CGI::Session::ActiveRecordStore::Session.respond_to?(:after_save_callback_chain) || CGI::Session::ActiveRecordStore::Session.after_save_callback_chain.detect{|cb| cb.method == :enhanced_write_lobs}.nil? class CGI::Session::ActiveRecordStore::Session after_save :enhanced_write_lobs end end end # RSI: load custom create, update, delete methods functionality # rescue LoadError if ruby-plsql gem cannot be loaded begin require 'active_record/connection_adapters/oracle_enhanced_procedures' rescue LoadError if defined?(RAILS_DEFAULT_LOGGER) RAILS_DEFAULT_LOGGER.info "INFO: ActiveRecord oracle_enhanced adapter could not load ruby-plsql gem. "+ "Custom create, update and delete methods will not be available." end end # RSI: load additional methods for composite_primary_keys support require 'active_record/connection_adapters/oracle_enhanced_cpk' # RSI: load patch for dirty tracking methods require 'active_record/connection_adapters/oracle_enhanced_dirty' # handles quoting of oracle reserved words require 'active_record/connection_adapters/oracle_enhanced_reserved_words'