#!/usr/bin/env rspec #encoding: utf-8 BEGIN { require 'pathname' basedir = Pathname( __FILE__ ).dirname.parent.parent libdir = basedir + 'lib' $LOAD_PATH.unshift( basedir.to_s ) unless $LOAD_PATH.include?( basedir.to_s ) $LOAD_PATH.unshift( libdir.to_s ) unless $LOAD_PATH.include?( libdir.to_s ) } require 'rspec' require 'spec/lib/helpers' require 'timeout' require 'socket' require 'pg' describe PG::Connection do before( :all ) do @conn = setup_testing_db( described_class.name ) end before( :each ) do |example| @conn.exec( 'BEGIN' ) unless example.metadata[:without_transaction] if PG.respond_to?( :library_version ) @conn.exec_params %Q{SET application_name TO '%s'} % [@conn.escape_string(example.description[0,60])] end end after( :each ) do |example| @conn.exec( 'ROLLBACK' ) unless example.metadata[:without_transaction] end after( :all ) do teardown_testing_db( @conn ) end # # Examples # it "can create a connection option string from a Hash of options" do optstring = described_class.parse_connect_args( :host => 'pgsql.example.com', :dbname => 'db01', 'sslmode' => 'require' ) optstring.should be_a( String ) optstring.should =~ /(^|\s)host='pgsql.example.com'/ optstring.should =~ /(^|\s)dbname='db01'/ optstring.should =~ /(^|\s)sslmode='require'/ end it "can create a connection option string from positional parameters" do optstring = described_class.parse_connect_args( 'pgsql.example.com', nil, '-c geqo=off', nil, 'sales' ) optstring.should be_a( String ) optstring.should =~ /(^|\s)host='pgsql.example.com'/ optstring.should =~ /(^|\s)dbname='sales'/ optstring.should =~ /(^|\s)options='-c geqo=off'/ optstring.should_not =~ /port=/ optstring.should_not =~ /tty=/ end it "can create a connection option string from a mix of positional and hash parameters" do optstring = described_class.parse_connect_args( 'pgsql.example.com', :dbname => 'licensing', :user => 'jrandom' ) optstring.should be_a( String ) optstring.should =~ /(^|\s)host='pgsql.example.com'/ optstring.should =~ /(^|\s)dbname='licensing'/ optstring.should =~ /(^|\s)user='jrandom'/ end it "escapes single quotes and backslashes in connection parameters" do described_class.parse_connect_args( "DB 'browser' \\" ). should =~ /host='DB \\'browser\\' \\\\'/ end it "connects with defaults if no connection parameters are given" do described_class.parse_connect_args.should == '' end it "connects successfully with connection string" do tmpconn = described_class.connect(@conninfo) tmpconn.status.should == PG::CONNECTION_OK tmpconn.finish end it "connects using 7 arguments converted to strings" do tmpconn = described_class.connect('localhost', @port, nil, nil, :test, nil, nil) tmpconn.status.should == PG::CONNECTION_OK tmpconn.finish end it "connects using a hash of connection parameters" do tmpconn = described_class.connect( :host => 'localhost', :port => @port, :dbname => :test) tmpconn.status.should == PG::CONNECTION_OK tmpconn.finish end it "connects using a hash of optional connection parameters", :postgresql_90 do tmpconn = described_class.connect( :host => 'localhost', :port => @port, :dbname => :test, :keepalives => 1) tmpconn.status.should == PG::CONNECTION_OK tmpconn.finish end it "raises an exception when connecting with an invalid number of arguments" do expect { described_class.connect( 1, 2, 3, 4, 5, 6, 7, 'extra' ) }.to raise_error( ArgumentError, /extra positional parameter/i ) end it "can connect asynchronously", :socket_io do tmpconn = described_class.connect_start( @conninfo ) tmpconn.should be_a( described_class ) socket = tmpconn.socket_io status = tmpconn.connect_poll while status != PG::PGRES_POLLING_OK if status == PG::PGRES_POLLING_READING select( [socket], [], [], 5.0 ) or raise "Asynchronous connection timed out!" elsif status == PG::PGRES_POLLING_WRITING select( [], [socket], [], 5.0 ) or raise "Asynchronous connection timed out!" end status = tmpconn.connect_poll end tmpconn.status.should == PG::CONNECTION_OK tmpconn.finish end it "can connect asynchronously for the duration of a block", :socket_io do conn = nil described_class.connect_start(@conninfo) do |tmpconn| tmpconn.should be_a( described_class ) conn = tmpconn socket = tmpconn.socket_io status = tmpconn.connect_poll while status != PG::PGRES_POLLING_OK if status == PG::PGRES_POLLING_READING if(not select([socket],[],[],5.0)) raise "Asynchronous connection timed out!" end elsif(status == PG::PGRES_POLLING_WRITING) if(not select([],[socket],[],5.0)) raise "Asynchronous connection timed out!" end end status = tmpconn.connect_poll end tmpconn.status.should == PG::CONNECTION_OK end conn.should be_finished() end it "raises proper error when sending fails" do conn = described_class.connect_start( '127.0.0.1', 54320, "", "", "me", "xxxx", "somedb" ) expect{ conn.exec 'SELECT 1' }.to raise_error(PG::UnableToSend, /no connection/) end it "doesn't leave stale server connections after finish" do described_class.connect(@conninfo).finish sleep 0.5 res = @conn.exec(%[SELECT COUNT(*) AS n FROM pg_stat_activity WHERE usename IS NOT NULL]) # there's still the global @conn, but should be no more res[0]['n'].should == '1' end EXPECTED_TRACE_OUTPUT = %{ To backend> Msg Q To backend> "SELECT 1 AS one" To backend> Msg complete, length 21 From backend> T From backend (#4)> 28 From backend (#2)> 1 From backend> "one" From backend (#4)> 0 From backend (#2)> 0 From backend (#4)> 23 From backend (#2)> 4 From backend (#4)> -1 From backend (#2)> 0 From backend> D From backend (#4)> 11 From backend (#2)> 1 From backend (#4)> 1 From backend (1)> 1 From backend> C From backend (#4)> 13 From backend> "SELECT 1" From backend> Z From backend (#4)> 5} + # JRuby will only print the msg header when it's ready to process # the message on the other hand, libpq will print the # ReadyForQuery msg header then realize it's in the Idle state # stop processing and wait for the user to consume the Result. # That's why we get a double 'Z' msg on MRI. if jruby? %{ From backend> T } else %{ From backend> Z From backend (#4)> 5 From backend> T } end it "trace and untrace client-server communication", :unix do # be careful to explicitly close files so that the # directory can be removed and we don't have to wait for # the GC to run. trace_file = TEST_DIRECTORY + "test_trace.out" trace_io = trace_file.open( 'w', 0600 ) @conn.trace( trace_io ) trace_io.close res = @conn.exec("SELECT 1 AS one") @conn.untrace res = @conn.exec("SELECT 2 AS two") trace_data = trace_file.read expected_trace_output = EXPECTED_TRACE_OUTPUT.dup.gsub( /^\t+/, '' ).lstrip # For PostgreSQL < 9.0, the output will be different: # -From backend (#4)> 13 # -From backend> "SELECT 1" # +From backend (#4)> 11 # +From backend> "SELECT" if @conn.server_version < 90000 expected_trace_output.sub!( /From backend \(#4\)> 13/, 'From backend (#4)> 11' ) expected_trace_output.sub!( /From backend> "SELECT 1"/, 'From backend> "SELECT"' ) end trace_data.should == expected_trace_output end it "allows a query to be cancelled" do error = false @conn.send_query("SELECT pg_sleep(1000)") @conn.cancel tmpres = @conn.get_result if(tmpres.result_status != PG::PGRES_TUPLES_OK) error = true end error.should == true end # We disable the following tests for JRuby. First because # Thread#kill isn't safe and on JRuby it can't effectively stop a # thread running native Java code. Second, JVM doesn't play nice # with signals especially USR1. Sending the signal to the JVM will # cause it to exit unless jruby? it "can stop a thread that runs a blocking query with async_exec" do start = Time.now t = Thread.new do @conn.async_exec( 'select pg_sleep(10)' ) end sleep 0.1 t.kill t.join (Time.now - start).should < 10 end it "should work together with signal handlers" do signal_received = false trap 'USR1' do signal_received = true end Thread.new do sleep 0.1 Process.kill("USR1", Process.pid) end @conn.exec("select pg_sleep(0.3)") signal_received.should be_true signal_received = false Thread.new do sleep 0.1 Process.kill("USR1", Process.pid) end @conn.async_exec("select pg_sleep(0.3)") signal_received.should be_true end end it "automatically rolls back a transaction started with Connection#transaction if an exception " + "is raised" do # abort the per-example transaction so we can test our own @conn.exec( 'ROLLBACK' ) res = nil @conn.exec( "CREATE TABLE pie ( flavor TEXT )" ) expect { res = @conn.transaction do @conn.exec( "INSERT INTO pie VALUES ('rhubarb'), ('cherry'), ('schizophrenia')" ) raise "Oh noes! All pie is gone!" end }.to raise_exception( RuntimeError, /all pie is gone/i ) res = @conn.exec( "SELECT * FROM pie" ) res.ntuples.should == 0 end it "returns the block result from Connection#transaction" do # abort the per-example transaction so we can test our own @conn.exec( 'ROLLBACK' ) res = @conn.transaction do "transaction result" end res.should == "transaction result" end it "not read past the end of a large object" do @conn.transaction do oid = @conn.lo_create( 0 ) fd = @conn.lo_open( oid, PG::INV_READ|PG::INV_WRITE ) @conn.lo_write( fd, "foobar" ) @conn.lo_read( fd, 10 ).should be_nil() @conn.lo_lseek( fd, 0, PG::SEEK_SET ) @conn.lo_read( fd, 10 ).should == 'foobar' end end it "supports parameters passed to #exec (backward compatibility)" do @conn.exec( "CREATE TABLE students ( name TEXT, age INTEGER )" ) @conn.exec( "INSERT INTO students VALUES( $1, $2 )", ['Wally', 8] ) @conn.exec( "INSERT INTO students VALUES( $1, $2 )", ['Sally', 6] ) @conn.exec( "INSERT INTO students VALUES( $1, $2 )", ['Dorothy', 4] ) res = @conn.exec( "SELECT name FROM students WHERE age >= $1", [6] ) res.values.should == [ ['Wally'], ['Sally'] ] end it "supports explicitly calling #exec_params" do @conn.exec( "CREATE TABLE students ( name TEXT, age INTEGER )" ) @conn.exec( "INSERT INTO students VALUES( $1, $2 )", ['Wally', 8] ) @conn.exec( "INSERT INTO students VALUES( $1, $2 )", ['Sally', 6] ) @conn.exec( "INSERT INTO students VALUES( $1, $2 )", ['Dorothy', 4] ) res = @conn.exec_params( "SELECT name FROM students WHERE age >= $1", [6] ) res.values.should == [ ['Wally'], ['Sally'] ] end it "can wait for NOTIFY events" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN woo' ) t = Thread.new do begin conn = described_class.connect( @conninfo ) sleep 1 conn.async_exec( 'NOTIFY woo' ) ensure conn.finish end end @conn.wait_for_notify( 10 ).should == 'woo' @conn.exec( 'UNLISTEN woo' ) t.join end it "calls a block for NOTIFY events if one is given" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN woo' ) t = Thread.new do begin conn = described_class.connect( @conninfo ) sleep 1 conn.async_exec( 'NOTIFY woo' ) ensure conn.finish end end eventpid = event = nil @conn.wait_for_notify( 10 ) {|*args| event, eventpid = args } event.should == 'woo' eventpid.should be_an( Integer ) @conn.exec( 'UNLISTEN woo' ) t.join end it "doesn't collapse sequential notifications" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN woo' ) @conn.exec( 'LISTEN war' ) @conn.exec( 'LISTEN woz' ) begin conn = described_class.connect( @conninfo ) conn.exec( 'NOTIFY woo' ) conn.exec( 'NOTIFY war' ) conn.exec( 'NOTIFY woz' ) ensure conn.finish end channels = [] 3.times do channels << @conn.wait_for_notify( 2 ) end channels.should have( 3 ).members channels.should include( 'woo', 'war', 'woz' ) @conn.exec( 'UNLISTEN woz' ) @conn.exec( 'UNLISTEN war' ) @conn.exec( 'UNLISTEN woo' ) end it "returns notifications which are already in the queue before wait_for_notify is called " + "without waiting for the socket to become readable" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN woo' ) begin conn = described_class.connect( @conninfo ) conn.exec( 'NOTIFY woo' ) ensure conn.finish end # Cause the notification to buffer, but not be read yet @conn.exec( 'SELECT 1' ) @conn.wait_for_notify( 10 ).should == 'woo' @conn.exec( 'UNLISTEN woo' ) end it "can receive notices while waiting for NOTIFY without exceeding the timeout", :postgresql_90 do notices = [] @conn.set_notice_processor do |msg| notices << [msg, Time.now] end st = Time.now @conn.send_query "SELECT pg_sleep(0.5); do $$ BEGIN RAISE NOTICE 'woohoo'; END; $$ LANGUAGE plpgsql;" @conn.wait_for_notify( 1 ).should be_nil notices.first.should_not be_nil et = Time.now (et - notices.first[1]).should >= 0.4 (et - st).should >= 0.9 (et - st).should < 1.4 end it "yields the result if block is given to exec" do rval = @conn.exec( "select 1234::int as a union select 5678::int as a" ) do |result| values = [] result.should be_kind_of( PG::Result ) result.ntuples.should == 2 result.each do |tuple| values << tuple['a'] end values end rval.should have( 2 ).members rval.should include( '5678', '1234' ) end it "can process #copy_data output queries" do rows = [] res2 = @conn.copy_data( "COPY (SELECT 1 UNION ALL SELECT 2) TO STDOUT" ) do |res| res.result_status.should == PG::PGRES_COPY_OUT res.nfields.should == 1 while row=@conn.get_copy_data rows << row end end rows.should == ["1\n", "2\n"] res2.result_status.should == PG::PGRES_COMMAND_OK verify_clean_exec_status end it "can handle incomplete #copy_data output queries" do expect { @conn.copy_data( "COPY (SELECT 1 UNION ALL SELECT 2) TO STDOUT" ) do |res| @conn.get_copy_data end }.to raise_error(PG::NotAllCopyDataRetrieved, /Not all/) verify_clean_exec_status end it "can handle client errors in #copy_data for output" do expect { @conn.copy_data( "COPY (SELECT 1 UNION ALL SELECT 2) TO STDOUT" ) do raise "boom" end }.to raise_error(RuntimeError, "boom") verify_clean_exec_status end it "can handle server errors in #copy_data for output" do @conn.exec "ROLLBACK" @conn.transaction do @conn.exec( "CREATE FUNCTION errfunc() RETURNS int AS $$ BEGIN RAISE 'test-error'; END; $$ LANGUAGE plpgsql;" ) expect { @conn.copy_data( "COPY (SELECT errfunc()) TO STDOUT" ) do |res| while @conn.get_copy_data end end }.to raise_error(PG::Error, /test-error/) end verify_clean_exec_status end it "can process #copy_data input queries" do @conn.exec( "CREATE TEMP TABLE copytable (col1 TEXT)" ) res2 = @conn.copy_data( "COPY copytable FROM STDOUT" ) do |res| res.result_status.should == PG::PGRES_COPY_IN res.nfields.should == 1 @conn.put_copy_data "1\n" @conn.put_copy_data "2\n" end res2.result_status.should == PG::PGRES_COMMAND_OK verify_clean_exec_status res = @conn.exec( "SELECT * FROM copytable ORDER BY col1" ) res.values.should == [["1"], ["2"]] end it "can handle client errors in #copy_data for input" do @conn.exec "ROLLBACK" @conn.transaction do @conn.exec( "CREATE TEMP TABLE copytable (col1 TEXT)" ) expect { @conn.copy_data( "COPY copytable FROM STDOUT" ) do |res| raise "boom" end }.to raise_error(RuntimeError, "boom") end verify_clean_exec_status end it "can handle server errors in #copy_data for input" do @conn.exec "ROLLBACK" @conn.transaction do @conn.exec( "CREATE TEMP TABLE copytable (col1 INT)" ) expect { @conn.copy_data( "COPY copytable FROM STDOUT" ) do |res| @conn.put_copy_data "xyz\n" end }.to raise_error(PG::Error, /invalid input syntax for integer/) end verify_clean_exec_status end it "should raise an error for non copy statements in #copy_data" do expect { @conn.copy_data( "SELECT 1" ){} }.to raise_error(ArgumentError, /no COPY/) verify_clean_exec_status end it "correctly finishes COPY queries passed to #async_exec" do @conn.async_exec( "COPY (SELECT 1 UNION ALL SELECT 2) TO STDOUT" ) results = [] begin data = @conn.get_copy_data( true ) if false == data @conn.block( 2.0 ) data = @conn.get_copy_data( true ) end results << data if data end until data.nil? results.should have( 2 ).members results.should include( "1\n", "2\n" ) end it "described_class#block shouldn't block a second thread" do start = Time.now t = Thread.new do @conn.send_query( "select pg_sleep(3)" ) @conn.block end sleep 0.5 t.should be_alive() @conn.cancel t.join (Time.now - start).should < 3 end it "described_class#block should allow a timeout" do @conn.send_query( "select pg_sleep(3)" ) start = Time.now @conn.block( 0.1 ) finish = Time.now (finish - start).should be_within( 0.05 ).of( 0.1 ) end it "can encrypt a string given a password and username" do described_class.encrypt_password("postgres", "postgres"). should =~ /\S+/ end it "raises an appropriate error if either of the required arguments for encrypt_password " + "is not valid" do expect { described_class.encrypt_password( nil, nil ) }.to raise_error( TypeError ) expect { described_class.encrypt_password( "postgres", nil ) }.to raise_error( TypeError ) expect { described_class.encrypt_password( nil, "postgres" ) }.to raise_error( TypeError ) end it "allows fetching a column of values from a result by column number" do res = @conn.exec( 'VALUES (1,2),(2,3),(3,4)' ) res.column_values( 0 ).should == %w[1 2 3] res.column_values( 1 ).should == %w[2 3 4] end it "allows fetching a column of values from a result by field name" do res = @conn.exec( 'VALUES (1,2),(2,3),(3,4)' ) res.field_values( 'column1' ).should == %w[1 2 3] res.field_values( 'column2' ).should == %w[2 3 4] end it "raises an error if selecting an invalid column index" do res = @conn.exec( 'VALUES (1,2),(2,3),(3,4)' ) expect { res.column_values( 20 ) }.to raise_error( IndexError ) end it "raises an error if selecting an invalid field name" do res = @conn.exec( 'VALUES (1,2),(2,3),(3,4)' ) expect { res.field_values( 'hUUuurrg' ) }.to raise_error( IndexError ) end it "raises an error if column index is not a number" do res = @conn.exec( 'VALUES (1,2),(2,3),(3,4)' ) expect { res.column_values( 'hUUuurrg' ) }.to raise_error( TypeError ) end it "can connect asynchronously", :socket_io do serv = TCPServer.new( '127.0.0.1', 54320 ) conn = described_class.connect_start( '127.0.0.1', 54320, "", "", "me", "xxxx", "somedb" ) [PG::PGRES_POLLING_WRITING, PG::CONNECTION_OK].should include conn.connect_poll select( nil, [conn.socket_io], nil, 0.2 ) serv.close if conn.connect_poll == PG::PGRES_POLLING_READING select( [conn.socket_io], nil, nil, 0.2 ) end conn.connect_poll.should == PG::PGRES_POLLING_FAILED end it "discards previous results (if any) before waiting on an #async_exec" it "ignore nil params" do res = @conn.async_exec 'select 1 as one', nil res[0]['one'].should == '1' end it "ignore nil params" do res = @conn.async_exec 'select $1::boolean as b', [false] res[0]['b'].should == 'f' end it "calls the block if one is provided to #async_exec" do result = nil @conn.async_exec( "select 47 as one" ) do |pg_res| result = pg_res[0] end result.should == { 'one' => '47' } end it "raises a rescue-able error if #finish is called twice", :without_transaction do conn = PG.connect( @conninfo ) conn.finish expect { conn.finish }.to raise_error( PG::ConnectionBad, /connection is closed/i ) end it "closes the IO fetched from #socket_io when the connection is closed", :without_transaction, :socket_io do conn = PG.connect( @conninfo ) io = conn.socket_io conn.finish io.should be_closed() expect { conn.socket_io }.to raise_error( PG::ConnectionBad, /connection is closed/i ) end it "closes the IO fetched from #socket_io when the connection is reset", :without_transaction, :socket_io do conn = PG.connect( @conninfo ) io = conn.socket_io conn.reset io.should be_closed() conn.socket_io.should_not equal( io ) conn.finish end it "block should raise ConnectionBad for a closed connection" do serv = TCPServer.new( '127.0.0.1', 54320 ) conn = described_class.connect_start( '127.0.0.1', 54320, "", "", "me", "xxxx", "somedb" ) while [PG::CONNECTION_STARTED, PG::CONNECTION_MADE].include?(conn.connect_poll) sleep 0.1 end serv.close expect{ conn.block }.to raise_error(PG::ConnectionBad, /server closed the connection unexpectedly|not connected/) expect{ conn.block }.to raise_error(PG::ConnectionBad, /can't get socket descriptor|not connected/) end context "under PostgreSQL 9", :postgresql_90 do before( :each ) do pending "only works with a PostgreSQL >= 9.0 server" if @conn.server_version < 9_00_00 end it "sets the fallback_application_name on new connections" do conn_string = PG::Connection.parse_connect_args( 'dbname=test' ) connection_string_should_contain_application_name(conn_string, $0) end it "sets a shortened fallback_application_name on new connections" do old_0 = $0 begin $0 = "/this/is/a/very/long/path/with/many/directories/to/our/beloved/ruby" conn_string = PG::Connection.parse_connect_args( 'dbname=test' ) connection_string_should_contain_application_name(conn_string, $0) ensure $0 = old_0 end end it "calls the block supplied to wait_for_notify with the notify" + " payload if it accepts any number of arguments" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN knees' ) conn = described_class.connect( @conninfo ) conn.exec( %Q{NOTIFY knees, 'skirt and boots'} ) conn.finish event, pid, msg = nil @conn.wait_for_notify( 10 ) do |*args| event, pid, msg = *args end @conn.exec( 'UNLISTEN knees' ) event.should == 'knees' pid.should be_a_kind_of( Integer ) msg.should == 'skirt and boots' end it "accepts nil as the timeout in #wait_for_notify " do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN knees' ) conn = described_class.connect( @conninfo ) conn.exec( %Q{NOTIFY knees} ) conn.finish event, pid = nil @conn.wait_for_notify( nil ) do |*args| event, pid = *args end @conn.exec( 'UNLISTEN knees' ) event.should == 'knees' pid.should be_a_kind_of( Integer ) end it "sends nil as the payload if the notification wasn't given one" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN knees' ) conn = described_class.connect( @conninfo ) conn.exec( %Q{NOTIFY knees} ) conn.finish payload = :notnil @conn.wait_for_notify( nil ) do |*args| payload = args[ 2 ] end @conn.exec( 'UNLISTEN knees' ) payload.should be_nil() end it "calls the block supplied to wait_for_notify with the notify payload if it accepts " + "two arguments" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN knees' ) conn = described_class.connect( @conninfo ) conn.exec( %Q{NOTIFY knees, 'skirt and boots'} ) conn.finish event, pid, msg = nil @conn.wait_for_notify( 10 ) do |arg1, arg2| event, pid, msg = arg1, arg2 end @conn.exec( 'UNLISTEN knees' ) event.should == 'knees' pid.should be_a_kind_of( Integer ) msg.should be_nil() end it "calls the block supplied to wait_for_notify with the notify payload if it " + "doesn't accept arguments" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN knees' ) conn = described_class.connect( @conninfo ) conn.exec( %Q{NOTIFY knees, 'skirt and boots'} ) conn.finish notification_received = false @conn.wait_for_notify( 10 ) do notification_received = true end @conn.exec( 'UNLISTEN knees' ) notification_received.should be_true() end it "calls the block supplied to wait_for_notify with the notify payload if it accepts " + "three arguments" do @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN knees' ) conn = described_class.connect( @conninfo ) conn.exec( %Q{NOTIFY knees, 'skirt and boots'} ) conn.finish event, pid, msg = nil @conn.wait_for_notify( 10 ) do |arg1, arg2, arg3| event, pid, msg = arg1, arg2, arg3 end @conn.exec( 'UNLISTEN knees' ) event.should == 'knees' pid.should be_a_kind_of( Integer ) msg.should == 'skirt and boots' end end context "under PostgreSQL 9.1 client library", :postgresql_91, :without_transaction do it "pings successfully with connection string" do ping = described_class.ping(@conninfo) ping.should == PG::PQPING_OK end it "pings using 7 arguments converted to strings" do ping = described_class.ping('localhost', @port, nil, nil, :test, nil, nil) ping.should == PG::PQPING_OK end it "pings using a hash of connection parameters" do ping = described_class.ping( :host => 'localhost', :port => @port, :dbname => :test) ping.should == PG::PQPING_OK end it "returns correct response when ping connection cannot be established" do ping = described_class.ping( :host => 'localhost', :port => 9999, :dbname => :test) ping.should == PG::PQPING_NO_RESPONSE end it "returns correct response when ping connection arguments are wrong" do ping = described_class.ping('localhost', 'localhost', nil, nil, :test, nil, nil) ping.should == PG::PQPING_NO_ATTEMPT end end context "under PostgreSQL 9.2 client library", :postgresql_92 do describe "set_single_row_mode" do it "raises an error when called at the wrong time" do expect { @conn.set_single_row_mode }.to raise_error(PG::Error) end it "should work in single row mode" do @conn.send_query( "SELECT generate_series(1,10)" ) @conn.set_single_row_mode results = [] loop do @conn.block res = @conn.get_result or break results << res end results.length.should == 11 results[0..-2].each do |res| res.result_status.should == PG::PGRES_SINGLE_TUPLE values = res.field_values('generate_series') values.length.should == 1 values.first.to_i.should > 0 end results.last.result_status.should == PG::PGRES_TUPLES_OK results.last.ntuples.should == 0 end it "should receive rows before entire query is finished" do @conn.send_query( "SELECT generate_series(0,999), NULL UNION ALL SELECT 1000, pg_sleep(1);" ) @conn.set_single_row_mode start_time = Time.now first_row_time = nil loop do res = @conn.get_result or break res.check first_row_time = Time.now unless first_row_time end (Time.now - start_time).should >= 1.0 (first_row_time - start_time).should < 1.0 end it "should receive rows before entire query fails" do @conn.exec( "CREATE FUNCTION errfunc() RETURNS int AS $$ BEGIN RAISE 'test-error'; END; $$ LANGUAGE plpgsql;" ) @conn.send_query( "SELECT generate_series(0,999), NULL UNION ALL SELECT 1000, errfunc();" ) @conn.set_single_row_mode first_result = nil expect do loop do res = @conn.get_result or break res.check first_result ||= res end end.to raise_error(PG::Error) first_result.kind_of?(PG::Result).should be_true first_result.result_status.should == PG::PGRES_SINGLE_TUPLE end end end context "multinationalization support", :ruby_19 do describe "rubyforge #22925: m17n support" do it "should return results in the same encoding as the client (iso-8859-1)" do out_string = nil @conn.transaction do |conn| conn.internal_encoding = 'iso8859-1' res = conn.exec("VALUES ('fantasia')", [], 0) out_string = res[0]['column1'] end out_string.should == 'fantasia' out_string.encoding.should == Encoding::ISO8859_1 end it "should return results in the same encoding as the client (utf-8)" do out_string = nil @conn.transaction do |conn| conn.internal_encoding = 'utf-8' res = conn.exec("VALUES ('世界線航跡蔵')", [], 0) out_string = res[0]['column1'] end out_string.encoding.should == Encoding::UTF_8 out_string.should == '世界線航跡蔵' end it "should return results in the same encoding as the client (EUC-JP)" do out_string = nil @conn.transaction do |conn| conn.internal_encoding = 'EUC-JP' stmt = "VALUES ('世界線航跡蔵')".encode('EUC-JP') res = conn.exec(stmt, [], 0) out_string = res[0]['column1'] end out_string.should == '世界線航跡蔵'.encode('EUC-JP') out_string.encoding.should == Encoding::EUC_JP end it "returns the results in the correct encoding even if the client_encoding has " + "changed since the results were fetched" do out_string = nil @conn.transaction do |conn| conn.internal_encoding = 'EUC-JP' stmt = "VALUES ('世界線航跡蔵')".encode('EUC-JP') res = conn.exec(stmt, [], 0) conn.internal_encoding = 'utf-8' out_string = res[0]['column1'] end out_string.should == '世界線航跡蔵'.encode('EUC-JP') out_string.encoding.should == Encoding::EUC_JP end it "the connection should return ASCII-8BIT when it's set to SQL_ASCII" do @conn.exec "SET client_encoding TO SQL_ASCII" @conn.internal_encoding.should == Encoding::ASCII_8BIT end it "works around the unsupported JOHAB encoding by returning stuff in 'ASCII_8BIT'" do pending "figuring out how to create a string in the JOHAB encoding" do out_string = nil @conn.transaction do |conn| conn.exec( "set client_encoding = 'JOHAB';" ) stmt = "VALUES ('foo')".encode('JOHAB') res = conn.exec( stmt, [], 0 ) out_string = res[0]['column1'] end out_string.should == 'foo'.encode( Encoding::ASCII_8BIT ) out_string.encoding.should == Encoding::ASCII_8BIT end end it "uses the client encoding for escaped string" do original = "string to\0 escape".force_encoding( "iso8859-1" ) @conn.set_client_encoding( "euc_jp" ) escaped = @conn.escape( original ) escaped.encoding.should == Encoding::EUC_JP escaped.should == "string to" end it "uses the client encoding for escaped literal", :postgresql_90 do original = "string to\0 escape".force_encoding( "iso8859-1" ) @conn.set_client_encoding( "euc_jp" ) escaped = @conn.escape_literal( original ) escaped.encoding.should == Encoding::EUC_JP escaped.should == "'string to'" end it "uses the client encoding for escaped identifier", :postgresql_90 do original = "string to\0 escape".force_encoding( "iso8859-1" ) @conn.set_client_encoding( "euc_jp" ) escaped = @conn.escape_identifier( original ) escaped.encoding.should == Encoding::EUC_JP escaped.should == "\"string to\"" end it "uses the client encoding for quote_ident" do original = "string to\0 escape".force_encoding( "iso8859-1" ) @conn.set_client_encoding( "euc_jp" ) escaped = @conn.quote_ident( original ) escaped.encoding.should == Encoding::EUC_JP escaped.should == "\"string to\"" end it "uses the previous string encoding for escaped string" do original = "string to\0 escape".force_encoding( "iso8859-1" ) @conn.set_client_encoding( "euc_jp" ) escaped = described_class.escape( original ) escaped.encoding.should == Encoding::ISO8859_1 escaped.should == "string to" end it "uses the previous string encoding for quote_ident" do original = "string to\0 escape".force_encoding( "iso8859-1" ) @conn.set_client_encoding( "euc_jp" ) escaped = described_class.quote_ident( original ) escaped.encoding.should == Encoding::ISO8859_1 escaped.should == "\"string to\"" end end describe "Ruby 1.9.x default_internal encoding" do it "honors the Encoding.default_internal if it's set and the synchronous interface is used" do @conn.transaction do |txn_conn| txn_conn.internal_encoding = Encoding::ISO8859_1 txn_conn.exec( "CREATE TABLE defaultinternaltest ( foo text )" ) txn_conn.exec( "INSERT INTO defaultinternaltest VALUES ('Grün und Weiß')" ) end begin prev_encoding = Encoding.default_internal Encoding.default_internal = Encoding::UTF_8 conn = PG.connect( @conninfo ) conn.internal_encoding.should == Encoding::UTF_8 res = conn.exec( "SELECT foo FROM defaultinternaltest" ) res[0]['foo'].encoding.should == Encoding::UTF_8 ensure conn.finish if conn Encoding.default_internal = prev_encoding end end it "allows users of the async interface to set the client_encoding to the default_internal" do begin prev_encoding = Encoding.default_internal Encoding.default_internal = Encoding::KOI8_R @conn.set_default_encoding @conn.internal_encoding.should == Encoding::KOI8_R ensure Encoding.default_internal = prev_encoding end end end it "encodes exception messages with the connection's encoding (#96)", :without_transaction do # Use a new connection so the client_encoding isn't set outside of this example conn = PG.connect( @conninfo ) conn.client_encoding = 'iso-8859-15' conn.transaction do conn.exec "CREATE TABLE foo (bar TEXT)" begin query = "INSERT INTO foo VALUES ('Côte d'Ivoire')".encode( 'iso-8859-15' ) conn.exec( query ) rescue => err err.message.encoding.should == Encoding::ISO8859_15 else fail "No exception raised?!" end end conn.finish if conn end it "receives properly encoded messages in the notice callbacks", :postgresql_90 do [:receiver, :processor].each do |kind| notices = [] @conn.internal_encoding = 'utf-8' if kind == :processor @conn.set_notice_processor do |msg| notices << msg end else @conn.set_notice_receiver do |result| notices << result.error_message end end 3.times do @conn.exec "do $$ BEGIN RAISE NOTICE '世界線航跡蔵'; END; $$ LANGUAGE plpgsql;" end notices.length.should == 3 notices.each do |notice| notice.should =~ /^NOTICE:.*世界線航跡蔵/ notice.encoding.should == Encoding::UTF_8 end @conn.set_notice_receiver @conn.set_notice_processor end end it "receives properly encoded text from wait_for_notify", :postgresql_90 do @conn.internal_encoding = 'utf-8' @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN "Möhre"' ) @conn.exec( %Q{NOTIFY "Möhre", '世界線航跡蔵'} ) event, pid, msg = nil @conn.wait_for_notify( 10 ) do |*args| event, pid, msg = *args end @conn.exec( 'UNLISTEN "Möhre"' ) event.should == "Möhre" event.encoding.should == Encoding::UTF_8 msg.should == '世界線航跡蔵' msg.encoding.should == Encoding::UTF_8 end it "returns properly encoded text from notifies", :postgresql_90 do @conn.internal_encoding = 'utf-8' @conn.exec( 'ROLLBACK' ) @conn.exec( 'LISTEN "Möhre"' ) @conn.exec( %Q{NOTIFY "Möhre", '世界線航跡蔵'} ) @conn.exec( 'UNLISTEN "Möhre"' ) notification = @conn.notifies notification[:relname].should == "Möhre" notification[:relname].encoding.should == Encoding::UTF_8 notification[:extra].should == '世界線航跡蔵' notification[:extra].encoding.should == Encoding::UTF_8 notification[:be_pid].should > 0 end end context "OS thread support", :ruby_19 do it "Connection#exec shouldn't block a second thread" do t = Thread.new do @conn.exec( "select pg_sleep(1)" ) end sleep 0.5 t.should be_alive() t.join end it "Connection.new shouldn't block a second thread" do serv = nil t = Thread.new do serv = TCPServer.new( '127.0.0.1', 54320 ) expect { described_class.new( '127.0.0.1', 54320, "", "", "me", "xxxx", "somedb" ) }.to raise_error(PG::ConnectionBad, /server closed the connection unexpectedly|reset by peer/) end sleep 0.5 t.should be_alive() serv.close t.join end end end