line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package Class::DBI::ClassGenerator; |
2
|
|
|
|
|
|
|
|
3
|
2
|
|
|
2
|
|
203770
|
use strict; |
|
2
|
|
|
|
|
6
|
|
|
2
|
|
|
|
|
93
|
|
4
|
2
|
|
|
2
|
|
13
|
use warnings; |
|
2
|
|
|
|
|
4
|
|
|
2
|
|
|
|
|
66
|
|
5
|
2
|
|
|
2
|
|
26306
|
use DBI; |
|
2
|
|
|
|
|
91715
|
|
|
2
|
|
|
|
|
156
|
|
6
|
|
|
|
|
|
|
|
7
|
2
|
|
|
2
|
|
23
|
use File::Spec; |
|
2
|
|
|
|
|
4
|
|
|
2
|
|
|
|
|
49
|
|
8
|
|
|
|
|
|
|
|
9
|
2
|
|
|
2
|
|
10
|
use vars qw($VERSION); |
|
2
|
|
|
|
|
4
|
|
|
2
|
|
|
|
|
2721
|
|
10
|
|
|
|
|
|
|
|
11
|
|
|
|
|
|
|
$VERSION = '1.04'; |
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
=head1 NAME |
14
|
|
|
|
|
|
|
|
15
|
|
|
|
|
|
|
Class::DBI::ClassGenerator - generate Class::DBI sub-class modules from a |
16
|
|
|
|
|
|
|
pre-exsting database's structure. |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
=head1 SUPPORT |
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
This module is unsupported, unloved, unmaintained, and DEPRECATED. No |
21
|
|
|
|
|
|
|
bugs will be fixed. No patches will be accepted. No users will be helped. |
22
|
|
|
|
|
|
|
All bug reports will be either ignored or rejected. |
23
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
I strongly recommend that you switch from using Class::DBI to using |
25
|
|
|
|
|
|
|
L instead, and L instead of this |
26
|
|
|
|
|
|
|
module. |
27
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
Unless, that is, someone else takes over ownership. |
29
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
=head1 SUBROUTINES |
31
|
|
|
|
|
|
|
|
32
|
|
|
|
|
|
|
=head2 create |
33
|
|
|
|
|
|
|
|
34
|
|
|
|
|
|
|
This takes the following named parameters: |
35
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
=over |
37
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
=item directory (compulsory) |
39
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
The name of the directory into which to drop the generated classes. If |
41
|
|
|
|
|
|
|
it doesn't exist it will be created. Sub-directories will be created |
42
|
|
|
|
|
|
|
under here as appropriate. |
43
|
|
|
|
|
|
|
|
44
|
|
|
|
|
|
|
directory => 'lib' |
45
|
|
|
|
|
|
|
|
46
|
|
|
|
|
|
|
=item connect_info (compulsory) |
47
|
|
|
|
|
|
|
|
48
|
|
|
|
|
|
|
An arrayref of the DSN, username and password to connect to the database. |
49
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
connect_info => ['dbi:mysql:dbname', 'username', 'password'] |
51
|
|
|
|
|
|
|
|
52
|
|
|
|
|
|
|
=item base_class (compulsory) |
53
|
|
|
|
|
|
|
|
54
|
|
|
|
|
|
|
The name of the base class that all your table classes will inherit their |
55
|
|
|
|
|
|
|
database connection from. |
56
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
base_class => 'MyApp::DB' |
58
|
|
|
|
|
|
|
|
59
|
|
|
|
|
|
|
=item tables (optional) |
60
|
|
|
|
|
|
|
|
61
|
|
|
|
|
|
|
A hashref whose keys are table names in the database and the values are |
62
|
|
|
|
|
|
|
the classnames you desire. |
63
|
|
|
|
|
|
|
|
64
|
|
|
|
|
|
|
tables => { |
65
|
|
|
|
|
|
|
artists => 'MyApp::Artist', |
66
|
|
|
|
|
|
|
tracks => 'MyApp::Track', |
67
|
|
|
|
|
|
|
albums => 'MyApp::Album', |
68
|
|
|
|
|
|
|
... |
69
|
|
|
|
|
|
|
} |
70
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
If you leave this out, the code will assume |
72
|
|
|
|
|
|
|
that you want classes for all tables, and that their names should be |
73
|
|
|
|
|
|
|
generated thus: |
74
|
|
|
|
|
|
|
|
75
|
|
|
|
|
|
|
The first character of the tablename is converted to uppercase; |
76
|
|
|
|
|
|
|
|
77
|
|
|
|
|
|
|
An underscore followed by a character becomes the character, in |
78
|
|
|
|
|
|
|
uppercase |
79
|
|
|
|
|
|
|
|
80
|
|
|
|
|
|
|
The base class name and :: is prepended. |
81
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
This is probably a close approximation for what you want anyway. |
83
|
|
|
|
|
|
|
|
84
|
|
|
|
|
|
|
=back |
85
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
It returns a list of all the files created. |
87
|
|
|
|
|
|
|
|
88
|
|
|
|
|
|
|
=cut |
89
|
|
|
|
|
|
|
|
90
|
|
|
|
|
|
|
sub create { |
91
|
5
|
|
|
5
|
1
|
5576
|
my %params = @_; |
92
|
5
|
100
|
|
|
|
32
|
die(__PACKAGE__."::create: no directory specified\n") |
93
|
|
|
|
|
|
|
unless($params{directory}); |
94
|
4
|
100
|
|
|
|
36
|
die(__PACKAGE__."::create: no connect_info specified\n") |
95
|
|
|
|
|
|
|
unless($params{connect_info}); |
96
|
3
|
100
|
|
|
|
18
|
die(__PACKAGE__."::create: no base class specified\n") |
97
|
|
|
|
|
|
|
unless($params{base_class}); |
98
|
|
|
|
|
|
|
|
99
|
2
|
|
|
|
|
7
|
mkdir($params{directory}); |
100
|
2
|
50
|
|
|
|
47
|
die("Couldn't create $params{directory}: $!\n") |
101
|
|
|
|
|
|
|
unless(-d $params{directory}); |
102
|
|
|
|
|
|
|
|
103
|
2
|
|
|
|
|
52
|
my $dbh = _get_dbh($params{connect_info}); |
104
|
2
|
|
|
|
|
954
|
my $db_driver = _get_db_driver($params{connect_info}); |
105
|
|
|
|
|
|
|
|
106
|
|
|
|
|
|
|
# get tables from DB if necessary |
107
|
2
|
|
|
|
|
8
|
$params{tables} = { |
108
|
|
|
|
|
|
|
map { |
109
|
2
|
100
|
|
|
|
15
|
$_ => _table_to_class($params{base_class}, $_) |
110
|
|
|
|
|
|
|
} $db_driver->_get_tables($dbh) |
111
|
|
|
|
|
|
|
} unless(ref($params{tables})); |
112
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
# get columns from DB |
114
|
3
|
|
|
|
|
18
|
$params{tables} = { |
115
|
|
|
|
|
|
|
map { |
116
|
2
|
|
|
|
|
7
|
$_ => { |
117
|
|
|
|
|
|
|
classname => $params{tables}->{$_}, |
118
|
|
|
|
|
|
|
columns => { $db_driver->_get_columns($dbh, $_) } |
119
|
|
|
|
|
|
|
} |
120
|
2
|
|
|
|
|
15
|
} keys %{$params{tables}} |
121
|
|
|
|
|
|
|
}; |
122
|
|
|
|
|
|
|
|
123
|
2
|
|
|
|
|
6
|
my @files_created = (); |
124
|
|
|
|
|
|
|
|
125
|
2
|
|
|
|
|
3
|
foreach my $table (keys %{$params{tables}}) { |
|
2
|
|
|
|
|
7
|
|
126
|
13
|
|
|
|
|
33
|
my $pks = join(' ', |
127
|
3
|
|
|
|
|
11
|
grep { $params{tables}->{$table}->{columns}->{$_}->{pk} } |
128
|
3
|
|
|
|
|
7
|
keys %{$params{tables}->{$table}->{columns}} |
129
|
|
|
|
|
|
|
); |
130
|
13
|
|
|
|
|
39
|
my $nonpks = join(' ', |
131
|
3
|
|
|
|
|
11
|
grep { !$params{tables}->{$table}->{columns}->{$_}->{pk} } |
132
|
3
|
|
|
|
|
42
|
keys %{$params{tables}->{$table}->{columns}} |
133
|
|
|
|
|
|
|
); |
134
|
3
|
|
|
|
|
45
|
my $classfile = File::Spec->catfile( |
135
|
|
|
|
|
|
|
$params{directory}, |
136
|
|
|
|
|
|
|
split('::', $params{tables}->{$table}->{classname}.'.pm') |
137
|
|
|
|
|
|
|
); |
138
|
3
|
|
|
|
|
46
|
_mkdir($params{directory}, $params{tables}->{$table}->{classname}); |
139
|
3
|
50
|
|
|
|
246
|
open(my $classfilefh, '>', $classfile) || |
140
|
|
|
|
|
|
|
die("Can't write $classfile: $!\n"); |
141
|
3
|
|
|
|
|
45
|
print $classfilefh "package ".$params{tables}->{$table}->{classname}.";\n"; |
142
|
3
|
|
|
|
|
9
|
print $classfilefh "use base '$params{base_class}';\n\n"; |
143
|
3
|
|
|
|
|
7
|
print $classfilefh "__PACKAGE__->table('$table');\n"; |
144
|
3
|
|
|
|
|
7
|
print $classfilefh "__PACKAGE__->columns(Primary => qw($pks));\n"; |
145
|
3
|
|
|
|
|
9
|
print $classfilefh "__PACKAGE__->columns(Others => qw($nonpks));\n"; |
146
|
3
|
|
|
|
|
125
|
close($classfilefh); |
147
|
|
|
|
|
|
|
# system("cat $classfile"); |
148
|
3
|
|
|
|
|
15
|
push @files_created, $classfile; |
149
|
|
|
|
|
|
|
} |
150
|
|
|
|
|
|
|
|
151
|
2
|
|
|
|
|
9
|
my $basefile = File::Spec->catfile( |
152
|
|
|
|
|
|
|
_mkdir($params{directory}, $params{base_class}), |
153
|
|
|
|
|
|
|
(split(/::/, $params{base_class}))[-1].'.pm' |
154
|
|
|
|
|
|
|
); |
155
|
2
|
50
|
|
|
|
165
|
open(my $basefilefh, '>', $basefile) || |
156
|
|
|
|
|
|
|
die("Can't write $basefile: $!\n"); |
157
|
2
|
|
|
|
|
12
|
print $basefilefh "package $params{base_class};\nuse base 'Class::DBI';\n\n"; |
158
|
2
|
|
|
|
|
10
|
print $basefilefh "$params{base_class}->connection('". |
159
|
2
|
|
|
|
|
13
|
join("', '", @{$params{connect_info}}). |
160
|
|
|
|
|
|
|
"');\n\n"; |
161
|
2
|
|
|
|
|
3
|
print $basefilefh "use $_;\n" foreach( |
|
3
|
|
|
|
|
15
|
|
162
|
|
|
|
|
|
|
map { |
163
|
2
|
|
|
|
|
6
|
$params{tables}->{$_}->{classname} |
164
|
|
|
|
|
|
|
} keys %{$params{tables}} |
165
|
|
|
|
|
|
|
); |
166
|
2
|
|
|
|
|
68
|
close($basefilefh); |
167
|
2
|
|
|
|
|
4
|
push @files_created, $basefile; |
168
|
|
|
|
|
|
|
# system("cat $basefile"); |
169
|
|
|
|
|
|
|
|
170
|
|
|
|
|
|
|
|
171
|
2
|
|
|
|
|
192
|
return @files_created; |
172
|
|
|
|
|
|
|
} |
173
|
|
|
|
|
|
|
|
174
|
|
|
|
|
|
|
# create a directory hierarchy for a class. Takes the base dir and |
175
|
|
|
|
|
|
|
# class name. Given, eg, ('lib', 'Foo::Bar::Baz') it will create |
176
|
|
|
|
|
|
|
# lib/Foo and lib/Foo/Bar. Returns the name of the last directory |
177
|
|
|
|
|
|
|
# created. |
178
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
sub _mkdir { |
180
|
6
|
|
|
6
|
|
4473
|
my($base, $class) = @_; |
181
|
6
|
|
|
|
|
23
|
my @components = split(/::/, $class); |
182
|
6
|
|
|
|
|
10
|
pop @components; # remove last bit - that's a filename |
183
|
6
|
|
|
|
|
10
|
my $dir = $base; |
184
|
6
|
|
|
|
|
17
|
while(@components) { |
185
|
13
|
|
|
|
|
82
|
$dir = File::Spec->catdir($dir, shift(@components)); |
186
|
13
|
|
50
|
|
|
1602
|
mkdir $dir || die("Couldn't create $dir: $!\n"); |
187
|
|
|
|
|
|
|
} |
188
|
6
|
|
|
|
|
41
|
return $dir; |
189
|
|
|
|
|
|
|
} |
190
|
|
|
|
|
|
|
|
191
|
|
|
|
|
|
|
# given a DSN/username/password arrayref, get a DBH |
192
|
3
|
|
|
3
|
|
10
|
sub _get_dbh { DBI->connect(@{$_[0]}); } |
|
3
|
|
|
|
|
23
|
|
193
|
|
|
|
|
|
|
|
194
|
|
|
|
|
|
|
# given a DSN/username/password arrayref, load and return the C::DBI::CG::DBD::blah |
195
|
|
|
|
|
|
|
sub _get_db_driver { |
196
|
3
|
|
|
3
|
|
1168155
|
my $dsn = shift; |
197
|
3
|
|
|
|
|
19
|
my $db_driver = __PACKAGE__.'::DBD::'. |
198
|
|
|
|
|
|
|
(split(':', $dsn->[0]))[1]; |
199
|
1
|
|
|
1
|
|
856
|
eval "use $db_driver"; |
|
1
|
|
|
1
|
|
4
|
|
|
1
|
|
|
1
|
|
20
|
|
|
1
|
|
|
|
|
8
|
|
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
13
|
|
|
1
|
|
|
|
|
8
|
|
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
17
|
|
|
3
|
|
|
|
|
470
|
|
200
|
3
|
50
|
|
|
|
14
|
die( |
201
|
|
|
|
|
|
|
__PACKAGE__. |
202
|
|
|
|
|
|
|
": can't find db-specific code for ". |
203
|
|
|
|
|
|
|
$dsn->[0]. |
204
|
|
|
|
|
|
|
"\n:$@\n" |
205
|
|
|
|
|
|
|
) if($@); |
206
|
3
|
|
|
|
|
8
|
return $db_driver; |
207
|
|
|
|
|
|
|
} |
208
|
|
|
|
|
|
|
|
209
|
|
|
|
|
|
|
# map a table name to a classname. Takes a base class name and a table |
210
|
|
|
|
|
|
|
# name, returns a classname |
211
|
|
|
|
|
|
|
sub _table_to_class { |
212
|
2
|
|
|
2
|
|
22
|
my($base, $table) = @_; |
213
|
2
|
|
|
|
|
11
|
$table =~ s/(^|_)(.)/uc($2)/eg; |
|
2
|
|
|
|
|
13
|
|
214
|
2
|
|
|
|
|
11
|
join('::', $base, $table); |
215
|
|
|
|
|
|
|
} |
216
|
|
|
|
|
|
|
|
217
|
|
|
|
|
|
|
=head1 BUGS and WARNINGS |
218
|
|
|
|
|
|
|
|
219
|
|
|
|
|
|
|
This should be considered to be pre-production code. It's probably chock |
220
|
|
|
|
|
|
|
full of exciting bugs. |
221
|
|
|
|
|
|
|
|
222
|
|
|
|
|
|
|
=head1 DATABASES SUPPORTED |
223
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
MySQL and SQLite are supported "out-of-the-box". Adding other databases |
225
|
|
|
|
|
|
|
is a simple matter of writing a "driver" module with two simple methods. |
226
|
|
|
|
|
|
|
You are encouraged to upload such modules to the CPAN yourself. |
227
|
|
|
|
|
|
|
|
228
|
|
|
|
|
|
|
L, for how to interrogate other |
229
|
|
|
|
|
|
|
databases. |
230
|
|
|
|
|
|
|
|
231
|
|
|
|
|
|
|
=head1 AUTHOR, COPYRIGHT and LICENCE |
232
|
|
|
|
|
|
|
|
233
|
|
|
|
|
|
|
Written by David Cantrell Edavid@cantrell.org.ukE |
234
|
|
|
|
|
|
|
|
235
|
|
|
|
|
|
|
Copyright 2008-2009 Outcome Technologies Ltd |
236
|
|
|
|
|
|
|
|
237
|
|
|
|
|
|
|
This software is free-as-in-speech software, and may be used, distributed, |
238
|
|
|
|
|
|
|
and modified under the terms of either the GNU General Public Licence |
239
|
|
|
|
|
|
|
version 2 or the Artistic Licence. It's up to you which one you use. The |
240
|
|
|
|
|
|
|
full text of the licences can be found in the files GPL2.txt and |
241
|
|
|
|
|
|
|
ARTISTIC.txt, respectively. |
242
|
|
|
|
|
|
|
|
243
|
|
|
|
|
|
|
=cut |
244
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
1; |