# NAME

SQL::Format - Yet another yet another SQL builder

# SYNOPSIS

   use SQL::Format;

   my ($stmt, @bind) = sqlf 'SELECT %c FROM %t WHERE %w' => (
       [qw/bar baz/], # %c
       'foo',         # %t
       {
           hoge => 'fuga',
           piyo => [qw/100 200 300/],
       },             # %w
   );
   # $stmt: SELECT `bar`, `baz` FROM `foo` WHERE (`hoge` = ?) AND (`piyo` IN (?, ?, ?))
   # @bind: ('fuga', 100, 200, 300);

   ($stmt, @bind) = sqlf 'SELECT %c FROM %t WHERE %w %o' => (
       '*',                # %c
       'foo',              # %t
       { hoge => 'fuga' }, # w
       {
           order_by => { bar => 'DESC' },
           limit    => 100,
           offset   => 10,
       },                  # %o
   );
   # $stmt: SELECT * FROM `foo` WHERE (`hoge` = ?) ORDER BY `bar` DESC LIMIT 100 OFFSET 10
   # @bind: (`fuga`)

   ($stmt, @bind) = sqlf 'UPDATE %t SET %s' => (
       foo => { bar => 'baz', 'hoge => 'fuga' },
   );
   # $stmt: UPDATE `foo` SET `bar` = ?, `hoge` = ?
   # @bind: ('baz', 'fuga')

   my $sqlf = SQL::Format->new(
       quote_char    => '',        # do not quote
       limit_dialect => 'LimitXY', # mysql style limit-offset
   );
   ($stmt, @bind) = $sqlf->select(foo => [qw/bar baz/], {
       hoge => 'fuga',
   }, {
       order_by => 'bar',
       limit    => 100,
       offset   => 10,
   });
   # $stmt: SELECT bar, baz FROM foo WHERE (hoge = ?) ORDER BY bar LIMIT 10, 100
   # @bind: ('fuga')

   ($stmt, @bind) = $sqlf->insert(foo => { bar => 'baz', hoge => 'fuga' });
   # $stmt: INSERT INTO foo (bar, hoge) VALUES (?, ?)
   # @bind: ('baz', 'fuga')

   ($stmt, @bind) = $sqlf->update(foo => { bar => 'xxx' }, { hoge => 'fuga' });
   # $stmt: UPDATE foo SET bar = ? WHERE hoge = ?
   # @bind: ('xxx', 'fuga')

   ($stmt, @bind) = $sqlf->delete(foo => { hoge => 'fuga' });
   # $stmt: DELETE FROM foo WHERE (hoge = ?)
   # @bind: ('fuga')

# DESCRIPTION

SQL::Format is a easy to SQL query building library.

__THIS MODULE IS ALPHA LEVEL INTERFACE!!__

# FUNCTIONS

## sqlf($format, @args)

Generate SQL from formatted output conversion.

   my ($stmt, @bind) = sqlf 'SELECT %c FROM %t WHERE %w' => (
       [qw/bar baz/],   # %c
       'foo',           # %t
       {
           hoge => 'fuga',
           piyo => [100, 200, 300],
       },               # %w
   );
   # $stmt: SELECT `foo` FROM `bar`, `baz WHERE (`hoge` = ?) AND (`piyo` IN (?, ?, ?))
   # @bind: ('fuga', 100, 200, 300)

Currently implemented formatters are:

- %t

   This format is a table name.

       ($stmt, @bind) = sqlf '%t', 'table_name';        # $stmt => `table_name`
       ($stmt, @bind) = sqlf '%t', [qw/tableA tableB/]; # $stmt => `tableA`, `tableB`
       ($stmt, @bind) = sqlf '%t', { tableA => 't1' };  # $stmt => `tableA` `t1`
       ($stmt, @bind) = sqlf '%t', {
           tableA => {
               index => { type => 'force', keys => [qw/key1 key2/] },
               alias => 't1',
       }; # $stmt: `tableA` `t1` FORCE INDEX (`key1`, `key2`)

- %c

   This format is a column name.

       ($stmt, @bind) = sqlf '%c', 'column_name';       # $stmt => `column_name`
       ($stmt, @bind) = sqlf '%c', [qw/colA colB/];     # $stmt => `colA`, `colB`
       ($stmt, @bind) = sqlf '%c', '*';                 # $stmt => *
       ($stmt, @bind) = sqlf '%c', [\'COUNT(*)', colC]; # $stmt => COUNT(*), `colC`

- %w

   This format is a where clause.

       ($stmt, @bind) = sqlf '%w', { foo => 'bar' };
       # $stmt: (`foo` = ?)
       # @bind: ("bar")

       ($stmt, @bind) = sqlf '%w', {
           foo => 'bar',
           baz => [qw/100 200 300/],
       };
       # $stmt: (`baz` IN (?, ?, ?) AND (`foo` = ?)
       # @bind: (100, 200, 300, 'bar')

- %o

   This format is a options. Currently specified are:

   - limit

       This option makes `LIMIT $n` clause.

           ($stmt, @bind) = sqlf '%o', { limit => 100 }; # $stmt => LIMIT 100

   - offset

       This option makes `OFFSET $n` clause. You must be specified both limit option.

           ($stmt, @bind) = sqlf '%o', { limit => 100, offset => 20 }; # $stmt => LIMIT 100 OFFSET 20

       You can change limit dialects from `$SQL::Format::LIMIT_DIALECT`.

   - order\_by

       This option makes `ORDER BY` clause.

           ($stmt, @bind) = sqlf '%o', { order_by => 'foo' };                       # $stmt => ORDER BY `foo`
           ($stmt, @bind) = sqlf '%o', { order_by => { foo => 'DESC' } };           # $stmt => ORDER BY `foo` DESC
           ($stmt, @bind) = sqlf '%o', { order_by => ['foo', { -asc => 'bar' } ] }; # $stmt => ORDER BY `foo`, `bar` ASC

   - group\_by

       This option makes `GROUP BY` clause. Argument value some as `order_by` option.

           ($stmt, @bind) = sqlf '%o', { group_by => { foo => 'DESC' } }; # $stmt => GROUP BY `foo` DESC

   - having

       This option makes `HAVING` clause. Argument value some as `where` clause.

           ($stmt, @bind) = sqlf '%o', { having => { foo => 'bar' } };
           # $stmt: HAVING (`foo` = ?)
           # @bind: ('bar')

- %j

   This format is join clause.

       ($stmt, @bind) = sqlf '%j', { table => 'bar', condition => 'foo.id = bar.id' };
       # $stmt: INNER JOIN `bar` ON (foo.id = bar.id)

       ($stmt, @bind) = sqlf '%j', {
           type      => 'left',
           table     => { bar => 'b' },
           condition => {
               'f.id'         => 'b.id',
               'f.updated_at' => \['UNIX_TIMESTAMP()', '2012-12-12']
               'f.created_at' => { '>' => 'b.created_at' },
           },
       };
       # $stmt: LEFT JOIN `bar` `b` ON (`f`.`id` = `b.id`)

- %s

   This format is set clause.

       ($stmt, @bind) = sqlf '%s', { bar => 'baz' };
       # $stmt: `bar` = ?
       # @bind: ('baz')

       ($stmt, @bind) = sqlf '%s', { bar => 'baz', 'hoge' => \'UNIX_TIMESTAMP()' };
       # $stmt: `bar` = ?, `hoge` = UNIX_TIMESTAMP()
       # @bind: ('baz')

       ($stmt, @bind) = sqlf '%s', {
           bar  => 'baz',
           hoge => \['CONCAT(?, ?)', 'ya', 'ppo'],
       };
       # $stmt: `bar` = ?, `hoge` = CONCAT(?, ?)
       # @bind: ('baz', 'ya', 'ppo')

For more examples, see also [SQL::Format::Spec](http://search.cpan.org/perldoc?SQL::Format::Spec).

You can change the behavior by changing the global variable.

- $SQL::Format::QUOTE\_CHAR : Str

   This is a quote character for table or column name.

   Default value is `"`"`.

- $SQL::Format::NAME\_SEP : Str

   This is a separate character for table or column name.

   Default value is `"."`.

- $SQL::Format::DELIMITER Str

   This is a delimiter for between columns.

   Default value is `", "`.

- $SQL::Format::LIMIT\_DIALECT : Str

   This is a types for dialects of limit-offset.

   You can choose are:

       LimitOffset  # LIMIT 100 OFFSET 20  (SQLite / PostgreSQL / MySQL)
       LimitXY      # LIMIT 20, 100        (MySQL / SQLite)
       LimitYX      # LIMIT 100, 20        (other)

   Default value is `LimitOffset"`.

# METHODS

## new(\[%options\])

Create a new instance of `SQL::Format`.

   my $sqlf = SQL::Format->new(
       quote_char    => '',
       limit_dialect => 'LimitXY',
   );

`%options` specify are:

- quote\_char : Str

   Default value is `$SQL::Format::QUOTE_CHAR`.

- name\_sep : Str

   This is a separate character for table or column name.

   Default value is `$SQL::Format::NAME_SEP`.

- delimiter: Str

   This is a delimiter for between columns.

   Default value is `$SQL::Format::DELIMITER`.

- limit\_dialect : Str

   This is a types for dialects of limit-offset.

   Default value is `$SQL::Format::LIMIT_DIALECT`.

## format($format, \\%args)

This method same as `sqlf` function.

   my ($stmt, @bind) = $self->format('SELECT %c FROM %t WHERE %w',
       [qw/bar baz/],
       'foo',
       { hoge => 'fuga' },
   );
   # $stmt: SELECT `bar`, `baz` FROM ` foo` WHERE (`hoge` = ?)
   # @bind: ('fuga')

## select($table|\\@table, $column|\\@columns \[, \\%where, \\%opts \])

This method returns SQL string and bind parameters for `SELECT` statement.

   my ($stmt, @bind) = $sqlf->select(foo => [qw/bar baz/], {
       hoge => 'fuga',
       piyo => [100, 200, 300],
   });
   # $stmt: SELECT `foo` FROM `bar`, `baz` WHERE (`hoge` = ?) AND (`piyo` IN (?, ?, ?))
   # @bind: ('fuga', 100, 200, 300)

Argument details are:

- $table | \\@table

   Same as `%t` format.

- $column | \\@columns

   Same as `%c` format.

- \\%where

   Same as `%w` format.

- \\%opts
   - $opts->{prefix}

       This is prefix for SELECT statement.

           my ($stmt, @bind) = $sqlf->select(foo => '*', { bar => 'baz' }, { prefix => 'SELECT SQL_CALC_FOUND_ROWS' });
           # $stmt: SELECT SQL_CALC_FOUND_ROWS * FROM `foo` WHERE (`bar` = ?)
           # @bind: ('baz')

       Default value is `SELECT`.

   - $opts->{suffix}

       Additional value for after the SELECT statement.

           my ($stmt, @bind) = $sqlf->select(foo => '*', { bar => 'baz' }, { suffix => 'FOR UPDATE' });
           # $stmt: SELECT * FROM `foo` WHERE (bar = ?) FOR UPDATE
           # @bind: ('baz')

       Default value is `''`

   - $opts->{limit}
   - $opts->{offset}
   - $opts->{order\_by}
   - $opts->{group\_by}
   - $opts->{having}
   - $opts->{join}

       See also `%o` format.

## insert($table, \\%values|\\@values \[, \\%opts \])

This method returns SQL string and bind parameters for `INSERT` statement.

   my ($stmt, @bind) = $sqlf->insert(foo => { bar => 'baz', hoge => 'fuga' });
   # $stmt: INSERT INTO `foo` (`bar`, `hoge`) VALUES (?, ?)
   # @bind: ('baz', 'fuga')

   my ($stmt, @bind) = $sqlf->insert(foo => [
       hoge => \'NOW()',
       fuga => \['UNIX_TIMESTAMP()', '2012-12-12 12:12:12'],
   ]);
   # $stmt: INSERT INTO `foo` (`hoge`, `fuga`) VALUES (NOW(), UNIX_TIMESTAMP(?))
   # @bind: ('2012-12-12 12:12:12')

Argument details are:

- $table

   This is a table name for target of INSERT.

- \\%values | \\@values

   This is a VALUES clause INSERT statement.

   Currently supported types are:

       # \%values case
       { foo => 'bar' }
       { foo => \'NOW()' }
       { foo => \['UNIX_TIMESTAMP()', '2012-12-12 12:12:12'] }

       # \@values case
       [ foo => 'bar' ]
       [ foo => \'NOW()' ]
       [ foo => \['UNIX_TIMESTAMP()', '2012-12-12 12:12:12'] ]

- \\%opts
   - $opts->{prefix}

       This is a prefix for INSERT statement.

           my ($stmt, @bind) = $sqlf->insert(foo => { bar => baz }, { prefix => 'INSERT IGNORE' });
           # $stmt: INSERT IGNORE INTO `foo` (`bar`) VALUES (?)
           # @bind: ('baz')

       Default value is `INSERT`.

## update($table, \\%set|\\@set \[, \\%where, \\%opts \])

This method returns SQL string and bind parameters for `UPDATE` statement.

   my ($stmt, @bind) = $sqlf->update(foo => { bar => 'baz' }, { hoge => 'fuga' });
   # $stmt: UPDATE `foo` SET `bar` = ? WHERE (`hoge` = ?)
   # @bind: ('baz', 'fuga')

Argument details are:

- $table

   This is a table name for target of UPDATE.

- \\%set | \\@set

   This is a SET clause for INSERT statement.

   Currently supported types are:

       # \%values case
       { foo => 'bar' }
       { foo => \'NOW()' }
       { foo => \['UNIX_TIMESTAMP()', '2012-12-12 12:12:12'] }

       # \@values case
       [ foo => 'bar' ]
       [ foo => \'NOW()' ]
       [ foo => \['UNIX_TIMESTAMP()', '2012-12-12 12:12:12'] ]

- \\%where

   Same as `%w` format.

- \\%opts
   - $opts->{prefix}

       This is a prefix for UPDATE statement.

           my ($stmt, @bind) = $sqlf->update(
               'foo'                                # table
               { bar    => 'baz' },                 # sets
               { hoge   => 'fuga' },                # where
               { prefix => 'UPDATE LOW_PRIORITY' }, # opts
           );
           # $stmt: UPDATE LOW_PRIORITY `foo` SET `bar` = ? WHERE (`hoge` = ?)
           # @bind: ('baz', 'fuga')

       Default value is `UPDATE`.

   - $opts->{order\_by}
   - $opts->{limit}

       See also `%o` format.

## delete($table \[, \\%where, \\%opts \])

This method returns SQL string and bind parameters for `DELETE` statement.

   my ($stmt, @bind) = $sqlf->delete(foo => { bar => 'baz' });
   # $stmt: DELETE FROM `foo` WHERE (`bar = ?)
   # @bind: ('baz')

Argument details are:

- $table

   This is a table name for target of DELETE.

- \\%where

   Same as `%w` format.

- \\%opts
   - $opts->{prefix}

       This is a prefix for DELETE statement.

           my ($stmt, @bind) = $sqlf->delete(foo => { bar => 'baz' }, { prefix => 'DELETE LOW_PRIORITY' });
           # $stmt: DELETE LOW_PRIORITY FROM `foo` WHERE (`bar` = ?)
           # @bind: ('baz')

       Default value is `DELETE`.

   - $opts->{order\_by}
   - $opts->{limit}

       See also `%o` format.

## insert\_multi($table, \\@cols, \\@values \[, \\%opts\])

This method returns SQL string and bind parameters for bulk insert.

   my ($stmt, @bind) = $self->insert_multi(
       foo => [qw/bar baz/],
       [
           [qw/hoge fuga/],
           [qw/fizz buzz/],
       ],
   );
   # $stmt: INSERT INTO `foo` (`bar`, `baz`) VALUES (?, ?), (?, ?)
   # @bind: (qw/hoge fuga fizz buzz/)

Argument details are:

- $table

   This is a table name for target of INSERT.

- \\@cols

   This is a columns for target of INSERT.

- \\@values

   This is a values parameters. Must be ARRAY within ARRAY.

       my ($stmt, @bind) = $sqlf->insert_multi(
           foo => [qw/bar baz/], [
               [qw/foo bar/],
               [\'NOW()', \['UNIX_TIMESTAMP(?)', '2012-12-12 12:12:12'] ],
           ],
       );
       # $stmt: INSERT INTO `foo` (`bar`, `baz`) VALUES (?, ?), (NOW(), UNIX_TIMESTAMP(?))
       # @bind: (qw/foo bar/, '2012-12-12 12:12:12')

- \\%opts
   - $opts->{prefix}

       This is a prefix for INSERT statement.

           my ($stmt, @bind) = $sqlf->insert_multi(..., { prefix => 'INSERT IGNORE INTO' });
           # $stmt: INSERT IGNORE INTO ...

       Default value is `INSERT INTO`.

   - $opts->{update}

       Some as `%s` format.

       If this value specified then add `ON DUPLICATE KEY UPDATE` statement.

           my ($stmt, @bind) = $sqlf->insert_multi(
               foo => [qw/bar baz/],
               [
                   [qw/hoge fuga/],
                   [qw/fizz buzz/],
               ],
               { update => { bar => 'piyo' } },
           );
           # $stmt: INSERT INTO `foo` (`bar`, `baz`) VALUES (?, ?), (?, ?) ON DUPLICATE KEY UPDATE `bar` = ?
           # @bind: (qw/hoge fuga fizz buzz piyo/)

## insert\_multi\_from\_hash($table, \\@values \[, \\%opts\])

This method is a wrapper for `insert_multi()`.

Argument dialects are:

- $table

   Same as `insert_multi()`

- \\@values

   This is a values parameters. Must be HASH within ARRAY.

       my ($stmt, @bind) = $sqlf->insert_multi_from_hash(foo => [
           { bar => 'hoge', baz => 'fuga' },
           { bar => 'fizz', baz => 'buzz' },
       ]);
       # $stmt: INSERT INTO `foo` (`bar`, `baz`) VALUES (?, ?), (?, ?)
       # @bind: (qw/hoge fuga fizz buzz/)

- \\%opts

   Same as `insert_multi()`

## insert\_on\_duplicate($table, \\%values|\\@values, \\%update\_values|\\@update\_values \[, \\%opts\])

This method generate "INSERT INTO ... ON DUPLICATE KEY UPDATE" query for MySQL.

   my ($stmt, @bind) = $sqlf->insert_on_duplicate(
       foo => {
           bar => 'hoge',
           baz => 'fuga',
       }, {
           bar => \'VALUES(bar)',
           baz => 'piyo',
       },
   );
   # $stmt: INSERT INTO `foo` (`bar`, `baz`) VALUES (?, ?) ON DUPLICATE KEY UPDATE `bar` = VALUES(bar), baz = 'piyo'
   # @bind: (qw/hoge fuga piyo/)

Argument details are:

- $table

   This is a table name for target of INSERT.

- \\%values|\\@values

   This is a values parameters.

- \\%update\_values|\\@update\_values

   This is a ON DUPLICATE KEY UPDATE parameters.

- \\%opts
   - $opts->{prefix}

       This is a prefix for INSERT statement.

           my ($stmt, @bind) = $sqlf->insert_on_duplicate(..., { prefix => 'INSERT IGNORE INTO' });
           # $stmt: INSERT IGNORE INTO ...

# AUTHOR

xaicron <xaicron {at} cpan.org>

# COPYRIGHT

Copyright 2012 - xaicron

# LICENSE

This library is free software; you can redistribute it and/or modify
it under the same terms as Perl itself.

# SEE ALSO

[SQL::Format::Spec](http://search.cpan.org/perldoc?SQL::Format::Spec)

[SQL::Maker](http://search.cpan.org/perldoc?SQL::Maker)

[SQL::Abstract](http://search.cpan.org/perldoc?SQL::Abstract)