add MooseX::Aliases (to allow for to/from and producer/parser)
[dbsrgits/SQL-Translator-2.0-ish.git] / lib / SQL / Translator.pm
CommitLineData
4f4fd192 1use MooseX::Declare;
2class SQL::Translator {
365e4efe 3 use TryCatch;
631efe08 4 use MooseX::Types::Moose qw(Bool HashRef Int Str Undef);
c6b7881f 5 use MooseX::Aliases;
ee1a2ac9 6 use SQL::Translator::Types qw(DBIHandle Parser Producer Schema);
7 use SQL::Translator::Object::Schema;
8
2d6021dc 9 our $VERSION = '0.001';
10
4f4fd192 11 has 'parser' => (
12 isa => Str,
ee1a2ac9 13 is => 'rw',
c6b7881f 14 alias => 'from',
4f4fd192 15 );
16
17 has 'producer' => (
18 isa => Str,
ee1a2ac9 19 is => 'rw',
c6b7881f 20 alias => 'to',
4f4fd192 21 );
22
23 has '_parser' => (
24 isa => Parser,
25 is => 'rw',
26 lazy_build => 1,
27 handles => [ qw(parse) ],
c6b7881f 28 predicate => 'has_parser',
4f4fd192 29 );
30
31 has '_producer' => (
32 isa => Producer,
33 is => 'rw',
34 lazy_build => 1,
35 handles => [ qw(produce) ],
c6b7881f 36 predicate => 'has_producer',
4f4fd192 37 );
38
39 has 'dbh' => (
40 isa => DBIHandle,
41 is => 'ro',
42 predicate => 'has_dbh',
43 );
ee1a2ac9 44
45 has 'schema' => (
46 isa => Schema,
47 is => 'rw',
5f60616d 48 lazy => 1,
c062b437 49 clearer => '_clear_schema',
5f60616d 50 default => sub { SQL::Translator::Object::Schema->new },
ee1a2ac9 51 );
52
53 has 'parser_args' => (
54 isa => HashRef,
55 is => 'rw',
23043f8d 56 predicate => 'has_parser_args',
ee1a2ac9 57 );
58
59 has 'producer_args' => (
60 isa => HashRef,
61 is => 'rw',
23043f8d 62 predicate => 'has_producer_args',
4f4fd192 63 );
64
ee1a2ac9 65 has 'add_drop_table' => (isa => Bool, is => 'rw', default => 0);
66 has 'no_comments' => (isa => Bool, is => 'rw', default => 0);
67 has 'show_warnings' => (isa => Bool, is => 'rw', default => 1);
68 has 'trace' => (isa => Bool, is => 'rw', default => 0);
6127f0bd 69 has 'quote_table_names' => (isa => Bool, is => 'rw', default => 0);
70 has 'quote_field_names' => (isa => Bool, is => 'rw', default => 0);
ee1a2ac9 71 has 'version' => (isa => Str, is => 'rw');
72 has 'filename' => (isa => Str, is => 'rw');
73
2d6021dc 74 has '_producer_mapping' => (
75 isa => HashRef,
76 is => 'ro',
77 default => sub { { MySQL => 'SQL::MySQL', SQLite => 'SQL::SQLite', PostgreSQL => 'SQL::PostgreSQL', XML => 'XML', YAML => 'YAML' } }
78 );
79
4f4fd192 80 method _build__parser {
81 my $class = 'SQL::Translator::Parser';
82
83 Class::MOP::load_class($class);
84
15adaf18 85 my $parser;
86 if ($self->has_dbh) {
ee1a2ac9 87 $parser = $class->new({ translator => $self, dbh => $self->dbh });
15adaf18 88 } else {
ee1a2ac9 89 $parser = $class->new({ translator => $self, type => $self->parser || '' });
15adaf18 90 }
4f4fd192 91
92 return $parser;
93 }
94
95 method _build__producer {
2d6021dc 96 my $mapping = $self->_producer_mapping;
97
4f4fd192 98 my $class = 'SQL::Translator::Producer';
2d6021dc 99 my $role = $class . '::' . $mapping->{$self->producer};
d1684085 100
4f4fd192 101 Class::MOP::load_class($class);
2d6021dc 102 Class::MOP::load_class($role);
4f4fd192 103
ee1a2ac9 104 my $producer = $class->new({ translator => $self });
4f4fd192 105 $role->meta->apply($producer);
106
107 return $producer;
c16f2fa9 108 }
ee1a2ac9 109
110 method translate(:$data, :$producer?, :$producer_args?, :$parser?, :$parser_args?) {
631efe08 111 my $return;
112
c6b7881f 113 $self->_clear_schema if defined $parser;
114
115 $parser ||= $self->parser unless $self->has_parser;
631efe08 116 if (defined $parser) {
ee1a2ac9 117 $self->_clear_parser;
118 $self->parser($parser);
d1684085 119 $self->parse($data);
631efe08 120 $return = $self->schema;
121 }
122
c6b7881f 123 $producer ||= $self->producer unless $self->has_producer;
631efe08 124 if (defined $producer) {
ee1a2ac9 125 $self->_clear_producer;
ee1a2ac9 126 $self->producer($producer);
631efe08 127 $return = $self->produce;
ee1a2ac9 128 }
631efe08 129
130 return $return;
ee1a2ac9 131 }
132
133 method parser_type { return $self->parser }
134 method producer_type { return $self->producer }
ffe3a4b1 135
631efe08 136 method engine_version(Int|Str|Undef $v, Str $target = 'perl') {
137 return undef unless $v;
138
ffe3a4b1 139 my @vers;
140
141 # X.Y.Z style
142 if ( $v =~ / ^ (\d+) \. (\d{1,3}) (?: \. (\d{1,3}) )? $ /x ) {
143 push @vers, $1, $2, $3;
144 }
145
146 # XYYZZ (mysql) style
147 elsif ( $v =~ / ^ (\d) (\d{2}) (\d{2}) $ /x ) {
148 push @vers, $1, $2, $3;
149 }
150
151 # XX.YYYZZZ (perl) style or simply X
152 elsif ( $v =~ / ^ (\d+) (?: \. (\d{3}) (\d{3}) )? $ /x ) {
153 push @vers, $1, $2, $3;
154 }
155 else {
156 #how do I croak sanely here?
157 die "Unparseable MySQL version '$v'";
158 }
159
160 if ($target eq 'perl') {
161 return sprintf ('%d.%03d%03d', map { $_ || 0 } (@vers) );
162 }
163 elsif ($target eq 'mysql') {
164 return sprintf ('%d%02d%02d', map { $_ || 0 } (@vers) );
165 }
166 else {
167 #how do I croak sanely here?
168 die "Unknown version target '$target'";
169 }
170 }
4f4fd192 171}