| Bigtop documentation | Contained in the Bigtop distribution. |
Bigtop::Backend::SQL::DB2 - backend to generate sql for DB2 database creation
If your bigtop file looks like this:
config {
SQL DB2 {}
}
app App::Name {
}
and there are table and/or sequence blocks in the app block, this module will make docs/schema.db2 (relative to the build_dir) when you type:
bigtop app.bigtop SQL
or
bigtop app.bigtop all
You can feed that file directly to any DB2 command interpreter, once you have created a database (and schema if needed).
This is a Bigtop backend which generates SQL DB2 can understand.
This module defines no keywords. Look in Bigtop::SQL for a list of the keywords you can use in table and sequence blocks.
Note that while DB2 supports sequences, this module does not. Using them is likely to be fatal (at least to your generated SQL).
This module does provide a couple of bits of shorthand (some aren't so short) for the arguments of the is field statement.
field id {
is int4, primary_key, auto;
}
This translates into:
id INTEGER PRIMARY KEY NOT NULL GENERATED ALWAYS AS IDENTITY
(START WITH 1, INCREMENT BY 1, NO CACHE)
You can also type 'assign_by_sequence' instead of 'auto'. That might aid understanding, if you can type it correctly.
Note that using 'primary_key' instead of the literal 'PRIMARY KEY' is important. It tells the SQL and the Model back ends that this is the primary key, allowing foreign keys to use REFERENCES and many other nice effects.
To keep podcoverage tests happy.
Tells tentmaker that I understand these config section backend block keywords:
no_gen
template
Tells tentmaker what this module makes. Summary: docs/schema.db2.
Called by Bigtop::Parser to get me to do my thing.
Called by Bigtop::Parser so the user can substitute an alternate template for the hard coded one here.
Phil Crow <crow.phil@gmail.com>
Copyright (C) 2009 by Phil Crow
This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself, either Perl version 5.8.6 or, at your option, any later version of Perl 5 you may have available.
| Bigtop documentation | Contained in the Bigtop distribution. |
package Bigtop::Backend::SQL::DB2; use strict; use warnings; use Bigtop::Backend::SQL; use Inline; sub what_do_you_make { return [ [ 'docs/schema.db2' => 'SQL for DB2 database creation' ], ]; } sub backend_block_keywords { return [ { keyword => 'no_gen', label => 'No Gen', descr => 'Skip everything for this backend', type => 'boolean' }, { keyword => 'template', label => 'Alternate Template', descr => 'A custom TT template.', type => 'text' }, ]; } sub gen_SQL { shift; my $base_dir = shift; my $tree = shift; # walk tree generating sql my $lookup = $tree->{application}{lookup}; my $sql = $tree->walk_postorder( 'output_sql_db2', $lookup ); my $sql_output = join '', @{ $sql }; # write the schema.postgres my $docs_dir = File::Spec->catdir( $base_dir, 'docs' ); mkdir $docs_dir; my $sql_file = File::Spec->catfile( $docs_dir, "schema.db2" ); open my $SQL, '>', $sql_file or die "Couldn't write $sql_file: $!\n"; print $SQL $sql_output; close $SQL or die "Couldn't close $sql_file: $!\n"; } our $template_is_setup = 0; our $default_template_text = <<'EO_TT_blocks'; [% BLOCK sql_block %] DROP TABLE [% name %]; CREATE [% keyword %] [% name %][% child_output %] [% END %] [% BLOCK table_body %] ( [% FOREACH child_element IN child_output %] [% child_element +%][% UNLESS loop.last %],[% END %] [% END %] ); [% END %] [% BLOCK pk_text %] PRIMARY KEY( [% FOREACH pk IN pks %][% pk %][% UNLESS loop.last %], [% END %][% END %] ) [%- END -%] [% BLOCK table_element_block %] [% name %] [% child_output %][% END %] [% BLOCK field_statement %] [% keywords.join( ' ' ) %] [% END %] [% BLOCK insert_statement %] INSERT INTO [% table %] ( [% columns.join( ', ' ) %] ) VALUES ( [% values.join( ', ' ) %] ); [% END %] [% BLOCK three_way %] DROP TABLE [% table_name %]; CREATE TABLE [% table_name %] ( id INTEGER PRIMARY KEY NOT NULL GENERATED ALWAYS AS IDENTITY ( START WITH 1, INCREMENT BY 1, NO CACHE ), [% FOREACH foreign_key_col IN foreign_key_cols %] [% foreign_key_col.name %] INTEGER[% IF foreign_key_col.foreign_pk %] REFERENCES [% foreign_key_col.table %]([% foreign_key_col.foreign_pk %])[% END %][% IF ! loop.last || other_fields.0 %],[% END +%] [% END %] [%- FOREACH other_field IN other_fields %] [% other_field %][% IF ! loop.last %],[% END +%] [% END -%] ); [% END %] EO_TT_blocks sub setup_template { my $class = shift; my $template_text = shift || $default_template_text; return if ( $template_is_setup ); Inline->bind( TT => $template_text, POST_CHOMP => 1, TRIM_LEADING_SPACE => 0, TRIM_TRAILING_SPACE => 0, ); $template_is_setup = 1; } package # table_block table_block; use strict; use warnings; sub output_sql_db2 { my $self = shift; my $child_output = shift; my $lookup = shift; return if ( $self->_skip_this_block ); my %output; foreach my $statement ( @{ $child_output } ) { my ( $type, $output ) = @{ $statement }; push @{ $output{ $type } }, $output; } my $pks = $self->find_primary_key( $self->{__NAME__}, $lookup ); if ( ref( $pks ) eq 'ARRAY' ) { # multi-column primary key my $pk_text = Bigtop::Backend::SQL::DB2::pk_text( { pks => $pks, } ); push @{ $output{ table_body } }, $pk_text; } elsif ( defined $pks ) { # single column, but treat it as a constraint push @{ $output{ table_body } }, " PRIMARY KEY( $pks )"; } my $child_out_str = Bigtop::Backend::SQL::DB2::table_body( { child_output => $output{table_body} } ); if ( defined $output{insert_statements} ) { $child_out_str .= "\n" . join "\n", @{ $output{insert_statements} }; } my $output = Bigtop::Backend::SQL::DB2::sql_block( { keyword => $self->get_create_keyword(), child_output => $child_out_str, name => $self->get_name(), } ); return [ $output ]; } package # table_element_block table_element_block; use strict; use warnings; sub output_sql_db2 { my $self = shift; my $child_output = shift; if ( defined $child_output) { my %output_pieces; foreach my $child_item ( @{ $child_output } ) { my ( $type, $output ) = %{ $child_item }; $output_pieces{ $type } = $output; } my $child_out_str = $output_pieces{ base_col_def }; if ( $output_pieces{ foreign_key_col } ) { unless ( $output_pieces{ foreign_table } ) { die "field '" . $self->get_name . "' in table '" . $self->get_table_name . "' has a foreign_key_col, but no refers_to\n" } $child_out_str .= ' REFERENCES ' . $output_pieces{ foreign_table } . "($output_pieces{ foreign_key_col })"; if ( $output_pieces{ on_delete } ) { $child_out_str .= "\n ON DELETE $output_pieces{ on_delete }"; } if ( $output_pieces{ on_update } ) { $child_out_str .= "\n ON UPDATE $output_pieces{ on_update }"; } } my $output = Bigtop::Backend::SQL::DB2::table_element_block( { name => $self->get_name(), child_output => $child_out_str } ); return [ [ table_body => $output ] ]; } else { return unless ( $self->{__TYPE__} eq 'data' ); my %text_column = ( varchar => 1, vlong => 1, date => 1, datetime => 1, 'time' => 1, timestamp => 1, ); my @columns; my @values; foreach my $insertion ( @{ $self->{__ARGS__} } ) { my ( $column, $value ) = %{ $insertion }; my $column_base_type_data = $self->{__PARENT__}->walk_postorder( 'get_column_base_type', { target => $column } ); my $column_type_base = lc $column_base_type_data->[0]; $column_type_base =~ s/\(.*//; if ( not defined $column_type_base or exists $text_column{$column_type_base} ) { $value = "'$value'"; } push @columns, $column; push @values, $value; } my $output = Bigtop::Backend::SQL::DB2::insert_statement( { table => $self->get_table_name, columns => \@columns, values => \@values, } ); return [ [ insert_statements => $output ] ]; } } sub get_column_base_type { my $self = shift; my $child_data = shift; my $data = shift; return unless defined $self->{__NAME__}; return unless $self->{__NAME__} eq $data->{ target }; return $child_data; } package # field_statement field_statement; use strict; use warnings; my $auto_text = 'NOT NULL GENERATED ALWAYS AS IDENTITY ' . '(START WITH 1, INCREMENT BY 1, NO CACHE)'; my %expansion_for = ( int4 => 'INTEGER', assign_by_sequence => $auto_text, auto => $auto_text, varchar => 'VARCHAR(255)', datetime => 'TIMESTAMP', vlong => 'LONG VARCHAR', boolean => 'INTEGER', ); sub db2_pk_text { my $self = shift; my $lookup = shift; my $table = $self->get_table_name(); my $pks = table_block->find_primary_key( $table, $lookup ); return ( ref( $pks ) eq 'ARRAY' ) ? '' : 'PRIMARY KEY'; } sub output_sql_db2 { my $self = shift; shift; # there is no child output my $lookup = shift; my $keyword = $self->get_name(); if ( $keyword eq 'is' ) { my @keywords; foreach my $arg ( @{ $self->{__DEF__}{__ARGS__} } ) { my $expanded_form = $expansion_for{$arg}; # primary_key is ignored here, single and multi-column are # applied as constraints if ( $arg eq 'primary_key' ) { # my $pk_text = $self->db2_pk_text( $lookup ); # push @keywords, $pk_text if $pk_text; } elsif ( defined $expanded_form ) { push @keywords, $expanded_form; } else { push @keywords, $arg; } } my $output = Bigtop::Backend::SQL::DB2::field_statement( { keywords => \@keywords } ); return [ { base_col_def => $output } ]; } elsif ( $keyword eq 'refers_to' ) { my $foreign_info = $self->{__DEF__}{__ARGS__}[0]; return unless ( ref( $foreign_info ) eq 'HASH' ); my ( $table, $col ) = %{ $foreign_info }; return [ { foreign_table => $table }, { foreign_key_col => $col }, ]; } elsif ( $keyword eq 'on_delete' ) { return [ { on_delete => $self->{__DEF__}{__ARGS__}[0] } ]; } elsif ( $keyword eq 'on_update' ) { return [ { on_update => $self->{__DEF__}{__ARGS__}[0] } ]; } } sub get_column_base_type { my $self = shift; shift; my $data = shift; return unless $self->{__KEYWORD__} eq 'is'; return [ $self->{__DEF__}{__ARGS__}[0] ]; } package # literal_block literal_block; use strict; use warnings; sub output_sql_db2 { my $self = shift; return $self->make_output( 'SQL' ); } package # join_table join_table; use strict; use warnings; sub output_sql_db2 { my $self = shift; my $child_output = shift; my $lookup = shift; my @foreign_keys; my @other_fields; my @inserts; foreach my $child_bit ( @{ $child_output } ) { if ( ref $child_bit eq 'ARRAY' ) { my ( $type, $new_item ) = @{ $child_bit }; if ( $type eq 'table_body' ) { push @other_fields, $new_item; } elsif ( $type eq 'insert_statements' ) { push @inserts, $new_item; } } else { push @foreign_keys, $child_bit; } } my @foreign_key_cols; my $schema = ''; foreach my $foreign_key ( @foreign_keys ) { my $col_name = $foreign_key; $col_name =~ s/(.*)\.//; $schema = $1 if $1; my $pk = table_block->find_primary_key( $foreign_key, $lookup ); $pk = undef unless scalar $pk; push @foreign_key_cols, { name => $col_name, table => $foreign_key, foreign_pk => $pk, }; } my $table_name = $self->{__NAME__}; if ( $schema and $table_name =~ /^$schema/ ) { $table_name =~ s/^$schema/$schema./; } my $three_way = Bigtop::Backend::SQL::DB2::three_way( { table_name => $table_name, foreign_key_cols => \@foreign_key_cols, other_fields => \@other_fields, } ); $three_way .= "\n" . join( "\n", @inserts ) . "\n" if @inserts; return [ $three_way ]; } package # join_table_statement join_table_statement; use strict; use warnings; sub output_sql_db2 { my $self = shift; my $child_output = shift; if ( $self->{__KEYWORD__} eq 'joins' ) { my @tables = %{ $self->{__DEF__}->get_first_arg() }; return \@tables; } elsif ( $self->{__KEYWORD__} eq 'data' ) { my @columns; my @values; foreach my $insertion ( @{ $self->{__DEF__} } ) { my ( $column, $value ) = %{ $insertion }; $value = "'$value'" unless $value =~ /^\d+$/; push @columns, $column; push @values, $value; } my $output = Bigtop::Backend::SQL::DB2::insert_statement( { table => $self->get_join_table_name, columns => \@columns, values => \@values, } ); return [ [ insert_statements => $output ] ]; } else { return; } } 1; __END__