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