use Test::More; use strict; use warnings; use utf8; use Encode qw/encode_utf8 decode_utf8/; use FindBin; use lib "$FindBin::Bin/common"; BEGIN { eval { require DBD::SQLite; 1 } or plan skip_all => 'DBD::SQLite required'; eval { DBD::SQLite->VERSION >= 1.25 } or plan skip_all => 'DBD::SQLite >= 1.25 required'; plan 'no_plan'; use_ok('DBIx::Custom::Next'); } $SIG{__WARN__} = sub { warn $_[0] unless $_[0] =~ /DEPRECATED/}; sub test { print "# $_[0]\n" } use DBIx::Custom::Next; { package DBIx::Custom::Next; has dsn => sub { 'dbi:SQLite:dbname=:memory:' } } # Constant my $create_table1 = 'create table table1 (key1 varchar, key2 varchar);'; my $create_table_quote = 'create table "table" ("select" varchar, "update" varchar)'; my $q = '"'; my $p = '"'; # Variables my $dbi; my $result; my $row; my $rows; my $binary; my $model; # Prepare table $dbi = DBIx::Custom::Next->connect; ### SQLite only test test 'option default'; $dbi = DBIx::Custom::Next->new; is_deeply($dbi->option, {}); test 'prefix'; $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table1') }; $dbi->execute('create table table1 (key1 varchar, key2 varchar, primary key(key1));'); $dbi->insert({key1 => 1, key2 => 2}, table => 'table1'); $dbi->insert({key1 => 1, key2 => 4}, table => 'table1', prefix => 'or replace'); $result = $dbi->execute('select * from table1;'); $rows = $result->all; is_deeply($rows, [{key1 => 1, key2 => 4}], "basic"); test 'insert created_at and updated_at scalar reference'; $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table1') }; $dbi->execute('create table table1 (key1, key2, key3)'); $dbi->now(\"datetime('now')"); $dbi->insert({key1 => \"datetime('now')"}, created_at => 'key2', updated_at => 'key3', table => 'table1'); $result = $dbi->select(table => 'table1'); $row = $result->one; is($row->{key1}, $row->{key2}); is($row->{key1}, $row->{key3}); $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table1') }; $dbi->execute('create table table1 (key1, key2, key3)'); $dbi->now(\"datetime('now')"); $model = $dbi->create_model(created_at => 'key2', updated_at => 'key3', table => 'table1'); $model->insert({key1 => \"datetime('now')"}); $result = $dbi->select(table => 'table1'); $row = $result->one; is($row->{key1}, $row->{key2}); is($row->{key1}, $row->{key3}); test 'insert created_at and updated_at scalar reference'; $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table1') }; $dbi->execute('create table table1 (key1, key2, key3)'); $dbi->now(\"datetime('now')"); $dbi->insert({key1 => \"datetime('now')"}, created_at => 'key2', updated_at => 'key3', table => 'table1'); $result = $dbi->select(table => 'table1'); $row = $result->one; is($row->{key1}, $row->{key2}); is($row->{key1}, $row->{key3}); test 'update updated_at scalar reference'; $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table1') }; $dbi->execute('create table table1 (key1, key2)'); $dbi->now(\"datetime('now')"); $dbi->insert({key1 => \"datetime('now')"}, updated_at => 'key2', table => 'table1'); $result = $dbi->select(table => 'table1'); $row = $result->one; is($row->{key1}, $row->{key2}); test 'update_or_insert created_at and updated_at'; eval { $dbi->execute('drop table table1') }; $dbi->execute('create table table1 (key1, key2, key3, key4)'); $dbi->now(\"datetime('now')"); $model = $dbi->create_model(created_at => 'key2', updated_at => 'key3', table => 'table1', primary_key => 'key4'); $model->update_or_insert({key1 => \"datetime('now')"}, id => 1); $result = $model->select(table => 'table1', id => 1); $row = $result->one; is($row->{key1}, $row->{key2}); is($row->{key1}, $row->{key3}); $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table1') }; $dbi->execute('create table table1 (key1, key2)'); $dbi->now(\"datetime('now')"); $model = $dbi->create_model(updated_at => 'key2', table => 'table1'); $model->insert({key1 => \"datetime('now')"}); $result = $dbi->select(table => 'table1'); $row = $result->one; is($row->{key1}, $row->{key2}); test 'DBIX_CUSTOM_DEBUG ok'; { local $ENV{DBIX_CUSTOM_DEBUG} = 1; $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table1') }; my $error; local $SIG{__WARN__} = sub { $error = shift; }; $dbi->execute('create table table1 (key1 varchar, key2 varchar, primary key(key1));'); ok($error); } test 'quote'; $dbi = DBIx::Custom::Next->connect; $dbi->quote('"'); eval { $dbi->execute("drop table ${q}table$p") }; $dbi->execute($create_table_quote); $dbi->insert({select => 1}, table => 'table'); $dbi->delete(table => 'table', where => {select => 1}); $result = $dbi->execute("select * from ${q}table$p"); $rows = $result->all; is_deeply($rows, [], "quote"); test 'finish statement handle'; $dbi = DBIx::Custom::Next->connect; $dbi->execute($create_table1); $dbi->insert({key1 => 1, key2 => 2}, table => 'table1'); $dbi->insert({key1 => 3, key2 => 4}, table => 'table1'); $result = $dbi->select(table => 'table1'); $row = $result->fetch_first; is_deeply($row, [1, 2], "row"); $row = $result->fetch; ok(!$row, "finished"); $result = $dbi->select(table => 'table1'); $row = $result->fetch_hash_first; is_deeply($row, {key1 => 1, key2 => 2}, "row"); $row = $result->fetch_hash; ok(!$row, "finished"); $dbi->execute('create table table2 (key1, key2);'); $result = $dbi->select(table => 'table2'); $row = $result->fetch_hash_first; ok(!$row, "no row fetch"); $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table1') }; $dbi->execute($create_table1); $dbi->insert({key1 => 1, key2 => 2}, table => 'table1'); $dbi->insert({key1 => 3, key2 => 4}, table => 'table1'); $dbi->insert({key1 => 5, key2 => 6}, table => 'table1'); $dbi->insert({key1 => 7, key2 => 8}, table => 'table1'); $dbi->insert({key1 => 9, key2 => 10}, table => 'table1'); $result = $dbi->select(table => 'table1'); $rows = $result->fetch_multi(2); is_deeply($rows, [[1, 2], [3, 4]], "fetch_multi first"); $rows = $result->fetch_multi(2); is_deeply($rows, [[5, 6], [7, 8]], "fetch_multi secound"); $rows = $result->fetch_multi(2); is_deeply($rows, [[9, 10]], "fetch_multi third"); $rows = $result->fetch_multi(2); ok(!$rows); $result = $dbi->select(table => 'table1'); eval {$result->fetch_multi}; like($@, qr/Row count must be specified/, "Not specified row count"); $result = $dbi->select(table => 'table1'); $rows = $result->fetch_hash_multi(2); is_deeply($rows, [{key1 => 1, key2 => 2}, {key1 => 3, key2 => 4}], "fetch_multi first"); $rows = $result->fetch_hash_multi(2); is_deeply($rows, [{key1 => 5, key2 => 6}, {key1 => 7, key2 => 8}], "fetch_multi secound"); $rows = $result->fetch_hash_multi(2); is_deeply($rows, [{key1 => 9, key2 => 10}], "fetch_multi third"); $rows = $result->fetch_hash_multi(2); ok(!$rows); $result = $dbi->select(table => 'table1'); eval {$result->fetch_hash_multi}; like($@, qr/Row count must be specified/, "Not specified row count"); test 'bind_type option'; $binary = pack("I3", 1, 2, 3); $dbi = DBIx::Custom::Next->connect(option => {sqlite_unicode => 1}); $dbi->execute('create table table1(key1, key2)'); $dbi->insert({key1 => $binary, key2 => 'あ'}, table => 'table1', bind_type => [key1 => DBI::SQL_BLOB]); $result = $dbi->select(table => 'table1'); $row = $result->one; is_deeply($row, {key1 => $binary, key2 => 'あ'}, "basic"); $result = $dbi->execute('select length(key1) as key1_length from table1'); $row = $result->one; is($row->{key1_length}, length $binary); test 'type_rule from'; $dbi = DBIx::Custom::Next->connect; $dbi->type_rule( from1 => { date => sub { uc $_[0] } } ); $dbi->execute("create table table1 (key1 Date, key2 datetime)"); $dbi->insert({key1 => 'a'}, table => 'table1'); $result = $dbi->select(table => 'table1'); is($result->fetch_first->[0], 'A'); $result = $dbi->select(table => 'table1'); is($result->one->{key1}, 'A'); test 'select limit'; eval { $dbi->execute('drop table table1') }; $dbi->execute($create_table1); $dbi->insert({key1 => 1, key2 => 2}, table => 'table1'); $dbi->insert({key1 => 3, key2 => 4}, table => 'table1'); $rows = $dbi->select(table => 'table1', append => "order by key1 desc limit 1")->all; is_deeply($rows, [{key1 => 3, key2 => 4}], "append statement"); # DEPRECATED! test test 'filter __ expression'; $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute('drop table table2') }; eval { $dbi->execute('drop table table3') }; $dbi->execute('create table table2 (id, name, table3_id)'); $dbi->execute('create table table3 (id, name)'); $dbi->insert({id => 1, name => 'a', table3_id => 2}, table => 'table2'); $dbi->insert({id => 2, name => 'b'}, table => 'table3'); $result = $dbi->select( table => 'table2', join => "inner join table3 on table2.table3_id = table3.id", column => ['table3.name as table3__name'] ); is($result->fetch_first->[0], 'b'); $result = $dbi->select( table => 'table2', join => "inner join table3 on table2.table3_id = table3.id", column => ['table3.name as table3__name'] ); is($result->fetch_first->[0], 'b'); $result = $dbi->select( table => 'table2', join => "inner join table3 on table2.table3_id = table3.id", column => ['table3.name as "table3.name"'] ); is($result->fetch_first->[0], 'b'); test 'quote'; $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute("drop table ${q}table$p") }; $dbi->quote('"'); $dbi->execute($create_table_quote); $dbi->insert({select => 1}, table => 'table'); $dbi->update({update => 2}, table => 'table', where => {'table.select' => 1}); $result = $dbi->execute("select * from ${q}table$p"); $rows = $result->all; is_deeply($rows, [{select => 1, update => 2}]); test 'join function'; $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute("drop table table1") }; eval { $dbi->execute("drop table table2") }; $dbi->execute($create_table1); $dbi->execute("create table table2 (key1, key3)"); $dbi->insert({key1 => 1, key2 => 2}, table => 'table1'); $dbi->insert({key1 => 1, key3 => 4}, table => 'table2'); $dbi->insert({key1 => 1, key3 => 1}, table => 'table2'); $result = $dbi->select( table => 'table1', column => [{table2 => ['key3']}], join => [ "left outer join table2 on coalesce(table1.key1, 0) = coalesce(table2.key1, 0) and table2.key3 > '3'" ] ); is_deeply($result->all, [{"table2.key3" => 4}]); $dbi = DBIx::Custom::Next->connect; eval { $dbi->execute("drop table table1") }; eval { $dbi->execute("drop table table2") }; $dbi->execute($create_table1); $dbi->execute("create table table2 (key1, key3)"); $dbi->insert({key1 => 1, key2 => 2}, table => 'table1'); $dbi->insert({key1 => 1, key3 => 4}, table => 'table2'); $dbi->insert({key1 => 1, key3 => 1}, table => 'table2'); $result = $dbi->select( table => 'table1', column => [{table2 => ['key3']}], join => [ "left outer join table2 on table2.key3 > '3' and coalesce(table1.key1, 0) = coalesce(table2.key1, 0)" ] ); is_deeply($result->all, [{"table2.key3" => 4}]);