require 'spec_helper' describe "OracleEnhancedConnection" do describe "create connection" do before(:all) do @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS) end before(:each) do @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS) unless @conn.active? end it "should create new connection" do @conn.should be_active end it "should ping active connection" do @conn.ping.should be_true end it "should not ping inactive connection" do @conn.logoff lambda { @conn.ping }.should raise_error(ActiveRecord::ConnectionAdapters::OracleEnhancedConnectionException) end it "should reset active connection" do @conn.reset! @conn.should be_active end it "should be in autocommit mode after connection" do @conn.should be_autocommit end end describe "create connection with NLS parameters" do after do ENV['NLS_DATE_FORMAT'] = nil end it "should use NLS_DATE_FORMAT environment variable" do ENV['NLS_DATE_FORMAT'] = 'YYYY-MM-DD' @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS) @conn.select("SELECT value FROM v$nls_parameters WHERE parameter = 'NLS_DATE_FORMAT'").should == [{'value' => 'YYYY-MM-DD'}] end it "should use configuration value and ignore NLS_DATE_FORMAT environment variable" do ENV['NLS_DATE_FORMAT'] = 'YYYY-MM-DD' @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS.merge(:nls_date_format => 'YYYY-MM-DD HH24:MI')) @conn.select("SELECT value FROM v$nls_parameters WHERE parameter = 'NLS_DATE_FORMAT'").should == [{'value' => 'YYYY-MM-DD HH24:MI'}] end it "should use default value when NLS_DATE_FORMAT environment variable is not set" do ENV['NLS_DATE_FORMAT'] = nil @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS) default = ActiveRecord::ConnectionAdapters::OracleEnhancedAdapter::DEFAULT_NLS_PARAMETERS[:nls_date_format] @conn.select("SELECT value FROM v$nls_parameters WHERE parameter = 'NLS_DATE_FORMAT'").should == [{'value' => default}] end end describe "with non-string parameters" do before(:all) do params = CONNECTION_PARAMS.dup params[:username] = params[:username].to_sym params[:password] = params[:password].to_sym params[:database] = params[:database].to_sym @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(params) end it "should create new connection" do @conn.should be_active end end describe "with slash-prefixed database name (service name)" do before(:all) do params = CONNECTION_PARAMS.dup params[:database] = "/#{params[:database]}" unless params[:database].match(/^\//) @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(params) end it "should create new connection" do @conn.should be_active end end if defined?(RUBY_ENGINE) && RUBY_ENGINE == 'jruby' describe "create JDBC connection" do it "should create new connection using :url" do params = CONNECTION_PARAMS.dup params[:url] = "jdbc:oracle:thin:@#{DATABASE_HOST && "//#{DATABASE_HOST}#{DATABASE_PORT && ":#{DATABASE_PORT}"}/"}#{DATABASE_NAME}" params[:host] = nil params[:database] = nil @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(params) @conn.should be_active end it "should create new connection using :url and tnsnames alias" do params = CONNECTION_PARAMS.dup params[:url] = "jdbc:oracle:thin:@#{DATABASE_NAME}" params[:host] = nil params[:database] = nil @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(params) @conn.should be_active end it "should create new connection using just tnsnames alias" do params = CONNECTION_PARAMS.dup params[:host] = nil @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(params) @conn.should be_active end it "should create a new connection using JNDI" do begin import 'oracle.jdbc.driver.OracleDriver' import 'org.apache.commons.pool.impl.GenericObjectPool' import 'org.apache.commons.dbcp.PoolingDataSource' import 'org.apache.commons.dbcp.PoolableConnectionFactory' import 'org.apache.commons.dbcp.DriverManagerConnectionFactory' rescue NameError => e return pending e.message end class InitialContextMock def initialize connection_pool = GenericObjectPool.new(nil) uri = "jdbc:oracle:thin:@#{DATABASE_HOST && "#{DATABASE_HOST}:"}#{DATABASE_PORT && "#{DATABASE_PORT}:"}#{DATABASE_NAME}" connection_factory = DriverManagerConnectionFactory.new(uri, DATABASE_USER, DATABASE_PASSWORD) poolable_connection_factory = PoolableConnectionFactory.new(connection_factory,connection_pool,nil,nil,false,true) @data_source = PoolingDataSource.new(connection_pool) @data_source.access_to_underlying_connection_allowed = true end def lookup(path) if (path == 'java:/comp/env') return self else return @data_source end end end javax.naming.InitialContext.stub!(:new).and_return(InitialContextMock.new) params = {} params[:jndi] = 'java:comp/env/jdbc/test' @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(params) @conn.should be_active end end it "should fall back to directly instantiating OracleDriver" do params = CONNECTION_PARAMS.dup params[:url] = "jdbc:oracle:thin:@#{DATABASE_HOST && "//#{DATABASE_HOST}#{DATABASE_PORT && ":#{DATABASE_PORT}"}/"}#{DATABASE_NAME}" params[:host] = nil params[:database] = nil java.sql.DriverManager.stub!(:getConnection).and_raise('no suitable driver found') @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(params) @conn.should be_active end end describe "SQL execution" do before(:all) do @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS) end it "should execute SQL statement" do @conn.exec("SELECT * FROM dual").should_not be_nil end it "should execute SQL select" do @conn.select("SELECT * FROM dual").should == [{'dummy' => 'X'}] end it "should execute SQL select and return also columns" do @conn.select("SELECT * FROM dual", nil, true).should == [ [{'dummy' => 'X'}], ['dummy'] ] end end describe "SQL with bind parameters" do before(:all) do @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS) end it "should execute SQL statement with bind parameter" do cursor = @conn.prepare("SELECT * FROM dual WHERE :1 = 1") cursor.bind_param(1, 1) cursor.exec cursor.get_col_names.should == ['DUMMY'] cursor.fetch.should == ["X"] cursor.close end it "should execute prepared statement with different bind parameters" do cursor = @conn.prepare("SELECT * FROM dual WHERE :1 = 1") cursor.bind_param(1, 1) cursor.exec cursor.fetch.should == ["X"] cursor.bind_param(1, 0) cursor.exec cursor.fetch.should be_nil cursor.close end end describe "SQL with bind parameters when NLS_NUMERIC_CHARACTERS is set to ', '" do before(:all) do ENV['NLS_NUMERIC_CHARACTERS'] = ", " @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS) @conn.exec "CREATE TABLE test_employees (age NUMBER(10,2))" end after(:all) do ENV['NLS_NUMERIC_CHARACTERS'] = nil @conn.exec "DROP TABLE test_employees" rescue nil end it "should execute prepared statement with decimal bind parameter " do cursor = @conn.prepare("INSERT INTO test_employees VALUES(:1)") column = ActiveRecord::ConnectionAdapters::OracleEnhancedColumn.new('age', nil, 'NUMBER(10,2)') column.type.should == :decimal cursor.bind_param(1, "1.5", column) cursor.exec cursor.close cursor = @conn.prepare("SELECT age FROM test_employees") cursor.exec cursor.fetch.should == [1.5] cursor.close end end describe "auto reconnection" do before(:all) do ActiveRecord::Base.establish_connection(CONNECTION_PARAMS) @conn = ActiveRecord::Base.connection.instance_variable_get("@connection") @sys_conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(SYS_CONNECTION_PARAMS) end before(:each) do ActiveRecord::Base.connection.reconnect! unless @conn.active? end def kill_current_session audsid = @conn.select("SELECT userenv('sessionid') audsid FROM dual").first['audsid'] sid_serial = @sys_conn.select("SELECT s.sid||','||s.serial# sid_serial FROM v$session s WHERE audsid = '#{audsid}'").first['sid_serial'] @sys_conn.exec "ALTER SYSTEM KILL SESSION '#{sid_serial}' IMMEDIATE" end it "should reconnect and execute SQL statement if connection is lost and auto retry is enabled" do # @conn.auto_retry = true ActiveRecord::Base.connection.auto_retry = true kill_current_session @conn.exec("SELECT * FROM dual").should_not be_nil end it "should not reconnect and execute SQL statement if connection is lost and auto retry is disabled" do # @conn.auto_retry = false ActiveRecord::Base.connection.auto_retry = false kill_current_session lambda { @conn.exec("SELECT * FROM dual") }.should raise_error end it "should reconnect and execute SQL select if connection is lost and auto retry is enabled" do # @conn.auto_retry = true ActiveRecord::Base.connection.auto_retry = true kill_current_session @conn.select("SELECT * FROM dual").should == [{'dummy' => 'X'}] end it "should not reconnect and execute SQL select if connection is lost and auto retry is disabled" do # @conn.auto_retry = false ActiveRecord::Base.connection.auto_retry = false kill_current_session lambda { @conn.select("SELECT * FROM dual") }.should raise_error end end describe "describe table" do before(:all) do @conn = ActiveRecord::ConnectionAdapters::OracleEnhancedConnection.create(CONNECTION_PARAMS) @owner = CONNECTION_PARAMS[:username].upcase end it "should describe existing table" do @conn.exec "CREATE TABLE test_employees (first_name VARCHAR2(20))" rescue nil @conn.describe("test_employees").should == [@owner, "TEST_EMPLOYEES"] @conn.exec "DROP TABLE test_employees" rescue nil end it "should not describe non-existing table" do lambda { @conn.describe("test_xxx") }.should raise_error(ActiveRecord::ConnectionAdapters::OracleEnhancedConnectionException) end it "should describe table in other schema" do @conn.describe("sys.dual").should == ["SYS", "DUAL"] end it "should describe existing view" do @conn.exec "CREATE TABLE test_employees (first_name VARCHAR2(20))" rescue nil @conn.exec "CREATE VIEW test_employees_v AS SELECT * FROM test_employees" rescue nil @conn.describe("test_employees_v").should == [@owner, "TEST_EMPLOYEES_V"] @conn.exec "DROP VIEW test_employees_v" rescue nil @conn.exec "DROP TABLE test_employees" rescue nil end it "should describe view in other schema" do @conn.describe("sys.v_$version").should == ["SYS", "V_$VERSION"] end it "should describe existing private synonym" do @conn.exec "CREATE SYNONYM test_dual FOR sys.dual" rescue nil @conn.describe("test_dual").should == ["SYS", "DUAL"] @conn.exec "DROP SYNONYM test_dual" rescue nil end it "should describe existing public synonym" do @conn.describe("all_tables").should == ["SYS", "ALL_TABLES"] end if defined?(OCI8) context "OCI8 adapter" do it "should not fallback to SELECT-based logic when querying non-existant table information" do @conn.should_not_receive(:select_one) @conn.describe("non_existant") rescue ActiveRecord::ConnectionAdapters::OracleEnhancedConnectionException end end end end end