require_relative 'helper' require 'stringio' describe 'Adapter' do supported_by Swift::Adapter::Postgres, Swift::Adapter::Mysql, Swift::Adapter::Sqlite3 do describe 'db' do before do @db = Swift.db @db.execute('drop table if exists users') serial = case @db when Swift::Adapter::Sqlite3 then 'integer primary key' else 'serial' end @db.execute %Q{create table users(id #{serial}, name text, email text, created_at timestamp)} end it 'yields db to block' do Swift.db do |db| assert_kind_of Swift::Adapter, db end end it 'records closed state' do assert !Swift.db.closed? Swift.db.close assert Swift.db.closed? end describe 'execute' do it 'executes without bind values' do assert @db.execute %q{select count(*) from users} end it 'executes with bind values' do assert @db.execute 'insert into users (name, created_at) values (?, current_timestamp)', 'Benny Arthurton' end end describe 'escape' do it 'escapes strings' do assert_match %r{foo.'bar}, @db.escape("foo'bar") end end describe 'prepared statements' do it 'executes via Adapter#prepare' do result = [] @db.prepare('select count(*) as n from users').execute.each {|r| result << r[:n] } assert_kind_of Fixnum, result[0] end it 'executes without bind values' do assert @db.prepare(%q{insert into users (name) values ('Apple Arthurton')}).execute end it 'executes with bind values' do assert @db.prepare(%q{insert into users (name) values (?)}).execute('Apple Arthurton') end it 'executes multiple times' do sth = @db.prepare(%q{insert into users (name, created_at) values (?, current_timestamp)}) assert sth.execute('Apple Arthurton') assert sth.execute('Benny Arthurton') end it 'has insert_id' do sql = case @db when Swift::Adapter::Postgres then %q{insert into users (name) values (?) returning id} else %q{insert into users (name) values (?)} end statement = @db.prepare(sql) assert 1, statement.execute('Connie Arthurton').insert_id end end describe 'executed prepared statements' do before do insert = @db.prepare(%q{insert into users (name, created_at) values (?, current_timestamp)}) insert.execute('Apple Arthurton') insert.execute('Benny Arthurton') @sth = @db.prepare('select * from users') @res = @sth.execute end it 'enumerates' do assert_kind_of Enumerable, @res end it 'enumerates block' do begin @sth.execute{|row| row} rescue => error flunk error.message else pass end end it 'returns hash tuples for enumerable methods' do assert_kind_of Hash, @res.first end it 'returns a result set on Adapter#execute{}' do @db.execute('select * from users') {|r| assert_kind_of Hash, r } end it 'returns a result set on Adapter#execute' do assert_respond_to @db.execute('select * from users'), :each end it 'returns fields' do assert_equal [ :id, :name, :email, :created_at ], @res.fields end end describe 'schema information' do it 'should list tables' do assert_equal %w(users), @db.tables end it 'should list fields in a table with types' do expect = {id: 'integer', name: 'text', email: 'text', created_at: 'timestamp'} assert_equal expect, @db.fields('users') end end describe 'bulk writes' do it 'writes from an IO object' do data = StringIO.new "Sally Arthurton\tsally@local\nJonas Arthurton\tjonas@local\n" assert_equal 2, @db.write('users', %w{name email}, data).affected_rows end it 'writes from a string' do data = "Sally Arthurton\tsally@local\nJonas Arthurton\tjonas@local\n" assert_equal 2, @db.write('users', %w{name email}, data).affected_rows end it 'writes with no columns specified' do ts = DateTime.parse('2010-01-01 00:00:00') data = "1\tSally Arthurton\tsally@local\t#{ts}\n" row = {id: 1, name: 'Sally Arthurton', email: 'sally@local', created_at: ts} assert_equal 1, @db.write('users', data).affected_rows assert_equal row, @db.execute('select * from users limit 1').first end end end end end