The Perl Toolchain Summit 2025 Needs You: You can help 🙏 Learn more

use strict;
use utf8;
new => 0,
wo => [qw/distinct for_update/],
rw => [qw/prefix/],
ro => [qw/quote_char name_sep new_line strict/],
);
use Scalar::Util ();
sub offset {
if (@_==1) {
return $_[0]->{offset};
} else {
$_[0]->{offset} = $_[1];
return $_[0];
}
}
sub limit {
if (@_==1) {
$_[0]->{limit};
} else {
$_[0]->{limit} = $_[1];
return $_[0];
}
}
sub new {
my $class = shift;
my %args = @_ == 1 ? %{$_[0]} : @_;
my $self = bless {
select => +[],
distinct => 0,
select_map => +{},
select_map_reverse => +{},
from => +[],
joins => +[],
index_hint => +{},
group_by => +[],
order_by => +[],
prefix => 'SELECT ',
new_line => "\n",
strict => 0,
%args
}, $class;
return $self;
}
sub new_condition {
my $self = shift;
SQL::Maker::Condition->new(
quote_char => $self->{quote_char},
name_sep => $self->{name_sep},
strict => $self->{strict},
);
}
sub bind {
my $self = shift;
my @bind;
push @bind, @{$self->{subqueries}} if $self->{subqueries};
push @bind, $self->{where}->bind if $self->{where};
push @bind, $self->{having}->bind if $self->{having};
return wantarray ? @bind : \@bind;
}
sub add_select {
my ($self, $term, $col) = @_;
push @{ $self->{select} }, $term;
if ($col) {
$col ||= $term;
$self->{select_map_reverse}->{$col} = $term;
}
return $self;
}
sub add_from {
my ($self, $table, $alias) = @_;
if ( Scalar::Util::blessed( $table ) and $table->can('as_sql') ) {
push @{ $self->{subqueries} }, $table->bind;
push @{$self->{from}}, [ \do{ '(' . $table->as_sql . ')' }, $alias ];
}
else {
push @{$self->{from}}, [$table, $alias];
}
return $self;
}
sub add_join {
my ($self, $table_ref, $joins) = @_;
my ($table, $alias) = ref($table_ref) eq 'ARRAY' ? @$table_ref : ($table_ref);
if ( Scalar::Util::blessed( $table ) and $table->can('as_sql') ) {
push @{ $self->{subqueries} }, $table->bind;
$table = \do{ '(' . $table->as_sql . ')' };
}
if ( Scalar::Util::blessed( $joins->{table} ) and $joins->{table}->can('as_sql') ) {
push @{ $self->{subqueries} }, $joins->{table}->bind;
$joins->{table} = \do{ '(' . $joins->{table}->as_sql . ')' };
}
push @{ $self->{joins} }, {
table => [ $table, $alias ],
joins => $joins,
};
return $self;
}
sub add_index_hint {
my ($self, $table, $hint) = @_;
my ($type, $list);
if (ref $hint eq 'HASH') {
# { type => '...', list => ['foo'] }
$type = $hint->{type} || 'USE';
$list = ref($hint->{list}) eq 'ARRAY' ? $hint->{list} : [ $hint->{list} ];
} else {
# ['foo, 'bar'] or just 'foo'
$type = 'USE';
$list = ref($hint) eq 'ARRAY' ? $hint : [ $hint ];
}
$self->{index_hint}->{$table} = {
type => $type,
list => $list,
};
return $self;
}
sub _quote {
my ($self, $label) = @_;
return $$label if ref $label eq 'SCALAR';
SQL::Maker::Util::quote_identifier($label, $self->{quote_char}, $self->{name_sep})
}
sub as_sql {
my $self = shift;
my $sql = '';
my $new_line = $self->new_line;
if (@{ $self->{select} }) {
$self->{select_map} = { reverse %{$self->{select_map_reverse}} };
$sql .= $self->{prefix};
$sql .= 'DISTINCT ' if $self->{distinct};
$sql .= join(', ', map {
my $alias = $self->{select_map}->{$_};
if (!$alias) {
$self->_quote($_)
} elsif ($alias && $_ =~ /(?:^|\.)\Q$alias\E$/) {
$self->_quote($_)
} else {
$self->_quote($_) . ' AS ' . $self->_quote($alias)
}
} @{ $self->{select} }) . $new_line;
}
$sql .= 'FROM ';
## Add any explicit JOIN statements before the non-joined tables.
if ($self->{joins} && @{ $self->{joins} }) {
my $initial_table_written = 0;
for my $j (@{ $self->{joins} }) {
my ($table, $join) = map { $j->{$_} } qw( table joins );
$table = $self->_add_index_hint(@$table); ## index hint handling
$sql .= $table unless $initial_table_written++;
$sql .= ' ' . uc($join->{type}) if $join->{type};
$sql .= ' JOIN ' . $self->_quote($join->{table});
$sql .= ' ' . $self->_quote($join->{alias}) if $join->{alias};
if ( defined $join->{condition} ) {
if (ref $join->{condition} && ref $join->{condition} eq 'ARRAY') {
$sql .= ' USING ('. join(', ', map { $self->_quote($_) } @{ $join->{condition} }) . ')';
}
elsif (ref $join->{condition} && ref $join->{condition} eq 'HASH') {
my @conds;
for my $key (keys %{ $join->{condition} }) {
push @conds, $self->_quote($key) . ' = ' . $self->_quote($join->{condition}{$key});
}
$sql .= ' ON ' . join(' AND ', @conds);
}
else {
$sql .= ' ON ' . $join->{condition};
}
}
}
$sql .= ', ' if @{ $self->{from} };
}
if ($self->{from} && @{ $self->{from} }) {
$sql .= join ', ',
map { $self->_add_index_hint($_->[0], $_->[1]) }
@{ $self->{from} };
}
$sql .= $new_line;
$sql .= $self->as_sql_where() if $self->{where};
$sql .= $self->as_sql_group_by if $self->{group_by};
$sql .= $self->as_sql_having if $self->{having};
$sql .= $self->as_sql_order_by if $self->{order_by};
$sql .= $self->as_sql_limit if defined $self->{limit};
$sql .= $self->as_sql_for_update;
$sql =~ s/${new_line}+$//;
return $sql;
}
sub as_sql_limit {
my $self = shift;
my $n = $self->{limit};
return '' unless defined $n;
die "Non-numerics in limit clause ($n)" if $n =~ /\D/;
return sprintf "LIMIT %d%s" . $self->new_line, $n,
($self->{offset} ? " OFFSET " . int($self->{offset}) : "");
}
sub add_order_by {
my ($self, $col, $type) = @_;
push @{$self->{order_by}}, [$col, $type];
return $self;
}
sub as_sql_order_by {
my ($self) = @_;
my @attrs = @{$self->{order_by}};
return '' unless @attrs;
return 'ORDER BY '
. join(', ', map {
my ($col, $type) = @$_;
if (ref $col) {
$$col
} else {
$type ? $self->_quote($col) . " $type" : $self->_quote($col)
}
} @attrs)
. $self->new_line;
}
sub add_group_by {
my ($self, $group, $order) = @_;
push @{$self->{group_by}}, $order ? $self->_quote($group) . " $order" : $self->_quote($group);
return $self;
}
sub as_sql_group_by {
my ($self,) = @_;
my $elems = $self->{group_by};
return '' if @$elems == 0;
return 'GROUP BY '
. join(', ', @$elems)
. $self->new_line;
}
sub set_where {
my ($self, $where) = @_;
$self->{where} = $where;
return $self;
}
sub add_where {
my ($self, $col, $val) = @_;
$self->{where} ||= $self->new_condition();
$self->{where}->add($col, $val);
return $self;
}
sub add_where_raw {
my ($self, $term, $bind) = @_;
$self->{where} ||= $self->new_condition();
$self->{where}->add_raw($term, $bind);
return $self;
}
sub as_sql_where {
my $self = shift;
my $where = $self->{where}->as_sql(undef, sub { $self->_quote($_[0]) });
$where ? "WHERE $where" . $self->new_line : '';
}
sub as_sql_having {
my $self = shift;
if ($self->{having}) {
'HAVING ' . $self->{having}->as_sql . $self->new_line;
} else {
''
}
}
sub add_having {
my ($self, $col, $val) = @_;
if (my $orig = $self->{select_map_reverse}->{$col}) {
$col = $orig;
}
$self->{having} ||= $self->new_condition();
$self->{having}->add($col, $val);
return $self;
}
sub add_having_raw {
my ($self, $term, $bind) = @_;
$self->{having} ||= $self->new_condition();
$self->{having}->add_raw($term, $bind);
return $self;
}
sub as_sql_for_update {
my $self = shift;
$self->{for_update} ? ' FOR UPDATE' : '';
}
sub _add_index_hint {
my ($self, $tbl_name, $alias) = @_;
my $quoted = $alias ? $self->_quote($tbl_name) . ' ' . $self->_quote($alias) : $self->_quote($tbl_name);
my $hint = $self->{index_hint}->{$tbl_name};
return $quoted unless $hint && ref($hint) eq 'HASH';
if ($hint->{list} && @{ $hint->{list} }) {
return $quoted . ' ' . uc($hint->{type} || 'USE') . ' INDEX (' .
join (',', map { $self->_quote($_) } @{ $hint->{list} }) .
')';
}
return $quoted;
}
1;
__END__
=head1 NAME
SQL::Maker::Select - dynamic SQL generator
=head1 SYNOPSIS
my $sql = SQL::Maker::Select->new()
->add_select('foo')
->add_select('bar')
->add_select('baz')
->add_from('table_name')
->as_sql;
# => "SELECT foo, bar, baz FROM table_name"
=head1 DESCRIPTION
=head1 METHODS
=over 4
=item C<< my $sql = $stmt->as_sql(); >>
Render the SQL string.
=item C<< my @bind = $stmt->bind(); >>
Get the bind variables.
=item C<< $stmt->add_select('*') >>
=item C<< $stmt->add_select($col => $alias) >>
=item C<< $stmt->add_select(\'COUNT(*)' => 'cnt') >>
Add a new select term. It's automatically quoted.
=item C<< $stmt->add_from($table :Str | $select :SQL::Maker::Select) : SQL::Maker::Select >>
Add a new FROM clause. You can specify the table name or an instance of L<SQL::Maker::Select> for a sub-query.
I<Return:> $stmt itself.
=item C<< $stmt->add_join(user => {type => 'inner', table => 'config', condition => 'user.user_id = config.user_id'}); >>
=item C<< $stmt->add_join(user => {type => 'inner', table => 'config', condition => {'user.user_id' => 'config.user_id'}); >>
=item C<< $stmt->add_join(user => {type => 'inner', table => 'config', condition => ['user_id']}); >>
=item C<< $stmt->add_join(user => {type => 'inner', table => $select :SQL::Maker::Select, condition => 'user.user_id = config.user_id'}); >>
Add a new JOIN clause. If you pass an arrayref for 'condition' then it uses 'USING'. If 'type' is omitted
it falls back to plain JOIN.
You can specify the table name or an instance of L<SQL::Maker::Select> for a sub-query in 'table' argument.
my $stmt = SQL::Maker::Select->new();
$stmt->add_join(
user => {
type => 'inner',
table => 'config',
condition => 'user.user_id = config.user_id',
}
);
$stmt->as_sql();
# => 'FROM user INNER JOIN config ON user.user_id = config.user_id'
my $stmt = SQL::Maker::Select->new(quote_char => '`', name_sep => '.');
$stmt->add_join(
user => {
type => 'inner',
table => 'config',
condition => {'user.user_id' => 'config.user_id'},
}
);
$stmt->as_sql();
# => 'FROM `user` INNER JOIN `config` ON `user`.`user_id` = `config`.`user_id`'
my $stmt = SQL::Maker::Select->new();
$stmt->add_select('name');
$stmt->add_join(
user => {
type => 'inner',
table => 'config',
condition => ['user_id'],
}
);
$stmt->as_sql();
# => 'SELECT name FROM user INNER JOIN config USING (user_id)'
my $subquery = SQL::Maker::Select->new();
$subquery->add_select('*');
$subquery->add_from( 'foo' );
$subquery->add_where( 'hoge' => 'fuga' );
my $stmt = SQL::Maker::Select->new();
$stmt->add_join(
[ $subquery, 'bar' ] => {
type => 'inner',
table => 'baz',
alias => 'b1',
condition => 'bar.baz_id = b1.baz_id'
},
);
$stmt->as_sql;
# => "FROM (SELECT * FROM foo WHERE (hoge = ?)) bar INNER JOIN baz b1 ON bar.baz_id = b1.baz_id";
my $subquery1 = SQL::Maker::Select->new();
$subquery1->add_select('*');
$subquery1->add_from( 'foo' );
$subquery1->add_where( 'hoge' => 'fuga' );
my $subquery2 = SQL::Maker::Select->new();
$subquery2->add_select('*');
$subquery2->add_from( 'bar' );
$subquery2->add_where( 'piyo' => 'gera' );
my $stmt = SQL::Maker::Select->new();
$stmt->add_join(
[ $subquery1, 'f1' ] => {
type => 'inner',
table => $subquery2,
alias => 'b1',
condition => 'f1.baz_id = b1.baz_id'
},
);
$stmt->as_sql;
# => "FROM (SELECT * FROM foo WHERE (hoge = ?)) f1 INNER JOIN (SELECT * FROM bar WHERE (piyo = ?)) b1 ON f1.baz_id = b1.baz_id";
=item C<< $stmt->add_index_hint(foo => {type => 'USE', list => ['index_hint']}); >>
=item C<< $stmt->add_index_hint(foo => 'index_hint'); >>
=item C<< $stmt->add_index_hint(foo => ['index_hint']); >>
my $stmt = SQL::Maker::Select->new();
$stmt->add_select('name');
$stmt->add_from('user');
$stmt->add_index_hint(user => {type => 'USE', list => ['index_hint']});
$stmt->as_sql();
# => "SELECT name FROM user USE INDEX (index_hint)"
=item C<< $stmt->add_where('foo_id' => 'bar'); >>
Add a new WHERE clause.
my $stmt = SQL::Maker::Select->new()
->add_select('c')
->add_from('foo')
->add_where('name' => 'john')
->add_where('type' => {IN => [qw/1 2 3/]})
->as_sql();
# => "SELECT c FROM foo WHERE (name = ?) AND (type IN (?, ?, ?))"
=item C<< $stmt->add_where_raw('id = ?', [1]) >>
Add a new WHERE clause from raw placeholder string and bind variables.
my $stmt = SQL::Maker::Select->new()
->add_select('c')
->add_from('foo')
->add_where_raw('EXISTS(SELECT * FROM bar WHERE name = ?)' => ['john'])
->add_where_raw('type IS NOT NULL')
->as_sql();
# => "SELECT c FROM foo WHERE (EXISTS(SELECT * FROM bar WHERE name = ?)) AND (type IS NOT NULL)"
=item C<< $stmt->set_where($condition) >>
Set the WHERE clause.
$condition should be instance of L<SQL::Maker::Condition>.
my $cond1 = SQL::Maker::Condition->new()
->add("name" => "john");
my $cond2 = SQL::Maker::Condition->new()
->add("type" => {IN => [qw/1 2 3/]});
my $stmt = SQL::Maker::Select->new()
->add_select('c')
->add_from('foo')
->set_where($cond1 & $cond2)
->as_sql();
# => "SELECT c FROM foo WHERE ((name = ?)) AND ((type IN (?, ?, ?)))"
=item C<< $stmt->add_order_by('foo'); >>
=item C<< $stmt->add_order_by({'foo' => 'DESC'}); >>
Add a new ORDER BY clause.
my $stmt = SQL::Maker::Select->new()
->add_select('c')
->add_from('foo')
->add_order_by('name' => 'DESC')
->add_order_by('id')
->as_sql();
# => "SELECT c FROM foo ORDER BY name DESC, id"
=item C<< $stmt->add_group_by('foo'); >>
Add a new GROUP BY clause.
my $stmt = SQL::Maker::Select->new()
->add_select('c')
->add_from('foo')
->add_group_by('id')
->as_sql();
# => "SELECT c FROM foo GROUP BY id"
my $stmt = SQL::Maker::Select->new()
->add_select('c')
->add_from('foo')
->add_group_by('id' => 'DESC')
->as_sql();
# => "SELECT c FROM foo GROUP BY id DESC"
=item C<< $stmt->limit(30) >>
=item C<< $stmt->offset(5) >>
Add LIMIT and OFFSET.
my $stmt = SQL::Maker::Select->new()
->add_select('c')
->add_from('foo')
->limit(30)
->offset(5)
->as_sql();
# => "SELECT c FROM foo LIMIT 30 OFFSET 5"
=item C<< $stmt->add_having(cnt => 2) >>
Add a HAVING clause.
my $stmt = SQL::Maker::Select->new()
->add_from('foo')
->add_select(\'COUNT(*)' => 'cnt')
->add_having(cnt => 2)
->as_sql();
# => "SELECT COUNT(*) AS cnt FROM foo HAVING (COUNT(*) = ?)"
=back
=head1 SEE ALSO
L<Data::ObjectDriver::SQL>