Aion-Query
view release on metacpan or search on metacpan
lib/Aion/Query.pm view on Meta::CPAN
use config {
DSN => undef,
DRV => 'mysql',
BASE => 'BASE',
HOST => undef,
PORT => undef,
SOCK => undef,
USER => 'root',
PASS => 123,
CONN => undef,
DEBUG => 0,
MAX_QUERY_ERROR => 1000,
BQ => 1,
};
# ФоÑмиÑÑÐµÑ DSN на оÑнове конÑига
our $DEFAULT_DSN;
sub default_dsn() {
$DEFAULT_DSN //= do {
if(defined DSN) {DSN}
elsif(DRV =~ /mysql|mariadb/i) {
my $sock = SOCK;
$sock //= "/var/run/mysqld/mysqld.sock" if !defined HOST;
"DBI:${\ DRV}:database=${\ BASE};${\
(defined(HOST)?
'host=' . HOST
. (defined(PORT)? ':' . PORT: ())
. ';': ()
)
}${\ (defined($sock)? 'mysql_socket=' . $sock: ()) }"
}
elsif(DRV =~ /sqlite/i) { "DBI:${\ DRV}:dbname=${\ BASE}" }
else { die "Using DSN! DRV: ${\ DRV} is'nt supported." }
}
}
my $CONN;
sub default_connect_options() {
return default_dsn, USER, PASS, $CONN //= CONN // do {
if(DRV =~ /mysql|mariadb/i) {[
"SET NAMES utf8",
"SET sql_mode='NO_AUTO_CREATE_USER,NO_ENGINE_SUBSTITUTION'",
]}
else {[]}
};
}
# ÐÐ¾Ð½Ð½ÐµÐºÑ Ðº базе и id коннекÑа
sub base_connect {
my ($dsn, $user, $password, $conn) = @_;
my $base = DBI->connect($dsn, $user, $password, {
RaiseError => 1,
PrintError => 0,
$dsn =~ /^DBI:mysql/i ? (mysql_enable_utf8 => 1): (),
}) or die "Connect to db failed";
$base->do($_) for @$conn;
return $base unless wantarray;
my ($base_connection_id) = $dsn =~ /^DBI:(mysql|mariadb)/i
? $base->selectrow_array("SELECT connection_id()")
: -1;
return $base, $base_connection_id;
}
# ÐÑовеÑка коннекÑа и пеÑеконнекÑ
sub connect_respavn {
my ($base) = @_;
$base->disconnect, undef $base if $base and !$base->ping;
($_[0], $_[1]) = base_connect(default_connect_options) if !$base;
return;
}
# РеÑÑаÑÑ ÐºÐ¾Ð½Ð½ÐµÐºÑа
sub connect_restart {
my ($base, $base_connection_id) = @_;
$base->disconnect if $base;
($_[0], $_[1]) = base_connect(default_connect_options());
return;
}
# ÐниÑиализаÑÐ¸Ñ ÐÐ
our $base; our $base_connection_id;
END {
$base->disconnect if $base;
}
# возможно вÑполнÑеÑÑÑ Ð·Ð°Ð¿ÑÐ¾Ñ - нÑжно его ÑбиÑÑ
sub query_stop {
return if $base_connection_id == -1;
# вÑпомогаÑелÑное подклÑÑение
my $signal = base_connect(default_connect_options());
$signal->do("KILL HARD " . ($base_connection_id + 0));
$signal->disconnect;
return;
}
# ÐапÑоÑÑ Ðº базе
our @DEBUG;
sub sql_debug(@) {
my ($fn, $query) = @_;
my $msg = "$fn: " . (ref $query? np($query): $query);
push @DEBUG, $msg;
print STDERR $msg, "\n" if DEBUG;
}
# sub debug_html {
# join "", map { ("<p class='debug'>", to_html($_), "</p>\n") } @DEBUG;
# }
# sub debug_text {
# return "" if !@DEBUG;
# join "", map { "$_\n\n" } @DEBUG, "";
# }
# sub debug_array {
# return if !@DEBUG;
# $_[0]->{SQL_DEBUG} = \@DEBUG;
lib/Aion/Query.pm view on Meta::CPAN
if(ref $x eq "ARRAY") {
[map _set_type($type, $_), @$x]
}
elsif(ref $x eq "HASH") {
+{ map ($_ => _set_type($type, $type->{$_})), keys %$x }
}
elsif(ref $type eq "SCALAR") {
\_set_type($type, $$x);
}
elsif($type eq "^") {
int $x
}
elsif($type eq "~") {
"$x"
}
elsif($type eq ".") {
$x+1.e-100
}
else {
die "_set_type($type): type does not exist"
}
}
sub _set_params {
my ($query, $param) = @_;
$query =~ s!:([~\.^])?([a-z_]\w*)!
exists $param->{$2}? do {
my $x = $param->{$2};
defined $1 ? quote _set_type($1, $x): quote $x
}: die "The :$1 parameter was not passed."!ige;
$query
}
# ÐÐµÐ»Ð°ÐµÑ Ð¿Ð¾Ð´ÑÑановки
sub query_prepare (@) {
my ($query, %param) = @_;
$query =~ s!
^(?<sep>[\ \t]*) (?<if>\w+)>> [\ \t]* (?<code>.*)
| ^(?<sep>[\ \t]*) (?<for>\w+)\*>> [\ \t]* (?<code>.*)
| (?<param> : [~\.^]? [a-z_]\w*)
!
exists $+{if}? ($param{$+{if}}? $+{sep} . _set_params($+{code}, \%param): ""):
exists $+{for}? do {
my ($sep, $param, $code) = @+{qw/sep for code/};
join "\n", map { local $param{'_'} = $_; _set_params("$sep$code", \%param) } @{$param{$param}}
}:
_set_params($+{param}, \%param)
!imgex;
$query
}
# ÐÑполнÑÐµÑ sql-запÑоÑ
sub query_do($;$) {
my ($query, $columns) = @_;
sql_debug query => $query;
connect_respavn($base, $base_connection_id);
my $res = eval {
if($query =~ /^\s*(select|show|desc(ribe)?)\b/in) {
my $r = @_>1? do {
my $sth = $base->prepare($query);
$sth->execute;
$_[1] = [@{$sth->{NAME}}];
my $res = $sth->fetchall_arrayref({});
$sth->finish;
$res
}: $base->selectall_arrayref($query, { Slice => {} });
if(defined $r and BQ) {
for my $row (@$r) {
for my $k (keys %$row) {
$row->{$k} =~ s/°([^\x7F]{1,7})\x7F/chr from_radix($1, 254)/ge if utf8::is_utf8($row->{$k});
}
}
}
$r
} else {
0 + $base->do($query)
}
};
die +(length($query)>MAX_QUERY_ERROR? substr($query, 0, MAX_QUERY_ERROR) . " ...": $query) . "\n\n$@" if $@;
$res
}
sub query_ref(@) {
my ($query, %kw) = @_;
my $map = delete $kw{MAP};
$query = query_prepare($query, %kw) if @_>1;
my $res = query_do($query);
if($map && ref $res eq "ARRAY") {
eval "require $map" or die unless UNIVERSAL::can($map, "new");
[map { $map->new(%$_) } @$res]
} else {
$res
}
}
sub query(@) {
my $ref = query_ref(@_);
wantarray && ref $ref? @$ref: $ref;
}
# ÐозвÑаÑÐ°ÐµÑ sth
sub query_sth(@) {
my ($query, %kw) = @_;
$query = query_prepare($query, %kw) if @_>1;
my $sth = $base->prepare($query);
$sth->execute;
$sth
}
# ÐÐ»Ñ ÑлайÑа
#
# query_slice word => "id", "SELECT word, id FROM word WHERE word in (1,2,3)" -> { 1 => 10, 2 => 20 }
#
# query_slice word => {}, "SELECT word, id FROM word WHERE word in (1,2,3)" -> { 1 => {id => 10, word => 1} }
#
# query_slice word => ["id"], "SELECT word, id FROM word WHERE word in (1,2,3)" -> { 1 => [10, 20], 2 => [30] }
#
# query_slice word => [], "SELECT word, id FROM word WHERE word in (1,2,3)" -> { 1 => [{id => 10, word => 1}, {id => 20, word => 2}] }
#
# query_slice word => [[]], "SELECT word, id FROM word WHERE word in (1,2,3)" -> [ [{id => 10, word => 1}, {id => 20, word => 2}], ... ]
#
# TODO: query_slice [] => word, "SELECT word, id FROM word WHERE word in (1,2,3)" -> [{id => 10, word => 1}, {id => 20, word => 2}]
#
# TODO: [ "id", "name", "jinni" ] -> [{ id=>1, items => [{ name => "hi!", items => [{ jinni=>2, items => [{...}] }] }] }]
lib/Aion/Query.pm view on Meta::CPAN
# query_col "SELECT id FROM word WHERE word in (1,2,3)" -> [1,2,3]
#
sub query_col(@);
sub query_col(@) {
return [query_col @_] if !wantarray;
my $rows = query_ref(@_);
die "Only one column is acceptable!" if @$rows and 1 != keys %{$rows->[0]};
map { my ($k, $v) = %$_; $v } @$rows
}
# ÐÑбÑаÑÑ ÑÑÑокÑ
#
# query_row_ref "SELECT id, word FROM word WHERE word = 1" -> {id=>1, word=>"ÑеÑебÑо"}
#
sub query_row_ref(@) {
my $rows = query_ref(@_);
die "A few lines!" if @$rows>1;
$rows->[0]
}
# ÐÑбÑаÑÑ ÑÑÑокÑ
#
# ($id, $word) = query_row_ref "SELECT id, word FROM word WHERE word = 1"
#
sub query_row(@) {
return query_row_ref(@_) unless wantarray;
my $sql = query_prepare(@_);
my $rows = query_do($sql, my $columns);
die "A few lines!" if @$rows > 1;
my $row = $rows->[0];
map $row->{$_}, @$columns
}
# ÐÑбÑаÑÑ Ð·Ð½Ð°Ñение
#
# query_scalar "SELECT word FROM word WHERE id = 1" -> "золоÑо"
#
sub query_scalar(@) {
my $rows = query_ref(@_);
die "A few lines!" if @$rows>1;
die "Only one column is acceptable! " . keys %{$rows->[0]} if @$rows and 1 != keys %{$rows->[0]};
my ($k, $v) = %{$rows->[0]};
$v
}
# СоздаÑÑ ÑаÑÑи sql-запÑоÑа Ð´Ð»Ñ ÑоÑÑиÑовки по ÑÑловиÑ, а не лимиÑÑ
#
# ("concat(size,',',likes)", "(size < 10 OR size = 10 AND likes >= 12)", ["size", "likes"]) = make_query_for_order "size desc, likes", "10,12"
#
# ("concat(size,',',likes)", 1) = make_query_for_order "size desc, likes", ""
#
sub make_query_for_order(@) {
my ($order, $next) = @_;
my @orders = split /\s*,\s*/, $order;
my @order_direct;
my @order_sel = map { my $x=$_; push @order_direct, $x=~s/\s+(asc|desc)\s*$//ie ? lc $1: "asc"; $x } @orders;
my $select = @order_sel==1? $order_sel[0]:
_check_drv($base, "mysql|mariadb")?
join("", "concat(", join(",',',", @order_sel), ")"):
join " || ',' || ", @order_sel
;
return $select, 1 if $next eq "";
my @next = split /,/, $next;
$next[$#orders] //= "";
@next = map quote($_), @next;
my @op = map { /^a/ ? ">": "<" } @order_direct;
# id -> id >= next[0]
# id, update -> id > next[0] OR id = next[0] and
my @whr;
for(my $i=0; $i<@orders; $i++) {
my @opr;
for(my $j=0; $j<=$i; $j++) {
#my $eq = $j == $#orders? "=": "";
if($j != $i) {
push @opr, "$order_sel[$j] = $next[$j]";
} elsif($j != $#orders) {
push @opr, "$order_sel[$j] $op[$j] $next[$j]";
} else {
push @opr, "$order_sel[$j] $op[$j]= $next[$j]";
}
}
push @whr, join " AND ", @opr;
}
my $where = join "\nOR ", map "$_", @whr;
return $select, "($where)", \@order_sel;
}
# УÑÑÐ°Ð½Ð°Ð²Ð»Ð¸Ð²Ð°ÐµÑ Ð¸Ð»Ð¸ возвÑаÑÐ°ÐµÑ ÐºÐ»ÑÑ Ð¸Ð· ÑаблиÑÑ settings
sub settings($;$) {
my ($id, $value) = @_;
if(@_ == 1) {
my $v = query_scalar("SELECT value FROM settings WHERE id=:id", id => $id);
return defined($v)? Aion::Format::Json::from_json($v): $v;
}
return remove("settings" => $id) if !defined $value;
store('settings',
id => $id,
value => Aion::Format::Json::to_json($value),
);
}
# возвÑаÑÐ°ÐµÑ Ð·Ð°Ð¿Ð¸ÑÑ Ð¿Ð¾ ÐµÑ pk
sub load_by_id(@) {
my ($tab, $pk, $fields, @options) = @_;
$fields //= "*";
query_row("SELECT $fields FROM $tab WHERE id=:id LIMIT 2", @options, id=>$pk)
}
# ÐÑовеÑÑÐµÑ Ð´ÑÐ°Ð¹Ð²ÐµÑ ÐРна имена
sub _check_drv {
my ($dbh, $drv) = @_;
$dbh->{Driver}{Name} =~ /^($drv)/ain
}
# ÐобавлÑÐµÑ Ð·Ð°Ð¿Ð¸ÑÑ Ð¸ возвÑаÑÐ°ÐµÑ ÐµÑ id
sub insert(@) {
my ($tab, %x) = @_;
if(_check_drv($base, "mysql|mariadb")) {
query "INSERT INTO $tab SET :set", set => \%x;
} else {
stores($tab, [\%x], insert => 1);
}
LAST_INSERT_ID()
}
# ÐбновлÑÐµÑ Ð·Ð°Ð¿Ð¸ÑÑ Ð¿Ð¾ ÐµÑ id
#
# update "tab" => 123, word => 123 -> 6
#
sub update(@) {
my ($tab, $id, %x) = @_;
die "Row $tab.id=$id is not!" if !query "UPDATE $tab SET :set WHERE id=:id", id=>$id, set => \%x;
$id
}
# УдалÑÐµÑ Ð·Ð°Ð¿Ð¸ÑÑ Ð¿Ð¾ ÐµÑ id
#
# remove "tab" => 123 -> 123
#
sub remove(@) {
my ($tab, $id) = @_;
die "Row $tab.id=$id does not exist!" if !query "DELETE FROM $tab WHERE id=:id", id=>$id;
$id
lib/Aion/Query.pm view on Meta::CPAN
{title => "Kiss in night", author_id => 1},
{title => "Mir", author_id => 1},
]},
{name => "Pushkin A.", id => 2, books => []},
{name => "Alice", id => 3, books => [
{title => "Mips as cpu", author_id => 3},
]},
];
$authors # --> $attaches
my $books = [
{title => "Kiss in night", author_id => 1},
{title => "Mips as cpu", author_id => 3},
{title => "Mir", author_id => 1},
];
\@books # --> $books
=head2 query_col ($query, %params)
Returns one column.
query_col "SELECT name FROM author ORDER BY name" # --> ["Alice", "Pushkin A.", "Pushkin A.S."]
eval {query_col "SELECT id, name FROM author"}; $@ # ~> Only one column is acceptable!
=head2 query_row ($query, %params)
Returns one row.
query_row "SELECT name FROM author WHERE id=2" # --> {name => "Pushkin A."}
my ($id, $name) = query_row "SELECT id, name FROM author WHERE id=2";
$id # -> 2
$name # => Pushkin A.
eval { query_row "SELECT id, name FROM author" }; $@ # ~> A few lines!
=head2 query_row_ref ($query, %params)
Like C<query_row>, but always returns a scalar.
my @x = query_row_ref "SELECT name FROM author WHERE id=2";
\@x # --> [{name => "Pushkin A."}]
eval {query_row_ref "SELECT name FROM author"}; $@ # ~> A few lines!
=head2 query_scalar ($query, %params)
Returns the first value. The query must return one row, otherwise it throws an exception.
query_scalar "SELECT name FROM author WHERE id=2" # => Pushkin A.
=head2 make_query_for_order ($order, $next)
Creates a page request condition not by offset, but by B<cursor pagination>.
To do this, it receives C<$order> of the SQL query and C<$next> - a link to the next page.
my ($select, $where, $order_sel) = make_query_for_order "name DESC, id ASC", undef;
$select # => name || ',' || id
$where # -> 1
$order_sel # -> undef
my @rows = query "SELECT $select as next FROM author WHERE $where LIMIT 2";
my $last = pop @rows;
($select, $where, $order_sel) = make_query_for_order "name DESC, id ASC", $last->{next};
$select # => name || ',' || id
$where # => (name < 'Pushkin A.'\nOR name = 'Pushkin A.' AND id >= '2')
$order_sel # --> [qw/name id/]
See also:
=over
=item 1. Article [Paging pages on social networks
sch(https://habr.com/ru/articles/674714/).
=item 2. LLL<https://metacpan.org/dist/SQL-SimpleOps/view/lib/SQL/SimpleOps.pod#SelectCursor>
=back
=head2 settings ($id, $value)
Sets or returns a key from the C<settings> table.
query "CREATE TABLE settings(
id TEXT PRIMARY KEY,
value TEXT NOT NULL
)";
settings "x1" # -> undef
settings "x1", 10 # -> 1
settings "x1" # -> 10
=head2 load_by_id ($tab, $pk, $fields, @options)
Returns a record by its ID.
load_by_id author => 2 # --> {id=>2, name=>"Pushkin A."}
load_by_id author => 2, "name as n" # --> {n=>"Pushkin A."}
load_by_id author => 2, "id+:x as n", x => 10 # --> {n=>12}
=head2 insert ($tab, %x)
Adds an entry and returns its ID.
insert 'author', name => 'Masha' # -> 4
=head2 update ($tab, $id, %params)
Updates a record by its ID and returns that ID.
update author => 3, name => 'Sasha' # -> 3
eval { update author => 5, name => 'Sasha' }; $@ # ~> Row author.id=5 is not!
=head2 remove ($tab, $id)
Delete a row from a table by its identifier and return this identifier.
remove "author", 4 # -> 4
eval { remove author => 4 }; $@ # ~> Row author.id=4 does not exist!
=head2 query_id ($tab, %params)
Returns an identifier based on other fields.
query_id 'author', name => 'Pushkin A.' # -> 2
( run in 0.805 second using v1.01-cache-2.11-cpan-df04353d9ac )