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