SEQUEL_ADAPTER_TEST = :postgres
require File.join(File.dirname(File.expand_path(__FILE__)), 'spec_helper.rb')
def DB.sqls
(@sqls ||= [])
end
logger = Object.new
def logger.method_missing(m, msg)
DB.sqls << msg
end
DB.loggers << logger
DB.extension :pg_array, :pg_hstore, :pg_range, :pg_interval, :pg_row, :pg_inet, :pg_json, :pg_enum
describe "PostgreSQL", '#create_table' do
before do
@db = DB
DB.sqls.clear
end
after do
@db.drop_table?(:tmp_dolls, :unlogged_dolls)
end
it "should create a temporary table" do
@db.create_table(:tmp_dolls, :temp => true){text :name}
check_sqls do
@db.sqls.must_equal ['CREATE TEMPORARY TABLE "tmp_dolls" ("name" text)']
end
end
it "temporary table should support :on_commit option" do
@db.drop_table?(:some_table)
@db.transaction do
@db.create_table(:some_table, :temp => true, :on_commit => :drop){text :name}
end
@db.table_exists?(:some_table).must_equal false
@db.transaction do
@db.create_table(:some_table, :temp => true, :on_commit => :delete_rows){text :name}
@db[:some_table].insert('a')
end
@db.table_exists?(:some_table).must_equal true
@db[:some_table].empty?.must_equal true
@db.drop_table(:some_table)
@db.transaction do
@db.create_table(:some_table, :temp => true, :on_commit => :preserve_rows){text :name}
@db[:some_table].insert('a')
end
@db.table_exists?(:some_table).must_equal true
@db[:some_table].count.must_equal 1
@db.drop_table(:some_table)
end
it "temporary table should accept :on_commit with :as option" do
@db.drop_table?(:some_table)
@db.transaction do
@db.create_table(:some_table, :temp => true, :on_commit => :drop, :as => 'select 1')
end
@db.table_exists?(:some_table).must_equal false
end
it ":on_commit should raise error if not used on a temporary table" do
proc{@db.create_table(:some_table, :on_commit => :drop)}.must_raise(Sequel::Error)
end
it ":on_commit should raise error if given unsupported value" do
proc{@db.create_table(:some_table, :temp => true, :on_commit => :unsupported){text :name}}.must_raise(Sequel::Error)
end
it "should create an unlogged table" do
@db.create_table(:unlogged_dolls, :unlogged => true){text :name}
check_sqls do
@db.sqls.must_equal ['CREATE UNLOGGED TABLE "unlogged_dolls" ("name" text)']
end
end
it "should create a table inheriting from another table" do
@db.create_table(:unlogged_dolls){text :name}
@db.create_table(:tmp_dolls, :inherits=>:unlogged_dolls){}
@db[:tmp_dolls].insert('a')
@db[:unlogged_dolls].all.must_equal [{:name=>'a'}]
end
it "should create a table inheriting from multiple tables" do
begin
@db.create_table(:unlogged_dolls){text :name}
@db.create_table(:tmp_dolls){text :bar}
@db.create_table!(:items, :inherits=>[:unlogged_dolls, :tmp_dolls]){text :foo}
@db[:items].insert(:name=>'a', :bar=>'b', :foo=>'c')
@db[:unlogged_dolls].all.must_equal [{:name=>'a'}]
@db[:tmp_dolls].all.must_equal [{:bar=>'b'}]
@db[:items].all.must_equal [{:name=>'a', :bar=>'b', :foo=>'c'}]
ensure
@db.drop_table?(:items)
end
end
it "should not allow to pass both :temp and :unlogged" do
proc do
@db.create_table(:temp_unlogged_dolls, :temp => true, :unlogged => true){text :name}
end.must_raise(Sequel::Error, "can't provide both :temp and :unlogged to create_table")
end
it "should support pg_loose_count extension" do
@db.extension :pg_loose_count
@db.create_table(:tmp_dolls){text :name}
@db.loose_count(:tmp_dolls).must_be_kind_of(Integer)
@db.loose_count(:tmp_dolls).must_equal 0
@db.loose_count(:public__tmp_dolls).must_equal 0
@db[:tmp_dolls].insert('a')
@db << 'VACUUM ANALYZE tmp_dolls'
@db.loose_count(:tmp_dolls).must_equal 1
@db.loose_count(:public__tmp_dolls).must_equal 1
end
end
describe "PostgreSQL views" do
before do
@db = DB
@db.drop_table?(:items, :cascade=>true)
@db.create_table(:items){Integer :number}
@db[:items].insert(10)
@db[:items].insert(20)
end
after do
@opts ||={}
@db.drop_view(:items_view, @opts.merge(:if_exists=>true, :cascade=>true)) rescue nil
@db.drop_table?(:items)
end
it "should support temporary views" do
@db.create_view(:items_view, @db[:items].where(:number=>10), :temp=>true)
@db[:items_view].map(:number).must_equal [10]
@db.create_or_replace_view(:items_view, @db[:items].where(:number=>20), :temp=>true)
@db[:items_view].map(:number).must_equal [20]
end
it "should support recursive views" do
@db.create_view(:items_view, @db[:items].where(:number=>10).union(@db[:items, :items_view].where(Sequel.-(:number, 5)=>:n).select(:number), :all=>true, :from_self=>false), :recursive=>[:n])
@db[:items_view].select_order_map(:n).must_equal [10]
@db[:items].insert(15)
@db[:items_view].select_order_map(:n).must_equal [10, 15, 20]
end if DB.server_version >= 90300
it "should support materialized views" do
@opts = {:materialized=>true}
@db.create_view(:items_view, @db[:items].where{number >= 10}, @opts)
@db[:items_view].select_order_map(:number).must_equal [10, 20]
@db[:items].insert(15)
@db[:items_view].select_order_map(:number).must_equal [10, 20]
@db.refresh_view(:items_view)
@db[:items_view].select_order_map(:number).must_equal [10, 15, 20]
end if DB.server_version >= 90300
it "should support refreshing materialized views concurrently" do
@opts = {:materialized=>true}
@db.create_view(:items_view, @db[:items].where{number >= 10}, @opts)
@db.refresh_view(:items_view)
proc{@db.refresh_view(:items_view, :concurrently=>true)}.must_raise(Sequel::DatabaseError)
@db.add_index :items_view, :number, :unique=>true
@db.refresh_view(:items_view, :concurrently=>true)
end if DB.server_version >= 90400
it "should support :if_exists=>true for not raising an error if the view does not exist" do
@db.drop_view(:items_view, :if_exists=>true)
end
end
describe "A PostgreSQL database" do
before(:all) do
@db = DB
@db.create_table!(:public__testfk){primary_key :id; foreign_key :i, :public__testfk}
end
after(:all) do
@db.drop_table?(:public__testfk)
end
it "should provide the server version" do
@db.server_version.must_be :>, 70000
end
it "should create a dataset using the VALUES clause via #values" do
@db.values([[1, 2], [3, 4]]).map([:column1, :column2]).must_equal [[1, 2], [3, 4]]
end
it "should support ordering and limiting with #values" do
@db.values([[1, 2], [3, 4]]).reverse(:column2, :column1).limit(1).map([:column1, :column2]).must_equal [[3, 4]]
@db.values([[1, 2], [3, 4]]).reverse(:column2, :column1).offset(1).map([:column1, :column2]).must_equal [[1, 2]]
end
it "should support subqueries with #values" do
@db.values([[1, 2]]).from_self.cross_join(@db.values([[3, 4]]).as(:x, [:c1, :c2])).map([:column1, :column2, :c1, :c2]).must_equal [[1, 2, 3, 4]]
end
it "should respect the :read_only option per-savepoint" do
proc{@db.transaction{@db.transaction(:savepoint=>true, :read_only=>true){@db[:public__testfk].insert}}}.must_raise(Sequel::DatabaseError)
proc{@db.transaction(:auto_savepoint=>true, :read_only=>true){@db.transaction(:read_only=>false){@db[:public__testfk].insert}}}.must_raise(Sequel::DatabaseError)
@db[:public__testfk].delete
@db.transaction{@db[:public__testfk].insert; @db.transaction(:savepoint=>true, :read_only=>true){@db[:public__testfk].all;}}
@db.transaction{@db.transaction(:savepoint=>true, :read_only=>true){}; @db[:public__testfk].insert}
@db.transaction{@db[:public__testfk].all; @db.transaction(:savepoint=>true, :read_only=>true){@db[:public__testfk].all;}}
end
it "should support disable_insert_returning" do
ds = @db[:public__testfk].disable_insert_returning
ds.delete
ds.insert.must_equal nil
id = ds.max(:id)
ds.select_order_map([:id, :i]).must_equal [[id, nil]]
ds.insert(:i=>id).must_equal nil
ds.select_order_map([:id, :i]).must_equal [[id, nil], [id+1, id]]
ds.insert_select(:i=>ds.max(:id)).must_equal nil
ds.select_order_map([:id, :i]).must_equal [[id, nil], [id+1, id]]
c = Class.new(Sequel::Model(ds))
c.class_eval do
def before_create
self.id = model.max(:id)+1
super
end
end
c.create(:i=>id+1).must_equal c.load(:id=>id+2, :i=>id+1)
ds.select_order_map([:id, :i]).must_equal [[id, nil], [id+1, id], [id+2, id+1]]
ds.delete
end
it "should support functions with and without quoting" do
ds = @db[:public__testfk]
ds.delete
ds.insert
ds.get{sum(1)}.must_equal 1
ds.get{Sequel.function('pg_catalog.sum', 1)}.must_equal 1
ds.get{sum.function(1)}.must_equal 1
ds.get{pg_catalog__sum.function(1)}.must_equal 1
ds.delete
end
it "should support a :qualify option to tables and views" do
@db.tables(:qualify=>true).must_include(Sequel.qualify(:public, :testfk))
begin
@db.create_view(:testfkv, @db[:testfk])
@db.views(:qualify=>true).must_include(Sequel.qualify(:public, :testfkv))
ensure
@db.drop_view(:testfkv)
end
end
it "should not typecast the int2vector type incorrectly" do
@db.get(Sequel.cast('10 20', :int2vector)).wont_equal 10
end
cspecify "should not typecast the money type incorrectly", [:do] do
@db.get(Sequel.cast('10.01', :money)).wont_equal 0
end
it "should correctly parse the schema" do
@db.schema(:public__testfk, :reload=>true).map{|c,s| [c, s[:oid]]}.must_equal [[:id, 23], [:i, 23]]
end
it "should parse foreign keys for tables in a schema" do
@db.foreign_key_list(:public__testfk).must_equal [{:on_delete=>:no_action, :on_update=>:no_action, :columns=>[:i], :key=>[:id], :deferrable=>false, :table=>Sequel.qualify(:public, :testfk), :name=>:testfk_i_fkey}]
end
it "should return uuid fields as strings" do
@db.get(Sequel.cast('550e8400-e29b-41d4-a716-446655440000', :uuid)).must_equal '550e8400-e29b-41d4-a716-446655440000'
end
it "should handle inserts with placeholder literal string tables" do
ds = @db.from(Sequel.lit('?', :testfk))
ds.delete
ds.insert(:id=>1)
ds.select_map(:id).must_equal [1]
end
it "should have notice receiver receive notices" do
a = nil
Sequel.connect(DB.opts.merge(:notice_receiver=>proc{|r| a = r.result_error_message})){|db| db.do("BEGIN\nRAISE WARNING 'foo';\nEND;")}
a.must_equal "WARNING: foo\n"
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG && DB.server_version >= 90000
# These only test the SQL created, because a true test using file_fdw or postgres_fdw
# requires superuser permissions, and you should not be running the tests as a superuser.
it "should support creating and dropping foreign tables" do
DB.send(:create_table_sql, :t, DB.create_table_generator{Integer :a}, :foreign=>:f, :options=>{:o=>1}).must_equal 'CREATE FOREIGN TABLE "t" ("a" integer) SERVER "f" OPTIONS (o \'1\')'
DB.send(:drop_table_sql, :t, :foreign=>true).must_equal 'DROP FOREIGN TABLE "t"'
end
end
describe "A PostgreSQL database with domain types" do
before(:all) do
@db = DB
@db << "DROP DOMAIN IF EXISTS positive_number CASCADE"
@db << "CREATE DOMAIN positive_number AS numeric(10,2) CHECK (VALUE > 0)"
@db.create_table!(:testfk){positive_number :id, :primary_key=>true}
end
after(:all) do
@db.drop_table?(:testfk)
@db << "DROP DOMAIN positive_number"
end
it "should correctly parse the schema" do
sch = @db.schema(:testfk, :reload=>true)
sch.first.last.delete(:domain_oid).must_be_kind_of(Integer)
sch.first.last[:db_domain_type].must_equal 'positive_number'
end
end
describe "A PostgreSQL dataset" do
before(:all) do
@db = DB
@d = @db[:test]
@db.create_table! :test do
text :name
integer :value, :index => true
end
end
before do
@d.delete
@db.sqls.clear
end
after do
@db.drop_table?(:atest)
end
after(:all) do
@db.drop_table?(:test)
end
it "should quote columns and tables using double quotes if quoting identifiers" do
check_sqls do
@d.select(:name).sql.must_equal 'SELECT "name" FROM "test"'
@d.select(Sequel.lit('COUNT(*)')).sql.must_equal 'SELECT COUNT(*) FROM "test"'
@d.select(Sequel.function(:max, :value)).sql.must_equal 'SELECT max("value") FROM "test"'
@d.select(Sequel.function(:NOW)).sql.must_equal 'SELECT NOW() FROM "test"'
@d.select(Sequel.function(:max, :items__value)).sql.must_equal 'SELECT max("items"."value") FROM "test"'
@d.order(Sequel.desc(:name)).sql.must_equal 'SELECT * FROM "test" ORDER BY "name" DESC'
@d.select(Sequel.lit('test.name AS item_name')).sql.must_equal 'SELECT test.name AS item_name FROM "test"'
@d.select(Sequel.lit('"name"')).sql.must_equal 'SELECT "name" FROM "test"'
@d.select(Sequel.lit('max(test."name") AS "max_name"')).sql.must_equal 'SELECT max(test."name") AS "max_name" FROM "test"'
@d.insert_sql(:x => :y).must_match(/\AINSERT INTO "test" \("x"\) VALUES \("y"\)( RETURNING NULL)?\z/)
@d.select(Sequel.function(:test, :abc, 'hello')).sql.must_equal "SELECT test(\"abc\", 'hello') FROM \"test\""
@d.select(Sequel.function(:test, :abc__def, 'hello')).sql.must_equal "SELECT test(\"abc\".\"def\", 'hello') FROM \"test\""
@d.select(Sequel.function(:test, :abc__def, 'hello').as(:x2)).sql.must_equal "SELECT test(\"abc\".\"def\", 'hello') AS \"x2\" FROM \"test\""
@d.insert_sql(:value => 333).must_match(/\AINSERT INTO "test" \("value"\) VALUES \(333\)( RETURNING NULL)?\z/)
end
end
it "should quote fields correctly when reversing the order if quoting identifiers" do
check_sqls do
@d.reverse_order(:name).sql.must_equal 'SELECT * FROM "test" ORDER BY "name" DESC'
@d.reverse_order(Sequel.desc(:name)).sql.must_equal 'SELECT * FROM "test" ORDER BY "name" ASC'
@d.reverse_order(:name, Sequel.desc(:test)).sql.must_equal 'SELECT * FROM "test" ORDER BY "name" DESC, "test" ASC'
@d.reverse_order(Sequel.desc(:name), :test).sql.must_equal 'SELECT * FROM "test" ORDER BY "name" ASC, "test" DESC'
end
end
it "should support regexps" do
@d << {:name => 'abc', :value => 1}
@d << {:name => 'bcd', :value => 2}
@d.filter(:name => /bc/).count.must_equal 2
@d.filter(:name => /^bc/).count.must_equal 1
end
it "should support NULLS FIRST and NULLS LAST" do
@d << {:name => 'abc'}
@d << {:name => 'bcd'}
@d << {:name => 'bcd', :value => 2}
@d.order(Sequel.asc(:value, :nulls=>:first), :name).select_map(:name).must_equal %w[abc bcd bcd]
@d.order(Sequel.asc(:value, :nulls=>:last), :name).select_map(:name).must_equal %w[bcd abc bcd]
@d.order(Sequel.asc(:value, :nulls=>:first), :name).reverse.select_map(:name).must_equal %w[bcd bcd abc]
end
it "should support selecting from LATERAL functions" do
@d.from{[generate_series(1,3,1).as(:a), pow(:a, 2).lateral.as(:b)]}.select_map([:a, :b])== [[1, 1], [2, 4], [3, 9]]
end if DB.server_version >= 90300
it "should support ordered-set and hypothetical-set aggregate functions" do
@d.from{generate_series(1,3,1).as(:a)}.select{(a.sql_number % 2).as(:a)}.from_self.get{mode{}.within_group(:a)}.must_equal 1
end if DB.server_version >= 90400
it "should support filtered aggregate functions" do
@d.from{generate_series(1,3,1).as(:a)}.select{(a.sql_number % 2).as(:a)}.from_self.get{count(:a).filter(:a=>1)}.must_equal 2
end if DB.server_version >= 90400
it "should support functions with ordinality" do
@d.from{generate_series(1,10,3).with_ordinality}.select_map([:generate_series, :ordinality]).must_equal [[1, 1], [4, 2], [7, 3], [10, 4]]
end if DB.server_version >= 90400
it "#lock should lock tables and yield if a block is given" do
@d.lock('EXCLUSIVE'){@d.insert(:name=>'a')}
end
it "should support exclusion constraints when creating or altering tables" do
@db.create_table!(:atest){Integer :t; exclude [[Sequel.desc(:t, :nulls=>:last), '=']], :using=>:btree, :where=>proc{t > 0}}
@db[:atest].insert(1)
@db[:atest].insert(2)
proc{@db[:atest].insert(2)}.must_raise(Sequel::Postgres::ExclusionConstraintViolation)
@db.create_table!(:atest){Integer :t}
@db.alter_table(:atest){add_exclusion_constraint [[:t, '=']], :using=>:btree, :name=>'atest_ex'}
@db[:atest].insert(1)
@db[:atest].insert(2)
proc{@db[:atest].insert(2)}.must_raise(Sequel::Postgres::ExclusionConstraintViolation)
@db.alter_table(:atest){drop_constraint 'atest_ex'}
end if DB.server_version >= 90000
it "should support deferrable exclusion constraints" do
@db.create_table!(:atest){Integer :t; exclude [[Sequel.desc(:t, :nulls=>:last), '=']], :using=>:btree, :where=>proc{t > 0}, :deferrable => true}
proc do
@db.transaction do
@db[:atest].insert(2)
@db[:atest].insert(2)
end
end.must_raise(Sequel::Postgres::ExclusionConstraintViolation)
end if DB.server_version >= 90000
it "should support Database#error_info for getting info hash on the given error" do
@db.create_table!(:atest){Integer :t; Integer :t2, :null=>false, :default=>1; constraint :f, :t=>0}
begin
@db[:atest].insert(1)
rescue => e
end
e.wont_equal nil
info = @db.error_info(e)
info[:schema].must_equal 'public'
info[:table].must_equal 'atest'
info[:constraint].must_equal 'f'
info[:column].must_equal nil
info[:type].must_equal nil
begin
@db[:atest].insert(0, nil)
rescue => e
end
e.wont_equal nil
info = @db.error_info(e.wrapped_exception)
info[:schema].must_equal 'public'
info[:table].must_equal 'atest'
info[:constraint].must_equal nil
info[:column].must_equal 't2'
info[:type].must_equal nil
end if DB.server_version >= 90300 && DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG && Object.const_defined?(:PG) && ::PG.const_defined?(:Constants) && ::PG::Constants.const_defined?(:PG_DIAG_SCHEMA_NAME)
it "should support Database#do for executing anonymous code blocks" do
@db.drop_table?(:btest)
@db.do "BEGIN EXECUTE 'CREATE TABLE btest (a INTEGER)'; EXECUTE 'INSERT INTO btest VALUES (1)'; END"
@db[:btest].select_map(:a).must_equal [1]
@db.do "BEGIN EXECUTE 'DROP TABLE btest; CREATE TABLE atest (a INTEGER)'; EXECUTE 'INSERT INTO atest VALUES (1)'; END", :language=>:plpgsql
@db[:atest].select_map(:a).must_equal [1]
end if DB.server_version >= 90000
it "should support adding foreign key constarints that are not yet valid, and validating them later" do
@db.create_table!(:atest){primary_key :id; Integer :fk}
@db[:atest].insert(1, 5)
@db.alter_table(:atest){add_foreign_key [:fk], :atest, :not_valid=>true, :name=>:atest_fk}
@db[:atest].insert(2, 1)
proc{@db[:atest].insert(3, 4)}.must_raise(Sequel::ForeignKeyConstraintViolation)
proc{@db.alter_table(:atest){validate_constraint :atest_fk}}.must_raise(Sequel::ForeignKeyConstraintViolation)
@db[:atest].where(:id=>1).update(:fk=>2)
@db.alter_table(:atest){validate_constraint :atest_fk}
@db.alter_table(:atest){validate_constraint :atest_fk}
end if DB.server_version >= 90200
it "should support adding check constarints that are not yet valid, and validating them later" do
@db.create_table!(:atest){Integer :a}
@db[:atest].insert(5)
@db.alter_table(:atest){add_constraint({:name=>:atest_check, :not_valid=>true}){a >= 10}}
@db[:atest].insert(10)
proc{@db[:atest].insert(6)}.must_raise(Sequel::CheckConstraintViolation)
proc{@db.alter_table(:atest){validate_constraint :atest_check}}.must_raise(Sequel::CheckConstraintViolation, Sequel::DatabaseError)
@db[:atest].where{a < 10}.update(:a=>Sequel.+(:a, 10))
@db.alter_table(:atest){validate_constraint :atest_check}
@db.alter_table(:atest){validate_constraint :atest_check}
end if DB.server_version >= 90200
it "should support :using when altering a column's type" do
@db.create_table!(:atest){Integer :t}
@db[:atest].insert(1262304000)
@db.alter_table(:atest){set_column_type :t, Time, :using=>Sequel.cast('epoch', Time) + Sequel.cast('1 second', :interval) * :t}
@db[:atest].get(Sequel.extract(:year, :t)).must_equal 2010
end
it "should support :using with a string when altering a column's type" do
@db.create_table!(:atest){Integer :t}
@db[:atest].insert(1262304000)
@db.alter_table(:atest){set_column_type :t, Time, :using=>"'epoch'::timestamp + '1 second'::interval * t"}
@db[:atest].get(Sequel.extract(:year, :t)).must_equal 2010
end
it "should be able to parse the default value for an interval type" do
@db.create_table!(:atest){interval :t, :default=>'1 week'}
@db.schema(:atest).first.last[:ruby_default].must_equal '7 days'
end
it "should have #transaction support various types of synchronous options" do
@db.transaction(:synchronous=>:on){}
@db.transaction(:synchronous=>true){}
@db.transaction(:synchronous=>:off){}
@db.transaction(:synchronous=>false){}
@db.sqls.grep(/synchronous/).must_equal ["SET LOCAL synchronous_commit = on", "SET LOCAL synchronous_commit = on", "SET LOCAL synchronous_commit = off", "SET LOCAL synchronous_commit = off"]
@db.sqls.clear
@db.transaction(:synchronous=>nil){}
check_sqls do
@db.sqls.must_equal ['BEGIN', 'COMMIT']
end
if @db.server_version >= 90100
@db.sqls.clear
@db.transaction(:synchronous=>:local){}
check_sqls do
@db.sqls.grep(/synchronous/).must_equal ["SET LOCAL synchronous_commit = local"]
end
if @db.server_version >= 90200
@db.sqls.clear
@db.transaction(:synchronous=>:remote_write){}
check_sqls do
@db.sqls.grep(/synchronous/).must_equal ["SET LOCAL synchronous_commit = remote_write"]
end
end
end
end
it "should have #transaction support read only transactions" do
@db.transaction(:read_only=>true){}
@db.transaction(:read_only=>false){}
@db.transaction(:isolation=>:serializable, :read_only=>true){}
@db.transaction(:isolation=>:serializable, :read_only=>false){}
@db.sqls.grep(/READ/).must_equal ["SET TRANSACTION READ ONLY", "SET TRANSACTION READ WRITE", "SET TRANSACTION ISOLATION LEVEL SERIALIZABLE READ ONLY", "SET TRANSACTION ISOLATION LEVEL SERIALIZABLE READ WRITE"]
end
it "should have #transaction support deferrable transactions" do
@db.transaction(:deferrable=>true){}
@db.transaction(:deferrable=>false){}
@db.transaction(:deferrable=>true, :read_only=>true){}
@db.transaction(:deferrable=>false, :read_only=>false){}
@db.transaction(:isolation=>:serializable, :deferrable=>true, :read_only=>true){}
@db.transaction(:isolation=>:serializable, :deferrable=>false, :read_only=>false){}
@db.sqls.grep(/DEF/).must_equal ["SET TRANSACTION DEFERRABLE", "SET TRANSACTION NOT DEFERRABLE", "SET TRANSACTION READ ONLY DEFERRABLE", "SET TRANSACTION READ WRITE NOT DEFERRABLE", "SET TRANSACTION ISOLATION LEVEL SERIALIZABLE READ ONLY DEFERRABLE", "SET TRANSACTION ISOLATION LEVEL SERIALIZABLE READ WRITE NOT DEFERRABLE"]
end if DB.server_version >= 90100
it "should support creating indexes concurrently" do
@db.add_index :test, [:name, :value], :concurrently=>true
check_sqls do
@db.sqls.must_equal ['CREATE INDEX CONCURRENTLY "test_name_value_index" ON "test" ("name", "value")']
end
end
it "should support dropping indexes only if they already exist" do
@db.add_index :test, [:name, :value], :name=>'tnv1'
@db.sqls.clear
@db.drop_index :test, [:name, :value], :if_exists=>true, :name=>'tnv1'
check_sqls do
@db.sqls.must_equal ['DROP INDEX IF EXISTS "tnv1"']
end
end
it "should support CASCADE when dropping indexes" do
@db.add_index :test, [:name, :value], :name=>'tnv2'
@db.sqls.clear
@db.drop_index :test, [:name, :value], :cascade=>true, :name=>'tnv2'
check_sqls do
@db.sqls.must_equal ['DROP INDEX "tnv2" CASCADE']
end
end
it "should support dropping indexes concurrently" do
@db.add_index :test, [:name, :value], :name=>'tnv2'
@db.sqls.clear
@db.drop_index :test, [:name, :value], :concurrently=>true, :name=>'tnv2'
check_sqls do
@db.sqls.must_equal ['DROP INDEX CONCURRENTLY "tnv2"']
end
end if DB.server_version >= 90200
it "#lock should lock table if inside a transaction" do
@db.transaction{@d.lock('EXCLUSIVE'); @d.insert(:name=>'a')}
end
it "#lock should return nil" do
@d.lock('EXCLUSIVE'){@d.insert(:name=>'a')}.must_equal nil
@db.transaction{@d.lock('EXCLUSIVE').must_equal nil; @d.insert(:name=>'a')}
end
it "should raise an error if attempting to update a joined dataset with a single FROM table" do
proc{@db[:test].join(:test, [:name]).update(:name=>'a')}.must_raise(Sequel::Error, 'Need multiple FROM tables if updating/deleting a dataset with JOINs')
end
it "should truncate with options" do
@d << { :name => 'abc', :value => 1}
@d.count.must_equal 1
@d.truncate(:cascade => true)
@d.count.must_equal 0
if @d.db.server_version > 80400
@d << { :name => 'abc', :value => 1}
@d.truncate(:cascade => true, :only=>true, :restart=>true)
@d.count.must_equal 0
end
end
it "should truncate multiple tables at once" do
tables = [:test, :test]
tables.each{|t| @d.from(t).insert}
@d.from(:test, :test).truncate
tables.each{|t| @d.from(t).count.must_equal 0}
end
end
describe "Dataset#distinct" do
before do
@db = DB
@db.create_table!(:a) do
Integer :a
Integer :b
end
@ds = @db[:a]
end
after do
@db.drop_table?(:a)
end
it "#distinct with arguments should return results distinct on those arguments" do
@ds.insert(20, 10)
@ds.insert(30, 10)
@ds.order(:b, :a).distinct.map(:a).must_equal [20, 30]
@ds.order(:b, Sequel.desc(:a)).distinct.map(:a).must_equal [30, 20]
@ds.order(:b, :a).distinct(:b).map(:a).must_equal [20]
@ds.order(:b, Sequel.desc(:a)).distinct(:b).map(:a).must_equal [30]
end
end
if DB.pool.respond_to?(:max_size) and DB.pool.max_size > 1
describe "Dataset#for_update support" do
before do
@db = DB.create_table!(:items) do
primary_key :id
Integer :number
String :name
end
@ds = DB[:items]
end
after do
DB.drop_table?(:items)
DB.disconnect
end
it "should handle FOR UPDATE" do
@ds.insert(:number=>20)
c, t = nil, nil
q = Queue.new
DB.transaction do
@ds.for_update.first(:id=>1)
t = Thread.new do
DB.transaction do
q.push nil
@ds.filter(:id=>1).update(:name=>'Jim')
c = @ds.first(:id=>1)
q.push nil
end
end
q.pop
@ds.filter(:id=>1).update(:number=>30)
end
q.pop
t.join
c.must_equal(:id=>1, :number=>30, :name=>'Jim')
end
it "should handle FOR SHARE" do
@ds.insert(:number=>20)
c, t = nil
q = Queue.new
DB.transaction do
@ds.for_share.first(:id=>1)
t = Thread.new do
DB.transaction do
c = @ds.for_share.filter(:id=>1).first
q.push nil
end
end
q.pop
@ds.filter(:id=>1).update(:name=>'Jim')
c.must_equal(:id=>1, :number=>20, :name=>nil)
end
t.join
end
end
end
describe "A PostgreSQL dataset with a timestamp field" do
before(:all) do
@db = DB
@db.create_table! :test3 do
Date :date
DateTime :time
end
@d = @db[:test3]
end
before do
@d.delete
end
after do
@db.convert_infinite_timestamps = false if @db.adapter_scheme == :postgres
end
after(:all) do
@db.drop_table?(:test3)
end
cspecify "should store milliseconds in time fields for Time objects", [:do], [:swift] do
t = Time.now
@d << {:time=>t}
t2 = @d.get(:time)
@d.literal(t2).must_equal @d.literal(t)
t2.strftime('%Y-%m-%d %H:%M:%S').must_equal t.strftime('%Y-%m-%d %H:%M:%S')
(t2.is_a?(Time) ? t2.usec : t2.strftime('%N').to_i/1000).must_equal t.usec
end
cspecify "should store milliseconds in time fields for DateTime objects", [:do], [:swift] do
t = DateTime.now
@d << {:time=>t}
t2 = @d.get(:time)
@d.literal(t2).must_equal @d.literal(t)
t2.strftime('%Y-%m-%d %H:%M:%S').must_equal t.strftime('%Y-%m-%d %H:%M:%S')
(t2.is_a?(Time) ? t2.usec : t2.strftime('%N').to_i/1000).must_equal t.strftime('%N').to_i/1000
end
if DB.adapter_scheme == :postgres
it "should handle infinite timestamps if convert_infinite_timestamps is set" do
@d << {:time=>Sequel.cast('infinity', DateTime)}
@db.convert_infinite_timestamps = :nil
@db[:test3].get(:time).must_equal nil
@db.convert_infinite_timestamps = :string
@db[:test3].get(:time).must_equal 'infinity'
@db.convert_infinite_timestamps = :float
@db[:test3].get(:time).must_equal 1.0/0.0
@db.convert_infinite_timestamps = 'nil'
@db[:test3].get(:time).must_equal nil
@db.convert_infinite_timestamps = 'string'
@db[:test3].get(:time).must_equal 'infinity'
@db.convert_infinite_timestamps = 'float'
@db[:test3].get(:time).must_equal 1.0/0.0
@db.convert_infinite_timestamps = 't'
@db[:test3].get(:time).must_equal 1.0/0.0
if ((Time.parse('infinity'); nil) rescue true)
# Skip for loose time parsing (e.g. old rbx)
@db.convert_infinite_timestamps = 'f'
proc{@db[:test3].get(:time)}.must_raise ArgumentError, Sequel::InvalidValue
@db.convert_infinite_timestamps = nil
proc{@db[:test3].get(:time)}.must_raise ArgumentError, Sequel::InvalidValue
@db.convert_infinite_timestamps = false
proc{@db[:test3].get(:time)}.must_raise ArgumentError, Sequel::InvalidValue
end
@d.update(:time=>Sequel.cast('-infinity', DateTime))
@db.convert_infinite_timestamps = :nil
@db[:test3].get(:time).must_equal nil
@db.convert_infinite_timestamps = :string
@db[:test3].get(:time).must_equal '-infinity'
@db.convert_infinite_timestamps = :float
@db[:test3].get(:time).must_equal(-1.0/0.0)
end
it "should handle conversions from infinite strings/floats in models" do
c = Class.new(Sequel::Model(:test3))
@db.convert_infinite_timestamps = :float
c.new(:time=>'infinity').time.must_equal 'infinity'
c.new(:time=>'-infinity').time.must_equal '-infinity'
c.new(:time=>1.0/0.0).time.must_equal 1.0/0.0
c.new(:time=>-1.0/0.0).time.must_equal(-1.0/0.0)
end
it "should handle infinite dates if convert_infinite_timestamps is set" do
@d << {:date=>Sequel.cast('infinity', Date)}
@db.convert_infinite_timestamps = :nil
@db[:test3].get(:date).must_equal nil
@db.convert_infinite_timestamps = :string
@db[:test3].get(:date).must_equal 'infinity'
@db.convert_infinite_timestamps = :float
@db[:test3].get(:date).must_equal 1.0/0.0
@d.update(:date=>Sequel.cast('-infinity', :timestamp))
@db.convert_infinite_timestamps = :nil
@db[:test3].get(:date).must_equal nil
@db.convert_infinite_timestamps = :string
@db[:test3].get(:date).must_equal '-infinity'
@db.convert_infinite_timestamps = :float
@db[:test3].get(:date).must_equal(-1.0/0.0)
end
it "should handle conversions from infinite strings/floats in models" do
c = Class.new(Sequel::Model(:test3))
@db.convert_infinite_timestamps = :float
c.new(:date=>'infinity').date.must_equal 'infinity'
c.new(:date=>'-infinity').date.must_equal '-infinity'
c.new(:date=>1.0/0.0).date.must_equal 1.0/0.0
c.new(:date=>-1.0/0.0).date.must_equal(-1.0/0.0)
end
end
it "explain and analyze should not raise errors" do
@d = DB[:test3]
@d.explain
@d.analyze
end
it "#locks should be a dataset returning database locks " do
@db.locks.must_be_kind_of(Sequel::Dataset)
@db.locks.all.must_be_kind_of(Array)
end
end
describe "A PostgreSQL database" do
before do
@db = DB
@db.create_table! :test2 do
text :name
integer :value
end
end
after do
@db.drop_table?(:test2)
end
it "should support column operations" do
@db.create_table!(:test2){text :name; integer :value}
@db[:test2] << {}
@db[:test2].columns.must_equal [:name, :value]
@db.add_column :test2, :xyz, :text, :default => '000'
@db[:test2].columns.must_equal [:name, :value, :xyz]
@db[:test2] << {:name => 'mmm', :value => 111}
@db[:test2].first[:xyz].must_equal '000'
@db[:test2].columns.must_equal [:name, :value, :xyz]
@db.drop_column :test2, :xyz
@db[:test2].columns.must_equal [:name, :value]
@db[:test2].delete
@db.add_column :test2, :xyz, :text, :default => '000'
@db[:test2] << {:name => 'mmm', :value => 111, :xyz => 'qqqq'}
@db[:test2].columns.must_equal [:name, :value, :xyz]
@db.rename_column :test2, :xyz, :zyx
@db[:test2].columns.must_equal [:name, :value, :zyx]
@db[:test2].first[:zyx].must_equal 'qqqq'
@db.add_column :test2, :xyz, :float
@db[:test2].delete
@db[:test2] << {:name => 'mmm', :value => 111, :xyz => 56.78}
@db.set_column_type :test2, :xyz, :integer
@db[:test2].first[:xyz].must_equal 57
end
end
describe "A PostgreSQL database" do
before do
@db = DB
@db.drop_table?(:posts)
@db.sqls.clear
end
after do
@db.drop_table?(:posts)
end
it "should support resetting the primary key sequence" do
@db.create_table(:posts){primary_key :a}
@db[:posts].insert(:a=>20).must_equal 20
@db[:posts].insert.must_equal 1
@db[:posts].insert.must_equal 2
@db[:posts].insert(:a=>10).must_equal 10
@db.reset_primary_key_sequence(:posts).must_equal 21
@db[:posts].insert.must_equal 21
@db[:posts].order(:a).map(:a).must_equal [1, 2, 10, 20, 21]
end
it "should support specifying Integer/Bignum/Fixnum types in primary keys and have them be auto incrementing" do
@db.create_table(:posts){primary_key :a, :type=>Integer}
@db[:posts].insert.must_equal 1
@db[:posts].insert.must_equal 2
@db.create_table!(:posts){primary_key :a, :type=>Fixnum}
@db[:posts].insert.must_equal 1
@db[:posts].insert.must_equal 2
@db.create_table!(:posts){primary_key :a, :type=>Bignum}
@db[:posts].insert.must_equal 1
@db[:posts].insert.must_equal 2
end
it "should not raise an error if attempting to resetting the primary key sequence for a table without a primary key" do
@db.create_table(:posts){Integer :a}
@db.reset_primary_key_sequence(:posts).must_equal nil
end
it "should support opclass specification" do
@db.create_table(:posts){text :title; text :body; integer :user_id; index(:user_id, :opclass => :int4_ops, :type => :btree)}
check_sqls do
@db.sqls.must_equal [
'CREATE TABLE "posts" ("title" text, "body" text, "user_id" integer)',
'CREATE INDEX "posts_user_id_index" ON "posts" USING btree ("user_id" int4_ops)'
]
end
end
it "should support fulltext indexes and searching" do
@db.create_table(:posts){text :title; text :body; full_text_index [:title, :body]; full_text_index :title, :language => 'french', :index_type=>:gist}
@db[:posts].insert(:title=>'ruby rails', :body=>'yowsa')
@db[:posts].insert(:title=>'sequel', :body=>'ruby')
@db[:posts].insert(:title=>'ruby scooby', :body=>'x')
@db[:posts].full_text_search(:title, 'rails').all.must_equal [{:title=>'ruby rails', :body=>'yowsa'}]
@db[:posts].full_text_search([:title, :body], ['yowsa', 'rails']).all.must_equal [:title=>'ruby rails', :body=>'yowsa']
@db[:posts].full_text_search(:title, 'scooby', :language => 'french').all.must_equal [{:title=>'ruby scooby', :body=>'x'}]
@db[:posts].full_text_search(:title, :$n).call(:select, :n=>'rails').must_equal [{:title=>'ruby rails', :body=>'yowsa'}]
@db[:posts].full_text_search(:title, :$n).prepare(:select, :fts_select).call(:n=>'rails').must_equal [{:title=>'ruby rails', :body=>'yowsa'}]
@db[:posts].insert(:title=>'jruby rubinius ruby maglev mri iron')
@db[:posts].insert(:title=>'ruby jruby maglev mri rubinius iron')
@db[:posts].full_text_search(:title, 'rubinius ruby', :phrase=>true).select_order_map(:title).must_equal ['jruby rubinius ruby maglev mri iron']
@db[:posts].full_text_search(:title, 'jruby maglev', :phrase=>true).select_order_map(:title).must_equal ['ruby jruby maglev mri rubinius iron']
@db[:posts].full_text_search(:title, 'rubinius ruby', :plain=>true).select_order_map(:title).must_equal ['jruby rubinius ruby maglev mri iron', 'ruby jruby maglev mri rubinius iron']
@db[:posts].full_text_search(:title, 'jruby maglev', :plain=>true).select_order_map(:title).must_equal ['jruby rubinius ruby maglev mri iron', 'ruby jruby maglev mri rubinius iron']
@db[:posts].full_text_search(Sequel.function(:to_tsvector, 'simple', :title), 'rails', :tsvector=>true).all.must_equal [{:title=>'ruby rails', :body=>'yowsa'}]
@db[:posts].full_text_search(:title, Sequel.function(:to_tsquery, 'simple', 'rails'), :tsquery=>true).all.must_equal [{:title=>'ruby rails', :body=>'yowsa'}]
proc{@db[:posts].full_text_search(Sequel.function(:to_tsvector, 'simple', :title), 'rubinius ruby', :tsvector=>true, :phrase=>true)}.must_raise(Sequel::Error)
proc{@db[:posts].full_text_search(:title, Sequel.function(:to_tsquery, 'simple', 'rails'), :tsquery=>true, :phrase=>true)}.must_raise(Sequel::Error)
@db[:posts].delete
t1 = "bork " * 1000 + "ruby sequel"
t2 = "ruby sequel " * 1000
@db[:posts].insert(:title=>t1)
@db[:posts].insert(:title=>t2)
@db[:posts].full_text_search(:title, 'ruby & sequel', :rank=>true).select_map(:title).must_equal [t1, t2]
end
it "should support spatial indexes" do
@db.create_table(:posts){box :geom; spatial_index [:geom]}
check_sqls do
@db.sqls.must_equal [
'CREATE TABLE "posts" ("geom" box)',
'CREATE INDEX "posts_geom_index" ON "posts" USING gist ("geom")'
]
end
end
it "should support indexes with index type" do
@db.create_table(:posts){varchar :title, :size => 5; index :title, :type => 'hash'}
check_sqls do
@db.sqls.must_equal [
'CREATE TABLE "posts" ("title" varchar(5))',
'CREATE INDEX "posts_title_index" ON "posts" USING hash ("title")'
]
end
end
it "should support unique indexes with index type" do
@db.create_table(:posts){varchar :title, :size => 5; index :title, :type => 'btree', :unique => true}
check_sqls do
@db.sqls.must_equal [
'CREATE TABLE "posts" ("title" varchar(5))',
'CREATE UNIQUE INDEX "posts_title_index" ON "posts" USING btree ("title")'
]
end
end
it "should support partial indexes" do
@db.create_table(:posts){varchar :title, :size => 5; index :title, :where => {:title => '5'}}
check_sqls do
@db.sqls.must_equal [
'CREATE TABLE "posts" ("title" varchar(5))',
'CREATE INDEX "posts_title_index" ON "posts" ("title") WHERE ("title" = \'5\')'
]
end
end
it "should support identifiers for table names in indicies" do
@db.create_table(Sequel::SQL::Identifier.new(:posts)){varchar :title, :size => 5; index :title, :where => {:title => '5'}}
check_sqls do
@db.sqls.must_equal [
'CREATE TABLE "posts" ("title" varchar(5))',
'CREATE INDEX "posts_title_index" ON "posts" ("title") WHERE ("title" = \'5\')'
]
end
end
it "should support renaming tables" do
@db.create_table!(:posts1){primary_key :a}
@db.rename_table(:posts1, :posts)
end
end
describe "Postgres::Dataset#import" do
before do
@db = DB
@db.create_table!(:test){primary_key :x; Integer :y}
@db.sqls.clear
@ds = @db[:test]
end
after do
@db.drop_table?(:test)
end
it "#import should a single insert statement" do
@ds.import([:x, :y], [[1, 2], [3, 4]])
check_sqls do
@db.sqls.must_equal ['BEGIN', 'INSERT INTO "test" ("x", "y") VALUES (1, 2), (3, 4)', 'COMMIT']
end
@ds.all.must_equal [{:x=>1, :y=>2}, {:x=>3, :y=>4}]
end
it "#import should work correctly when returning primary keys" do
@ds.import([:x, :y], [[1, 2], [3, 4]], :return=>:primary_key).must_equal [1, 3]
@ds.all.must_equal [{:x=>1, :y=>2}, {:x=>3, :y=>4}]
end
it "#import should work correctly when returning primary keys with :slice option" do
@ds.import([:x, :y], [[1, 2], [3, 4]], :return=>:primary_key, :slice=>1).must_equal [1, 3]
@ds.all.must_equal [{:x=>1, :y=>2}, {:x=>3, :y=>4}]
end
it "#import should work correctly with an arbitrary returning value" do
@ds.returning(:y, :x).import([:x, :y], [[1, 2], [3, 4]]).must_equal [{:y=>2, :x=>1}, {:y=>4, :x=>3}]
@ds.all.must_equal [{:x=>1, :y=>2}, {:x=>3, :y=>4}]
end
end
describe "Postgres::Dataset#insert" do
before do
@db = DB
@db.create_table!(:test5){primary_key :xid; Integer :value}
@db.sqls.clear
@ds = @db[:test5]
end
after do
@db.drop_table?(:test5)
end
it "should work with static SQL" do
@ds.with_sql('INSERT INTO test5 (value) VALUES (10)').insert.must_equal nil
@db['INSERT INTO test5 (value) VALUES (20)'].insert.must_equal nil
@ds.all.must_equal [{:xid=>1, :value=>10}, {:xid=>2, :value=>20}]
end
it "should insert correctly if using a column array and a value array" do
@ds.insert([:value], [10]).must_equal 1
@ds.all.must_equal [{:xid=>1, :value=>10}]
end
it "should use INSERT RETURNING" do
@ds.insert(:value=>10).must_equal 1
check_sqls do
@db.sqls.last.must_equal 'INSERT INTO "test5" ("value") VALUES (10) RETURNING "xid"'
end
end
it "should have insert_select insert the record and return the inserted record" do
h = @ds.insert_select(:value=>10)
h[:value].must_equal 10
@ds.first(:xid=>h[:xid])[:value].must_equal 10
end
it "should have insert_select respect existing returning clause" do
h = @ds.returning(:value___v, :xid___x).insert_select(:value=>10)
h[:v].must_equal 10
@ds.first(:xid=>h[:x])[:value].must_equal 10
end
it "should have prepared insert_select respect existing returning clause" do
h = @ds.returning(:value___v, :xid___x).prepare(:insert_select, :insert_select, :value=>10).call
h[:v].must_equal 10
@ds.first(:xid=>h[:x])[:value].must_equal 10
end
it "should correctly return the inserted record's primary key value" do
value1 = 10
id1 = @ds.insert(:value=>value1)
@ds.first(:xid=>id1)[:value].must_equal value1
value2 = 20
id2 = @ds.insert(:value=>value2)
@ds.first(:xid=>id2)[:value].must_equal value2
end
it "should return nil if the table has no primary key" do
@db.create_table!(:test5){String :name; Integer :value}
@ds.delete
@ds.insert(:name=>'a').must_equal nil
end
end
describe "Postgres::Database schema qualified tables" do
before do
@db = DB
@db << "CREATE SCHEMA schema_test"
@db.instance_variable_set(:@primary_keys, {})
@db.instance_variable_set(:@primary_key_sequences, {})
end
after do
@db << "DROP SCHEMA schema_test CASCADE"
end
it "should be able to create, drop, select and insert into tables in a given schema" do
@db.create_table(:schema_test__schema_test){primary_key :i}
@db[:schema_test__schema_test].first.must_equal nil
@db[:schema_test__schema_test].insert(:i=>1).must_equal 1
@db[:schema_test__schema_test].first.must_equal(:i=>1)
@db.from(Sequel.lit('schema_test.schema_test')).first.must_equal(:i=>1)
@db.drop_table(:schema_test__schema_test)
@db.create_table(Sequel.qualify(:schema_test, :schema_test)){integer :i}
@db[:schema_test__schema_test].first.must_equal nil
@db.from(Sequel.lit('schema_test.schema_test')).first.must_equal nil
@db.drop_table(Sequel.qualify(:schema_test, :schema_test))
end
it "#tables should not include tables in a default non-public schema" do
@db.create_table(:schema_test__schema_test){integer :i}
@db.tables(:schema=>:schema_test).must_include(:schema_test)
@db.tables.wont_include(:pg_am)
@db.tables.wont_include(:domain_udt_usage)
end
it "#tables should return tables in the schema provided by the :schema argument" do
@db.create_table(:schema_test__schema_test){integer :i}
@db.tables(:schema=>:schema_test).must_equal [:schema_test]
end
it "#schema should not include columns from tables in a default non-public schema" do
@db.create_table(:schema_test__domains){integer :i}
sch = @db.schema(:schema_test__domains)
cs = sch.map{|x| x.first}
cs.must_include(:i)
cs.wont_include(:data_type)
end
it "#schema should only include columns from the table in the given :schema argument" do
@db.create_table!(:domains){integer :d}
@db.create_table(:schema_test__domains){integer :i}
sch = @db.schema(:domains, :schema=>:schema_test)
cs = sch.map{|x| x.first}
cs.must_include(:i)
cs.wont_include(:d)
@db.drop_table(:domains)
end
it "#schema should not include columns in tables from other domains by default" do
@db.create_table!(:public__domains){integer :d}
@db.create_table(:schema_test__domains){integer :i}
begin
@db.schema(:domains).map{|x| x.first}.must_equal [:d]
@db.schema(:schema_test__domains).map{|x| x.first}.must_equal [:i]
ensure
@db.drop_table?(:public__domains)
end
end
it "#table_exists? should see if the table is in a given schema" do
@db.create_table(:schema_test__schema_test){integer :i}
@db.table_exists?(:schema_test__schema_test).must_equal true
end
it "should be able to add and drop indexes in a schema" do
@db.create_table(:schema_test__schema_test){Integer :i, :index=>true}
@db.indexes(:schema_test__schema_test).keys.must_equal [:schema_test_schema_test_i_index]
@db.drop_index :schema_test__schema_test, :i
@db.indexes(:schema_test__schema_test).keys.must_equal []
end
it "should be able to get primary keys for tables in a given schema" do
@db.create_table(:schema_test__schema_test){primary_key :i}
@db.primary_key(:schema_test__schema_test).must_equal 'i'
end
it "should be able to get serial sequences for tables in a given schema" do
@db.create_table(:schema_test__schema_test){primary_key :i}
@db.primary_key_sequence(:schema_test__schema_test).must_equal '"schema_test"."schema_test_i_seq"'
end
it "should be able to get serial sequences for tables that have spaces in the name in a given schema" do
@db.create_table(:"schema_test__schema test"){primary_key :i}
@db.primary_key_sequence(:"schema_test__schema test").must_equal '"schema_test"."schema test_i_seq"'
end
it "should be able to get custom sequences for tables in a given schema" do
@db << "CREATE SEQUENCE schema_test.kseq"
@db.create_table(:schema_test__schema_test){integer :j; primary_key :k, :type=>:integer, :default=>Sequel.lit("nextval('schema_test.kseq'::regclass)")}
@db.primary_key_sequence(:schema_test__schema_test).must_equal '"schema_test".kseq'
end
it "should be able to get custom sequences for tables that have spaces in the name in a given schema" do
@db << "CREATE SEQUENCE schema_test.\"ks eq\""
@db.create_table(:"schema_test__schema test"){integer :j; primary_key :k, :type=>:integer, :default=>Sequel.lit("nextval('schema_test.\"ks eq\"'::regclass)")}
@db.primary_key_sequence(:"schema_test__schema test").must_equal '"schema_test"."ks eq"'
end
it "should handle schema introspection cases with tables with same name in multiple schemas" do
begin
@db.create_table(:schema_test__schema_test) do
primary_key :id
foreign_key :i, :schema_test__schema_test, :index=>{:name=>:schema_test_sti}
end
@db.create_table!(:public__schema_test) do
primary_key :id
foreign_key :j, :public__schema_test, :index=>{:name=>:public_test_sti}
end
h = @db.schema(:schema_test)
h.length.must_equal 2
h.last.first.must_equal :j
@db.indexes(:schema_test).must_equal(:public_test_sti=>{:unique=>false, :columns=>[:j], :deferrable=>nil})
@db.foreign_key_list(:schema_test).must_equal [{:on_update=>:no_action, :columns=>[:j], :deferrable=>false, :key=>[:id], :table=>:schema_test, :on_delete=>:no_action, :name=>:schema_test_j_fkey}]
ensure
@db.drop_table?(:public__schema_test)
end
end
end
describe "Postgres::Database schema qualified tables and eager graphing" do
before(:all) do
@db = DB
@db.run "DROP SCHEMA s CASCADE" rescue nil
@db.run "CREATE SCHEMA s"
@db.create_table(:s__bands){primary_key :id; String :name}
@db.create_table(:s__albums){primary_key :id; String :name; foreign_key :band_id, :s__bands}
@db.create_table(:s__tracks){primary_key :id; String :name; foreign_key :album_id, :s__albums}
@db.create_table(:s__members){primary_key :id; String :name; foreign_key :band_id, :s__bands}
@Band = Class.new(Sequel::Model(:s__bands))
@Album = Class.new(Sequel::Model(:s__albums))
@Track = Class.new(Sequel::Model(:s__tracks))
@Member = Class.new(Sequel::Model(:s__members))
def @Band.name; :Band; end
def @Album.name; :Album; end
def @Track.name; :Track; end
def @Member.name; :Member; end
@Band.one_to_many :albums, :class=>@Album, :order=>:name
@Band.one_to_many :members, :class=>@Member, :order=>:name
@Album.many_to_one :band, :class=>@Band, :order=>:name
@Album.one_to_many :tracks, :class=>@Track, :order=>:name
@Track.many_to_one :album, :class=>@Album, :order=>:name
@Member.many_to_one :band, :class=>@Band, :order=>:name
@Member.many_to_many :members, :class=>@Member, :join_table=>:s__bands, :right_key=>:id, :left_key=>:id, :left_primary_key=>:band_id, :right_primary_key=>:band_id, :order=>:name
@Band.many_to_many :tracks, :class=>@Track, :join_table=>:s__albums, :right_key=>:id, :right_primary_key=>:album_id, :order=>:name
@b1 = @Band.create(:name=>"BM")
@b2 = @Band.create(:name=>"J")
@a1 = @Album.create(:name=>"BM1", :band=>@b1)
@a2 = @Album.create(:name=>"BM2", :band=>@b1)
@a3 = @Album.create(:name=>"GH", :band=>@b2)
@a4 = @Album.create(:name=>"GHL", :band=>@b2)
@t1 = @Track.create(:name=>"BM1-1", :album=>@a1)
@t2 = @Track.create(:name=>"BM1-2", :album=>@a1)
@t3 = @Track.create(:name=>"BM2-1", :album=>@a2)
@t4 = @Track.create(:name=>"BM2-2", :album=>@a2)
@m1 = @Member.create(:name=>"NU", :band=>@b1)
@m2 = @Member.create(:name=>"TS", :band=>@b1)
@m3 = @Member.create(:name=>"NS", :band=>@b2)
@m4 = @Member.create(:name=>"JC", :band=>@b2)
end
after(:all) do
@db.run "DROP SCHEMA s CASCADE"
end
it "should return all eager graphs correctly" do
bands = @Band.order(:bands__name).eager_graph(:albums).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands = @Band.order(:bands__name).eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.albums.map{|y| y.tracks}}.must_equal [[[@t1, @t2], [@t3, @t4]], [[], []]]
bands = @Band.order(:bands__name).eager_graph({:albums=>:tracks}, :members).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.albums.map{|y| y.tracks}}.must_equal [[[@t1, @t2], [@t3, @t4]], [[], []]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
end
it "should have eager graphs work with previous joins" do
bands = @Band.order(:bands__name).select_all(:s__bands).join(:s__members, :band_id=>:id).from_self(:alias=>:bands0).eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.albums.map{|y| y.tracks}}.must_equal [[[@t1, @t2], [@t3, @t4]], [[], []]]
end
it "should have eager graphs work with joins with the same tables" do
bands = @Band.order(:bands__name).select_all(:s__bands).join(:s__members, :band_id=>:id).eager_graph({:albums=>:tracks}, :members).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.albums.map{|y| y.tracks}}.must_equal [[[@t1, @t2], [@t3, @t4]], [[], []]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
end
it "should have eager graphs work with self referential associations" do
bands = @Band.order(:bands__name).eager_graph(:tracks=>{:album=>:band}).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
bands.map{|x| x.tracks.map{|y| y.album}}.must_equal [[@a1, @a1, @a2, @a2], []]
bands.map{|x| x.tracks.map{|y| y.album.band}}.must_equal [[@b1, @b1, @b1, @b1], []]
members = @Member.order(:members__name).eager_graph(:members).all
members.must_equal [@m4, @m3, @m1, @m2]
members.map{|x| x.members}.must_equal [[@m4, @m3], [@m4, @m3], [@m1, @m2], [@m1, @m2]]
members = @Member.order(:members__name).eager_graph(:band, :members=>:band).all
members.must_equal [@m4, @m3, @m1, @m2]
members.map{|x| x.band}.must_equal [@b2, @b2, @b1, @b1]
members.map{|x| x.members}.must_equal [[@m4, @m3], [@m4, @m3], [@m1, @m2], [@m1, @m2]]
members.map{|x| x.members.map{|y| y.band}}.must_equal [[@b2, @b2], [@b2, @b2], [@b1, @b1], [@b1, @b1]]
end
it "should have eager graphs work with a from_self dataset" do
bands = @Band.order(:bands__name).from_self.eager_graph(:tracks=>{:album=>:band}).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
bands.map{|x| x.tracks.map{|y| y.album}}.must_equal [[@a1, @a1, @a2, @a2], []]
bands.map{|x| x.tracks.map{|y| y.album.band}}.must_equal [[@b1, @b1, @b1, @b1], []]
end
it "should have eager graphs work with different types of aliased from tables" do
bands = @Band.order(:tracks__name).from(:s__bands___tracks).eager_graph(:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
bands = @Band.order(:tracks__name).from(Sequel.expr(:s__bands).as(:tracks)).eager_graph(:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
bands = @Band.order(:tracks__name).from(Sequel.expr(:s__bands).as(Sequel.identifier(:tracks))).eager_graph(:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
bands = @Band.order(:tracks__name).from(Sequel.expr(:s__bands).as('tracks')).eager_graph(:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
end
it "should have eager graphs work with join tables with aliases" do
bands = @Band.order(:bands__name).eager_graph(:members).join(:s__albums___tracks, :band_id=>Sequel.qualify(:s__bands, :id)).eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
bands = @Band.order(:bands__name).eager_graph(:members).join(Sequel.as(:s__albums, :tracks), :band_id=>Sequel.qualify(:s__bands, :id)).eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
bands = @Band.order(:bands__name).eager_graph(:members).join(Sequel.as(:s__albums, 'tracks'), :band_id=>Sequel.qualify(:s__bands, :id)).eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
bands = @Band.order(:bands__name).eager_graph(:members).join(Sequel.as(:s__albums, Sequel.identifier(:tracks)), :band_id=>Sequel.qualify(:s__bands, :id)).eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
bands = @Band.order(:bands__name).eager_graph(:members).join(:s__albums, {:band_id=>Sequel.qualify(:s__bands, :id)}, :table_alias=>:tracks).eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
bands = @Band.order(:bands__name).eager_graph(:members).join(:s__albums, {:band_id=>Sequel.qualify(:s__bands, :id)}, :table_alias=>'tracks').eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
bands = @Band.order(:bands__name).eager_graph(:members).join(:s__albums, {:band_id=>Sequel.qualify(:s__bands, :id)}, :table_alias=>Sequel.identifier(:tracks)).eager_graph(:albums=>:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.albums}.must_equal [[@a1, @a2], [@a3, @a4]]
bands.map{|x| x.members}.must_equal [[@m1, @m2], [@m4, @m3]]
end
it "should have eager graphs work with different types of qualified from tables" do
bands = @Band.order(:bands__name).from(Sequel.qualify(:s, :bands)).eager_graph(:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
bands = @Band.order(:bands__name).from(Sequel.identifier(:bands).qualify(:s)).eager_graph(:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
bands = @Band.order(:bands__name).from(Sequel::SQL::QualifiedIdentifier.new(:s, 'bands')).eager_graph(:tracks).all
bands.must_equal [@b1, @b2]
bands.map{|x| x.tracks}.must_equal [[@t1, @t2, @t3, @t4], []]
end
end
if DB.server_version >= 80300
describe "PostgreSQL tsearch2" do
before(:all) do
DB.create_table! :test6 do
text :title
text :body
full_text_index [:title, :body]
end
@ds = DB[:test6]
end
after do
DB[:test6].delete
end
after(:all) do
DB.drop_table?(:test6)
end
it "should search by indexed column" do
record = {:title => "oopsla conference", :body => "test"}
@ds << record
@ds.full_text_search(:title, "oopsla").all.must_include(record)
end
it "should join multiple coumns with spaces to search by last words in row" do
record = {:title => "multiple words", :body => "are easy to search"}
@ds << record
@ds.full_text_search([:title, :body], "words").all.must_include(record)
end
it "should return rows with a NULL in one column if a match in another column" do
record = {:title => "multiple words", :body =>nil}
@ds << record
@ds.full_text_search([:title, :body], "words").all.must_include(record)
end
end
end
if DB.dataset.supports_window_functions?
describe "Postgres::Dataset named windows" do
before do
@db = DB
@db.create_table!(:i1){Integer :id; Integer :group_id; Integer :amount}
@ds = @db[:i1].order(:id)
@ds.insert(:id=>1, :group_id=>1, :amount=>1)
@ds.insert(:id=>2, :group_id=>1, :amount=>10)
@ds.insert(:id=>3, :group_id=>1, :amount=>100)
@ds.insert(:id=>4, :group_id=>2, :amount=>1000)
@ds.insert(:id=>5, :group_id=>2, :amount=>10000)
@ds.insert(:id=>6, :group_id=>2, :amount=>100000)
end
after do
@db.drop_table?(:i1)
end
it "should give correct results for window functions" do
@ds.window(:win, :partition=>:group_id, :order=>:id).select(:id){sum(:amount).over(:window=>win)}.all.
must_equal [{:sum=>1, :id=>1}, {:sum=>11, :id=>2}, {:sum=>111, :id=>3}, {:sum=>1000, :id=>4}, {:sum=>11000, :id=>5}, {:sum=>111000, :id=>6}]
@ds.window(:win, :partition=>:group_id).select(:id){sum(:amount).over(:window=>win, :order=>id)}.all.
must_equal [{:sum=>1, :id=>1}, {:sum=>11, :id=>2}, {:sum=>111, :id=>3}, {:sum=>1000, :id=>4}, {:sum=>11000, :id=>5}, {:sum=>111000, :id=>6}]
@ds.window(:win, {}).select(:id){sum(:amount).over(:window=>:win, :order=>id)}.all.
must_equal [{:sum=>1, :id=>1}, {:sum=>11, :id=>2}, {:sum=>111, :id=>3}, {:sum=>1111, :id=>4}, {:sum=>11111, :id=>5}, {:sum=>111111, :id=>6}]
@ds.window(:win, :partition=>:group_id).select(:id){sum(:amount).over(:window=>:win, :order=>id, :frame=>:all)}.all.
must_equal [{:sum=>111, :id=>1}, {:sum=>111, :id=>2}, {:sum=>111, :id=>3}, {:sum=>111000, :id=>4}, {:sum=>111000, :id=>5}, {:sum=>111000, :id=>6}]
end
end
end
describe "Postgres::Database functions, languages, schemas, and triggers" do
before do
@d = DB
end
after do
@d.drop_function('tf', :if_exists=>true, :cascade=>true)
@d.drop_function('tf', :if_exists=>true, :cascade=>true, :args=>%w'integer integer')
@d.drop_language(:plpgsql, :if_exists=>true, :cascade=>true) if @d.server_version < 90000
@d.drop_schema(:sequel, :if_exists=>true, :cascade=>true)
@d.drop_table?(:test)
end
it "#create_function and #drop_function should create and drop functions" do
proc{@d['SELECT tf()'].all}.must_raise(Sequel::DatabaseError)
args = ['tf', 'SELECT 1', {:returns=>:integer}]
@d.send(:create_function_sql, *args).must_match(/\A\s*CREATE FUNCTION tf\(\)\s+RETURNS integer\s+LANGUAGE SQL\s+AS 'SELECT 1'\s*\z/)
@d.create_function(*args)
@d['SELECT tf()'].all.must_equal [{:tf=>1}]
@d.send(:drop_function_sql, 'tf').must_equal 'DROP FUNCTION tf()'
@d.drop_function('tf')
proc{@d['SELECT tf()'].all}.must_raise(Sequel::DatabaseError)
end
it "#create_function and #drop_function should support options" do
args = ['tf', 'SELECT $1 + $2', {:args=>[[:integer, :a], :integer], :replace=>true, :returns=>:integer, :language=>'SQL', :behavior=>:immutable, :strict=>true, :security_definer=>true, :cost=>2, :set=>{:search_path => 'public'}}]
@d.send(:create_function_sql,*args).must_match(/\A\s*CREATE OR REPLACE FUNCTION tf\(a integer, integer\)\s+RETURNS integer\s+LANGUAGE SQL\s+IMMUTABLE\s+STRICT\s+SECURITY DEFINER\s+COST 2\s+SET search_path = public\s+AS 'SELECT \$1 \+ \$2'\s*\z/)
@d.create_function(*args)
# Make sure replace works
@d.create_function(*args)
@d['SELECT tf(1, 2)'].all.must_equal [{:tf=>3}]
args = ['tf', {:if_exists=>true, :cascade=>true, :args=>[[:integer, :a], :integer]}]
@d.send(:drop_function_sql,*args).must_equal 'DROP FUNCTION IF EXISTS tf(a integer, integer) CASCADE'
@d.drop_function(*args)
# Make sure if exists works
@d.drop_function(*args)
end
it "#create_language and #drop_language should create and drop languages" do
@d.send(:create_language_sql, :plpgsql).must_equal 'CREATE LANGUAGE plpgsql'
@d.create_language(:plpgsql, :replace=>true) if @d.server_version < 90000
proc{@d.create_language(:plpgsql)}.must_raise(Sequel::DatabaseError)
@d.send(:drop_language_sql, :plpgsql).must_equal 'DROP LANGUAGE plpgsql'
@d.drop_language(:plpgsql) if @d.server_version < 90000
proc{@d.drop_language(:plpgsql)}.must_raise(Sequel::DatabaseError) if @d.server_version < 90000
@d.send(:create_language_sql, :plpgsql, :replace=>true, :trusted=>true, :handler=>:a, :validator=>:b).must_equal(@d.server_version >= 90000 ? 'CREATE OR REPLACE TRUSTED LANGUAGE plpgsql HANDLER a VALIDATOR b' : 'CREATE TRUSTED LANGUAGE plpgsql HANDLER a VALIDATOR b')
@d.send(:drop_language_sql, :plpgsql, :if_exists=>true, :cascade=>true).must_equal 'DROP LANGUAGE IF EXISTS plpgsql CASCADE'
# Make sure if exists works
@d.drop_language(:plpgsql, :if_exists=>true, :cascade=>true) if @d.server_version < 90000
end
it "#create_schema and #drop_schema should create and drop schemas" do
@d.send(:create_schema_sql, :sequel).must_equal 'CREATE SCHEMA "sequel"'
@d.send(:create_schema_sql, :sequel, :if_not_exists=>true, :owner=>:foo).must_equal 'CREATE SCHEMA IF NOT EXISTS "sequel" AUTHORIZATION "foo"'
@d.send(:drop_schema_sql, :sequel).must_equal 'DROP SCHEMA "sequel"'
@d.send(:drop_schema_sql, :sequel, :if_exists=>true, :cascade=>true).must_equal 'DROP SCHEMA IF EXISTS "sequel" CASCADE'
@d.create_schema(:sequel)
@d.create_schema(:sequel, :if_not_exists=>true) if @d.server_version >= 90300
@d.create_table(:sequel__test){Integer :a}
@d.drop_schema(:sequel, :if_exists=>true, :cascade=>true)
end
it "#create_trigger and #drop_trigger should create and drop triggers" do
@d.create_language(:plpgsql) if @d.server_version < 90000
@d.create_function(:tf, 'BEGIN IF NEW.value IS NULL THEN RAISE EXCEPTION \'Blah\'; END IF; RETURN NEW; END;', :language=>:plpgsql, :returns=>:trigger)
@d.send(:create_trigger_sql, :test, :identity, :tf, :each_row=>true).must_equal 'CREATE TRIGGER identity BEFORE INSERT OR UPDATE OR DELETE ON "test" FOR EACH ROW EXECUTE PROCEDURE tf()'
@d.create_table(:test){String :name; Integer :value}
@d.create_trigger(:test, :identity, :tf, :each_row=>true)
@d[:test].insert(:name=>'a', :value=>1)
@d[:test].filter(:name=>'a').all.must_equal [{:name=>'a', :value=>1}]
proc{@d[:test].filter(:name=>'a').update(:value=>nil)}.must_raise(Sequel::DatabaseError)
@d[:test].filter(:name=>'a').all.must_equal [{:name=>'a', :value=>1}]
@d[:test].filter(:name=>'a').update(:value=>3)
@d[:test].filter(:name=>'a').all.must_equal [{:name=>'a', :value=>3}]
@d.send(:drop_trigger_sql, :test, :identity).must_equal 'DROP TRIGGER identity ON "test"'
@d.drop_trigger(:test, :identity)
@d.send(:create_trigger_sql, :test, :identity, :tf, :after=>true, :events=>:insert, :args=>[1, 'a']).must_equal 'CREATE TRIGGER identity AFTER INSERT ON "test" EXECUTE PROCEDURE tf(1, \'a\')'
@d.send(:drop_trigger_sql, :test, :identity, :if_exists=>true, :cascade=>true).must_equal 'DROP TRIGGER IF EXISTS identity ON "test" CASCADE'
# Make sure if exists works
@d.drop_trigger(:test, :identity, :if_exists=>true, :cascade=>true)
if @d.supports_trigger_conditions?
@d.send(:create_trigger_sql, :test, :identity, :tf, :each_row=>true, :when=> {:new__name => 'b'}).must_equal %q{CREATE TRIGGER identity BEFORE INSERT OR UPDATE OR DELETE ON "test" FOR EACH ROW WHEN ("new"."name" = 'b') EXECUTE PROCEDURE tf()}
@d.create_trigger(:test, :identity, :tf, :each_row=>true, :events => :update, :when=> {:new__name => 'b'})
@d[:test].filter(:name=>'a').update(:value=>nil)
@d[:test].filter(:name=>'a').all.must_equal [{:name=>'a', :value=>nil}]
proc{@d[:test].filter(:name=>'a').update(:name=>'b')}.must_raise(Sequel::DatabaseError)
@d[:test].filter(:name=>'a').all.must_equal [{:name=>'a', :value=>nil}]
@d.drop_trigger(:test, :identity)
end
end
end
if DB.adapter_scheme == :postgres
describe "Postgres::Dataset #use_cursor" do
before(:all) do
@db = DB
@db.create_table!(:test_cursor){Integer :x}
@db.sqls.clear
@ds = @db[:test_cursor]
@db.transaction{1001.times{|i| @ds.insert(i)}}
end
after(:all) do
@db.drop_table?(:test_cursor)
end
it "should return the same results as the non-cursor use" do
@ds.all.must_equal @ds.use_cursor.all
end
it "should not swallow errors if closing cursor raises an error" do
proc do
@db.synchronize do |c|
@ds.use_cursor.each do |r|
@db.run "CLOSE sequel_cursor"
raise ArgumentError
end
end
end.must_raise(ArgumentError)
end
it "should respect the :rows_per_fetch option" do
@db.sqls.clear
@ds.use_cursor.all
check_sqls do
@db.sqls.length.must_equal 6
@db.sqls.clear
end
@ds.use_cursor(:rows_per_fetch=>100).all
check_sqls do
@db.sqls.length.must_equal 15
end
end
it "should respect the :hold=>true option for creating the cursor WITH HOLD and not using a transaction" do
@ds.use_cursor.each{@db.in_transaction?.must_equal true}
check_sqls{@db.sqls.any?{|s| s =~ /WITH HOLD/}.must_equal false}
@ds.use_cursor(:hold=>true).each{@db.in_transaction?.must_equal false}
check_sqls{@db.sqls.any?{|s| s =~ /WITH HOLD/}.must_equal true}
end
it "should support updating individual rows based on a cursor" do
@db.transaction(:rollback=>:always) do
@ds.use_cursor(:rows_per_fetch=>1).each do |row|
@ds.where_current_of.update(:x=>Sequel.*(row[:x], 10))
end
@ds.select_order_map(:x).must_equal((0..1000).map{|x| x * 10})
end
@ds.select_order_map(:x).must_equal((0..1000).to_a)
end
it "should respect the :cursor_name option" do
one_rows = []
two_rows = []
@ds.order(:x).use_cursor(:cursor_name => 'cursor_one').each do |one|
one_rows << one
if one[:x] % 1000 == 500
two_rows = []
@ds.order(:x).use_cursor(:cursor_name => 'cursor_two').each do |two|
two_rows << two
end
end
end
one_rows.must_equal two_rows
end
it "should handle returning inside block" do
def @ds.check_return
use_cursor.each{|r| return}
end
@ds.check_return
@ds.all.must_equal @ds.use_cursor.all
end
end
describe "Postgres::PG_NAMED_TYPES" do
before(:all) do
@db = DB
@old_cp = @db.conversion_procs[1013]
@db.conversion_procs.delete(1013)
Sequel::Postgres::PG_NAMED_TYPES[:oidvector] = lambda{|v| v.reverse}
@db.reset_conversion_procs
@db.register_array_type('oidvector')
end
after(:all) do
Sequel::Postgres::PG_NAMED_TYPES.delete(:oidvector)
@db.conversion_procs.delete(30)
@db.conversion_procs[1013] = @old_cp
@db.drop_table?(:foo)
@db.drop_enum(:foo_enum)
end
it "should look up conversion procs by name" do
@db.create_table!(:foo){oidvector :bar}
@db[:foo].insert(Sequel.cast('21', :oidvector))
@db[:foo].get(:bar).must_equal '12'
end
it "should handle array types of named types" do
@db.create_table!(:foo){column :bar, 'oidvector[]'}
@db[:foo].insert(Sequel.pg_array(['21'], :oidvector))
@db[:foo].get(:bar).must_equal ['12']
end
it "should work with conversion procs on enums" do
@db.drop_enum(:foo_enum) rescue nil
@db.create_enum(:foo_enum, %w(foo bar))
@db.add_named_conversion_proc(:foo_enum){|string| string.reverse}
@db.create_table!(:foo){foo_enum :bar}
@db[:foo].insert(:bar => 'foo')
@db[:foo].get(:bar).must_equal 'foo'.reverse
end
end
end
if ((DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG) || DB.adapter_scheme == :jdbc) && DB.server_version >= 90000
describe "Postgres::Database#copy_into" do
before(:all) do
@db = DB
@db.create_table!(:test_copy){Integer :x; Integer :y}
@ds = @db[:test_copy].order(:x, :y)
end
before do
@db[:test_copy].delete
end
after(:all) do
@db.drop_table?(:test_copy)
end
it "should work with a :data option containing data in PostgreSQL text format" do
@db.copy_into(:test_copy, :data=>"1\t2\n3\t4\n")
@ds.select_map([:x, :y]).must_equal [[1, 2], [3, 4]]
end
it "should work with :format=>:csv option and :data option containing data in CSV format" do
@db.copy_into(:test_copy, :format=>:csv, :data=>"1,2\n3,4\n")
@ds.select_map([:x, :y]).must_equal [[1, 2], [3, 4]]
end
it "should respect given :options" do
@db.copy_into(:test_copy, :options=>"FORMAT csv, HEADER TRUE", :data=>"x,y\n1,2\n3,4\n")
@ds.select_map([:x, :y]).must_equal [[1, 2], [3, 4]]
end
it "should respect given :options options when :format is used" do
@db.copy_into(:test_copy, :options=>"QUOTE '''', DELIMITER '|'", :format=>:csv, :data=>"'1'|'2'\n'3'|'4'\n")
@ds.select_map([:x, :y]).must_equal [[1, 2], [3, 4]]
end
it "should accept :columns option to online copy the given columns" do
@db.copy_into(:test_copy, :data=>"1\t2\n3\t4\n", :columns=>[:y, :x])
@ds.select_map([:x, :y]).must_equal [[2, 1], [4, 3]]
end
it "should accept a block and use returned values for the copy in data stream" do
buf = ["1\t2\n", "3\t4\n"]
@db.copy_into(:test_copy){buf.shift}
@ds.select_map([:x, :y]).must_equal [[1, 2], [3, 4]]
end
it "should work correctly with a block and :format=>:csv" do
buf = ["1,2\n", "3,4\n"]
@db.copy_into(:test_copy, :format=>:csv){buf.shift}
@ds.select_map([:x, :y]).must_equal [[1, 2], [3, 4]]
end
it "should accept an enumerable as the :data option" do
@db.copy_into(:test_copy, :data=>["1\t2\n", "3\t4\n"])
@ds.select_map([:x, :y]).must_equal [[1, 2], [3, 4]]
end
it "should have an exception, cause a rollback of copied data and still have a usable connection" do
2.times do
sent = false
proc{@db.copy_into(:test_copy){raise ArgumentError if sent; sent = true; "1\t2\n"}}.must_raise(ArgumentError)
@ds.select_map([:x, :y]).must_equal []
end
end
it "should handle database errors with a rollback of copied data and still have a usable connection" do
2.times do
proc{@db.copy_into(:test_copy, :data=>["1\t2\n", "3\ta\n"])}.must_raise(Sequel::DatabaseError)
@ds.select_map([:x, :y]).must_equal []
end
end
it "should raise an Error if both :data and a block are provided" do
proc{@db.copy_into(:test_copy, :data=>["1\t2\n", "3\t4\n"]){}}.must_raise(Sequel::Error)
end
it "should raise an Error if neither :data or a block are provided" do
proc{@db.copy_into(:test_copy)}.must_raise(Sequel::Error)
end
end
describe "Postgres::Database#copy_table" do
before(:all) do
@db = DB
@db.create_table!(:test_copy){Integer :x; Integer :y}
ds = @db[:test_copy]
ds.insert(1, 2)
ds.insert(3, 4)
end
after(:all) do
@db.drop_table?(:test_copy)
end
it "without a block or options should return a text version of the table as a single string" do
@db.copy_table(:test_copy).must_equal "1\t2\n3\t4\n"
end
it "without a block and with :format=>:csv should return a csv version of the table as a single string" do
@db.copy_table(:test_copy, :format=>:csv).must_equal "1,2\n3,4\n"
end
it "should treat string as SQL code" do
@db.copy_table('COPY "test_copy" TO STDOUT').must_equal "1\t2\n3\t4\n"
end
it "should respect given :options options" do
@db.copy_table(:test_copy, :options=>"FORMAT csv, HEADER TRUE").must_equal "x,y\n1,2\n3,4\n"
end
it "should respect given :options options when :format is used" do
@db.copy_table(:test_copy, :format=>:csv, :options=>"QUOTE '''', FORCE_QUOTE *").must_equal "'1','2'\n'3','4'\n"
end
it "should accept dataset as first argument" do
@db.copy_table(@db[:test_copy].cross_join(:test_copy___tc).order(:test_copy__x, :test_copy__y, :tc__x, :tc__y)).must_equal "1\t2\t1\t2\n1\t2\t3\t4\n3\t4\t1\t2\n3\t4\t3\t4\n"
end
it "with a block and no options should yield each row as a string in text format" do
buf = []
@db.copy_table(:test_copy){|b| buf << b}
buf.must_equal ["1\t2\n", "3\t4\n"]
end
it "with a block and :format=>:csv should yield each row as a string in csv format" do
buf = []
@db.copy_table(:test_copy, :format=>:csv){|b| buf << b}
buf.must_equal ["1,2\n", "3,4\n"]
end
it "should work fine when using a block that is terminated early with a following copy_table" do
buf = []
proc{@db.copy_table(:test_copy, :format=>:csv){|b| buf << b; break}}.must_raise(Sequel::DatabaseDisconnectError)
buf.must_equal ["1,2\n"]
buf.clear
proc{@db.copy_table(:test_copy, :format=>:csv){|b| buf << b; raise ArgumentError}}.must_raise(Sequel::DatabaseDisconnectError)
buf.must_equal ["1,2\n"]
buf.clear
@db.copy_table(:test_copy){|b| buf << b}
buf.must_equal ["1\t2\n", "3\t4\n"]
end
it "should work fine when using a block that is terminated early with a following regular query" do
buf = []
proc{@db.copy_table(:test_copy, :format=>:csv){|b| buf << b; break}}.must_raise(Sequel::DatabaseDisconnectError)
buf.must_equal ["1,2\n"]
buf.clear
proc{@db.copy_table(:test_copy, :format=>:csv){|b| buf << b; raise ArgumentError}}.must_raise(Sequel::DatabaseDisconnectError)
buf.must_equal ["1,2\n"]
@db[:test_copy].select_order_map(:x).must_equal [1, 3]
end
end
end
if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG && DB.server_version >= 90000
describe "Postgres::Database LISTEN/NOTIFY" do
before(:all) do
@db = DB
end
it "should support listen and notify" do
notify_pid = @db.synchronize{|conn| conn.backend_pid}
called = false
@db.listen('foo', :after_listen=>proc{@db.notify('foo')}) do |ev, pid, payload|
ev.must_equal 'foo'
pid.must_equal notify_pid
['', nil].must_include(payload)
called = true
end.must_equal 'foo'
called.must_equal true
# Check weird identifier names
called = false
@db.listen('FOO bar', :after_listen=>proc{@db.notify('FOO bar')}) do |ev, pid, payload|
ev.must_equal 'FOO bar'
pid.must_equal notify_pid
['', nil].must_include(payload)
called = true
end.must_equal 'FOO bar'
called.must_equal true
# Check identifier symbols
called = false
@db.listen(:foo, :after_listen=>proc{@db.notify(:foo)}) do |ev, pid, payload|
ev.must_equal 'foo'
pid.must_equal notify_pid
['', nil].must_include(payload)
called = true
end.must_equal 'foo'
called.must_equal true
called = false
@db.listen('foo', :after_listen=>proc{@db.notify('foo', :payload=>'bar')}) do |ev, pid, payload|
ev.must_equal 'foo'
pid.must_equal notify_pid
payload.must_equal 'bar'
called = true
end.must_equal 'foo'
called.must_equal true
@db.listen('foo', :after_listen=>proc{@db.notify('foo')}).must_equal 'foo'
called = false
called2 = false
i = 0
@db.listen(['foo', 'bar'], :after_listen=>proc{@db.notify('foo', :payload=>'bar'); @db.notify('bar', :payload=>'foo')}, :loop=>proc{i+=1}) do |ev, pid, payload|
if !called
ev.must_equal 'foo'
pid.must_equal notify_pid
payload.must_equal 'bar'
called = true
else
ev.must_equal 'bar'
pid.must_equal notify_pid
payload.must_equal 'foo'
called2 = true
break
end
end.must_equal nil
called.must_equal true
called2.must_equal true
i.must_equal 1
end
it "should accept a :timeout option in listen" do
@db.listen('foo2', :timeout=>0.001).must_equal nil
called = false
@db.listen('foo2', :timeout=>0.001){|ev, pid, payload| called = true}.must_equal nil
called.must_equal false
i = 0
@db.listen('foo2', :timeout=>0.001, :loop=>proc{i+=1; throw :stop if i > 3}){|ev, pid, payload| called = true}.must_equal nil
i.must_equal 4
called = false
i = 0
@db.listen('foo2', :timeout=>proc{i+=1; 0.001}){|ev, pid, payload| called = true}.must_equal nil
called.must_equal false
i.must_equal 1
i = 0
t = 0
@db.listen('foo2', :timeout=>proc{t+=1; 0.001}, :loop=>proc{i+=1; throw :stop if i > 3}){|ev, pid, payload| called = true}.must_equal nil
called.must_equal false
t.must_equal 4
end unless RUBY_PLATFORM =~ /mingw/ # Ruby freezes on this spec on this platform/version
end
end
describe 'PostgreSQL special float handling' do
before do
@db = DB
@db.create_table!(:test5){Float :value}
@db.sqls.clear
@ds = @db[:test5]
end
after do
@db.drop_table?(:test5)
end
check_sqls do
it 'should quote NaN' do
nan = 0.0/0.0
@ds.insert_sql(:value => nan).must_equal %q{INSERT INTO "test5" ("value") VALUES ('NaN')}
end
it 'should quote +Infinity' do
inf = 1.0/0.0
@ds.insert_sql(:value => inf).must_equal %q{INSERT INTO "test5" ("value") VALUES ('Infinity')}
end
it 'should quote -Infinity' do
inf = -1.0/0.0
@ds.insert_sql(:value => inf).must_equal %q{INSERT INTO "test5" ("value") VALUES ('-Infinity')}
end
end
if DB.adapter_scheme == :postgres
it 'inserts NaN' do
nan = 0.0/0.0
@ds.insert(:value=>nan)
@ds.all[0][:value].nan?.must_equal true
end
it 'inserts +Infinity' do
inf = 1.0/0.0
@ds.insert(:value=>inf)
@ds.all[0][:value].infinite?.must_be :>, 0
end
it 'inserts -Infinity' do
inf = -1.0/0.0
@ds.insert(:value=>inf)
@ds.all[0][:value].infinite?.must_be :<, 0
end
end
end
describe 'PostgreSQL array handling' do
before(:all) do
@db = DB
@ds = @db[:items]
@native = DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
@tp = lambda{@db.schema(:items).map{|a| a.last[:type]}}
end
after do
@db.drop_table?(:items)
end
it 'insert and retrieve integer and float arrays of various sizes' do
@db.create_table!(:items) do
column :i2, 'int2[]'
column :i4, 'int4[]'
column :i8, 'int8[]'
column :r, 'real[]'
column :dp, 'double precision[]'
end
@tp.call.must_equal [:smallint_array, :integer_array, :bigint_array, :real_array, :float_array]
@ds.insert(Sequel.pg_array([1], :int2), Sequel.pg_array([nil, 2], :int4), Sequel.pg_array([3, nil], :int8), Sequel.pg_array([4, nil, 4.5], :real), Sequel.pg_array([5, nil, 5.5], "double precision"))
@ds.count.must_equal 1
rs = @ds.all
if @native
rs.must_equal [{:i2=>[1], :i4=>[nil, 2], :i8=>[3, nil], :r=>[4.0, nil, 4.5], :dp=>[5.0, nil, 5.5]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@ds.delete
@ds.insert(Sequel.pg_array([[1], [2]], :int2), Sequel.pg_array([[nil, 2], [3, 4]], :int4), Sequel.pg_array([[3, nil], [nil, nil]], :int8), Sequel.pg_array([[4, nil], [nil, 4.5]], :real), Sequel.pg_array([[5, nil], [nil, 5.5]], "double precision"))
rs = @ds.all
if @native
rs.must_equal [{:i2=>[[1], [2]], :i4=>[[nil, 2], [3, 4]], :i8=>[[3, nil], [nil, nil]], :r=>[[4, nil], [nil, 4.5]], :dp=>[[5, nil], [nil, 5.5]]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'insert and retrieve decimal arrays' do
@db.create_table!(:items) do
column :n, 'numeric[]'
end
@tp.call.must_equal [:decimal_array]
@ds.insert(Sequel.pg_array([BigDecimal.new('1.000000000000000000001'), nil, BigDecimal.new('1')], :numeric))
@ds.count.must_equal 1
rs = @ds.all
if @native
rs.must_equal [{:n=>[BigDecimal.new('1.000000000000000000001'), nil, BigDecimal.new('1')]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@ds.delete
@ds.insert(Sequel.pg_array([[BigDecimal.new('1.0000000000000000000000000000001'), nil], [nil, BigDecimal.new('1')]], :numeric))
rs = @ds.all
if @native
rs.must_equal [{:n=>[[BigDecimal.new('1.0000000000000000000000000000001'), nil], [nil, BigDecimal.new('1')]]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'insert and retrieve string arrays' do
@db.create_table!(:items) do
column :c, 'char(4)[]'
column :vc, 'varchar[]'
column :t, 'text[]'
end
@tp.call.must_equal [:character_array, :varchar_array, :string_array]
@ds.insert(Sequel.pg_array(['a', nil, 'NULL', 'b"\'c'], 'char(4)'), Sequel.pg_array(['a', nil, 'NULL', 'b"\'c', '', ''], :varchar), Sequel.pg_array(['a', nil, 'NULL', 'b"\'c'], :text))
@ds.count.must_equal 1
rs = @ds.all
if @native
rs.must_equal [{:c=>['a ', nil, 'NULL', 'b"\'c'], :vc=>['a', nil, 'NULL', 'b"\'c', '', ''], :t=>['a', nil, 'NULL', 'b"\'c']}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@ds.delete
@ds.insert(Sequel.pg_array([[['a'], [nil]], [['NULL'], ['b"\'c']]], 'char(4)'), Sequel.pg_array([[['a[],\\[\\]\\,\\""NULL",'], ['']], [['NULL'], ['b"\'c']]], :varchar), Sequel.pg_array([[['a'], [nil]], [['NULL'], ['b"\'c']]], :text))
rs = @ds.all
if @native
rs.must_equal [{:c=>[[['a '], [nil]], [['NULL'], ['b"\'c']]], :vc=>[[['a[],\\[\\]\\,\\""NULL",'], ['']], [['NULL'], ['b"\'c']]], :t=>[[['a'], [nil]], [['NULL'], ['b"\'c']]]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'insert and retrieve arrays of other types' do
@db.create_table!(:items) do
column :b, 'bool[]'
column :d, 'date[]'
column :t, 'time[]'
column :ts, 'timestamp[]'
column :tstz, 'timestamptz[]'
end
@tp.call.must_equal [:boolean_array, :date_array, :time_array, :datetime_array, :datetime_timezone_array]
d = Date.today
t = Sequel::SQLTime.create(10, 20, 30)
ts = Time.local(2011, 1, 2, 3, 4, 5)
@ds.insert(Sequel.pg_array([true, false], :bool), Sequel.pg_array([d, nil], :date), Sequel.pg_array([t, nil], :time), Sequel.pg_array([ts, nil], :timestamp), Sequel.pg_array([ts, nil], :timestamptz))
@ds.count.must_equal 1
rs = @ds.all
if @native
rs.must_equal [{:b=>[true, false], :d=>[d, nil], :t=>[t, nil], :ts=>[ts, nil], :tstz=>[ts, nil]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@db.create_table!(:items) do
column :ba, 'bytea[]'
column :tz, 'timetz[]'
column :o, 'oid[]'
end
@tp.call.must_equal [:blob_array, :time_timezone_array, :oid_array]
@ds.insert(Sequel.pg_array([Sequel.blob("a\0"), nil], :bytea), Sequel.pg_array([t, nil], :timetz), Sequel.pg_array([1, 2, 3], :oid))
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.must_equal [{:ba=>[Sequel.blob("a\0"), nil], :tz=>[t, nil], :o=>[1, 2, 3]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@db.create_table!(:items) do
column :x, 'xml[]'
column :m, 'money[]'
column :b, 'bit[]'
column :vb, 'bit varying[]'
column :u, 'uuid[]'
column :xi, 'xid[]'
column :c, 'cid[]'
column :n, 'name[]'
column :o, 'oidvector[]'
end
@tp.call.must_equal [:xml_array, :money_array, :bit_array, :varbit_array, :uuid_array, :xid_array, :cid_array, :name_array, :oidvector_array]
@ds.insert(Sequel.pg_array([''], :xml),
Sequel.pg_array(['1'], :money),
Sequel.pg_array(['1'], :bit),
Sequel.pg_array(['10'], :varbit),
Sequel.pg_array(['c0f24910-39e7-11e4-916c-0800200c9a66'], :uuid),
Sequel.pg_array(['12'], :xid),
Sequel.pg_array(['12'], :cid),
Sequel.pg_array(['N'], :name),
Sequel.pg_array(['1 2'], :oidvector))
@ds.count.must_equal 1
if @native
rs = @ds.all
r = rs.first
m = r.delete(:m)
m.class.must_equal(Sequel::Postgres::PGArray)
m.to_a.must_be_kind_of(Array)
m.first.must_be_kind_of(String)
r.must_be(:==, :x=>[''], :b=>['1'], :vb=>['10'], :u=>['c0f24910-39e7-11e4-916c-0800200c9a66'], :xi=>['12'], :c=>['12'], :n=>['N'], :o=>['1 2'])
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
r[:m] = m
@ds.delete
@ds.insert(r)
@ds.all.must_equal rs
end
end
it 'insert and retrieve empty arrays' do
@db.create_table!(:items) do
column :n, 'integer[]'
end
@ds.insert(:n=>Sequel.pg_array([], :integer))
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.must_equal [{:n=>[]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'convert ruby array :default values' do
@db.create_table!(:items) do
column :n, 'integer[]', :default=>[]
end
@ds.insert
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.must_equal [{:n=>[]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'insert and retrieve custom array types' do
int2vector = Class.new do
attr_reader :array
def initialize(array)
@array = array
end
def sql_literal_append(ds, sql)
sql << "'#{array.join(' ')}'"
end
def ==(other)
if other.is_a?(self.class)
array == other.array
else
super
end
end
end
@db.register_array_type(:int2vector){|s| int2vector.new(s.split.map{|i| i.to_i})}
@db.create_table!(:items) do
column :b, 'int2vector[]'
end
@tp.call.must_equal [:int2vector_array]
int2v = int2vector.new([1, 2])
@ds.insert(Sequel.pg_array([int2v], :int2vector))
@ds.count.must_equal 1
rs = @ds.all
if @native
rs.must_equal [{:b=>[int2v]}]
rs.first.values.each{|v| v.class.must_equal(Sequel::Postgres::PGArray)}
rs.first.values.each{|v| v.to_a.must_be_kind_of(Array)}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'use arrays in bound variables' do
@db.create_table!(:items) do
column :i, 'int4[]'
end
@ds.call(:insert, {:i=>[1,2]}, {:i=>:$i})
@ds.get(:i).must_equal [1, 2]
@ds.filter(:i=>:$i).call(:first, :i=>[1,2]).must_equal(:i=>[1,2])
@ds.filter(:i=>:$i).call(:first, :i=>[1,3]).must_equal nil
# NULL values
@ds.delete
@ds.call(:insert, {:i=>[nil,nil]}, {:i=>:$i})
@ds.first.must_equal(:i=>[nil, nil])
@db.create_table!(:items) do
column :i, 'text[]'
end
a = ["\"\\\\\"{}\n\t\r \v\b123afP", 'NULL', nil, '']
@ds.call(:insert, {:i=>:$i}, :i=>Sequel.pg_array(a))
@ds.get(:i).must_equal a
@ds.filter(:i=>:$i).call(:first, :i=>a).must_equal(:i=>a)
@ds.filter(:i=>:$i).call(:first, :i=>['', nil, nil, 'a']).must_equal nil
@db.create_table!(:items) do
column :i, 'date[]'
end
a = [Date.today]
@ds.call(:insert, {:i=>:$i}, :i=>Sequel.pg_array(a, 'date'))
@ds.get(:i).must_equal a
@ds.filter(:i=>:$i).call(:first, :i=>a).must_equal(:i=>a)
@ds.filter(:i=>:$i).call(:first, :i=>Sequel.pg_array([Date.today-1], 'date')).must_equal nil
@db.create_table!(:items) do
column :i, 'timestamp[]'
end
a = [Time.local(2011, 1, 2, 3, 4, 5)]
@ds.call(:insert, {:i=>:$i}, :i=>Sequel.pg_array(a, 'timestamp'))
@ds.get(:i).must_equal a
@ds.filter(:i=>:$i).call(:first, :i=>a).must_equal(:i=>a)
@ds.filter(:i=>:$i).call(:first, :i=>Sequel.pg_array([a.first-1], 'timestamp')).must_equal nil
@db.create_table!(:items) do
column :i, 'boolean[]'
end
a = [true, false]
@ds.call(:insert, {:i=>:$i}, :i=>Sequel.pg_array(a, 'boolean'))
@ds.get(:i).must_equal a
@ds.filter(:i=>:$i).call(:first, :i=>a).must_equal(:i=>a)
@ds.filter(:i=>:$i).call(:first, :i=>Sequel.pg_array([false, true], 'boolean')).must_equal nil
@db.create_table!(:items) do
column :i, 'bytea[]'
end
a = [Sequel.blob("a\0'\"")]
@ds.call(:insert, {:i=>:$i}, :i=>Sequel.pg_array(a, 'bytea'))
@ds.get(:i).must_equal a
@ds.filter(:i=>:$i).call(:first, :i=>a).must_equal(:i=>a)
@ds.filter(:i=>:$i).call(:first, :i=>Sequel.pg_array([Sequel.blob("b\0")], 'bytea')).must_equal nil
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'with models' do
@db.create_table!(:items) do
primary_key :id
column :i, 'integer[]'
column :f, 'double precision[]'
column :d, 'numeric[]'
column :t, 'text[]'
end
c = Class.new(Sequel::Model(@db[:items]))
c.plugin :pg_typecast_on_load, :i, :f, :d, :t unless @native
h = {:i=>[1,2, nil], :f=>[[1, 2.5], [3, 4.5]], :d=>[1, BigDecimal.new('1.000000000000000000001')], :t=>[%w'a b c', ['NULL', nil, '1']]}
o = c.create(h)
o.i.must_equal [1, 2, nil]
o.f.must_equal [[1, 2.5], [3, 4.5]]
o.d.must_equal [BigDecimal.new('1'), BigDecimal.new('1.000000000000000000001')]
o.t.must_equal [%w'a b c', ['NULL', nil, '1']]
c.where(:i=>o.i, :f=>o.f, :d=>o.d, :t=>o.t).all.must_equal [o]
o2 = c.new(h)
c.where(:i=>o2.i, :f=>o2.f, :d=>o2.d, :t=>o2.t).all.must_equal [o]
@db.create_table!(:items) do
primary_key :id
column :i, 'int2[]'
column :f, 'real[]'
column :d, 'numeric(30,28)[]'
column :t, 'varchar[]'
end
c = Class.new(Sequel::Model(@db[:items]))
c.plugin :pg_typecast_on_load, :i, :f, :d, :t unless @native
o = c.create(:i=>[1,2, nil], :f=>[[1, 2.5], [3, 4.5]], :d=>[1, BigDecimal.new('1.000000000000000000001')], :t=>[%w'a b c', ['NULL', nil, '1']])
o.i.must_equal [1, 2, nil]
o.f.must_equal [[1, 2.5], [3, 4.5]]
o.d.must_equal [BigDecimal.new('1'), BigDecimal.new('1.000000000000000000001')]
o.t.must_equal [%w'a b c', ['NULL', nil, '1']]
c.where(:i=>o.i, :f=>o.f, :d=>o.d, :t=>o.t).all.must_equal [o]
o2 = c.new(h)
c.where(:i=>o2.i, :f=>o2.f, :d=>o2.d, :t=>o2.t).all.must_equal [o]
end
it 'operations/functions with pg_array_ops' do
Sequel.extension :pg_array_ops
@db.create_table!(:items){column :i, 'integer[]'; column :i2, 'integer[]'; column :i3, 'integer[]'; column :i4, 'integer[]'; column :i5, 'integer[]'}
@ds.insert(Sequel.pg_array([1, 2, 3]), Sequel.pg_array([2, 1]), Sequel.pg_array([4, 4]), Sequel.pg_array([[5, 5], [4, 3]]), Sequel.pg_array([1, nil, 5]))
@ds.get(Sequel.pg_array(:i) > :i3).must_equal false
@ds.get(Sequel.pg_array(:i3) > :i).must_equal true
@ds.get(Sequel.pg_array(:i) >= :i3).must_equal false
@ds.get(Sequel.pg_array(:i) >= :i).must_equal true
@ds.get(Sequel.pg_array(:i3) < :i).must_equal false
@ds.get(Sequel.pg_array(:i) < :i3).must_equal true
@ds.get(Sequel.pg_array(:i3) <= :i).must_equal false
@ds.get(Sequel.pg_array(:i) <= :i).must_equal true
@ds.get(Sequel.expr(5=>Sequel.pg_array(:i).any)).must_equal false
@ds.get(Sequel.expr(1=>Sequel.pg_array(:i).any)).must_equal true
@ds.get(Sequel.expr(1=>Sequel.pg_array(:i3).all)).must_equal false
@ds.get(Sequel.expr(4=>Sequel.pg_array(:i3).all)).must_equal true
@ds.get(Sequel.expr(1=>Sequel.pg_array(:i)[1..1].any)).must_equal true
@ds.get(Sequel.expr(2=>Sequel.pg_array(:i)[1..1].any)).must_equal false
@ds.get(Sequel.pg_array(:i2)[1]).must_equal 2
@ds.get(Sequel.pg_array(:i2)[1]).must_equal 2
@ds.get(Sequel.pg_array(:i2)[2]).must_equal 1
@ds.get(Sequel.pg_array(:i4)[2][1]).must_equal 4
@ds.get(Sequel.pg_array(:i4)[2][2]).must_equal 3
@ds.get(Sequel.pg_array(:i).contains(:i2)).must_equal true
@ds.get(Sequel.pg_array(:i).contains(:i3)).must_equal false
@ds.get(Sequel.pg_array(:i2).contained_by(:i)).must_equal true
@ds.get(Sequel.pg_array(:i).contained_by(:i2)).must_equal false
@ds.get(Sequel.pg_array(:i).overlaps(:i2)).must_equal true
@ds.get(Sequel.pg_array(:i2).overlaps(:i3)).must_equal false
@ds.get(Sequel.pg_array(:i).dims).must_equal '[1:3]'
@ds.get(Sequel.pg_array(:i).length).must_equal 3
@ds.get(Sequel.pg_array(:i).lower).must_equal 1
if @db.server_version >= 80400
@ds.select(Sequel.pg_array(:i).unnest).from_self.count.must_equal 3
end
if @db.server_version >= 90000
@ds.get(Sequel.pg_array(:i5).join).must_equal '15'
@ds.get(Sequel.pg_array(:i5).join(':')).must_equal '1:5'
@ds.get(Sequel.pg_array(:i5).join(':', '*')).must_equal '1:*:5'
end
if @db.server_version >= 90300
@ds.get(Sequel.pg_array(:i5).remove(1).length).must_equal 2
@ds.get(Sequel.pg_array(:i5).replace(1, 4).contains([1])).must_equal false
@ds.get(Sequel.pg_array(:i5).replace(1, 4).contains([4])).must_equal true
end
if @db.server_version >= 90400
@ds.get(Sequel.pg_array(:i).cardinality).must_equal 3
@ds.get(Sequel.pg_array(:i4).cardinality).must_equal 4
@ds.get(Sequel.pg_array(:i5).cardinality).must_equal 3
@ds.from{Sequel.pg_array([1,2,3]).op.unnest([4,5,6], [7,8]).as(:t1, [:a, :b, :c])}.select_order_map([:a, :b, :c]).must_equal [[1, 4, 7], [2, 5, 8], [3, 6, nil]]
end
if @native
@ds.get(Sequel.pg_array(:i).push(4)).must_equal [1, 2, 3, 4]
@ds.get(Sequel.pg_array(:i).unshift(4)).must_equal [4, 1, 2, 3]
@ds.get(Sequel.pg_array(:i).concat(:i2)).must_equal [1, 2, 3, 2, 1]
end
if @db.type_supported?(:hstore)
Sequel.extension :pg_hstore_ops
@db.get(Sequel.pg_array(['a', 'b']).op.hstore['a']).must_equal 'b'
@db.get(Sequel.pg_array(['a', 'b']).op.hstore(['c', 'd'])['a']).must_equal 'c'
end
end
end
describe 'PostgreSQL hstore handling' do
before(:all) do
@db = DB
@ds = @db[:items]
@h = {'a'=>'b', 'c'=>nil, 'd'=>'NULL', 'e'=>'\\\\" \\\' ,=>'}
@native = DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
end
after do
@db.drop_table?(:items)
end
it 'insert and retrieve hstore values' do
@db.create_table!(:items) do
column :h, :hstore
end
@ds.insert(Sequel.hstore(@h))
@ds.count.must_equal 1
if @native
rs = @ds.all
v = rs.first[:h]
v.must_equal @h
v.class.must_equal(Sequel::Postgres::HStore)
v.to_hash.must_be_kind_of(Hash)
v.to_hash.must_equal @h
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'insert and retrieve hstore[] values' do
@db.create_table!(:items) do
column :h, 'hstore[]'
end
@ds.insert(Sequel.pg_array([Sequel.hstore(@h)], :hstore))
@ds.count.must_equal 1
if @native
rs = @ds.all
v = rs.first[:h].first
v.class.must_equal(Sequel::Postgres::HStore)
v.to_hash.must_be_kind_of(Hash)
v.to_hash.must_equal @h
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'use hstore in bound variables' do
@db.create_table!(:items) do
column :i, :hstore
end
@ds.call(:insert, {:i=>Sequel.hstore(@h)}, {:i=>:$i})
@ds.get(:i).must_equal @h
@ds.filter(:i=>:$i).call(:first, :i=>Sequel.hstore(@h)).must_equal(:i=>@h)
@ds.filter(:i=>:$i).call(:first, :i=>Sequel.hstore({})).must_equal nil
@ds.delete
@ds.call(:insert, {:i=>Sequel.hstore('a'=>nil)}, {:i=>:$i})
@ds.get(:i).must_equal Sequel.hstore('a'=>nil)
@ds.delete
@ds.call(:insert, {:i=>@h}, {:i=>:$i})
@ds.get(:i).must_equal @h
@ds.filter(:i=>:$i).call(:first, :i=>@h).must_equal(:i=>@h)
@ds.filter(:i=>:$i).call(:first, :i=>{}).must_equal nil
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'with models and associations' do
@db.create_table!(:items) do
primary_key :id
column :h, :hstore
end
c = Class.new(Sequel::Model(@db[:items])) do
def self.name
'Item'
end
unrestrict_primary_key
def item_id
h['item_id'].to_i if h
end
def left_item_id
h['left_item_id'].to_i if h
end
end
Sequel.extension :pg_hstore_ops
c.plugin :many_through_many
c.plugin :pg_typecast_on_load, :h unless @native
h = {'item_id'=>"2", 'left_item_id'=>"1"}
o2 = c.create(:id=>2)
o = c.create(:id=>1, :h=>h)
o.h.must_equal h
c.many_to_one :item, :class=>c, :key_column=>Sequel.cast(Sequel.hstore(:h)['item_id'], Integer)
c.one_to_many :items, :class=>c, :key=>Sequel.cast(Sequel.hstore(:h)['item_id'], Integer), :key_method=>:item_id
c.many_to_many :related_items, :class=>c, :join_table=>:items___i, :left_key=>Sequel.cast(Sequel.hstore(:h)['left_item_id'], Integer), :right_key=>Sequel.cast(Sequel.hstore(:h)['item_id'], Integer)
c.many_to_one :other_item, :class=>c, :key=>:id, :primary_key_method=>:item_id, :primary_key=>Sequel.cast(Sequel.hstore(:h)['item_id'], Integer), :reciprocal=>:other_items
c.one_to_many :other_items, :class=>c, :primary_key=>:item_id, :key=>:id, :primary_key_column=>Sequel.cast(Sequel.hstore(:h)['item_id'], Integer), :reciprocal=>:other_item
c.many_to_many :other_related_items, :class=>c, :join_table=>:items___i, :left_key=>:id, :right_key=>:id,
:left_primary_key_column=>Sequel.cast(Sequel.hstore(:h)['left_item_id'], Integer),
:left_primary_key=>:left_item_id,
:right_primary_key=>Sequel.cast(Sequel.hstore(:h)['left_item_id'], Integer),
:right_primary_key_method=>:left_item_id
c.many_through_many :mtm_items, [
[:items, Sequel.cast(Sequel.hstore(:h)['item_id'], Integer), Sequel.cast(Sequel.hstore(:h)['left_item_id'], Integer)],
[:items, Sequel.cast(Sequel.hstore(:h)['left_item_id'], Integer), Sequel.cast(Sequel.hstore(:h)['left_item_id'], Integer)]
],
:class=>c,
:left_primary_key_column=>Sequel.cast(Sequel.hstore(:h)['item_id'], Integer),
:left_primary_key=>:item_id,
:right_primary_key=>Sequel.cast(Sequel.hstore(:h)['left_item_id'], Integer),
:right_primary_key_method=>:left_item_id
# Lazily Loading
o.item.must_equal o2
o2.items.must_equal [o]
o.related_items.must_equal [o2]
o2.other_item.must_equal o
o.other_items.must_equal [o2]
o.other_related_items.must_equal [o]
o.mtm_items.must_equal [o]
# Eager Loading via eager
os = c.eager(:item, :related_items, :other_items, :other_related_items, :mtm_items).where(:id=>1).all.first
os.item.must_equal o2
os.related_items.must_equal [o2]
os.other_items.must_equal [o2]
os.other_related_items.must_equal [o]
os.mtm_items.must_equal [o]
os = c.eager(:items, :other_item).where(:id=>2).all.first
os.items.must_equal [o]
os.other_item.must_equal o
# Eager Loading via eager_graph
c.eager_graph(:item).where(:items__id=>1).all.first.item.must_equal o2
c.eager_graph(:items).where(:items__id=>2).all.first.items.must_equal [o]
c.eager_graph(:related_items).where(:items__id=>1).all.first.related_items.must_equal [o2]
c.eager_graph(:other_item).where(:items__id=>2).all.first.other_item.must_equal o
c.eager_graph(:other_items).where(:items__id=>1).all.first.other_items.must_equal [o2]
c.eager_graph(:other_related_items).where(:items__id=>1).all.first.other_related_items.must_equal [o]
c.eager_graph(:mtm_items).where(:items__id=>1).all.first.mtm_items.must_equal [o]
# Filter By Associations - Model Instances
c.filter(:item=>o2).all.must_equal [o]
c.filter(:items=>o).all.must_equal [o2]
c.filter(:related_items=>o2).all.must_equal [o]
c.filter(:other_item=>o).all.must_equal [o2]
c.filter(:other_items=>o2).all.must_equal [o]
c.filter(:other_related_items=>o).all.must_equal [o]
c.filter(:mtm_items=>o).all.must_equal [o]
# Filter By Associations - Model Datasets
c.filter(:item=>c.filter(:id=>o2.id)).all.must_equal [o]
c.filter(:items=>c.filter(:id=>o.id)).all.must_equal [o2]
c.filter(:related_items=>c.filter(:id=>o2.id)).all.must_equal [o]
c.filter(:other_item=>c.filter(:id=>o.id)).all.must_equal [o2]
c.filter(:other_items=>c.filter(:id=>o2.id)).all.must_equal [o]
c.filter(:other_related_items=>c.filter(:id=>o.id)).all.must_equal [o]
c.filter(:mtm_items=>c.filter(:id=>o.id)).all.must_equal [o]
end
it 'operations/functions with pg_hstore_ops' do
Sequel.extension :pg_hstore_ops, :pg_array_ops
@db.create_table!(:items){hstore :h1; hstore :h2; hstore :h3; String :t}
@ds.insert(Sequel.hstore('a'=>'b', 'c'=>nil), Sequel.hstore('a'=>'b'), Sequel.hstore('d'=>'e'))
h1 = Sequel.hstore(:h1)
h2 = Sequel.hstore(:h2)
h3 = Sequel.hstore(:h3)
@ds.get(h1['a']).must_equal 'b'
@ds.get(h1['d']).must_equal nil
@ds.get(h2.concat(h3).keys.length).must_equal 2
@ds.get(h1.concat(h3).keys.length).must_equal 3
@ds.get(h2.merge(h3).keys.length).must_equal 2
@ds.get(h1.merge(h3).keys.length).must_equal 3
@ds.get(h1.contain_all(%w'a c')).must_equal true
@ds.get(h1.contain_all(%w'a d')).must_equal false
@ds.get(h1.contain_any(%w'a d')).must_equal true
@ds.get(h1.contain_any(%w'e d')).must_equal false
@ds.get(h1.contains(h2)).must_equal true
@ds.get(h1.contains(h3)).must_equal false
@ds.get(h2.contained_by(h1)).must_equal true
@ds.get(h2.contained_by(h3)).must_equal false
@ds.get(h1.defined('a')).must_equal true
@ds.get(h1.defined('c')).must_equal false
@ds.get(h1.defined('d')).must_equal false
@ds.get(h1.delete('a')['c']).must_equal nil
@ds.get(h1.delete(%w'a d')['c']).must_equal nil
@ds.get(h1.delete(h2)['c']).must_equal nil
@ds.from(Sequel.hstore('a'=>'b', 'c'=>nil).op.each).order(:key).all.must_equal [{:key=>'a', :value=>'b'}, {:key=>'c', :value=>nil}]
@ds.get(h1.has_key?('c')).must_equal true
@ds.get(h1.include?('c')).must_equal true
@ds.get(h1.key?('c')).must_equal true
@ds.get(h1.member?('c')).must_equal true
@ds.get(h1.exist?('c')).must_equal true
@ds.get(h1.has_key?('d')).must_equal false
@ds.get(h1.include?('d')).must_equal false
@ds.get(h1.key?('d')).must_equal false
@ds.get(h1.member?('d')).must_equal false
@ds.get(h1.exist?('d')).must_equal false
@ds.get(h1.hstore.hstore.hstore.keys.length).must_equal 2
@ds.get(h1.keys.length).must_equal 2
@ds.get(h2.keys.length).must_equal 1
@ds.get(h1.akeys.length).must_equal 2
@ds.get(h2.akeys.length).must_equal 1
@ds.from(Sequel.hstore('t'=>'s').op.populate(Sequel::SQL::Cast.new(nil, :items))).select_map(:t).must_equal ['s']
@ds.from(:items___i).select(Sequel.hstore('t'=>'s').op.record_set(:i).as(:r)).from_self(:alias=>:s).select(Sequel.lit('(r).*')).from_self.select_map(:t).must_equal ['s']
@ds.from(Sequel.hstore('t'=>'s', 'a'=>'b').op.skeys.as(:s)).select_order_map(:s).must_equal %w'a t'
@ds.from((Sequel.hstore('t'=>'s', 'a'=>'b').op - 'a').skeys.as(:s)).select_order_map(:s).must_equal %w't'
@ds.get(h1.slice(%w'a c').keys.length).must_equal 2
@ds.get(h1.slice(%w'd c').keys.length).must_equal 1
@ds.get(h1.slice(%w'd e').keys.length).must_equal nil
@ds.from(Sequel.hstore('t'=>'s', 'a'=>'b').op.svals.as(:s)).select_order_map(:s).must_equal %w'b s'
@ds.get(h1.to_array.length).must_equal 4
@ds.get(h2.to_array.length).must_equal 2
@ds.get(h1.to_matrix.length).must_equal 2
@ds.get(h2.to_matrix.length).must_equal 1
@ds.get(h1.values.length).must_equal 2
@ds.get(h2.values.length).must_equal 1
@ds.get(h1.avals.length).must_equal 2
@ds.get(h2.avals.length).must_equal 1
end
end if DB.type_supported?(:hstore)
describe 'PostgreSQL json type' do
before(:all) do
@db = DB
@ds = @db[:items]
@a = [1, 2, {'a'=>'b'}, 3.0]
@h = {'a'=>'b', '1'=>[3, 4, 5]}
@native = DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
end
after do
@db.drop_table?(:items)
end
json_types = [:json]
json_types << :jsonb if DB.server_version >= 90400
json_types.each do |json_type|
json_array_type = "#{json_type}[]"
pg_json = lambda{|v| Sequel.send(:"pg_#{json_type}", v)}
it 'insert and retrieve json values' do
hash_class = json_type == :jsonb ? Sequel::Postgres::JSONBHash : Sequel::Postgres::JSONHash
array_class = json_type == :jsonb ? Sequel::Postgres::JSONBArray : Sequel::Postgres::JSONArray
@db.create_table!(:items){column :j, json_type}
@ds.insert(pg_json.call(@h))
@ds.count.must_equal 1
if @native
rs = @ds.all
v = rs.first[:j]
v.class.must_equal(hash_class)
v.to_hash.must_be_kind_of(Hash)
v.must_equal @h
v.to_hash.must_equal @h
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@ds.delete
@ds.insert(pg_json.call(@a))
@ds.count.must_equal 1
if @native
rs = @ds.all
v = rs.first[:j]
v.class.must_equal(array_class)
v.to_a.must_be_kind_of(Array)
v.must_equal @a
v.to_a.must_equal @a
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'insert and retrieve json[] values' do
@db.create_table!(:items){column :j, json_array_type}
j = Sequel.pg_array([pg_json.call('a'=>1), pg_json.call(['b', 2])])
@ds.insert(j)
@ds.count.must_equal 1
if @native
rs = @ds.all
v = rs.first[:j]
v.class.must_equal(Sequel::Postgres::PGArray)
v.to_a.must_be_kind_of(Array)
v.must_equal j
v.to_a.must_equal j
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'with models' do
@db.create_table!(:items) do
primary_key :id
column :h, json_type
end
c = Class.new(Sequel::Model(@db[:items]))
c.plugin :pg_typecast_on_load, :h unless @native
c.create(:h=>pg_json.call(@h)).h.must_equal @h
c.create(:h=>pg_json.call(@a)).h.must_equal @a
end
it 'use json in bound variables' do
@db.create_table!(:items){column :i, json_type}
@ds.call(:insert, {:i=>pg_json.call(@h)}, {:i=>:$i})
@ds.get(:i).must_equal @h
@ds.delete
@ds.call(:insert, {:i=>pg_json.call('a'=>nil)}, {:i=>:$i})
@ds.get(:i).must_equal pg_json.call('a'=>nil)
@db.create_table!(:items){column :i, json_array_type}
j = Sequel.pg_array([pg_json.call('a'=>1), pg_json.call(['b', 2])], json_type)
@ds.call(:insert, {:i=>j}, {:i=>:$i})
@ds.get(:i).must_equal j
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'operations/functions with pg_json_ops' do
Sequel.extension :pg_json_ops
jo = pg_json.call('a'=>1, 'b'=>{'c'=>2, 'd'=>{'e'=>3}}).op
ja = pg_json.call([2, 3, %w'a b']).op
@db.get(jo['a']).must_equal 1
@db.get(jo['b']['c']).must_equal 2
@db.get(jo[%w'b c']).must_equal 2
@db.get(jo['b'].get_text(%w'd e')).must_equal "3"
@db.get(jo[%w'b d'].get_text('e')).must_equal "3"
@db.get(ja[1]).must_equal 3
@db.get(ja[%w'2 1']).must_equal 'b'
@db.get(jo.extract('a')).must_equal 1
@db.get(jo.extract('b').extract('c')).must_equal 2
@db.get(jo.extract('b', 'c')).must_equal 2
@db.get(jo.extract('b', 'd', 'e')).must_equal 3
@db.get(jo.extract_text('b', 'd')).gsub(' ', '').must_equal '{"e":3}'
@db.get(jo.extract_text('b', 'd', 'e')).must_equal '3'
@db.get(ja.array_length).must_equal 3
@db.from(ja.array_elements.as(:v)).select_map(:v).must_equal [2, 3, %w'a b']
if DB.server_version >= 90400
@db.get(jo.typeof).must_equal 'object'
@db.get(ja.typeof).must_equal 'array'
@db.from(ja.array_elements_text.as(:v)).select_map(:v).map{|s| s.gsub(' ', '')}.must_equal ['2', '3', '["a","b"]']
@db.from(jo.to_record.as(:v, [Sequel.lit('a integer'), Sequel.lit('b text')])).select_map(:a).must_equal [1]
@db.from(pg_json.call([{'a'=>1, 'b'=>1}]).op.to_recordset.as(:v, [Sequel.lit('a integer'), Sequel.lit('b integer')])).select_map(:a).must_equal [1]
if json_type == :jsonb
@db.get(jo.has_key?('a')).must_equal true
@db.get(jo.has_key?('c')).must_equal false
@db.get(pg_json.call(['2', '3', %w'a b']).op.include?('2')).must_equal true
@db.get(pg_json.call(['2', '3', %w'a b']).op.include?('4')).must_equal false
@db.get(jo.contain_all(['a', 'b'])).must_equal true
@db.get(jo.contain_all(['a', 'c'])).must_equal false
@db.get(jo.contain_all(['d', 'c'])).must_equal false
@db.get(jo.contain_any(['a', 'b'])).must_equal true
@db.get(jo.contain_any(['a', 'c'])).must_equal true
@db.get(jo.contain_any(['d', 'c'])).must_equal false
@db.get(jo.contains(jo)).must_equal true
@db.get(jo.contained_by(jo)).must_equal true
@db.get(jo.contains('a'=>1)).must_equal true
@db.get(jo.contained_by('a'=>1)).must_equal false
@db.get(pg_json.call('a'=>1).op.contains(jo)).must_equal false
@db.get(pg_json.call('a'=>1).op.contained_by(jo)).must_equal true
@db.get(ja.contains(ja)).must_equal true
@db.get(ja.contained_by(ja)).must_equal true
@db.get(ja.contains([2,3])).must_equal true
@db.get(ja.contained_by([2,3])).must_equal false
@db.get(pg_json.call([2,3]).op.contains(ja)).must_equal false
@db.get(pg_json.call([2,3]).op.contained_by(ja)).must_equal true
end
end
@db.from(jo.keys.as(:k)).select_order_map(:k).must_equal %w'a b'
@db.from(jo.each).select_order_map(:key).must_equal %w'a b'
@db.from(jo.each).order(:key).select_map(:value).must_equal [1, {'c'=>2, 'd'=>{'e'=>3}}]
@db.from(jo.each_text).select_order_map(:key).must_equal %w'a b'
@db.from(jo.each_text).order(:key).where(:key=>'b').get(:value).gsub(' ', '').must_match(/\{"d":\{"e":3\},"c":2\}|\{"c":2,"d":\{"e":3\}\}/)
Sequel.extension :pg_row_ops
@db.create_table!(:items) do
Integer :a
String :b
end
j = Sequel.pg_json('a'=>1, 'b'=>'c').op
@db.get(j.populate(Sequel.cast(nil, :items)).pg_row[:a]).must_equal 1
@db.get(j.populate(Sequel.cast(nil, :items)).pg_row[:b]).must_equal 'c'
j = Sequel.pg_json([{'a'=>1, 'b'=>'c'}, {'a'=>2, 'b'=>'d'}]).op
@db.from(j.populate_set(Sequel.cast(nil, :items))).select_order_map(:a).must_equal [1, 2]
@db.from(j.populate_set(Sequel.cast(nil, :items))).select_order_map(:b).must_equal %w'c d'
end if DB.server_version >= 90300 && (DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc)
end
end if DB.server_version >= 90200
describe 'PostgreSQL inet/cidr types' do
ipv6_broken = (IPAddr.new('::1'); false) rescue true
before(:all) do
@db = DB
@ds = @db[:items]
@v4 = '127.0.0.1'
@v4nm = '127.0.0.0/8'
@v6 = '2001:4f8:3:ba:2e0:81ff:fe22:d1f1'
@v6nm = '2001:4f8:3:ba::/64'
@ipv4 = IPAddr.new(@v4)
@ipv4nm = IPAddr.new(@v4nm)
unless ipv6_broken
@ipv6 = IPAddr.new(@v6)
@ipv6nm = IPAddr.new(@v6nm)
end
@native = DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
end
after do
@db.drop_table?(:items)
end
it 'insert and retrieve inet/cidr values' do
@db.create_table!(:items){inet :i; cidr :c}
@ds.insert(@ipv4, @ipv4nm)
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.first[:i].must_equal @ipv4
rs.first[:c].must_equal @ipv4nm
rs.first[:i].must_be_kind_of(IPAddr)
rs.first[:c].must_be_kind_of(IPAddr)
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
unless ipv6_broken
@ds.delete
@ds.insert(@ipv6, @ipv6nm)
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.first[:j]
rs.first[:i].must_equal @ipv6
rs.first[:c].must_equal @ipv6nm
rs.first[:i].must_be_kind_of(IPAddr)
rs.first[:c].must_be_kind_of(IPAddr)
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
end
it 'insert and retrieve inet/cidr/macaddr array values' do
@db.create_table!(:items){column :i, 'inet[]'; column :c, 'cidr[]'; column :m, 'macaddr[]'}
@ds.insert(Sequel.pg_array([@ipv4], 'inet'), Sequel.pg_array([@ipv4nm], 'cidr'), Sequel.pg_array(['12:34:56:78:90:ab'], 'macaddr'))
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.first.values.all?{|c| c.is_a?(Sequel::Postgres::PGArray)}.must_equal true
rs.first[:i].first.must_equal @ipv4
rs.first[:c].first.must_equal @ipv4nm
rs.first[:m].first.must_equal '12:34:56:78:90:ab'
rs.first[:i].first.must_be_kind_of(IPAddr)
rs.first[:c].first.must_be_kind_of(IPAddr)
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'use ipaddr in bound variables' do
@db.create_table!(:items){inet :i; cidr :c}
@ds.call(:insert, {:i=>@ipv4, :c=>@ipv4nm}, {:i=>:$i, :c=>:$c})
@ds.get(:i).must_equal @ipv4
@ds.get(:c).must_equal @ipv4nm
@ds.filter(:i=>:$i, :c=>:$c).call(:first, :i=>@ipv4, :c=>@ipv4nm).must_equal(:i=>@ipv4, :c=>@ipv4nm)
@ds.filter(:i=>:$i, :c=>:$c).call(:first, :i=>@ipv6, :c=>@ipv6nm).must_equal nil
@ds.filter(:i=>:$i, :c=>:$c).call(:delete, :i=>@ipv4, :c=>@ipv4nm).must_equal 1
unless ipv6_broken
@ds.call(:insert, {:i=>@ipv6, :c=>@ipv6nm}, {:i=>:$i, :c=>:$c})
@ds.get(:i).must_equal @ipv6
@ds.get(:c).must_equal @ipv6nm
@ds.filter(:i=>:$i, :c=>:$c).call(:first, :i=>@ipv6, :c=>@ipv6nm).must_equal(:i=>@ipv6, :c=>@ipv6nm)
@ds.filter(:i=>:$i, :c=>:$c).call(:first, :i=>@ipv4, :c=>@ipv4nm).must_equal nil
@ds.filter(:i=>:$i, :c=>:$c).call(:delete, :i=>@ipv6, :c=>@ipv6nm).must_equal 1
end
@db.create_table!(:items){column :i, 'inet[]'; column :c, 'cidr[]'; column :m, 'macaddr[]'}
@ds.call(:insert, {:i=>[@ipv4], :c=>[@ipv4nm], :m=>['12:34:56:78:90:ab']}, {:i=>:$i, :c=>:$c, :m=>:$m})
@ds.filter(:i=>:$i, :c=>:$c, :m=>:$m).call(:first, :i=>[@ipv4], :c=>[@ipv4nm], :m=>['12:34:56:78:90:ab']).must_equal(:i=>[@ipv4], :c=>[@ipv4nm], :m=>['12:34:56:78:90:ab'])
@ds.filter(:i=>:$i, :c=>:$c, :m=>:$m).call(:first, :i=>[], :c=>[], :m=>[]).must_equal nil
@ds.filter(:i=>:$i, :c=>:$c, :m=>:$m).call(:delete, :i=>[@ipv4], :c=>[@ipv4nm], :m=>['12:34:56:78:90:ab']).must_equal 1
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'with models' do
@db.create_table!(:items) do
primary_key :id
inet :i
cidr :c
end
c = Class.new(Sequel::Model(@db[:items]))
c.plugin :pg_typecast_on_load, :i, :c unless @native
c.create(:i=>@v4, :c=>@v4nm).values.values_at(:i, :c).must_equal [@ipv4, @ipv4nm]
unless ipv6_broken
c.create(:i=>@ipv6, :c=>@ipv6nm).values.values_at(:i, :c).must_equal [@ipv6, @ipv6nm]
end
end
it 'operations/functions with pg_inet_ops' do
Sequel.extension :pg_inet_ops
@db.get(Sequel.pg_inet_op('1.2.3.4') << '1.2.3.0/24').must_equal true
@db.get(Sequel.pg_inet_op('1.2.3.4') << '1.2.3.4/32').must_equal false
@db.get(Sequel.pg_inet_op('1.2.3.4') << '1.2.2.0/24').must_equal false
@db.get(Sequel.pg_inet_op('1.2.3.4').contained_by('1.2.3.0/24')).must_equal true
@db.get(Sequel.pg_inet_op('1.2.3.4').contained_by('1.2.3.4/32')).must_equal false
@db.get(Sequel.pg_inet_op('1.2.3.4').contained_by('1.2.2.0/24')).must_equal false
@db.get(Sequel.pg_inet_op('1.2.3.4').contained_by_or_equals('1.2.3.0/24')).must_equal true
@db.get(Sequel.pg_inet_op('1.2.3.4').contained_by_or_equals('1.2.3.4/32')).must_equal true
@db.get(Sequel.pg_inet_op('1.2.3.4').contained_by_or_equals('1.2.2.0/24')).must_equal false
@db.get(Sequel.pg_inet_op('1.2.3.0/24') >> '1.2.3.4').must_equal true
@db.get(Sequel.pg_inet_op('1.2.3.0/24') >> '1.2.2.4').must_equal false
@db.get(Sequel.pg_inet_op('1.2.3.0/24').contains('1.2.3.4')).must_equal true
@db.get(Sequel.pg_inet_op('1.2.3.0/24').contains('1.2.2.4')).must_equal false
@db.get(Sequel.pg_inet_op('1.2.3.0/24').contains_or_equals('1.2.3.4')).must_equal true
@db.get(Sequel.pg_inet_op('1.2.3.0/24').contains_or_equals('1.2.2.4')).must_equal false
@db.get(Sequel.pg_inet_op('1.2.3.0/24').contains_or_equals('1.2.3.0/24')).must_equal true
@db.get(Sequel.pg_inet_op('1.2.3.0/32') + 1).must_equal IPAddr.new('1.2.3.1/32')
@db.get(Sequel.pg_inet_op('1.2.3.1/32') - 1).must_equal IPAddr.new('1.2.3.0/32')
@db.get(Sequel.pg_inet_op('1.2.3.1/32') - '1.2.3.0/32').must_equal 1
@db.get(Sequel.pg_inet_op('1.2.3.0/32') & '1.2.0.4/32').must_equal IPAddr.new('1.2.0.0/32')
@db.get(Sequel.pg_inet_op('1.2.0.0/32') | '0.0.3.4/32').must_equal IPAddr.new('1.2.3.4/32')
@db.get(~Sequel.pg_inet_op('0.0.0.0/32')).must_equal IPAddr.new('255.255.255.255/32')
@db.get(Sequel.pg_inet_op('1.2.3.4/24').abbrev).must_equal '1.2.3.4/24'
@db.get(Sequel.pg_inet_op('1.2.3.4/24').broadcast).must_equal IPAddr.new('1.2.3.255/24')
@db.get(Sequel.pg_inet_op('1.2.3.4/24').family).must_equal 4
@db.get(Sequel.pg_inet_op('1.2.3.4/24').host).must_equal '1.2.3.4'
@db.get(Sequel.pg_inet_op('1.2.3.4/24').hostmask).must_equal IPAddr.new('0.0.0.255/32')
@db.get(Sequel.pg_inet_op('1.2.3.4/24').masklen).must_equal 24
@db.get(Sequel.pg_inet_op('1.2.3.4/24').netmask).must_equal IPAddr.new('255.255.255.0/32')
@db.get(Sequel.pg_inet_op('1.2.3.4/24').network).must_equal IPAddr.new('1.2.3.0/24')
@db.get(Sequel.pg_inet_op('1.2.3.4/24').set_masklen(16)).must_equal IPAddr.new('1.2.3.4/16')
@db.get(Sequel.pg_inet_op('1.2.3.4/32').text).must_equal '1.2.3.4/32'
if @db.server_version >= 90400
@db.get(Sequel.pg_inet_op('1.2.3.0/24').contains_or_contained_by('1.2.0.0/16')).must_equal true
@db.get(Sequel.pg_inet_op('1.2.0.0/16').contains_or_contained_by('1.2.3.0/24')).must_equal true
@db.get(Sequel.pg_inet_op('1.3.0.0/16').contains_or_contained_by('1.2.3.0/24')).must_equal false
end
end
end
describe 'PostgreSQL range types' do
before(:all) do
@db = DB
@ds = @db[:items]
@map = {:i4=>'int4range', :i8=>'int8range', :n=>'numrange', :d=>'daterange', :t=>'tsrange', :tz=>'tstzrange'}
@r = {:i4=>1...2, :i8=>2...3, :n=>BigDecimal.new('1.0')..BigDecimal.new('2.0'), :d=>Date.today...(Date.today+1), :t=>Time.local(2011, 1)..Time.local(2011, 2), :tz=>Time.local(2011, 1)..Time.local(2011, 2)}
@ra = {}
@pgr = {}
@pgra = {}
@r.each{|k, v| @ra[k] = Sequel.pg_array([v], @map[k])}
@r.each{|k, v| @pgr[k] = Sequel.pg_range(v)}
@r.each{|k, v| @pgra[k] = Sequel.pg_array([Sequel.pg_range(v)], @map[k])}
@native = DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
end
after do
@db.drop_table?(:items)
end
it 'insert and retrieve range type values' do
@db.create_table!(:items){int4range :i4; int8range :i8; numrange :n; daterange :d; tsrange :t; tstzrange :tz}
[@r, @pgr].each do |input|
h = {}
input.each{|k, v| h[k] = Sequel.cast(v, @map[k])}
@ds.insert(h)
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.first.each do |k, v|
v.class.must_equal(Sequel::Postgres::PGRange)
v.to_range.must_be_kind_of(Range)
v.must_be :==, @r[k]
v.to_range.must_equal @r[k]
end
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@ds.delete
end
end
it 'insert and retrieve arrays of range type values' do
@db.create_table!(:items){column :i4, 'int4range[]'; column :i8, 'int8range[]'; column :n, 'numrange[]'; column :d, 'daterange[]'; column :t, 'tsrange[]'; column :tz, 'tstzrange[]'}
[@ra, @pgra].each do |input|
@ds.insert(input)
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.first.each do |k, v|
v.class.must_equal(Sequel::Postgres::PGArray)
v.to_a.must_be_kind_of(Array)
v.first.class.must_equal(Sequel::Postgres::PGRange)
v.first.to_range.must_be_kind_of(Range)
v.must_be :==, @ra[k].to_a
v.first.must_be :==, @r[k]
end
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@ds.delete
end
end
it 'use range types in bound variables' do
@db.create_table!(:items){int4range :i4; int8range :i8; numrange :n; daterange :d; tsrange :t; tstzrange :tz}
h = {}
@r.keys.each{|k| h[k] = :"$#{k}"}
r2 = {}
@r.each{|k, v| r2[k] = Range.new(v.begin, v.end+2)}
@ds.call(:insert, @r, h)
@ds.first.must_be :==, @r
@ds.filter(h).call(:first, @r).must_be :==, @r
@ds.filter(h).call(:first, @pgr).must_be :==, @r
@ds.filter(h).call(:first, r2).must_equal nil
@ds.filter(h).call(:delete, @r).must_equal 1
@db.create_table!(:items){column :i4, 'int4range[]'; column :i8, 'int8range[]'; column :n, 'numrange[]'; column :d, 'daterange[]'; column :t, 'tsrange[]'; column :tz, 'tstzrange[]'}
@r.each{|k, v| r2[k] = [Range.new(v.begin, v.end+2)]}
@ds.call(:insert, @ra, h)
@ds.filter(h).call(:first, @ra).each{|k, v| v.must_be :==, @ra[k].to_a}
@ds.filter(h).call(:first, @pgra).each{|k, v| v.must_be :==, @ra[k].to_a}
@ds.filter(h).call(:first, r2).must_equal nil
@ds.filter(h).call(:delete, @ra).must_equal 1
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'with models' do
@db.create_table!(:items){primary_key :id; int4range :i4; int8range :i8; numrange :n; daterange :d; tsrange :t; tstzrange :tz}
c = Class.new(Sequel::Model(@db[:items]))
c.plugin :pg_typecast_on_load, :i4, :i8, :n, :d, :t, :tz unless @native
v = c.create(@r).values
v.delete(:id)
v.must_be :==, @r
@db.create_table!(:items){primary_key :id; column :i4, 'int4range[]'; column :i8, 'int8range[]'; column :n, 'numrange[]'; column :d, 'daterange[]'; column :t, 'tsrange[]'; column :tz, 'tstzrange[]'}
c = Class.new(Sequel::Model(@db[:items]))
c.plugin :pg_typecast_on_load, :i4, :i8, :n, :d, :t, :tz unless @native
v = c.create(@ra).values
v.delete(:id)
v.each{|k,v1| v1.must_be :==, @ra[k].to_a}
end
it 'operations/functions with pg_range_ops' do
Sequel.extension :pg_range_ops
@db.get(Sequel.pg_range(1..5, :int4range).op.contains(2..4)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.contains(3..6)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.contains(0..6)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.contained_by(0..6)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.contained_by(3..6)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.contained_by(2..4)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.overlaps(5..6)).must_equal true
@db.get(Sequel.pg_range(1...5, :int4range).op.overlaps(5..6)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.left_of(6..10)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.left_of(5..10)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.left_of(-1..0)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.left_of(-1..3)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.right_of(6..10)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.right_of(5..10)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.right_of(-1..0)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.right_of(-1..3)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.ends_before(6..10)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.ends_before(5..10)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.ends_before(-1..0)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.ends_before(-1..3)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.ends_before(-1..7)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.starts_after(6..10)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.starts_after(5..10)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.starts_after(3..10)).must_equal false
@db.get(Sequel.pg_range(1..5, :int4range).op.starts_after(-1..10)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.starts_after(-1..0)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.starts_after(-1..3)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.starts_after(-5..-1)).must_equal true
@db.get(Sequel.pg_range(1..5, :int4range).op.adjacent_to(6..10)).must_equal true
@db.get(Sequel.pg_range(1...5, :int4range).op.adjacent_to(6..10)).must_equal false
@db.get((Sequel.pg_range(1..5, :int4range).op + (6..10)).adjacent_to(6..10)).must_equal false
@db.get((Sequel.pg_range(1..5, :int4range).op + (6..10)).adjacent_to(11..20)).must_equal true
@db.get((Sequel.pg_range(1..5, :int4range).op * (2..6)).adjacent_to(6..10)).must_equal true
@db.get((Sequel.pg_range(1..4, :int4range).op * (2..6)).adjacent_to(6..10)).must_equal false
@db.get((Sequel.pg_range(1..5, :int4range).op - (2..6)).adjacent_to(2..10)).must_equal true
@db.get((Sequel.pg_range(0..4, :int4range).op - (3..6)).adjacent_to(4..10)).must_equal false
@db.get(Sequel.pg_range(0..4, :int4range).op.lower).must_equal 0
@db.get(Sequel.pg_range(0..4, :int4range).op.upper).must_equal 5
@db.get(Sequel.pg_range(0..4, :int4range).op.isempty).must_equal false
@db.get(Sequel::Postgres::PGRange.empty(:int4range).op.isempty).must_equal true
@db.get(Sequel.pg_range(1..5, :numrange).op.lower_inc).must_equal true
@db.get(Sequel::Postgres::PGRange.new(1, 5, :exclude_begin=>true, :db_type=>:numrange).op.lower_inc).must_equal false
@db.get(Sequel.pg_range(1..5, :numrange).op.upper_inc).must_equal true
@db.get(Sequel.pg_range(1...5, :numrange).op.upper_inc).must_equal false
@db.get(Sequel::Postgres::PGRange.new(1, 5, :db_type=>:int4range).op.lower_inf).must_equal false
@db.get(Sequel::Postgres::PGRange.new(nil, 5, :db_type=>:int4range).op.lower_inf).must_equal true
@db.get(Sequel::Postgres::PGRange.new(1, 5, :db_type=>:int4range).op.upper_inf).must_equal false
@db.get(Sequel::Postgres::PGRange.new(1, nil, :db_type=>:int4range).op.upper_inf).must_equal true
end
end if DB.server_version >= 90200
describe 'PostgreSQL interval types' do
before(:all) do
@db = DB
@ds = @db[:items]
@native = DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
end
after(:all) do
Sequel::Postgres::PG_TYPES.delete(1186)
end
after do
@db.drop_table?(:items)
end
it 'insert and retrieve interval values' do
@db.create_table!(:items){interval :i}
[
['0', '00:00:00', 0, []],
['1', '00:00:01', 1, [[:seconds, 1]]],
['1 microsecond', '00:00:00.000001', 0.000001, [[:seconds, 0.000001]]],
['1 millisecond', '00:00:00.001', 0.001, [[:seconds, 0.001]]],
['1 second', '00:00:01', 1, [[:seconds, 1]]],
['1 minute', '00:01:00', 60, [[:seconds, 60]]],
['1 hour', '01:00:00', 3600, [[:seconds, 3600]]],
['123000 hours', '123000:00:00', 442800000, [[:seconds, 442800000]]],
['1 day', '1 day', 86400, [[:days, 1]]],
['1 week', '7 days', 86400*7, [[:days, 7]]],
['1 month', '1 mon', 86400*30, [[:months, 1]]],
['1 year', '1 year', 31557600, [[:years, 1]]],
['1 decade', '10 years', 31557600*10, [[:years, 10]]],
['1 century', '100 years', 31557600*100, [[:years, 100]]],
['1 millennium', '1000 years', 31557600*1000, [[:years, 1000]]],
['1 year 2 months 3 weeks 4 days 5 hours 6 minutes 7 seconds', '1 year 2 mons 25 days 05:06:07', 31557600 + 2*86400*30 + 3*86400*7 + 4*86400 + 5*3600 + 6*60 + 7, [[:years, 1], [:months, 2], [:days, 25], [:seconds, 18367]]],
['-1 year +2 months -3 weeks +4 days -5 hours +6 minutes -7 seconds', '-10 mons -17 days -04:54:07', -10*86400*30 - 3*86400*7 + 4*86400 - 5*3600 + 6*60 - 7, [[:months, -10], [:days, -17], [:seconds, -17647]]],
['+2 years -1 months +3 weeks -4 days +5 hours -6 minutes +7 seconds', '1 year 11 mons 17 days 04:54:07', 31557600 + 11*86400*30 + 3*86400*7 - 4*86400 + 5*3600 - 6*60 + 7, [[:years, 1], [:months, 11], [:days, 17], [:seconds, 17647]]],
].each do |instr, outstr, value, parts|
@ds.insert(instr)
@ds.count.must_equal 1
if @native
@ds.get(Sequel.cast(:i, String)).must_equal outstr
rs = @ds.all
rs.first[:i].is_a?(ActiveSupport::Duration).must_equal true
rs.first[:i].must_equal ActiveSupport::Duration.new(value, parts)
rs.first[:i].parts.sort_by{|k,v| k.to_s}.reject{|k,v| v == 0}.must_equal parts.sort_by{|k,v| k.to_s}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
@ds.delete
end
end
it 'insert and retrieve interval array values' do
@db.create_table!(:items){column :i, 'interval[]'}
@ds.insert(Sequel.pg_array(['1 year 2 months 3 weeks 4 days 5 hours 6 minutes 7 seconds'], 'interval'))
@ds.count.must_equal 1
if @native
rs = @ds.all
rs.first[:i].is_a?(Sequel::Postgres::PGArray).must_equal true
rs.first[:i].first.is_a?(ActiveSupport::Duration).must_equal true
rs.first[:i].first.must_equal ActiveSupport::Duration.new(31557600 + 2*86400*30 + 3*86400*7 + 4*86400 + 5*3600 + 6*60 + 7, [[:years, 1], [:months, 2], [:days, 25], [:seconds, 18367]])
rs.first[:i].first.parts.sort_by{|k,v| k.to_s}.must_equal [[:years, 1], [:months, 2], [:days, 25], [:seconds, 18367]].sort_by{|k,v| k.to_s}
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
end
end
it 'use intervals in bound variables' do
@db.create_table!(:items){interval :i}
@ds.insert('1 year 2 months 3 weeks 4 days 5 hours 6 minutes 7 seconds')
d = @ds.get(:i)
@ds.delete
@ds.call(:insert, {:i=>d}, {:i=>:$i})
@ds.get(:i).must_equal d
@ds.filter(:i=>:$i).call(:first, :i=>d).must_equal(:i=>d)
@ds.filter(:i=>:$i).call(:first, :i=>'0').must_equal nil
@ds.filter(:i=>:$i).call(:delete, :i=>d).must_equal 1
@db.create_table!(:items){column :i, 'interval[]'}
@ds.call(:insert, {:i=>[d]}, {:i=>:$i})
@ds.filter(:i=>:$i).call(:first, :i=>[d]).must_equal(:i=>[d])
@ds.filter(:i=>:$i).call(:first, :i=>[]).must_equal nil
@ds.filter(:i=>:$i).call(:delete, :i=>[d]).must_equal 1
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'with models' do
@db.create_table!(:items) do
primary_key :id
interval :i
end
c = Class.new(Sequel::Model(@db[:items]))
c.plugin :pg_typecast_on_load, :i, :c unless @native
v = c.create(:i=>'1 year 2 mons 25 days 05:06:07').i
v.is_a?(ActiveSupport::Duration).must_equal true
v.must_equal ActiveSupport::Duration.new(31557600 + 2*86400*30 + 3*86400*7 + 4*86400 + 5*3600 + 6*60 + 7, [[:years, 1], [:months, 2], [:days, 25], [:seconds, 18367]])
v.parts.sort_by{|k,_| k.to_s}.must_equal [[:years, 1], [:months, 2], [:days, 25], [:seconds, 18367]].sort_by{|k,_| k.to_s}
end
end if (begin require 'active_support/duration'; require 'active_support/inflector'; require 'active_support/core_ext/string/inflections'; true; rescue LoadError; false end)
describe 'PostgreSQL row-valued/composite types' do
before(:all) do
@db = DB
Sequel.extension :pg_array_ops, :pg_row_ops
@ds = @db[:person]
@db.create_table!(:address) do
String :street
String :city
String :zip
end
@db.create_table!(:person) do
Integer :id
address :address
end
@db.create_table!(:company) do
Integer :id
column :employees, 'person[]'
end
@db.register_row_type(:address)
@db.register_row_type(Sequel.qualify(:public, :person))
@db.register_row_type(:public__company)
@native = DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
end
after(:all) do
@db.drop_table?(:company, :person, :address)
@db.row_types.clear
@db.reset_conversion_procs if @native
end
after do
[:company, :person, :address].each{|t| @db[t].delete}
end
it 'insert and retrieve row types' do
@ds.insert(:id=>1, :address=>Sequel.pg_row(['123 Sesame St', 'Somewhere', '12345']))
@ds.count.must_equal 1
if @native
# Single row valued type
rs = @ds.all
v = rs.first[:address]
v.class.superclass.must_equal(Sequel::Postgres::PGRow::HashRow)
v.to_hash.must_be_kind_of(Hash)
v.to_hash.must_equal(:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345')
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
# Nested row value type
p = @ds.get(:person)
p[:id].must_equal 1
p[:address].must_equal v
end
end
it 'insert and retrieve row types containing domains' do
begin
@db << "DROP DOMAIN IF EXISTS positive_integer CASCADE"
@db << "CREATE DOMAIN positive_integer AS integer CHECK (VALUE > 0)"
@db.create_table!(:domain_check) do
positive_integer :id
end
@db.register_row_type(:domain_check)
@db.get(@db.row_type(:domain_check, [1])).must_equal(:id=>1)
ensure
@db.drop_table(:domain_check)
@db << "DROP DOMAIN positive_integer"
end
end if DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
it 'insert and retrieve arrays of row types' do
@ds = @db[:company]
@ds.insert(:id=>1, :employees=>Sequel.pg_array([@db.row_type(:person, [1, Sequel.pg_row(['123 Sesame St', 'Somewhere', '12345'])])]))
@ds.count.must_equal 1
if @native
v = @ds.get(:company)
v.class.superclass.must_equal(Sequel::Postgres::PGRow::HashRow)
v.to_hash.must_be_kind_of(Hash)
v[:id].must_equal 1
employees = v[:employees]
employees.class.must_equal(Sequel::Postgres::PGArray)
employees.to_a.must_be_kind_of(Array)
employees.must_equal [{:id=>1, :address=>{:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345'}}]
@ds.delete
@ds.insert(v[:id], v[:employees])
@ds.get(:company).must_equal v
end
end
it 'use row types in bound variables' do
@ds.call(:insert, {:address=>Sequel.pg_row(['123 Sesame St', 'Somewhere', '12345'])}, {:address=>:$address, :id=>1})
@ds.get(:address).must_equal(:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345')
@ds.filter(:address=>Sequel.cast(:$address, :address)).call(:first, :address=>Sequel.pg_row(['123 Sesame St', 'Somewhere', '12345']))[:id].must_equal 1
@ds.filter(:address=>Sequel.cast(:$address, :address)).call(:first, :address=>Sequel.pg_row(['123 Sesame St', 'Somewhere', '12356'])).must_equal nil
@ds.delete
@ds.call(:insert, {:address=>Sequel.pg_row([nil, nil, nil])}, {:address=>:$address, :id=>1})
@ds.get(:address).must_equal(:street=>nil, :city=>nil, :zip=>nil)
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'use arrays of row types in bound variables' do
@ds = @db[:company]
@ds.call(:insert, {:employees=>Sequel.pg_array([@db.row_type(:person, [1, Sequel.pg_row(['123 Sesame St', 'Somewhere', '12345'])])])}, {:employees=>:$employees, :id=>1})
@ds.get(:company).must_equal(:id=>1, :employees=>[{:id=>1, :address=>{:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345'}}])
@ds.filter(:employees=>Sequel.cast(:$employees, 'person[]')).call(:first, :employees=>Sequel.pg_array([@db.row_type(:person, [1, Sequel.pg_row(['123 Sesame St', 'Somewhere', '12345'])])]))[:id].must_equal 1
@ds.filter(:employees=>Sequel.cast(:$employees, 'person[]')).call(:first, :employees=>Sequel.pg_array([@db.row_type(:person, [1, Sequel.pg_row(['123 Sesame St', 'Somewhere', '12356'])])])).must_equal nil
@ds.delete
@ds.call(:insert, {:employees=>Sequel.pg_array([@db.row_type(:person, [1, Sequel.pg_row([nil, nil, nil])])])}, {:employees=>:$employees, :id=>1})
@ds.get(:employees).must_equal [{:address=>{:city=>nil, :zip=>nil, :street=>nil}, :id=>1}]
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'operations/functions with pg_row_ops' do
@ds.insert(:id=>1, :address=>Sequel.pg_row(['123 Sesame St', 'Somewhere', '12345']))
@ds.get(Sequel.pg_row(:address)[:street]).must_equal '123 Sesame St'
@ds.get(Sequel.pg_row(:address)[:city]).must_equal 'Somewhere'
@ds.get(Sequel.pg_row(:address)[:zip]).must_equal '12345'
@ds = @db[:company]
@ds.insert(:id=>1, :employees=>Sequel.pg_array([@db.row_type(:person, [1, Sequel.pg_row(['123 Sesame St', 'Somewhere', '12345'])])]))
@ds.get(Sequel.pg_row(:company)[:id]).must_equal 1
if @native
@ds.get(Sequel.pg_row(:company)[:employees]).must_equal [{:id=>1, :address=>{:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345'}}]
@ds.get(Sequel.pg_row(:company)[:employees][1]).must_equal(:id=>1, :address=>{:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345'})
@ds.get(Sequel.pg_row(:company)[:employees][1][:address]).must_equal(:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345')
end
@ds.get(Sequel.pg_row(:company)[:employees][1][:id]).must_equal 1
@ds.get(Sequel.pg_row(:company)[:employees][1][:address][:street]).must_equal '123 Sesame St'
@ds.get(Sequel.pg_row(:company)[:employees][1][:address][:city]).must_equal 'Somewhere'
@ds.get(Sequel.pg_row(:company)[:employees][1][:address][:zip]).must_equal '12345'
end
describe "#splat and #*" do
before(:all) do
@db.create_table!(:a){Integer :a}
@db.create_table!(:b){a :b; Integer :a}
@db.register_row_type(:a)
@db.register_row_type(:b)
@db[:b].insert(:a=>1, :b=>@db.row_type(:a, [2]))
end
after(:all) do
@db.drop_table?(:b, :a)
end
it "splat should reference the table type" do
@db[:b].select(:a).first.must_equal(:a=>1)
@db[:b].select(:b__a).first.must_equal(:a=>1)
@db[:b].select(Sequel.pg_row(:b)[:a]).first.must_equal(:a=>2)
@db[:b].select(Sequel.pg_row(:b).splat[:a]).first.must_equal(:a=>1)
if @native
@db[:b].select(:b).first.must_equal(:b=>{:a=>2})
@db[:b].select(Sequel.pg_row(:b).splat).first.must_equal(:a=>1, :b=>{:a=>2})
@db[:b].select(Sequel.pg_row(:b).splat(:b)).first.must_equal(:b=>{:a=>1, :b=>{:a=>2}})
end
end
it "* should expand the table type into separate columns" do
ds = @db[:b].select(Sequel.pg_row(:b).splat(:b)).from_self(:alias=>:t)
if @native
ds.first.must_equal(:b=>{:a=>1, :b=>{:a=>2}})
ds.select(Sequel.pg_row(:b).*).first.must_equal(:a=>1, :b=>{:a=>2})
ds.select(Sequel.pg_row(:b)[:b]).first.must_equal(:b=>{:a=>2})
ds.select(Sequel.pg_row(:t__b).*).first.must_equal(:a=>1, :b=>{:a=>2})
ds.select(Sequel.pg_row(:t__b)[:b]).first.must_equal(:b=>{:a=>2})
end
ds.select(Sequel.pg_row(:b)[:a]).first.must_equal(:a=>1)
ds.select(Sequel.pg_row(:t__b)[:a]).first.must_equal(:a=>1)
end
end
describe "with models" do
before(:all) do
class Address < Sequel::Model(:address)
plugin :pg_row
end
class Person < Sequel::Model(:person)
plugin :pg_row
end
class Company < Sequel::Model(:company)
plugin :pg_row
end
@a = Address.new(:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345')
@es = Sequel.pg_array([Person.new(:id=>1, :address=>@a)])
end
after(:all) do
Object.send(:remove_const, :Address) rescue nil
Object.send(:remove_const, :Person) rescue nil
Object.send(:remove_const, :Company) rescue nil
end
it 'insert and retrieve row types as model objects' do
@ds.insert(:id=>1, :address=>@a)
@ds.count.must_equal 1
if @native
# Single row valued type
rs = @ds.all
v = rs.first[:address]
v.must_be_kind_of(Address)
v.must_equal @a
@ds.delete
@ds.insert(rs.first)
@ds.all.must_equal rs
# Nested row value type
p = @ds.get(:person)
p.must_be_kind_of(Person)
p.id.must_equal 1
p.address.must_be_kind_of(Address)
p.address.must_equal @a
end
end
it 'insert and retrieve arrays of row types as model objects' do
@ds = @db[:company]
@ds.insert(:id=>1, :employees=>@es)
@ds.count.must_equal 1
if @native
v = @ds.get(:company)
v.must_be_kind_of(Company)
v.id.must_equal 1
employees = v[:employees]
employees.class.must_equal(Sequel::Postgres::PGArray)
employees.to_a.must_be_kind_of(Array)
employees.must_equal @es
@ds.delete
@ds.insert(v.id, v.employees)
@ds.get(:company).must_equal v
end
end
it 'use model objects in bound variables' do
@ds.call(:insert, {:address=>@a}, {:address=>:$address, :id=>1})
@ds.get(:address).must_equal @a
@ds.filter(:address=>Sequel.cast(:$address, :address)).call(:first, :address=>@a)[:id].must_equal 1
@ds.filter(:address=>Sequel.cast(:$address, :address)).call(:first, :address=>Address.new(:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12356')).must_equal nil
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'use arrays of model objects in bound variables' do
@ds = @db[:company]
@ds.call(:insert, {:employees=>@es}, {:employees=>:$employees, :id=>1})
@ds.get(:company).must_equal Company.new(:id=>1, :employees=>@es)
@ds.filter(:employees=>Sequel.cast(:$employees, 'person[]')).call(:first, :employees=>@es)[:id].must_equal 1
@ds.filter(:employees=>Sequel.cast(:$employees, 'person[]')).call(:first, :employees=>Sequel.pg_array([@db.row_type(:person, [1, Sequel.pg_row(['123 Sesame St', 'Somewhere', '12356'])])])).must_equal nil
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG
it 'model typecasting' do
Person.plugin :pg_typecast_on_load, :address unless @native
a = Address.new(:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345')
o = Person.create(:id=>1, :address=>['123 Sesame St', 'Somewhere', '12345'])
o.address.must_equal a
o = Person.create(:id=>1, :address=>{:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345'})
o.address.must_equal a
o = Person.create(:id=>1, :address=>a)
o.address.must_equal a
Company.plugin :pg_typecast_on_load, :employees unless @native
e = Person.new(:id=>1, :address=>a)
o = Company.create(:id=>1, :employees=>[{:id=>1, :address=>{:street=>'123 Sesame St', :city=>'Somewhere', :zip=>'12345'}}])
o.employees.must_equal [e]
o = Company.create(:id=>1, :employees=>[e])
o.employees.must_equal [e]
end
end
end
describe 'pg_static_cache_updater extension' do
before(:all) do
@db = DB
@db.extension :pg_static_cache_updater
@db.drop_function(@db.default_static_cache_update_name, :cascade=>true, :if_exists=>true)
@db.create_static_cache_update_function
@db.create_table!(:things) do
primary_key :id
String :name
end
@Thing = Class.new(Sequel::Model(:things))
@Thing.plugin :static_cache
@db.create_static_cache_update_trigger(:things)
end
after(:all) do
@db.drop_table(:things)
@db.drop_function(@db.default_static_cache_update_name)
end
it "should reload model static cache when underlying table changes" do
@Thing.all.must_equal []
q = Queue.new
q1 = Queue.new
@db.listen_for_static_cache_updates(@Thing, :timeout=>0, :loop=>proc{q.push(nil); q1.pop.call}, :before_thread_exit=>proc{q.push(nil)})
q.pop
q1.push(proc{@db[:things].insert(1, 'A')})
q.pop
@Thing.all.must_equal [@Thing.load(:id=>1, :name=>'A')]
q1.push(proc{@db[:things].update(:name=>'B')})
q.pop
@Thing.all.must_equal [@Thing.load(:id=>1, :name=>'B')]
q1.push(proc{@db[:things].delete})
q.pop
@Thing.all.must_equal []
q1.push(proc{throw :stop})
q.pop
end
end if DB.adapter_scheme == :postgres && SEQUEL_POSTGRES_USES_PG && DB.server_version >= 90000
describe 'PostgreSQL enum types' do
before do
@db = DB
@db.create_enum(:test_enum, %w'a b c d')
@db.create_table!(:test_enumt) do
test_enum :t
end
end
after do
@db.drop_table?(:test_enumt)
@db.drop_enum(:test_enum)
end
it "should return correct entries in the schema" do
s = @db.schema(:test_enumt)
s.first.last[:type].must_equal :enum
s.first.last[:enum_values].must_equal %w'a b c d'
end
it "should add array parsers for enum values" do
@db.get(Sequel.pg_array(%w'a b', :test_enum)).must_equal %w'a b'
end if DB.adapter_scheme == :postgres || DB.adapter_scheme == :jdbc
it "should set up model typecasting correctly" do
c = Class.new(Sequel::Model(:test_enumt))
o = c.new
o.t = :a
o.t.must_equal 'a'
end
it "should add values to existing enum" do
@db.add_enum_value(:test_enum, 'e')
@db.add_enum_value(:test_enum, 'f', :after=>'a')
@db.add_enum_value(:test_enum, 'g', :before=>'b')
@db.add_enum_value(:test_enum, 'a', :if_not_exists=>true) if @db.server_version >= 90300
@db.schema(:test_enumt, :reload=>true).first.last[:enum_values].must_equal %w'a f g b c d e'
end if DB.server_version >= 90100
end
describe "PostgreSQL stored procedures for datasets" do
before do
require 'sequel/adapters/utils/stored_procedures'
@db = DB
@db.create_table!(:items) do
primary_key :id
integer :numb
end
@db.execute(<<-SQL)
create or replace function insert_item(numb bigint)
returns items.id%type
as $$
declare
l_id items.id%type;
begin
l_id := 1;
insert into items(id, numb) values(l_id, numb);
return l_id;
end;
$$ language plpgsql;
SQL
@ds = @db[:items]
end
after do
@db.drop_function("insert_item", :if_exists=>true)
@db.drop_table?(:items)
end
it "should correctly call stored procedure for inserting record" do
result = @ds.call_sproc(:insert, :insert_item, 100)
result.must_equal nil
@ds.call(:all).must_equal [{:id=>1, :numb=>100}]
end
end if DB.adapter_scheme == :jdbc